blob: d09411f5db5daf5e9567786f2e5fa0a803b258f2 [file] [log] [blame]
Saurabh Shah86c17292013-02-08 15:24:13 -08001/*
Arun Kumar K.Rc62935a2013-12-03 16:47:47 -08002 * Copyright (c) 2013-14, The Linux Foundation. All rights reserved.
Saurabh Shah86c17292013-02-08 15:24:13 -08003 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above
10 * copyright notice, this list of conditions and the following
11 * disclaimer in the documentation and/or other materials provided
12 * with the distribution.
13 * * Neither the name of The Linux Foundation nor the names of its
14 * contributors may be used to endorse or promote products derived
15 * from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR CLIENTS; LOSS OF USE, DATA, OR PROFITS; OR
24 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29
30#include <hwc_qclient.h>
31#include <IQService.h>
32#include <hwc_utils.h>
Zohaib Alam1bb65612013-09-28 03:38:20 -040033#include <mdp_version.h>
Naseer Ahmed35a268c2014-06-24 19:07:13 -040034#include <hwc_mdpcomp.h>
Tatenda Chipeperekwa06af9cb2014-08-26 14:51:05 -070035#include <hwc_virtual.h>
Saurabh Shah86c17292013-02-08 15:24:13 -080036
37#define QCLIENT_DEBUG 0
38
39using namespace android;
40using namespace qService;
Arun Kumar K.Rc62935a2013-12-03 16:47:47 -080041using namespace qhwc;
Saurabh Shah86c17292013-02-08 15:24:13 -080042
43namespace qClient {
44
45// ----------------------------------------------------------------------------
Saurabh Shah7128e502013-02-20 13:24:48 -080046QClient::QClient(hwc_context_t *ctx) : mHwcContext(ctx),
47 mMPDeathNotifier(new MPDeathNotifier(ctx))
Saurabh Shah86c17292013-02-08 15:24:13 -080048{
49 ALOGD_IF(QCLIENT_DEBUG, "QClient Constructor invoked");
50}
51
52QClient::~QClient()
53{
54 ALOGD_IF(QCLIENT_DEBUG,"QClient Destructor invoked");
55}
56
Naseer Ahmed4957c522013-11-12 18:07:15 -050057static void securing(hwc_context_t *ctx, uint32_t startEnd) {
58 Locker::Autolock _sl(ctx->mDrawLock);
Saurabh Shahc2125772013-03-15 16:49:50 -070059 //The only way to make this class in this process subscribe to media
60 //player's death.
61 IMediaDeathNotifier::getMediaPlayerService();
62
Naseer Ahmed4957c522013-11-12 18:07:15 -050063 ctx->mSecuring = startEnd;
Saurabh Shah86c17292013-02-08 15:24:13 -080064 //We're done securing
65 if(startEnd == IQService::END)
Naseer Ahmed4957c522013-11-12 18:07:15 -050066 ctx->mSecureMode = true;
67 if(ctx->proc)
68 ctx->proc->invalidate(ctx->proc);
Saurabh Shah86c17292013-02-08 15:24:13 -080069}
70
Naseer Ahmed4957c522013-11-12 18:07:15 -050071static void unsecuring(hwc_context_t *ctx, uint32_t startEnd) {
72 Locker::Autolock _sl(ctx->mDrawLock);
73 ctx->mSecuring = startEnd;
Saurabh Shah86c17292013-02-08 15:24:13 -080074 //We're done unsecuring
75 if(startEnd == IQService::END)
Naseer Ahmed4957c522013-11-12 18:07:15 -050076 ctx->mSecureMode = false;
77 if(ctx->proc)
78 ctx->proc->invalidate(ctx->proc);
Saurabh Shah86c17292013-02-08 15:24:13 -080079}
80
Saurabh Shah7128e502013-02-20 13:24:48 -080081void QClient::MPDeathNotifier::died() {
Saurabh Shahb39f8152013-08-22 10:21:44 -070082 Locker::Autolock _sl(mHwcContext->mDrawLock);
Saurabh Shah7128e502013-02-20 13:24:48 -080083 ALOGD_IF(QCLIENT_DEBUG, "Media Player died");
84 mHwcContext->mSecuring = false;
85 mHwcContext->mSecureMode = false;
86 if(mHwcContext->proc)
87 mHwcContext->proc->invalidate(mHwcContext->proc);
88}
89
Naseer Ahmed4957c522013-11-12 18:07:15 -050090static android::status_t screenRefresh(hwc_context_t *ctx) {
Jeykumar Sankaran9f59a762013-02-28 10:45:56 -080091 status_t result = NO_INIT;
Naseer Ahmed4957c522013-11-12 18:07:15 -050092 if(ctx->proc) {
93 ctx->proc->invalidate(ctx->proc);
Jeykumar Sankaran9f59a762013-02-28 10:45:56 -080094 result = NO_ERROR;
95 }
Jeykumar Sankaran9f59a762013-02-28 10:45:56 -080096 return result;
97}
Arun Kumar K.Rffef7482013-04-10 14:17:22 -070098
Naseer Ahmed4957c522013-11-12 18:07:15 -050099static void setExtOrientation(hwc_context_t *ctx, uint32_t orientation) {
100 ctx->mExtOrientation = orientation;
Arun Kumar K.Rffef7482013-04-10 14:17:22 -0700101}
102
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500103static void isExternalConnected(hwc_context_t* ctx, Parcel* outParcel) {
104 int connected;
105 connected = ctx->dpyAttr[HWC_DISPLAY_EXTERNAL].connected ? 1 : 0;
106 outParcel->writeInt32(connected);
107}
108
109static void getDisplayAttributes(hwc_context_t* ctx, const Parcel* inParcel,
110 Parcel* outParcel) {
111 int dpy = inParcel->readInt32();
112 outParcel->writeInt32(ctx->dpyAttr[dpy].vsync_period);
Dileep Kumar Reddie351d842014-03-25 10:39:21 +0530113 if (ctx->dpyAttr[dpy].customFBSize) {
114 outParcel->writeInt32(ctx->dpyAttr[dpy].xres_new);
115 outParcel->writeInt32(ctx->dpyAttr[dpy].yres_new);
116 } else {
117 outParcel->writeInt32(ctx->dpyAttr[dpy].xres);
118 outParcel->writeInt32(ctx->dpyAttr[dpy].yres);
119 }
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500120 outParcel->writeFloat(ctx->dpyAttr[dpy].xdpi);
121 outParcel->writeFloat(ctx->dpyAttr[dpy].ydpi);
122 //XXX: Need to check what to return for HDMI
123 outParcel->writeInt32(ctx->mMDP.panel);
124}
Arun Kumar K.R2aa44c62014-01-21 23:08:28 -0800125static void setHSIC(const Parcel* inParcel) {
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500126 int dpy = inParcel->readInt32();
Arun Kumar K.R2aa44c62014-01-21 23:08:28 -0800127 ALOGD_IF(0, "In %s: dpy = %d", __FUNCTION__, dpy);
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500128 HSICData_t hsic_data;
129 hsic_data.hue = inParcel->readInt32();
130 hsic_data.saturation = inParcel->readFloat();
131 hsic_data.intensity = inParcel->readInt32();
132 hsic_data.contrast = inParcel->readFloat();
133 //XXX: Actually set the HSIC data through ABL lib
134}
135
136
Naseer Ahmed4957c522013-11-12 18:07:15 -0500137static void setBufferMirrorMode(hwc_context_t *ctx, uint32_t enable) {
138 ctx->mBufferMirrorMode = enable;
Arun Kumar K.Rfb5bfa62013-07-25 03:10:51 -0700139}
140
Arun Kumar K.R8e7a62f2013-12-06 18:55:41 -0800141static status_t getDisplayVisibleRegion(hwc_context_t* ctx, int dpy,
142 Parcel* outParcel) {
143 // Get the info only if the dpy is valid
144 if(dpy >= HWC_DISPLAY_PRIMARY && dpy <= HWC_DISPLAY_VIRTUAL) {
145 Locker::Autolock _sl(ctx->mDrawLock);
146 if(dpy && (ctx->mExtOrientation || ctx->mBufferMirrorMode)) {
147 // Return the destRect on external, if external orienation
148 // is enabled
149 outParcel->writeInt32(ctx->dpyAttr[dpy].mDstRect.left);
150 outParcel->writeInt32(ctx->dpyAttr[dpy].mDstRect.top);
151 outParcel->writeInt32(ctx->dpyAttr[dpy].mDstRect.right);
152 outParcel->writeInt32(ctx->dpyAttr[dpy].mDstRect.bottom);
153 } else {
154 outParcel->writeInt32(ctx->mViewFrame[dpy].left);
155 outParcel->writeInt32(ctx->mViewFrame[dpy].top);
156 outParcel->writeInt32(ctx->mViewFrame[dpy].right);
157 outParcel->writeInt32(ctx->mViewFrame[dpy].bottom);
158 }
159 return NO_ERROR;
160 } else {
161 ALOGE("In %s: invalid dpy index %d", __FUNCTION__, dpy);
162 return BAD_VALUE;
163 }
164}
165
Arun Kumar K.Rc62935a2013-12-03 16:47:47 -0800166static void pauseWFD(hwc_context_t *ctx, uint32_t pause) {
Raj kamal59fea562014-04-01 16:52:19 +0530167 /* TODO: Will remove pauseWFD once all the clients start using
168 * setWfdStatus to indicate the status of WFD display
169 */
Arun Kumar K.Rc62935a2013-12-03 16:47:47 -0800170 int dpy = HWC_DISPLAY_VIRTUAL;
171 if(pause) {
172 //WFD Pause
173 handle_pause(ctx, dpy);
174 } else {
175 //WFD Resume
176 handle_resume(ctx, dpy);
177 }
178}
179
Raj kamal59fea562014-04-01 16:52:19 +0530180static void setWfdStatus(hwc_context_t *ctx, uint32_t wfdStatus) {
181
182 ALOGD_IF(HWC_WFDDISPSYNC_LOG,
183 "%s: Received a binder call that WFD state is %s",
184 __FUNCTION__,getExternalDisplayState(wfdStatus));
185 int dpy = HWC_DISPLAY_VIRTUAL;
186
187 if(wfdStatus == EXTERNAL_OFFLINE) {
188 ctx->mWfdSyncLock.lock();
189 ctx->mWfdSyncLock.signal();
190 ctx->mWfdSyncLock.unlock();
191 } else if(wfdStatus == EXTERNAL_PAUSE) {
192 handle_pause(ctx, dpy);
193 } else if(wfdStatus == EXTERNAL_RESUME) {
194 handle_resume(ctx, dpy);
195 }
196}
197
Ramkumar Radhakrishnan0a021a82014-05-19 19:53:56 -0700198
199static status_t setViewFrame(hwc_context_t* ctx, const Parcel* inParcel) {
200 int dpy = inParcel->readInt32();
201 if(dpy >= HWC_DISPLAY_PRIMARY && dpy <= HWC_DISPLAY_VIRTUAL) {
202 Locker::Autolock _sl(ctx->mDrawLock);
203 ctx->mViewFrame[dpy].left = inParcel->readInt32();
204 ctx->mViewFrame[dpy].top = inParcel->readInt32();
205 ctx->mViewFrame[dpy].right = inParcel->readInt32();
206 ctx->mViewFrame[dpy].bottom = inParcel->readInt32();
207 ALOGD_IF(QCLIENT_DEBUG, "%s: mViewFrame[%d] = [%d %d %d %d]",
208 __FUNCTION__, dpy,
209 ctx->mViewFrame[dpy].left, ctx->mViewFrame[dpy].top,
210 ctx->mViewFrame[dpy].right, ctx->mViewFrame[dpy].bottom);
211 return NO_ERROR;
212 } else {
213 ALOGE("In %s: invalid dpy index %d", __FUNCTION__, dpy);
214 return BAD_VALUE;
215 }
216}
217
Naseer Ahmed35a268c2014-06-24 19:07:13 -0400218static void toggleDynamicDebug(hwc_context_t* ctx, const Parcel* inParcel) {
219 int debug_type = inParcel->readInt32();
220 bool enable = !!inParcel->readInt32();
221 ALOGD("%s: debug_type: %d enable:%d",
222 __FUNCTION__, debug_type, enable);
223 Locker::Autolock _sl(ctx->mDrawLock);
224 switch (debug_type) {
225 //break is ignored for DEBUG_ALL to toggle all of them at once
226 case IQService::DEBUG_ALL:
227 case IQService::DEBUG_MDPCOMP:
228 qhwc::MDPComp::dynamicDebug(enable);
229 if (debug_type != IQService::DEBUG_ALL)
230 break;
231 case IQService::DEBUG_VSYNC:
232 ctx->vstate.debug = enable;
233 if (debug_type != IQService::DEBUG_ALL)
234 break;
Tatenda Chipeperekwa06af9cb2014-08-26 14:51:05 -0700235 case IQService::DEBUG_VD:
Manoj Kumar AVM9591a5e2014-08-21 22:50:21 -0700236 HWCVirtualVDS::dynamicDebug(enable);
Tatenda Chipeperekwa06af9cb2014-08-26 14:51:05 -0700237 if (debug_type != IQService::DEBUG_ALL)
238 break;
Naseer Ahmed35a268c2014-06-24 19:07:13 -0400239 }
240}
241
Naseer Ahmed4957c522013-11-12 18:07:15 -0500242status_t QClient::notifyCallback(uint32_t command, const Parcel* inParcel,
243 Parcel* outParcel) {
Arun Kumar K.R8e7a62f2013-12-06 18:55:41 -0800244 status_t ret = NO_ERROR;
Naseer Ahmed4957c522013-11-12 18:07:15 -0500245
Naseer Ahmed4957c522013-11-12 18:07:15 -0500246 switch(command) {
247 case IQService::SECURING:
248 securing(mHwcContext, inParcel->readInt32());
249 break;
250 case IQService::UNSECURING:
251 unsecuring(mHwcContext, inParcel->readInt32());
252 break;
253 case IQService::SCREEN_REFRESH:
254 return screenRefresh(mHwcContext);
255 break;
256 case IQService::EXTERNAL_ORIENTATION:
257 setExtOrientation(mHwcContext, inParcel->readInt32());
258 break;
259 case IQService::BUFFER_MIRRORMODE:
260 setBufferMirrorMode(mHwcContext, inParcel->readInt32());
261 break;
Arun Kumar K.R8e7a62f2013-12-06 18:55:41 -0800262 case IQService::GET_DISPLAY_VISIBLE_REGION:
263 ret = getDisplayVisibleRegion(mHwcContext, inParcel->readInt32(),
264 outParcel);
265 break;
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500266 case IQService::CHECK_EXTERNAL_STATUS:
267 isExternalConnected(mHwcContext, outParcel);
268 break;
269 case IQService::GET_DISPLAY_ATTRIBUTES:
270 getDisplayAttributes(mHwcContext, inParcel, outParcel);
271 break;
272 case IQService::SET_HSIC_DATA:
Arun Kumar K.R2aa44c62014-01-21 23:08:28 -0800273 setHSIC(inParcel);
Raj kamal59fea562014-04-01 16:52:19 +0530274 break;
Arun Kumar K.Rc62935a2013-12-03 16:47:47 -0800275 case IQService::PAUSE_WFD:
276 pauseWFD(mHwcContext, inParcel->readInt32());
Naseer Ahmed78c952e2013-11-25 18:12:23 -0500277 break;
Raj kamal59fea562014-04-01 16:52:19 +0530278 case IQService::SET_WFD_STATUS:
279 setWfdStatus(mHwcContext,inParcel->readInt32());
280 break;
Ramkumar Radhakrishnan0a021a82014-05-19 19:53:56 -0700281 case IQService::SET_VIEW_FRAME:
282 setViewFrame(mHwcContext, inParcel);
283 break;
Naseer Ahmed35a268c2014-06-24 19:07:13 -0400284 case IQService::DYNAMIC_DEBUG:
285 toggleDynamicDebug(mHwcContext, inParcel);
286 break;
Naseer Ahmed4957c522013-11-12 18:07:15 -0500287 default:
Arun Kumar K.R8e7a62f2013-12-06 18:55:41 -0800288 ret = NO_ERROR;
Naseer Ahmed4957c522013-11-12 18:07:15 -0500289 }
Arun Kumar K.R8e7a62f2013-12-06 18:55:41 -0800290 return ret;
Naseer Ahmed4957c522013-11-12 18:07:15 -0500291}
292
Saurabh Shah86c17292013-02-08 15:24:13 -0800293}