/* ** Copyright (c) 2011,2015 The Linux Foundation. All rights reserved. ** ** 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. */ /*#error uncomment this for compiler test!*/ //#define ALOG_NDEBUG 0 #define ALOG_NIDEBUG 0 #define LOG_TAG "QCameraHWI_Record" #include <utils/Log.h> #include <utils/threads.h> #include <cutils/properties.h> #include <fcntl.h> #include <sys/mman.h> #include "QCameraStream.h" #define LIKELY(exp) __builtin_expect(!!(exp), 1) #define UNLIKELY(exp) __builtin_expect(!!(exp), 0) /* QCameraStream_record class implementation goes here*/ /* following code implement the video streaming capture & encoding logic of this class*/ // --------------------------------------------------------------------------- // QCameraStream_record createInstance() // --------------------------------------------------------------------------- namespace android { QCameraStream* QCameraStream_record::createInstance(int cameraId, camera_mode_t mode) { ALOGV("%s: BEGIN", __func__); QCameraStream* pme = new QCameraStream_record(cameraId, mode); ALOGV("%s: END", __func__); return pme; } // --------------------------------------------------------------------------- // QCameraStream_record deleteInstance() // --------------------------------------------------------------------------- void QCameraStream_record::deleteInstance(QCameraStream *ptr) { ALOGV("%s: BEGIN", __func__); if (ptr){ ptr->release(); delete ptr; ptr = NULL; } ALOGV("%s: END", __func__); } // --------------------------------------------------------------------------- // QCameraStream_record Constructor // --------------------------------------------------------------------------- QCameraStream_record::QCameraStream_record(int cameraId, camera_mode_t mode) :QCameraStream(cameraId,mode), mDebugFps(false) { mHalCamCtrl = NULL; char value[PROPERTY_VALUE_MAX]; ALOGV("%s: BEGIN", __func__); property_get("persist.debug.sf.showfps", value, "0"); mDebugFps = atoi(value); ALOGV("%s: END", __func__); } // --------------------------------------------------------------------------- // QCameraStream_record Destructor // --------------------------------------------------------------------------- QCameraStream_record::~QCameraStream_record() { ALOGV("%s: BEGIN", __func__); if(mActive) { stop(); } if(mInit) { release(); } mInit = false; mActive = false; ALOGV("%s: END", __func__); } // --------------------------------------------------------------------------- // QCameraStream_record Callback from mm_camera // --------------------------------------------------------------------------- static void record_notify_cb(mm_camera_ch_data_buf_t *bufs_new, void *user_data) { QCameraStream_record *pme = (QCameraStream_record *)user_data; mm_camera_ch_data_buf_t *bufs_used = 0; ALOGV("%s: BEGIN", __func__); /* * Call Function Process Video Data */ pme->processRecordFrame(bufs_new); ALOGV("%s: END", __func__); } // --------------------------------------------------------------------------- // QCameraStream_record // --------------------------------------------------------------------------- status_t QCameraStream_record::init() { status_t ret = NO_ERROR; ALOGV("%s: BEGIN", __func__); /* * Acquiring Video Channel */ ret = QCameraStream::initChannel (mCameraId, MM_CAMERA_CH_VIDEO_MASK); if (NO_ERROR!=ret) { ALOGE("%s ERROR: Can't init native cammera preview ch\n",__func__); return ret; } /* * Register the Callback with camera */ (void) cam_evt_register_buf_notify(mCameraId, MM_CAMERA_CH_VIDEO, record_notify_cb, MM_CAMERA_REG_BUF_CB_INFINITE, 0, this); mInit = true; ALOGV("%s: END", __func__); return ret; } // --------------------------------------------------------------------------- // QCameraStream_record // --------------------------------------------------------------------------- status_t QCameraStream_record::start() { status_t ret = NO_ERROR; ALOGV("%s: BEGIN", __func__); Mutex::Autolock lock(mStopCallbackLock); if(!mInit) { ALOGE("%s ERROR: Record buffer not registered",__func__); return BAD_VALUE; } setFormat(MM_CAMERA_CH_VIDEO_MASK , (cam_format_t)0); //mRecordFreeQueueLock.lock(); //mRecordFreeQueue.clear(); //mRecordFreeQueueLock.unlock(); /* * Allocating Encoder Frame Buffers */ ret = initEncodeBuffers(); if (NO_ERROR!=ret) { ALOGE("%s ERROR: Buffer Allocation Failed\n",__func__); goto error; } ret = cam_config_prepare_buf(mCameraId, &mRecordBuf); if(ret != MM_CAMERA_OK) { ALOGE("%s ERROR: Reg Record buf err=%d\n", __func__, ret); ret = BAD_VALUE; goto error; }else{ ret = NO_ERROR; } /* * Start Video Streaming */ ret = cam_ops_action(mCameraId, true, MM_CAMERA_OPS_VIDEO, 0); if (MM_CAMERA_OK != ret) { ALOGE ("%s ERROR: Video streaming start err=%d\n", __func__, ret); ret = BAD_VALUE; goto error; }else{ ALOGV("%s : Video streaming Started",__func__); ret = NO_ERROR; } mActive = true; ALOGV("%s: END", __func__); return ret; error: releaseEncodeBuffer(); ALOGV("%s: END", __func__); return ret; } void QCameraStream_record::releaseEncodeBuffer() { for(int cnt = 0; cnt < mHalCamCtrl->mRecordingMemory.buffer_count; cnt++) { if (NO_ERROR != mHalCamCtrl->sendUnMappingBuf(MSM_V4L2_EXT_CAPTURE_MODE_VIDEO, cnt, mCameraId, CAM_SOCK_MSG_TYPE_FD_UNMAPPING)) ALOGE("%s: Unmapping Video Data Failed", __func__); if (mHalCamCtrl->mStoreMetaDataInFrame) { struct encoder_media_buffer_type * packet = (struct encoder_media_buffer_type *) mHalCamCtrl->mRecordingMemory.metadata_memory[cnt]->data; native_handle_delete(const_cast<native_handle_t *>(packet->meta_handle)); mHalCamCtrl->mRecordingMemory.metadata_memory[cnt]->release( mHalCamCtrl->mRecordingMemory.metadata_memory[cnt]); } mHalCamCtrl->mRecordingMemory.camera_memory[cnt]->release( mHalCamCtrl->mRecordingMemory.camera_memory[cnt]); close(mHalCamCtrl->mRecordingMemory.fd[cnt]); mHalCamCtrl->mRecordingMemory.fd[cnt] = -1; #ifdef USE_ION mHalCamCtrl->deallocate_ion_memory(&mHalCamCtrl->mRecordingMemory, cnt); #endif } memset(&mHalCamCtrl->mRecordingMemory, 0, sizeof(mHalCamCtrl->mRecordingMemory)); //mNumRecordFrames = 0; delete[] recordframes; if (mRecordBuf.video.video.buf.mp) delete[] mRecordBuf.video.video.buf.mp; } // --------------------------------------------------------------------------- // QCameraStream_record // --------------------------------------------------------------------------- void QCameraStream_record::stop() { status_t ret = NO_ERROR; ALOGV("%s: BEGIN", __func__); if(!mActive) { ALOGE("%s : Record stream not started",__func__); return; } mActive = false; Mutex::Autolock lock(mStopCallbackLock); #if 0 //mzhu, when stop recording, all frame will be dirty. no need to queue frame back to kernel any more mRecordFreeQueueLock.lock(); while(!mRecordFreeQueue.isEmpty()) { ALOGV("%s : Pre-releasing of Encoder buffers!\n", __FUNCTION__); mm_camera_ch_data_buf_t releasedBuf = mRecordFreeQueue.itemAt(0); mRecordFreeQueue.removeAt(0); mRecordFreeQueueLock.unlock(); ALOGV("%s (%d): releasedBuf.idx = %d\n", __FUNCTION__, __LINE__, releasedBuf.video.video.idx); if(MM_CAMERA_OK != cam_evt_buf_done(mCameraId,&releasedBuf)) ALOGE("%s : Buf Done Failed",__func__); } mRecordFreeQueueLock.unlock(); #if 0 while (!mRecordFreeQueue.isEmpty()) { ALOGE("%s : Waiting for Encoder to release all buffer!\n", __FUNCTION__); } #endif #endif // mzhu /* unregister the notify fn from the mmmm_camera_t object * call stop() in parent class to stop the monitor thread */ ret = cam_ops_action(mCameraId, false, MM_CAMERA_OPS_VIDEO, 0); if (MM_CAMERA_OK != ret) { ALOGE ("%s ERROR: Video streaming Stop err=%d\n", __func__, ret); } ret = cam_config_unprepare_buf(mCameraId, MM_CAMERA_CH_VIDEO); if(ret != MM_CAMERA_OK){ ALOGE("%s ERROR: Ureg video buf \n", __func__); } releaseEncodeBuffer(); mActive = false; ALOGV("%s: END", __func__); } // --------------------------------------------------------------------------- // QCameraStream_record // --------------------------------------------------------------------------- void QCameraStream_record::release() { status_t ret = NO_ERROR; ALOGV("%s: BEGIN", __func__); if(mActive) { stop(); } if(!mInit) { ALOGE("%s : Record stream not initialized",__func__); return; } ret= QCameraStream::deinitChannel(mCameraId, MM_CAMERA_CH_VIDEO); if(ret != MM_CAMERA_OK) { ALOGE("%s:Deinit Video channel failed=%d\n", __func__, ret); } (void)cam_evt_register_buf_notify(mCameraId, MM_CAMERA_CH_VIDEO, NULL, (mm_camera_register_buf_cb_type_t)NULL, NULL, NULL); mInit = false; ALOGV("%s: END", __func__); } status_t QCameraStream_record::processRecordFrame(void *data) { ALOGV("%s : BEGIN",__func__); mm_camera_ch_data_buf_t* frame = (mm_camera_ch_data_buf_t*) data; Mutex::Autolock lock(mStopCallbackLock); if(!mActive) { ALOGE("Recording Stopped. Returning callback"); return NO_ERROR; } if (UNLIKELY(mDebugFps)) { debugShowVideoFPS(); } mHalCamCtrl->dumpFrameToFile(frame->video.video.frame, HAL_DUMP_FRM_VIDEO); mHalCamCtrl->mCallbackLock.lock(); camera_data_timestamp_callback rcb = mHalCamCtrl->mDataCbTimestamp; void *rdata = mHalCamCtrl->mCallbackCookie; mHalCamCtrl->mCallbackLock.unlock(); nsecs_t timeStamp = nsecs_t(frame->video.video.frame->ts.tv_sec)*1000000000LL + \ frame->video.video.frame->ts.tv_nsec; ALOGV("Send Video frame to services/encoder TimeStamp : %lld",timeStamp); mRecordedFrames[frame->video.video.idx] = *frame; #ifdef USE_ION struct ion_flush_data cache_inv_data; int ion_fd; ion_fd = frame->video.video.frame->ion_dev_fd; cache_inv_data.vaddr = (void *)frame->video.video.frame->buffer; cache_inv_data.fd = frame->video.video.frame->fd; cache_inv_data.handle = frame->video.video.frame->fd_data.handle; cache_inv_data.length = frame->video.video.frame->ion_alloc.len; if (mHalCamCtrl->cache_ops(ion_fd, &cache_inv_data, ION_IOC_CLEAN_CACHES) < 0) ALOGE("%s: Cache clean for Video buffer %p fd = %d failed", __func__, cache_inv_data.vaddr, cache_inv_data.fd); #endif if (mHalCamCtrl->mStoreMetaDataInFrame) { mStopCallbackLock.unlock(); if(mActive && (rcb != NULL) && (mHalCamCtrl->mMsgEnabled & CAMERA_MSG_VIDEO_FRAME)) { rcb(timeStamp, CAMERA_MSG_VIDEO_FRAME, mHalCamCtrl->mRecordingMemory.metadata_memory[frame->video.video.idx], 0, mHalCamCtrl->mCallbackCookie); } } else { mStopCallbackLock.unlock(); if(mActive && (rcb != NULL) && (mHalCamCtrl->mMsgEnabled & CAMERA_MSG_VIDEO_FRAME)) { rcb(timeStamp, CAMERA_MSG_VIDEO_FRAME, mHalCamCtrl->mRecordingMemory.camera_memory[frame->video.video.idx], 0, mHalCamCtrl->mCallbackCookie); } } ALOGV("%s : END",__func__); return NO_ERROR; } //Record Related Functions status_t QCameraStream_record::initEncodeBuffers() { ALOGV("%s : BEGIN",__func__); status_t ret = NO_ERROR; const char *pmem_region; uint32_t frame_len; uint8_t num_planes; uint32_t planes[VIDEO_MAX_PLANES]; //cam_ctrl_dimension_t dim; int width = 0; /* width of channel */ int height = 0; /* height of channel */ int buf_cnt; pmem_region = "/dev/pmem_adsp"; memset(&mHalCamCtrl->mRecordingMemory, 0, sizeof(mHalCamCtrl->mRecordingMemory)); memset(&dim, 0, sizeof(cam_ctrl_dimension_t)); ret = cam_config_get_parm(mCameraId, MM_CAMERA_PARM_DIMENSION, &dim); if (MM_CAMERA_OK != ret) { ALOGE("%s: ERROR - can't get camera dimension!", __func__); return BAD_VALUE; } else { width = dim.video_width; height = dim.video_height; } num_planes = 2; planes[0] = dim.video_frame_offset.mp[0].len; planes[1] = dim.video_frame_offset.mp[1].len; frame_len = dim.video_frame_offset.frame_len; buf_cnt = VIDEO_BUFFER_COUNT; if(mHalCamCtrl->isLowPowerCamcorder()) { ALOGV("%s: lower power camcorder selected", __func__); buf_cnt = VIDEO_BUFFER_COUNT_LOW_POWER_CAMCORDER; } recordframes = new msm_frame[buf_cnt]; memset(recordframes,0,sizeof(struct msm_frame) * buf_cnt); mRecordBuf.video.video.buf.mp = new mm_camera_mp_buf_t[buf_cnt * sizeof(mm_camera_mp_buf_t)]; if (!mRecordBuf.video.video.buf.mp) { ALOGE("%s Error allocating memory for mplanar struct ", __func__); return BAD_VALUE; } memset(mRecordBuf.video.video.buf.mp, 0, buf_cnt * sizeof(mm_camera_mp_buf_t)); memset(&mHalCamCtrl->mRecordingMemory, 0, sizeof(mHalCamCtrl->mRecordingMemory)); for (int i=0; i<MM_CAMERA_MAX_NUM_FRAMES;i++) { mHalCamCtrl->mRecordingMemory.main_ion_fd[i] = -1; mHalCamCtrl->mRecordingMemory.fd[i] = -1; } mHalCamCtrl->mRecordingMemory.buffer_count = buf_cnt; mHalCamCtrl->mRecordingMemory.size = frame_len; mHalCamCtrl->mRecordingMemory.cbcr_offset = planes[0]; for (int cnt = 0; cnt < mHalCamCtrl->mRecordingMemory.buffer_count; cnt++) { #ifdef USE_ION if(mHalCamCtrl->allocate_ion_memory(&mHalCamCtrl->mRecordingMemory, cnt, ((0x1 << CAMERA_ION_HEAP_ID) | (0x1 << CAMERA_ION_FALLBACK_HEAP_ID))) < 0) { ALOGE("%s ION alloc failed\n", __func__); return UNKNOWN_ERROR; } #else mHalCamCtrl->mRecordingMemory.fd[cnt] = open("/dev/pmem_adsp", O_RDWR|O_SYNC); if(mHalCamCtrl->mRecordingMemory.fd[cnt] <= 0) { ALOGE("%s: no pmem for frame %d", __func__, cnt); return UNKNOWN_ERROR; } #endif mHalCamCtrl->mRecordingMemory.camera_memory[cnt] = mHalCamCtrl->mGetMemory(mHalCamCtrl->mRecordingMemory.fd[cnt], mHalCamCtrl->mRecordingMemory.size, 1, (void *)this); if (mHalCamCtrl->mStoreMetaDataInFrame) { mHalCamCtrl->mRecordingMemory.metadata_memory[cnt] = mHalCamCtrl->mGetMemory(-1, sizeof(struct encoder_media_buffer_type), 1, (void *)this); struct encoder_media_buffer_type * packet = (struct encoder_media_buffer_type *) mHalCamCtrl->mRecordingMemory.metadata_memory[cnt]->data; packet->meta_handle = native_handle_create(1, 2); //1 fd, 1 offset and 1 size packet->buffer_type = kMetadataBufferTypeCameraSource; native_handle_t * nh = const_cast<native_handle_t *>(packet->meta_handle); nh->data[0] = mHalCamCtrl->mRecordingMemory.fd[cnt]; nh->data[1] = 0; nh->data[2] = mHalCamCtrl->mRecordingMemory.size; } recordframes[cnt].fd = mHalCamCtrl->mRecordingMemory.fd[cnt]; recordframes[cnt].buffer = (uint32_t)mHalCamCtrl->mRecordingMemory.camera_memory[cnt]->data; recordframes[cnt].y_off = 0; recordframes[cnt].cbcr_off = mHalCamCtrl->mRecordingMemory.cbcr_offset; recordframes[cnt].path = OUTPUT_TYPE_V; recordframes[cnt].fd_data = mHalCamCtrl->mRecordingMemory.ion_info_fd[cnt]; recordframes[cnt].ion_alloc = mHalCamCtrl->mRecordingMemory.alloc[cnt]; recordframes[cnt].ion_dev_fd = mHalCamCtrl->mRecordingMemory.main_ion_fd[cnt]; if (NO_ERROR != mHalCamCtrl->sendMappingBuf(MSM_V4L2_EXT_CAPTURE_MODE_VIDEO, cnt, recordframes[cnt].fd, mHalCamCtrl->mRecordingMemory.size, mCameraId, CAM_SOCK_MSG_TYPE_FD_MAPPING)) ALOGE("%s: sending mapping data Msg Failed", __func__); ALOGV("initRecord : record heap , video buffers buffer=%lu fd=%d y_off=%d cbcr_off=%d\n", (unsigned long)recordframes[cnt].buffer, recordframes[cnt].fd, recordframes[cnt].y_off, recordframes[cnt].cbcr_off); //mNumRecordFrames++; mRecordBuf.video.video.buf.mp[cnt].frame = recordframes[cnt]; mRecordBuf.video.video.buf.mp[cnt].frame_offset = 0; mRecordBuf.video.video.buf.mp[cnt].num_planes = num_planes; /* Plane 0 needs to be set seperately. Set other planes * in a loop. */ mRecordBuf.video.video.buf.mp[cnt].planes[0].reserved[0] = mRecordBuf.video.video.buf.mp[cnt].frame_offset; mRecordBuf.video.video.buf.mp[cnt].planes[0].length = planes[0]; mRecordBuf.video.video.buf.mp[cnt].planes[0].m.userptr = recordframes[cnt].fd; for (int j = 1; j < num_planes; j++) { mRecordBuf.video.video.buf.mp[cnt].planes[j].length = planes[j]; mRecordBuf.video.video.buf.mp[cnt].planes[j].m.userptr = recordframes[cnt].fd; mRecordBuf.video.video.buf.mp[cnt].planes[j].reserved[0] = mRecordBuf.video.video.buf.mp[cnt].planes[j-1].reserved[0] + mRecordBuf.video.video.buf.mp[cnt].planes[j-1].length; } mRecordBuf.video.video.no_enqueue_flag[cnt] = 0; } //memset(&mRecordBuf, 0, sizeof(mRecordBuf)); mRecordBuf.ch_type = MM_CAMERA_CH_VIDEO; mRecordBuf.video.video.num = mHalCamCtrl->mRecordingMemory.buffer_count;//kRecordBufferCount; //mRecordBuf.video.video.frame_offset = &record_offset[0]; //mRecordBuf.video.video.frame = &recordframes[0]; ALOGV("%s : END",__func__); return NO_ERROR; } void QCameraStream_record::releaseRecordingFrame(const void *opaque) { ALOGV("%s : BEGIN, opaque = 0x%p",__func__, opaque); if(!mActive) { ALOGE("%s : Recording already stopped!!! Leak???",__func__); return; } for(int cnt = 0; cnt < mHalCamCtrl->mRecordingMemory.buffer_count; cnt++) { if (mHalCamCtrl->mStoreMetaDataInFrame) { if(mHalCamCtrl->mRecordingMemory.metadata_memory[cnt] && mHalCamCtrl->mRecordingMemory.metadata_memory[cnt]->data == opaque) { /* found the match */ if(MM_CAMERA_OK != cam_evt_buf_done(mCameraId, &mRecordedFrames[cnt])) ALOGE("%s : Buf Done Failed",__func__); ALOGV("%s : END",__func__); return; } } else { if(mHalCamCtrl->mRecordingMemory.camera_memory[cnt] && mHalCamCtrl->mRecordingMemory.camera_memory[cnt]->data == opaque) { /* found the match */ if(MM_CAMERA_OK != cam_evt_buf_done(mCameraId, &mRecordedFrames[cnt])) ALOGE("%s : Buf Done Failed",__func__); ALOGV("%s : END",__func__); return; } } } ALOGE("%s: cannot find the matched frame with opaue = 0x%p", __func__, opaque); } void QCameraStream_record::debugShowVideoFPS() const { static int mFrameCount; static int mLastFrameCount = 0; static nsecs_t mLastFpsTime = 0; static float mFps = 0; mFrameCount++; nsecs_t now = systemTime(); nsecs_t diff = now - mLastFpsTime; if (diff > ms2ns(250)) { mFps = ((mFrameCount - mLastFrameCount) * float(s2ns(1))) / diff; ALOGV("Video Frames Per Second: %.4f", mFps); mLastFpsTime = now; mLastFrameCount = mFrameCount; } } #if 0 sp<IMemoryHeap> QCameraStream_record::getHeap() const { return mRecordHeap != NULL ? mRecordHeap->mHeap : NULL; } #endif status_t QCameraStream_record::takeLiveSnapshot(){ return true; } }//namespace android