video_device.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. package usbcapture
  2. import (
  3. "bufio"
  4. "bytes"
  5. "fmt"
  6. "os/exec"
  7. "regexp"
  8. "strconv"
  9. "strings"
  10. )
  11. /*
  12. 1920 x 1080 60fps = 55Mbps //Edge not support
  13. 1920 x 1080 30fps = 50Mbps
  14. 1920 x 1080 25fps = 40Mbps
  15. 1920 x 1080 20fps = 30Mbps
  16. 1920 x 1080 10fps = 15Mbps
  17. 1360 x 768 60fps = 28Mbps
  18. 1360 x 768 30fps = 25Mbps
  19. 1360 x 768 25fps = 20Mbps
  20. 1360 x 768 20fps = 18Mbps
  21. 1360 x 768 10fps = 10Mbps
  22. */
  23. // Struct to store the size and fps info
  24. type FormatInfo struct {
  25. Format string
  26. Sizes []SizeInfo
  27. }
  28. type SizeInfo struct {
  29. Width int
  30. Height int
  31. FPS []int
  32. }
  33. // CheckVideoCaptureDevice checks if the given video device is a video capture device
  34. func checkVideoCaptureDevice(device string) (bool, error) {
  35. // Run v4l2-ctl to get device capabilities
  36. cmd := exec.Command("v4l2-ctl", "--device", device, "--all")
  37. output, err := cmd.CombinedOutput()
  38. if err != nil {
  39. return false, fmt.Errorf("failed to execute v4l2-ctl: %w", err)
  40. }
  41. // Convert output to string and check for the "Video Capture" capability
  42. outputStr := string(output)
  43. if strings.Contains(outputStr, "Video Capture") {
  44. return true, nil
  45. }
  46. return false, nil
  47. }
  48. func deviceSupportResolution(devicePath string, resolution *CaptureResolution) (bool, error) {
  49. formatInfo, err := GetV4L2FormatInfo(devicePath)
  50. if err != nil {
  51. return false, err
  52. }
  53. // Yes, this is an O(N^3) operation, but a video decices rarely have supported resolution
  54. // more than 20 combinations. The compute time should be fine
  55. for _, res := range formatInfo {
  56. for _, size := range res.Sizes {
  57. //Check if there is a matching resolution
  58. if size.Height == resolution.Height && size.Width == resolution.Width {
  59. //Matching resolution. Check if the required FPS is supported
  60. for _, fps := range size.FPS {
  61. if fps == resolution.FPS {
  62. return true, nil
  63. }
  64. }
  65. }
  66. }
  67. }
  68. return false, nil
  69. }
  70. func PrintV4L2FormatInfo(devicePath string) {
  71. // Check if the device is a video capture device
  72. isCapture, err := checkVideoCaptureDevice(devicePath)
  73. if err != nil {
  74. fmt.Printf("Error checking device: %v\n", err)
  75. return
  76. }
  77. if !isCapture {
  78. fmt.Printf("Device %s is not a video capture device\n", devicePath)
  79. return
  80. }
  81. // Get format info
  82. formats, err := GetV4L2FormatInfo(devicePath)
  83. if err != nil {
  84. fmt.Printf("Error getting format info: %v\n", err)
  85. return
  86. }
  87. // Print format info
  88. for _, format := range formats {
  89. fmt.Printf("Format: %s\n", format.Format)
  90. for _, size := range format.Sizes {
  91. fmt.Printf(" Size: %dx%d\n", size.Width, size.Height)
  92. fmt.Printf(" FPS: %v\n", size.FPS)
  93. }
  94. }
  95. }
  96. // Function to run the v4l2-ctl command and parse the output
  97. func GetV4L2FormatInfo(devicePath string) ([]FormatInfo, error) {
  98. // Run the v4l2-ctl command to list formats
  99. cmd := exec.Command("v4l2-ctl", "--list-formats-ext", "-d", devicePath)
  100. var out bytes.Buffer
  101. cmd.Stdout = &out
  102. err := cmd.Run()
  103. if err != nil {
  104. return nil, err
  105. }
  106. // Parse the output
  107. var formats []FormatInfo
  108. var currentFormat *FormatInfo
  109. scanner := bufio.NewScanner(&out)
  110. formatRegex := regexp.MustCompile(`\[(\d+)\]: '(\S+)'`)
  111. sizeRegex := regexp.MustCompile(`Size: Discrete (\d+)x(\d+)`)
  112. intervalRegex := regexp.MustCompile(`Interval: Discrete (\d+\.\d+)s \((\d+\.\d+) fps\)`)
  113. for scanner.Scan() {
  114. line := scanner.Text()
  115. // Match format line
  116. if matches := formatRegex.FindStringSubmatch(line); matches != nil {
  117. if currentFormat != nil {
  118. formats = append(formats, *currentFormat)
  119. }
  120. // Start a new format entry
  121. currentFormat = &FormatInfo{
  122. Format: matches[2],
  123. }
  124. }
  125. // Match size line
  126. if matches := sizeRegex.FindStringSubmatch(line); matches != nil {
  127. width, _ := strconv.Atoi(matches[1])
  128. height, _ := strconv.Atoi(matches[2])
  129. // Initialize the size entry
  130. sizeInfo := SizeInfo{
  131. Width: width,
  132. Height: height,
  133. }
  134. // Match FPS intervals for the current size
  135. for scanner.Scan() {
  136. line = scanner.Text()
  137. if fpsMatches := intervalRegex.FindStringSubmatch(line); fpsMatches != nil {
  138. fps, _ := strconv.ParseFloat(fpsMatches[2], 32)
  139. sizeInfo.FPS = append(sizeInfo.FPS, int(fps))
  140. } else {
  141. // Stop parsing FPS intervals when no more matches are found
  142. break
  143. }
  144. }
  145. // Add the size information to the current format
  146. currentFormat.Sizes = append(currentFormat.Sizes, sizeInfo)
  147. }
  148. }
  149. // Append the last format if present
  150. if currentFormat != nil {
  151. formats = append(formats, *currentFormat)
  152. }
  153. if err := scanner.Err(); err != nil {
  154. return nil, err
  155. }
  156. return formats, nil
  157. }