/* * 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 #include #include #include #include #include #include //v4l includes #include #include "webrtc/system_wrappers/include/trace.h" namespace webrtc { namespace videocapturemodule { VideoCaptureModule::DeviceInfo* VideoCaptureImpl::CreateDeviceInfo(const int32_t id) { return new videocapturemodule::DeviceInfoLinux(id); } DeviceInfoLinux::DeviceInfoLinux(const int32_t id) : DeviceInfoImpl(id) { } int32_t DeviceInfoLinux::Init() { return 0; } DeviceInfoLinux::~DeviceInfoLinux() { } uint32_t DeviceInfoLinux::NumberOfDevices() { WEBRTC_TRACE(webrtc::kTraceApiCall, webrtc::kTraceVideoCapture, _id, "%s", __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*/) { WEBRTC_TRACE(webrtc::kTraceApiCall, webrtc::kTraceVideoCapture, _id, "%s", __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) { WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, _id, "error in querying the device capability for device %s. errno = %d", device, 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 { WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, _id, "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 { WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, _id, "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) { WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, _id, "Device name too long"); return -1; } WEBRTC_TRACE(webrtc::kTraceInfo, webrtc::kTraceVideoCapture, _id, "CreateCapabilityMap called for device %s", 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) { WEBRTC_TRACE(webrtc::kTraceError, webrtc::kTraceVideoCapture, _id, "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); WEBRTC_TRACE(webrtc::kTraceInfo, webrtc::kTraceVideoCapture, _id, "CreateCapabilityMap %u", static_cast(_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; cap.expectedCaptureDelay = 120; if (videoFormats[fmts] == V4L2_PIX_FMT_YUYV) { cap.rawType = kVideoYUY2; } else if (videoFormats[fmts] == V4L2_PIX_FMT_YUV420) { cap.rawType = kVideoI420; } else if (videoFormats[fmts] == V4L2_PIX_FMT_MJPEG) { cap.rawType = kVideoMJPEG; } else if (videoFormats[fmts] == V4L2_PIX_FMT_UYVY) { cap.rawType = kVideoUYVY; } // get fps of current camera mode // V4l2 does not have a stable method of knowing so we just guess. if(cap.width >= 800 && cap.rawType != kVideoMJPEG) { cap.maxFPS = 15; } else { cap.maxFPS = 30; } _captureCapabilities.push_back(cap); index++; WEBRTC_TRACE(webrtc::kTraceInfo, webrtc::kTraceVideoCapture, _id, "Camera capability, width:%d height:%d type:%d fps:%d", cap.width, cap.height, cap.rawType, cap.maxFPS); } } } } WEBRTC_TRACE(webrtc::kTraceInfo, webrtc::kTraceVideoCapture, _id, "CreateCapabilityMap %u", static_cast(_captureCapabilities.size())); return _captureCapabilities.size(); } } // namespace videocapturemodule } // namespace webrtc