| /* |
| * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "webrtc/modules/video_capture/linux/device_info_linux.h" |
| |
| #include <errno.h> |
| #include <fcntl.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <sys/ioctl.h> |
| #include <sys/stat.h> |
| #include <unistd.h> |
| //v4l includes |
| #include <linux/videodev2.h> |
| |
| #include "webrtc/rtc_base/logging.h" |
| |
| |
| namespace webrtc |
| { |
| namespace videocapturemodule |
| { |
| VideoCaptureModule::DeviceInfo* |
| VideoCaptureImpl::CreateDeviceInfo() |
| { |
| return new videocapturemodule::DeviceInfoLinux(); |
| } |
| |
| DeviceInfoLinux::DeviceInfoLinux() |
| : DeviceInfoImpl() |
| { |
| } |
| |
| int32_t DeviceInfoLinux::Init() |
| { |
| return 0; |
| } |
| |
| DeviceInfoLinux::~DeviceInfoLinux() |
| { |
| } |
| |
| uint32_t DeviceInfoLinux::NumberOfDevices() |
| { |
| LOG(LS_INFO) << __FUNCTION__; |
| |
| uint32_t count = 0; |
| char device[20]; |
| int fd = -1; |
| |
| /* detect /dev/video [0-63]VideoCaptureModule entries */ |
| for (int n = 0; n < 64; n++) |
| { |
| sprintf(device, "/dev/video%d", n); |
| if ((fd = open(device, O_RDONLY)) != -1) |
| { |
| close(fd); |
| count++; |
| } |
| } |
| |
| return count; |
| } |
| |
| int32_t DeviceInfoLinux::GetDeviceName( |
| uint32_t deviceNumber, |
| char* deviceNameUTF8, |
| uint32_t deviceNameLength, |
| char* deviceUniqueIdUTF8, |
| uint32_t deviceUniqueIdUTF8Length, |
| char* /*productUniqueIdUTF8*/, |
| uint32_t /*productUniqueIdUTF8Length*/) |
| { |
| LOG(LS_INFO) << __FUNCTION__; |
| |
| // Travel through /dev/video [0-63] |
| uint32_t count = 0; |
| char device[20]; |
| int fd = -1; |
| bool found = false; |
| for (int n = 0; n < 64; n++) |
| { |
| sprintf(device, "/dev/video%d", n); |
| if ((fd = open(device, O_RDONLY)) != -1) |
| { |
| if (count == deviceNumber) { |
| // Found the device |
| found = true; |
| break; |
| } else { |
| close(fd); |
| count++; |
| } |
| } |
| } |
| |
| if (!found) |
| return -1; |
| |
| // query device capabilities |
| struct v4l2_capability cap; |
| if (ioctl(fd, VIDIOC_QUERYCAP, &cap) < 0) |
| { |
| LOG(LS_INFO) << "error in querying the device capability for device " |
| << device << ". errno = " << errno; |
| close(fd); |
| return -1; |
| } |
| |
| close(fd); |
| |
| char cameraName[64]; |
| memset(deviceNameUTF8, 0, deviceNameLength); |
| memcpy(cameraName, cap.card, sizeof(cap.card)); |
| |
| if (deviceNameLength >= strlen(cameraName)) |
| { |
| memcpy(deviceNameUTF8, cameraName, strlen(cameraName)); |
| } |
| else |
| { |
| LOG(LS_INFO) << "buffer passed is too small"; |
| return -1; |
| } |
| |
| if (cap.bus_info[0] != 0) // may not available in all drivers |
| { |
| // copy device id |
| if (deviceUniqueIdUTF8Length >= strlen((const char*) cap.bus_info)) |
| { |
| memset(deviceUniqueIdUTF8, 0, deviceUniqueIdUTF8Length); |
| memcpy(deviceUniqueIdUTF8, cap.bus_info, |
| strlen((const char*) cap.bus_info)); |
| } |
| else |
| { |
| LOG(LS_INFO) << "buffer passed is too small"; |
| return -1; |
| } |
| } |
| |
| return 0; |
| } |
| |
| int32_t DeviceInfoLinux::CreateCapabilityMap( |
| const char* deviceUniqueIdUTF8) |
| { |
| int fd; |
| char device[32]; |
| bool found = false; |
| |
| const int32_t deviceUniqueIdUTF8Length = |
| (int32_t) strlen((char*) deviceUniqueIdUTF8); |
| if (deviceUniqueIdUTF8Length > kVideoCaptureUniqueNameLength) |
| { |
| LOG(LS_INFO) << "Device name too long"; |
| return -1; |
| } |
| LOG(LS_INFO) << "CreateCapabilityMap called for device " |
| << deviceUniqueIdUTF8; |
| |
| /* detect /dev/video [0-63] entries */ |
| for (int n = 0; n < 64; ++n) |
| { |
| sprintf(device, "/dev/video%d", n); |
| fd = open(device, O_RDONLY); |
| if (fd == -1) |
| continue; |
| |
| // query device capabilities |
| struct v4l2_capability cap; |
| if (ioctl(fd, VIDIOC_QUERYCAP, &cap) == 0) |
| { |
| if (cap.bus_info[0] != 0) |
| { |
| if (strncmp((const char*) cap.bus_info, |
| (const char*) deviceUniqueIdUTF8, |
| strlen((const char*) deviceUniqueIdUTF8)) == 0) //match with device id |
| { |
| found = true; |
| break; // fd matches with device unique id supplied |
| } |
| } |
| else //match for device name |
| { |
| if (IsDeviceNameMatches((const char*) cap.card, |
| (const char*) deviceUniqueIdUTF8)) |
| { |
| found = true; |
| break; |
| } |
| } |
| } |
| close(fd); // close since this is not the matching device |
| } |
| |
| if (!found) |
| { |
| LOG(LS_INFO) << "no matching device found"; |
| return -1; |
| } |
| |
| // now fd will point to the matching device |
| // reset old capability list. |
| _captureCapabilities.clear(); |
| |
| int size = FillCapabilities(fd); |
| close(fd); |
| |
| // Store the new used device name |
| _lastUsedDeviceNameLength = deviceUniqueIdUTF8Length; |
| _lastUsedDeviceName = (char*) realloc(_lastUsedDeviceName, |
| _lastUsedDeviceNameLength + 1); |
| memcpy(_lastUsedDeviceName, deviceUniqueIdUTF8, _lastUsedDeviceNameLength + 1); |
| |
| LOG(LS_INFO) << "CreateCapabilityMap " << _captureCapabilities.size(); |
| |
| return size; |
| } |
| |
| bool DeviceInfoLinux::IsDeviceNameMatches(const char* name, |
| const char* deviceUniqueIdUTF8) |
| { |
| if (strncmp(deviceUniqueIdUTF8, name, strlen(name)) == 0) |
| return true; |
| return false; |
| } |
| |
| int32_t DeviceInfoLinux::FillCapabilities(int fd) |
| { |
| |
| // set image format |
| struct v4l2_format video_fmt; |
| memset(&video_fmt, 0, sizeof(struct v4l2_format)); |
| |
| video_fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; |
| video_fmt.fmt.pix.sizeimage = 0; |
| |
| int totalFmts = 4; |
| unsigned int videoFormats[] = { |
| V4L2_PIX_FMT_MJPEG, |
| V4L2_PIX_FMT_YUV420, |
| V4L2_PIX_FMT_YUYV, |
| V4L2_PIX_FMT_UYVY }; |
| |
| int sizes = 13; |
| unsigned int size[][2] = { { 128, 96 }, { 160, 120 }, { 176, 144 }, |
| { 320, 240 }, { 352, 288 }, { 640, 480 }, |
| { 704, 576 }, { 800, 600 }, { 960, 720 }, |
| { 1280, 720 }, { 1024, 768 }, { 1440, 1080 }, |
| { 1920, 1080 } }; |
| |
| int index = 0; |
| for (int fmts = 0; fmts < totalFmts; fmts++) |
| { |
| for (int i = 0; i < sizes; i++) |
| { |
| video_fmt.fmt.pix.pixelformat = videoFormats[fmts]; |
| video_fmt.fmt.pix.width = size[i][0]; |
| video_fmt.fmt.pix.height = size[i][1]; |
| |
| if (ioctl(fd, VIDIOC_TRY_FMT, &video_fmt) >= 0) |
| { |
| if ((video_fmt.fmt.pix.width == size[i][0]) |
| && (video_fmt.fmt.pix.height == size[i][1])) |
| { |
| VideoCaptureCapability cap; |
| cap.width = video_fmt.fmt.pix.width; |
| cap.height = video_fmt.fmt.pix.height; |
| if (videoFormats[fmts] == V4L2_PIX_FMT_YUYV) |
| { |
| cap.videoType = VideoType::kYUY2; |
| } |
| else if (videoFormats[fmts] == V4L2_PIX_FMT_YUV420) |
| { |
| cap.videoType = VideoType::kI420; |
| } |
| else if (videoFormats[fmts] == V4L2_PIX_FMT_MJPEG) |
| { |
| cap.videoType = VideoType::kMJPEG; |
| } |
| else if (videoFormats[fmts] == V4L2_PIX_FMT_UYVY) |
| { |
| cap.videoType = VideoType::kUYVY; |
| } |
| |
| // get fps of current camera mode |
| // V4l2 does not have a stable method of knowing so we just guess. |
| if (cap.width >= 800 && |
| cap.videoType != VideoType::kMJPEG) { |
| cap.maxFPS = 15; |
| } |
| else |
| { |
| cap.maxFPS = 30; |
| } |
| |
| _captureCapabilities.push_back(cap); |
| index++; |
| LOG(LS_VERBOSE) << "Camera capability, width:" << cap.width |
| << " height:" << cap.height << " type:" |
| << static_cast<int32_t>(cap.videoType) |
| << " fps:" << cap.maxFPS; |
| } |
| } |
| } |
| } |
| |
| LOG(LS_INFO) << "CreateCapabilityMap " << _captureCapabilities.size(); |
| return _captureCapabilities.size(); |
| } |
| |
| } // namespace videocapturemodule |
| } // namespace webrtc |