/* ** Copyright (c) 2011-2012 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_Mem" #include <utils/Log.h> #include <utils/Errors.h> #include <utils/threads.h> //#include <binder/MemoryHeapPmem.h> #include <utils/String16.h> #include <sys/types.h> #include <sys/stat.h> #include <unistd.h> #include <fcntl.h> #include <cutils/properties.h> #include <math.h> #if HAVE_ANDROID_OS #include <linux/android_pmem.h> #endif #include <linux/ioctl.h> #include "QCameraParameters.h" #include <media/mediarecorder.h> #include <gralloc_priv.h> #include "QCameraHWI_Mem.h" #define CAMERA_HAL_UNUSED(expr) do { (void)(expr); } while (0) /* QCameraHardwareInterface class implementation goes here*/ /* following code implement the contol logic of this class*/ namespace android { static bool register_buf(int size, int frame_size, int cbcr_offset, int yoffset, int pmempreviewfd, uint32_t offset, uint8_t *buf, int pmem_type, bool vfe_can_write, bool register_buffer = true); #if 0 MMCameraDL::MMCameraDL(){ ALOGV("MMCameraDL: E"); libmmcamera = NULL; #if DLOPEN_LIBMMCAMERA libmmcamera = ::dlopen("liboemcamera.so", RTLD_NOW); #endif ALOGV("Open MM camera DL libeomcamera loaded at %p ", libmmcamera); ALOGV("MMCameraDL: X"); } void * MMCameraDL::pointer(){ return libmmcamera; } MMCameraDL::~MMCameraDL(){ ALOGV("~MMCameraDL: E"); LINK_mm_camera_destroy(); if (libmmcamera != NULL) { ::dlclose(libmmcamera); ALOGV("closed MM Camera DL "); } libmmcamera = NULL; ALOGV("~MMCameraDL: X"); } wp<MMCameraDL> MMCameraDL::instance; Mutex MMCameraDL::singletonLock; sp<MMCameraDL> MMCameraDL::getInstance(){ Mutex::Autolock instanceLock(singletonLock); sp<MMCameraDL> mmCamera = instance.promote(); if(mmCamera == NULL){ mmCamera = new MMCameraDL(); instance = mmCamera; } return mmCamera; } #endif MemPool::MemPool(int buffer_size, int num_buffers, int frame_size, const char *name) : mBufferSize(buffer_size), mNumBuffers(num_buffers), mFrameSize(frame_size), mBuffers(NULL), mName(name) { int page_size_minus_1 = getpagesize() - 1; mAlignedBufferSize = (buffer_size + page_size_minus_1) & (~page_size_minus_1); } void MemPool::completeInitialization() { // If we do not know how big the frame will be, we wait to allocate // the buffers describing the individual frames until we do know their // size. if (mFrameSize > 0) { mBuffers = new sp<MemoryBase>[mNumBuffers]; for (int i = 0; i < mNumBuffers; i++) { mBuffers[i] = new MemoryBase(mHeap, i * mAlignedBufferSize, mFrameSize); } } } AshmemPool::AshmemPool(int buffer_size, int num_buffers, int frame_size, const char *name) : MemPool(buffer_size, num_buffers, frame_size, name) { ALOGV("constructing MemPool %s backed by ashmem: " "%d frames @ %d uint8_ts, " "buffer size %d", mName, num_buffers, frame_size, buffer_size); int page_mask = getpagesize() - 1; int ashmem_size = buffer_size * num_buffers; ashmem_size += page_mask; ashmem_size &= ~page_mask; mHeap = new MemoryHeapBase(ashmem_size); completeInitialization(); } static bool register_buf(int size, int frame_size, int cbcr_offset, int yoffset, int pmempreviewfd, uint32_t offset, uint8_t *buf, int pmem_type, bool vfe_can_write, bool register_buffer) { /*TODO*/ /* struct msm_pmem_info pmemBuf; CAMERA_HAL_UNUSED(frame_size); pmemBuf.type = pmem_type; pmemBuf.fd = pmempreviewfd; pmemBuf.offset = offset; pmemBuf.len = size; pmemBuf.vaddr = buf; pmemBuf.y_off = yoffset; pmemBuf.cbcr_off = cbcr_offset; pmemBuf.active = vfe_can_write; ALOGV("register_buf: reg = %d buffer = %p", !register_buffer, buf); if(native_start_ops(register_buffer ? CAMERA_OPS_REGISTER_BUFFER : CAMERA_OPS_UNREGISTER_BUFFER ,(void *)&pmemBuf) < 0) { ALOGE("register_buf: MSM_CAM_IOCTL_(UN)REGISTER_PMEM error %s", strerror(errno)); return false; }*/ return true; } #if 0 bool register_record_buffers(bool register_buffer) { ALOGI("%s: (%d) E", __FUNCTION__, register_buffer); struct msm_pmem_info pmemBuf; for (int cnt = 0; cnt < VIDEO_BUFFER_COUNT; ++cnt) { pmemBuf.type = MSM_PMEM_VIDEO; pmemBuf.fd = mRecordHeap->mHeap->getHeapID(); pmemBuf.offset = mRecordHeap->mAlignedBufferSize * cnt; pmemBuf.len = mRecordHeap->mBufferSize; pmemBuf.vaddr = (uint8_t *)mRecordHeap->mHeap->base() + mRecordHeap->mAlignedBufferSize * cnt; pmemBuf.y_off = 0; pmemBuf.cbcr_off = recordframes[0].cbcr_off; if(register_buffer == true) { pmemBuf.active = (cnt<ACTIVE_VIDEO_BUFFERS); if( (mVpeEnabled) && (cnt == kRecordBufferCount-1)) { pmemBuf.type = MSM_PMEM_VIDEO_VPE; pmemBuf.active = 1; } } else { pmemBuf.active = false; } ALOGV("register_buf: reg = %d buffer = %p", !register_buffer, (void *)pmemBuf.vaddr); if(native_start_ops(register_buffer ? CAMERA_OPS_REGISTER_BUFFER : CAMERA_OPS_UNREGISTER_BUFFER ,(void *)&pmemBuf) < 0) { ALOGE("register_buf: MSM_CAM_IOCTL_(UN)REGISTER_PMEM error %s", strerror(errno)); return false; } } return true; } #endif #if 0 PmemPool::PmemPool(const char *pmem_pool, int flags, int pmem_type, int buffer_size, int num_buffers, int frame_size, int cbcr_offset, int yOffset, const char *name) : MemPool(buffer_size,num_buffers,frame_size,name), mPmemType(pmem_type), mCbCrOffset(cbcr_offset), myOffset(yOffset) { ALOGI("constructing MemPool %s backed by pmem pool %s: " "%d frames @ %d bytes, buffer size %d", mName, pmem_pool, num_buffers, frame_size, buffer_size); //mMMCameraDLRef = MMCameraDL::getInstance(); // Make a new mmap'ed heap that can be shared across processes. // mAlignedBufferSize is already in 4k aligned. (do we need total size necessary to be in power of 2??) mAlignedSize = mAlignedBufferSize * num_buffers; sp<MemoryHeapBase> masterHeap = new MemoryHeapBase(pmem_pool, mAlignedSize, flags); if (masterHeap->getHeapID() < 0) { ALOGE("failed to construct master heap for pmem pool %s", pmem_pool); masterHeap.clear(); return; } sp<MemoryHeapPmem> pmemHeap = new MemoryHeapPmem(masterHeap, flags); if (pmemHeap->getHeapID() >= 0) { pmemHeap->slap(); masterHeap.clear(); mHeap = pmemHeap; pmemHeap.clear(); mFd = mHeap->getHeapID(); if (::ioctl(mFd, PMEM_GET_SIZE, &mSize)) { ALOGE("pmem pool %s ioctl(PMEM_GET_SIZE) error %s (%d)", pmem_pool, ::strerror(errno), errno); mHeap.clear(); return; } ALOGE("pmem pool %s ioctl(fd = %d, PMEM_GET_SIZE) is %ld", pmem_pool, mFd, mSize.len); ALOGE("mBufferSize=%d, mAlignedBufferSize=%d\n", mBufferSize, mAlignedBufferSize); #if 0 // Unregister preview buffers with the camera drivers. Allow the VFE to write // to all preview buffers except for the last one. // Only Register the preview, snapshot and thumbnail buffers with the kernel. if( (strcmp("postview", mName) != 0) ){ int num_buf = num_buffers; if(!strcmp("preview", mName)) num_buf = kPreviewBufferCount; ALOGD("num_buffers = %d", num_buf); for (int cnt = 0; cnt < num_buf; ++cnt) { int active = 1; if(pmem_type == MSM_PMEM_VIDEO){ active = (cnt<ACTIVE_VIDEO_BUFFERS); //When VPE is enabled, set the last record //buffer as active and pmem type as PMEM_VIDEO_VPE //as this is a requirement from VPE operation. //No need to set this pmem type to VIDEO_VPE while unregistering, //because as per camera stack design: "the VPE AXI is also configured //when VFE is configured for VIDEO, which is as part of preview //initialization/start. So during this VPE AXI config camera stack //will lookup the PMEM_VIDEO_VPE buffer and give it as o/p of VPE and //change it's type to PMEM_VIDEO". if( (mVpeEnabled) && (cnt == kRecordBufferCount-1)) { active = 1; pmem_type = MSM_PMEM_VIDEO_VPE; } ALOGV(" pmempool creating video buffers : active %d ", active); } else if (pmem_type == MSM_PMEM_PREVIEW){ active = (cnt < ACTIVE_PREVIEW_BUFFERS); } else if ((pmem_type == MSM_PMEM_MAINIMG) || (pmem_type == MSM_PMEM_THUMBNAIL)){ active = (cnt < ACTIVE_ZSL_BUFFERS); } register_buf(mBufferSize, mFrameSize, mCbCrOffset, myOffset, mHeap->getHeapID(), mAlignedBufferSize * cnt, (uint8_t *)mHeap->base() + mAlignedBufferSize * cnt, pmem_type, active); } } #endif completeInitialization(); } else ALOGE("pmem pool %s error: could not create master heap!", pmem_pool); ALOGI("%s: (%s) X ", __FUNCTION__, mName); } #endif PmemPool::~PmemPool() { ALOGV("%s: %s E", __FUNCTION__, mName); #if 0 if (mHeap != NULL) { // Unregister preview buffers with the camera drivers. // Only Unregister the preview, snapshot and thumbnail // buffers with the kernel. if( (strcmp("postview", mName) != 0) ){ int num_buffers = mNumBuffers; if(!strcmp("preview", mName)) num_buffers = PREVIEW_BUFFER_COUNT; for (int cnt = 0; cnt < num_buffers; ++cnt) { register_buf(mBufferSize, mFrameSize, mCbCrOffset, myOffset, mHeap->getHeapID(), mAlignedBufferSize * cnt, (uint8_t *)mHeap->base() + mAlignedBufferSize * cnt, mPmemType, false, false /* unregister */); } } } mMMCameraDLRef.clear(); #endif ALOGV("%s: %s X", __FUNCTION__, mName); } MemPool::~MemPool() { ALOGV("destroying MemPool %s", mName); if (mFrameSize > 0) delete [] mBuffers; mHeap.clear(); ALOGV("destroying MemPool %s completed", mName); } status_t MemPool::dump(int fd, const Vector<String16>& args) const { const size_t SIZE = 256; char buffer[SIZE]; String8 result; CAMERA_HAL_UNUSED(args); snprintf(buffer, 255, "QualcommCameraHardware::AshmemPool::dump\n"); result.append(buffer); if (mName) { snprintf(buffer, 255, "mem pool name (%s)\n", mName); result.append(buffer); } if (mHeap != 0) { snprintf(buffer, 255, "heap base(%p), size(%d), flags(%d), device(%s)\n", mHeap->getBase(), mHeap->getSize(), mHeap->getFlags(), mHeap->getDevice()); result.append(buffer); } snprintf(buffer, 255, "buffer size (%d), number of buffers (%d), frame size(%d)", mBufferSize, mNumBuffers, mFrameSize); result.append(buffer); write(fd, result.string(), result.size()); return NO_ERROR; } };