/* * Copyright (C) Texas Instruments - http://www.ti.com/ * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /** * @file V4LCameraAdapter.cpp * * This file maps the Camera Hardware Interface to V4L2. * */ #include "V4LCameraAdapter.h" #include "CameraHal.h" #include "TICameraParameters.h" #include <signal.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <unistd.h> #include <errno.h> #include <sys/ioctl.h> #include <sys/mman.h> #include <sys/select.h> #include <linux/videodev.h> #include <cutils/properties.h> #define UNLIKELY( exp ) (__builtin_expect( (exp) != 0, false )) static int mDebugFps = 0; #define Q16_OFFSET 16 #define HERE(Msg) {CAMHAL_LOGEB("--===line %d, %s===--\n", __LINE__, Msg);} namespace android { #undef LOG_TAG ///Maintain a separate tag for V4LCameraAdapter logs to isolate issues OMX specific #define LOG_TAG "CameraHAL" //frames skipped before recalculating the framerate #define FPS_PERIOD 30 Mutex gAdapterLock; const char *device = DEVICE; /*--------------------Camera Adapter Class STARTS here-----------------------------*/ status_t V4LCameraAdapter::initialize(CameraProperties::Properties* caps) { LOG_FUNCTION_NAME; char value[PROPERTY_VALUE_MAX]; property_get("debug.camera.showfps", value, "0"); mDebugFps = atoi(value); int ret = NO_ERROR; // Allocate memory for video info structure mVideoInfo = (struct VideoInfo *) calloc (1, sizeof (struct VideoInfo)); if(!mVideoInfo) { return NO_MEMORY; } if ((mCameraHandle = open(device, O_RDWR)) == -1) { CAMHAL_LOGEB("Error while opening handle to V4L2 Camera: %s", strerror(errno)); return -EINVAL; } ret = ioctl (mCameraHandle, VIDIOC_QUERYCAP, &mVideoInfo->cap); if (ret < 0) { CAMHAL_LOGEA("Error when querying the capabilities of the V4L Camera"); return -EINVAL; } if ((mVideoInfo->cap.capabilities & V4L2_CAP_VIDEO_CAPTURE) == 0) { CAMHAL_LOGEA("Error while adapter initialization: video capture not supported."); return -EINVAL; } if (!(mVideoInfo->cap.capabilities & V4L2_CAP_STREAMING)) { CAMHAL_LOGEA("Error while adapter initialization: Capture device does not support streaming i/o"); return -EINVAL; } // Initialize flags mPreviewing = false; mVideoInfo->isStreaming = false; mRecording = false; LOG_FUNCTION_NAME_EXIT; return ret; } status_t V4LCameraAdapter::fillThisBuffer(void* frameBuf, CameraFrame::FrameType frameType) { status_t ret = NO_ERROR; if ( !mVideoInfo->isStreaming ) { return NO_ERROR; } int i = mPreviewBufs.valueFor(( unsigned int )frameBuf); if(i<0) { return BAD_VALUE; } mVideoInfo->buf.index = i; mVideoInfo->buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->buf.memory = V4L2_MEMORY_MMAP; ret = ioctl(mCameraHandle, VIDIOC_QBUF, &mVideoInfo->buf); if (ret < 0) { CAMHAL_LOGEA("Init: VIDIOC_QBUF Failed"); return -1; } nQueued++; return ret; } status_t V4LCameraAdapter::setParameters(const CameraParameters ¶ms) { LOG_FUNCTION_NAME; status_t ret = NO_ERROR; int width, height; params.getPreviewSize(&width, &height); CAMHAL_LOGDB("Width * Height %d x %d format 0x%x", width, height, DEFAULT_PIXEL_FORMAT); mVideoInfo->width = width; mVideoInfo->height = height; mVideoInfo->framesizeIn = (width * height << 1); mVideoInfo->formatIn = DEFAULT_PIXEL_FORMAT; mVideoInfo->format.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->format.fmt.pix.width = width; mVideoInfo->format.fmt.pix.height = height; mVideoInfo->format.fmt.pix.pixelformat = DEFAULT_PIXEL_FORMAT; ret = ioctl(mCameraHandle, VIDIOC_S_FMT, &mVideoInfo->format); if (ret < 0) { CAMHAL_LOGEB("Open: VIDIOC_S_FMT Failed: %s", strerror(errno)); return ret; } // Udpate the current parameter set mParams = params; LOG_FUNCTION_NAME_EXIT; return ret; } void V4LCameraAdapter::getParameters(CameraParameters& params) { LOG_FUNCTION_NAME; // Return the current parameter set params = mParams; LOG_FUNCTION_NAME_EXIT; } ///API to give the buffers to Adapter status_t V4LCameraAdapter::useBuffers(CameraMode mode, void* bufArr, int num, size_t length, unsigned int queueable) { status_t ret = NO_ERROR; LOG_FUNCTION_NAME; Mutex::Autolock lock(mLock); switch(mode) { case CAMERA_PREVIEW: ret = UseBuffersPreview(bufArr, num); break; //@todo Insert Image capture case here case CAMERA_VIDEO: //@warn Video capture is not fully supported yet ret = UseBuffersPreview(bufArr, num); break; } LOG_FUNCTION_NAME_EXIT; return ret; } status_t V4LCameraAdapter::UseBuffersPreview(void* bufArr, int num) { int ret = NO_ERROR; if(NULL == bufArr) { return BAD_VALUE; } //First allocate adapter internal buffers at V4L level for USB Cam //These are the buffers from which we will copy the data into overlay buffers /* Check if camera can handle NB_BUFFER buffers */ mVideoInfo->rb.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->rb.memory = V4L2_MEMORY_MMAP; mVideoInfo->rb.count = num; ret = ioctl(mCameraHandle, VIDIOC_REQBUFS, &mVideoInfo->rb); if (ret < 0) { CAMHAL_LOGEB("VIDIOC_REQBUFS failed: %s", strerror(errno)); return ret; } for (int i = 0; i < num; i++) { memset (&mVideoInfo->buf, 0, sizeof (struct v4l2_buffer)); mVideoInfo->buf.index = i; mVideoInfo->buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->buf.memory = V4L2_MEMORY_MMAP; ret = ioctl (mCameraHandle, VIDIOC_QUERYBUF, &mVideoInfo->buf); if (ret < 0) { CAMHAL_LOGEB("Unable to query buffer (%s)", strerror(errno)); return ret; } mVideoInfo->mem[i] = mmap (0, mVideoInfo->buf.length, PROT_READ | PROT_WRITE, MAP_SHARED, mCameraHandle, mVideoInfo->buf.m.offset); if (mVideoInfo->mem[i] == MAP_FAILED) { CAMHAL_LOGEB("Unable to map buffer (%s)", strerror(errno)); return -1; } uint32_t *ptr = (uint32_t*) bufArr; //Associate each Camera internal buffer with the one from Overlay mPreviewBufs.add((int)ptr[i], i); } // Update the preview buffer count mPreviewBufferCount = num; return ret; } status_t V4LCameraAdapter::startPreview() { status_t ret = NO_ERROR; Mutex::Autolock lock(mPreviewBufsLock); if(mPreviewing) { return BAD_VALUE; } for (int i = 0; i < mPreviewBufferCount; i++) { mVideoInfo->buf.index = i; mVideoInfo->buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->buf.memory = V4L2_MEMORY_MMAP; ret = ioctl(mCameraHandle, VIDIOC_QBUF, &mVideoInfo->buf); if (ret < 0) { CAMHAL_LOGEA("VIDIOC_QBUF Failed"); return -EINVAL; } nQueued++; } enum v4l2_buf_type bufType; if (!mVideoInfo->isStreaming) { bufType = V4L2_BUF_TYPE_VIDEO_CAPTURE; ret = ioctl (mCameraHandle, VIDIOC_STREAMON, &bufType); if (ret < 0) { CAMHAL_LOGEB("StartStreaming: Unable to start capture: %s", strerror(errno)); return ret; } mVideoInfo->isStreaming = true; } // Create and start preview thread for receiving buffers from V4L Camera mPreviewThread = new PreviewThread(this); CAMHAL_LOGDA("Created preview thread"); //Update the flag to indicate we are previewing mPreviewing = true; return ret; } status_t V4LCameraAdapter::stopPreview() { enum v4l2_buf_type bufType; int ret = NO_ERROR; Mutex::Autolock lock(mPreviewBufsLock); if(!mPreviewing) { return NO_INIT; } if (mVideoInfo->isStreaming) { bufType = V4L2_BUF_TYPE_VIDEO_CAPTURE; ret = ioctl (mCameraHandle, VIDIOC_STREAMOFF, &bufType); if (ret < 0) { CAMHAL_LOGEB("StopStreaming: Unable to stop capture: %s", strerror(errno)); return ret; } mVideoInfo->isStreaming = false; } mVideoInfo->buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->buf.memory = V4L2_MEMORY_MMAP; nQueued = 0; nDequeued = 0; /* Unmap buffers */ for (int i = 0; i < mPreviewBufferCount; i++) if (munmap(mVideoInfo->mem[i], mVideoInfo->buf.length) < 0) CAMHAL_LOGEA("Unmap failed"); mPreviewBufs.clear(); mPreviewThread->requestExitAndWait(); mPreviewThread.clear(); return ret; } char * V4LCameraAdapter::GetFrame(int &index) { int ret; mVideoInfo->buf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; mVideoInfo->buf.memory = V4L2_MEMORY_MMAP; /* DQ */ ret = ioctl(mCameraHandle, VIDIOC_DQBUF, &mVideoInfo->buf); if (ret < 0) { CAMHAL_LOGEA("GetFrame: VIDIOC_DQBUF Failed"); return NULL; } nDequeued++; index = mVideoInfo->buf.index; return (char *)mVideoInfo->mem[mVideoInfo->buf.index]; } //API to get the frame size required to be allocated. This size is used to override the size passed //by camera service when VSTAB/VNF is turned ON for example status_t V4LCameraAdapter::getFrameSize(size_t &width, size_t &height) { status_t ret = NO_ERROR; // Just return the current preview size, nothing more to do here. mParams.getPreviewSize(( int * ) &width, ( int * ) &height); LOG_FUNCTION_NAME_EXIT; return ret; } status_t V4LCameraAdapter::getFrameDataSize(size_t &dataFrameSize, size_t bufferCount) { // We don't support meta data, so simply return return NO_ERROR; } status_t V4LCameraAdapter::getPictureBufferSize(size_t &length, size_t bufferCount) { // We don't support image capture yet, safely return from here without messing up return NO_ERROR; } static void debugShowFPS() { static int mFrameCount = 0; static int mLastFrameCount = 0; static nsecs_t mLastFpsTime = 0; static float mFps = 0; mFrameCount++; if (!(mFrameCount & 0x1F)) { nsecs_t now = systemTime(); nsecs_t diff = now - mLastFpsTime; mFps = ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff; mLastFpsTime = now; mLastFrameCount = mFrameCount; ALOGD("Camera %d Frames, %f FPS", mFrameCount, mFps); } // XXX: mFPS has the value we want } status_t V4LCameraAdapter::recalculateFPS() { float currentFPS; mFrameCount++; if ( ( mFrameCount % FPS_PERIOD ) == 0 ) { nsecs_t now = systemTime(); nsecs_t diff = now - mLastFPSTime; currentFPS = ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff; mLastFPSTime = now; mLastFrameCount = mFrameCount; if ( 1 == mIter ) { mFPS = currentFPS; } else { //cumulative moving average mFPS = mLastFPS + (currentFPS - mLastFPS)/mIter; } mLastFPS = mFPS; mIter++; } return NO_ERROR; } void V4LCameraAdapter::onOrientationEvent(uint32_t orientation, uint32_t tilt) { LOG_FUNCTION_NAME; LOG_FUNCTION_NAME_EXIT; } V4LCameraAdapter::V4LCameraAdapter(size_t sensor_index) { LOG_FUNCTION_NAME; // Nothing useful to do in the constructor LOG_FUNCTION_NAME_EXIT; } V4LCameraAdapter::~V4LCameraAdapter() { LOG_FUNCTION_NAME; // Close the camera handle and free the video info structure close(mCameraHandle); if (mVideoInfo) { free(mVideoInfo); mVideoInfo = NULL; } LOG_FUNCTION_NAME_EXIT; } /* Preview Thread */ // --------------------------------------------------------------------------- int V4LCameraAdapter::previewThread() { status_t ret = NO_ERROR; int width, height; CameraFrame frame; if (mPreviewing) { int index = 0; char *fp = this->GetFrame(index); if(!fp) { return BAD_VALUE; } uint8_t* ptr = (uint8_t*) mPreviewBufs.keyAt(index); int width, height; uint16_t* dest = (uint16_t*)ptr; uint16_t* src = (uint16_t*) fp; mParams.getPreviewSize(&width, &height); for(int i=0;i<height;i++) { for(int j=0;j<width;j++) { //*dest = *src; //convert from YUYV to UYVY supported in Camera service *dest = (((*src & 0xFF000000)>>24)<<16)|(((*src & 0x00FF0000)>>16)<<24) | (((*src & 0xFF00)>>8)<<0)|(((*src & 0x00FF)>>0)<<8); src++; dest++; } dest += 4096/2-width; } mParams.getPreviewSize(&width, &height); frame.mFrameType = CameraFrame::PREVIEW_FRAME_SYNC; frame.mBuffer = ptr; frame.mLength = width*height*2; frame.mAlignment = width*2; frame.mOffset = 0; frame.mTimestamp = systemTime(SYSTEM_TIME_MONOTONIC);; ret = sendFrameToSubscribers(&frame); } return ret; } extern "C" CameraAdapter* CameraAdapter_Factory() { CameraAdapter *adapter = NULL; Mutex::Autolock lock(gAdapterLock); LOG_FUNCTION_NAME; adapter = new V4LCameraAdapter(sensor_index); if ( adapter ) { CAMHAL_LOGDB("New OMX Camera adapter instance created for sensor %d",sensor_index); } else { CAMHAL_LOGEA("Camera adapter create failed!"); } LOG_FUNCTION_NAME_EXIT; return adapter; } extern "C" int CameraAdapter_Capabilities(CameraProperties::Properties* properties_array, const unsigned int starting_camera, const unsigned int max_camera) { int num_cameras_supported = 0; CameraProperties::Properties* properties = NULL; LOG_FUNCTION_NAME; if(!properties_array) { return -EINVAL; } // TODO: Need to tell camera properties what other cameras we can support if (starting_camera + num_cameras_supported < max_camera) { num_cameras_supported++; properties = properties_array + starting_camera; properties->set(CameraProperties::CAMERA_NAME, "USBCamera"); } LOG_FUNCTION_NAME_EXIT; return num_cameras_supported; } }; /*--------------------Camera Adapter Class ENDS here-----------------------------*/