1 /* 2 * Copyright (C) 2012 Intel Corporation. All rights reserved. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 * 16 */ 17 18 #include <inttypes.h> 19 20 #include <media/hardware/HardwareAPI.h> 21 #include <system/graphics.h> 22 #include "isv_bufmanager.h" 23 #ifndef TARGET_VPP_USE_GEN 24 #include "hal_public.h" 25 #endif 26 27 //#define LOG_NDEBUG 0 28 #undef LOG_TAG 29 #define LOG_TAG "isv-omxil" 30 31 using namespace android; 32 33 #define GRALLOC_SUB_BUFFER_MAX 3 34 #define RANDOM_BUFFER_SIZE 200 35 static char random_buf[RANDOM_BUFFER_SIZE]; 36 37 ISVBuffer::~ISVBuffer() { 38 if (mWorker != NULL) { 39 ALOGV("%s: mSurface %d", __func__, mSurface); 40 mWorker->freeSurface(&mSurface); 41 } 42 } 43 44 status_t ISVBuffer::initBufferInfo(uint32_t hackFormat) 45 { 46 if (mType == ISV_BUFFER_METADATA) { 47 VideoDecoderOutputMetaData *metaData = 48 reinterpret_cast<VideoDecoderOutputMetaData*>(mBuffer); 49 50 if (metaData->eType != kMetadataBufferTypeGrallocSource) { 51 ALOGE("%s: unsupported meta data format eType = %d", __func__, metaData->eType); 52 return UNKNOWN_ERROR; 53 } 54 55 if (mGrallocHandle != 0) { 56 if ((unsigned long)metaData->pHandle != mGrallocHandle) { 57 if (STATUS_OK != mWorker->freeSurface(&mSurface)) { 58 ALOGE("%s: free surface %d failed.", __func__, mSurface); 59 return UNKNOWN_ERROR; 60 } 61 } else 62 return OK; 63 } 64 mGrallocHandle = (unsigned long)metaData->pHandle; 65 } else { 66 if (mSurface != -1) 67 return OK; 68 mGrallocHandle = mBuffer; 69 } 70 71 int32_t err = 0; 72 if (!mpGralloc) { 73 err = hw_get_module(GRALLOC_HARDWARE_MODULE_ID, (hw_module_t const**)&mpGralloc); 74 if (0 != err) 75 return UNKNOWN_ERROR; 76 } 77 #ifdef TARGET_VPP_USE_GEN 78 ufo_buffer_details_t info; 79 80 memset(&info, 0, sizeof(ufo_buffer_details_t)); 81 err = mpGralloc->perform(mpGralloc, INTEL_UFO_GRALLOC_MODULE_PERFORM_GET_BO_INFO, mGrallocHandle, &info); 82 83 if (0 != err) { 84 ALOGE("%s: can't get graphic buffer info", __func__); 85 } 86 mWidth = info.width; 87 mHeight = info.height; 88 mStride = info.pitch; 89 mColorFormat = info.format; 90 #else 91 IMG_native_handle_t* grallocHandle = (IMG_native_handle_t*)mGrallocHandle; 92 mStride = grallocHandle->aiStride[0]; 93 mSurfaceHeight = grallocHandle->iHeight; 94 mColorFormat = (hackFormat != 0) ? hackFormat : grallocHandle->iFormat; 95 #endif 96 if (mWorker == NULL) { 97 ALOGE("%s: mWorker == NULL!!", __func__); 98 return UNKNOWN_ERROR; 99 } 100 101 if (STATUS_OK != mWorker->allocSurface(&mWidth, &mHeight, mStride, mColorFormat, mGrallocHandle, &mSurface)) { 102 ALOGE("%s: alloc surface failed, mGrallocHandle %lu", __func__, mGrallocHandle); 103 return UNKNOWN_ERROR; 104 } 105 106 ALOGD_IF(ISV_BUFFER_MANAGER_DEBUG, 107 "%s: mWidth %d, mHeight %d, mStride %d, mColorFormat %d, " 108 "mGrallocHandle %p, mSurface %d", 109 __func__, mWidth, mHeight, mStride, mColorFormat, 110 reinterpret_cast<void*>(mGrallocHandle), mSurface); 111 return OK; 112 } 113 114 status_t ISVBuffer::clearIfNeed() 115 { 116 #ifndef TARGET_VPP_USE_GEN 117 static bool bRandomBufferInit = false; 118 if (!bRandomBufferInit) { 119 time_t my_time; 120 srand((unsigned)time(&my_time)); 121 for (int32_t i = 0; i < RANDOM_BUFFER_SIZE; i++) 122 random_buf[i] = (char)(((double)rand()/(double)RAND_MAX) * 255.0); 123 bRandomBufferInit = true; 124 } 125 126 if ((mFlags & ISV_BUFFER_NEED_CLEAR) && mpGralloc) { 127 int32_t usage = GRALLOC_USAGE_HW_TEXTURE | GRALLOC_USAGE_SW_READ_OFTEN | GRALLOC_USAGE_SW_WRITE_OFTEN; 128 void *vaddr[GRALLOC_SUB_BUFFER_MAX]; 129 130 int32_t err = mpGralloc->lock(mpGralloc, (buffer_handle_t)mGrallocHandle, usage, 0, 0, mStride, mSurfaceHeight, &vaddr[0]); 131 132 if (0 != err) { 133 ALOGE("%s: get graphic buffer ptr failed", __func__); 134 return UNKNOWN_ERROR; 135 } 136 137 int32_t buffer_size = mStride * mSurfaceHeight * 3 / 2; 138 char* ptr = (char*)vaddr[0]; 139 for (int32_t i = 0; i < buffer_size/RANDOM_BUFFER_SIZE; i++) { 140 memcpy(ptr, random_buf, sizeof(random_buf)); 141 ptr += sizeof(random_buf); 142 } 143 mpGralloc->unlock(mpGralloc, (buffer_handle_t)mGrallocHandle); 144 ALOGD_IF(ISV_BUFFER_MANAGER_DEBUG, "%s: clear isv buffer %p finished, buffer size %d", __func__, this, buffer_size); 145 mFlags &= ~ISV_BUFFER_NEED_CLEAR; 146 } 147 #endif 148 return OK; 149 } 150 151 status_t ISVBufferManager::setBufferCount(int32_t size) 152 { 153 Mutex::Autolock autoLock(mBufferLock); 154 #if 0 155 if (!mBuffers.isEmpty()) { 156 ALOGE("%s: the buffer queue should be empty before we set its size", __func__); 157 return STATUS_ERROR; 158 } 159 #endif 160 mBuffers.setCapacity(size); 161 162 return OK; 163 } 164 165 status_t ISVBufferManager::freeBuffer(unsigned long handle) 166 { 167 Mutex::Autolock autoLock(mBufferLock); 168 for (uint32_t i = 0; i < mBuffers.size(); i++) { 169 ISVBuffer* isvBuffer = mBuffers.itemAt(i); 170 if (isvBuffer->getHandle() == handle) { 171 delete isvBuffer; 172 mBuffers.removeAt(i); 173 ALOGD_IF(ISV_BUFFER_MANAGER_DEBUG, "%s: remove handle 0x%08lx, and then mBuffers.size() %d", __func__, 174 handle, mBuffers.size()); 175 return OK; 176 } 177 } 178 179 ALOGW("%s: can't find buffer %lu", __func__, handle); 180 return UNKNOWN_ERROR; 181 } 182 183 status_t ISVBufferManager::useBuffer(unsigned long handle) 184 { 185 Mutex::Autolock autoLock(mBufferLock); 186 if (handle == 0 || mBuffers.size() >= mBuffers.capacity()) 187 return BAD_VALUE; 188 189 for (uint32_t i = 0; i < mBuffers.size(); i++) { 190 ISVBuffer* isvBuffer = mBuffers.itemAt(i); 191 if (isvBuffer->getHandle() == handle) { 192 ALOGE("%s: this buffer 0x%08lx has already been registered", __func__, handle); 193 return UNKNOWN_ERROR; 194 } 195 } 196 197 ISVBuffer* isvBuffer = new ISVBuffer(mWorker, handle, 198 mMetaDataMode ? ISVBuffer::ISV_BUFFER_METADATA : ISVBuffer::ISV_BUFFER_GRALLOC, 199 mNeedClearBuffers ? ISVBuffer::ISV_BUFFER_NEED_CLEAR : 0); 200 201 ALOGD_IF(ISV_BUFFER_MANAGER_DEBUG, 202 "%s: add handle 0x%08lx, and then mBuffers.size() %d", __func__, 203 handle, mBuffers.size()); 204 mBuffers.push_back(isvBuffer); 205 return OK; 206 207 } 208 209 status_t ISVBufferManager::useBuffer(const sp<ANativeWindowBuffer> nativeBuffer) 210 { 211 Mutex::Autolock autoLock(mBufferLock); 212 if (nativeBuffer == NULL || mBuffers.size() >= mBuffers.capacity()) 213 return BAD_VALUE; 214 215 for (uint32_t i = 0; i < mBuffers.size(); i++) { 216 ISVBuffer* isvBuffer = mBuffers.itemAt(i); 217 if (isvBuffer->getHandle() == (unsigned long)nativeBuffer->handle) { 218 ALOGE( 219 "%s: this buffer 0x%08" PRIxPTR " has already been registered", 220 __func__, reinterpret_cast<uintptr_t>(nativeBuffer->handle)); 221 return UNKNOWN_ERROR; 222 } 223 } 224 225 ISVBuffer* isvBuffer = new ISVBuffer(mWorker, 226 (unsigned long)nativeBuffer->handle, (unsigned long)nativeBuffer->handle, 227 nativeBuffer->width, nativeBuffer->height, 228 nativeBuffer->stride, nativeBuffer->format, 229 mMetaDataMode ? ISVBuffer::ISV_BUFFER_METADATA : ISVBuffer::ISV_BUFFER_GRALLOC, 230 mNeedClearBuffers ? ISVBuffer::ISV_BUFFER_NEED_CLEAR : 0); 231 232 ALOGD_IF(ISV_BUFFER_MANAGER_DEBUG, 233 "%s: add handle 0x%08" PRIxPTR ", and then mBuffers.size() %d", 234 __func__, reinterpret_cast<uintptr_t>(nativeBuffer->handle), 235 mBuffers.size()); 236 mBuffers.push_back(isvBuffer); 237 return OK; 238 } 239 240 ISVBuffer* ISVBufferManager::mapBuffer(unsigned long handle) 241 { 242 Mutex::Autolock autoLock(mBufferLock); 243 for (uint32_t i = 0; i < mBuffers.size(); i++) { 244 ISVBuffer* isvBuffer = mBuffers.itemAt(i); 245 if (isvBuffer->getHandle() == handle) 246 return isvBuffer; 247 } 248 return NULL; 249 } 250 251 status_t ISVBufferManager::setBuffersFlag(uint32_t flag) 252 { 253 Mutex::Autolock autoLock(mBufferLock); 254 255 if (flag & ISVBuffer::ISV_BUFFER_NEED_CLEAR) { 256 if (mBuffers.size() == 0) 257 mNeedClearBuffers = true; 258 else { 259 for (uint32_t i = 0; i < mBuffers.size(); i++) { 260 ISVBuffer* isvBuffer = mBuffers.itemAt(i); 261 isvBuffer->setFlag(ISVBuffer::ISV_BUFFER_NEED_CLEAR); 262 } 263 } 264 } 265 return OK; 266 } 267