video_device.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  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 PrintV4L2FormatInfo(devicePath string) {
  49. // Check if the device is a video capture device
  50. isCapture, err := checkVideoCaptureDevice(devicePath)
  51. if err != nil {
  52. fmt.Printf("Error checking device: %v\n", err)
  53. return
  54. }
  55. if !isCapture {
  56. fmt.Printf("Device %s is not a video capture device\n", devicePath)
  57. return
  58. }
  59. // Get format info
  60. formats, err := GetV4L2FormatInfo(devicePath)
  61. if err != nil {
  62. fmt.Printf("Error getting format info: %v\n", err)
  63. return
  64. }
  65. // Print format info
  66. for _, format := range formats {
  67. fmt.Printf("Format: %s\n", format.Format)
  68. for _, size := range format.Sizes {
  69. fmt.Printf(" Size: %dx%d\n", size.Width, size.Height)
  70. fmt.Printf(" FPS: %v\n", size.FPS)
  71. }
  72. }
  73. }
  74. // Function to run the v4l2-ctl command and parse the output
  75. func GetV4L2FormatInfo(devicePath string) ([]FormatInfo, error) {
  76. // Run the v4l2-ctl command to list formats
  77. cmd := exec.Command("v4l2-ctl", "--list-formats-ext", "-d", devicePath)
  78. var out bytes.Buffer
  79. cmd.Stdout = &out
  80. err := cmd.Run()
  81. if err != nil {
  82. return nil, err
  83. }
  84. // Parse the output
  85. var formats []FormatInfo
  86. var currentFormat *FormatInfo
  87. scanner := bufio.NewScanner(&out)
  88. formatRegex := regexp.MustCompile(`\[(\d+)\]: '(\S+)'`)
  89. sizeRegex := regexp.MustCompile(`Size: Discrete (\d+)x(\d+)`)
  90. intervalRegex := regexp.MustCompile(`Interval: Discrete (\d+\.\d+)s \((\d+\.\d+) fps\)`)
  91. for scanner.Scan() {
  92. line := scanner.Text()
  93. // Match format line
  94. if matches := formatRegex.FindStringSubmatch(line); matches != nil {
  95. if currentFormat != nil {
  96. formats = append(formats, *currentFormat)
  97. }
  98. // Start a new format entry
  99. currentFormat = &FormatInfo{
  100. Format: matches[2],
  101. }
  102. }
  103. // Match size line
  104. if matches := sizeRegex.FindStringSubmatch(line); matches != nil {
  105. width, _ := strconv.Atoi(matches[1])
  106. height, _ := strconv.Atoi(matches[2])
  107. // Initialize the size entry
  108. sizeInfo := SizeInfo{
  109. Width: width,
  110. Height: height,
  111. }
  112. // Match FPS intervals for the current size
  113. for scanner.Scan() {
  114. line = scanner.Text()
  115. if fpsMatches := intervalRegex.FindStringSubmatch(line); fpsMatches != nil {
  116. fps, _ := strconv.ParseInt(fpsMatches[2], 10, 0)
  117. sizeInfo.FPS = append(sizeInfo.FPS, int(fps))
  118. } else {
  119. // Stop parsing FPS intervals when no more matches are found
  120. break
  121. }
  122. }
  123. // Add the size information to the current format
  124. currentFormat.Sizes = append(currentFormat.Sizes, sizeInfo)
  125. }
  126. }
  127. // Append the last format if present
  128. if currentFormat != nil {
  129. formats = append(formats, *currentFormat)
  130. }
  131. if err := scanner.Err(); err != nil {
  132. return nil, err
  133. }
  134. return formats, nil
  135. }