/* // 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 <fcntl.h> #include <errno.h> #include <common/utils/HwcTrace.h> #include <IDisplayDevice.h> #include <DrmConfig.h> #include <common/base/Drm.h> #include <Hwcomposer.h> namespace android { namespace intel { Drm::Drm() : mDrmFd(0), mLock(), mInitialized(false) { memset(&mOutputs, 0, sizeof(mOutputs)); } Drm::~Drm() { WARN_IF_NOT_DEINIT(); } bool Drm::initialize() { if (mInitialized) { WLOGTRACE("Drm object has been initialized"); return true; } const char *path = DrmConfig::getDrmPath(); mDrmFd = open(path, O_RDWR, 0); if (mDrmFd < 0) { ELOGTRACE("failed to open Drm, error: %s", strerror(errno)); return false; } DLOGTRACE("mDrmFd = %d", mDrmFd); memset(&mOutputs, 0, sizeof(mOutputs)); mInitialized = true; return true; } void Drm::deinitialize() { for (int i = 0; i < OUTPUT_MAX; i++) { resetOutput(i); } if (mDrmFd) { close(mDrmFd); mDrmFd = 0; } mInitialized = false; } bool Drm::detect(int device) { RETURN_FALSE_IF_NOT_INIT(); Mutex::Autolock _l(mLock); int outputIndex = getOutputIndex(device); if (outputIndex < 0 ) { return false; } resetOutput(outputIndex); // get drm resources drmModeResPtr resources = drmModeGetResources(mDrmFd); if (!resources) { ELOGTRACE("fail to get drm resources, error: %s", strerror(errno)); return false; } drmModeConnectorPtr connector = NULL; DrmOutput *output = &mOutputs[outputIndex]; bool ret = false; // find connector for the given device for (int i = 0; i < resources->count_connectors; i++) { if (!resources->connectors || !resources->connectors[i]) { ELOGTRACE("fail to get drm resources connectors, error: %s", strerror(errno)); continue; } connector = drmModeGetConnector(mDrmFd, resources->connectors[i]); if (!connector) { ELOGTRACE("drmModeGetConnector failed"); continue; } if (connector->connector_type != DrmConfig::getDrmConnector(device)) { drmModeFreeConnector(connector); continue; } if (connector->connection != DRM_MODE_CONNECTED) { ILOGTRACE("device %d is not connected", device); drmModeFreeConnector(connector); ret = true; break; } output->connector = connector; output->connected = true; // get proper encoder for the given connector if (connector->encoder_id) { ILOGTRACE("Drm connector has encoder attached on device %d", device); output->encoder = drmModeGetEncoder(mDrmFd, connector->encoder_id); if (!output->encoder) { ELOGTRACE("failed to get encoder from a known encoder id"); // fall through to get an encoder } } if (!output->encoder) { ILOGTRACE("getting encoder for device %d", device); drmModeEncoderPtr encoder; for (int j = 0; j < resources->count_encoders; j++) { if (!resources->encoders || !resources->encoders[j]) { ELOGTRACE("fail to get drm resources encoders, error: %s", strerror(errno)); continue; } encoder = drmModeGetEncoder(mDrmFd, resources->encoders[i]); if (!encoder) { ELOGTRACE("drmModeGetEncoder failed"); continue; } if (encoder->encoder_type == DrmConfig::getDrmEncoder(device)) { output->encoder = encoder; break; } drmModeFreeEncoder(encoder); encoder = NULL; } } if (!output->encoder) { ELOGTRACE("failed to get drm encoder"); break; } // get an attached crtc or spare crtc if (output->encoder->crtc_id) { ILOGTRACE("Drm encoder has crtc attached on device %d", device); output->crtc = drmModeGetCrtc(mDrmFd, output->encoder->crtc_id); if (!output->crtc) { ELOGTRACE("failed to get crtc from a known crtc id"); // fall through to get a spare crtc } } if (!output->crtc) { ILOGTRACE("getting crtc for device %d", device); drmModeCrtcPtr crtc; for (int j = 0; j < resources->count_crtcs; j++) { if (!resources->crtcs || !resources->crtcs[j]) { ELOGTRACE("fail to get drm resources crtcs, error: %s", strerror(errno)); continue; } crtc = drmModeGetCrtc(mDrmFd, resources->crtcs[j]); if (!crtc) { ELOGTRACE("drmModeGetCrtc failed"); continue; } // check if legal crtc to the encoder if (output->encoder->possible_crtcs & (1<<j)) { if (crtc->buffer_id == 0) { output->crtc = crtc; break; } } drmModeFreeCrtc(crtc); } } if (!output->crtc) { ELOGTRACE("failed to get drm crtc"); break; } // current mode if (output->crtc->mode_valid) { ILOGTRACE("mode is valid, kernel mode settings"); memcpy(&output->mode, &output->crtc->mode, sizeof(drmModeModeInfo)); //output->fbId = output->crtc->buffer_id; ret = true; } else { ELOGTRACE("mode is invalid. Kernel mode setting is not completed"); ret = false; } if (outputIndex == OUTPUT_PRIMARY) { if (!readIoctl(DRM_PSB_PANEL_ORIENTATION, &output->panelOrientation, sizeof(int))) { ELOGTRACE("failed to get device %d orientation", device); output->panelOrientation = PANEL_ORIENTATION_0; } } else { output->panelOrientation = PANEL_ORIENTATION_0; } break; } if (!ret) { if (output->connector == NULL && outputIndex != OUTPUT_PRIMARY) { // a fatal failure on primary device // non fatal on secondary device WLOGTRACE("device %d is disabled?", device); ret = true; } resetOutput(outputIndex); } else if (output->connected) { ILOGTRACE("mode is: %dx%d@%dHz", output->mode.hdisplay, output->mode.vdisplay, output->mode.vrefresh); } drmModeFreeResources(resources); return ret; } bool Drm::isSameDrmMode(drmModeModeInfoPtr value, drmModeModeInfoPtr base) const { if (base->hdisplay == value->hdisplay && base->vdisplay == value->vdisplay && base->vrefresh == value->vrefresh && (base->flags & value->flags) == value->flags) { VLOGTRACE("Drm mode is not changed"); return true; } return false; } bool Drm::setDrmMode(int device, drmModeModeInfo& value) { RETURN_FALSE_IF_NOT_INIT(); Mutex::Autolock _l(mLock); if (device != IDisplayDevice::DEVICE_EXTERNAL) { WLOGTRACE("Setting mode on invalid device %d", device); return false; } int outputIndex = getOutputIndex(device); if (outputIndex < 0 ) { ELOGTRACE("invalid device"); return false; } DrmOutput *output= &mOutputs[outputIndex]; if (!output->connected) { ELOGTRACE("device is not connected"); return false; } if (output->connector->count_modes <= 0) { ELOGTRACE("invalid count of modes"); return false; } drmModeModeInfoPtr mode; int index = 0; for (int i = 0; i < output->connector->count_modes; i++) { mode = &output->connector->modes[i]; if (mode->type & DRM_MODE_TYPE_PREFERRED) { index = i; } if (isSameDrmMode(&value, mode)) { index = i; break; } } mode = &output->connector->modes[index]; return setDrmMode(outputIndex, mode); } bool Drm::setRefreshRate(int device, int hz) { RETURN_FALSE_IF_NOT_INIT(); Mutex::Autolock _l(mLock); if (device != IDisplayDevice::DEVICE_EXTERNAL) { WLOGTRACE("Setting mode on invalid device %d", device); return false; } int outputIndex = getOutputIndex(device); if (outputIndex < 0 ) { ELOGTRACE("invalid device"); return false; } DrmOutput *output= &mOutputs[outputIndex]; if (!output->connected) { ELOGTRACE("device is not connected"); return false; } if (output->connector->count_modes <= 0) { ELOGTRACE("invalid count of modes"); return false; } drmModeModeInfoPtr mode; int index = 0; for (int i = 0; i < output->connector->count_modes; i++) { mode = &output->connector->modes[i]; if (mode->type & DRM_MODE_TYPE_PREFERRED) { index = i; } if (mode->hdisplay == output->mode.hdisplay && mode->vdisplay == output->mode.vdisplay && mode->vrefresh == (uint32_t)hz) { index = i; break; } } mode = &output->connector->modes[index]; return setDrmMode(outputIndex, mode); } bool Drm::writeReadIoctl(unsigned long cmd, void *data, unsigned long size) { int err; if (mDrmFd <= 0) { ELOGTRACE("drm is not initialized"); return false; } if (!data || !size) { ELOGTRACE("invalid parameters"); return false; } err = drmCommandWriteRead(mDrmFd, cmd, data, size); if (err) { WLOGTRACE("failed to call %ld ioctl with failure %d", cmd, err); return false; } return true; } bool Drm::writeIoctl(unsigned long cmd, void *data, unsigned long size) { int err; if (mDrmFd <= 0) { ELOGTRACE("drm is not initialized"); return false; } if (!data || !size) { ELOGTRACE("invalid parameters"); return false; } err = drmCommandWrite(mDrmFd, cmd, data, size); if (err) { WLOGTRACE("failed to call %ld ioctl with failure %d", cmd, err); return false; } return true; } bool Drm::readIoctl(unsigned long cmd, void *data, unsigned long size) { int err; if (mDrmFd <= 0) { ELOGTRACE("drm is not initialized"); return false; } if (!data || !size) { ELOGTRACE("invalid parameters"); return false; } err = drmCommandRead(mDrmFd, cmd, data, size); if (err) { WLOGTRACE("failed to call %ld ioctl with failure %d", cmd, err); return false; } return true; } int Drm::getDrmFd() const { return mDrmFd; } bool Drm::getModeInfo(int device, drmModeModeInfo& mode) { Mutex::Autolock _l(mLock); int outputIndex = getOutputIndex(device); if (outputIndex < 0 ) { return false; } DrmOutput *output= &mOutputs[outputIndex]; if (output->connected == false) { ELOGTRACE("device is not connected"); return false; } if (output->mode.hdisplay == 0 || output->mode.vdisplay == 0) { ELOGTRACE("invalid width or height"); return false; } memcpy(&mode, &output->mode, sizeof(drmModeModeInfo)); #ifdef INTEL_SUPPORT_HDMI_PRIMARY // FIXME: use default fb size instead of hdmi mode, because to // support hdmi primary, we cannot report dynamic mode to SF. mode.hdisplay = DEFAULT_DRM_FB_WIDTH; mode.vdisplay = DEFAULT_DRM_FB_HEIGHT; #endif return true; } bool Drm::getPhysicalSize(int device, uint32_t& width, uint32_t& height) { Mutex::Autolock _l(mLock); int outputIndex = getOutputIndex(device); if (outputIndex < 0 ) { return false; } DrmOutput *output= &mOutputs[outputIndex]; if (output->connected == false) { ELOGTRACE("device is not connected"); return false; } width = output->connector->mmWidth; height = output->connector->mmHeight; return true; } bool Drm::getDisplayResolution(int device, uint32_t& width, uint32_t& height) { Mutex::Autolock _l(mLock); int outputIndex = getOutputIndex(device); if (outputIndex < 0) { return false; } DrmOutput *output= &mOutputs[outputIndex]; if (output->connected == false) { ELOGTRACE("device is not connected"); return false; } width = output->mode.hdisplay; height = output->mode.vdisplay; if (!width || !height) { ELOGTRACE("invalid width or height"); return false; } return true; } bool Drm::isConnected(int device) { Mutex::Autolock _l(mLock); int output = getOutputIndex(device); if (output < 0 ) { return false; } return mOutputs[output].connected; } bool Drm::setDpmsMode(int device, int mode) { Mutex::Autolock _l(mLock); int output = getOutputIndex(device); if (output < 0 ) { return false; } if (mode != IDisplayDevice::DEVICE_DISPLAY_OFF && mode != IDisplayDevice::DEVICE_DISPLAY_STANDBY && mode != IDisplayDevice::DEVICE_DISPLAY_ON) { ELOGTRACE("invalid mode %d", mode); return false; } DrmOutput *out = &mOutputs[output]; if (!out->connected) { ELOGTRACE("device is not connected"); return false; } drmModePropertyPtr props; for (int i = 0; i < out->connector->count_props; i++) { props = drmModeGetProperty(mDrmFd, out->connector->props[i]); if (!props) { continue; } if (strcmp(props->name, "DPMS") == 0) { int ret = drmModeConnectorSetProperty( mDrmFd, out->connector->connector_id, props->prop_id, (mode == IDisplayDevice::DEVICE_DISPLAY_ON) ? DRM_MODE_DPMS_ON : IDisplayDevice::DEVICE_DISPLAY_STANDBY == mode ? DRM_MODE_DPMS_STANDBY : DRM_MODE_DPMS_OFF); drmModeFreeProperty(props); if (ret != 0) { ELOGTRACE("unable to set DPMS %d", mode); return false; } else { return true; } } drmModeFreeProperty(props); } return false; } void Drm::resetOutput(int index) { DrmOutput *output = &mOutputs[index]; output->connected = false; memset(&output->mode, 0, sizeof(drmModeModeInfo)); if (output->connector) { drmModeFreeConnector(output->connector); output->connector = 0; } if (output->encoder) { drmModeFreeEncoder(output->encoder); output->encoder = 0; } if (output->crtc) { drmModeFreeCrtc(output->crtc); output->crtc = 0; } if (output->fbId) { drmModeRmFB(mDrmFd, output->fbId); output->fbId = 0; } if (output->fbHandle) { Hwcomposer::getInstance().getBufferManager()->freeFrameBuffer(output->fbHandle); output->fbHandle = 0; } } bool Drm::initDrmMode(int outputIndex) { DrmOutput *output= &mOutputs[outputIndex]; if (output->connector->count_modes <= 0) { ELOGTRACE("invalid count of modes"); return false; } drmModeModeInfoPtr mode; int index = 0; for (int i = 0; i < output->connector->count_modes; i++) { mode = &output->connector->modes[i]; if (mode->type & DRM_MODE_TYPE_PREFERRED) { index = i; break; } } return setDrmMode(outputIndex, &output->connector->modes[index]); } bool Drm::setDrmMode(int index, drmModeModeInfoPtr mode) { DrmOutput *output = &mOutputs[index]; int oldFbId = 0; int oldFbHandle = 0; // reuse current frame buffer if there is no resolution change int fbId = -1; drmModeModeInfo currentMode; memcpy(¤tMode, &output->mode, sizeof(drmModeModeInfo)); if (isSameDrmMode(mode, ¤tMode)) return true; if (currentMode.hdisplay != mode->hdisplay || currentMode.vdisplay != mode->vdisplay) { oldFbId = output->fbId; oldFbHandle = output->fbHandle; // allocate frame buffer int stride = 0; #ifdef INTEL_SUPPORT_HDMI_PRIMARY output->fbHandle = Hwcomposer::getInstance().getBufferManager()->allocFrameBuffer( DEFAULT_DRM_FB_WIDTH, DEFAULT_DRM_FB_HEIGHT, &stride); #else output->fbHandle = Hwcomposer::getInstance().getBufferManager()->allocFrameBuffer( mode->hdisplay, mode->vdisplay, &stride); #endif if (output->fbHandle == 0) { ELOGTRACE("failed to allocate frame buffer"); return false; } int ret = 0; ret = drmModeAddFB( mDrmFd, #ifdef INTEL_SUPPORT_HDMI_PRIMARY DEFAULT_DRM_FB_WIDTH, DEFAULT_DRM_FB_HEIGHT, #else mode->hdisplay, mode->vdisplay, #endif DrmConfig::getFrameBufferDepth(), DrmConfig::getFrameBufferBpp(), stride, output->fbHandle, &output->fbId); if (ret != 0) { ELOGTRACE("drmModeAddFB failed, error: %d", ret); return false; } fbId = output->fbId; } ILOGTRACE("mode set: %dx%d@%dHz", mode->hdisplay, mode->vdisplay, mode->vrefresh); int ret = drmModeSetCrtc(mDrmFd, output->crtc->crtc_id, fbId, 0, 0, &output->connector->connector_id, 1, mode); if (ret == 0) { //save mode memcpy(&output->mode, mode, sizeof(drmModeModeInfo)); } else { ELOGTRACE("drmModeSetCrtc failed. error: %d", ret); } if (oldFbId) { drmModeRmFB(mDrmFd, oldFbId); } if (oldFbHandle) { Hwcomposer::getInstance().getBufferManager()->freeFrameBuffer(oldFbHandle); } return ret == 0; } int Drm::getOutputIndex(int device) { switch (device) { case IDisplayDevice::DEVICE_PRIMARY: return OUTPUT_PRIMARY; case IDisplayDevice::DEVICE_EXTERNAL: return OUTPUT_EXTERNAL; default: ELOGTRACE("invalid display device"); break; } return -1; } int Drm::getPanelOrientation(int device) { int outputIndex = getOutputIndex(device); if (outputIndex < 0) { ELOGTRACE("invalid device"); return PANEL_ORIENTATION_0; } DrmOutput *output= &mOutputs[outputIndex]; if (output->connected == false) { ELOGTRACE("device is not connected"); return PANEL_ORIENTATION_0; } return output->panelOrientation; } // HWC 1.4 requires that we return all of the compatible configs in getDisplayConfigs // this is needed so getActiveConfig/setActiveConfig work correctly. It is up to the // user space to decide what speed to send. drmModeModeInfoPtr Drm::detectAllConfigs(int device, int *modeCount) { RETURN_NULL_IF_NOT_INIT(); Mutex::Autolock _l(mLock); if (modeCount != NULL) *modeCount = 0; else return NULL; int outputIndex = getOutputIndex(device); if (outputIndex < 0) { ELOGTRACE("invalid device"); return NULL; } DrmOutput *output= &mOutputs[outputIndex]; if (!output->connected) { ELOGTRACE("device is not connected"); return NULL; } if (output->connector->count_modes <= 0) { ELOGTRACE("invalid count of modes"); return NULL; } *modeCount = output->connector->count_modes; return output->connector->modes; } } // namespace intel } // namespace android