/* * Copyright (c) 2014 - 2016, 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 "hwc_display_virtual.h" #include "hwc_debugger.h" #define __CLASS__ "HWCDisplayVirtual" namespace sdm { int HWCDisplayVirtual::Create(CoreInterface *core_intf, hwc_procs_t const **hwc_procs, uint32_t primary_width, uint32_t primary_height, hwc_display_contents_1_t *content_list, HWCDisplay **hwc_display) { int status = 0; HWCDisplayVirtual *hwc_display_virtual = new HWCDisplayVirtual(core_intf, hwc_procs); uint32_t virtual_width = 0, virtual_height = 0; status = hwc_display_virtual->Init(); if (status) { delete hwc_display_virtual; return status; } status = hwc_display_virtual->SetPowerMode(HWC_POWER_MODE_NORMAL); if (status) { Destroy(hwc_display_virtual); return status; } // TODO(user): Need to update resolution(and not aligned resolution) on writeback. status = hwc_display_virtual->SetOutputSliceFromMetadata(content_list); if (status) { Destroy(hwc_display_virtual); return status; } hwc_display_virtual->GetPanelResolution(&virtual_width, &virtual_height); if (content_list->numHwLayers < 1) { Destroy(hwc_display_virtual); return -1; } hwc_layer_1_t &fb_layer = content_list->hwLayers[content_list->numHwLayers-1]; int fb_width = fb_layer.displayFrame.right - fb_layer.displayFrame.left; int fb_height = fb_layer.displayFrame.bottom - fb_layer.displayFrame.top; status = hwc_display_virtual->SetFrameBufferResolution(UINT32(fb_width), UINT32(fb_height)); if (status) { Destroy(hwc_display_virtual); return status; } *hwc_display = static_cast(hwc_display_virtual); return 0; } void HWCDisplayVirtual::Destroy(HWCDisplay *hwc_display) { hwc_display->Deinit(); delete hwc_display; } HWCDisplayVirtual::HWCDisplayVirtual(CoreInterface *core_intf, hwc_procs_t const **hwc_procs) : HWCDisplay(core_intf, hwc_procs, kVirtual, HWC_DISPLAY_VIRTUAL, false), dump_output_layer_(false), output_buffer_(NULL) { } int HWCDisplayVirtual::Init() { output_buffer_ = new LayerBuffer(); if (!output_buffer_) { return -ENOMEM; } return HWCDisplay::Init(); } int HWCDisplayVirtual::Deinit() { int status = 0; status = HWCDisplay::Deinit(); if (status) { return status; } if (output_buffer_) { delete output_buffer_; } return status; } int HWCDisplayVirtual::Prepare(hwc_display_contents_1_t *content_list) { int status = 0; status = SetOutputSliceFromMetadata(content_list); if (status) { return status; } if (display_paused_) { MarkLayersForGPUBypass(content_list); return status; } status = AllocateLayerStack(content_list); if (status) { return status; } status = SetOutputBuffer(content_list); if (status) { return status; } status = PrePrepareLayerStack(content_list); if (status) { return status; } status = PrepareLayerStack(content_list); if (status) { return status; } return 0; } int HWCDisplayVirtual::Commit(hwc_display_contents_1_t *content_list) { int status = 0; if (display_paused_) { DisplayError error = display_intf_->Flush(); if (error != kErrorNone) { DLOGE("Flush failed. Error = %d", error); } return status; } status = HWCDisplay::CommitLayerStack(content_list); if (status) { return status; } DumpOutputBuffer(content_list); status = HWCDisplay::PostCommitLayerStack(content_list); if (status) { return status; } return 0; } int HWCDisplayVirtual::SetOutputSliceFromMetadata(hwc_display_contents_1_t *content_list) { const private_handle_t *output_handle = static_cast(content_list->outbuf); DisplayError error = kErrorNone; int status = 0; if (output_handle) { int output_handle_format = output_handle->format; if (output_handle_format == HAL_PIXEL_FORMAT_RGBA_8888) { output_handle_format = HAL_PIXEL_FORMAT_RGBX_8888; } LayerBufferFormat format = GetSDMFormat(output_handle_format, output_handle->flags); if (format == kFormatInvalid) { return -EINVAL; } int active_width; int active_height; AdrenoMemInfo::getInstance().getAlignedWidthAndHeight(output_handle, active_width, active_height); if ((active_width != INT(output_buffer_->width)) || (active_height!= INT(output_buffer_->height)) || (format != output_buffer_->format)) { // Populate virtual display attributes based on displayFrame of FBT. // For DRC, use width and height populated in metadata (unaligned values) // for setting attributes of virtual display. This is needed because if // we use aligned width and height, scaling will be required for FBT layer. DisplayConfigVariableInfo variable_info; hwc_layer_1_t &fbt_layer = content_list->hwLayers[content_list->numHwLayers-1]; hwc_rect_t &frame = fbt_layer.displayFrame; int fbt_width = frame.right - frame.left; int fbt_height = frame.bottom - frame.top; const MetaData_t *meta_data = reinterpret_cast(output_handle->base_metadata); if (meta_data && meta_data->operation & UPDATE_BUFFER_GEOMETRY) { variable_info.x_pixels = UINT32(meta_data->bufferDim.sliceWidth); variable_info.y_pixels = UINT32(meta_data->bufferDim.sliceHeight); } else { variable_info.x_pixels = UINT32(fbt_width); variable_info.y_pixels = UINT32(fbt_height); } // TODO(user): Need to get the framerate of primary display and update it. variable_info.fps = 60; error = display_intf_->SetActiveConfig(&variable_info); if (error != kErrorNone) { return -EINVAL; } status = SetOutputBuffer(content_list); if (status) { return status; } } } return 0; } int HWCDisplayVirtual::SetOutputBuffer(hwc_display_contents_1_t *content_list) { int status = 0; const private_handle_t *output_handle = static_cast(content_list->outbuf); // Fill output buffer parameters (width, height, format, plane information, fence) output_buffer_->acquire_fence_fd = content_list->outbufAcquireFenceFd; if (output_handle) { int output_handle_format = output_handle->format; if (output_handle_format == HAL_PIXEL_FORMAT_RGBA_8888) { output_handle_format = HAL_PIXEL_FORMAT_RGBX_8888; } output_buffer_->format = GetSDMFormat(output_handle_format, output_handle->flags); if (output_buffer_->format == kFormatInvalid) { return -EINVAL; } int output_buffer_width, output_buffer_height; AdrenoMemInfo::getInstance().getAlignedWidthAndHeight(output_handle, output_buffer_width, output_buffer_height); output_buffer_->width = UINT32(output_buffer_width); output_buffer_->height = UINT32(output_buffer_height); output_buffer_->flags.secure = 0; output_buffer_->flags.video = 0; // TZ Protected Buffer - L1 if (output_handle->flags & private_handle_t::PRIV_FLAGS_SECURE_BUFFER) { output_buffer_->flags.secure = 1; } // ToDo: Need to extend for non-RGB formats output_buffer_->planes[0].fd = output_handle->fd; output_buffer_->planes[0].offset = output_handle->offset; output_buffer_->planes[0].stride = UINT32(output_handle->width); } layer_stack_.output_buffer = output_buffer_; return status; } void HWCDisplayVirtual::DumpOutputBuffer(hwc_display_contents_1_t *content_list) { const private_handle_t *output_handle = (const private_handle_t *)(content_list->outbuf); char dir_path[PATH_MAX]; if (!dump_frame_count_ || flush_ || !dump_output_layer_) { return; } snprintf(dir_path, sizeof(dir_path), "/data/misc/display/frame_dump_%s", GetDisplayString()); if (mkdir(dir_path, 777) != 0 && errno != EEXIST) { DLOGW("Failed to create %s directory errno = %d, desc = %s", dir_path, errno, strerror(errno)); return; } // if directory exists already, need to explicitly change the permission. if (errno == EEXIST && chmod(dir_path, 0777) != 0) { DLOGW("Failed to change permissions on %s directory", dir_path); return; } if (output_handle && output_handle->base) { char dump_file_name[PATH_MAX]; size_t result = 0; if (content_list->outbufAcquireFenceFd >= 0) { int error = sync_wait(content_list->outbufAcquireFenceFd, 1000); if (error < 0) { DLOGW("sync_wait error errno = %d, desc = %s", errno, strerror(errno)); return; } } snprintf(dump_file_name, sizeof(dump_file_name), "%s/output_layer_%dx%d_%s_frame%d.raw", dir_path, output_handle->width, output_handle->height, GetHALPixelFormatString(output_handle->format), dump_frame_index_); FILE* fp = fopen(dump_file_name, "w+"); if (fp) { result = fwrite(reinterpret_cast(output_handle->base), output_handle->size, 1, fp); fclose(fp); } DLOGI("Frame Dump of %s is %s", dump_file_name, result ? "Successful" : "Failed"); } } void HWCDisplayVirtual::SetFrameDumpConfig(uint32_t count, uint32_t bit_mask_layer_type) { HWCDisplay::SetFrameDumpConfig(count, bit_mask_layer_type); dump_output_layer_ = ((bit_mask_layer_type & (1 << OUTPUT_LAYER_DUMP)) != 0); DLOGI("output_layer_dump_enable %d", dump_output_layer_); } } // namespace sdm