/* * Copyright (C) 2010 The Android Open Source Project * Copyright (C) 2012, The Linux Foundation. All rights reserved. * * Not a Contribution, Apache license notifications and license are * retained for attribution purposes only. * 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. */ #define UEVENT_DEBUG 0 #include #include #include #include #include #include #include "hwc_utils.h" #include "external.h" namespace qhwc { #define HWC_UEVENT_THREAD_NAME "hwcUeventThread" const char* MSMFB_HDMI_NODE = "fb1"; static void handle_uevent(hwc_context_t* ctx, const char* udata, int len) { int vsync = 0; char* hdmi; int64_t timestamp = 0; const char *str = udata; if(!strcasestr(str, "@/devices/virtual/graphics/fb")) { ALOGD_IF(UEVENT_DEBUG, "%s: Not Ext Disp Event ", __FUNCTION__); return; } hdmi = strcasestr(str, MSMFB_HDMI_NODE); // parse HDMI events // The event will be of the form: // change@/devices/virtual/graphics/fb1 ACTION=change // DEVPATH=/devices/virtual/graphics/fb1 // SUBSYSTEM=graphics HDCP_STATE=FAIL MAJOR=29 // for now just parsing onlin/offline info is enough if (hdmi) { str = udata; int connected = -1; //some event other than online and offline .. e.g if(!(strncmp(str,"online@",strlen("online@")))) { //Disabled until SF calls unblank ctx->dpyAttr[HWC_DISPLAY_EXTERNAL].isActive = false; connected = 1; } else if(!(strncmp(str,"offline@",strlen("offline@")))) { //Disabled until SF calls unblank ctx->dpyAttr[HWC_DISPLAY_EXTERNAL].isActive = false; connected = 0; } if(connected != -1) { //either we got online or offline ctx->dpyAttr[HWC_DISPLAY_EXTERNAL].connected = connected; ctx->mExtDisplay->setExternalDisplay(connected); ALOGD("%s sending hotplug: connected = %d", __FUNCTION__,connected); Locker::Autolock _l(ctx->mExtSetLock); //hwc comp could be on //TODO ideally should be done on "connected" not "online" ctx->proc->hotplug(ctx->proc, HWC_DISPLAY_EXTERNAL, connected); } } } static void *uevent_loop(void *param) { int len = 0; static char udata[PAGE_SIZE]; hwc_context_t * ctx = reinterpret_cast(param); char thread_name[64] = HWC_UEVENT_THREAD_NAME; prctl(PR_SET_NAME, (unsigned long) &thread_name, 0, 0, 0); setpriority(PRIO_PROCESS, 0, HAL_PRIORITY_URGENT_DISPLAY); uevent_init(); while(1) { len = uevent_next_event(udata, sizeof(udata) - 2); handle_uevent(ctx, udata, len); } return NULL; } void init_uevent_thread(hwc_context_t* ctx) { pthread_t uevent_thread; int ret; ALOGI("Initializing UEVENT Thread"); ret = pthread_create(&uevent_thread, NULL, uevent_loop, (void*) ctx); if (ret) { ALOGE("%s: failed to create %s: %s", __FUNCTION__, HWC_UEVENT_THREAD_NAME, strerror(ret)); } } }; //namespace