blob: 4b7f3a22315401be8ed2294e8fd31cda956e3498 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2007 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17// tag as surfaceflinger
18#define LOG_TAG "SurfaceFlinger"
19
20#include <stdint.h>
21#include <sys/types.h>
22
Mathias Agopian07952722009-05-19 19:08:10 -070023#include <binder/Parcel.h>
24#include <binder/IMemory.h>
25#include <binder/IPCThreadState.h>
26#include <binder/IServiceManager.h>
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080027
28#include <ui/ISurfaceComposer.h>
29#include <ui/DisplayInfo.h>
30
31// ---------------------------------------------------------------------------
32
33#define LIKELY( exp ) (__builtin_expect( (exp) != 0, true ))
34#define UNLIKELY( exp ) (__builtin_expect( (exp) != 0, false ))
35
36// ---------------------------------------------------------------------------
37
38namespace android {
39
40class BpSurfaceComposer : public BpInterface<ISurfaceComposer>
41{
42public:
43 BpSurfaceComposer(const sp<IBinder>& impl)
44 : BpInterface<ISurfaceComposer>(impl)
45 {
46 }
47
48 virtual sp<ISurfaceFlingerClient> createConnection()
49 {
50 uint32_t n;
51 Parcel data, reply;
52 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
53 remote()->transact(BnSurfaceComposer::CREATE_CONNECTION, data, &reply);
54 return interface_cast<ISurfaceFlingerClient>(reply.readStrongBinder());
55 }
56
57 virtual sp<IMemory> getCblk() const
58 {
59 Parcel data, reply;
60 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
61 remote()->transact(BnSurfaceComposer::GET_CBLK, data, &reply);
62 return interface_cast<IMemory>(reply.readStrongBinder());
63 }
64
65 virtual void openGlobalTransaction()
66 {
67 Parcel data, reply;
68 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
69 remote()->transact(BnSurfaceComposer::OPEN_GLOBAL_TRANSACTION, data, &reply);
70 }
71
72 virtual void closeGlobalTransaction()
73 {
74 Parcel data, reply;
75 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
76 remote()->transact(BnSurfaceComposer::CLOSE_GLOBAL_TRANSACTION, data, &reply);
77 }
78
79 virtual status_t freezeDisplay(DisplayID dpy, uint32_t flags)
80 {
81 Parcel data, reply;
82 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
83 data.writeInt32(dpy);
84 data.writeInt32(flags);
85 remote()->transact(BnSurfaceComposer::FREEZE_DISPLAY, data, &reply);
86 return reply.readInt32();
87 }
88
89 virtual status_t unfreezeDisplay(DisplayID dpy, uint32_t flags)
90 {
91 Parcel data, reply;
92 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
93 data.writeInt32(dpy);
94 data.writeInt32(flags);
95 remote()->transact(BnSurfaceComposer::UNFREEZE_DISPLAY, data, &reply);
96 return reply.readInt32();
97 }
98
Mathias Agopianeb0c86e2009-03-27 18:11:38 -070099 virtual int setOrientation(DisplayID dpy, int orientation, uint32_t flags)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800100 {
101 Parcel data, reply;
102 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
103 data.writeInt32(dpy);
104 data.writeInt32(orientation);
Mathias Agopianeb0c86e2009-03-27 18:11:38 -0700105 data.writeInt32(flags);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800106 remote()->transact(BnSurfaceComposer::SET_ORIENTATION, data, &reply);
107 return reply.readInt32();
108 }
109
110 virtual void bootFinished()
111 {
112 Parcel data, reply;
113 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
114 remote()->transact(BnSurfaceComposer::BOOT_FINISHED, data, &reply);
115 }
116
117 virtual status_t requestGPU(
118 const sp<IGPUCallback>& callback, gpu_info_t* gpu)
119 {
120 Parcel data, reply;
121 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
122 data.writeStrongBinder(callback->asBinder());
123 remote()->transact(BnSurfaceComposer::REQUEST_GPU, data, &reply);
124 gpu->regs = interface_cast<IMemory>(reply.readStrongBinder());
125 gpu->count = reply.readInt32();
126
127 // FIXME: for now, we don't dynamically allocate the regions array
128 size_t maxCount = sizeof(gpu->regions)/sizeof(*gpu->regions);
129 if (gpu->count > maxCount)
130 return BAD_VALUE;
131
132 for (size_t i=0 ; i<gpu->count ; i++) {
133 gpu->regions[i].region = interface_cast<IMemory>(reply.readStrongBinder());
134 gpu->regions[i].reserved = reply.readInt32();
135 }
136 return reply.readInt32();
137 }
138
139 virtual status_t revokeGPU()
140 {
141 Parcel data, reply;
142 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
143 remote()->transact(BnSurfaceComposer::REVOKE_GPU, data, &reply);
144 return reply.readInt32();
145 }
146
147 virtual void signal() const
148 {
149 Parcel data, reply;
150 data.writeInterfaceToken(ISurfaceComposer::getInterfaceDescriptor());
151 remote()->transact(BnSurfaceComposer::SIGNAL, data, &reply, IBinder::FLAG_ONEWAY);
152 }
153};
154
155IMPLEMENT_META_INTERFACE(SurfaceComposer, "android.ui.ISurfaceComposer");
156
157// ----------------------------------------------------------------------
158
159#define CHECK_INTERFACE(interface, data, reply) \
160 do { if (!data.enforceInterface(interface::getInterfaceDescriptor())) { \
161 LOGW("Call incorrectly routed to " #interface); \
162 return PERMISSION_DENIED; \
163 } } while (0)
164
165status_t BnSurfaceComposer::onTransact(
166 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
167{
168 status_t err = BnInterface<ISurfaceComposer>::onTransact(code, data, reply, flags);
169 if (err == NO_ERROR)
170 return err;
171
172 CHECK_INTERFACE(ISurfaceComposer, data, reply);
173
174 switch(code) {
175 case CREATE_CONNECTION: {
176 sp<IBinder> b = createConnection()->asBinder();
177 reply->writeStrongBinder(b);
178 } break;
179 case OPEN_GLOBAL_TRANSACTION: {
180 openGlobalTransaction();
181 } break;
182 case CLOSE_GLOBAL_TRANSACTION: {
183 closeGlobalTransaction();
184 } break;
185 case SET_ORIENTATION: {
186 DisplayID dpy = data.readInt32();
187 int orientation = data.readInt32();
Mathias Agopianeb0c86e2009-03-27 18:11:38 -0700188 uint32_t flags = data.readInt32();
189 reply->writeInt32( setOrientation(dpy, orientation, flags) );
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800190 } break;
191 case FREEZE_DISPLAY: {
192 DisplayID dpy = data.readInt32();
193 uint32_t flags = data.readInt32();
194 reply->writeInt32( freezeDisplay(dpy, flags) );
195 } break;
196 case UNFREEZE_DISPLAY: {
197 DisplayID dpy = data.readInt32();
198 uint32_t flags = data.readInt32();
199 reply->writeInt32( unfreezeDisplay(dpy, flags) );
200 } break;
201 case BOOT_FINISHED: {
202 bootFinished();
203 } break;
204 case REVOKE_GPU: {
205 reply->writeInt32( revokeGPU() );
206 } break;
207 case SIGNAL: {
208 signal();
209 } break;
210 case GET_CBLK: {
211 sp<IBinder> b = getCblk()->asBinder();
212 reply->writeStrongBinder(b);
213 } break;
214 case REQUEST_GPU: {
215 // TODO: this should be protected by a permission
216 gpu_info_t info;
217 sp<IGPUCallback> callback
218 = interface_cast<IGPUCallback>(data.readStrongBinder());
219 status_t res = requestGPU(callback, &info);
220
221 // FIXME: for now, we don't dynamically allocate the regions array
222 size_t maxCount = sizeof(info.regions)/sizeof(*info.regions);
223 if (info.count > maxCount)
224 return BAD_VALUE;
225
226 reply->writeStrongBinder(info.regs->asBinder());
227 reply->writeInt32(info.count);
228 for (size_t i=0 ; i<info.count ; i++) {
229 reply->writeStrongBinder(info.regions[i].region->asBinder());
230 reply->writeInt32(info.regions[i].reserved);
231 }
232 reply->writeInt32(res);
233 } break;
234 default:
235 return UNKNOWN_TRANSACTION;
236 }
237 return NO_ERROR;
238}
239
240// ----------------------------------------------------------------------------
241
242enum {
243 // Note: BOOT_FINISHED must remain this value, it is called by ActivityManagerService.
244 GPU_LOST = IBinder::FIRST_CALL_TRANSACTION
245};
246
247class BpGPUCallback : public BpInterface<IGPUCallback>
248{
249public:
250 BpGPUCallback(const sp<IBinder>& impl)
251 : BpInterface<IGPUCallback>(impl)
252 {
253 }
254
255 virtual void gpuLost()
256 {
257 Parcel data, reply;
258 data.writeInterfaceToken(IGPUCallback::getInterfaceDescriptor());
259 remote()->transact(GPU_LOST, data, &reply, IBinder::FLAG_ONEWAY);
260 }
261};
262
263IMPLEMENT_META_INTERFACE(GPUCallback, "android.ui.IGPUCallback");
264
265status_t BnGPUCallback::onTransact(
266 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
267{
268 switch(code) {
269 case GPU_LOST: {
270 CHECK_INTERFACE(IGPUCallback, data, reply);
271 gpuLost();
272 return NO_ERROR;
273 } break;
274 default:
275 return BBinder::onTransact(code, data, reply, flags);
276 }
277}
278
279};