mirror of
				https://github.com/pion/mediadevices.git
				synced 2025-10-26 18:10:23 +08:00 
			
		
		
		
	Compare commits
	
		
			47 Commits
		
	
	
		
			v0.1.16
			...
			merged-for
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 58767d8fb4 | ||
|   | 662d1ac3a7 | ||
|   | b1f7693135 | ||
|   | d8ff6be0f3 | ||
|   | 41da6ab56a | ||
|   | 419afd453a | ||
|   | 80be2a7a57 | ||
|   | e316b30964 | ||
|   | 596b8c4e11 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | be5f684ea6 | ||
|   | a88c2daf89 | ||
|   | 1f313a9d61 | ||
|   | 19eaf375ff | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b3c94a1f7b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 86cb9f8ce8 | ||
|   | f8d1f974cf | ||
|   | 809f74cafc | ||
|   | eb2db82766 | ||
|   | b4c6eb5409 | ||
|   | b263026d52 | ||
|   | 070ab924f9 | ||
|   | 23177a5d75 | ||
|   | 3a04686875 | ||
|   | 5f95b84719 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8919ba4fe5 | ||
|   | 153c36e461 | ||
|   | 10769b702e | ||
|   | 2948735964 | ||
|   | ba848b3416 | ||
|   | 94c6b66e46 | ||
|   | 96fd92142c | ||
|   | d71b72c64d | ||
|   | 8c2c8a9b27 | ||
|   | 1e03f61b4b | ||
|   | b863c105c8 | ||
|   | 6411b00e93 | ||
|   | acd2cb992b | ||
|   | dafd208de7 | ||
|   | fcec5a9149 | ||
|   | 3d3830f7ff | ||
|   | 655b513810 | ||
|   | eaaaacfc6b | ||
|   | fa95e47bad | ||
|   | 020de77bc9 | ||
|   | 33b6412c26 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f29d08ae6b | ||
|   | b5b0653697 | 
							
								
								
									
										10
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -13,7 +13,7 @@ jobs: | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         go: [ '1.15', '1.14' ] | ||||
|         go: [ '1.16', '1.15' ] | ||||
|     name: Linux Go ${{ matrix.go }} | ||||
|     steps: | ||||
|       - name: Checkout | ||||
| @@ -32,14 +32,14 @@ jobs: | ||||
|             libx264-dev | ||||
|       - name: Run Test Suite | ||||
|         run: make test | ||||
|       - uses: codecov/codecov-action@v1	 | ||||
|         if: matrix.go == '1.15' | ||||
|       - uses: codecov/codecov-action@v2	 | ||||
|         if: matrix.go == '1.16' | ||||
|   build-darwin: | ||||
|     runs-on: macos-latest | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         go: [ '1.15', '1.14' ] | ||||
|         go: [ '1.16', '1.15' ] | ||||
|     name: Darwin Go ${{ matrix.go }} | ||||
|     steps: | ||||
|       - name: Checkout | ||||
| @@ -66,7 +66,7 @@ jobs: | ||||
|       - name: Setup Go | ||||
|         uses: actions/setup-go@v2 | ||||
|         with: | ||||
|           go-version: '1.15' | ||||
|           go-version: '1.16' | ||||
|       - name: Installing go-licenses | ||||
|         run: go get github.com/google/go-licenses | ||||
|       - name: Checking licenses | ||||
|   | ||||
							
								
								
									
										10
									
								
								.idea/mediadevices.iml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								.idea/mediadevices.iml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <module type="JAVA_MODULE" version="4"> | ||||
|   <component name="Go" enabled="true" /> | ||||
|   <component name="NewModuleRootManager" inherit-compiler-output="true"> | ||||
|     <exclude-output /> | ||||
|     <content url="file://$MODULE_DIR$" /> | ||||
|     <orderEntry type="inheritedJdk" /> | ||||
|     <orderEntry type="sourceFolder" forTests="false" /> | ||||
|   </component> | ||||
| </module> | ||||
							
								
								
									
										6
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="ProjectRootManager"> | ||||
|     <output url="file://$PROJECT_DIR$/out" /> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										8
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="ProjectModuleManager"> | ||||
|     <modules> | ||||
|       <module fileurl="file://$PROJECT_DIR$/.idea/mediadevices.iml" filepath="$PROJECT_DIR$/.idea/mediadevices.iml" /> | ||||
|     </modules> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="VcsDirectoryMappings"> | ||||
|     <mapping directory="" vcs="Git" /> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										63
									
								
								.idea/workspace.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								.idea/workspace.xml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="AutoImportSettings"> | ||||
