/* * Copyright (c) 2009-2011 Intel Corporation. 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. */ //#define LOG_NDEBUG 0 #define LOG_TAG "IntelMetadataBuffer" #include <wrs_omxil_core/log.h> #include "IntelMetadataBuffer.h" #include <string.h> #include <stdio.h> #ifdef INTEL_VIDEO_XPROC_SHARING #include <binder/IServiceManager.h> #include <binder/MemoryBase.h> #include <binder/Parcel.h> #include <utils/List.h> #include <utils/threads.h> #include <ui/GraphicBuffer.h> //#define TEST struct ShareMemMap { uint32_t sessionflag; intptr_t value; intptr_t value_backup; uint32_t type; sp<MemoryBase> membase; sp<GraphicBuffer> gbuffer; }; List <ShareMemMap *> gShareMemMapList; Mutex gShareMemMapListLock; enum { SHARE_MEM = IBinder::FIRST_CALL_TRANSACTION, GET_MEM, CLEAR_MEM, }; enum { ST_MEMBASE = 0, ST_GFX, ST_MAX, }; #define REMOTE_PROVIDER 0x80000000 #define REMOTE_CONSUMER 0x40000000 static ShareMemMap* ReadMemObjFromBinder(const Parcel& data, uint32_t sessionflag, intptr_t value) { uint32_t type = data.readInt32(); if (type >= ST_MAX) return NULL; ShareMemMap* map = new ShareMemMap; map->sessionflag = sessionflag; map->type = type; map->value_backup = value; map->membase = NULL; map->gbuffer= NULL; // LOGI("ReadMemObjFromBinder"); if (type == ST_MEMBASE) /*offset, size, heap*/ { ssize_t offset = data.readInt32(); size_t size = data.readInt32(); sp<IMemoryHeap> heap = interface_cast<IMemoryHeap>(data.readStrongBinder()); sp<MemoryBase> mem = new MemoryBase(heap, offset, size); if (mem == NULL) { delete map; return NULL; } map->value = (intptr_t)( mem->pointer() + 0x0FFF) & ~0x0FFF; map->membase = mem; #ifdef TEST ALOGI("membase heapID:%d, pointer:%x data:%x, aligned value:%x", \ heap->getHeapID(), mem->pointer(), *((intptr_t *)(mem->pointer())), map->value); #endif } else if (type == ST_GFX) /*graphicbuffer*/ { sp<GraphicBuffer> buffer = new GraphicBuffer(); if (buffer == NULL) { delete map; return NULL; } data.read(*buffer); map->value = (intptr_t)buffer->handle; map->gbuffer = buffer; #ifdef TEST void* usrptr[3]; buffer->lock(GraphicBuffer::USAGE_HW_TEXTURE | GraphicBuffer::USAGE_SW_READ_OFTEN, &usrptr[0]); buffer->unlock(); ALOGI("gfx handle:%p data:%x", (intptr_t)buffer->handle, *((intptr_t *)usrptr[0])); #endif } gShareMemMapListLock.lock(); gShareMemMapList.push_back(map); gShareMemMapListLock.unlock(); return map; } static status_t WriteMemObjToBinder(Parcel& data, ShareMemMap* smem) { if (smem->type >= ST_MAX) return BAD_VALUE; // LOGI("WriteMemObjToBinder"); data.writeInt32(smem->type); if (smem->type == ST_MEMBASE) /*offset, size, heap*/ { ssize_t offset; size_t size; sp<IMemoryHeap> heap = smem->membase->getMemory(&offset, &size); data.writeInt32(offset); data.writeInt32(size); data.writeStrongBinder(heap->asBinder()); #ifdef TEST ALOGI("membase heapID:%d pointer:%x data:%x", \ heap->getHeapID(), smem->membase->pointer(), *((int *)(smem->membase->pointer()))); #endif } else if (smem->type == ST_GFX) /*graphicbuffer*/ data.write(*(smem->gbuffer)); return NO_ERROR; } static void ClearLocalMem(uint32_t sessionflag) { List<ShareMemMap *>::iterator node; gShareMemMapListLock.lock(); for(node = gShareMemMapList.begin(); node != gShareMemMapList.end(); ) { if ((*node)->sessionflag == sessionflag) //remove all buffers belong to this session { (*node)->membase = NULL; (*node)->gbuffer = NULL; delete (*node); node = gShareMemMapList.erase(node); } else node ++; } gShareMemMapListLock.unlock(); } static ShareMemMap* FindShareMem(uint32_t sessionflag, intptr_t value, bool isBackup) { List<ShareMemMap *>::iterator node; gShareMemMapListLock.lock(); for(node = gShareMemMapList.begin(); node != gShareMemMapList.end(); node++) { if (isBackup) { if ((*node)->sessionflag == sessionflag && (*node)->value_backup == value) { gShareMemMapListLock.unlock(); return (*node); } } else if ((*node)->sessionflag == sessionflag && (*node)->value == value) { gShareMemMapListLock.unlock(); return (*node); } } gShareMemMapListLock.unlock(); return NULL; } static ShareMemMap* PopShareMem(uint32_t sessionflag, intptr_t value) { List<ShareMemMap *>::iterator node; gShareMemMapListLock.lock(); for(node = gShareMemMapList.begin(); node != gShareMemMapList.end(); node++) { if ((*node)->sessionflag == sessionflag && (*node)->value == value) { gShareMemMapList.erase(node); gShareMemMapListLock.unlock(); return (*node); } } gShareMemMapListLock.unlock(); return NULL; } static void PushShareMem(ShareMemMap* &smem) { gShareMemMapListLock.lock(); gShareMemMapList.push_back(smem); gShareMemMapListLock.unlock(); } static sp<IBinder> GetIntelBufferSharingService() { sp<IServiceManager> sm = defaultServiceManager(); sp<IBinder> binder = sm->checkService(String16("media.IntelBufferSharing")); if (binder == 0) ALOGE("media.IntelBufferSharing service is not published"); return binder; } IntelBufferSharingService* IntelBufferSharingService::gBufferService = NULL; status_t IntelBufferSharingService::instantiate(){ status_t ret = NO_ERROR; if (gBufferService == NULL) { gBufferService = new IntelBufferSharingService(); ret = defaultServiceManager()->addService(String16("media.IntelBufferSharing"), gBufferService); LOGI("IntelBufferSharingService::instantiate() ret = %d\n", ret); } return ret; } status_t IntelBufferSharingService::onTransact(uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) { //TODO: if pid is int32? pid_t pid = data.readInt32(); uint32_t sessionflag = data.readInt32(); switch(code) { case SHARE_MEM: { if (pid == getpid()) //in same process, should not use binder { ALOGE("onTransact in same process, wrong sessionflag?"); return UNKNOWN_ERROR; } intptr_t value = data.readIntPtr(); // LOGI("onTransact SHARE_MEM value=%x", value); //different process ShareMemMap* map = ReadMemObjFromBinder(data, sessionflag, value); if (map == NULL) return UNKNOWN_ERROR; reply->writeIntPtr(map->value); return NO_ERROR; } case CLEAR_MEM: { // LOGI("onTransact CLEAR_MEM sessionflag=%x", sessionflag); if (pid == getpid()) //in same process, should not use binder { //same process, return same pointer in data ALOGE("onTransact CLEAR_MEM in same process, wrong sessionflag?"); return UNKNOWN_ERROR; } ClearLocalMem(sessionflag); return NO_ERROR; } case GET_MEM: { if (pid == getpid()) //in same process, should not use binder { ALOGE("onTransact GET_MEM in same process, wrong sessionflag?"); return UNKNOWN_ERROR; } intptr_t value = data.readIntPtr(); // LOGI("onTransact GET_MEM value=%x", value); ShareMemMap* smem = FindShareMem(sessionflag, value, false); if (smem && (NO_ERROR == WriteMemObjToBinder(*reply, smem))) return NO_ERROR; else ALOGE("onTransact GET_MEM: Not find mem"); return UNKNOWN_ERROR; } default: return BBinder::onTransact(code, data, reply, flags); } return NO_ERROR; } #endif IntelMetadataBuffer::IntelMetadataBuffer() { mType = IntelMetadataBufferTypeCameraSource; mValue = 0; mInfo = NULL; mExtraValues = NULL; mExtraValues_Count = 0; mBytes = NULL; mSize = 0; #ifdef INTEL_VIDEO_XPROC_SHARING mSessionFlag = 0; #endif } IntelMetadataBuffer::IntelMetadataBuffer(IntelMetadataBufferType type, intptr_t value) { mType = type; mValue = value; mInfo = NULL; mExtraValues = NULL; mExtraValues_Count = 0; mBytes = NULL; mSize = 0; #ifdef INTEL_VIDEO_XPROC_SHARING mSessionFlag = 0; #endif } IntelMetadataBuffer::~IntelMetadataBuffer() { if (mInfo) delete mInfo; if (mExtraValues) delete[] mExtraValues; if (mBytes) delete[] mBytes; } IntelMetadataBuffer::IntelMetadataBuffer(const IntelMetadataBuffer& imb) :mType(imb.mType), mValue(imb.mValue), mInfo(NULL), mExtraValues(NULL), mExtraValues_Count(imb.mExtraValues_Count), mBytes(NULL), mSize(imb.mSize) #ifdef INTEL_VIDEO_XPROC_SHARING ,mSessionFlag(imb.mSessionFlag) #endif { if (imb.mInfo) mInfo = new ValueInfo(*imb.mInfo); if (imb.mExtraValues) { mExtraValues = new intptr_t[mExtraValues_Count]; memcpy(mExtraValues, imb.mExtraValues, sizeof(mValue) * mExtraValues_Count); } if (imb.mBytes) { mBytes = new uint8_t[mSize]; memcpy(mBytes, imb.mBytes, mSize); } } const IntelMetadataBuffer& IntelMetadataBuffer::operator=(const IntelMetadataBuffer& imb) { mType = imb.mType; mValue = imb.mValue; mInfo = NULL; mExtraValues = NULL; mExtraValues_Count = imb.mExtraValues_Count; mBytes = NULL; mSize = imb.mSize; #ifdef INTEL_VIDEO_XPROC_SHARING mSessionFlag = imb.mSessionFlag; #endif if (imb.mInfo) mInfo = new ValueInfo(*imb.mInfo); if (imb.mExtraValues) { mExtraValues = new intptr_t[mExtraValues_Count]; memcpy(mExtraValues, imb.mExtraValues, sizeof(mValue) * mExtraValues_Count); } if (imb.mBytes) { mBytes = new uint8_t[mSize]; memcpy(mBytes, imb.mBytes, mSize); } return *this; } IMB_Result IntelMetadataBuffer::GetType(IntelMetadataBufferType& type) { type = mType; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::SetType(IntelMetadataBufferType type) { if (type < IntelMetadataBufferTypeLast) mType = type; else return IMB_INVAL_PARAM; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::GetValue(intptr_t& value) { value = mValue; #ifndef INTEL_VIDEO_XPROC_SHARING return IMB_SUCCESS; #else if ((mSessionFlag & REMOTE_CONSUMER) == 0) //no sharing or is local consumer return IMB_SUCCESS; //try to find if it is already cached. ShareMemMap* smem = FindShareMem(mSessionFlag, mValue, true); if(smem) { value = smem->value; return IMB_SUCCESS; } //is remote provider and not find from cache, then pull from service sp<IBinder> binder = GetIntelBufferSharingService(); if (binder == 0) return IMB_NO_SERVICE; //Detect IntelBufferSharingService, share mem to service Parcel data, reply; //send pid, sessionflag, and memtype pid_t pid = getpid(); //TODO: if pid is int32? data.writeInt32(pid); data.writeInt32(mSessionFlag); data.writeIntPtr(mValue); //do transcation if (binder->transact(GET_MEM, data, &reply) != NO_ERROR) return IMB_SERVICE_FAIL; //get type/Mem OBJ smem = ReadMemObjFromBinder(reply, mSessionFlag, mValue); if (smem) value = smem->value; else return IMB_SERVICE_FAIL; return IMB_SUCCESS; #endif } IMB_Result IntelMetadataBuffer::SetValue(intptr_t value) { mValue = value; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::GetValueInfo(ValueInfo* &info) { info = mInfo; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::SetValueInfo(ValueInfo* info) { if (info) { if (mInfo == NULL) mInfo = new ValueInfo; memcpy(mInfo, info, sizeof(ValueInfo)); } else return IMB_INVAL_PARAM; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::GetExtraValues(intptr_t* &values, uint32_t& num) { values = mExtraValues; num = mExtraValues_Count; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::SetExtraValues(intptr_t* values, uint32_t num) { if (values && num > 0) { if (mExtraValues && mExtraValues_Count != num) { delete[] mExtraValues; mExtraValues = NULL; } if (mExtraValues == NULL) mExtraValues = new intptr_t[num]; memcpy(mExtraValues, values, sizeof(intptr_t) * num); mExtraValues_Count = num; } else return IMB_INVAL_PARAM; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::UnSerialize(uint8_t* data, uint32_t size) { if (!data || size == 0) return IMB_INVAL_PARAM; IntelMetadataBufferType type; intptr_t value; uint32_t extrasize = size - sizeof(type) - sizeof(value); ValueInfo* info = NULL; intptr_t* ExtraValues = NULL; uint32_t ExtraValues_Count = 0; memcpy(&type, data, sizeof(type)); data += sizeof(type); memcpy(&value, data, sizeof(value)); data += sizeof(value); switch (type) { case IntelMetadataBufferTypeCameraSource: case IntelMetadataBufferTypeEncoder: case IntelMetadataBufferTypeUser: { if (extrasize >0 && extrasize < sizeof(ValueInfo)) return IMB_INVAL_BUFFER; if (extrasize > sizeof(ValueInfo)) //has extravalues { if ( (extrasize - sizeof(ValueInfo)) % sizeof(mValue) != 0 ) return IMB_INVAL_BUFFER; ExtraValues_Count = (extrasize - sizeof(ValueInfo)) / sizeof(mValue); } if (extrasize > 0) { info = new ValueInfo; memcpy(info, data, sizeof(ValueInfo)); data += sizeof(ValueInfo); } if (ExtraValues_Count > 0) { ExtraValues = new intptr_t[ExtraValues_Count]; memcpy(ExtraValues, data, ExtraValues_Count * sizeof(mValue)); } break; } case IntelMetadataBufferTypeGrallocSource: if (extrasize > 0) return IMB_INVAL_BUFFER; break; default: return IMB_INVAL_BUFFER; } //store data mType = type; mValue = value; if (mInfo) delete mInfo; mInfo = info; if (mExtraValues) delete[] mExtraValues; mExtraValues = ExtraValues; mExtraValues_Count = ExtraValues_Count; #ifdef INTEL_VIDEO_XPROC_SHARING if (mInfo != NULL) mSessionFlag = mInfo->sessionFlag; #endif return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::Serialize(uint8_t* &data, uint32_t& size) { if (mBytes == NULL) { if (mType == IntelMetadataBufferTypeGrallocSource && mInfo) return IMB_INVAL_PARAM; //assemble bytes according members mSize = sizeof(mType) + sizeof(mValue); if (mInfo) { mSize += sizeof(ValueInfo); if (mExtraValues) mSize += sizeof(mValue) * mExtraValues_Count; } mBytes = new uint8_t[mSize]; uint8_t *ptr = mBytes; memcpy(ptr, &mType, sizeof(mType)); ptr += sizeof(mType); memcpy(ptr, &mValue, sizeof(mValue)); ptr += sizeof(mValue); if (mInfo) { #ifdef INTEL_VIDEO_XPROC_SHARING mInfo->sessionFlag = mSessionFlag; #endif memcpy(ptr, mInfo, sizeof(ValueInfo)); ptr += sizeof(ValueInfo); if (mExtraValues) memcpy(ptr, mExtraValues, mExtraValues_Count * sizeof(mValue)); } } data = mBytes; size = mSize; return IMB_SUCCESS; } uint32_t IntelMetadataBuffer::GetMaxBufferSize() { return 256; } #ifdef INTEL_VIDEO_XPROC_SHARING IMB_Result IntelMetadataBuffer::GetSessionFlag(uint32_t& sessionflag) { sessionflag = mSessionFlag; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::SetSessionFlag(uint32_t sessionflag) { mSessionFlag = sessionflag; return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::ShareValue(sp<MemoryBase> mem) { mValue = (intptr_t)((intptr_t) ( mem->pointer() + 0x0FFF) & ~0x0FFF); if ( !(mSessionFlag & REMOTE_PROVIDER) && !(mSessionFlag & REMOTE_CONSUMER)) //no sharing return IMB_SUCCESS; if (mSessionFlag & REMOTE_PROVIDER) //is remote provider { sp<IBinder> binder = GetIntelBufferSharingService(); if (binder == 0) return IMB_NO_SERVICE; //Detect IntelBufferSharingService, share mem to service Parcel data, reply; //send pid, sessionflag, and value pid_t pid = getpid(); //TODO: if pid is int32? data.writeInt32(pid); data.writeInt32(mSessionFlag); data.writeIntPtr(mValue); //send type/obj (offset/size/MemHeap) ShareMemMap smem; smem.membase = mem; smem.type = ST_MEMBASE; if (WriteMemObjToBinder(data, &smem) != NO_ERROR) return IMB_SERVICE_FAIL; //do transcation if (binder->transact(SHARE_MEM, data, &reply) != NO_ERROR) return IMB_SERVICE_FAIL; //set new value gotten from peer mValue = reply.readIntPtr(); // LOGI("ShareValue(membase) Get reply from sevice, new value:%x\n", mValue); } else //is local provider , direct access list { ShareMemMap* smem = new ShareMemMap; smem->sessionflag = mSessionFlag; smem->value = mValue; smem->value_backup = mValue; smem->type = ST_MEMBASE; smem->membase = mem; smem->gbuffer = NULL; PushShareMem(smem); } return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::ShareValue(sp<GraphicBuffer> gbuffer) { mValue = (intptr_t)gbuffer->handle; if ( !(mSessionFlag & REMOTE_PROVIDER) && !(mSessionFlag & REMOTE_CONSUMER)) //no sharing return IMB_SUCCESS; if (mSessionFlag & REMOTE_PROVIDER == 0) //is remote provider { sp<IBinder> binder = GetIntelBufferSharingService(); if (binder == 0) return IMB_NO_SERVICE; Parcel data, reply; //send pid, sessionflag, and memtype pid_t pid = getpid(); //TODO: if pid is int32 ? data.writeInt32(pid); data.writeInt32(mSessionFlag); data.writeIntPtr(mValue); //send value/graphicbuffer obj ShareMemMap smem; smem.gbuffer = gbuffer; smem.type = ST_GFX; if (WriteMemObjToBinder(data, &smem) != NO_ERROR) return IMB_SERVICE_FAIL; //do transcation if (binder->transact(SHARE_MEM, data, &reply) != NO_ERROR) return IMB_SERVICE_FAIL; //set new value gotten from peer mValue = reply.readIntPtr(); // LOGI("ShareValue(gfx) Get reply from sevice, new value:%x\n", mValue); } else //is local provider, direct access list { ShareMemMap* smem = new ShareMemMap; smem->sessionflag = mSessionFlag; smem->value = mValue; smem->value_backup = mValue; smem->type = ST_GFX; smem->membase = NULL; smem->gbuffer = gbuffer; PushShareMem(smem); } return IMB_SUCCESS; } IMB_Result IntelMetadataBuffer::ClearContext(uint32_t sessionflag, bool isProvider) { if ( !(sessionflag & REMOTE_PROVIDER) && !(sessionflag & REMOTE_CONSUMER)) //no sharing return IMB_SUCCESS; //clear local firstly ClearLocalMem(sessionflag); //clear mem on service if it is remote user if ((isProvider && (sessionflag & REMOTE_PROVIDER)) || (!isProvider && (sessionflag & REMOTE_CONSUMER))) { // LOGI("CLEAR_MEM sessionflag=%x", sessionflag); sp<IBinder> binder = GetIntelBufferSharingService(); if (binder == 0) return IMB_NO_SERVICE; //Detect IntelBufferSharingService, unshare mem from service Parcel data, reply; //send pid and sessionflag pid_t pid = getpid(); //TODO: if pid is int32? data.writeInt32(pid); data.writeInt32(sessionflag); if (binder->transact(CLEAR_MEM, data, &reply) != NO_ERROR) return IMB_SERVICE_FAIL; } return IMB_SUCCESS; } uint32_t IntelMetadataBuffer::MakeSessionFlag(bool romoteProvider, bool remoteConsumer, uint16_t sindex) { uint32_t sessionflag = 0; if (romoteProvider) sessionflag |= REMOTE_PROVIDER; if (remoteConsumer) sessionflag |= REMOTE_CONSUMER; return sessionflag + sindex; } #endif