video_device.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. package usbcapture
  2. import (
  3. "bufio"
  4. "bytes"
  5. "fmt"
  6. "os/exec"
  7. "regexp"
  8. "strconv"
  9. "strings"
  10. )
  11. // Struct to store the size and fps info
  12. type FormatInfo struct {
  13. Format string
  14. Sizes []SizeInfo
  15. }
  16. type SizeInfo struct {
  17. Width int
  18. Height int
  19. FPS []float64
  20. }
  21. // CheckVideoCaptureDevice checks if the given video device is a video capture device
  22. func checkVideoCaptureDevice(device string) (bool, error) {
  23. // Run v4l2-ctl to get device capabilities
  24. cmd := exec.Command("v4l2-ctl", "--device", device, "--all")
  25. output, err := cmd.CombinedOutput()
  26. if err != nil {
  27. return false, fmt.Errorf("failed to execute v4l2-ctl: %w", err)
  28. }
  29. // Convert output to string and check for the "Video Capture" capability
  30. outputStr := string(output)
  31. if strings.Contains(outputStr, "Video Capture") {
  32. return true, nil
  33. }
  34. return false, nil
  35. }
  36. // Function to run the v4l2-ctl command and parse the output
  37. func getV4L2FormatInfo(devicePath string) ([]FormatInfo, error) {
  38. // Run the v4l2-ctl command to list formats
  39. cmd := exec.Command("v4l2-ctl", "--list-formats-ext", "-d", devicePath)
  40. var out bytes.Buffer
  41. cmd.Stdout = &out
  42. err := cmd.Run()
  43. if err != nil {
  44. return nil, err
  45. }
  46. // Parse the output
  47. var formats []FormatInfo
  48. var currentFormat *FormatInfo
  49. scanner := bufio.NewScanner(&out)
  50. formatRegex := regexp.MustCompile(`\[(\d+)\]: '(\S+)'`)
  51. sizeRegex := regexp.MustCompile(`Size: Discrete (\d+)x(\d+)`)
  52. intervalRegex := regexp.MustCompile(`Interval: Discrete (\d+\.\d+)s \((\d+\.\d+) fps\)`)
  53. for scanner.Scan() {
  54. line := scanner.Text()
  55. // Match format line
  56. if matches := formatRegex.FindStringSubmatch(line); matches != nil {
  57. if currentFormat != nil {
  58. formats = append(formats, *currentFormat)
  59. }
  60. // Start a new format entry
  61. currentFormat = &FormatInfo{
  62. Format: matches[2],
  63. }
  64. }
  65. // Match size line
  66. if matches := sizeRegex.FindStringSubmatch(line); matches != nil {
  67. width, _ := strconv.Atoi(matches[1])
  68. height, _ := strconv.Atoi(matches[2])
  69. // Initialize the size entry
  70. sizeInfo := SizeInfo{
  71. Width: width,
  72. Height: height,
  73. }
  74. // Match FPS intervals for the current size
  75. for scanner.Scan() {
  76. line = scanner.Text()
  77. if fpsMatches := intervalRegex.FindStringSubmatch(line); fpsMatches != nil {
  78. fps, _ := strconv.ParseFloat(fpsMatches[2], 64)
  79. sizeInfo.FPS = append(sizeInfo.FPS, fps)
  80. } else {
  81. // Stop parsing FPS intervals when no more matches are found
  82. break
  83. }
  84. }
  85. // Add the size information to the current format
  86. currentFormat.Sizes = append(currentFormat.Sizes, sizeInfo)
  87. }
  88. }
  89. // Append the last format if present
  90. if currentFormat != nil {
  91. formats = append(formats, *currentFormat)
  92. }
  93. if err := scanner.Err(); err != nil {
  94. return nil, err
  95. }
  96. return formats, nil
  97. }