|     <option name="autoReloadType" value="SELECTIVE" /> | ||||
|   </component> | ||||
|   <component name="ChangeListManager"> | ||||
|     <list default="true" id="8804a8cb-7b92-421b-8786-b7715667b867" name="Changes" comment=""> | ||||
|       <change afterPath="$PROJECT_DIR$/.idea/mediadevices.iml" afterDir="false" /> | ||||
|       <change afterPath="$PROJECT_DIR$/.idea/modules.xml" afterDir="false" /> | ||||
|       <change afterPath="$PROJECT_DIR$/.idea/vcs.xml" afterDir="false" /> | ||||
|       <change beforePath="$PROJECT_DIR$/.idea/workspace.xml" beforeDir="false" afterPath="$PROJECT_DIR$/.idea/workspace.xml" afterDir="false" /> | ||||
|       <change beforePath="$PROJECT_DIR$/go.mod" beforeDir="false" afterPath="$PROJECT_DIR$/go.mod" afterDir="false" /> | ||||
|     </list> | ||||
|     <option name="SHOW_DIALOG" value="false" /> | ||||
|     <option name="HIGHLIGHT_CONFLICTS" value="true" /> | ||||
|     <option name="HIGHLIGHT_NON_ACTIVE_CHANGELIST" value="false" /> | ||||
|     <option name="LAST_RESOLUTION" value="IGNORE" /> | ||||
|   </component> | ||||
|   <component name="GOROOT" url="file:///usr/local/opt/go/libexec" /> | ||||
|   <component name="Git.Settings"> | ||||
|     <option name="RECENT_GIT_ROOT_PATH" value="$PROJECT_DIR$" /> | ||||
|   </component> | ||||
|   <component name="GoLibraries"> | ||||
|     <option name="indexEntireGoPath" value="false" /> | ||||
|   </component> | ||||
|   <component name="KubernetesApiPersistence"> | ||||
|     <option name="context" value="crane-nts-0" /> | ||||
|     <option name="namespace" value="default" /> | ||||
|   </component> | ||||
|   <component name="ProjectId" id="25msmAX4e3Virjg5KBrchpLrlbl" /> | ||||
|   <component name="ProjectViewState"> | ||||
|     <option name="autoscrollFromSource" value="true" /> | ||||
|     <option name="autoscrollToSource" value="true" /> | ||||
|     <option name="hideEmptyMiddlePackages" value="true" /> | ||||
|     <option name="showLibraryContents" value="true" /> | ||||
|   </component> | ||||
|   <component name="PropertiesComponent"> | ||||
|     <property name="RunOnceActivity.OpenProjectViewOnStart" value="true" /> | ||||
|     <property name="RunOnceActivity.ShowReadmeOnStart" value="true" /> | ||||
|     <property name="WebServerToolWindowFactoryState" value="false" /> | ||||
|     <property name="go.formatter.settings.were.checked" value="true" /> | ||||
|     <property name="go.import.settings.migrated" value="true" /> | ||||
|     <property name="go.sdk.automatically.set" value="true" /> | ||||
|   </component> | ||||
|   <component name="SpellCheckerSettings" RuntimeDictionaries="0" Folders="0" CustomDictionaries="0" DefaultDictionary="application-level" UseSingleDictionary="true" transferred="true" /> | ||||
|   <component name="TaskManager"> | ||||
|     <task active="true" id="Default" summary="Default task"> | ||||
|       <changelist id="8804a8cb-7b92-421b-8786-b7715667b867" name="Changes" comment="" /> | ||||
|       <created>1646143753802</created> | ||||
|       <option name="number" value="Default" /> | ||||
|       <option name="presentableId" value="Default" /> | ||||
|       <updated>1646143753802</updated> | ||||
|       <workItem from="1646143757180" duration="56000" /> | ||||
|     </task> | ||||
|     <servers /> | ||||
|   </component> | ||||
|   <component name="TypeScriptGeneratedFilesManager"> | ||||
|     <option name="version" value="3" /> | ||||
|   </component> | ||||
|   <component name="VgoProject"> | ||||
|     <integration-enabled>true</integration-enabled> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										3
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								Makefile
									
									
									
									
									
								
							| @@ -25,6 +25,7 @@ codec_list := $(shell ls $(codec_dir)/*/Makefile) | ||||
| codec_list := $(codec_list:$(codec_dir)/%/Makefile=%) | ||||
| targets := $(foreach codec, $(codec_list), $(addprefix $(cmd_build)-$(codec)-, $(supported_platforms))) | ||||
| pkgs_without_mmal := $(shell go list ./... | grep -v mmal) | ||||
| pkgs_without_cgo := $(shell go list ./... | grep -v pkg/codec | grep -v pkg/driver | grep -v pkg/avfoundation) | ||||
|  | ||||
| define BUILD_TEMPLATE | ||||
| ifneq (,$$(findstring $(2)-$(3),$$(supported_platforms))) | ||||
| @@ -74,7 +75,7 @@ $(cmd_test): | ||||
| 	go vet $(pkgs_without_mmal) | ||||
| 	go build $(pkgs_without_mmal) | ||||
| 	# go build without CGO | ||||
| 	CGO_ENABLED=0 go build . pkg/... | ||||
| 	CGO_ENABLED=0 go build $(pkgs_without_cgo) | ||||
| 	# go build with CGO | ||||
| 	CGO_ENABLED=1 go build $(pkgs_without_mmal) | ||||
| 	$(MAKE) --directory=$(examples_dir) | ||||
|   | ||||
							
								
								
									
										1
									
								
								examples/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								examples/.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | ||||
| go.sum | ||||
| @@ -5,4 +5,4 @@ examples := $(filter-out internal,$(examples)) | ||||
| all: $(examples) | ||||
|  | ||||
| $(examples): | ||||
| 	cd $@ && go build | ||||
| 	cd $@ && go build -mod=mod | ||||
|   | ||||
| @@ -2,8 +2,10 @@ module github.com/pion/mediadevices/examples | ||||
|  | ||||
| go 1.14 | ||||
|  | ||||
| // Please don't commit require entries of examples. | ||||
| // `git checkout master examples/go.mod` to revert this file. | ||||
| require github.com/pion/mediadevices v0.0.0 | ||||
| require ( | ||||
| 	github.com/esimov/pigo v1.4.3 | ||||
| 	github.com/pion/mediadevices v0.0.0 | ||||
| 	github.com/pion/webrtc/v3 v3.0.20 | ||||
| ) | ||||
|  | ||||
| replace github.com/pion/mediadevices v0.0.0 => ../ | ||||
|   | ||||
							
								
								
									
										166
									
								
								examples/go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								examples/go.sum
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | ||||
| github.com/BurntSushi/xgb v0.0.0-20210121224620-deaf085860bc/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | ||||
| github.com/blackjack/webcam v0.0.0-20200313125108-10ed912a8539 h1:1aIqYfg9s9RETAJHGfVKZW4ok0b22p4QTwk8MsdRtPs= | ||||
| github.com/blackjack/webcam v0.0.0-20200313125108-10ed912a8539/go.mod h1:G0X+rEqYPWSq0dG8OMf8M446MtKytzpPjgS3HbdOJZ4= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/disintegration/imaging v1.6.1/go.mod h1:xuIt+sRxDFrHS0drzXUlCJthkJ8k7lkkUojDSR247MQ= | ||||
| github.com/esimov/pigo v1.4.3 h1:xl098Z9CHmouywvyRZepuKx8aSWHBs/0lZtp7Yt5g28= | ||||
| github.com/esimov/pigo v1.4.3/go.mod h1:aOTYpOWsqniACzXKdSOGkqI6CnWQpP8tFjgtUOARoEs= | ||||
| github.com/fogleman/gg v1.0.0/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
| github.com/gen2brain/malgo v0.10.29 h1:bTYiUTUKJsEomNby+W0hgyLrOttUXIk4lTEnKA54iqM= | ||||
| github.com/gen2brain/malgo v0.10.29/go.mod h1:zHSUNZAXfCeNsZou0RtQ6Zk7gDYLIcKOrUWtAdksnEs= | ||||
| github.com/gen2brain/shm v0.0.0-20200228170931-49f9650110c5/go.mod h1:uF6rMu/1nvu+5DpiRLwusA6xB8zlkNoGzKn8lmYONUo= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/uuid v1.2.0 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/kbinani/screenshot v0.0.0-20210326165202-b96eb3309bb0/go.mod h1:ZceVWGtzUZmxyN+/1I+oG31oOm1dOA2QUNbua9TLVdE= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1 h1:Fmg33tUaq4/8ym9TJN1x7sLJnHVwhP33CNkpYV/7rwI= | ||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/lxn/win v0.0.0-20210218163916-a377121e959e/go.mod h1:KxxjdtRkfNoYDCUP5ryK7XJJNTnpC8atvtmTheChOtk= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.16.1/go.mod h1:CObGmKUOKaSC0RjmoAK7tKyn4Azo5P2IWuoMnvwxz1E= | ||||
| github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= | ||||
| github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= | ||||
| github.com/onsi/gomega v1.11.0/go.mod h1:azGKhqFUon9Vuj0YmTfLSmx0FUwqXYSTl5re8lQLTUg= | ||||
| github.com/pion/datachannel v1.4.21 h1:3ZvhNyfmxsAqltQrApLPQMhSFNA+aT87RqyCq4OXmf0= | ||||
| github.com/pion/datachannel v1.4.21/go.mod h1:oiNyP4gHx2DIwRzX/MFyH0Rz/Gz05OgBlayAI2hAWjg= | ||||
| github.com/pion/dtls/v2 v2.0.9 h1:7Ow+V++YSZQMYzggI0P9vLJz/hUFcffsfGMfT/Qy+u8= | ||||
| github.com/pion/dtls/v2 v2.0.9/go.mod h1:O0Wr7si/Zj5/EBFlDzDd6UtVxx25CE1r7XM7BQKYQho= | ||||
| github.com/pion/ice/v2 v2.1.7 h1:FjgDfUNrVYTxQabJrkBX6ld12tvYbgzHenqPh3PJF6E= | ||||
| github.com/pion/ice/v2 v2.1.7/go.mod h1:kV4EODVD5ux2z8XncbLHIOtcXKtYXVgLVCeVqnpoeP0= | ||||
| github.com/pion/interceptor v0.0.12 h1:eC1iVneBIAQJEfaNAfDqAncJWhMDAnaXPRCJsltdokE= | ||||
| github.com/pion/interceptor v0.0.12/go.mod h1:qzeuWuD/ZXvPqOnxNcnhWfkCZ2e1kwwslicyyPnhoK4= | ||||
| github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY= | ||||
| github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms= | ||||
| github.com/pion/mdns v0.0.5 h1:Q2oj/JB3NqfzY9xGZ1fPzZzK7sDSD8rZPOvcIQ10BCw= | ||||
| github.com/pion/mdns v0.0.5/go.mod h1:UgssrvdD3mxpi8tMxAXbsppL3vJ4Jipw1mTCW+al01g= | ||||
| github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= | ||||
| github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= | ||||
| github.com/pion/rtcp v1.2.6 h1:1zvwBbyd0TeEuuWftrd/4d++m+/kZSeiguxU61LFWpo= | ||||
| github.com/pion/rtcp v1.2.6/go.mod h1:52rMNPWFsjr39z9B9MhnkqhPLoeHTv1aN63o/42bWE0= | ||||
| github.com/pion/rtp v1.6.2 h1:iGBerLX6JiDjB9NXuaPzHyxHFG9JsIEdgwTC0lp5n/U= | ||||
| github.com/pion/rtp v1.6.2/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||
| github.com/pion/rtp v1.6.5 h1:o2cZf8OascA5HF/b0PAbTxRKvOWxTQxWYt7SlToxFGI= | ||||
| github.com/pion/rtp v1.6.5/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||
| github.com/pion/sctp v1.7.10/go.mod h1:EhpTUQu1/lcK3xI+eriS6/96fWetHGCvBi9MSsnaBN0= | ||||
| github.com/pion/sctp v1.7.12 h1:GsatLufywVruXbZZT1CKg+Jr8ZTkwiPnmUC/oO9+uuY= | ||||
| github.com/pion/sctp v1.7.12/go.mod h1:xFe9cLMZ5Vj6eOzpyiKjT9SwGM4KpK/8Jbw5//jc+0s= | ||||
| github.com/pion/sdp/v3 v3.0.4 h1:2Kf+dgrzJflNCSw3TV5v2VLeI0s/qkzy2r5jlR0wzf8= | ||||
| github.com/pion/sdp/v3 v3.0.4/go.mod h1:bNiSknmJE0HYBprTHXKPQ3+JjacTv5uap92ueJZKsRk= | ||||
| github.com/pion/srtp/v2 v2.0.2 h1:664iGzVmaY7KYS5M0gleY0DscRo9ReDfTxQrq4UgGoU= | ||||
| github.com/pion/srtp/v2 v2.0.2/go.mod h1:VEyLv4CuxrwGY8cxM+Ng3bmVy8ckz/1t6A0q/msKOw0= | ||||
| github.com/pion/stun v0.3.5 h1:uLUCBCkQby4S1cf6CGuR9QrVOKcvUwFeemaC865QHDg= | ||||
| github.com/pion/stun v0.3.5/go.mod h1:gDMim+47EeEtfWogA37n6qXZS88L5V6LqFcf+DZA2UA= | ||||
| github.com/pion/transport v0.10.1/go.mod h1:PBis1stIILMiis0PewDw91WJeLJkyIMcEk+DwKOzf4A= | ||||
| github.com/pion/transport v0.12.2/go.mod h1:N3+vZQD9HlDP5GWkZ85LohxNsDcNgofQmyL6ojX5d8Q= | ||||
| github.com/pion/transport v0.12.3 h1:vdBfvfU/0Wq8kd2yhUMSDB/x+O4Z9MYVl2fJ5BT4JZw= | ||||
| github.com/pion/transport v0.12.3/go.mod h1:OViWW9SP2peE/HbwBvARicmAVnesphkNkCVZIWJ6q9A= | ||||
| github.com/pion/turn/v2 v2.0.5 h1:iwMHqDfPEDEOFzwWKT56eFmh6DYC6o/+xnLAEzgISbA= | ||||
| github.com/pion/turn/v2 v2.0.5/go.mod h1:APg43CFyt/14Uy7heYUOGWdkem/Wu4PhCO/bjyrTqMw= | ||||
| github.com/pion/udp v0.1.1 h1:8UAPvyqmsxK8oOjloDk4wUt63TzFe9WEJkg5lChlj7o= | ||||
| github.com/pion/udp v0.1.1/go.mod h1:6AFo+CMdKQm7UiA0eUPA8/eVCTx8jBIITLZHc9DWX5M= | ||||
| github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/sclevine/agouti v3.0.0+incompatible/go.mod h1:b4WX9W9L1sfQKXeJf1mUTLZKJ48R1S7H23Ji7oFO5Bw= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2 h1:It14KIkyBFYkHkwZ7k45minvA9aorojkyjGk9KJ5B/w= | ||||
| golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= | ||||
| golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= | ||||
| golang.org/x/image v0.0.0-20210220032944-ac19c3e999fb h1:fqpd0EBDzlHRCjiphRR5Zo/RSWWQlWv34418dnEixWk= | ||||
| golang.org/x/image v0.0.0-20210220032944-ac19c3e999fb/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201201195509-5d6afe98e0b7/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210331212208-0fccb6fa2b5c/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= | ||||
| golang.org/x/net v0.0.0-20210420210106-798c2154c571 h1:Q6Bg8xzKzpFPU4Oi1sBnBTHBwlMsLeEXpu4hYBY8rAg= | ||||
| golang.org/x/net v0.0.0-20210420210106-798c2154c571/go.mod h1:72T/g9IO56b78aLF+1Kcs5dz7/ng1VjMUvfKvpfy+jM= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201018230417-eeed37f84f13/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe h1:WdX7u8s3yOigWAhHEaDl8r9G+4XwFQEQFtBMYyN+kXQ= | ||||
| golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/term v0.0.0-20191110171634-ad39bd3f0407/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= | ||||
| google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= | ||||
| google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= | ||||
| gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= | ||||
| gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
							
								
								
									
										46
									
								
								examples/vnc/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								examples/vnc/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| ## Instructions | ||||
|  | ||||
| ### Install required codecs | ||||
|  | ||||
| In this example, we'll be using x264 and opus as our video and audio codecs. Therefore, we need to make sure that these codecs are installed within our system.  | ||||
|  | ||||
| Installation steps: | ||||
|  | ||||
| * [x264](https://github.com/pion/mediadevices#x264) | ||||
|  | ||||
| ### Download vnc example | ||||
|  | ||||
| ``` | ||||
| git clone https://github.com/pion/mediadevices.git | ||||
| ``` | ||||
|  | ||||
| #### Compile vnc example | ||||
|  | ||||
| ``` | ||||
| cd mediadevices/examples/vnc && go build | ||||
| ``` | ||||
|  | ||||
| ### Open example page | ||||
|  | ||||
| [jsfiddle.net](https://jsfiddle.net/gh/get/library/pure/pion/mediadevices/tree/master/examples/internal/jsfiddle/audio-and-video) you should see two text-areas and a 'Start Session' button | ||||
|  | ||||
| ### Run the webrtc example with your browsers SessionDescription as stdin | ||||
|  | ||||
| In the jsfiddle the top textarea is your browser, copy that, and store the session description in an environment variable, `export SDP=<put_the_sdp_here>` | ||||
|  | ||||
| Run `echo $SDP | ./vnc` | ||||
|  | ||||
| In Windows | ||||
|  | ||||
| ```powershell | ||||
| type sdp.txt| .\vnc.exe | ||||
| ``` | ||||
| ### Input webrtc's SessionDescription into your browser | ||||
|  | ||||
| Copy the text that `./webrtc` just emitted and copy into second text area | ||||
|  | ||||
| ### Hit 'Start Session' in jsfiddle, enjoy your video! | ||||
|  | ||||
| A video should start playing in your browser above the input boxes, and will continue playing until you close the application. | ||||
|  | ||||
| Congrats, you have used pion-MediaDevices! Now start building something cool | ||||
							
								
								
									
										127
									
								
								examples/vnc/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								examples/vnc/main.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/pion/mediadevices/pkg/driver" | ||||
| 	"github.com/pion/mediadevices/pkg/driver/vncdriver" | ||||
|  | ||||
| 	"github.com/pion/mediadevices" | ||||
| 	"github.com/pion/mediadevices/examples/internal/signal" | ||||
| 	"github.com/pion/webrtc/v3" | ||||
|  | ||||
| 	// If you don't like x264, you can also use vpx by importing as below | ||||
| 	// "github.com/pion/mediadevices/pkg/codec/vpx" // This is required to use VP8/VP9 video encoder | ||||
| 	// or you can also use openh264 for alternative h264 implementation | ||||
| 	// "github.com/pion/mediadevices/pkg/codec/openh264" | ||||
| 	// or if you use a raspberry pi like, you can use mmal for using its hardware encoder | ||||
| 	// "github.com/pion/mediadevices/pkg/codec/mmal" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/x264" // This is required to use h264 video encoder | ||||
|  | ||||
| 	// Note: If you don't have a camera or microphone or your adapters are not supported, | ||||
| 	//       you can always swap your adapters with our dummy adapters below. | ||||
| 	// _ "github.com/pion/mediadevices/pkg/driver/videotest" | ||||
| 	// _ "github.com/pion/mediadevices/pkg/driver/audiotest" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
| 	config := webrtc.Configuration{ | ||||
| 		ICEServers: []webrtc.ICEServer{ | ||||
| 			{ | ||||
| 				URLs: []string{"stun:stun.l.google.com:19302"}, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| 	driver.GetManager().Register( | ||||
| 		vncdriver.NewVnc("127.0.0.1:5900"), | ||||
| 		driver.Info{Label: "VNC", DeviceType: driver.Camera, Priority: driver.PriorityLow}, | ||||
| 	) | ||||
| 	// Wait for the offer to be pasted | ||||
| 	offer := webrtc.SessionDescription{} | ||||
| 	signal.Decode(signal.MustReadStdin(), &offer) | ||||
|  | ||||
| 	// Create a new RTCPeerConnection | ||||
| 	x264Params, err := x264.NewParams() | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	x264Params.BitRate = 500_000 // 500kbps | ||||
|  | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
| 	codecSelector := mediadevices.NewCodecSelector( | ||||
| 		mediadevices.WithVideoEncoders(&x264Params), | ||||
| 	) | ||||
|  | ||||
| 	mediaEngine := webrtc.MediaEngine{} | ||||
| 	codecSelector.Populate(&mediaEngine) | ||||
| 	api := webrtc.NewAPI(webrtc.WithMediaEngine(&mediaEngine)) | ||||
| 	peerConnection, err := api.NewPeerConnection(config) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Set the handler for ICE connection state | ||||
| 	// This will notify you when the peer has connected/disconnected | ||||
| 	peerConnection.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) { | ||||
| 		fmt.Printf("Connection State has changed %s \n", connectionState.String()) | ||||
| 	}) | ||||
|  | ||||
| 	s, err := mediadevices.GetUserMedia(mediadevices.MediaStreamConstraints{ | ||||
| 		Video: func(c *mediadevices.MediaTrackConstraints) { | ||||
|  | ||||
| 		}, | ||||
| 		Codec: codecSelector, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	for _, track := range s.GetTracks() { | ||||
| 		track.OnEnded(func(err error) { | ||||
| 			fmt.Printf("Track (ID: %s) ended with error: %v\n", | ||||
| 				track.ID(), err) | ||||
| 		}) | ||||
|  | ||||
| 		_, err = peerConnection.AddTransceiverFromTrack(track, | ||||
| 			webrtc.RtpTransceiverInit{ | ||||
| 				Direction: webrtc.RTPTransceiverDirectionSendonly, | ||||
| 			}, | ||||
| 		) | ||||
| 		if err != nil { | ||||
| 			panic(err) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Set the remote SessionDescription | ||||
| 	err = peerConnection.SetRemoteDescription(offer) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Create an answer | ||||
| 	answer, err := peerConnection.CreateAnswer(nil) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Create channel that is blocked until ICE Gathering is complete | ||||
| 	gatherComplete := webrtc.GatheringCompletePromise(peerConnection) | ||||
|  | ||||
| 	// Sets the LocalDescription, and starts our UDP listeners | ||||
| 	err = peerConnection.SetLocalDescription(answer) | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	// Block until ICE Gathering is complete, disabling trickle ICE | ||||
| 	// we do this because we only can exchange one signaling message | ||||
| 	// in a production application you should exchange ICE Candidates via OnICECandidate | ||||
| 	<-gatherComplete | ||||
|  | ||||
| 	// Output the answer in base64 so we can paste it in browser | ||||
| 	fmt.Println(signal.Encode(*peerConnection.LocalDescription())) | ||||
|  | ||||
| 	// Block forever | ||||
| 	select {} | ||||
| } | ||||
							
								
								
									
										19
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								go.mod
									
									
									
									
									
								
							| @@ -2,18 +2,17 @@ module github.com/pion/mediadevices | ||||
|  | ||||
| go 1.13 | ||||
|  | ||||
| replace github.com/pion/webrtc/v3 => github.com/EmrysMyrddin/webrtc/v3 v3.0.30-0.20220301133300-cf76910357bd | ||||
|  | ||||
| require ( | ||||
| 	github.com/BurntSushi/xgb v0.0.0-20201008132610-5f9e7b3c49cd // indirect | ||||
| 	github.com/blackjack/webcam v0.0.0-20200313125108-10ed912a8539 | ||||
| 	github.com/gen2brain/malgo v0.10.29 | ||||
| 	github.com/gen2brain/shm v0.0.0-20200228170931-49f9650110c5 // indirect | ||||
| 	github.com/google/uuid v1.2.0 | ||||
| 	github.com/kbinani/screenshot v0.0.0-20191211154542-3a185f1ce18f | ||||
| 	github.com/lxn/win v0.0.0-20201111105847-2a20daff6a55 // indirect | ||||
| 	github.com/gen2brain/malgo v0.10.35 | ||||
| 	github.com/google/uuid v1.3.0 | ||||
| 	github.com/kbinani/screenshot v0.0.0-20210720154843-7d3a670d8329 | ||||
| 	github.com/pion/logging v0.2.2 | ||||
| 	github.com/pion/rtp v1.6.2 | ||||
| 	github.com/pion/webrtc/v3 v3.0.5 | ||||
| 	github.com/satori/go.uuid v1.2.0 | ||||
| 	golang.org/x/image v0.0.0-20201208152932-35266b937fa6 | ||||
| 	github.com/pion/rtcp v1.2.9 | ||||
| 	github.com/pion/rtp v1.7.4 | ||||
| 	github.com/pion/webrtc/v3 v3.1.10 | ||||
| 	golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d | ||||
| 	gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c // indirect | ||||
| ) | ||||
|   | ||||
							
								
								
									
										141
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										141
									
								
								go.sum
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | ||||
| github.com/BurntSushi/xgb v0.0.0-20201008132610-5f9e7b3c49cd h1:u7K2oMFMd8APDV3fM1j2rO3U/XJf1g1qC3DDTKou8iM= | ||||
| github.com/BurntSushi/xgb v0.0.0-20201008132610-5f9e7b3c49cd/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= | ||||
| github.com/EmrysMyrddin/webrtc/v3 v3.0.30-0.20220301133300-cf76910357bd h1:LMu4e+BuBW8N1TnzXXz4bB/O47fQzw0CqfyROgzMSx8= | ||||
| github.com/EmrysMyrddin/webrtc/v3 v3.0.30-0.20220301133300-cf76910357bd/go.mod h1:mO/yv7fBN3Lp7YNlnYcTj1jtpvNvssJG+7eh6itZ4xM= | ||||
| github.com/blackjack/webcam v0.0.0-20200313125108-10ed912a8539 h1:1aIqYfg9s9RETAJHGfVKZW4ok0b22p4QTwk8MsdRtPs= | ||||
| github.com/blackjack/webcam v0.0.0-20200313125108-10ed912a8539/go.mod h1:G0X+rEqYPWSq0dG8OMf8M446MtKytzpPjgS3HbdOJZ4= | ||||
| github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| @@ -7,10 +7,11 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= | ||||
| github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= | ||||
| github.com/gen2brain/malgo v0.10.29 h1:bTYiUTUKJsEomNby+W0hgyLrOttUXIk4lTEnKA54iqM= | ||||
| github.com/gen2brain/malgo v0.10.29/go.mod h1:zHSUNZAXfCeNsZou0RtQ6Zk7gDYLIcKOrUWtAdksnEs= | ||||
| github.com/gen2brain/malgo v0.10.35 h1:D6aNo/Q0SnzQLHomTydTXxj4AJFdGJcVoE7I8JxPoUo= | ||||
| github.com/gen2brain/malgo v0.10.35/go.mod h1:zHSUNZAXfCeNsZou0RtQ6Zk7gDYLIcKOrUWtAdksnEs= | ||||
| github.com/gen2brain/shm v0.0.0-20200228170931-49f9650110c5 h1:Y5Q2mEwfzjMt5+3u70Gtw93ZOu2UuPeeeTBDntF7FoY= | ||||
| github.com/gen2brain/shm v0.0.0-20200228170931-49f9650110c5/go.mod h1:uF6rMu/1nvu+5DpiRLwusA6xB8zlkNoGzKn8lmYONUo= | ||||
| github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= | ||||
| github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||
| github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= | ||||
| github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= | ||||
| @@ -18,102 +19,106 @@ github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrU | ||||
| github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= | ||||
| github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= | ||||
| github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= | ||||
| github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= | ||||
| github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= | ||||
| github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= | ||||
| github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/uuid v1.1.5/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/uuid v1.2.0 h1:qJYtXnJRWmpe7m/3XlyhrsLrEURqHRM2kxzoxXqyUDs= | ||||
| github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= | ||||
| github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= | ||||
| github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= | ||||
| github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= | ||||
| github.com/kbinani/screenshot v0.0.0-20191211154542-3a185f1ce18f h1:5hWo+DzJQSOBl6X+TDac0SPWffRonuRJ2///OYtYRT8= | ||||
| github.com/kbinani/screenshot v0.0.0-20191211154542-3a185f1ce18f/go.mod h1:f8GY5V3lRzakvEyr49P7hHRYoHtPr8zvj/7JodCoRzw= | ||||
| github.com/jezek/xgb v0.0.0-20210312150743-0e0f116e1240 h1:dy+DS31tGEGCsZzB45HmJJNHjur8GDgtRNX9U7HnSX4= | ||||
| github.com/jezek/xgb v0.0.0-20210312150743-0e0f116e1240/go.mod h1:3P4UH/k22rXyHIJD2w4h2XMqPX4Of/eySEZq9L6wqc4= | ||||
| github.com/kbinani/screenshot v0.0.0-20210720154843-7d3a670d8329 h1:qq2nCpSrXrmvDGRxW0ruW9BVEV1CN2a9YDOExdt+U0o= | ||||
| github.com/kbinani/screenshot v0.0.0-20210720154843-7d3a670d8329/go.mod h1:2VPVQDR4wO7KXHwP+DAypEy67rXf+okUx2zjgpCxZw4= | ||||
| github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= | ||||
| github.com/kr/pretty v0.2.1 h1:Fmg33tUaq4/8ym9TJN1x7sLJnHVwhP33CNkpYV/7rwI= | ||||
| github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= | ||||
| github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= | ||||
| github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE= | ||||
| github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= | ||||
| github.com/lxn/win v0.0.0-20201111105847-2a20daff6a55 h1:4BxFx5XCtXc+nFtXDGDW+Uu5sPtsAbvPh6RObj3fG9o= | ||||
| github.com/lxn/win v0.0.0-20201111105847-2a20daff6a55/go.mod h1:KxxjdtRkfNoYDCUP5ryK7XJJNTnpC8atvtmTheChOtk= | ||||
| github.com/lxn/win v0.0.0-20210218163916-a377121e959e h1:H+t6A/QJMbhCSEH5rAuRxh+CtW96g0Or0Fxa9IKr4uc= | ||||
| github.com/lxn/win v0.0.0-20210218163916-a377121e959e/go.mod h1:KxxjdtRkfNoYDCUP5ryK7XJJNTnpC8atvtmTheChOtk= | ||||
| github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= | ||||
| github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= | ||||
| github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= | ||||
| github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= | ||||
| github.com/onsi/ginkgo v1.14.2/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY= | ||||
| github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= | ||||
| github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= | ||||
| github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= | ||||
| github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= | ||||
| github.com/onsi/gomega v1.10.3/go.mod h1:V9xEwhxec5O8UDM77eCW8vLymOMltsqPVYWrpDsH8xc= | ||||
| github.com/pion/datachannel v1.4.21 h1:3ZvhNyfmxsAqltQrApLPQMhSFNA+aT87RqyCq4OXmf0= | ||||
| github.com/pion/datachannel v1.4.21/go.mod h1:oiNyP4gHx2DIwRzX/MFyH0Rz/Gz05OgBlayAI2hAWjg= | ||||
| github.com/pion/dtls/v2 v2.0.4 h1:WuUcqi6oYMu/noNTz92QrF1DaFj4eXbhQ6dzaaAwOiI= | ||||
| github.com/pion/dtls/v2 v2.0.4/go.mod h1:qAkFscX0ZHoI1E07RfYPoRw3manThveu+mlTDdOxoGI= | ||||
| github.com/pion/ice/v2 v2.0.15 h1:KZrwa2ciL9od8+TUVJiYTNsCW9J5lktBjGwW1MacEnQ= | ||||
| github.com/pion/ice/v2 v2.0.15/go.mod h1:ZIiVGevpgAxF/cXiIVmuIUtCb3Xs4gCzCbXB6+nFkSI= | ||||
| github.com/pion/interceptor v0.0.9 h1:fk5hTdyLO3KURQsf/+RjMpEm4NE3yeTY9Kh97b5BvwA= | ||||
| github.com/pion/interceptor v0.0.9/go.mod h1:dHgEP5dtxOTf21MObuBAjJeAayPxLUAZjerGH8Xr07c= | ||||
| github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= | ||||
| github.com/pion/datachannel v1.5.2 h1:piB93s8LGmbECrpO84DnkIVWasRMk3IimbcXkTQLE6E= | ||||
| github.com/pion/datachannel v1.5.2/go.mod h1:FTGQWaHrdCwIJ1rw6xBIfZVkslikjShim5yr05XFuCQ= | ||||
| github.com/pion/dtls/v2 v2.1.2/go.mod h1:o6+WvyLDAlXF7YiPB/RlskRoeK+/JtuaZa5emwQcWus= | ||||
| github.com/pion/dtls/v2 v2.1.3 h1:3UF7udADqous+M2R5Uo2q/YaP4EzUoWKdfX2oscCUio= | ||||
| github.com/pion/dtls/v2 v2.1.3/go.mod h1:o6+WvyLDAlXF7YiPB/RlskRoeK+/JtuaZa5emwQcWus= | ||||
| github.com/pion/ice/v2 v2.2.1 h1:R3MeuJZpU1ty3diPqpD5OxaxcZ15eprAc+EtUiSoFxg= | ||||
| github.com/pion/ice/v2 v2.2.1/go.mod h1:Op8jlPtjeiycsXh93Cs4jK82C9j/kh7vef6ztIOvtIQ= | ||||
| github.com/pion/interceptor v0.1.7 h1:HThW0tIIKT9RRoDWGURe8rlZVOx0fJHxBHpA0ej0+bo= | ||||
| github.com/pion/interceptor v0.1.7/go.mod h1:Lh3JSl/cbJ2wP8I3ccrjh1K/deRGRn3UlSPuOTiHb6U= | ||||
| github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY= | ||||
| github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms= | ||||
| github.com/pion/mdns v0.0.4 h1:O4vvVqr4DGX63vzmO6Fw9vpy3lfztVWHGCQfyw0ZLSY= | ||||
| github.com/pion/mdns v0.0.4/go.mod h1:R1sL0p50l42S5lJs91oNdUL58nm0QHrhxnSegr++qC0= | ||||
| github.com/pion/mdns v0.0.5 h1:Q2oj/JB3NqfzY9xGZ1fPzZzK7sDSD8rZPOvcIQ10BCw= | ||||
| github.com/pion/mdns v0.0.5/go.mod h1:UgssrvdD3mxpi8tMxAXbsppL3vJ4Jipw1mTCW+al01g= | ||||
| github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= | ||||
| github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= | ||||
| github.com/pion/rtcp v1.2.6 h1:1zvwBbyd0TeEuuWftrd/4d++m+/kZSeiguxU61LFWpo= | ||||
| github.com/pion/rtcp v1.2.6/go.mod h1:52rMNPWFsjr39z9B9MhnkqhPLoeHTv1aN63o/42bWE0= | ||||
| github.com/pion/rtp v1.6.2 h1:iGBerLX6JiDjB9NXuaPzHyxHFG9JsIEdgwTC0lp5n/U= | ||||
| github.com/pion/rtp v1.6.2/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||
| github.com/pion/sctp v1.7.10/go.mod h1:EhpTUQu1/lcK3xI+eriS6/96fWetHGCvBi9MSsnaBN0= | ||||
| github.com/pion/sctp v1.7.11 h1:UCnj7MsobLKLuP/Hh+JMiI/6W5Bs/VF45lWKgHFjSIE= | ||||
| github.com/pion/sctp v1.7.11/go.mod h1:EhpTUQu1/lcK3xI+eriS6/96fWetHGCvBi9MSsnaBN0= | ||||
| github.com/pion/rtcp v1.2.9 h1:1ujStwg++IOLIEoOiIQ2s+qBuJ1VN81KW+9pMPsif+U= | ||||
| github.com/pion/rtcp v1.2.9/go.mod h1:qVPhiCzAm4D/rxb6XzKeyZiQK69yJpbUDJSF7TgrqNo= | ||||
| github.com/pion/rtp v1.7.0/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||
| github.com/pion/rtp v1.7.4 h1:4dMbjb1SuynU5OpA3kz1zHK+u+eOCQjW3MAeVHf1ODA= | ||||
| github.com/pion/rtp v1.7.4/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= | ||||
| github.com/pion/sctp v1.8.0/go.mod h1:xFe9cLMZ5Vj6eOzpyiKjT9SwGM4KpK/8Jbw5//jc+0s= | ||||
| github.com/pion/sctp v1.8.2 h1:yBBCIrUMJ4yFICL3RIvR4eh/H2BTTvlligmSTy+3kiA= | ||||
| github.com/pion/sctp v1.8.2/go.mod h1:xFe9cLMZ5Vj6eOzpyiKjT9SwGM4KpK/8Jbw5//jc+0s= | ||||
| github.com/pion/sdp/v3 v3.0.4 h1:2Kf+dgrzJflNCSw3TV5v2VLeI0s/qkzy2r5jlR0wzf8= | ||||
| github.com/pion/sdp/v3 v3.0.4/go.mod h1:bNiSknmJE0HYBprTHXKPQ3+JjacTv5uap92ueJZKsRk= | ||||
| github.com/pion/srtp/v2 v2.0.1 h1:kgfh65ob3EcnFYA4kUBvU/menCp9u7qaJLXwWgpobzs= | ||||
| github.com/pion/srtp/v2 v2.0.1/go.mod h1:c8NWHhhkFf/drmHTAblkdu8++lsISEBBdAuiyxgqIsE= | ||||
| github.com/pion/srtp/v2 v2.0.5 h1:ks3wcTvIUE/GHndO3FAvROQ9opy0uLELpwHJaQ1yqhQ= | ||||
| github.com/pion/srtp/v2 v2.0.5/go.mod h1:8k6AJlal740mrZ6WYxc4Dg6qDqqhxoRG2GSjlUhDF0A= | ||||
| github.com/pion/stun v0.3.5 h1:uLUCBCkQby4S1cf6CGuR9QrVOKcvUwFeemaC865QHDg= | ||||
| github.com/pion/stun v0.3.5/go.mod h1:gDMim+47EeEtfWogA37n6qXZS88L5V6LqFcf+DZA2UA= | ||||
| github.com/pion/transport v0.8.10/go.mod h1:tBmha/UCjpum5hqTWhfAEs3CO4/tHSg0MYRhSzR+CZ8= | ||||
| github.com/pion/transport v0.10.0/go.mod h1:BnHnUipd0rZQyTVB2SBGojFHT9CBt5C5TcsJSQGkvSE= | ||||
| github.com/pion/transport v0.10.1/go.mod h1:PBis1stIILMiis0PewDw91WJeLJkyIMcEk+DwKOzf4A= | ||||
| github.com/pion/transport v0.12.1/go.mod h1:N3+vZQD9HlDP5GWkZ85LohxNsDcNgofQmyL6ojX5d8Q= | ||||
| github.com/pion/transport v0.12.2 h1:WYEjhloRHt1R86LhUKjC5y+P52Y11/QqEUalvtzVoys= | ||||
| github.com/pion/transport v0.12.2/go.mod h1:N3+vZQD9HlDP5GWkZ85LohxNsDcNgofQmyL6ojX5d8Q= | ||||
| github.com/pion/turn/v2 v2.0.5 h1:iwMHqDfPEDEOFzwWKT56eFmh6DYC6o/+xnLAEzgISbA= | ||||
| github.com/pion/turn/v2 v2.0.5/go.mod h1:APg43CFyt/14Uy7heYUOGWdkem/Wu4PhCO/bjyrTqMw= | ||||
| github.com/pion/udp v0.1.0 h1:uGxQsNyrqG3GLINv36Ff60covYmfrLoxzwnCsIYspXI= | ||||
| github.com/pion/udp v0.1.0/go.mod h1:BPELIjbwE9PRbd/zxI/KYBnbo7B6+oA6YuEaNE8lths= | ||||
| github.com/pion/webrtc/v3 v3.0.5 h1:utennp7RwX+2mtyMzoOXE03IUIckiHBigjarRJZ2DqY= | ||||
| github.com/pion/webrtc/v3 v3.0.5/go.mod h1:/EDCREM8y+JrJSkoCRHpoz//qtuBCOYV4E96vEK3bz0= | ||||
| github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= | ||||
| github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= | ||||
| github.com/pion/transport v0.12.3/go.mod h1:OViWW9SP2peE/HbwBvARicmAVnesphkNkCVZIWJ6q9A= | ||||
| github.com/pion/transport v0.13.0 h1:KWTA5ZrQogizzYwPEciGtHPLwpAjE91FgXnyu+Hv2uY= | ||||
| github.com/pion/transport v0.13.0/go.mod h1:yxm9uXpK9bpBBWkITk13cLo1y5/ur5VQpG22ny6EP7g= | ||||
| github.com/pion/turn/v2 v2.0.8 h1:KEstL92OUN3k5k8qxsXHpr7WWfrdp7iJZHx99ud8muw= | ||||
| github.com/pion/turn/v2 v2.0.8/go.mod h1:+y7xl719J8bAEVpSXBXvTxStjJv3hbz9YFflvkpcGPw= | ||||
| github.com/pion/udp v0.1.1 h1:8UAPvyqmsxK8oOjloDk4wUt63TzFe9WEJkg5lChlj7o= | ||||
| github.com/pion/udp v0.1.1/go.mod h1:6AFo+CMdKQm7UiA0eUPA8/eVCTx8jBIITLZHc9DWX5M= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/satori/go.uuid v1.2.0 h1:0uYX9dsZ2yD7q2RtLRtPSdGDWzjeM3TbMJP9utgA0ww= | ||||
| github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= | ||||
| github.com/sclevine/agouti v3.0.0+incompatible/go.mod h1:b4WX9W9L1sfQKXeJf1mUTLZKJ48R1S7H23Ji7oFO5Bw= | ||||
| github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= | ||||
| github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= | ||||
| github.com/stretchr/testify v1.6.1 h1:hDPOHmpOpP40lSULcqw7IrRb/u7w6RpDC9399XyoNd0= | ||||
| github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= | ||||
| github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= | ||||
| github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= | ||||
| golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= | ||||
| golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= | ||||
| golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897 h1:pLI5jrR7OSLijeIDcmRxNmw2api+jEfxLoykJVice/E= | ||||
| golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= | ||||
| golang.org/x/image v0.0.0-20201208152932-35266b937fa6 h1:nfeHNc1nAqecKCy2FCy4HY+soOOe5sDLJ/gZLbx6GYI= | ||||
| golang.org/x/image v0.0.0-20201208152932-35266b937fa6/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0= | ||||
| golang.org/x/crypto v0.0.0-20220131195533-30dcbda58838 h1:71vQrMauZZhcTVK6KdYM+rklehEEwb3E+ZhaE5jrPrE= | ||||
| golang.org/x/crypto v0.0.0-20220131195533-30dcbda58838/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= | ||||
| golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d h1:RNPAfi2nHY7C2srAV8A49jpsYr0ADedCk1wq6fTMTvs= | ||||
| golang.org/x/image v0.0.0-20210628002857-a66eb6448b8d/go.mod h1:023OzeP/+EPmXeapQh35lcL3II3LrY8Ic+EFFKVhULM= | ||||
| golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= | ||||
| golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||
| golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= | ||||
| golang.org/x/net v0.0.0-20191126235420-ef20fe5d7933/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= | ||||
| golang.org/x/net v0.0.0-20200520004742-59133d7f0dd7/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= | ||||
| golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= | ||||
| golang.org/x/net v0.0.0-20201006153459-a7d1128ccaa0/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201201195509-5d6afe98e0b7 h1:3uJsdck53FDIpWwLeAXlia9p4C8j0BO2xZrqzKpL0D8= | ||||
| golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20201201195509-5d6afe98e0b7/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777 h1:003p0dJM77cxMSyCPFphvZf/Y5/NXf5fzg6ufd1/Oew= | ||||
| golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= | ||||
| golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= | ||||
| golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20211201190559-0a0e4e1bb54c/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= | ||||
| golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd h1:O7DYs+zxREGLKzKoMQrtrEacpb0ZVXA5rIwylE2Xchk= | ||||
| golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= | ||||
| golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||
| golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
| golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| @@ -121,16 +126,25 @@ golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7w | ||||
| golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201018230417-eeed37f84f13/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68 h1:nxC68pudNYkKU6jWhgrqdreuFiOQWj1Fs7T3VrH4Pjw= | ||||
| golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||
| golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e h1:fLOSk5Q00efkSvAm+4xcoXD+RRmLmmulPn5I3Y9F2EM= | ||||
| golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||
| golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= | ||||
| golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= | ||||
| golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||
| golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= | ||||
| golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= | ||||
| golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= | ||||
| golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||
| golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= | ||||
| golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= | ||||
| golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE= | ||||
| golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= | ||||
| @@ -140,6 +154,8 @@ google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQ | ||||
| google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= | ||||
| google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= | ||||
| google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= | ||||
| google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= | ||||
| google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= | ||||
| gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||
| gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= | ||||
| @@ -149,5 +165,6 @@ gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWD | ||||
| gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||
| gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c h1:dUUwHk2QECo/6vqA44rthZ8ie2QXMNeKRTHCNY2nXvo= | ||||
| gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= | ||||
|   | ||||
							
								
								
									
										24
									
								
								ioreader.go
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								ioreader.go
									
									
									
									
									
								
							| @@ -1,5 +1,7 @@ | ||||
| package mediadevices | ||||
|  | ||||
| import "github.com/pion/mediadevices/pkg/codec" | ||||
|  | ||||
| type EncodedBuffer struct { | ||||
| 	Data    []byte | ||||
| 	Samples uint32 | ||||
| @@ -8,11 +10,13 @@ type EncodedBuffer struct { | ||||
| type EncodedReadCloser interface { | ||||
| 	Read() (EncodedBuffer, func(), error) | ||||
| 	Close() error | ||||
| 	codec.Controllable | ||||
| } | ||||
|  | ||||
| type encodedReadCloserImpl struct { | ||||
| 	readFn  func() (EncodedBuffer, func(), error) | ||||
| 	closeFn func() error | ||||
| 	readFn       func() (EncodedBuffer, func(), error) | ||||
| 	closeFn      func() error | ||||
| 	controllerFn func() codec.EncoderController | ||||
| } | ||||
|  | ||||
| func (r *encodedReadCloserImpl) Read() (EncodedBuffer, func(), error) { | ||||
| @@ -23,9 +27,14 @@ func (r *encodedReadCloserImpl) Close() error { | ||||
| 	return r.closeFn() | ||||
| } | ||||
|  | ||||
| func (r *encodedReadCloserImpl) Controller() codec.EncoderController { | ||||
| 	return r.controllerFn() | ||||
| } | ||||
|  | ||||
| type encodedIOReadCloserImpl struct { | ||||
| 	readFn  func([]byte) (int, error) | ||||
| 	closeFn func() error | ||||
| 	readFn     func([]byte) (int, error) | ||||
| 	closeFn    func() error | ||||
| 	controller func() codec.EncoderController | ||||
| } | ||||
|  | ||||
| func newEncodedIOReadCloserImpl(reader EncodedReadCloser) *encodedIOReadCloserImpl { | ||||
| @@ -48,7 +57,8 @@ func newEncodedIOReadCloserImpl(reader EncodedReadCloser) *encodedIOReadCloserIm | ||||
| 			encoded.Data = encoded.Data[n:] | ||||
| 			return n, nil | ||||
| 		}, | ||||
| 		closeFn: reader.Close, | ||||
| 		closeFn:    reader.Close, | ||||
| 		controller: reader.Controller, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @@ -59,3 +69,7 @@ func (r *encodedIOReadCloserImpl) Read(b []byte) (int, error) { | ||||
| func (r *encodedIOReadCloserImpl) Close() error { | ||||
| 	return r.closeFn() | ||||
| } | ||||
|  | ||||
| func (r *encodedIOReadCloserImpl) Controller() codec.EncoderController { | ||||
| 	return r.controller() | ||||
| } | ||||
|   | ||||
| @@ -19,6 +19,10 @@ func (track *mockMediaStreamTrack) StreamID() string { | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (track *mockMediaStreamTrack) RID() string { | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| func (track *mockMediaStreamTrack) Close() error { | ||||
| 	return nil | ||||
| } | ||||
|   | ||||
| @@ -1,17 +1,17 @@ | ||||
| // MIT License | ||||
| //  | ||||
| // | ||||
| // Copyright (c) 2019-2020 Pion | ||||
| //  | ||||
| // | ||||
| // Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| // of this software and associated documentation files (the "Software"), to deal | ||||
| // in the Software without restriction, including without limitation the rights | ||||
| // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| // copies of the Software, and to permit persons to whom the Software is | ||||
| // furnished to do so, subject to the following conditions: | ||||
| //  | ||||
| // | ||||
| // The above copyright notice and this permission notice shall be included in all | ||||
| // copies or substantial portions of the Software. | ||||
| //  | ||||
| // | ||||
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| @@ -76,29 +76,23 @@ didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer | ||||
|         !CMSampleBufferDataIsReady(sampleBuffer)) { | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|  | ||||
|     CVImageBufferRef imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer); | ||||
|     if (imageBuffer == NULL) { | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|  | ||||
|     imageBuffer = CVBufferRetain(imageBuffer); | ||||
|     CVReturn ret = | ||||
|         CVPixelBufferLockBaseAddress(imageBuffer, kCVPixelBufferLock_ReadOnly); | ||||
|     if (ret != kCVReturnSuccess) { | ||||
|       return; | ||||
|     } | ||||
|      | ||||
|     size_t heightY = CVPixelBufferGetHeightOfPlane(imageBuffer, 0); | ||||
|     size_t bytesPerRowY = CVPixelBufferGetBytesPerRowOfPlane(imageBuffer, 0); | ||||
|      | ||||
|     size_t heightUV = CVPixelBufferGetHeightOfPlane(imageBuffer, 1); | ||||
|     size_t bytesPerRowUV = CVPixelBufferGetBytesPerRowOfPlane(imageBuffer, 1); | ||||
|      | ||||
|     int len = (int)((heightY * bytesPerRowY) + (2 * heightUV * bytesPerRowUV)); | ||||
|  | ||||
|     void *buf = CVPixelBufferGetBaseAddressOfPlane(imageBuffer, 0); | ||||
|     _mCallback(_mPUserData, buf, len); | ||||
|      | ||||
|     size_t dataSize = CVPixelBufferGetDataSize(imageBuffer); | ||||
|     _mCallback(_mPUserData, buf, (int)dataSize); | ||||
|  | ||||
|     CVPixelBufferUnlockBaseAddress(imageBuffer, 0); | ||||
|     CVBufferRelease(imageBuffer); | ||||
| } | ||||
| @@ -136,8 +130,8 @@ STATUS frameFormatToFourCC(AVBindFrameFormat format, FourCharCode *pFourCC) { | ||||
|     // Useful mapping reference from ffmpeg: | ||||
|     // https://github.com/FFmpeg/FFmpeg/blob/c810a9502cebe32e1dd08ee3d0d17053dde44aa9/libavdevice/avfoundation.m#L53-L80 | ||||
|     switch (format) { | ||||
|         case AVBindFrameFormatNV21: | ||||
|             *pFourCC = kCVPixelFormatType_420YpCbCr8BiPlanarFullRange; | ||||
|         case AVBindFrameFormatI420: | ||||
|             *pFourCC = kCVPixelFormatType_420YpCbCr8Planar; | ||||
|             break; | ||||
|         case AVBindFrameFormatNV12: | ||||
|             *pFourCC = kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange; | ||||
| @@ -158,9 +152,10 @@ STATUS frameFormatToFourCC(AVBindFrameFormat format, FourCharCode *pFourCC) { | ||||
| STATUS frameFormatFromFourCC(FourCharCode fourCC, AVBindFrameFormat *pFormat) { | ||||
|     STATUS retStatus = STATUS_OK; | ||||
|     switch (fourCC) { | ||||
|         case kCVPixelFormatType_420YpCbCr8BiPlanarFullRange: | ||||
|             *pFormat = AVBindFrameFormatNV21; | ||||
|         case kCVPixelFormatType_420YpCbCr8Planar: | ||||
|             *pFormat = AVBindFrameFormatI420; | ||||
|             break; | ||||
|         case kCVPixelFormatType_420YpCbCr8BiPlanarFullRange: | ||||
|         case kCVPixelFormatType_420YpCbCr8BiPlanarVideoRange: | ||||
|             *pFormat = AVBindFrameFormatNV12; | ||||
|             break; | ||||
| @@ -171,8 +166,8 @@ STATUS frameFormatFromFourCC(FourCharCode fourCC, AVBindFrameFormat *pFormat) { | ||||
|             *pFormat = AVBindFrameFormatYUY2; | ||||
|             break; | ||||
|          // TODO: Add the rest of frame formats | ||||
|          default: | ||||
|              retStatus = STATUS_UNSUPPORTED_FRAME_FORMAT; | ||||
|         default: | ||||
|             retStatus = STATUS_UNSUPPORTED_FRAME_FORMAT; | ||||
|      } | ||||
|     return retStatus; | ||||
| } | ||||
| @@ -184,7 +179,7 @@ STATUS AVBindDevices(AVBindMediaType mediaType, PAVBindDevice *ppDevices, int *p | ||||
|     NSAutoreleasePool *refPool = [[NSAutoreleasePool alloc] init]; | ||||
|     CHK(mediaType == AVBindMediaTypeVideo || mediaType == AVBindMediaTypeAudio, STATUS_UNSUPPORTED_MEDIA_TYPE); | ||||
|     CHK(ppDevices != NULL && pLen != NULL, STATUS_NULL_ARG); | ||||
|      | ||||
|  | ||||
|     PAVBindDevice pDevice; | ||||
|     AVMediaType _mediaType = mediaType == AVBindMediaTypeVideo ? AVMediaTypeVideo : AVMediaTypeAudio; | ||||
|     NSArray *refAllTypes = @[ | ||||
| @@ -196,22 +191,22 @@ STATUS AVBindDevices(AVBindMediaType mediaType, PAVBindDevice *ppDevices, int *p | ||||
|         discoverySessionWithDeviceTypes: refAllTypes | ||||
|         mediaType: _mediaType | ||||
|         position: AVCaptureDevicePositionUnspecified]; | ||||
|      | ||||
|  | ||||
|     int i = 0; | ||||
|     for (AVCaptureDevice *refDevice in refSession.devices) { | ||||
|         if (i >= MAX_DEVICES) { | ||||
|             break; | ||||
|         } | ||||
|          | ||||
|  | ||||
|         pDevice = devices + i; | ||||
|         strncpy(pDevice->uid, refDevice.uniqueID.UTF8String, MAX_DEVICE_UID_CHARS); | ||||
|         pDevice->uid[MAX_DEVICE_UID_CHARS] = '\0'; | ||||
|         i++; | ||||
|     } | ||||
|      | ||||
|  | ||||
|     *ppDevices = devices; | ||||
|     *pLen = i; | ||||
|      | ||||
|  | ||||
| cleanup: | ||||
|     [refPool drain]; | ||||
|     return retStatus; | ||||
| @@ -231,7 +226,7 @@ STATUS AVBindSessionInit(AVBindDevice device, PAVBindSession *ppSessionResult) { | ||||
|     pSession->device = device; | ||||
|     pSession->refCaptureSession = NULL; | ||||
|     *ppSessionResult = pSession; | ||||
|      | ||||
|  | ||||
| cleanup: | ||||
|     return retStatus; | ||||
| } | ||||
| @@ -258,15 +253,15 @@ STATUS AVBindSessionOpen(PAVBindSession pSession, | ||||
|     STATUS retStatus = STATUS_OK; | ||||
|     NSAutoreleasePool *refPool = [[NSAutoreleasePool alloc] init]; | ||||
|     CHK(pSession != NULL && dataCallback != NULL, STATUS_NULL_ARG); | ||||
|      | ||||
|  | ||||
|     AVCaptureDeviceInput *refInput; | ||||
|     NSError *refErr = NULL; | ||||
|     NSString *refUID = [NSString stringWithUTF8String: pSession->device.uid]; | ||||
|     AVCaptureDevice *refDevice = [AVCaptureDevice deviceWithUniqueID: refUID]; | ||||
|      | ||||
|  | ||||
|     refInput = [[AVCaptureDeviceInput alloc] initWithDevice: refDevice error: &refErr]; | ||||
|     CHK(refErr == NULL, STATUS_DEVICE_INIT_FAILED); | ||||
|      | ||||
|  | ||||
|     AVCaptureSession *refCaptureSession = [[AVCaptureSession alloc] init]; | ||||
|     refCaptureSession.sessionPreset = AVCaptureSessionPresetMedium; | ||||
|     [refCaptureSession addInput: refInput]; | ||||
| @@ -275,7 +270,7 @@ STATUS AVBindSessionOpen(PAVBindSession pSession, | ||||
|         VideoDataDelegate *pDelegate = [[VideoDataDelegate alloc] | ||||
|                                         init: dataCallback | ||||
|                                         withUserData: pUserData]; | ||||
|          | ||||
|  | ||||
|         AVCaptureVideoDataOutput *pOutput = [[AVCaptureVideoDataOutput alloc] init]; | ||||
|         FourCharCode fourCC; | ||||
|         CHK_STATUS(frameFormatToFourCC(property.frameFormat, &fourCC)); | ||||
| @@ -293,10 +288,10 @@ STATUS AVBindSessionOpen(PAVBindSession pSession, | ||||
|     } else { | ||||
|         // TODO: implement audio pipeline | ||||
|     } | ||||
|      | ||||
|  | ||||
|     pSession->refCaptureSession = [refCaptureSession retain]; | ||||
|     [refCaptureSession startRunning]; | ||||
|      | ||||
|  | ||||
| cleanup: | ||||
|     [refPool drain]; | ||||
|     return retStatus; | ||||
| @@ -307,11 +302,11 @@ STATUS AVBindSessionClose(PAVBindSession pSession) { | ||||
|     STATUS retStatus = STATUS_OK; | ||||
|     CHK(pSession != NULL, STATUS_NULL_ARG); | ||||
|     CHK(pSession->refCaptureSession != NULL, STATUS_OK); | ||||
|      | ||||
|  | ||||
|     [pSession->refCaptureSession stopRunning]; | ||||
|     [pSession->refCaptureSession release]; | ||||
|     pSession->refCaptureSession = NULL; | ||||
|          | ||||
|  | ||||
| cleanup: | ||||
|     return retStatus; | ||||
| } | ||||
| @@ -330,7 +325,7 @@ STATUS AVBindSessionProperties(PAVBindSession pSession, PAVBindMediaProperty *pp | ||||
|     STATUS retStatus = STATUS_OK; | ||||
|     NSAutoreleasePool *refPool = [[NSAutoreleasePool alloc] init]; | ||||
|     CHK(pSession != NULL && ppProperties != NULL && pLen != NULL, STATUS_NULL_ARG); | ||||
|      | ||||
|  | ||||
|     NSString *refDeviceUID = [NSString stringWithUTF8String: pSession->device.uid]; | ||||
|     AVCaptureDevice *refDevice = [AVCaptureDevice deviceWithUniqueID: refDeviceUID]; | ||||
|     FourCharCode fourCC; | ||||
| @@ -346,14 +341,14 @@ STATUS AVBindSessionProperties(PAVBindSession pSession, PAVBindMediaProperty *pp | ||||
|             NSLog(@"[WARNING] skipping the rest of properties due to MAX_PROPERTIES"); | ||||
|             break; | ||||
|         } | ||||
|          | ||||
|  | ||||
|         if ([refFormat.mediaType isEqual:AVMediaTypeVideo]) { | ||||
|             fourCC = CMFormatDescriptionGetMediaSubType(refFormat.formatDescription); | ||||
|             if (frameFormatFromFourCC(fourCC, &pProperty->frameFormat) != STATUS_OK) { | ||||
|                 NSLog(@"[WARNING] skipping %@ %dx%d since it's not supported", FourCCString(fourCC), videoDimensions.width, videoDimensions.height); | ||||
|                 continue; | ||||
|             } | ||||
|              | ||||
|  | ||||
|             videoFormat = (CMVideoFormatDescriptionRef) refFormat.formatDescription; | ||||
|             videoDimensions = CMVideoFormatDescriptionGetDimensions(videoFormat); | ||||
|             pProperty->height = videoDimensions.height; | ||||
| @@ -361,16 +356,16 @@ STATUS AVBindSessionProperties(PAVBindSession pSession, PAVBindMediaProperty *pp | ||||
|         } else { | ||||
|             // TODO: Get audio properties | ||||
|         } | ||||
|          | ||||
|  | ||||
|         pProperty++; | ||||
|         len++; | ||||
|     } | ||||
|      | ||||
|  | ||||
|     *ppProperties = pSession->properties; | ||||
|     *pLen = len; | ||||
|      | ||||
|  | ||||
| cleanup: | ||||
|      | ||||
|  | ||||
|     [refPool drain]; | ||||
|     return retStatus; | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| package avfoundation | ||||
|  | ||||
| // extern void onData(void*, void*, int); | ||||
| import "C" | ||||
| import ( | ||||
| 	"sync" | ||||
| @@ -18,11 +17,10 @@ type handleID int | ||||
|  | ||||
| //export onData | ||||
| func onData(userData unsafe.Pointer, buf unsafe.Pointer, length C.int) { | ||||
| 	data := C.GoBytes(buf, length) | ||||
|  | ||||
| 	handleNum := (*C.int)(userData) | ||||
| 	cb, ok := lookup(handleID(*handleNum)) | ||||
| 	if ok { | ||||
| 		data := C.GoBytes(buf, length) | ||||
| 		cb(data) | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| package codec | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/io/audio" | ||||
| 	"github.com/pion/mediadevices/pkg/io/video" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| @@ -13,6 +15,9 @@ import ( | ||||
| type RTPCodec struct { | ||||
| 	webrtc.RTPCodecParameters | ||||
| 	rtp.Payloader | ||||
|  | ||||
| 	// Latency of static frame size codec. | ||||
| 	Latency time.Duration | ||||
| } | ||||
|  | ||||
| // NewRTPH264Codec is a helper to create an H264 codec | ||||
| @@ -107,15 +112,37 @@ type VideoEncoderBuilder interface { | ||||
| 	BuildVideoEncoder(r video.Reader, p prop.Media) (ReadCloser, error) | ||||
| } | ||||
|  | ||||
| // ReadCloser is an io.ReadCloser with methods for rate limiting: SetBitRate and ForceKeyFrame | ||||
| // ReadCloser is an io.ReadCloser with a controller | ||||
| type ReadCloser interface { | ||||
| 	Read() (b []byte, release func(), err error) | ||||
| 	Close() error | ||||
| 	Controllable | ||||
| } | ||||
|  | ||||
| // EncoderController is the interface allowing to control the encoder behaviour after it's initialisation. | ||||
| // It will possibly have common control method in the future. | ||||
| // A controller can have optional methods represented by *Controller interfaces | ||||
| type EncoderController interface{} | ||||
|  | ||||
| // Controllable is a interface representing a encoder which can be controlled | ||||
| // after it's initialisation with an EncoderController | ||||
| type Controllable interface { | ||||
| 	Controller() EncoderController | ||||
| } | ||||
|  | ||||
| // KeyFrameController is a interface representing an encoder that can be forced to produce key frame on demand | ||||
| type KeyFrameController interface { | ||||
| 	EncoderController | ||||
| 	// ForceKeyFrame forces the next frame to be a keyframe, aka intra-frame. | ||||
| 	ForceKeyFrame() error | ||||
| } | ||||
|  | ||||
| // BitRateController is a interface representing an encoder which can have a variable bit rate | ||||
| type BitRateController interface { | ||||
| 	EncoderController | ||||
| 	// SetBitRate sets current target bitrate, lower bitrate means smaller data will be transmitted | ||||
| 	// but this also means that the quality will also be lower. | ||||
| 	SetBitRate(int) error | ||||
| 	// ForceKeyFrame forces the next frame to be a keyframe, aka intra-frame. | ||||
| 	ForceKeyFrame() error | ||||
| } | ||||
|  | ||||
| // BaseParams represents an codec's encoding properties | ||||
|   | ||||
							
								
								
									
										121
									
								
								pkg/codec/internal/codectest/codectest.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								pkg/codec/internal/codectest/codectest.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,121 @@ | ||||
| // Package codectest provides shared test for codec implementations. | ||||
| package codectest | ||||
|  | ||||
| import ( | ||||
| 	"image" | ||||
| 	"io" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/io/audio" | ||||
| 	"github.com/pion/mediadevices/pkg/io/video" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| 	"github.com/pion/mediadevices/pkg/wave" | ||||
| ) | ||||
|  | ||||
| func assertNoPanic(t *testing.T, fn func() error, msg string) error { | ||||
| 	defer func() { | ||||
| 		if r := recover(); r != nil { | ||||
| 			t.Errorf("panic: %v: %s", r, msg) | ||||
| 		} | ||||
| 	}() | ||||
| 	return fn() | ||||
| } | ||||
|  | ||||
| func AudioEncoderSimpleReadTest(t *testing.T, c codec.AudioEncoderBuilder, p prop.Media, w wave.Audio) { | ||||
| 	var eof bool | ||||
| 	enc, err := c.BuildAudioEncoder(audio.ReaderFunc(func() (wave.Audio, func(), error) { | ||||
| 		if eof { | ||||
| 			return nil, nil, io.EOF | ||||
| 		} | ||||
| 		return w, nil, nil | ||||
| 	}), p) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	for i := 0; i < 16; i++ { | ||||
| 		b, release, err := enc.Read() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		if len(b) == 0 { | ||||
| 			t.Fatal("Encoded frame is empty") | ||||
| 		} | ||||
| 		release() | ||||
| 	} | ||||
|  | ||||
| 	eof = true | ||||
| 	if _, _, err := enc.Read(); err != io.EOF { | ||||
| 		t.Fatalf("Expected EOF, got %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if err := enc.Close(); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func VideoEncoderSimpleReadTest(t *testing.T, c codec.VideoEncoderBuilder, p prop.Media, img image.Image) { | ||||
| 	var eof bool | ||||
| 	enc, err := c.BuildVideoEncoder(video.ReaderFunc(func() (image.Image, func(), error) { | ||||
| 		if eof { | ||||
| 			return nil, nil, io.EOF | ||||
| 		} | ||||
| 		return img, nil, nil | ||||
| 	}), p) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	for i := 0; i < 16; i++ { | ||||
| 		b, release, err := enc.Read() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		if len(b) == 0 { | ||||
| 			t.Errorf("Encoded frame is empty (%d)", i) | ||||
| 		} | ||||
| 		release() | ||||
| 	} | ||||
|  | ||||
| 	eof = true | ||||
| 	if _, _, err := enc.Read(); err != io.EOF { | ||||
| 		t.Fatalf("Expected EOF, got %v", err) | ||||
| 	} | ||||
|  | ||||
| 	if err := enc.Close(); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func AudioEncoderCloseTwiceTest(t *testing.T, c codec.AudioEncoderBuilder, p prop.Media) { | ||||
| 	enc, err := c.BuildAudioEncoder(audio.ReaderFunc(func() (wave.Audio, func(), error) { | ||||
| 		return nil, nil, io.EOF | ||||
| 	}), p) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	if err := assertNoPanic(t, enc.Close, "on first Close()"); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if err := assertNoPanic(t, enc.Close, "on second Close()"); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func VideoEncoderCloseTwiceTest(t *testing.T, c codec.VideoEncoderBuilder, p prop.Media) { | ||||
| 	enc, err := c.BuildVideoEncoder(video.ReaderFunc(func() (image.Image, func(), error) { | ||||
| 		return nil, nil, io.EOF | ||||
| 	}), p) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	if err := assertNoPanic(t, enc.Close, "on first Close()"); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if err := assertNoPanic(t, enc.Close, "on second Close()"); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| } | ||||
| @@ -91,12 +91,8 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, err | ||||
| } | ||||
|  | ||||
| func (e *encoder) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| } | ||||
|  | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoder) Close() error { | ||||
|   | ||||
							
								
								
									
										65
									
								
								pkg/codec/mmal/mmal_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								pkg/codec/mmal/mmal_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| package mmal | ||||
|  | ||||
| import ( | ||||
| 	"image" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	t.Run("SimpleRead", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.VideoEncoderSimpleReadTest(t, &p, | ||||
| 			prop.Media{ | ||||
| 				Video: prop.Video{ | ||||
| 					Width:       256, | ||||
| 					Height:      144, | ||||
| 					FrameFormat: frame.FormatI420, | ||||
| 				}, | ||||
| 			}, | ||||
| 			image.NewYCbCr( | ||||
| 				image.Rect(0, 0, 256, 144), | ||||
| 				image.YCbCrSubsampleRatio420, | ||||
| 			), | ||||
| 		) | ||||
| 	}) | ||||
| 	t.Run("CloseTwice", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.VideoEncoderCloseTwiceTest(t, &p, prop.Media{ | ||||
| 			Video: prop.Video{ | ||||
| 				Width:       640, | ||||
| 				Height:      480, | ||||
| 				FrameRate:   30, | ||||
| 				FrameFormat: frame.FormatI420, | ||||
| 			}, | ||||
| 		}) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement key frame control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
| @@ -80,6 +80,11 @@ Slice enc_encode(Encoder *e, Frame f, int *eresult) { | ||||
|   SFrameBSInfo info = {0}; | ||||
|   Slice payload = {0}; | ||||
|  | ||||
|   if(e->force_key_frame == 1) { | ||||
|     info.eFrameType = videoFrameTypeI; | ||||
|     e->force_key_frame = 0; | ||||
|   } | ||||
|  | ||||
|   pic.iPicWidth = f.width; | ||||
|   pic.iPicHeight = f.height; | ||||
|   pic.iColorFormat = videoFormatI420; | ||||
|   | ||||
| @@ -27,6 +27,7 @@ typedef struct Encoder { | ||||
|   ISVCEncoder *engine; | ||||
|   unsigned char *buff; | ||||
|   int buff_size; | ||||
|   int force_key_frame; | ||||
| } Encoder; | ||||
|  | ||||
| Encoder *enc_new(const EncoderOptions params, int *eresult); | ||||
|   | ||||
| @@ -79,18 +79,23 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, nil | ||||
| } | ||||
|  | ||||
| func (e *encoder) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	e.engine.force_key_frame = C.int(1) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoder) Close() error { | ||||
| 	e.mu.Lock() | ||||
| 	defer e.mu.Unlock() | ||||
|  | ||||
| 	if e.closed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	e.closed = true | ||||
|  | ||||
| 	var rv C.int | ||||
|   | ||||
							
								
								
									
										63
									
								
								pkg/codec/openh264/openh264_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								pkg/codec/openh264/openh264_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| package openh264 | ||||
|  | ||||
| import ( | ||||
| 	"image" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	t.Run("SimpleRead", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.VideoEncoderSimpleReadTest(t, &p, | ||||
| 			prop.Media{ | ||||
| 				Video: prop.Video{ | ||||
| 					Width:       256, | ||||
| 					Height:      144, | ||||
| 					FrameFormat: frame.FormatI420, | ||||
| 				}, | ||||
| 			}, | ||||
| 			image.NewYCbCr( | ||||
| 				image.Rect(0, 0, 256, 144), | ||||
| 				image.YCbCrSubsampleRatio420, | ||||
| 			), | ||||
| 		) | ||||
| 	}) | ||||
| 	t.Run("CloseTwice", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.VideoEncoderCloseTwiceTest(t, &p, prop.Media{ | ||||
| 			Video: prop.Video{ | ||||
| 				Width:       640, | ||||
| 				Height:      480, | ||||
| 				FrameRate:   30, | ||||
| 				FrameFormat: frame.FormatI420, | ||||
| 			}, | ||||
| 		}) | ||||
| 	}) | ||||
| } | ||||
| @@ -3,7 +3,6 @@ package opus | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"math" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/io/audio" | ||||
| @@ -28,8 +27,6 @@ type encoder struct { | ||||
| 	engine *C.OpusEncoder | ||||
| } | ||||
|  | ||||
| var latencies = []float64{5, 10, 20, 40, 60} | ||||
|  | ||||
| func newEncoder(r audio.Reader, p prop.Media, params Params) (codec.ReadCloser, error) { | ||||
| 	var cerror C.int | ||||
|  | ||||
| @@ -37,10 +34,6 @@ func newEncoder(r audio.Reader, p prop.Media, params Params) (codec.ReadCloser, | ||||
| 		return nil, fmt.Errorf("opus: inProp.SampleRate is required") | ||||
| 	} | ||||
|  | ||||
| 	if p.Latency == 0 { | ||||
| 		p.Latency = 20 | ||||
| 	} | ||||
|  | ||||
| 	if params.BitRate == 0 { | ||||
| 		params.BitRate = 32000 | ||||
| 	} | ||||
| @@ -49,19 +42,8 @@ func newEncoder(r audio.Reader, p prop.Media, params Params) (codec.ReadCloser, | ||||
| 		params.ChannelMixer = &mixer.MonoMixer{} | ||||
| 	} | ||||
|  | ||||
| 	// Select the nearest supported latency | ||||
| 	var targetLatency float64 | ||||
| 	// TODO: use p.Latency.Milliseconds() after Go 1.12 EOL | ||||
| 	latencyInMS := float64(p.Latency.Nanoseconds() / 1000000) | ||||
| 	nearestDist := math.Inf(+1) | ||||
| 	for _, latency := range latencies { | ||||
| 		dist := math.Abs(latency - latencyInMS) | ||||
| 		if dist >= nearestDist { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		nearestDist = dist | ||||
| 		targetLatency = latency | ||||
| 	if !params.Latency.Validate() { | ||||
| 		return nil, fmt.Errorf("opus: unsupported latency %v", params.Latency) | ||||
| 	} | ||||
|  | ||||
| 	channels := p.ChannelCount | ||||
| @@ -77,7 +59,7 @@ func newEncoder(r audio.Reader, p prop.Media, params Params) (codec.ReadCloser, | ||||
| 	} | ||||
|  | ||||
| 	rMix := audio.NewChannelMixer(channels, params.ChannelMixer) | ||||
| 	rBuf := audio.NewBuffer(int(targetLatency * float64(p.SampleRate) / 1000)) | ||||
| 	rBuf := audio.NewBuffer(params.Latency.samples(p.SampleRate)) | ||||
| 	e := encoder{ | ||||
| 		engine: engine, | ||||
| 		reader: rMix(rBuf(r)), | ||||
| @@ -139,11 +121,14 @@ func (e *encoder) SetBitRate(bitRate int) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoder) Close() error { | ||||
| 	if e.engine == nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	C.opus_encoder_destroy(e.engine) | ||||
| 	e.engine = nil | ||||
| 	return nil | ||||
|   | ||||
							
								
								
									
										60
									
								
								pkg/codec/opus/opus_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								pkg/codec/opus/opus_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| package opus | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| 	"github.com/pion/mediadevices/pkg/wave" | ||||
| ) | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement key frame control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	t.Run("SimpleRead", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.AudioEncoderSimpleReadTest(t, &p, | ||||
| 			prop.Media{ | ||||
| 				Audio: prop.Audio{ | ||||
| 					SampleRate:   48000, | ||||
| 					ChannelCount: 2, | ||||
| 				}, | ||||
| 			}, | ||||
| 			wave.NewInt16Interleaved(wave.ChunkInfo{ | ||||
| 				Len:          960, | ||||
| 				SamplingRate: 48000, | ||||
| 				Channels:     2, | ||||
| 			}), | ||||
| 		) | ||||
| 	}) | ||||
| 	t.Run("CloseTwice", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		codectest.AudioEncoderCloseTwiceTest(t, &p, prop.Media{ | ||||
| 			Audio: prop.Audio{ | ||||
| 				SampleRate:   48000, | ||||
| 				ChannelCount: 2, | ||||
| 			}, | ||||
| 		}) | ||||
| 	}) | ||||
| } | ||||
| @@ -1,27 +1,69 @@ | ||||
| package opus | ||||
|  | ||||
| import ( | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/io/audio" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| 	"github.com/pion/mediadevices/pkg/wave/mixer" | ||||
| ) | ||||
|  | ||||
| // Latency is a type of OPUS codec frame duration. | ||||
| type Latency time.Duration | ||||
|  | ||||
| // Latency values available in OPUS codec. | ||||
| const ( | ||||
| 	Latency2500us Latency = Latency(2500 * time.Microsecond) | ||||
| 	Latency5ms    Latency = Latency(5 * time.Millisecond) | ||||
| 	Latency10ms   Latency = Latency(10 * time.Millisecond) | ||||
| 	Latency20ms   Latency = Latency(20 * time.Millisecond) | ||||
| 	Latency40ms   Latency = Latency(40 * time.Millisecond) | ||||
| 	Latency60ms   Latency = Latency(60 * time.Millisecond) | ||||
| ) | ||||
|  | ||||
| // Validate that the Latency is allowed in OPUS. | ||||
| func (l Latency) Validate() bool { | ||||
| 	switch l { | ||||
| 	case Latency2500us, Latency5ms, Latency10ms, Latency20ms, Latency40ms, Latency60ms: | ||||
| 		return true | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Duration returns latency in time.Duration. | ||||
| func (l Latency) Duration() time.Duration { | ||||
| 	return time.Duration(l) | ||||
| } | ||||
|  | ||||
| // samples returns number of samples for given sample rate. | ||||
| func (l Latency) samples(sampleRate int) int { | ||||
| 	return int(l.Duration() * time.Duration(sampleRate) / time.Second) | ||||
| } | ||||
|  | ||||
| // Params stores opus specific encoding parameters. | ||||
| type Params struct { | ||||
| 	codec.BaseParams | ||||
| 	// ChannelMixer is a mixer to be used if number of given and expected channels differ. | ||||
| 	ChannelMixer mixer.ChannelMixer | ||||
|  | ||||
| 	// Expected latency of the codec. | ||||
| 	Latency Latency | ||||
| } | ||||
|  | ||||
| // NewParams returns default opus codec specific parameters. | ||||
| func NewParams() (Params, error) { | ||||
| 	return Params{}, nil | ||||
| 	return Params{ | ||||
| 		Latency: Latency20ms, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| // RTPCodec represents the codec metadata | ||||
| func (p *Params) RTPCodec() *codec.RTPCodec { | ||||
| 	return codec.NewRTPOpusCodec(48000) | ||||
| 	c := codec.NewRTPOpusCodec(48000) | ||||
| 	c.Latency = time.Duration(p.Latency) | ||||
| 	return c | ||||
| } | ||||
|  | ||||
| // BuildAudioEncoder builds opus encoder with given params | ||||
|   | ||||
							
								
								
									
										49
									
								
								pkg/codec/opus/params_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								pkg/codec/opus/params_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| package opus | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestLatency_Validate(t *testing.T) { | ||||
| 	t.Run("Valid", func(t *testing.T) { | ||||
| 		for _, l := range []Latency{ | ||||
| 			Latency2500us, Latency5ms, Latency10ms, Latency20ms, Latency40ms, Latency60ms, | ||||
| 		} { | ||||
| 			if !l.Validate() { | ||||
| 				t.Errorf("Defined Latency(%v) must be valid", l) | ||||
| 			} | ||||
| 		} | ||||
| 	}) | ||||
| 	t.Run("Invalid", func(t *testing.T) { | ||||
| 		for _, l := range []Latency{ | ||||
| 			0, Latency(time.Second), | ||||
| 		} { | ||||
| 			if l.Validate() { | ||||
| 				t.Errorf("Latency(%v) must be valid", l) | ||||
| 			} | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestLatency_samples(t *testing.T) { | ||||
| 	testCases := []struct { | ||||
| 		latency    Latency | ||||
| 		sampleRate int | ||||
| 		samples    int | ||||
| 	}{ | ||||
| 		{Latency5ms, 48000, 240}, | ||||
| 		{Latency20ms, 16000, 320}, | ||||
| 		{Latency20ms, 48000, 960}, | ||||
| 	} | ||||
| 	for _, testCase := range testCases { | ||||
| 		testCase := testCase | ||||
| 		t.Run(fmt.Sprintf("%v_%d", time.Duration(testCase.latency), testCase.sampleRate), func(t *testing.T) { | ||||
| 			samples := testCase.latency.samples(testCase.sampleRate) | ||||
| 			if samples != testCase.samples { | ||||
| 				t.Errorf("Expected samples: %d, got: %d", testCase.samples, samples) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -1,3 +1,4 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| // Package vaapi implements hardware accelerated codecs. | ||||
|   | ||||
							
								
								
									
										70
									
								
								pkg/codec/vaapi/vaapi_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								pkg/codec/vaapi/vaapi_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| package vaapi | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"image" | ||||
| 	"os" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	if _, err := os.Stat("/dev/dri/card0"); errors.Is(err, os.ErrNotExist) { | ||||
| 		t.Skip("/dev/dri/card0 not found") | ||||
| 	} | ||||
|  | ||||
| 	for name, factory := range map[string]func() (codec.VideoEncoderBuilder, error){ | ||||
| 		"VP8": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP8Params() | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 		"VP9": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP9Params() | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 	} { | ||||
| 		factory := factory | ||||
| 		t.Run(name, func(t *testing.T) { | ||||
| 			t.Run("SimpleRead", func(t *testing.T) { | ||||
| 				p, err := factory() | ||||
| 				if err != nil { | ||||
| 					t.Fatal(err) | ||||
| 				} | ||||
| 				codectest.VideoEncoderSimpleReadTest(t, p, | ||||
| 					prop.Media{ | ||||
| 						Video: prop.Video{ | ||||
| 							Width:       256, | ||||
| 							Height:      144, | ||||
| 							FrameFormat: frame.FormatI420, | ||||
| 						}, | ||||
| 					}, | ||||
| 					image.NewYCbCr( | ||||
| 						image.Rect(0, 0, 256, 144), | ||||
| 						image.YCbCrSubsampleRatio420, | ||||
| 					), | ||||
| 				) | ||||
| 			}) | ||||
| 			t.Run("CloseTwice", func(t *testing.T) { | ||||
| 				p, err := factory() | ||||
| 				if err != nil { | ||||
| 					t.Fatal(err) | ||||
| 				} | ||||
| 				codectest.VideoEncoderCloseTwiceTest(t, p, prop.Media{ | ||||
| 					Video: prop.Video{ | ||||
| 						Width:       640, | ||||
| 						Height:      480, | ||||
| 						FrameRate:   30, | ||||
| 						FrameFormat: frame.FormatI420, | ||||
| 					}, | ||||
| 				}) | ||||
| 			}) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
| @@ -1,3 +1,4 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| package vaapi | ||||
| @@ -540,18 +541,18 @@ func (e *encoderVP8) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, err | ||||
| } | ||||
|  | ||||
| func (e *encoderVP8) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| } | ||||
|  | ||||
| func (e *encoderVP8) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoderVP8) Close() error { | ||||
| 	e.mu.Lock() | ||||
| 	defer e.mu.Unlock() | ||||
|  | ||||
| 	if e.closed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	C.vaDestroySurfaces(e.display, &e.surfs[0], C.int(len(e.surfs))) | ||||
| 	C.vaDestroyContext(e.display, e.ctxID) | ||||
| 	C.vaDestroyConfig(e.display, e.confID) | ||||
|   | ||||
							
								
								
									
										27
									
								
								pkg/codec/vaapi/vp8_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								pkg/codec/vaapi/vp8_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| package vaapi | ||||
|  | ||||
| import ( | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoderVP8{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement key frame control | ||||
|  | ||||
| 	e := &encoderVP8{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
| @@ -1,3 +1,4 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| package vaapi | ||||
| @@ -475,18 +476,18 @@ func (e *encoderVP9) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, err | ||||
| } | ||||
|  | ||||
| func (e *encoderVP9) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| } | ||||
|  | ||||
| func (e *encoderVP9) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoderVP9) Close() error { | ||||
| 	e.mu.Lock() | ||||
| 	defer e.mu.Unlock() | ||||
|  | ||||
| 	if e.closed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	C.vaDestroySurfaces(e.display, &e.surfs[0], C.int(len(e.surfs))) | ||||
| 	C.vaDestroyContext(e.display, e.ctxID) | ||||
| 	C.vaDestroyConfig(e.display, e.confID) | ||||
|   | ||||
							
								
								
									
										22
									
								
								pkg/codec/vaapi/vp9_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								pkg/codec/vaapi/vp9_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| //go:build dragonfly || freebsd || linux || netbsd || openbsd || solaris | ||||
| // +build dragonfly freebsd linux netbsd openbsd solaris | ||||
|  | ||||
| package vaapi | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoderVP9{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement key frame control | ||||
|  | ||||
| 	e := &encoderVP9{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
| @@ -16,6 +16,7 @@ type Params struct { | ||||
| 	RateControlOvershootPercent  uint | ||||
| 	RateControlMinQuantizer      uint | ||||
| 	RateControlMaxQuantizer      uint | ||||
| 	LagInFrames                  uint | ||||
| 	ErrorResilient               ErrorResilientMode | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -23,6 +23,9 @@ package vpx | ||||
| // int pktSz(vpx_codec_cx_pkt_t *pkt) { | ||||
| //   return pkt->data.frame.sz; | ||||
| // } | ||||
| // vpx_codec_frame_flags_t pktFrameFlags(vpx_codec_cx_pkt_t *pkt) { | ||||
| //   return pkt->data.frame.flags; | ||||
| // } | ||||
| // | ||||
| // // Alloc helpers | ||||
| // vpx_codec_ctx_t *newCtx() { | ||||
| @@ -61,15 +64,17 @@ import ( | ||||
| ) | ||||
|  | ||||
| type encoder struct { | ||||
| 	codec      *C.vpx_codec_ctx_t | ||||
| 	raw        *C.vpx_image_t | ||||
| 	cfg        *C.vpx_codec_enc_cfg_t | ||||
| 	r          video.Reader | ||||
| 	frameIndex int | ||||
| 	tStart     int | ||||
| 	tLastFrame int | ||||
| 	frame      []byte | ||||
| 	deadline   int | ||||
| 	codec           *C.vpx_codec_ctx_t | ||||
| 	raw             *C.vpx_image_t | ||||
| 	cfg             *C.vpx_codec_enc_cfg_t | ||||
| 	r               video.Reader | ||||
| 	frameIndex      int | ||||
| 	tStart          int | ||||
| 	tLastFrame      int | ||||
| 	frame           []byte | ||||
| 	deadline        int | ||||
| 	requireKeyFrame bool | ||||
| 	isKeyFrame      bool | ||||
|  | ||||
| 	mu     sync.Mutex | ||||
| 	closed bool | ||||
| @@ -141,6 +146,7 @@ func newParams(codecIface *C.vpx_codec_iface_t) (Params, error) { | ||||
| 		RateControlOvershootPercent:  uint(cfg.rc_overshoot_pct), | ||||
| 		RateControlMinQuantizer:      uint(cfg.rc_min_quantizer), | ||||
| 		RateControlMaxQuantizer:      uint(cfg.rc_max_quantizer), | ||||
| 		LagInFrames:                  uint(cfg.g_lag_in_frames), | ||||
| 		ErrorResilient:               ErrorResilientMode(cfg.g_error_resilient), | ||||
| 	}, nil | ||||
| } | ||||
| @@ -171,6 +177,7 @@ func newEncoder(r video.Reader, p prop.Media, params Params, codecIface *C.vpx_c | ||||
| 	cfg.g_h = C.uint(p.Height) | ||||
| 	cfg.g_timebase.num = 1 | ||||
| 	cfg.g_timebase.den = 1000 | ||||
| 	cfg.g_lag_in_frames = C.uint(params.LagInFrames) | ||||
| 	cfg.rc_target_bitrate = C.uint(params.BitRate) / 1000 | ||||
| 	cfg.kf_max_dist = C.uint(params.KeyFrameInterval) | ||||
|  | ||||
| @@ -254,6 +261,9 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 		duration = 1 | ||||
| 	} | ||||
| 	var flags int | ||||
| 	if e.requireKeyFrame { | ||||
| 		flags = flags | C.VPX_EFLAG_FORCE_KF | ||||
| 	} | ||||
| 	if ec := C.encode_wrapper( | ||||
| 		e.codec, e.raw, | ||||
| 		C.long(t-e.tStart), C.ulong(duration), C.long(flags), C.ulong(e.deadline), | ||||
| @@ -262,6 +272,7 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 		return nil, func() {}, fmt.Errorf("vpx_codec_encode failed (%d)", ec) | ||||
| 	} | ||||
|  | ||||
| 	e.requireKeyFrame = false | ||||
| 	e.frameIndex++ | ||||
| 	e.tLastFrame = t | ||||
|  | ||||
| @@ -273,6 +284,7 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 			break | ||||
| 		} | ||||
| 		if pkt.kind == C.VPX_CODEC_CX_FRAME_PKT { | ||||
| 			e.isKeyFrame = C.pktFrameFlags(pkt)&C.VPX_FRAME_IS_KEY == C.VPX_FRAME_IS_KEY | ||||
| 			encoded := C.GoBytes(unsafe.Pointer(C.pktBuf(pkt)), C.pktSz(pkt)) | ||||
| 			e.frame = append(e.frame, encoded...) | ||||
| 		} | ||||
| @@ -283,18 +295,25 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, err | ||||
| } | ||||
|  | ||||
| func (e *encoder) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	e.mu.Lock() | ||||
| 	defer e.mu.Unlock() | ||||
| 	e.requireKeyFrame = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoder) Close() error { | ||||
| 	e.mu.Lock() | ||||
| 	defer e.mu.Unlock() | ||||
|  | ||||
| 	if e.closed { | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
| 	e.closed = true | ||||
|  | ||||
| 	C.free(unsafe.Pointer(e.raw)) | ||||
|   | ||||
| @@ -7,11 +7,63 @@ import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/io/video" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	for name, factory := range map[string]func() (codec.VideoEncoderBuilder, error){ | ||||
| 		"VP8": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP8Params() | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 		"VP9": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP9Params() | ||||
| 			p.LagInFrames = 0 | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 	} { | ||||
| 		factory := factory | ||||
| 		t.Run(name, func(t *testing.T) { | ||||
| 			t.Run("SimpleRead", func(t *testing.T) { | ||||
| 				p, err := factory() | ||||
| 				if err != nil { | ||||
| 					t.Fatal(err) | ||||
| 				} | ||||
| 				codectest.VideoEncoderSimpleReadTest(t, p, | ||||
| 					prop.Media{ | ||||
| 						Video: prop.Video{ | ||||
| 							Width:       256, | ||||
| 							Height:      144, | ||||
| 							FrameFormat: frame.FormatI420, | ||||
| 						}, | ||||
| 					}, | ||||
| 					image.NewYCbCr( | ||||
| 						image.Rect(0, 0, 256, 144), | ||||
| 						image.YCbCrSubsampleRatio420, | ||||
| 					), | ||||
| 				) | ||||
| 			}) | ||||
| 			t.Run("CloseTwice", func(t *testing.T) { | ||||
| 				p, err := factory() | ||||
| 				if err != nil { | ||||
| 					t.Fatal(err) | ||||
| 				} | ||||
| 				codectest.VideoEncoderCloseTwiceTest(t, p, prop.Media{ | ||||
| 					Video: prop.Video{ | ||||
| 						Width:       640, | ||||
| 						Height:      480, | ||||
| 						FrameRate:   30, | ||||
| 						FrameFormat: frame.FormatI420, | ||||
| 					}, | ||||
| 				}) | ||||
| 			}) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestImageSizeChange(t *testing.T) { | ||||
| 	for name, factory := range map[string]func() (codec.VideoEncoderBuilder, error){ | ||||
| 		"VP8": func() (codec.VideoEncoderBuilder, error) { | ||||
| @@ -20,6 +72,8 @@ func TestImageSizeChange(t *testing.T) { | ||||
| 		}, | ||||
| 		"VP9": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP9Params() | ||||
| 			// Disable latency to ease test and begin to receive packets for each input frame | ||||
| 			p.LagInFrames = 0 | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 	} { | ||||
| @@ -87,3 +141,88 @@ func TestImageSizeChange(t *testing.T) { | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestRequestKeyFrame(t *testing.T) { | ||||
| 	for name, factory := range map[string]func() (codec.VideoEncoderBuilder, error){ | ||||
| 		"VP8": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP8Params() | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 		"VP9": func() (codec.VideoEncoderBuilder, error) { | ||||
| 			p, err := NewVP9Params() | ||||
| 			// Disable latency to ease test and begin to receive packets for each input frame | ||||
| 			p.LagInFrames = 0 | ||||
| 			return &p, err | ||||
| 		}, | ||||
| 	} { | ||||
| 		factory := factory | ||||
| 		t.Run(name, func(t *testing.T) { | ||||
| 			param, err := factory() | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
|  | ||||
| 			var initialWidth, initialHeight, width, height int = 320, 240, 320, 240 | ||||
|  | ||||
| 			var cnt uint32 | ||||
| 			r, err := param.BuildVideoEncoder( | ||||
| 				video.ReaderFunc(func() (image.Image, func(), error) { | ||||
| 					i := atomic.AddUint32(&cnt, 1) | ||||
| 					if i == 3 { | ||||
| 						return nil, nil, io.EOF | ||||
| 					} | ||||
| 					return image.NewYCbCr( | ||||
| 						image.Rect(0, 0, width, height), | ||||
| 						image.YCbCrSubsampleRatio420, | ||||
| 					), func() {}, nil | ||||
| 				}), | ||||
| 				prop.Media{ | ||||
| 					Video: prop.Video{ | ||||
| 						Width:       initialWidth, | ||||
| 						Height:      initialHeight, | ||||
| 						FrameRate:   1, | ||||
| 						FrameFormat: frame.FormatI420, | ||||
| 					}, | ||||
| 				}, | ||||
| 			) | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 			_, rel, err := r.Read() | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 			rel() | ||||
| 			r.Controller().(codec.KeyFrameController).ForceKeyFrame() | ||||
| 			_, rel, err = r.Read() | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 			if !r.(*encoder).isKeyFrame { | ||||
| 				t.Fatal("Not a key frame") | ||||
| 			} | ||||
| 			rel() | ||||
| 			_, _, err = r.Read() | ||||
| 			if err != io.EOF { | ||||
| 				t.Fatal(err) | ||||
| 			} | ||||
| 		}) | ||||
|  | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -124,12 +124,8 @@ func (e *encoder) Read() ([]byte, func(), error) { | ||||
| 	return encoded, func() {}, err | ||||
| } | ||||
|  | ||||
| func (e *encoder) SetBitRate(b int) error { | ||||
| 	panic("SetBitRate is not implemented") | ||||
| } | ||||
|  | ||||
| func (e *encoder) ForceKeyFrame() error { | ||||
| 	panic("ForceKeyFrame is not implemented") | ||||
| func (e *encoder) Controller() codec.EncoderController { | ||||
| 	return e | ||||
| } | ||||
|  | ||||
| func (e *encoder) Close() error { | ||||
|   | ||||
							
								
								
									
										67
									
								
								pkg/codec/x264/x264_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								pkg/codec/x264/x264_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | ||||
| package x264 | ||||
|  | ||||
| import ( | ||||
| 	"image" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/mediadevices/pkg/codec/internal/codectest" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestEncoder(t *testing.T) { | ||||
| 	t.Run("SimpleRead", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		p.BitRate = 200000 | ||||
| 		codectest.VideoEncoderSimpleReadTest(t, &p, | ||||
| 			prop.Media{ | ||||
| 				Video: prop.Video{ | ||||
| 					Width:       256, | ||||
| 					Height:      144, | ||||
| 					FrameFormat: frame.FormatI420, | ||||
| 				}, | ||||
| 			}, | ||||
| 			image.NewYCbCr( | ||||
| 				image.Rect(0, 0, 256, 144), | ||||
| 				image.YCbCrSubsampleRatio420, | ||||
| 			), | ||||
| 		) | ||||
| 	}) | ||||
| 	t.Run("CloseTwice", func(t *testing.T) { | ||||
| 		p, err := NewParams() | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		p.BitRate = 200000 | ||||
| 		codectest.VideoEncoderCloseTwiceTest(t, &p, prop.Media{ | ||||
| 			Video: prop.Video{ | ||||
| 				Width:       640, | ||||
| 				Height:      480, | ||||
| 				FrameRate:   30, | ||||
| 				FrameFormat: frame.FormatI420, | ||||
| 			}, | ||||
| 		}) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestShouldImplementKeyFrameControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement key frame control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.KeyFrameController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestShouldImplementBitRateControl(t *testing.T) { | ||||
| 	t.SkipNow() // TODO: Implement bit rate control | ||||
|  | ||||
| 	e := &encoder{} | ||||
| 	if _, ok := e.Controller().(codec.BitRateController); !ok { | ||||
| 		t.Error() | ||||
| 	} | ||||
| } | ||||
| @@ -1 +1,16 @@ | ||||
| /* | ||||
| Package camera provides a video camera driver. | ||||
|  | ||||
| Device Label Generation Rules | ||||
|  | ||||
| On Linux, the device label will be in the format of: | ||||
| 	pci-0000:00:00.0-usb-0:0:0.0-video-index0;video0 | ||||
| If /dev/v4l/by-path/* is not available (for example in a docker container without | ||||
| bindings in /dev/v4l/by-path/), it will be: | ||||
| 	video0;video0 | ||||
| */ | ||||
| package camera | ||||
|  | ||||
| // LabelSeparator is used to separate labels for a driver that | ||||
| // is found from multiple locations on a host. | ||||
| const LabelSeparator = ";" | ||||
|   | ||||
| @@ -56,7 +56,7 @@ func BenchmarkRead(b *testing.B) { | ||||
|  | ||||
| 			b.ResetTimer() | ||||
| 			for i := 0; i < b.N; i++ { | ||||
| 				_, err := r.Read() | ||||
| 				_, _, err := r.Read() | ||||
| 				if err != nil { | ||||
| 					b.Fatalf("Failed to read: %v", err) | ||||
| 				} | ||||
|   | ||||
							
								
								
									
										63
									
								
								pkg/driver/camera/camera_darwin_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								pkg/driver/camera/camera_darwin_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| // +build darwin | ||||
|  | ||||
| // $ go test -v . -tags darwin -run="^TestCameraFrameFormatSupport$" | ||||
|  | ||||
| package camera | ||||
|  | ||||
| import ( | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/avfoundation" | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| func TestCameraFrameFormatSupport(t *testing.T) { | ||||
| 	devices, err := avfoundation.Devices(avfoundation.Video) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if len(devices) > 0 { | ||||
| 		c := newCamera(devices[0]) | ||||
| 		if err := c.Open(); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		defer c.Close() | ||||
|  | ||||
| 		supportedFormats := make(map[frame.Format]struct{}) | ||||
| 		for _, p := range c.Properties() { | ||||
| 			supportedFormats[p.FrameFormat] = struct{}{} | ||||
| 		} | ||||
|  | ||||
| 		for _, format := range []frame.Format{ | ||||
| 			frame.FormatI420, | ||||
| 			frame.FormatNV12, | ||||
| 			frame.FormatNV21, | ||||
| 			frame.FormatYUY2, | ||||
| 			frame.FormatUYVY, | ||||
| 		} { | ||||
| 			if _, ok := supportedFormats[format]; !ok { | ||||
| 				t.Logf("[%v] UNSUPPORTED", format) | ||||
| 				continue | ||||
| 			} | ||||
| 			r, err := c.VideoRecord(prop.Media{ | ||||
| 				Video: prop.Video{ | ||||
| 					Width:       640, | ||||
| 					Height:      480, | ||||
| 					FrameFormat: format, | ||||
| 				}}) | ||||
| 			if err != nil { | ||||
| 				t.Logf("[%v] Failed to capture image: %v", format, err) | ||||
| 				continue | ||||
| 			} | ||||
| 			for i := 0; i < 10; i++ { | ||||
| 				_, _, err := r.Read() | ||||
| 				if err != nil { | ||||
| 					t.Logf("[%v] Failed to read: %v", format, err) | ||||
| 					continue | ||||
| 				} | ||||
| 			} | ||||
| 			t.Logf("[%v] OK", format) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @@ -73,51 +73,51 @@ type camera struct { | ||||
|  | ||||
| func init() { | ||||
| 	discovered := make(map[string]struct{}) | ||||
| 	discover(discovered, "/dev/v4l/by-path/*") | ||||
| 	discover(discovered, "/dev/video*") | ||||
| } | ||||
|  | ||||
| 	discover := func(pattern string) { | ||||
| 		devices, err := filepath.Glob(pattern) | ||||
| 		if err != nil { | ||||
| 			// No v4l device. | ||||
| 			return | ||||
| 		} | ||||
| 		for _, device := range devices { | ||||
| 			label := filepath.Base(device) | ||||
| 			reallink, err := os.Readlink(device) | ||||
| 			if err != nil { | ||||
| 				reallink = label | ||||
| 			} else { | ||||
| 				reallink = filepath.Base(reallink) | ||||
| 			} | ||||
|  | ||||
| 			if _, ok := discovered[reallink]; ok { | ||||
| 				continue | ||||
| 			} | ||||
|  | ||||
| 			discovered[reallink] = struct{}{} | ||||
| 			cam := newCamera(device) | ||||
| 			priority := driver.PriorityNormal | ||||
| 			if label == prioritizedDevice { | ||||
| 				priority = driver.PriorityHigh | ||||
| 			} | ||||
| 			driver.GetManager().Register(cam, driver.Info{ | ||||
| 				Label:      label, | ||||
| 				DeviceType: driver.Camera, | ||||
| 				Priority:   priority, | ||||
| 			}) | ||||
| 		} | ||||
| func discover(discovered map[string]struct{}, pattern string) { | ||||
| 	devices, err := filepath.Glob(pattern) | ||||
| 	if err != nil { | ||||
| 		// No v4l device. | ||||
| 		return | ||||
| 	} | ||||
| 	for _, device := range devices { | ||||
| 		label := filepath.Base(device) | ||||
| 		reallink, err := os.Readlink(device) | ||||
| 		if err != nil { | ||||
| 			reallink = label | ||||
| 		} else { | ||||
| 			reallink = filepath.Base(reallink) | ||||
| 		} | ||||
| 		if _, ok := discovered[reallink]; ok { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 	discover("/dev/v4l/by-path/*") | ||||
| 	discover("/dev/video*") | ||||
| 		discovered[reallink] = struct{}{} | ||||
| 		cam := newCamera(device) | ||||
| 		priority := driver.PriorityNormal | ||||
| 		if reallink == prioritizedDevice { | ||||
| 			priority = driver.PriorityHigh | ||||
| 		} | ||||
| 		driver.GetManager().Register(cam, driver.Info{ | ||||
| 			Label:      label + LabelSeparator + reallink, | ||||
| 			DeviceType: driver.Camera, | ||||
| 			Priority:   priority, | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func newCamera(path string) *camera { | ||||
| 	formats := map[webcam.PixelFormat]frame.Format{ | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_YUV420): frame.FormatI420, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_NV21):   frame.FormatNV21, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_NV12):   frame.FormatNV12, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_YUYV):   frame.FormatYUYV, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_UYVY):   frame.FormatUYVY, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_NV12):   frame.FormatNV21, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_MJPEG):  frame.FormatMJPEG, | ||||
| 		webcam.PixelFormat(C.V4L2_PIX_FMT_Z16):    frame.FormatZ16, | ||||
| 	} | ||||
|  | ||||
| 	reversedFormats := make(map[frame.Format]webcam.PixelFormat) | ||||
|   | ||||
							
								
								
									
										72
									
								
								pkg/driver/camera/camera_linux_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								pkg/driver/camera/camera_linux_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | ||||
| package camera | ||||
|  | ||||
| import ( | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/driver" | ||||
| ) | ||||
|  | ||||
| func TestDiscover(t *testing.T) { | ||||
| 	const ( | ||||
| 		shortName  = "unittest-video0" | ||||
| 		shortName2 = "unittest-video1" | ||||
| 		longName   = "unittest-long-device-name:0:1:2:3" | ||||
| 	) | ||||
|  | ||||
| 	dir, err := ioutil.TempDir("", "") | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	defer os.RemoveAll(dir) | ||||
|  | ||||
| 	byPathDir := filepath.Join(dir, "v4l", "by-path") | ||||
| 	if err := os.MkdirAll(byPathDir, 0755); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if err := ioutil.WriteFile(filepath.Join(dir, shortName), []byte{}, 0644); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if err := ioutil.WriteFile(filepath.Join(dir, shortName2), []byte{}, 0644); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if err := os.Symlink( | ||||
| 		filepath.Join(dir, shortName), | ||||
| 		filepath.Join(byPathDir, longName), | ||||
| 	); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	discovered := make(map[string]struct{}) | ||||
| 	discover(discovered, filepath.Join(byPathDir, "*")) | ||||
| 	discover(discovered, filepath.Join(dir, "unittest-video*")) | ||||
|  | ||||
| 	drvs := driver.GetManager().Query(func(d driver.Driver) bool { | ||||
| 		// Ignore real cameras. | ||||
| 		return d.Info().DeviceType == driver.Camera && strings.Contains(d.Info().Label, "unittest") | ||||
| 	}) | ||||
| 	if len(drvs) != 2 { | ||||
| 		t.Fatalf("Expected 2 driver, got %d drivers", len(drvs)) | ||||
| 	} | ||||
|  | ||||
| 	labels := []string{ | ||||
| 		drvs[0].Info().Label, | ||||
| 		drvs[1].Info().Label, | ||||
| 	} | ||||
| 	// Returned drivers are unordered. Sort to get static result. | ||||
| 	sort.Sort(sort.StringSlice(labels)) | ||||
|  | ||||
| 	expected := longName + LabelSeparator + shortName | ||||
| 	if label := labels[0]; label != expected { | ||||
| 		t.Errorf("Expected label: %s, got: %s", expected, label) | ||||
| 	} | ||||
|  | ||||
| 	expectedNoLink := shortName2 + LabelSeparator + shortName2 | ||||
| 	if label := labels[1]; label != expectedNoLink { | ||||
| 		t.Errorf("Expected label: %s, got: %s", expectedNoLink, label) | ||||
| 	} | ||||
| } | ||||
| @@ -155,9 +155,6 @@ func (m *microphone) AudioRecord(inputProp prop.Media) (audio.Reader, error) { | ||||
| 		return decodedChunk, func() {}, err | ||||
| 	}) | ||||
|  | ||||
| 	// FIXME: The current audio detection and audio encoder can only work with a static latency. Since the latency from the driver | ||||
| 	//        can fluctuate, we need to stabilize it. Maybe there's a better way for doing this? | ||||
| 	reader = audio.NewBuffer(int(inputProp.Latency.Seconds() * float64(inputProp.SampleRate)))(reader) | ||||
| 	return reader, nil | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package screen | ||||
|  | ||||
| // #cgo pkg-config: x11 xext | ||||
| // #include <stdint.h> | ||||
| // #include <string.h> | ||||
| // #include <sys/shm.h> | ||||
| // #include <X11/Xlib.h> | ||||
| // #define XUTIL_DEFINE_FUNCTIONS | ||||
| @@ -125,6 +126,14 @@ func (c colorFunc) RGBA() (r, g, b, a uint32) { | ||||
|  | ||||
| func (s *shmImage) At(x, y int) color.Color { | ||||
| 	switch s.pixFmt { | ||||
| 	case pixFmtRGB24: | ||||
| 		addr := (x + y*int(s.img.width)) * 4 | ||||
| 		r := uint32(s.b[addr]) * 0x100 | ||||
| 		g := uint32(s.b[addr+1]) * 0x100 | ||||
| 		b := uint32(s.b[addr+2]) * 0x100 | ||||
| 		return colorFunc(func() (_, _, _, _ uint32) { | ||||
| 			return r, g, b, 0xFFFF | ||||
| 		}) | ||||
| 	case pixFmtBGR24: | ||||
| 		addr := (x + y*int(s.img.width)) * 4 | ||||
| 		b := uint32(s.b[addr]) * 0x100 | ||||
| @@ -133,6 +142,15 @@ func (s *shmImage) At(x, y int) color.Color { | ||||
| 		return colorFunc(func() (_, _, _, _ uint32) { | ||||
| 			return r, g, b, 0xFFFF | ||||
| 		}) | ||||
| 	case pixFmtRGB16: | ||||
| 		addr := (x + y*int(s.img.width)) * 2 | ||||
| 		b1, b2 := s.b[addr], s.b[addr+1] | ||||
| 		r := uint32(b1>>3) * 0x100 | ||||
| 		g := uint32((b1&0x7)<<3|(b2&0xE0)>>5) * 0x100 | ||||
| 		b := uint32(b2&0x1F) * 0x100 | ||||
| 		return colorFunc(func() (_, _, _, _ uint32) { | ||||
| 			return r, g, b, 0xFFFF | ||||
| 		}) | ||||
| 	case pixFmtBGR16: | ||||
| 		addr := (x + y*int(s.img.width)) * 2 | ||||
| 		b1, b2 := s.b[addr], s.b[addr+1] | ||||
| @@ -149,12 +167,25 @@ func (s *shmImage) At(x, y int) color.Color { | ||||
|  | ||||
| func (s *shmImage) RGBAAt(x, y int) color.RGBA { | ||||
| 	switch s.pixFmt { | ||||
| 	case pixFmtRGB24: | ||||
| 		addr := (x + y*int(s.img.width)) * 4 | ||||
| 		r := s.b[addr] | ||||
| 		g := s.b[addr+1] | ||||
| 		b := s.b[addr+2] | ||||
| 		return color.RGBA{R: r, G: g, B: b, A: 0xFF} | ||||
| 	case pixFmtBGR24: | ||||
| 		addr := (x + y*int(s.img.width)) * 4 | ||||
| 		b := s.b[addr] | ||||
| 		g := s.b[addr+1] | ||||
| 		r := s.b[addr+2] | ||||
| 		return color.RGBA{R: r, G: g, B: b, A: 0xFF} | ||||
| 	case pixFmtRGB16: | ||||
| 		addr := (x + y*int(s.img.width)) * 2 | ||||
| 		b1, b2 := s.b[addr], s.b[addr+1] | ||||
| 		r := b1 >> 3 | ||||
| 		g := (b1&0x7)<<3 | (b2&0xE0)>>5 | ||||
| 		b := b2 & 0x1F | ||||
| 		return color.RGBA{R: r, G: g, B: b, A: 0xFF} | ||||
| 	case pixFmtBGR16: | ||||
| 		addr := (x + y*int(s.img.width)) * 2 | ||||
| 		b1, b2 := s.b[addr], s.b[addr+1] | ||||
| @@ -178,11 +209,17 @@ func (s *shmImage) ToRGBA(dst *image.RGBA) *image.RGBA { | ||||
| 		dst.Pix = dst.Pix[:l] | ||||
| 	} | ||||
| 	switch s.pixFmt { | ||||
| 	case pixFmtRGB24: | ||||
| 		C.memcpy(unsafe.Pointer(&dst.Pix[0]), unsafe.Pointer(s.img.data), C.size_t(len(dst.Pix))) | ||||
| 		return dst | ||||
| 	case pixFmtBGR24: | ||||
| 		C.copyBGR24(unsafe.Pointer(&dst.Pix[0]), s.img.data, C.ulong(len(dst.Pix))) | ||||
| 		C.copyBGR24(unsafe.Pointer(&dst.Pix[0]), s.img.data, C.size_t(len(dst.Pix))) | ||||
| 		return dst | ||||
| 	case pixFmtRGB16: | ||||
| 		C.memcpy(unsafe.Pointer(&dst.Pix[0]), unsafe.Pointer(s.img.data), C.size_t(len(dst.Pix))) | ||||
| 		return dst | ||||
| 	case pixFmtBGR16: | ||||
| 		C.copyBGR16(unsafe.Pointer(&dst.Pix[0]), s.img.data, C.ulong(len(dst.Pix))) | ||||
| 		C.copyBGR16(unsafe.Pointer(&dst.Pix[0]), s.img.data, C.size_t(len(dst.Pix))) | ||||
| 		return dst | ||||
| 	default: | ||||
| 		panic("unsupported pixel format") | ||||
| @@ -199,8 +236,12 @@ func newShmImage(dp *C.Display, screen int) (*shmImage, error) { | ||||
| 	s := &shmImage{dp: dp} | ||||
|  | ||||
| 	switch { | ||||
| 	case v.red_mask == 0xFF && v.green_mask == 0xFF00 && v.blue_mask == 0xFF0000: | ||||
| 		s.pixFmt = pixFmtRGB24 | ||||
| 	case v.red_mask == 0xFF0000 && v.green_mask == 0xFF00 && v.blue_mask == 0xFF: | ||||
| 		s.pixFmt = pixFmtBGR24 | ||||
| 	case v.red_mask == 0x1F && v.green_mask == 0x7E0 && v.blue_mask == 0xF800: | ||||
| 		s.pixFmt = pixFmtRGB16 | ||||
| 	case v.red_mask == 0xF800 && v.green_mask == 0x7E0 && v.blue_mask == 0x1F: | ||||
| 		s.pixFmt = pixFmtBGR16 | ||||
| 	default: | ||||
| @@ -209,7 +250,7 @@ func newShmImage(dp *C.Display, screen int) (*shmImage, error) { | ||||
| 		return nil, errors.New("unsupported pixel format") | ||||
| 	} | ||||
|  | ||||
| 	s.shm.shmid = C.shmget(C.IPC_PRIVATE, C.ulong(w*h*4+8), C.IPC_CREAT|0600) | ||||
| 	s.shm.shmid = C.shmget(C.IPC_PRIVATE, C.size_t(w*h*4+8), C.IPC_CREAT|0600) | ||||
| 	if s.shm.shmid == -1 { | ||||
| 		return nil, errors.New("failed to get shared memory") | ||||
| 	} | ||||
| @@ -279,5 +320,5 @@ func (r *reader) Close() { | ||||
|  | ||||
| // cAlign64 is fot testing | ||||
| func cAlign64(ptr uintptr) uintptr { | ||||
| 	return uintptr(C.align64ForTest(C.ulong(uintptr(ptr)))) | ||||
| 	return uintptr(C.align64ForTest(C.size_t(uintptr(ptr)))) | ||||
| } | ||||
|   | ||||
							
								
								
									
										21
									
								
								pkg/driver/vncdriver/vnc/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								pkg/driver/vncdriver/vnc/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| The MIT License (MIT) | ||||
|  | ||||
| Copyright (c) 2013 Mitchell Hashimoto | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in | ||||
| all copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||
| THE SOFTWARE. | ||||
							
								
								
									
										16
									
								
								pkg/driver/vncdriver/vnc/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								pkg/driver/vncdriver/vnc/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| # VNC Library for Go | ||||
|  | ||||
| go-vnc is a VNC library for Go, initially supporting VNC clients but | ||||
| with the goal of eventually implementing a VNC server. | ||||
|  | ||||
| This library implements [RFC 6143](http://tools.ietf.org/html/rfc6143). | ||||
|  | ||||
| ## Usage & Installation | ||||
|  | ||||
| The library is installable via standard `go get`. The package name is `vnc`. | ||||
|  | ||||
| ``` | ||||
| $ go get github.com/mitchellh/go-vnc | ||||
| ``` | ||||
|  | ||||
| Documentation is available on GoDoc: http://godoc.org/github.com/mitchellh/go-vnc | ||||
							
								
								
									
										494
									
								
								pkg/driver/vncdriver/vnc/client.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										494
									
								
								pkg/driver/vncdriver/vnc/client.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,494 @@ | ||||
| // Package vnc implements a VNC client. | ||||
| // | ||||
| // References: | ||||
| //   [PROTOCOL]: http://tools.ietf.org/html/rfc6143 | ||||
| package vnc | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/binary" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"net" | ||||
| 	"unicode" | ||||
| ) | ||||
|  | ||||
| type ClientConn struct { | ||||
| 	c      net.Conn | ||||
| 	config *ClientConfig | ||||
|  | ||||
| 	// If the pixel format uses a color map, then this is the color | ||||
| 	// map that is used. This should not be modified directly, since | ||||
| 	// the data comes from the server. | ||||
| 	ColorMap [256]Color | ||||
|  | ||||
| 	// Encodings supported by the client. This should not be modified | ||||
| 	// directly. Instead, SetEncodings should be used. | ||||
| 	Encs []Encoding | ||||
|  | ||||
| 	// Width of the frame buffer in pixels, sent from the server. | ||||
| 	FrameBufferWidth uint16 | ||||
|  | ||||
| 	// Height of the frame buffer in pixels, sent from the server. | ||||
| 	FrameBufferHeight uint16 | ||||
|  | ||||
| 	// Name associated with the desktop, sent from the server. | ||||
| 	DesktopName string | ||||
|  | ||||
| 	// The pixel format associated with the connection. This shouldn't | ||||
| 	// be modified. If you wish to set a new pixel format, use the | ||||
| 	// SetPixelFormat method. | ||||
| 	PixelFormat PixelFormat | ||||
| } | ||||
|  | ||||
| // A ClientConfig structure is used to configure a ClientConn. After | ||||
| // one has been passed to initialize a connection, it must not be modified. | ||||
| type ClientConfig struct { | ||||
| 	// A slice of ClientAuth methods. Only the first instance that is | ||||
| 	// suitable by the server will be used to authenticate. | ||||
| 	Auth []ClientAuth | ||||
|  | ||||
| 	// Exclusive determines whether the connection is shared with other | ||||
| 	// clients. If true, then all other clients connected will be | ||||
| 	// disconnected when a connection is established to the VNC server. | ||||
| 	Exclusive bool | ||||
|  | ||||
| 	// The channel that all messages received from the server will be | ||||
| 	// sent on. If the channel blocks, then the goroutine reading data | ||||
| 	// from the VNC server may block indefinitely. It is up to the user | ||||
| 	// of the library to ensure that this channel is properly read. | ||||
| 	// If this is not set, then all messages will be discarded. | ||||
| 	ServerMessageCh chan<- ServerMessage | ||||
|  | ||||
| 	// A slice of supported messages that can be read from the server. | ||||
| 	// This only needs to contain NEW server messages, and doesn't | ||||
| 	// need to explicitly contain the RFC-required messages. | ||||
| 	ServerMessages []ServerMessage | ||||
| } | ||||
|  | ||||
| func Client(c net.Conn, cfg *ClientConfig) (*ClientConn, error) { | ||||
| 	conn := &ClientConn{ | ||||
| 		c:      c, | ||||
| 		config: cfg, | ||||
| 	} | ||||
|  | ||||
| 	if err := conn.handshake(); err != nil { | ||||
| 		conn.Close() | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	go conn.mainLoop() | ||||
|  | ||||
| 	return conn, nil | ||||
| } | ||||
|  | ||||
| func (c *ClientConn) Close() error { | ||||
| 	return c.c.Close() | ||||
| } | ||||
|  | ||||
| // CutText tells the server that the client has new text in its cut buffer. | ||||
| // The text string MUST only contain Latin-1 characters. This encoding | ||||
| // is compatible with Go's native string format, but can only use up to | ||||
| // unicode.MaxLatin values. | ||||
| // | ||||
| // See RFC 6143 Section 7.5.6 | ||||
| func (c *ClientConn) CutText(text string) error { | ||||
| 	var buf bytes.Buffer | ||||
|  | ||||
| 	// This is the fixed size data we'll send | ||||
| 	fixedData := []interface{}{ | ||||
| 		uint8(6), | ||||
| 		uint8(0), | ||||
| 		uint8(0), | ||||
| 		uint8(0), | ||||
| 		uint32(len(text)), | ||||
| 	} | ||||
|  | ||||
| 	for _, val := range fixedData { | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, val); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, char := range text { | ||||
| 		if char > unicode.MaxLatin1 { | ||||
| 			return fmt.Errorf("Character '%d' is not valid Latin-1", char) | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, uint8(char)); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	dataLength := 8 + len(text) | ||||
| 	if _, err := c.c.Write(buf.Bytes()[0:dataLength]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Requests a framebuffer update from the server. There may be an indefinite | ||||
| // time between the request and the actual framebuffer update being | ||||
| // received. | ||||
| // | ||||
| // See RFC 6143 Section 7.5.3 | ||||
| func (c *ClientConn) FramebufferUpdateRequest(incremental bool, x, y, width, height uint16) error { | ||||
| 	var buf bytes.Buffer | ||||
| 	var incrementalByte uint8 = 0 | ||||
|  | ||||
| 	if incremental { | ||||
| 		incrementalByte = 1 | ||||
| 	} | ||||
|  | ||||
| 	data := []interface{}{ | ||||
| 		uint8(3), | ||||
| 		incrementalByte, | ||||
| 		x, y, width, height, | ||||
| 	} | ||||
|  | ||||
| 	for _, val := range data { | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, val); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if _, err := c.c.Write(buf.Bytes()[0:10]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // KeyEvent indiciates a key press or release and sends it to the server. | ||||
| // The key is indicated using the X Window System "keysym" value. Use | ||||
| // Google to find a reference of these values. To simulate a key press, | ||||
| // you must send a key with both a down event, and a non-down event. | ||||
| // | ||||
| // See 7.5.4. | ||||
| func (c *ClientConn) KeyEvent(keysym uint32, down bool) error { | ||||
| 	var downFlag uint8 = 0 | ||||
| 	if down { | ||||
| 		downFlag = 1 | ||||
| 	} | ||||
|  | ||||
| 	data := []interface{}{ | ||||
| 		uint8(4), | ||||
| 		downFlag, | ||||
| 		uint8(0), | ||||
| 		uint8(0), | ||||
| 		keysym, | ||||
| 	} | ||||
|  | ||||
| 	for _, val := range data { | ||||
| 		if err := binary.Write(c.c, binary.BigEndian, val); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // PointerEvent indicates that pointer movement or a pointer button | ||||
| // press or release. | ||||
| // | ||||
| // The mask is a bitwise mask of various ButtonMask values. When a button | ||||
| // is set, it is pressed, when it is unset, it is released. | ||||
| // | ||||
| // See RFC 6143 Section 7.5.5 | ||||
| func (c *ClientConn) PointerEvent(mask ButtonMask, x, y uint16) error { | ||||
| 	var buf bytes.Buffer | ||||
|  | ||||
| 	data := []interface{}{ | ||||
| 		uint8(5), | ||||
| 		uint8(mask), | ||||
| 		x, | ||||
| 		y, | ||||
| 	} | ||||
|  | ||||
| 	for _, val := range data { | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, val); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if _, err := c.c.Write(buf.Bytes()[0:6]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SetEncodings sets the encoding types in which the pixel data can | ||||
| // be sent from the server. After calling this method, the encs slice | ||||
| // given should not be modified. | ||||
| // | ||||
| // See RFC 6143 Section 7.5.2 | ||||
| func (c *ClientConn) SetEncodings(encs []Encoding) error { | ||||
| 	data := make([]interface{}, 3+len(encs)) | ||||
| 	data[0] = uint8(2) | ||||
| 	data[1] = uint8(0) | ||||
| 	data[2] = uint16(len(encs)) | ||||
|  | ||||
| 	for i, enc := range encs { | ||||
| 		data[3+i] = int32(enc.Type()) | ||||
| 	} | ||||
|  | ||||
| 	var buf bytes.Buffer | ||||
| 	for _, val := range data { | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, val); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	dataLength := 4 + (4 * len(encs)) | ||||
| 	if _, err := c.c.Write(buf.Bytes()[0:dataLength]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	c.Encs = encs | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SetPixelFormat sets the format in which pixel values should be sent | ||||
| // in FramebufferUpdate messages from the server. | ||||
| // | ||||
| // See RFC 6143 Section 7.5.1 | ||||
| func (c *ClientConn) SetPixelFormat(format *PixelFormat) error { | ||||
| 	var keyEvent [20]byte | ||||
| 	keyEvent[0] = 0 | ||||
|  | ||||
| 	pfBytes, err := writePixelFormat(format) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Copy the pixel format bytes into the proper slice location | ||||
| 	copy(keyEvent[4:], pfBytes) | ||||
|  | ||||
| 	// Send the data down the connection | ||||
| 	if _, err := c.c.Write(keyEvent[:]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Reset the color map as according to RFC. | ||||
| 	var newColorMap [256]Color | ||||
| 	c.ColorMap = newColorMap | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| const pvLen = 12 // ProtocolVersion message length. | ||||
|  | ||||
| func parseProtocolVersion(pv []byte) (uint, uint, error) { | ||||
| 	var major, minor uint | ||||
|  | ||||
| 	if len(pv) < pvLen { | ||||
| 		return 0, 0, fmt.Errorf("ProtocolVersion message too short (%v < %v)", len(pv), pvLen) | ||||
| 	} | ||||
|  | ||||
| 	l, err := fmt.Sscanf(string(pv), "RFB %d.%d\n", &major, &minor) | ||||
| 	if l != 2 { | ||||
| 		return 0, 0, fmt.Errorf("error parsing ProtocolVersion.") | ||||
| 	} | ||||
| 	if err != nil { | ||||
| 		return 0, 0, err | ||||
| 	} | ||||
|  | ||||
| 	return major, minor, nil | ||||
| } | ||||
|  | ||||
| func (c *ClientConn) handshake() error { | ||||
| 	var protocolVersion [pvLen]byte | ||||
|  | ||||
| 	// 7.1.1, read the ProtocolVersion message sent by the server. | ||||
| 	if _, err := io.ReadFull(c.c, protocolVersion[:]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	maxMajor, maxMinor, err := parseProtocolVersion(protocolVersion[:]) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if maxMajor < 3 { | ||||
| 		return fmt.Errorf("unsupported major version, less than 3: %d", maxMajor) | ||||
| 	} | ||||
| 	if maxMinor < 3 { | ||||
| 		return fmt.Errorf("unsupported minor version, less than 3: %d", maxMinor) | ||||
| 	} | ||||
|  | ||||
| 	// Respond with the version we will support | ||||
| 	if maxMinor<8 { | ||||
| 		if _, err = c.c.Write([]byte("RFB 003.003\n")); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		var numSecurityTypes uint32 | ||||
| 		if err = binary.Read(c.c, binary.BigEndian, &numSecurityTypes); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if numSecurityTypes == 0 { | ||||
| 			return fmt.Errorf("no security types: %s", c.readErrorReason()) | ||||
| 		} | ||||
| 	}else{ | ||||
| 		if _, err = c.c.Write([]byte("RFB 003.008\n")); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 		// 7.1.2 Security Handshake from server | ||||
| 		var numSecurityTypes uint8 | ||||
| 		if err = binary.Read(c.c, binary.BigEndian, &numSecurityTypes); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if numSecurityTypes == 0 { | ||||
| 			return fmt.Errorf("no security types: %s", c.readErrorReason()) | ||||
| 		} | ||||
|  | ||||
| 		securityTypes := make([]uint8, numSecurityTypes) | ||||
| 		if err = binary.Read(c.c, binary.BigEndian, &securityTypes); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		clientSecurityTypes := c.config.Auth | ||||
| 		if clientSecurityTypes == nil { | ||||
| 			clientSecurityTypes = []ClientAuth{new(ClientAuthNone)} | ||||
| 		} | ||||
|  | ||||
| 		var auth ClientAuth | ||||
| 	FindAuth: | ||||
| 		for _, curAuth := range clientSecurityTypes { | ||||
| 			for _, securityType := range securityTypes { | ||||
| 				if curAuth.SecurityType() == securityType { | ||||
| 					// We use the first matching supported authentication | ||||
| 					auth = curAuth | ||||
| 					break FindAuth | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if auth == nil { | ||||
| 			return fmt.Errorf("no suitable auth schemes found. server supported: %#v", securityTypes) | ||||
| 		} | ||||
|  | ||||
| 		// Respond back with the security type we'll use | ||||
| 		if err = binary.Write(c.c, binary.BigEndian, auth.SecurityType()); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err = auth.Handshake(c.c); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		// 7.1.3 SecurityResult Handshake | ||||
| 		var securityResult uint32 | ||||
| 		if err = binary.Read(c.c, binary.BigEndian, &securityResult); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if securityResult == 1 { | ||||
| 			return fmt.Errorf("security handshake failed: %s", c.readErrorReason()) | ||||
| 		} | ||||
| 	} | ||||
| 	// 7.3.1 ClientInit | ||||
| 	var sharedFlag uint8 = 1 | ||||
| 	if c.config.Exclusive { | ||||
| 		sharedFlag = 0 | ||||
| 	} | ||||
|  | ||||
| 	if err = binary.Write(c.c, binary.BigEndian, sharedFlag); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 7.3.2 ServerInit | ||||
| 	if err = binary.Read(c.c, binary.BigEndian, &c.FrameBufferWidth); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err = binary.Read(c.c, binary.BigEndian, &c.FrameBufferHeight); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// Read the pixel format | ||||
| 	if err = readPixelFormat(c.c, &c.PixelFormat); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	var nameLength uint32 | ||||
| 	if err = binary.Read(c.c, binary.BigEndian, &nameLength); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	nameBytes := make([]uint8, nameLength) | ||||
| 	if err = binary.Read(c.c, binary.BigEndian, &nameBytes); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	c.DesktopName = string(nameBytes) | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // mainLoop reads messages sent from the server and routes them to the | ||||
| // proper channels for users of the client to read. | ||||
| func (c *ClientConn) mainLoop() { | ||||
| 	defer c.Close() | ||||
|  | ||||
| 	// Build the map of available server messages | ||||
| 	typeMap := make(map[uint8]ServerMessage) | ||||
|  | ||||
| 	defaultMessages := []ServerMessage{ | ||||
| 		new(FramebufferUpdateMessage), | ||||
| 		new(SetColorMapEntriesMessage), | ||||
| 		new(BellMessage), | ||||
| 		new(ServerCutTextMessage), | ||||
| 	} | ||||
|  | ||||
| 	for _, msg := range defaultMessages { | ||||
| 		typeMap[msg.Type()] = msg | ||||
| 	} | ||||
|  | ||||
| 	if c.config.ServerMessages != nil { | ||||
| 		for _, msg := range c.config.ServerMessages { | ||||
| 			typeMap[msg.Type()] = msg | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for { | ||||
| 		var messageType uint8 | ||||
| 		if err := binary.Read(c.c, binary.BigEndian, &messageType); err != nil { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		msg, ok := typeMap[messageType] | ||||
| 		if !ok { | ||||
| 			// Unsupported message type! Bad! | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		parsedMsg, err := msg.Read(c, c.c) | ||||
| 		if err != nil { | ||||
| 			break | ||||
| 		} | ||||
|  | ||||
| 		if c.config.ServerMessageCh == nil { | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		c.config.ServerMessageCh <- parsedMsg | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (c *ClientConn) readErrorReason() string { | ||||
| 	var reasonLen uint32 | ||||
| 	if err := binary.Read(c.c, binary.BigEndian, &reasonLen); err != nil { | ||||
| 		return "<error>" | ||||
| 	} | ||||
|  | ||||
| 	reason := make([]uint8, reasonLen) | ||||
| 	if err := binary.Read(c.c, binary.BigEndian, &reason); err != nil { | ||||
| 		return "<error>" | ||||
| 	} | ||||
|  | ||||
| 	return string(reason) | ||||
| } | ||||
							
								
								
									
										124
									
								
								pkg/driver/vncdriver/vnc/client_auth.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								pkg/driver/vncdriver/vnc/client_auth.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| package vnc | ||||
|  | ||||
| import ( | ||||
| 	"net" | ||||
|  | ||||
| 	"crypto/des" | ||||
| 	"encoding/binary" | ||||
| ) | ||||
|  | ||||
| // A ClientAuth implements a method of authenticating with a remote server. | ||||
| type ClientAuth interface { | ||||
| 	// SecurityType returns the byte identifier sent by the server to | ||||
| 	// identify this authentication scheme. | ||||
| 	SecurityType() uint8 | ||||
|  | ||||
| 	// Handshake is called when the authentication handshake should be | ||||
| 	// performed, as part of the general RFB handshake. (see 7.2.1) | ||||
| 	Handshake(net.Conn) error | ||||
| } | ||||
|  | ||||
| // ClientAuthNone is the "none" authentication. See 7.2.1 | ||||
| type ClientAuthNone byte | ||||
|  | ||||
| func (*ClientAuthNone) SecurityType() uint8 { | ||||
| 	return 1 | ||||
| } | ||||
|  | ||||
| func (*ClientAuthNone) Handshake(net.Conn) error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // PasswordAuth is VNC authentication, 7.2.2 | ||||
| type PasswordAuth struct { | ||||
| 	Password string | ||||
| } | ||||
|  | ||||
| func (p *PasswordAuth) SecurityType() uint8 { | ||||
| 	return 2 | ||||
| } | ||||
|  | ||||
| func (p *PasswordAuth) Handshake(c net.Conn) error { | ||||
| 	randomValue := make([]uint8, 16) | ||||
| 	if err := binary.Read(c, binary.BigEndian, &randomValue); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	crypted, err := p.encrypt(p.Password, randomValue) | ||||
|  | ||||
| 	if (err != nil) { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := binary.Write(c, binary.BigEndian, &crypted); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (p *PasswordAuth) reverseBits(b byte) byte { | ||||
| 	var reverse = [256]int{ | ||||
| 		0, 128, 64, 192, 32, 160, 96, 224, | ||||
| 		16, 144, 80, 208, 48, 176, 112, 240, | ||||
| 		8, 136, 72, 200, 40, 168, 104, 232, | ||||
| 		24, 152, 88, 216, 56, 184, 120, 248, | ||||
| 		4, 132, 68, 196, 36, 164, 100, 228, | ||||
| 		20, 148, 84, 212, 52, 180, 116, 244, | ||||
| 		12, 140, 76, 204, 44, 172, 108, 236, | ||||
| 		28, 156, 92, 220, 60, 188, 124, 252, | ||||
| 		2, 130, 66, 194, 34, 162, 98, 226, | ||||
| 		18, 146, 82, 210, 50, 178, 114, 242, | ||||
| 		10, 138, 74, 202, 42, 170, 106, 234, | ||||
| 		26, 154, 90, 218, 58, 186, 122, 250, | ||||
| 		6, 134, 70, 198, 38, 166, 102, 230, | ||||
| 		22, 150, 86, 214, 54, 182, 118, 246, | ||||
| 		14, 142, 78, 206, 46, 174, 110, 238, | ||||
| 		30, 158, 94, 222, 62, 190, 126, 254, | ||||
| 		1, 129, 65, 193, 33, 161, 97, 225, | ||||
| 		17, 145, 81, 209, 49, 177, 113, 241, | ||||
| 		9, 137, 73, 201, 41, 169, 105, 233, | ||||
| 		25, 153, 89, 217, 57, 185, 121, 249, | ||||
| 		5, 133, 69, 197, 37, 165, 101, 229, | ||||
| 		21, 149, 85, 213, 53, 181, 117, 245, | ||||
| 		13, 141, 77, 205, 45, 173, 109, 237, | ||||
| 		29, 157, 93, 221, 61, 189, 125, 253, | ||||
| 		3, 131, 67, 195, 35, 163, 99, 227, | ||||
| 		19, 147, 83, 211, 51, 179, 115, 243, | ||||
| 		11, 139, 75, 203, 43, 171, 107, 235, | ||||
| 		27, 155, 91, 219, 59, 187, 123, 251, | ||||
| 		7, 135, 71, 199, 39, 167, 103, 231, | ||||
| 		23, 151, 87, 215, 55, 183, 119, 247, | ||||
| 		15, 143, 79, 207, 47, 175, 111, 239, | ||||
| 		31, 159, 95, 223, 63, 191, 127, 255, | ||||
| 	} | ||||
|  | ||||
| 	return byte(reverse[int(b)]) | ||||
| } | ||||
|  | ||||
| func (p *PasswordAuth) encrypt(key string, bytes []byte) ([]byte, error) { | ||||
| 	keyBytes := []byte{0,0,0,0,0,0,0,0} | ||||
|  | ||||
| 	if len(key) > 8 { | ||||
| 		key = key[:8] | ||||
| 	} | ||||
|  | ||||
| 	for i := 0; i < len(key); i++ { | ||||
| 		keyBytes[i] = p.reverseBits(key[i]) | ||||
| 	} | ||||
|  | ||||
| 	block, err := des.NewCipher(keyBytes) | ||||
|  | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	result1 := make([]byte, 8) | ||||
| 	block.Encrypt(result1, bytes) | ||||
| 	result2 := make([]byte, 8) | ||||
| 	block.Encrypt(result2, bytes[8:]) | ||||
|  | ||||
| 	crypted := append(result1, result2...) | ||||
|  | ||||
| 	return crypted, nil | ||||
| } | ||||
							
								
								
									
										6
									
								
								pkg/driver/vncdriver/vnc/color.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								pkg/driver/vncdriver/vnc/color.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| package vnc | ||||
|  | ||||
| // Color represents a single color in a color map. | ||||
| type Color struct { | ||||
| 	R, G, B uint16 | ||||
| } | ||||
							
								
								
									
										186
									
								
								pkg/driver/vncdriver/vnc/encoding.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										186
									
								
								pkg/driver/vncdriver/vnc/encoding.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,186 @@ | ||||
| package vnc | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"compress/zlib" | ||||
| 	"encoding/binary" | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| // An Encoding implements a method for encoding pixel data that is | ||||
| // sent by the server to the client. | ||||
| type Encoding interface { | ||||
| 	// The number that uniquely identifies this encoding type. | ||||
| 	Type() int32 | ||||
|  | ||||
| 	// Read reads the contents of the encoded pixel data from the reader. | ||||
| 	// This should return a new Encoding implementation that contains | ||||
| 	// the proper data. | ||||
| 	Read(*ClientConn, *Rectangle, io.Reader) (Encoding, error) | ||||
| } | ||||
|  | ||||
| // RawEncoding is raw pixel data sent by the server. | ||||
| // | ||||
| // See RFC 6143 Section 7.7.1 | ||||
| type RawEncoding struct { | ||||
| 	Colors []Color | ||||
| 	RawPixel []uint32	//RGBA | ||||
| } | ||||
|  | ||||
| func (*RawEncoding) Type() int32 { | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (*RawEncoding) Read(c *ClientConn, rect *Rectangle, r io.Reader) (Encoding, error) { | ||||
| 	bytesPerPixel := c.PixelFormat.BPP / 8 | ||||
| 	pixelBytes := make([]uint8, bytesPerPixel) | ||||
|  | ||||
| 	var byteOrder binary.ByteOrder = binary.LittleEndian | ||||
| 	if c.PixelFormat.BigEndian { | ||||
| 		byteOrder = binary.BigEndian | ||||
| 	} | ||||
|  | ||||
| 	colors := make([]Color, int(rect.Height)*int(rect.Width)) | ||||
| 	rawPixels:=make([]uint32,int(rect.Height)*int(rect.Width)) | ||||
| 	for y := uint16(0); y < rect.Height; y++ { | ||||
| 		for x := uint16(0); x < rect.Width; x++ { | ||||
| 			if _, err := io.ReadFull(r, pixelBytes); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
|  | ||||
| 			var rawPixel uint32 | ||||
| 			if c.PixelFormat.BPP == 8 { | ||||
| 				rawPixel = uint32(pixelBytes[0]) | ||||
| 			} else if c.PixelFormat.BPP == 16 { | ||||
| 				rawPixel = uint32(byteOrder.Uint16(pixelBytes)) | ||||
| 			} else if c.PixelFormat.BPP == 32 { | ||||
| 				rawPixel = byteOrder.Uint32(pixelBytes) | ||||
| 			} | ||||
| 			//rawPixels[int(y)*int(rect.Width)+int(x)]=rawPixel | ||||
| 			color := &colors[int(y)*int(rect.Width)+int(x)] | ||||
| 			if c.PixelFormat.TrueColor { | ||||
| 				color.R = uint16((rawPixel >> c.PixelFormat.RedShift) & uint32(c.PixelFormat.RedMax)) | ||||
| 				color.G = uint16((rawPixel >> c.PixelFormat.GreenShift) & uint32(c.PixelFormat.GreenMax)) | ||||
| 				color.B = uint16((rawPixel >> c.PixelFormat.BlueShift) & uint32(c.PixelFormat.BlueMax)) | ||||
| 			} else { | ||||
| 				*color = c.ColorMap[rawPixel] | ||||
| 			} | ||||
| 			rawPixels[int(y)*int(rect.Width)+int(x)]=uint32(color.B)<<16 | uint32(color.G)<<8 | uint32(color.R) | ||||
| 			//fmt.Printf("%x %x",rawPixel,rawPixels[int(y)*int(rect.Width)+int(x)]) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &RawEncoding{colors,rawPixels}, nil | ||||
| } | ||||
| // ZlibEncoding is raw pixel data sent by the server compressed by Zlib. | ||||
| // | ||||
| // A single Zlib stream is created. There is only a single header for a framebuffer request response. | ||||
| type ZlibEncoding struct { | ||||
| 	Colors  []Color | ||||
| 	RawPixel[] uint32 | ||||
| 	ZStream *bytes.Buffer | ||||
| 	ZReader io.ReadCloser | ||||
| } | ||||
|  | ||||
| func (*ZlibEncoding) Type() int32 { | ||||
| 	return 6 | ||||
| } | ||||
|  | ||||
| func (ze *ZlibEncoding) Read(c *ClientConn, rect *Rectangle, r io.Reader) (Encoding, error) { | ||||
| 	bytesPerPixel := c.PixelFormat.BPP / 8 | ||||
| 	pixelBytes := make([]uint8, bytesPerPixel) | ||||
|  | ||||
| 	var byteOrder binary.ByteOrder = binary.LittleEndian | ||||
| 	if c.PixelFormat.BigEndian { | ||||
| 		byteOrder = binary.BigEndian | ||||
| 	} | ||||
|  | ||||
| 	// Format | ||||
| 	// 4 bytes        | uint32 | length | ||||
| 	// 'length' bytes | []byte | zlibData | ||||
|  | ||||
| 	// Read zlib length | ||||
| 	var zipLength uint32 | ||||
| 	err := binary.Read(r, binary.BigEndian, &zipLength) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Read all compressed data | ||||
| 	zBytes := make([]byte, zipLength) | ||||
| 	if _, err := io.ReadFull(r, zBytes); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Create new zlib stream if needed | ||||
| 	if ze.ZStream == nil { | ||||
| 		// Create and save the buffer | ||||
| 		ze.ZStream = new(bytes.Buffer) | ||||
| 		ze.ZStream.Write(zBytes) | ||||
|  | ||||
| 		// Create a reader for the buffer | ||||
| 		ze.ZReader, err = zlib.NewReader(ze.ZStream) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		// This is needed to avoid 'zlib missing header' | ||||
| 	} else { | ||||
| 		// Just append if already created | ||||
| 		ze.ZStream.Write(zBytes) | ||||
| 	} | ||||
|  | ||||
| 	// Calculate zlib decompressed size | ||||
| 	sizeToRead := int(rect.Height) * int(rect.Width) * int(bytesPerPixel) | ||||
|  | ||||
| 	// Create buffer for bytes | ||||
| 	colorBytes := make([]byte, sizeToRead) | ||||
|  | ||||
| 	// Read all data from zlib stream | ||||
| 	read, err := io.ReadFull(ze.ZReader, colorBytes) | ||||
| 	if read != sizeToRead || err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Create buffer for raw encoding | ||||
| 	colorReader := bytes.NewReader(colorBytes) | ||||
|  | ||||
| 	colors := make([]Color, int(rect.Height)*int(rect.Width)) | ||||
| 	rawPixels:=make([]uint32,int(rect.Height)*int(rect.Width)) | ||||
| 	for y := uint16(0); y < rect.Height; y++ { | ||||
| 		for x := uint16(0); x < rect.Width; x++ { | ||||
| 			if _, err := io.ReadFull(colorReader, pixelBytes); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
|  | ||||
| 			var rawPixel uint32 | ||||
| 			if c.PixelFormat.BPP == 8 { | ||||
| 				rawPixel = uint32(pixelBytes[0]) | ||||
| 			} else if c.PixelFormat.BPP == 16 { | ||||
| 				rawPixel = uint32(byteOrder.Uint16(pixelBytes)) | ||||
| 			} else if c.PixelFormat.BPP == 32 { | ||||
| 				rawPixel = byteOrder.Uint32(pixelBytes) | ||||
| 			} | ||||
|  | ||||
| 			color := &colors[int(y)*int(rect.Width)+int(x)] | ||||
| 			if c.PixelFormat.TrueColor { | ||||
| 				color.R = uint16((rawPixel >> c.PixelFormat.RedShift) & uint32(c.PixelFormat.RedMax)) | ||||
| 				color.G = uint16((rawPixel >> c.PixelFormat.GreenShift) & uint32(c.PixelFormat.GreenMax)) | ||||
| 				color.B = uint16((rawPixel >> c.PixelFormat.BlueShift) & uint32(c.PixelFormat.BlueMax)) | ||||
| 			} else { | ||||
| 				*color = c.ColorMap[rawPixel] | ||||
| 			} | ||||
| 			rawPixels[int(y)*int(rect.Width)+int(x)]=uint32(color.B)<<16 | uint32(color.G)<<8 | uint32(color.R) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &ZlibEncoding{Colors: colors,RawPixel: rawPixels}, nil | ||||
| } | ||||
|  | ||||
| func (ze *ZlibEncoding) Close() { | ||||
| 	if ze.ZStream != nil { | ||||
| 		ze.ZStream = nil | ||||
| 		ze.ZReader.Close() | ||||
| 		ze.ZReader = nil | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										151
									
								
								pkg/driver/vncdriver/vnc/pixel_format.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								pkg/driver/vncdriver/vnc/pixel_format.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| package vnc | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/binary" | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| // PixelFormat describes the way a pixel is formatted for a VNC connection. | ||||
| // | ||||
| // See RFC 6143 Section 7.4 for information on each of the fields. | ||||
| type PixelFormat struct { | ||||
| 	BPP        uint8 | ||||
| 	Depth      uint8 | ||||
| 	BigEndian  bool | ||||
| 	TrueColor  bool | ||||
| 	RedMax     uint16 | ||||
| 	GreenMax   uint16 | ||||
| 	BlueMax    uint16 | ||||
| 	RedShift   uint8 | ||||
| 	GreenShift uint8 | ||||
| 	BlueShift  uint8 | ||||
| } | ||||
|  | ||||
| func readPixelFormat(r io.Reader, result *PixelFormat) error { | ||||
| 	var rawPixelFormat [16]byte | ||||
| 	if _, err := io.ReadFull(r, rawPixelFormat[:]); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	var pfBoolByte uint8 | ||||
| 	brPF := bytes.NewReader(rawPixelFormat[:]) | ||||
| 	if err := binary.Read(brPF, binary.BigEndian, &result.BPP); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := binary.Read(brPF, binary.BigEndian, &result.Depth); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := binary.Read(brPF, binary.BigEndian, &pfBoolByte); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if pfBoolByte != 0 { | ||||
| 		// Big endian is true | ||||
| 		result.BigEndian = true | ||||
| 	} | ||||
|  | ||||
| 	if err := binary.Read(brPF, binary.BigEndian, &pfBoolByte); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if pfBoolByte != 0 { | ||||
| 		// True Color is true. So we also have to read all the color max & shifts. | ||||
| 		result.TrueColor = true | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.RedMax); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.GreenMax); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.BlueMax); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.RedShift); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.GreenShift); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Read(brPF, binary.BigEndian, &result.BlueShift); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func writePixelFormat(format *PixelFormat) ([]byte, error) { | ||||
| 	var buf bytes.Buffer | ||||
|  | ||||
| 	// Byte 1 | ||||
| 	if err := binary.Write(&buf, binary.BigEndian, format.BPP); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Byte 2 | ||||
| 	if err := binary.Write(&buf, binary.BigEndian, format.Depth); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var boolByte byte | ||||
| 	if format.BigEndian { | ||||
| 		boolByte = 1 | ||||
| 	} else { | ||||
| 		boolByte = 0 | ||||
| 	} | ||||
|  | ||||
| 	// Byte 3 (BigEndian) | ||||
| 	if err := binary.Write(&buf, binary.BigEndian, boolByte); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	if format.TrueColor { | ||||
| 		boolByte = 1 | ||||
| 	} else { | ||||
| 		boolByte = 0 | ||||
| 	} | ||||
|  | ||||
| 	// Byte 4 (TrueColor) | ||||
| 	if err := binary.Write(&buf, binary.BigEndian, boolByte); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// If we have true color enabled then we have to fill in the rest of the | ||||
| 	// structure with the color values. | ||||
| 	if format.TrueColor { | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.RedMax); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.GreenMax); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.BlueMax); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.RedShift); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.GreenShift); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
|  | ||||
| 		if err := binary.Write(&buf, binary.BigEndian, format.BlueShift); err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return buf.Bytes()[0:16], nil | ||||
| } | ||||
							
								
								
									
										16
									
								
								pkg/driver/vncdriver/vnc/pointer.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								pkg/driver/vncdriver/vnc/pointer.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| package vnc | ||||
|  | ||||
| // ButtonMask represents a mask of pointer presses/releases. | ||||
| type ButtonMask uint8 | ||||
|  | ||||
| // All available button mask components. | ||||
| const ( | ||||
| 	ButtonLeft ButtonMask = 1 << iota | ||||
| 	ButtonMiddle | ||||
| 	ButtonRight | ||||
| 	Button4 | ||||
| 	Button5 | ||||
| 	Button6 | ||||
| 	Button7 | ||||
| 	Button8 | ||||
| ) | ||||
							
								
								
									
										192
									
								
								pkg/driver/vncdriver/vnc/server_messages.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								pkg/driver/vncdriver/vnc/server_messages.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | ||||
| package vnc | ||||
|  | ||||
| import ( | ||||
| 	"encoding/binary" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| // A ServerMessage implements a message sent from the server to the client. | ||||
| type ServerMessage interface { | ||||
| 	// The type of the message that is sent down on the wire. | ||||
| 	Type() uint8 | ||||
|  | ||||
| 	// Read reads the contents of the message from the reader. At the point | ||||
| 	// this is called, the message type has already been read from the reader. | ||||
| 	// This should return a new ServerMessage that is the appropriate type. | ||||
| 	Read(*ClientConn, io.Reader) (ServerMessage, error) | ||||
| } | ||||
|  | ||||
| // FramebufferUpdateMessage consists of a sequence of rectangles of | ||||
| // pixel data that the client should put into its framebuffer. | ||||
| type FramebufferUpdateMessage struct { | ||||
| 	Rectangles []Rectangle | ||||
| } | ||||
|  | ||||
| // Rectangle represents a rectangle of pixel data. | ||||
| type Rectangle struct { | ||||
| 	X      uint16 | ||||
| 	Y      uint16 | ||||
| 	Width  uint16 | ||||
| 	Height uint16 | ||||
| 	Enc    Encoding | ||||
| } | ||||
|  | ||||
| func (*FramebufferUpdateMessage) Type() uint8 { | ||||
| 	return 0 | ||||
| } | ||||
|  | ||||
| func (*FramebufferUpdateMessage) Read(c *ClientConn, r io.Reader) (ServerMessage, error) { | ||||
| 	// Read off the padding | ||||
| 	var padding [1]byte | ||||
| 	if _, err := io.ReadFull(r, padding[:]); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var numRects uint16 | ||||
| 	if err := binary.Read(r, binary.BigEndian, &numRects); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// Build the map of encodings supported | ||||
| 	encMap := make(map[int32]Encoding) | ||||
| 	for _, enc := range c.Encs { | ||||
| 		encMap[enc.Type()] = enc | ||||
| 	} | ||||
|  | ||||
| 	// We must always support the raw encoding | ||||
| 	rawEnc := new(RawEncoding) | ||||
| 	encMap[rawEnc.Type()] = rawEnc | ||||
|  | ||||
| 	rects := make([]Rectangle, numRects) | ||||
| 	for i := uint16(0); i < numRects; i++ { | ||||
| 		var encodingType int32 | ||||
|  | ||||
| 		rect := &rects[i] | ||||
| 		data := []interface{}{ | ||||
| 			&rect.X, | ||||
| 			&rect.Y, | ||||
| 			&rect.Width, | ||||
| 			&rect.Height, | ||||
| 			&encodingType, | ||||
| 		} | ||||
|  | ||||
| 		for _, val := range data { | ||||
| 			if err := binary.Read(r, binary.BigEndian, val); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		enc, ok := encMap[encodingType] | ||||
| 		if !ok { | ||||
| 			return nil, fmt.Errorf("unsupported encoding type: %d", encodingType) | ||||
| 		} | ||||
|  | ||||
| 		var err error | ||||
| 		rect.Enc, err = enc.Read(c, rect, r) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return &FramebufferUpdateMessage{rects}, nil | ||||
| } | ||||
|  | ||||
| // SetColorMapEntriesMessage is sent by the server to set values into | ||||
| // the color map. This message will automatically update the color map | ||||
| // for the associated connection, but contains the color change data | ||||
| // if the consumer wants to read it. | ||||
| // | ||||
| // See RFC 6143 Section 7.6.2 | ||||
| type SetColorMapEntriesMessage struct { | ||||
| 	FirstColor uint16 | ||||
| 	Colors     []Color | ||||
| } | ||||
|  | ||||
| func (*SetColorMapEntriesMessage) Type() uint8 { | ||||
| 	return 1 | ||||
| } | ||||
|  | ||||
| func (*SetColorMapEntriesMessage) Read(c *ClientConn, r io.Reader) (ServerMessage, error) { | ||||
| 	// Read off the padding | ||||
| 	var padding [1]byte | ||||
| 	if _, err := io.ReadFull(r, padding[:]); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var result SetColorMapEntriesMessage | ||||
| 	if err := binary.Read(r, binary.BigEndian, &result.FirstColor); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var numColors uint16 | ||||
| 	if err := binary.Read(r, binary.BigEndian, &numColors); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	result.Colors = make([]Color, numColors) | ||||
| 	for i := uint16(0); i < numColors; i++ { | ||||
|  | ||||
| 		color := &result.Colors[i] | ||||
| 		data := []interface{}{ | ||||
| 			&color.R, | ||||
| 			&color.G, | ||||
| 			&color.B, | ||||
| 		} | ||||
|  | ||||
| 		for _, val := range data { | ||||
| 			if err := binary.Read(r, binary.BigEndian, val); err != nil { | ||||
| 				return nil, err | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		// Update the connection's color map | ||||
| 		c.ColorMap[result.FirstColor+i] = *color | ||||
| 	} | ||||
|  | ||||
| 	return &result, nil | ||||
| } | ||||
|  | ||||
| // Bell signals that an audible bell should be made on the client. | ||||
| // | ||||
| // See RFC 6143 Section 7.6.3 | ||||
| type BellMessage byte | ||||
|  | ||||
| func (*BellMessage) Type() uint8 { | ||||
| 	return 2 | ||||
| } | ||||
|  | ||||
| func (*BellMessage) Read(*ClientConn, io.Reader) (ServerMessage, error) { | ||||
| 	return new(BellMessage), nil | ||||
| } | ||||
|  | ||||
| // ServerCutTextMessage indicates the server has new text in the cut buffer. | ||||
| // | ||||
| // See RFC 6143 Section 7.6.4 | ||||
| type ServerCutTextMessage struct { | ||||
| 	Text string | ||||
| } | ||||
|  | ||||
| func (*ServerCutTextMessage) Type() uint8 { | ||||
| 	return 3 | ||||
| } | ||||
|  | ||||
| func (*ServerCutTextMessage) Read(c *ClientConn, r io.Reader) (ServerMessage, error) { | ||||
| 	// Read off the padding | ||||
| 	var padding [3]byte | ||||
| 	if _, err := io.ReadFull(r, padding[:]); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	var textLength uint32 | ||||
| 	if err := binary.Read(r, binary.BigEndian, &textLength); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	textBytes := make([]uint8, textLength) | ||||
| 	if err := binary.Read(r, binary.BigEndian, &textBytes); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &ServerCutTextMessage{string(textBytes)}, nil | ||||
| } | ||||
							
								
								
									
										177
									
								
								pkg/driver/vncdriver/vncdriver.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										177
									
								
								pkg/driver/vncdriver/vncdriver.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,177 @@ | ||||
| // Package videotest provides vncDevice video driver for testing. | ||||
| package vncdriver | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"encoding/binary" | ||||
| 	"fmt" | ||||
| 	"github.com/pion/mediadevices/pkg/driver/vncdriver/vnc" | ||||
| 	"image" | ||||
| 	"io" | ||||
| 	"net" | ||||
| 	"sync" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/pion/mediadevices/pkg/frame" | ||||
| 	"github.com/pion/mediadevices/pkg/io/video" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| ) | ||||
|  | ||||
| type vncDevice struct { | ||||
| 	closed   <-chan struct{} | ||||
| 	cancel   func() | ||||
| 	tick     *time.Ticker | ||||
| 	h, w     int | ||||
| 	rawPixel []byte | ||||
| 	mutex    sync.Mutex | ||||
| 	vClient  *vnc.ClientConn | ||||
| 	vncAddr  string | ||||
| } | ||||
|  | ||||
| func NewVnc(vncAddr string) *vncDevice { | ||||
| 	return &vncDevice{vncAddr: vncAddr} | ||||
| } | ||||
| func (d *vncDevice) PointerEvent(mask uint8, x, y uint16) { | ||||
| 	if d.vClient!=nil{ | ||||
| 		d.vClient.PointerEvent(vnc.ButtonMask(mask), x, y) | ||||
| 	} | ||||
| } | ||||
| func (d *vncDevice) KeyEvent(keysym uint32, down bool) { | ||||
| 	if d.vClient!=nil { | ||||
| 		d.vClient.KeyEvent(keysym, down) | ||||
| 	} | ||||
| } | ||||
| func (d *vncDevice) Open() error { | ||||
| 	if d.vClient != nil { | ||||
| 		return nil | ||||
| 	} | ||||
| 	ctx, cancel := context.WithCancel(context.Background()) | ||||
| 	d.closed = ctx.Done() | ||||
| 	d.cancel = cancel | ||||
| 	msg := make(chan vnc.ServerMessage, 1) | ||||
| 	conf := vnc.ClientConfig{ | ||||
| 		ServerMessageCh: msg, | ||||
| 		Exclusive:       false, | ||||
| 	} | ||||
| 	d.mutex.Lock() | ||||
| 	defer d.mutex.Unlock() | ||||
| 	conn, err := net.Dial("tcp", d.vncAddr) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	d.vClient, err = vnc.Client(conn, &conf) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	d.vClient.SetEncodings([]vnc.Encoding{ | ||||
| 		&vnc.ZlibEncoding{}, | ||||
| 		&vnc.RawEncoding{}, | ||||
| 	}) | ||||
| 	d.w = int(d.vClient.FrameBufferWidth) | ||||
| 	d.h = int(d.vClient.FrameBufferHeight) | ||||
|  | ||||
| 	d.rawPixel = make([]byte, d.h*d.w*4) | ||||
|  | ||||
| 	go func(ctx context.Context) { | ||||
| 		c, cancel := context.WithCancel(ctx) | ||||
| 		defer cancel() | ||||
| 		if d.vClient == nil { | ||||
| 			return | ||||
| 		} | ||||
| 		d.vClient.FramebufferUpdateRequest(true, 0, 0, uint16(d.w), uint16(d.h)) | ||||
| 		for { | ||||
| 			select { | ||||
| 			case <-c.Done(): | ||||
| 				return | ||||
| 			case msg := <-msg: | ||||
| 				switch t := msg.(type) { | ||||
| 				case *vnc.FramebufferUpdateMessage: | ||||
| 					for _, rect := range t.Rectangles { | ||||
| 						var pix []uint32 | ||||
| 						switch t := rect.Enc.(type) { | ||||
| 						case *vnc.RawEncoding: | ||||
| 							pix = t.RawPixel | ||||
| 						case *vnc.ZlibEncoding: | ||||
| 							pix = t.RawPixel | ||||
| 						} | ||||
| 						for y := int(rect.Y); y < int(rect.Height+rect.Y); y++ { | ||||
| 							for x := int(rect.X); x < int(rect.Width+rect.X); x++ { | ||||
| 								binary.LittleEndian.PutUint32(d.rawPixel[(y*d.w+x)*4:], pix[(y-int(rect.Y))*int(rect.Width)+(x-int(rect.X))]) | ||||
| 								//BigEndian | ||||
| 							} | ||||
| 						} | ||||
|  | ||||
| 					} | ||||
| 					//time.Sleep(33 * time.Millisecond) | ||||
| 					d.vClient.FramebufferUpdateRequest(true, 0, 0, uint16(d.w), uint16(d.h)) | ||||
| 					break | ||||
| 				default: | ||||
|  | ||||
| 				} | ||||
| 			case <-time.After(10 * time.Second): | ||||
| 				//fmt.Println("Timeout FramebufferUpdate") | ||||
| 				if d.vClient.FramebufferUpdateRequest(true, 0, 0, uint16(d.w), uint16(d.h)) != nil { | ||||
| 					d.cancel() | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 			} | ||||
| 		} | ||||
| 	}(ctx) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *vncDevice) Close() error { | ||||
| 	d.cancel() | ||||
| 	if d.tick != nil { | ||||
| 		d.tick.Stop() | ||||
| 	} | ||||
| 	d.mutex.Lock() | ||||
| 	defer d.mutex.Unlock() | ||||
| 	if d.vClient != nil { | ||||
| 		d.vClient.Close() | ||||
| 		d.vClient = nil | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (d *vncDevice) VideoRecord(p prop.Media) (video.Reader, error) { | ||||
| 	if p.FrameRate == 0 { | ||||
| 		p.FrameRate = 15 | ||||
| 	} | ||||
|  | ||||
| 	tick := time.NewTicker(time.Duration(float32(time.Second) / p.FrameRate)) | ||||
| 	d.tick = tick | ||||
| 	closed := d.closed | ||||
| 	pixs := make([]byte, d.h*d.w*4) | ||||
| 	r := video.ReaderFunc(func() (image.Image, func(), error) { | ||||
| 		select { | ||||
| 		case <-closed: | ||||
| 			fmt.Println("Stop Record Video By VideoRecord") | ||||
| 			return nil, func() {}, io.EOF | ||||
| 		default: | ||||
| 		} | ||||
|  | ||||
| 		<-tick.C | ||||
| 		copy(pixs, d.rawPixel) | ||||
| 		return &image.RGBA{ | ||||
| 			Pix:    pixs, | ||||
| 			Stride: 4, | ||||
| 			Rect:   image.Rect(0, 0, d.w, d.h), | ||||
| 		}, func() {}, nil | ||||
| 	}) | ||||
|  | ||||
| 	return r, nil | ||||
| } | ||||
|  | ||||
| func (d *vncDevice) Properties() []prop.Media { | ||||
| 	return []prop.Media{ | ||||
| 		{ | ||||
| 			Video: prop.Video{ | ||||
| 				Width:       d.w, | ||||
| 				Height:      d.h, | ||||
| 				FrameFormat: frame.FormatRGBA, | ||||
| 			}, | ||||
| 		}, | ||||
| 	} | ||||
| } | ||||
| @@ -1,14 +1,19 @@ | ||||
| package driver | ||||
|  | ||||
| import ( | ||||
| 	"github.com/google/uuid" | ||||
| 	"github.com/pion/mediadevices/pkg/io/audio" | ||||
| 	"github.com/pion/mediadevices/pkg/io/video" | ||||
| 	"github.com/pion/mediadevices/pkg/prop" | ||||
| 	uuid "github.com/satori/go.uuid" | ||||
| ) | ||||
|  | ||||
| func wrapAdapter(a Adapter, info Info) Driver { | ||||
| 	id := uuid.NewV4().String() | ||||
| 	generator, err := uuid.NewRandom() | ||||
| 	if err != nil { | ||||
| 		panic(err) | ||||
| 	} | ||||
|  | ||||
| 	id := generator.String() | ||||
| 	d := &adapterWrapper{ | ||||
| 		Adapter: a, | ||||
| 		id:      id, | ||||
|   | ||||
| @@ -25,6 +25,9 @@ const ( | ||||
|  | ||||
| 	// FormatMJPEG https://www.fourcc.org/mjpg/ | ||||
| 	FormatMJPEG = "MJPEG" | ||||
|  | ||||
| 	// FormatZ16 https://www.kernel.org/doc/html/v5.9/userspace-api/media/v4l/pixfmt-z16.html | ||||
| 	FormatZ16 = "Z16" | ||||
| ) | ||||
|  | ||||
| const FormatYUYV = FormatYUY2 | ||||
| @@ -36,6 +39,7 @@ var decoderMap = map[Format]decoderFunc{ | ||||
| 	FormatYUY2:  decodeYUY2, | ||||
| 	FormatUYVY:  decodeUYVY, | ||||
| 	FormatMJPEG: decodeMJPEG, | ||||
| 	FormatZ16:   decodeZ16, | ||||
| } | ||||
|  | ||||
| func NewDecoder(f Format) (Decoder, error) { | ||||
|   | ||||
| @@ -17,7 +17,7 @@ func decodeYUY2(frame []byte, width, height int) (image.Image, func(), error) { | ||||
| 	ci := yi / 2 | ||||
| 	fi := yi + 2*ci | ||||
|  | ||||
| 	if len(frame) != fi { | ||||
| 	if len(frame) < fi { | ||||
| 		return nil, func() {}, fmt.Errorf("frame length (%d) less than expected (%d)", len(frame), fi) | ||||
| 	} | ||||
|  | ||||
| @@ -49,7 +49,7 @@ func decodeUYVY(frame []byte, width, height int) (image.Image, func(), error) { | ||||
| 	ci := yi / 2 | ||||
| 	fi := yi + 2*ci | ||||
|  | ||||
| 	if len(frame) != fi { | ||||
| 	if len(frame) < fi { | ||||
| 		return nil, func() {}, fmt.Errorf("frame length (%d) less than expected (%d)", len(frame), fi) | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -12,7 +12,7 @@ func decodeYUY2(frame []byte, width, height int) (image.Image, func(), error) { | ||||
| 	ci := yi / 2 | ||||
| 	fi := yi + 2*ci | ||||
|  | ||||
| 	if len(frame) != fi { | ||||
| 	if len(frame) < fi { | ||||
| 		return nil, func() {}, fmt.Errorf("frame length (%d) less than expected (%d)", len(frame), fi) | ||||
| 	} | ||||
|  | ||||
| @@ -47,7 +47,7 @@ func decodeUYVY(frame []byte, width, height int) (image.Image, func(), error) { | ||||
| 	ci := yi / 2 | ||||
| 	fi := yi + 2*ci | ||||
|  | ||||
| 	if len(frame) != fi { | ||||
| 	if len(frame) < fi { | ||||
| 		return nil, func() {}, fmt.Errorf("frame length (%d) less than expected (%d)", len(frame), fi) | ||||
| 	} | ||||
|  | ||||
|   | ||||
							
								
								
									
										36
									
								
								pkg/frame/z16.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								pkg/frame/z16.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| package frame | ||||
|  | ||||
| import ( | ||||
| 	"encoding/binary" | ||||
| 	"fmt" | ||||
| 	"image" | ||||
| 	"image/color" | ||||
| ) | ||||
|  | ||||
| func decodeZ16(frame []byte, width, height int) (image.Image, func(), error) { | ||||
| 	expectedSize := 2 * (width * height) | ||||
| 	if expectedSize  != len(frame) { | ||||
| 		return nil, func() {}, fmt.Errorf("frame length (%d) not expected size (%d)", len(frame), expectedSize) | ||||
| 	} | ||||
| 	img := image.NewGray16(image.Rect(0, 0, width, height)) | ||||
| 	/* | ||||
| 		v4l specifies images in terms of series of lines which is perplexing because the | ||||
| 		example in https://www.kernel.org/doc/html/v5.9/userspace-api/media/v4l/pixfmt-z16.html | ||||
| 		seems to swap the subscript of each depth value. | ||||
|  | ||||
| 		Clear example: | ||||
| 		Width: 3, Height: 4 | ||||
| 		[Z_low(x_0,y_0), Z_high(x_0,y_0), Z_low(x_1,y_0), Z_high(x_1,y_0), Z_low(x_2,y_0), Z_high(x_2,y_0), | ||||
| 		 Z_low(x_0,y_1), Z_high(x_0,y_1), Z_low(x_1,y_1), Z_high(x_1,y_1), Z_low(x_2,y_1), Z_high(x_2,y_1), | ||||
| 		 Z_low(x_0,y_2), Z_high(x_0,y_2), Z_low(x_1,y_2), Z_high(x_1,y_2), Z_low(x_2,y_2), Z_high(x_2,y_2), | ||||
| 		 Z_low(x_0,y_3), Z_high(x_0,y_3), Z_low(x_1,y_3), Z_high(x_1,y_3), Z_low(x_2,y_3), Z_high(x_2,y_3)] | ||||
| 	*/ | ||||
| 	for x := 0; x < width; x++ { | ||||
| 		for y := 0; y < height; y++ { | ||||
| 			idx := 2 * (x + (y * width)) | ||||
| 			z := binary.LittleEndian.Uint16(frame[idx : idx+2]) | ||||
| 			img.SetGray16(x, y, color.Gray16{Y: z}) | ||||
| 		} | ||||
| 	} | ||||
| 	return img, func() {}, nil | ||||
| } | ||||
							
								
								
									
										45
									
								
								pkg/frame/z16_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								pkg/frame/z16_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| package frame | ||||
|  | ||||
| import ( | ||||
| 	"image" | ||||
| 	"image/color" | ||||
| 	"reflect" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestDecodeZ16(t *testing.T) { | ||||
| 	const ( | ||||
| 		width  = 2 | ||||
| 		height = 3 | ||||
| 	) | ||||
| 	decoder, err := NewDecoder(FormatZ16) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	img, _, err := decoder.Decode([]byte{0x00}, width, height) | ||||
| 	if err == nil { | ||||
| 		t.Errorf("expected to get a frame length mismatch") | ||||
| 	} | ||||
|  | ||||
| 	input := []byte{ | ||||
| 		0x0c, 0x00, 0x20, 0x03, | ||||
| 		0xa3, 0x01, 0x10, 0x00, | ||||
| 		0x56, 0x09, 0x5d, 0x00, | ||||
| 	} | ||||
| 	expected := image.NewGray16(image.Rect(0, 0, width, height)) | ||||
| 	expected.Stride = width * 2 | ||||
| 	expected.SetGray16(0, 0, color.Gray16{Y: 12}) | ||||
| 	expected.SetGray16(1, 0, color.Gray16{Y: 800}) | ||||
| 	expected.SetGray16(0, 1, color.Gray16{Y: 419}) | ||||
| 	expected.SetGray16(1, 1, color.Gray16{Y: 16}) | ||||
| 	expected.SetGray16(0, 2, color.Gray16{Y: 2390}) | ||||
| 	expected.SetGray16(1, 2, color.Gray16{Y: 93}) | ||||
|  | ||||
| 	img, _, err = decoder.Decode(input, width, height) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	if !reflect.DeepEqual(expected, img) { | ||||
| 		t.Errorf("Wrong decode result,\nexpected:\n%+v\ngot:\n%+v", expected, img) | ||||
| 	} | ||||
| } | ||||
| @@ -107,12 +107,14 @@ func TestBroadcast(t *testing.T) { | ||||
|  | ||||
| 							fps := float64(count) / duration.Seconds() | ||||
| 							if fps < pauseCond.expectedFPS-2 || fps > pauseCond.expectedFPS+2 { | ||||
| 								t.Fatal("Unexpected average FPS") | ||||
| 								t.Error("Unexpected average FPS") | ||||
| 								return | ||||
| 							} | ||||
|  | ||||
| 							droppedFramesPerSecond := float64(droppedFrames) / duration.Seconds() | ||||
| 							if droppedFramesPerSecond < pauseCond.expectedDrop-2 || droppedFramesPerSecond > pauseCond.expectedDrop+2 { | ||||
| 								t.Fatal("Unexpected drop count") | ||||
| 								t.Error("Unexpected drop count") | ||||
| 								return | ||||
| 							} | ||||
|  | ||||
| 							fpsChan <- []float64{fps, droppedFramesPerSecond, float64(lastFrameCount)} | ||||
|   | ||||
| @@ -144,9 +144,11 @@ func newInt16InterleavedDecoder() (Decoder, Format) { | ||||
| 		container := NewInt16Interleaved(chunkInfo) | ||||
|  | ||||
| 		if endian == hostEndian { | ||||
| 			data := container.Data | ||||
| 			dst := *(*[]byte)(unsafe.Pointer(&data)) | ||||
| 			hdr := (*reflect.SliceHeader)(unsafe.Pointer(&dst)) | ||||
| 			n := len(chunk) | ||||
| 			h := reflect.SliceHeader{Data: uintptr(unsafe.Pointer(&container.Data[0])), Len: n, Cap: n} | ||||
| 			dst := *(*[]byte)(unsafe.Pointer(&h)) | ||||
| 			hdr.Len, hdr.Cap = n, n | ||||
| 			copy(dst, chunk) | ||||
| 			return container, nil | ||||
| 		} | ||||
| @@ -188,9 +190,11 @@ func newInt16NonInterleavedDecoder() (Decoder, Format) { | ||||
|  | ||||
| 		if endian == hostEndian { | ||||
| 			for ch := 0; ch < channels; ch++ { | ||||
| 				data := container.Data[ch] | ||||
| 				dst := *(*[]byte)(unsafe.Pointer(&data)) | ||||
| 				hdr := (*reflect.SliceHeader)(unsafe.Pointer(&dst)) | ||||
| 				hdr.Len, hdr.Cap = chunkLen, chunkLen | ||||
| 				offset := ch * chunkLen | ||||
| 				h := reflect.SliceHeader{Data: uintptr(unsafe.Pointer(&container.Data[ch][0])), Len: chunkLen, Cap: chunkLen} | ||||
| 				dst := *(*[]byte)(unsafe.Pointer(&h)) | ||||
| 				copy(dst, chunk[offset:offset+chunkLen]) | ||||
| 			} | ||||
| 			return container, nil | ||||
| @@ -228,9 +232,11 @@ func newFloat32InterleavedDecoder() (Decoder, Format) { | ||||
| 		container := NewFloat32Interleaved(chunkInfo) | ||||
|  | ||||
| 		if endian == hostEndian { | ||||
| 			data := container.Data | ||||
| 			dst := *(*[]byte)(unsafe.Pointer(&data)) | ||||
| 			hdr := (*reflect.SliceHeader)(unsafe.Pointer(&dst)) | ||||
| 			n := len(chunk) | ||||
| 			h := reflect.SliceHeader{Data: uintptr(unsafe.Pointer(&container.Data[0])), Len: n, Cap: n} | ||||
| 			dst := *(*[]byte)(unsafe.Pointer(&h)) | ||||
| 			hdr.Len, hdr.Cap = n, n | ||||
| 			copy(dst, chunk) | ||||
| 			return container, nil | ||||
| 		} | ||||
| @@ -272,9 +278,11 @@ func newFloat32NonInterleavedDecoder() (Decoder, Format) { | ||||
|  | ||||
| 		if endian == hostEndian { | ||||
| 			for ch := 0; ch < channels; ch++ { | ||||
| 				data := container.Data[ch] | ||||
| 				dst := *(*[]byte)(unsafe.Pointer(&data)) | ||||
| 				hdr := (*reflect.SliceHeader)(unsafe.Pointer(&dst)) | ||||
| 				hdr.Len, hdr.Cap = chunkLen, chunkLen | ||||
| 				offset := ch * chunkLen | ||||
| 				h := reflect.SliceHeader{Data: uintptr(unsafe.Pointer(&container.Data[ch][0])), Len: chunkLen, Cap: chunkLen} | ||||
| 				dst := *(*[]byte)(unsafe.Pointer(&h)) | ||||
| 				copy(dst, chunk[offset:offset+chunkLen]) | ||||
| 			} | ||||
| 			return container, nil | ||||
|   | ||||
							
								
								
									
										15
									
								
								rtpreader.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								rtpreader.go
									
									
									
									
									
								
							| @@ -1,15 +1,20 @@ | ||||
| package mediadevices | ||||
|  | ||||
| import "github.com/pion/rtp" | ||||
| import ( | ||||
| 	"github.com/pion/mediadevices/pkg/codec" | ||||
| 	"github.com/pion/rtp" | ||||
| ) | ||||
|  | ||||
| type RTPReadCloser interface { | ||||
| 	Read() (pkts []*rtp.Packet, release func(), err error) | ||||
| 	Close() error | ||||
| 	codec.Controllable | ||||
| } | ||||
|  | ||||
| type rtpReadCloserImpl struct { | ||||
| 	readFn  func() ([]*rtp.Packet, func(), error) | ||||
| 	closeFn func() error | ||||
| 	readFn       func() ([]*rtp.Packet, func(), error) | ||||
| 	closeFn      func() error | ||||
| 	controllerFn func() codec.EncoderController | ||||
| } | ||||
|  | ||||
| func (r *rtpReadCloserImpl) Read() ([]*rtp.Packet, func(), error) { | ||||
| @@ -19,3 +24,7 @@ func (r *rtpReadCloserImpl) Read() ([]*rtp.Packet, func(), error) { | ||||
| func (r *rtpReadCloserImpl) Close() error { | ||||
| 	return r.closeFn() | ||||
| } | ||||
|  | ||||
| func (r *rtpReadCloserImpl) Controller() codec.EncoderController { | ||||
| 	return r.controllerFn() | ||||
| } | ||||
|   | ||||
							
								
								
									
										80
									
								
								track.go
									
									
									
									
									
								
							
							
						
						
									
										80
									
								
								track.go
									
									
									
									
									
								
							| @@ -3,6 +3,7 @@ package mediadevices | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"github.com/pion/rtcp" | ||||
| 	"image" | ||||
| 	"io" | ||||
| 	"strings" | ||||
| @@ -56,6 +57,8 @@ type Track interface { | ||||
| 	Kind() webrtc.RTPCodecType | ||||
| 	// StreamID is the group this track belongs too. This must be unique | ||||
| 	StreamID() string | ||||
| 	// RID is the RTP Stearm ID for this track. This is only used for Simulcast | ||||
| 	RID() string | ||||
| 	// Bind binds the current track source to the given peer connection. In Pion/webrtc v3, the bind | ||||
| 	// call will happen automatically after the SDP negotiation. Users won't need to call this manually. | ||||
| 	Bind(webrtc.TrackLocalContext) (webrtc.RTPCodecParameters, error) | ||||
| @@ -64,6 +67,8 @@ type Track interface { | ||||
| 	Unbind(webrtc.TrackLocalContext) error | ||||
| 	// NewRTPReader creates a new reader from the source. The reader will encode the source, and packetize | ||||
| 	// the encoded data in RTP format with given mtu size. | ||||
| 	// | ||||
| 	// Note: `mtu int` will be changed to `mtu uint16` in a future update. | ||||
| 	NewRTPReader(codecName string, ssrc uint32, mtu int) (RTPReadCloser, error) | ||||
| 	// NewEncodedReader creates a EncodedReadCloser that reads the encoded data in codecName format | ||||
| 	NewEncodedReader(codecName string) (EncodedReadCloser, error) | ||||
| @@ -113,6 +118,11 @@ func (track *baseTrack) StreamID() string { | ||||
| 	return generator.String() | ||||
| } | ||||
|  | ||||
| // RID is only relevant if you wish to use Simulcast | ||||
| func (track *baseTrack) RID() string { | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| // OnEnded sets an error handler. When a track has been created and started, if an | ||||
| // error occurs, handler will get called with the error given to the parameter. | ||||
| func (track *baseTrack) OnEnded(handler func(error)) { | ||||
| @@ -148,6 +158,7 @@ func (track *baseTrack) bind(ctx webrtc.TrackLocalContext, specializedTrack Trac | ||||
| 	defer track.mu.Unlock() | ||||
|  | ||||
| 	signalCh := make(chan chan<- struct{}) | ||||
| 	var stopRead chan struct{} | ||||
| 	track.activePeerConnections[ctx.ID()] = signalCh | ||||
|  | ||||
| 	var encodedReader RTPReadCloser | ||||
| @@ -173,6 +184,7 @@ func (track *baseTrack) bind(ctx webrtc.TrackLocalContext, specializedTrack Trac | ||||
| 		var doneCh chan<- struct{} | ||||
| 		writer := ctx.WriteStream() | ||||
| 		defer func() { | ||||
| 			close(stopRead) | ||||
| 			encodedReader.Close() | ||||
|  | ||||
| 			// When there's another call to unbind, it won't block since we mark the signalCh to be closed | ||||
| @@ -205,25 +217,65 @@ func (track *baseTrack) bind(ctx webrtc.TrackLocalContext, specializedTrack Trac | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	keyFrameController, ok := encodedReader.Controller().(codec.KeyFrameController) | ||||
| 	if ok { | ||||
| 		stopRead = make(chan struct{}) | ||||
| 		go func() { | ||||
| 			reader := ctx.ReadStream() | ||||
| 			for { | ||||
| 				select { | ||||
| 				case <-stopRead: | ||||
| 					return | ||||
| 				default: | ||||
| 				} | ||||
|  | ||||
| 				pkts, _, err := reader.ReadRTCP() | ||||
| 				if err != nil { | ||||
| 					track.onError(err) | ||||
| 					return | ||||
| 				} | ||||
|  | ||||
| 				for _, pkt := range pkts { | ||||
| 					switch pkt.(type) { | ||||
| 					case *rtcp.PictureLossIndication, *rtcp.FullIntraRequest: | ||||
| 						if err := keyFrameController.ForceKeyFrame(); err != nil { | ||||
| 							track.onError(err) | ||||
| 							return | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		}() | ||||
| 	} | ||||
|  | ||||
| 	return selectedCodec, nil | ||||
| } | ||||
|  | ||||
| func (track *baseTrack) unbind(ctx webrtc.TrackLocalContext) error { | ||||
| 	track.mu.Lock() | ||||
| 	defer track.mu.Unlock() | ||||
|  | ||||
| 	ch, ok := track.activePeerConnections[ctx.ID()] | ||||
| 	if !ok { | ||||
| 		return errNotFoundPeerConnection | ||||
| 	ch, err := track.removeActivePeerConnection(ctx.ID()) | ||||
| 	if err != err { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	doneCh := make(chan struct{}) | ||||
| 	ch <- doneCh | ||||
| 	<-doneCh | ||||
| 	delete(track.activePeerConnections, ctx.ID()) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (track *baseTrack) removeActivePeerConnection(id string) (chan<- chan<- struct{}, error) { | ||||
| 	track.mu.Lock() | ||||
| 	defer track.mu.Unlock() | ||||
|  | ||||
| 	ch, ok := track.activePeerConnections[id] | ||||
| 	if !ok { | ||||
| 		return nil, errNotFoundPeerConnection | ||||
| 	} | ||||
| 	delete(track.activePeerConnections, id) | ||||
|  | ||||
| 	return ch, nil | ||||
| } | ||||
|  | ||||
| func newTrackFromDriver(d driver.Driver, constraints MediaTrackConstraints, selector *CodecSelector) (Track, error) { | ||||
| 	if err := d.Open(); err != nil { | ||||
| 		return nil, err | ||||
| @@ -316,7 +368,8 @@ func (track *VideoTrack) newEncodedReader(codecNames ...string) (EncodedReadClos | ||||
| 			} | ||||
| 			return buffer, release, err | ||||
| 		}, | ||||
| 		closeFn: encodedReader.Close, | ||||
| 		closeFn:      encodedReader.Close, | ||||
| 		controllerFn: encodedReader.Controller, | ||||
| 	}, selectedCodec, nil | ||||
| } | ||||
|  | ||||
| @@ -339,7 +392,7 @@ func (track *VideoTrack) NewRTPReader(codecName string, ssrc uint32, mtu int) (R | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	packetizer := rtp.NewPacketizer(mtu, uint8(selectedCodec.PayloadType), ssrc, selectedCodec.Payloader, rtp.NewRandomSequencer(), selectedCodec.ClockRate) | ||||
| 	packetizer := rtp.NewPacketizer(uint16(mtu), uint8(selectedCodec.PayloadType), ssrc, selectedCodec.Payloader, rtp.NewRandomSequencer(), selectedCodec.ClockRate) | ||||
|  | ||||
| 	return &rtpReadCloserImpl{ | ||||
| 		readFn: func() ([]*rtp.Packet, func(), error) { | ||||
| @@ -354,7 +407,8 @@ func (track *VideoTrack) NewRTPReader(codecName string, ssrc uint32, mtu int) (R | ||||
| 			pkts := packetizer.Packetize(encoded.Data, encoded.Samples) | ||||
| 			return pkts, release, err | ||||
| 		}, | ||||
| 		closeFn: encodedReader.Close, | ||||
| 		closeFn:      encodedReader.Close, | ||||
| 		controllerFn: encodedReader.Controller, | ||||
| 	}, nil | ||||
| } | ||||
|  | ||||
| @@ -365,7 +419,7 @@ type AudioTrack struct { | ||||
| 	*audio.Broadcaster | ||||
| } | ||||
|  | ||||
| // NewAudioTrack constructs a new VideoTrack | ||||
| // NewAudioTrack constructs a new AudioTrack | ||||
| func NewAudioTrack(source AudioSource, selector *CodecSelector) Track { | ||||
| 	return newAudioTrackFromReader(source, source, selector) | ||||
| } | ||||
| @@ -425,7 +479,7 @@ func (track *AudioTrack) newEncodedReader(codecNames ...string) (EncodedReadClos | ||||
| 		return nil, nil, err | ||||
| 	} | ||||
|  | ||||
| 	sample := newAudioSampler(selectedCodec.ClockRate, inputProp.Latency) | ||||
| 	sample := newAudioSampler(selectedCodec.ClockRate, selectedCodec.Latency) | ||||
|  | ||||
| 	return &encodedReadCloserImpl{ | ||||
| 		readFn: func() (EncodedBuffer, func(), error) { | ||||
| @@ -459,7 +513,7 @@ func (track *AudioTrack) NewRTPReader(codecName string, ssrc uint32, mtu int) (R | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	packetizer := rtp.NewPacketizer(mtu, uint8(selectedCodec.PayloadType), ssrc, selectedCodec.Payloader, rtp.NewRandomSequencer(), selectedCodec.ClockRate) | ||||
| 	packetizer := rtp.NewPacketizer(uint16(mtu), uint8(selectedCodec.PayloadType), ssrc, selectedCodec.Payloader, rtp.NewRandomSequencer(), selectedCodec.ClockRate) | ||||
|  | ||||
| 	return &rtpReadCloserImpl{ | ||||
| 		readFn: func() ([]*rtp.Packet, func(), error) { | ||||
|   | ||||
		Reference in New Issue
	
	Block a user