/* * Copyright (C) 2008 The Android Open Source Project * * 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. */ #include #include #include #include #include #include #include #include "allocator.h" #include "gr.h" #include "gpu.h" /*****************************************************************************/ static int gralloc_alloc_buffer(alloc_device_t* dev, size_t size, int usage, buffer_handle_t* pHandle); /*****************************************************************************/ int fb_device_open(const hw_module_t* module, const char* name, hw_device_t** device); static int gralloc_device_open(const hw_module_t* module, const char* name, hw_device_t** device); extern int gralloc_lock(gralloc_module_t const* module, buffer_handle_t handle, int usage, int l, int t, int w, int h, void** vaddr); extern int gralloc_unlock(gralloc_module_t const* module, buffer_handle_t handle); extern int gralloc_register_buffer(gralloc_module_t const* module, buffer_handle_t handle); extern int gralloc_unregister_buffer(gralloc_module_t const* module, buffer_handle_t handle); extern int gralloc_perform(struct gralloc_module_t const* module, int operation, ... ); /*****************************************************************************/ /* On-device dependency implementation */ class PmemAllocatorDepsDeviceImpl : public PmemUserspaceAllocator::Deps, public PmemKernelAllocator::Deps { const private_module_t* module; virtual size_t getPmemTotalSize(int fd, size_t* size) { int err = 0; #ifndef TARGET_MSM7x27 pmem_region region; err = ioctl(fd, PMEM_GET_TOTAL_SIZE, ®ion); if (err == 0) { *size = region.len; } #else #ifdef USE_ASHMEM if(module != NULL) *size = module->info.xres * module->info.yres * 2 * 2; else return -ENOMEM; #else *size = 23<<20; //23MB for 7x27 #endif #endif return err; } virtual int connectPmem(int fd, int master_fd) { return ioctl(fd, PMEM_CONNECT, master_fd); } virtual int mapPmem(int fd, int offset, size_t size) { struct pmem_region sub = { offset, size }; return ioctl(fd, PMEM_MAP, &sub); } virtual int unmapPmem(int fd, int offset, size_t size) { struct pmem_region sub = { offset, size }; return ioctl(fd, PMEM_UNMAP, &sub); } virtual int alignPmem(int fd, size_t size, int align) { struct pmem_allocation allocation; allocation.size = size; allocation.align = align; return ioctl(fd, PMEM_ALLOCATE_ALIGNED, &allocation); } virtual int cleanPmem(int fd, unsigned long base, int offset, size_t size) { struct pmem_addr pmem_addr; pmem_addr.vaddr = base; pmem_addr.offset = offset; pmem_addr.length = size; return ioctl(fd, PMEM_CLEAN_INV_CACHES, &pmem_addr); } virtual int getErrno() { return errno; } virtual void* mmap(void* start, size_t length, int prot, int flags, int fd, off_t offset) { return ::mmap(start, length, prot, flags, fd, offset); } virtual int munmap(void* start, size_t length) { return ::munmap(start, length); } virtual int open(const char* pathname, int flags, int mode) { return ::open(pathname, flags, mode); } virtual int close(int fd) { return ::close(fd); } public: void setModule(const private_module_t* m) { module = m; } }; class GpuContextDepsDeviceImpl : public gpu_context_t::Deps { public: virtual int ashmem_create_region(const char *name, size_t size) { return ::ashmem_create_region(name, size); } virtual int mapFrameBufferLocked(struct private_module_t* module) { return ::mapFrameBufferLocked(module); } virtual int terminateBuffer(gralloc_module_t const* module, private_handle_t* hnd) { return ::terminateBuffer(module, hnd); } virtual int close(int fd) { return ::close(fd); } }; static PmemAllocatorDepsDeviceImpl pmemAllocatorDeviceDepsImpl; static GpuContextDepsDeviceImpl gpuContextDeviceDepsImpl; /*****************************************************************************/ static SimpleBestFitAllocator pmemAllocMgr; static PmemUserspaceAllocator pmemAllocator(pmemAllocatorDeviceDepsImpl, pmemAllocMgr, "/dev/pmem"); static PmemKernelAllocator pmemAdspAllocator(pmemAllocatorDeviceDepsImpl); /*****************************************************************************/ static struct hw_module_methods_t gralloc_module_methods = { open: gralloc_device_open }; struct private_module_t HAL_MODULE_INFO_SYM = { base: { common: { tag: HARDWARE_MODULE_TAG, version_major: 1, version_minor: 0, id: GRALLOC_HARDWARE_MODULE_ID, name: "Graphics Memory Allocator Module", author: "The Android Open Source Project", methods: &gralloc_module_methods }, registerBuffer: gralloc_register_buffer, unregisterBuffer: gralloc_unregister_buffer, lock: gralloc_lock, unlock: gralloc_unlock, perform: gralloc_perform, }, framebuffer: 0, fbFormat: 0, flags: 0, numBuffers: 0, bufferMask: 0, lock: PTHREAD_MUTEX_INITIALIZER, currentBuffer: 0, }; /*****************************************************************************/ int gralloc_device_open(const hw_module_t* module, const char* name, hw_device_t** device) { int status = -EINVAL; if (!strcmp(name, GRALLOC_HARDWARE_GPU0)) { const private_module_t* m = reinterpret_cast( module); pmemAllocatorDeviceDepsImpl.setModule(m); gpu_context_t *dev; dev = new gpu_context_t(gpuContextDeviceDepsImpl, pmemAllocator, pmemAdspAllocator, m); *device = &dev->common; status = 0; } else { status = fb_device_open(module, name, device); } return status; }