|
@@ -8,6 +8,7 @@ import (
|
|
|
"mime/multipart"
|
|
|
"net/http"
|
|
|
"net/textproto"
|
|
|
+ "os"
|
|
|
"strings"
|
|
|
"syscall"
|
|
|
|
|
@@ -15,38 +16,72 @@ import (
|
|
|
"github.com/vladimirvivien/go4vl/v4l2"
|
|
|
)
|
|
|
|
|
|
-var (
|
|
|
- camera *device.Device
|
|
|
- frames <-chan []byte
|
|
|
- fps uint32 = 30
|
|
|
-
|
|
|
- 1920 x 1080 60fps = 55Mbps
|
|
|
- 1920 x 1080 30fps = 50Mbps
|
|
|
- 1920 x 1080 25fps = 40Mbps
|
|
|
- 1920 x 1080 20fps = 30Mbps
|
|
|
- 1920 x 1080 10fps = 15Mbps
|
|
|
-
|
|
|
- 1360 x 768 60fps = 28Mbps
|
|
|
- 1360 x 768 30fps = 25Mbps
|
|
|
- 1360 x 768 25fps = 20Mbps
|
|
|
- 1360 x 768 20fps = 18Mbps
|
|
|
- 1360 x 768 10fps = 10Mbps
|
|
|
- */
|
|
|
- pixfmt v4l2.FourCCType
|
|
|
- width = 1920
|
|
|
- height = 1080
|
|
|
- streamInfo string
|
|
|
-)
|
|
|
+type Config struct {
|
|
|
+ DeviceName string
|
|
|
+ Resolution *SizeInfo
|
|
|
+}
|
|
|
+
|
|
|
+type Instance struct {
|
|
|
+ Config *Config
|
|
|
+ SupportedResolutions []FormatInfo
|
|
|
+ Capturing bool
|
|
|
+ camera *device.Device
|
|
|
+ frames_buff <-chan []byte
|
|
|
+ pixfmt v4l2.FourCCType
|
|
|
+ width int
|
|
|
+ height int
|
|
|
+ streamInfo string
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+func NewInstance(config *Config) (*Instance, error) {
|
|
|
+ if config == nil {
|
|
|
+ return nil, fmt.Errorf("config cannot be nil")
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ if _, err := os.Stat(config.DeviceName); os.IsNotExist(err) {
|
|
|
+ return nil, fmt.Errorf("video device %s does not exist", config.DeviceName)
|
|
|
+ } else if err != nil {
|
|
|
+ return nil, fmt.Errorf("failed to check video device: %w", err)
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ isValidDevice, err := checkVideoCaptureDevice(config.DeviceName)
|
|
|
+ if err != nil {
|
|
|
+ return nil, fmt.Errorf("failed to check video device: %w", err)
|
|
|
+ }
|
|
|
+
|
|
|
+ if !isValidDevice {
|
|
|
+ return nil, fmt.Errorf("device %s is not a video capture device", config.DeviceName)
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ formatInfo, err := GetV4L2FormatInfo(config.DeviceName)
|
|
|
+ if err != nil {
|
|
|
+ return nil, fmt.Errorf("failed to get video device format info: %w", err)
|
|
|
+ }
|
|
|
+
|
|
|
+ if len(formatInfo) == 0 {
|
|
|
+ return nil, fmt.Errorf("no supported formats found for device %s", config.DeviceName)
|
|
|
+ }
|
|
|
+
|
|
|
+ return &Instance{
|
|
|
+ Config: config,
|
|
|
+ Capturing: false,
|
|
|
+ SupportedResolutions: formatInfo,
|
|
|
+ }, nil
|
|
|
+}
|
|
|
|
|
|
|
|
|
-func serveVideoStream(w http.ResponseWriter, req *http.Request) {
|
|
|
+func (i *Instance) ServeVideoStream(w http.ResponseWriter, req *http.Request) {
|
|
|
mimeWriter := multipart.NewWriter(w)
|
|
|
w.Header().Set("Content-Type", fmt.Sprintf("multipart/x-mixed-replace; boundary=%s", mimeWriter.Boundary()))
|
|
|
partHeader := make(textproto.MIMEHeader)
|
|
|
partHeader.Add("Content-Type", "image/jpeg")
|
|
|
|
|
|
var frame []byte
|
|
|
- for frame = range frames {
|
|
|
+ for frame = range i.frames_buff {
|
|
|
if len(frame) == 0 {
|
|
|
log.Print("skipping empty frame")
|
|
|
continue
|
|
@@ -69,46 +104,64 @@ func serveVideoStream(w http.ResponseWriter, req *http.Request) {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-
|
|
|
-func startVideoServer() {
|
|
|
- port := ":9090"
|
|
|
- devName := "/dev/video0"
|
|
|
- frameRate := int(fps)
|
|
|
+
|
|
|
+func (i *Instance) StartVideoCapture(selectedFPS int) error {
|
|
|
+ if i.Capturing {
|
|
|
+ return fmt.Errorf("video capture already started")
|
|
|
+ }
|
|
|
+
|
|
|
+ devName := i.Config.DeviceName
|
|
|
+ frameRate := 25
|
|
|
buffSize := 8
|
|
|
format := "mjpeg"
|
|
|
|
|
|
+ if i.Config.Resolution == nil {
|
|
|
+ return fmt.Errorf("resolution not provided")
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ if i.Config.Resolution != nil {
|
|
|
+ for _, size := range i.Config.Resolution.FPS {
|
|
|
+ if size == selectedFPS {
|
|
|
+ frameRate = size
|
|
|
+ break
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if frameRate != selectedFPS {
|
|
|
+ log.Printf("selected FPS %d is not supported, using default %d", selectedFPS, frameRate)
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ log.Printf("no resolution provided, using default %d", frameRate)
|
|
|
+ }
|
|
|
+
|
|
|
|
|
|
isCaptureDev, err := checkVideoCaptureDevice(devName)
|
|
|
if err != nil {
|
|
|
- panic(err)
|
|
|
+ return fmt.Errorf("failed to check video device: %w", err)
|
|
|
}
|
|
|
if !isCaptureDev {
|
|
|
- panic("target device is not a capture-able device")
|
|
|
+ return fmt.Errorf("device %s is not a video capture device", devName)
|
|
|
}
|
|
|
+
|
|
|
+
|
|
|
camera, err := device.Open(devName,
|
|
|
device.WithIOType(v4l2.IOTypeMMAP),
|
|
|
- device.WithPixFormat(v4l2.PixFormat{PixelFormat: getFormatType(format), Width: uint32(width), Height: uint32(height), Field: v4l2.FieldAny}),
|
|
|
+ device.WithPixFormat(v4l2.PixFormat{
|
|
|
+ PixelFormat: getFormatType(format),
|
|
|
+ Width: uint32(i.Config.Resolution.Width),
|
|
|
+ Height: uint32(i.Config.Resolution.Height),
|
|
|
+ Field: v4l2.FieldAny,
|
|
|
+ }),
|
|
|
device.WithFPS(uint32(frameRate)),
|
|
|
device.WithBufferSize(uint32(buffSize)),
|
|
|
)
|
|
|
|
|
|
if err != nil {
|
|
|
- log.Fatalf("failed to open device: %s", err)
|
|
|
+ return fmt.Errorf("failed to open video device: %w", err)
|
|
|
}
|
|
|
- defer camera.Close()
|
|
|
|
|
|
-
|
|
|
- formatInfo, err := getV4L2FormatInfo(devName)
|
|
|
- if err != nil {
|
|
|
- log.Fatal(err)
|
|
|
- }
|
|
|
- for _, format := range formatInfo {
|
|
|
- fmt.Printf("Format: %s\n", format.Format)
|
|
|
- for _, size := range format.Sizes {
|
|
|
- fmt.Printf(" Size: %dx%d\n", size.Width, size.Height)
|
|
|
- fmt.Printf(" FPS: %v\n", size.FPS)
|
|
|
- }
|
|
|
- }
|
|
|
+ i.camera = camera
|
|
|
|
|
|
caps := camera.Capability()
|
|
|
log.Printf("device [%s] opened\n", devName)
|
|
@@ -122,8 +175,11 @@ func startVideoServer() {
|
|
|
}
|
|
|
log.Printf("Current format: %s", currFmt)
|
|
|
|
|
|
- pixfmt = currFmt.PixelFormat
|
|
|
- streamInfo = fmt.Sprintf("%s - %s [%dx%d] %d fps",
|
|
|
+ i.pixfmt = currFmt.PixelFormat
|
|
|
+ i.width = int(currFmt.Width)
|
|
|
+ i.height = int(currFmt.Height)
|
|
|
+
|
|
|
+ i.streamInfo = fmt.Sprintf("%s - %s [%dx%d] %d fps",
|
|
|
caps.Card,
|
|
|
v4l2.PixelFormats[currFmt.PixelFormat],
|
|
|
currFmt.Width, currFmt.Height, frameRate,
|
|
@@ -140,17 +196,40 @@ func startVideoServer() {
|
|
|
}()
|
|
|
|
|
|
|
|
|
- frames = camera.GetOutput()
|
|
|
+ i.frames_buff = camera.GetOutput()
|
|
|
|
|
|
log.Printf("device capture started (buffer size set %d)", camera.BufferCount())
|
|
|
- log.Printf("starting server on port %s", port)
|
|
|
- log.Println("use url path /webcam")
|
|
|
+ i.Capturing = true
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+func (i *Instance) GetStreamInfo() string {
|
|
|
+ return i.streamInfo
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+func (i *Instance) IsCapturing() bool {
|
|
|
+ return i.Capturing
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+func (i *Instance) StopCapture() error {
|
|
|
+ if i.camera != nil {
|
|
|
+ i.camera.Stop()
|
|
|
+ i.camera.Close()
|
|
|
+ i.camera = nil
|
|
|
+ }
|
|
|
+ i.Capturing = false
|
|
|
+ return nil
|
|
|
+}
|
|
|
|
|
|
-
|
|
|
- http.HandleFunc("/stream", serveVideoStream)
|
|
|
- if err := http.ListenAndServe(port, nil); err != nil {
|
|
|
- log.Fatal(err)
|
|
|
+
|
|
|
+func (i *Instance) Close() error {
|
|
|
+ if i.camera != nil {
|
|
|
+ i.StopCapture()
|
|
|
}
|
|
|
+ return nil
|
|
|
}
|
|
|
|
|
|
func getFormatType(fmtStr string) v4l2.FourCCType {
|