/* * Copyright (c) 2015 - 2017, The Linux Foundation. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * Neither the name of The Linux Foundation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include "hwc_debugger.h" #include "hwc_buffer_allocator.h" #define __CLASS__ "HWCBufferAllocator" namespace sdm { HWCBufferAllocator::HWCBufferAllocator() { alloc_controller_ = gralloc::IAllocController::getInstance(); } DisplayError HWCBufferAllocator::AllocateBuffer(BufferInfo *buffer_info) { gralloc::alloc_data data; const BufferConfig &buffer_config = buffer_info->buffer_config; AllocatedBufferInfo *alloc_buffer_info = &buffer_info->alloc_buffer_info; MetaBufferInfo *meta_buffer_info = new MetaBufferInfo(); if (!meta_buffer_info) { return kErrorMemory; } int alloc_flags = INT(GRALLOC_USAGE_PRIVATE_IOMMU_HEAP); int error = 0; int width = INT(buffer_config.width); int height = INT(buffer_config.height); int format; if (buffer_config.secure_camera) { alloc_flags = GRALLOC_USAGE_HW_CAMERA_WRITE; alloc_flags |= (GRALLOC_USAGE_PROTECTED | GRALLOC_USAGE_HW_COMPOSER); data.align = SZ_2M; } else if (buffer_config.secure) { alloc_flags = INT(GRALLOC_USAGE_PRIVATE_MM_HEAP); alloc_flags |= INT(GRALLOC_USAGE_PROTECTED); data.align = SECURE_ALIGN; } else { data.align = UINT32(getpagesize()); } if (buffer_config.cache == false) { // Allocate uncached buffers alloc_flags |= GRALLOC_USAGE_PRIVATE_UNCACHED; } error = SetBufferInfo(buffer_config.format, &format, &alloc_flags); if (error != 0) { delete meta_buffer_info; return kErrorParameters; } int aligned_width = 0, aligned_height = 0; uint32_t buffer_size = getBufferSizeAndDimensions(width, height, format, alloc_flags, aligned_width, aligned_height); buffer_size = ROUND_UP(buffer_size, data.align) * buffer_config.buffer_count; data.base = 0; data.fd = -1; data.offset = 0; data.size = buffer_size; data.uncached = !buffer_config.cache; error = alloc_controller_->allocate(data, alloc_flags); if (error != 0) { DLOGE("Error allocating memory size %d uncached %d", data.size, data.uncached); delete meta_buffer_info; return kErrorMemory; } alloc_buffer_info->fd = data.fd; // TODO(user): define stride for all planes and fix stride in bytes alloc_buffer_info->stride = UINT32(aligned_width); alloc_buffer_info->aligned_width = UINT32(aligned_width); alloc_buffer_info->aligned_height = UINT32(aligned_height); alloc_buffer_info->size = buffer_size; meta_buffer_info->base_addr = data.base; meta_buffer_info->alloc_type = data.allocType; buffer_info->private_data = meta_buffer_info; return kErrorNone; } DisplayError HWCBufferAllocator::FreeBuffer(BufferInfo *buffer_info) { int ret = 0; AllocatedBufferInfo *alloc_buffer_info = &buffer_info->alloc_buffer_info; // Deallocate the buffer, only if the buffer fd is valid. if (alloc_buffer_info->fd > 0) { MetaBufferInfo *meta_buffer_info = static_cast (buffer_info->private_data); gralloc::IMemAlloc *memalloc = alloc_controller_->getAllocator(meta_buffer_info->alloc_type); if (memalloc == NULL) { DLOGE("Memalloc handle is NULL, alloc type %d", meta_buffer_info->alloc_type); return kErrorResources; } ret = memalloc->free_buffer(meta_buffer_info->base_addr, alloc_buffer_info->size, 0, alloc_buffer_info->fd); if (ret != 0) { DLOGE("Error freeing buffer base_addr %p size %d fd %d", meta_buffer_info->base_addr, alloc_buffer_info->size, alloc_buffer_info->fd); return kErrorMemory; } alloc_buffer_info->fd = -1; alloc_buffer_info->stride = 0; alloc_buffer_info->aligned_width = 0; alloc_buffer_info->aligned_height = 0; alloc_buffer_info->size = 0; meta_buffer_info->base_addr = NULL; meta_buffer_info->alloc_type = 0; delete meta_buffer_info; meta_buffer_info = NULL; } return kErrorNone; } uint32_t HWCBufferAllocator::GetBufferSize(BufferInfo *buffer_info) { uint32_t align = UINT32(getpagesize()); const BufferConfig &buffer_config = buffer_info->buffer_config; int alloc_flags = INT(GRALLOC_USAGE_PRIVATE_IOMMU_HEAP); int width = INT(buffer_config.width); int height = INT(buffer_config.height); int format; if (buffer_config.secure_camera) { alloc_flags = GRALLOC_USAGE_HW_CAMERA_WRITE; alloc_flags |= (GRALLOC_USAGE_PROTECTED | GRALLOC_USAGE_HW_COMPOSER); align = SZ_2M; } else if (buffer_config.secure) { alloc_flags = INT(GRALLOC_USAGE_PRIVATE_MM_HEAP); alloc_flags |= INT(GRALLOC_USAGE_PROTECTED); align = SECURE_ALIGN; } if (buffer_config.cache == false) { // Allocate uncached buffers alloc_flags |= GRALLOC_USAGE_PRIVATE_UNCACHED; } if (SetBufferInfo(buffer_config.format, &format, &alloc_flags) < 0) { return 0; } int aligned_width = 0; int aligned_height = 0; uint32_t buffer_size = getBufferSizeAndDimensions(width, height, format, alloc_flags, aligned_width, aligned_height); buffer_size = ROUND_UP(buffer_size, align) * buffer_config.buffer_count; return buffer_size; } int HWCBufferAllocator::SetBufferInfo(LayerBufferFormat format, int *target, int *flags) { switch (format) { case kFormatRGBA8888: *target = HAL_PIXEL_FORMAT_RGBA_8888; break; case kFormatRGBX8888: *target = HAL_PIXEL_FORMAT_RGBX_8888; break; case kFormatRGB888: *target = HAL_PIXEL_FORMAT_RGB_888; break; case kFormatRGB565: *target = HAL_PIXEL_FORMAT_RGB_565; break; case kFormatBGR565: *target = HAL_PIXEL_FORMAT_BGR_565; break; case kFormatBGRA8888: *target = HAL_PIXEL_FORMAT_BGRA_8888; break; case kFormatYCrCb420PlanarStride16: *target = HAL_PIXEL_FORMAT_YV12; break; case kFormatYCrCb420SemiPlanar: *target = HAL_PIXEL_FORMAT_YCrCb_420_SP; break; case kFormatYCbCr420SemiPlanar: *target = HAL_PIXEL_FORMAT_YCbCr_420_SP; break; case kFormatYCbCr422H2V1Packed: *target = HAL_PIXEL_FORMAT_YCbCr_422_I; break; case kFormatCbYCrY422H2V1Packed: *target = HAL_PIXEL_FORMAT_CbYCrY_422_I; break; case kFormatYCbCr422H2V1SemiPlanar: *target = HAL_PIXEL_FORMAT_YCbCr_422_SP; break; case kFormatYCbCr420SemiPlanarVenus: *target = HAL_PIXEL_FORMAT_YCbCr_420_SP_VENUS; break; case kFormatYCrCb420SemiPlanarVenus: *target = HAL_PIXEL_FORMAT_YCrCb_420_SP_VENUS; break; case kFormatYCbCr420SPVenusUbwc: *target = HAL_PIXEL_FORMAT_YCbCr_420_SP_VENUS_UBWC; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatRGBA5551: *target = HAL_PIXEL_FORMAT_RGBA_5551; break; case kFormatRGBA4444: *target = HAL_PIXEL_FORMAT_RGBA_4444; break; case kFormatRGBA1010102: *target = HAL_PIXEL_FORMAT_RGBA_1010102; break; case kFormatARGB2101010: *target = HAL_PIXEL_FORMAT_ARGB_2101010; break; case kFormatRGBX1010102: *target = HAL_PIXEL_FORMAT_RGBX_1010102; break; case kFormatXRGB2101010: *target = HAL_PIXEL_FORMAT_XRGB_2101010; break; case kFormatBGRA1010102: *target = HAL_PIXEL_FORMAT_BGRA_1010102; break; case kFormatABGR2101010: *target = HAL_PIXEL_FORMAT_ABGR_2101010; break; case kFormatBGRX1010102: *target = HAL_PIXEL_FORMAT_BGRX_1010102; break; case kFormatXBGR2101010: *target = HAL_PIXEL_FORMAT_XBGR_2101010; break; case kFormatYCbCr420P010: *target = HAL_PIXEL_FORMAT_YCbCr_420_P010; break; case kFormatYCbCr420TP10Ubwc: *target = HAL_PIXEL_FORMAT_YCbCr_420_TP10_UBWC; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatYCbCr420P010Ubwc: *target = HAL_PIXEL_FORMAT_YCbCr_420_P010_UBWC; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatRGBA8888Ubwc: *target = HAL_PIXEL_FORMAT_RGBA_8888; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatRGBX8888Ubwc: *target = HAL_PIXEL_FORMAT_RGBX_8888; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatBGR565Ubwc: *target = HAL_PIXEL_FORMAT_BGR_565; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatRGBA1010102Ubwc: *target = HAL_PIXEL_FORMAT_RGBA_1010102; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; case kFormatRGBX1010102Ubwc: *target = HAL_PIXEL_FORMAT_RGBX_1010102; *flags |= GRALLOC_USAGE_PRIVATE_ALLOC_UBWC; break; default: DLOGE("Unsupported format = 0x%x", format); return -EINVAL; } return 0; } DisplayError HWCBufferAllocator::GetAllocatedBufferInfo(const BufferConfig &buffer_config, AllocatedBufferInfo *allocated_buffer_info) { int width = INT(buffer_config.width); int height = INT(buffer_config.height); int alloc_flags = INT(GRALLOC_USAGE_PRIVATE_IOMMU_HEAP); if (buffer_config.secure) { alloc_flags = INT(GRALLOC_USAGE_PRIVATE_MM_HEAP); alloc_flags |= INT(GRALLOC_USAGE_PROTECTED); } if (buffer_config.cache == false) { // Allocate uncached buffers alloc_flags |= GRALLOC_USAGE_PRIVATE_UNCACHED; } int format; int error = SetBufferInfo(buffer_config.format, &format, &alloc_flags); if (error) { DLOGE("Failed: format = %d or width = %d height = %d", buffer_config.format, width, height); return kErrorNotSupported; } int width_aligned = 0, height_aligned = 0; uint32_t buffer_size = 0; buffer_size = getBufferSizeAndDimensions(width, height, format, alloc_flags, width_aligned, height_aligned); allocated_buffer_info->stride = UINT32(width_aligned); allocated_buffer_info->aligned_width = UINT32(width_aligned); allocated_buffer_info->aligned_height = UINT32(height_aligned); allocated_buffer_info->size = UINT32(buffer_size); allocated_buffer_info->format = buffer_config.format; return kErrorNone; } DisplayError HWCBufferAllocator::GetBufferLayout(const AllocatedBufferInfo &buf_info, uint32_t stride[4], uint32_t offset[4], uint32_t *num_planes) { private_handle_t hnd(-1, 0, 0, 0, 0, 0, 0); int format = HAL_PIXEL_FORMAT_RGBA_8888; int flags = 0; SetBufferInfo(buf_info.format, &format, &flags); // Setup only the required stuff, skip rest hnd.format = format; hnd.width = buf_info.aligned_width; hnd.height = buf_info.aligned_height; if (flags & GRALLOC_USAGE_PRIVATE_ALLOC_UBWC) { hnd.flags = private_handle_t::PRIV_FLAGS_UBWC_ALIGNED; } int ret = getBufferLayout(&hnd, stride, offset, num_planes); if (ret < 0) { DLOGE("getBufferLayout failed"); return kErrorParameters; } return kErrorNone; } } // namespace sdm