mirror of
https://github.com/pion/mediadevices.git
synced 2025-10-05 16:46:51 +08:00
145 lines
3.0 KiB
Go
145 lines
3.0 KiB
Go
package camera
|
|
|
|
// #include <linux/videodev2.h>
|
|
import "C"
|
|
|
|
import (
|
|
"image"
|
|
"io"
|
|
"io/ioutil"
|
|
|
|
"github.com/blackjack/webcam"
|
|
"github.com/pion/mediadevices/pkg/driver"
|
|
"github.com/pion/mediadevices/pkg/frame"
|
|
"github.com/pion/mediadevices/pkg/io/video"
|
|
"github.com/pion/mediadevices/pkg/prop"
|
|
)
|
|
|
|
// Camera implementation using v4l2
|
|
// Reference: https://linuxtv.org/downloads/v4l-dvb-apis/uapi/v4l/videodev.html#videodev
|
|
type camera struct {
|
|
path string
|
|
cam *webcam.Webcam
|
|
formats map[webcam.PixelFormat]frame.Format
|
|
reversedFormats map[frame.Format]webcam.PixelFormat
|
|
started bool
|
|
}
|
|
|
|
func init() {
|
|
searchPath := "/dev/v4l/by-path/"
|
|
devices, err := ioutil.ReadDir(searchPath)
|
|
if err != nil {
|
|
// No v4l device.
|
|
return
|
|
}
|
|
for _, device := range devices {
|
|
cam := newCamera(searchPath + device.Name())
|
|
driver.GetManager().Register(cam, device.Name())
|
|
}
|
|
}
|
|
|
|
func newCamera(path string) *camera {
|
|
formats := map[webcam.PixelFormat]frame.Format{
|
|
webcam.PixelFormat(C.V4L2_PIX_FMT_YUYV): frame.FormatYUYV,
|
|
webcam.PixelFormat(C.V4L2_PIX_FMT_NV12): frame.FormatNV21,
|
|
webcam.PixelFormat(C.V4L2_PIX_FMT_MJPEG): frame.FormatMJPEG,
|
|
}
|
|
|
|
reversedFormats := make(map[frame.Format]webcam.PixelFormat)
|
|
for k, v := range formats {
|
|
reversedFormats[v] = k
|
|
}
|
|
|
|
return &camera{
|
|
path: path,
|
|
formats: formats,
|
|
reversedFormats: reversedFormats,
|
|
}
|
|
}
|
|
|
|
func (c *camera) Open() error {
|
|
cam, err := webcam.Open(c.path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.cam = cam
|
|
return nil
|
|
}
|
|
|
|
func (c *camera) Close() error {
|
|
if c.cam == nil {
|
|
return nil
|
|
}
|
|
|
|
if c.started {
|
|
c.cam.StopStreaming()
|
|
}
|
|
c.cam.Close()
|
|
return nil
|
|
}
|
|
|
|
func (c *camera) VideoRecord(p prop.Media) (video.Reader, error) {
|
|
decoder, err := frame.NewDecoder(p.FrameFormat)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pf := c.reversedFormats[p.FrameFormat]
|
|
_, _, _, err = c.cam.SetImageFormat(pf, uint32(p.Width), uint32(p.Height))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := c.cam.StartStreaming(); err != nil {
|
|
return nil, err
|
|
}
|
|
c.started = true
|
|
|
|
r := video.ReaderFunc(func() (img image.Image, err error) {
|
|
// Wait until a frame is ready
|
|
for {
|
|
err := c.cam.WaitForFrame(5)
|
|
switch err.(type) {
|
|
case nil:
|
|
case *webcam.Timeout:
|
|
continue
|
|
default:
|
|
// Camera has been stopped.
|
|
return nil, io.EOF
|
|
}
|
|
|
|
b, err := c.cam.ReadFrame()
|
|
if err != nil {
|
|
// Camera has been stopped.
|
|
return nil, io.EOF
|
|
}
|
|
|
|
// Frame is not ready.
|
|
if len(b) == 0 {
|
|
continue
|
|
}
|
|
|
|
return decoder.Decode(b, p.Width, p.Height)
|
|
}
|
|
})
|
|
|
|
return r, nil
|
|
}
|
|
|
|
func (c *camera) Properties() []prop.Media {
|
|
properties := make([]prop.Media, 0)
|
|
for format := range c.cam.GetSupportedFormats() {
|
|
for _, frameSize := range c.cam.GetSupportedFrameSizes(format) {
|
|
properties = append(properties, prop.Media{
|
|
Video: prop.Video{
|
|
Width: int(frameSize.MaxWidth),
|
|
Height: int(frameSize.MaxHeight),
|
|
FrameFormat: c.formats[format],
|
|
},
|
|
})
|
|
}
|
|
}
|
|
return properties
|
|
}
|