blob: 526328146d82c0c99c837ac803f9bb8113e51acd [file] [log] [blame]
Mathias Agopiana4e19522013-07-31 20:09:53 -07001/*
2 * Copyright (C) 2013 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#define EGL_EGLEXT_PROTOTYPES
18
19#include <EGL/egl.h>
20#include <EGL/eglext.h>
21
22
23#include <stdint.h>
24#include <sys/types.h>
25
26#include <utils/Errors.h>
27
28#include <binder/Parcel.h>
29#include <binder/IInterface.h>
30
31#include <gui/IConsumerListener.h>
32#include <gui/IGraphicBufferConsumer.h>
33
34#include <ui/GraphicBuffer.h>
35#include <ui/Fence.h>
36
37#include <system/window.h>
38
39namespace android {
40// ---------------------------------------------------------------------------
41
42IGraphicBufferConsumer::BufferItem::BufferItem() :
43 mTransform(0),
44 mScalingMode(NATIVE_WINDOW_SCALING_MODE_FREEZE),
45 mTimestamp(0),
46 mFrameNumber(0),
47 mBuf(INVALID_BUFFER_SLOT),
48 mIsDroppable(false),
49 mAcquireCalled(false) {
50 mCrop.makeInvalid();
51}
52
53size_t IGraphicBufferConsumer::BufferItem::getPodSize() const {
54 size_t c = sizeof(mCrop) +
55 sizeof(mTransform) +
56 sizeof(mScalingMode) +
57 sizeof(mTimestamp) +
58 sizeof(mFrameNumber) +
59 sizeof(mBuf) +
60 sizeof(mIsDroppable) +
61 sizeof(mAcquireCalled);
62 return c;
63}
64
65size_t IGraphicBufferConsumer::BufferItem::getFlattenedSize() const {
66 size_t c = 0;
67 if (mGraphicBuffer != 0) {
68 c += mGraphicBuffer->getFlattenedSize();
69 FlattenableUtils::align<4>(c);
70 }
71 if (mFence != 0) {
72 c += mFence->getFlattenedSize();
73 FlattenableUtils::align<4>(c);
74 }
75 return sizeof(int32_t) + c + getPodSize();
76}
77
78size_t IGraphicBufferConsumer::BufferItem::getFdCount() const {
79 size_t c = 0;
80 if (mGraphicBuffer != 0) {
81 c += mGraphicBuffer->getFdCount();
82 }
83 if (mFence != 0) {
84 c += mFence->getFdCount();
85 }
86 return c;
87}
88
89status_t IGraphicBufferConsumer::BufferItem::flatten(
90 void*& buffer, size_t& size, int*& fds, size_t& count) const {
91
92 // make sure we have enough space
93 if (count < BufferItem::getFlattenedSize()) {
94 return NO_MEMORY;
95 }
96
97 // content flags are stored first
98 uint32_t& flags = *static_cast<uint32_t*>(buffer);
99
100 // advance the pointer
101 FlattenableUtils::advance(buffer, size, sizeof(uint32_t));
102
103 flags = 0;
104 if (mGraphicBuffer != 0) {
105 status_t err = mGraphicBuffer->flatten(buffer, size, fds, count);
106 if (err) return err;
107 size -= FlattenableUtils::align<4>(buffer);
108 flags |= 1;
109 }
110 if (mFence != 0) {
111 status_t err = mFence->flatten(buffer, size, fds, count);
112 if (err) return err;
113 size -= FlattenableUtils::align<4>(buffer);
114 flags |= 2;
115 }
116
117 // check we have enough space (in case flattening the fence/graphicbuffer lied to us)
118 if (size < getPodSize()) {
119 return NO_MEMORY;
120 }
121
122 FlattenableUtils::write(buffer, size, mCrop);
123 FlattenableUtils::write(buffer, size, mTransform);
124 FlattenableUtils::write(buffer, size, mScalingMode);
125 FlattenableUtils::write(buffer, size, mTimestamp);
126 FlattenableUtils::write(buffer, size, mFrameNumber);
127 FlattenableUtils::write(buffer, size, mBuf);
128 FlattenableUtils::write(buffer, size, mIsDroppable);
129 FlattenableUtils::write(buffer, size, mAcquireCalled);
130
131 return NO_ERROR;
132}
133
134status_t IGraphicBufferConsumer::BufferItem::unflatten(
135 void const*& buffer, size_t& size, int const*& fds, size_t& count) {
136
137 if (size < sizeof(uint32_t))
138 return NO_MEMORY;
139
140 uint32_t flags = 0;
141 FlattenableUtils::read(buffer, size, flags);
142
143 if (flags & 1) {
144 mGraphicBuffer = new GraphicBuffer();
145 status_t err = mGraphicBuffer->unflatten(buffer, size, fds, count);
146 if (err) return err;
147 size -= FlattenableUtils::align<4>(buffer);
148 }
149
150 if (flags & 2) {
151 mFence = new Fence();
152 status_t err = mFence->unflatten(buffer, size, fds, count);
153 if (err) return err;
154 size -= FlattenableUtils::align<4>(buffer);
155 }
156
157 // check we have enough space
158 if (size < getPodSize()) {
159 return NO_MEMORY;
160 }
161
162 FlattenableUtils::read(buffer, size, mCrop);
163 FlattenableUtils::read(buffer, size, mTransform);
164 FlattenableUtils::read(buffer, size, mScalingMode);
165 FlattenableUtils::read(buffer, size, mTimestamp);
166 FlattenableUtils::read(buffer, size, mFrameNumber);
167 FlattenableUtils::read(buffer, size, mBuf);
168 FlattenableUtils::read(buffer, size, mIsDroppable);
169 FlattenableUtils::read(buffer, size, mAcquireCalled);
170
171 return NO_ERROR;
172}
173
174// ---------------------------------------------------------------------------
175
176enum {
177 ACQUIRE_BUFFER = IBinder::FIRST_CALL_TRANSACTION,
178 RELEASE_BUFFER,
179 CONSUMER_CONNECT,
180 CONSUMER_DISCONNECT,
181 GET_RELEASED_BUFFERS,
182 SET_DEFAULT_BUFFER_SIZE,
183 SET_DEFAULT_MAX_BUFFER_COUNT,
184 DISABLE_ASYNC_BUFFER,
185 SET_MAX_ACQUIRED_BUFFER_COUNT,
186 SET_CONSUMER_NAME,
187 SET_DEFAULT_BUFFER_FORMAT,
188 SET_CONSUMER_USAGE_BITS,
189 SET_TRANSFORM_HINT
190};
191
192
193class BpGraphicBufferConsumer : public BpInterface<IGraphicBufferConsumer>
194{
195public:
196 BpGraphicBufferConsumer(const sp<IBinder>& impl)
197 : BpInterface<IGraphicBufferConsumer>(impl)
198 {
199 }
200
201 virtual status_t acquireBuffer(BufferItem *buffer, nsecs_t presentWhen) {
202 Parcel data, reply;
203 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
204 data.writeInt64(presentWhen);
205 status_t result = remote()->transact(ACQUIRE_BUFFER, data, &reply);
206 if (result != NO_ERROR) {
207 return result;
208 }
209 result = reply.read(*buffer);
210 if (result != NO_ERROR) {
211 return result;
212 }
213 return reply.readInt32();
214 }
215
216 virtual status_t releaseBuffer(int buf, uint64_t frameNumber,
217 EGLDisplay display, EGLSyncKHR fence,
218 const sp<Fence>& releaseFence) {
219 Parcel data, reply;
220 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
221 data.writeInt32(buf);
222 data.writeInt64(frameNumber);
223 data.write(*releaseFence);
224 status_t result = remote()->transact(RELEASE_BUFFER, data, &reply);
225 if (result != NO_ERROR) {
226 return result;
227 }
228 return reply.readInt32();
229 }
230
231 virtual status_t consumerConnect(const sp<IConsumerListener>& consumer, bool controlledByApp) {
232 Parcel data, reply;
233 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
234 data.writeStrongBinder(consumer->asBinder());
235 data.writeInt32(controlledByApp);
236 status_t result = remote()->transact(CONSUMER_CONNECT, data, &reply);
237 if (result != NO_ERROR) {
238 return result;
239 }
240 return reply.readInt32();
241 }
242
243 virtual status_t consumerDisconnect() {
244 Parcel data, reply;
245 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
246 status_t result = remote()->transact(CONSUMER_DISCONNECT, data, &reply);
247 if (result != NO_ERROR) {
248 return result;
249 }
250 return reply.readInt32();
251 }
252
253 virtual status_t getReleasedBuffers(uint32_t* slotMask) {
254 Parcel data, reply;
255 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
256 status_t result = remote()->transact(GET_RELEASED_BUFFERS, data, &reply);
257 if (result != NO_ERROR) {
258 return result;
259 }
260 *slotMask = reply.readInt32();
261 return reply.readInt32();
262 }
263
264 virtual status_t setDefaultBufferSize(uint32_t w, uint32_t h) {
265 Parcel data, reply;
266 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
267 data.writeInt32(w);
268 data.writeInt32(h);
269 status_t result = remote()->transact(SET_DEFAULT_BUFFER_SIZE, data, &reply);
270 if (result != NO_ERROR) {
271 return result;
272 }
273 return reply.readInt32();
274 }
275
276 virtual status_t setDefaultMaxBufferCount(int bufferCount) {
277 Parcel data, reply;
278 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
279 data.writeInt32(bufferCount);
280 status_t result = remote()->transact(SET_DEFAULT_MAX_BUFFER_COUNT, data, &reply);
281 if (result != NO_ERROR) {
282 return result;
283 }
284 return reply.readInt32();
285 }
286
287 virtual status_t disableAsyncBuffer() {
288 Parcel data, reply;
289 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
290 status_t result = remote()->transact(DISABLE_ASYNC_BUFFER, data, &reply);
291 if (result != NO_ERROR) {
292 return result;
293 }
294 return reply.readInt32();
295 }
296
297 virtual status_t setMaxAcquiredBufferCount(int maxAcquiredBuffers) {
298 Parcel data, reply;
299 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
300 data.writeInt32(maxAcquiredBuffers);
301 status_t result = remote()->transact(SET_MAX_ACQUIRED_BUFFER_COUNT, data, &reply);
302 if (result != NO_ERROR) {
303 return result;
304 }
305 return reply.readInt32();
306 }
307
308 virtual void setConsumerName(const String8& name) {
309 Parcel data, reply;
310 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
311 data.writeString8(name);
312 remote()->transact(SET_CONSUMER_NAME, data, &reply);
313 }
314
315 virtual status_t setDefaultBufferFormat(uint32_t defaultFormat) {
316 Parcel data, reply;
317 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
318 data.writeInt32(defaultFormat);
319 status_t result = remote()->transact(SET_DEFAULT_BUFFER_FORMAT, data, &reply);
320 if (result != NO_ERROR) {
321 return result;
322 }
323 return reply.readInt32();
324 }
325
326 virtual status_t setConsumerUsageBits(uint32_t usage) {
327 Parcel data, reply;
328 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
329 data.writeInt32(usage);
330 status_t result = remote()->transact(SET_CONSUMER_USAGE_BITS, data, &reply);
331 if (result != NO_ERROR) {
332 return result;
333 }
334 return reply.readInt32();
335 }
336
337 virtual status_t setTransformHint(uint32_t hint) {
338 Parcel data, reply;
339 data.writeInterfaceToken(IGraphicBufferConsumer::getInterfaceDescriptor());
340 data.writeInt32(hint);
341 status_t result = remote()->transact(SET_TRANSFORM_HINT, data, &reply);
342 if (result != NO_ERROR) {
343 return result;
344 }
345 return reply.readInt32();
346 }
347};
348
349IMPLEMENT_META_INTERFACE(GraphicBufferConsumer, "android.gui.IGraphicBufferConsumer");
350
351// ----------------------------------------------------------------------
352
353status_t BnGraphicBufferConsumer::onTransact(
354 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
355{
356 switch(code) {
357 case ACQUIRE_BUFFER: {
358 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
359 BufferItem item;
360 int64_t presentWhen = data.readInt64();
361 status_t result = acquireBuffer(&item, presentWhen);
362 status_t err = reply->write(item);
363 if (err) return err;
364 reply->writeInt32(result);
365 return NO_ERROR;
366 } break;
367 case RELEASE_BUFFER: {
368 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
369 int buf = data.readInt32();
370 uint64_t frameNumber = data.readInt64();
371 sp<Fence> releaseFence = new Fence();
372 status_t err = data.read(*releaseFence);
373 if (err) return err;
374 status_t result = releaseBuffer(buf, frameNumber,
375 EGL_NO_DISPLAY, EGL_NO_SYNC_KHR, releaseFence);
376 reply->writeInt32(result);
377 return NO_ERROR;
378 } break;
379 case CONSUMER_CONNECT: {
380 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
381 sp<IConsumerListener> consumer = IConsumerListener::asInterface( data.readStrongBinder() );
382 bool controlledByApp = data.readInt32();
383 status_t result = consumerConnect(consumer, controlledByApp);
384 reply->writeInt32(result);
385 return NO_ERROR;
386 } break;
387 case CONSUMER_DISCONNECT: {
388 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
389 status_t result = consumerDisconnect();
390 reply->writeInt32(result);
391 return NO_ERROR;
392 } break;
393 case GET_RELEASED_BUFFERS: {
394 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
395 uint32_t slotMask;
396 status_t result = getReleasedBuffers(&slotMask);
397 reply->writeInt32(slotMask);
398 reply->writeInt32(result);
399 return NO_ERROR;
400 } break;
401 case SET_DEFAULT_BUFFER_SIZE: {
402 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
403 uint32_t w = data.readInt32();
404 uint32_t h = data.readInt32();
405 status_t result = setDefaultBufferSize(w, h);
406 reply->writeInt32(result);
407 return NO_ERROR;
408 } break;
409 case SET_DEFAULT_MAX_BUFFER_COUNT: {
410 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
411 uint32_t bufferCount = data.readInt32();
412 status_t result = setDefaultMaxBufferCount(bufferCount);
413 reply->writeInt32(result);
414 return NO_ERROR;
415 } break;
416 case DISABLE_ASYNC_BUFFER: {
417 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
418 status_t result = disableAsyncBuffer();
419 reply->writeInt32(result);
420 return NO_ERROR;
421 } break;
422 case SET_MAX_ACQUIRED_BUFFER_COUNT: {
423 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
424 uint32_t maxAcquiredBuffers = data.readInt32();
425 status_t result = setMaxAcquiredBufferCount(maxAcquiredBuffers);
426 reply->writeInt32(result);
427 return NO_ERROR;
428 } break;
429 case SET_CONSUMER_NAME: {
430 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
431 setConsumerName( data.readString8() );
432 return NO_ERROR;
433 } break;
434 case SET_DEFAULT_BUFFER_FORMAT: {
435 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
436 uint32_t defaultFormat = data.readInt32();
437 status_t result = setDefaultBufferFormat(defaultFormat);
438 reply->writeInt32(result);
439 return NO_ERROR;
440 } break;
441 case SET_CONSUMER_USAGE_BITS: {
442 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
443 uint32_t usage = data.readInt32();
444 status_t result = setConsumerUsageBits(usage);
445 reply->writeInt32(result);
446 return NO_ERROR;
447 } break;
448 case SET_TRANSFORM_HINT: {
449 CHECK_INTERFACE(IGraphicBufferConsumer, data, reply);
450 uint32_t hint = data.readInt32();
451 status_t result = setTransformHint(hint);
452 reply->writeInt32(result);
453 return NO_ERROR;
454 } break;
455 }
456 return BBinder::onTransact(code, data, reply, flags);
457}
458
459}; // namespace android