/* // Copyright (c) 2014 Intel Corporation // // 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 <hardware/hardware.h> #include <string.h> #include <stdio.h> #include <fcntl.h> #include <errno.h> #include <HwcTrace.h> #include <Hwcomposer.h> #define GET_HWC_RETURN_X_IF_NULL(X) \ CTRACE(); \ Hwcomposer *hwc = static_cast<Hwcomposer*>(dev); \ do {\ if (!hwc) { \ ETRACE("invalid HWC device."); \ return X; \ } \ } while (0) #define GET_HWC_RETURN_ERROR_IF_NULL() GET_HWC_RETURN_X_IF_NULL(-EINVAL) #define GET_HWC_RETURN_VOID_IF_NULL() GET_HWC_RETURN_X_IF_NULL() namespace android { namespace intel { static int hwc_prepare(struct hwc_composer_device_1 *dev, size_t numDisplays, hwc_display_contents_1_t** displays) { GET_HWC_RETURN_ERROR_IF_NULL(); if (!hwc->prepare(numDisplays, displays)) { ETRACE("failed to prepare"); return -EINVAL; } return 0; } static int hwc_set(struct hwc_composer_device_1 *dev, size_t numDisplays, hwc_display_contents_1_t **displays) { GET_HWC_RETURN_ERROR_IF_NULL(); if (!hwc->commit(numDisplays, displays)) { ETRACE("failed to commit"); return -EINVAL; } return 0; } static void hwc_dump(struct hwc_composer_device_1 *dev, char *buff, int buff_len) { GET_HWC_RETURN_VOID_IF_NULL(); hwc->dump(buff, buff_len, 0); } void hwc_registerProcs(struct hwc_composer_device_1 *dev, hwc_procs_t const *procs) { GET_HWC_RETURN_VOID_IF_NULL(); hwc->registerProcs(procs); } static int hwc_device_close(struct hw_device_t *dev) { CTRACE(); Hwcomposer::releaseInstance(); return 0; } static int hwc_query(struct hwc_composer_device_1 *dev, int what, int* value) { ATRACE("what = %d", what); return -EINVAL; } static int hwc_eventControl(struct hwc_composer_device_1 *dev, int disp, int event, int enabled) { bool ret; GET_HWC_RETURN_ERROR_IF_NULL(); switch (event) { case HWC_EVENT_VSYNC: ret = hwc->vsyncControl(disp, enabled); if (ret == false) { ETRACE("failed to control vsync"); return -EINVAL; } break; default: WTRACE("unsupported event %d", event); break; } return 0; } static int hwc_blank(hwc_composer_device_1_t *dev, int disp, int blank) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->blank(disp, blank); if (ret == false) { ETRACE("failed to blank disp %d, blank %d", disp, blank); return -EINVAL; } return 0; } static int hwc_getDisplayConfigs(hwc_composer_device_1_t *dev, int disp, uint32_t *configs, size_t *numConfigs) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->getDisplayConfigs(disp, configs, numConfigs); if (ret == false) { WTRACE("failed to get configs of disp %d", disp); return -EINVAL; } return 0; } static int hwc_getDisplayAttributes(hwc_composer_device_1_t *dev, int disp, uint32_t config, const uint32_t *attributes, int32_t *values) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->getDisplayAttributes(disp, config, attributes, values); if (ret == false) { WTRACE("failed to get attributes of disp %d", disp); return -EINVAL; } return 0; } static int hwc_compositionComplete(hwc_composer_device_1_t *dev, int disp) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->compositionComplete(disp); if (ret == false) { ETRACE("failed for disp %d", disp); return -EINVAL; } return 0; } static int hwc_setPowerMode(hwc_composer_device_1_t *dev, int disp, int mode) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->setPowerMode(disp, mode); if (ret == false) { WTRACE("failed to set power mode of disp %d", disp); return -EINVAL; } return 0; } static int hwc_getActiveConfig(hwc_composer_device_1_t *dev, int disp) { GET_HWC_RETURN_ERROR_IF_NULL(); int ret = hwc->getActiveConfig(disp); if (ret == -1) { WTRACE("failed to get active config of disp %d", disp); return -EINVAL; } return ret; } static int hwc_setActiveConfig(hwc_composer_device_1_t *dev, int disp, int index) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->setActiveConfig(disp, index); if (ret == false) { WTRACE("failed to set active config of disp %d", disp); return -EINVAL; } return 0; } static int hwc_setCursorPositionAsync(hwc_composer_device_1_t *dev, int disp, int x, int y) { GET_HWC_RETURN_ERROR_IF_NULL(); bool ret = hwc->setCursorPositionAsync(disp, x, y); if (ret == false) { WTRACE("failed to set cursor position of disp %d", disp); return -EINVAL; } return 0; } //------------------------------------------------------------------------------ static int hwc_device_open(const struct hw_module_t* module, const char* name, struct hw_device_t** device) { if (!name) { ETRACE("invalid name."); return -EINVAL; } ATRACE("open device %s", name); if (strcmp(name, HWC_HARDWARE_COMPOSER) != 0) { ETRACE("try to open unknown HWComposer %s", name); return -EINVAL; } Hwcomposer& hwc = Hwcomposer::getInstance(); // initialize our state here if (hwc.initialize() == false) { ETRACE("failed to intialize HWComposer"); Hwcomposer::releaseInstance(); return -EINVAL; } // initialize the procs hwc.hwc_composer_device_1_t::common.tag = HARDWARE_DEVICE_TAG; hwc.hwc_composer_device_1_t::common.module = const_cast<hw_module_t*>(module); hwc.hwc_composer_device_1_t::common.close = hwc_device_close; hwc.hwc_composer_device_1_t::prepare = hwc_prepare; hwc.hwc_composer_device_1_t::set = hwc_set; hwc.hwc_composer_device_1_t::dump = hwc_dump; hwc.hwc_composer_device_1_t::registerProcs = hwc_registerProcs; hwc.hwc_composer_device_1_t::query = hwc_query; hwc.hwc_composer_device_1_t::blank = hwc_blank; hwc.hwc_composer_device_1_t::eventControl = hwc_eventControl; hwc.hwc_composer_device_1_t::getDisplayConfigs = hwc_getDisplayConfigs; hwc.hwc_composer_device_1_t::getDisplayAttributes = hwc_getDisplayAttributes; // This is used to hack FBO switch flush issue in SurfaceFlinger. hwc.hwc_composer_device_1_t::reserved_proc[0] = (void*)hwc_compositionComplete; hwc.hwc_composer_device_1_t::common.version = HWC_DEVICE_API_VERSION_1_4; hwc.hwc_composer_device_1_t::setPowerMode = hwc_setPowerMode; hwc.hwc_composer_device_1_t::getActiveConfig = hwc_getActiveConfig; hwc.hwc_composer_device_1_t::setActiveConfig = hwc_setActiveConfig; // Todo: add hwc_setCursorPositionAsync after supporting patches hwc.hwc_composer_device_1_t::setCursorPositionAsync = NULL; *device = &hwc.hwc_composer_device_1_t::common; return 0; } } // namespace intel } // namespace android static struct hw_module_methods_t hwc_module_methods = { open: android::intel::hwc_device_open }; hwc_module_t HAL_MODULE_INFO_SYM = { common: { tag: HARDWARE_MODULE_TAG, version_major: 1, version_minor: 4, id: HWC_HARDWARE_MODULE_ID, name: "Intel Hardware Composer", author: "Intel", methods: &hwc_module_methods, dso: NULL, reserved: {0}, } };