blob: f725c9774e303d05d3d6c2c740a4b99613e9894d [file] [log] [blame]
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001/*
2**
3** Copyright (C) 2008 The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
Mark Salyzyn34fb2962014-06-18 16:30:56 -070018#include <inttypes.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080019#include <stdint.h>
20#include <sys/types.h>
Mark Salyzyn34fb2962014-06-18 16:30:56 -070021
Mathias Agopian75624082009-05-19 19:08:10 -070022#include <binder/Parcel.h>
Chris Watkins99f31602015-03-20 13:06:33 -070023#include <media/IDataSource.h>
Andreas Huber1b86fe02014-01-29 11:13:26 -080024#include <media/IMediaHTTPService.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080025#include <media/IMediaMetadataRetriever.h>
Andreas Huberaf8791e2011-03-21 10:25:44 -070026#include <utils/String8.h>
Sangkyu Leed01c1482013-02-08 16:26:39 +090027#include <utils/KeyedVector.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080028
Dave Sparksec4dde72009-11-23 16:51:15 -080029// The binder is supposed to propagate the scheduler group across
30// the binder interface so that remote calls are executed with
31// the same priority as local calls. This is currently not working
32// so this change puts in a temporary hack to fix the issue with
33// metadata retrieval which can be a huge CPU hit if done on a
34// foreground thread.
35#ifndef DISABLE_GROUP_SCHEDULE_HACK
36
Dave Sparksf311c552009-11-23 19:51:33 -080037#undef LOG_TAG
Dave Sparksec4dde72009-11-23 16:51:15 -080038#define LOG_TAG "IMediaMetadataRetriever"
39#include <utils/Log.h>
40#include <cutils/sched_policy.h>
41
42namespace android {
43
44static void sendSchedPolicy(Parcel& data)
45{
46 SchedPolicy policy;
Glenn Kasten0512ab52011-05-04 17:58:57 -070047 get_sched_policy(gettid(), &policy);
Dave Sparksec4dde72009-11-23 16:51:15 -080048 data.writeInt32(policy);
49}
50
51static void setSchedPolicy(const Parcel& data)
52{
53 SchedPolicy policy = (SchedPolicy) data.readInt32();
Glenn Kasten0512ab52011-05-04 17:58:57 -070054 set_sched_policy(gettid(), policy);
Dave Sparksec4dde72009-11-23 16:51:15 -080055}
56static void restoreSchedPolicy()
57{
Glenn Kasten0512ab52011-05-04 17:58:57 -070058 set_sched_policy(gettid(), SP_FOREGROUND);
Dave Sparksec4dde72009-11-23 16:51:15 -080059}
60}; // end namespace android
61#endif
62
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080063namespace android {
64
65enum {
66 DISCONNECT = IBinder::FIRST_CALL_TRANSACTION,
67 SET_DATA_SOURCE_URL,
68 SET_DATA_SOURCE_FD,
Chris Watkins99f31602015-03-20 13:06:33 -070069 SET_DATA_SOURCE_CALLBACK,
James Dong16afe2f2010-12-02 17:42:08 -080070 GET_FRAME_AT_TIME,
Chong Zhangd3e0d862017-10-03 13:17:13 -070071 GET_IMAGE_AT_INDEX,
72 GET_FRAME_AT_INDEX,
Dave Sparksec4dde72009-11-23 16:51:15 -080073 EXTRACT_ALBUM_ART,
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080074 EXTRACT_METADATA,
75};
76
77class BpMediaMetadataRetriever: public BpInterface<IMediaMetadataRetriever>
78{
79public:
Chih-Hung Hsieh090ef602016-04-27 10:39:54 -070080 explicit BpMediaMetadataRetriever(const sp<IBinder>& impl)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080081 : BpInterface<IMediaMetadataRetriever>(impl)
82 {
83 }
84
85 // disconnect from media metadata retriever service
86 void disconnect()
87 {
88 Parcel data, reply;
89 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
90 remote()->transact(DISCONNECT, data, &reply);
91 }
92
Andreas Huberaf8791e2011-03-21 10:25:44 -070093 status_t setDataSource(
Andreas Huber1b86fe02014-01-29 11:13:26 -080094 const sp<IMediaHTTPService> &httpService,
95 const char *srcUrl,
96 const KeyedVector<String8, String8> *headers)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080097 {
98 Parcel data, reply;
99 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
Andreas Huber1b86fe02014-01-29 11:13:26 -0800100 data.writeInt32(httpService != NULL);
101 if (httpService != NULL) {
Marco Nelissen06b46062014-11-14 07:58:25 -0800102 data.writeStrongBinder(IInterface::asBinder(httpService));
Andreas Huber1b86fe02014-01-29 11:13:26 -0800103 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800104 data.writeCString(srcUrl);
Andreas Huberaf8791e2011-03-21 10:25:44 -0700105
106 if (headers == NULL) {
107 data.writeInt32(0);
108 } else {
109 // serialize the headers
Glenn Kastene03dd222014-01-28 11:04:39 -0800110 data.writeInt64(headers->size());
Andreas Huberaf8791e2011-03-21 10:25:44 -0700111 for (size_t i = 0; i < headers->size(); ++i) {
112 data.writeString8(headers->keyAt(i));
113 data.writeString8(headers->valueAt(i));
114 }
115 }
116
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800117 remote()->transact(SET_DATA_SOURCE_URL, data, &reply);
118 return reply.readInt32();
119 }
120
121 status_t setDataSource(int fd, int64_t offset, int64_t length)
122 {
123 Parcel data, reply;
124 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
125 data.writeFileDescriptor(fd);
126 data.writeInt64(offset);
127 data.writeInt64(length);
128 remote()->transact(SET_DATA_SOURCE_FD, data, &reply);
129 return reply.readInt32();
130 }
131
Chong Zhang24c15772017-07-26 16:25:28 -0700132 status_t setDataSource(const sp<IDataSource>& source, const char *mime)
Chris Watkins99f31602015-03-20 13:06:33 -0700133 {
134 Parcel data, reply;
135 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
136 data.writeStrongBinder(IInterface::asBinder(source));
Chong Zhang24c15772017-07-26 16:25:28 -0700137
138 if (mime != NULL) {
139 data.writeInt32(1);
140 data.writeCString(mime);
141 } else {
142 data.writeInt32(0);
143 }
Chris Watkins99f31602015-03-20 13:06:33 -0700144 remote()->transact(SET_DATA_SOURCE_CALLBACK, data, &reply);
145 return reply.readInt32();
146 }
147
Chong Zhang24c15772017-07-26 16:25:28 -0700148 sp<IMemory> getFrameAtTime(int64_t timeUs, int option, int colorFormat, bool metaOnly)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800149 {
Chong Zhang24c15772017-07-26 16:25:28 -0700150 ALOGV("getTimeAtTime: time(%" PRId64 " us), option(%d), colorFormat(%d) metaOnly(%d)",
151 timeUs, option, colorFormat, metaOnly);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800152 Parcel data, reply;
153 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
James Dong16afe2f2010-12-02 17:42:08 -0800154 data.writeInt64(timeUs);
155 data.writeInt32(option);
Chong Zhang24c15772017-07-26 16:25:28 -0700156 data.writeInt32(colorFormat);
157 data.writeInt32(metaOnly);
Dave Sparksec4dde72009-11-23 16:51:15 -0800158#ifndef DISABLE_GROUP_SCHEDULE_HACK
159 sendSchedPolicy(data);
160#endif
James Dong16afe2f2010-12-02 17:42:08 -0800161 remote()->transact(GET_FRAME_AT_TIME, data, &reply);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800162 status_t ret = reply.readInt32();
163 if (ret != NO_ERROR) {
164 return NULL;
165 }
166 return interface_cast<IMemory>(reply.readStrongBinder());
167 }
168
Chong Zhangd3e0d862017-10-03 13:17:13 -0700169 sp<IMemory> getImageAtIndex(int index, int colorFormat, bool metaOnly)
170 {
171 ALOGV("getImageAtIndex: index %d, colorFormat(%d) metaOnly(%d)",
172 index, colorFormat, metaOnly);
173 Parcel data, reply;
174 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
175 data.writeInt32(index);
176 data.writeInt32(colorFormat);
177 data.writeInt32(metaOnly);
178#ifndef DISABLE_GROUP_SCHEDULE_HACK
179 sendSchedPolicy(data);
180#endif
181 remote()->transact(GET_IMAGE_AT_INDEX, data, &reply);
182 status_t ret = reply.readInt32();
183 if (ret != NO_ERROR) {
184 return NULL;
185 }
186 return interface_cast<IMemory>(reply.readStrongBinder());
187 }
188
189 status_t getFrameAtIndex(std::vector<sp<IMemory> > *frames,
190 int frameIndex, int numFrames, int colorFormat, bool metaOnly)
191 {
192 ALOGV("getFrameAtIndex: frameIndex(%d), numFrames(%d), colorFormat(%d) metaOnly(%d)",
193 frameIndex, numFrames, colorFormat, metaOnly);
194 Parcel data, reply;
195 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
196 data.writeInt32(frameIndex);
197 data.writeInt32(numFrames);
198 data.writeInt32(colorFormat);
199 data.writeInt32(metaOnly);
200#ifndef DISABLE_GROUP_SCHEDULE_HACK
201 sendSchedPolicy(data);
202#endif
203 remote()->transact(GET_FRAME_AT_INDEX, data, &reply);
204 status_t ret = reply.readInt32();
205 if (ret != NO_ERROR) {
206 return ret;
207 }
208 int retNumFrames = reply.readInt32();
209 if (retNumFrames < numFrames) {
210 numFrames = retNumFrames;
211 }
212 for (int i = 0; i < numFrames; i++) {
213 frames->push_back(interface_cast<IMemory>(reply.readStrongBinder()));
214 }
215 return OK;
216 }
217
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800218 sp<IMemory> extractAlbumArt()
219 {
220 Parcel data, reply;
221 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
Dave Sparksec4dde72009-11-23 16:51:15 -0800222#ifndef DISABLE_GROUP_SCHEDULE_HACK
223 sendSchedPolicy(data);
224#endif
225 remote()->transact(EXTRACT_ALBUM_ART, data, &reply);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800226 status_t ret = reply.readInt32();
227 if (ret != NO_ERROR) {
228 return NULL;
229 }
230 return interface_cast<IMemory>(reply.readStrongBinder());
231 }
232
233 const char* extractMetadata(int keyCode)
234 {
235 Parcel data, reply;
236 data.writeInterfaceToken(IMediaMetadataRetriever::getInterfaceDescriptor());
Dave Sparksec4dde72009-11-23 16:51:15 -0800237#ifndef DISABLE_GROUP_SCHEDULE_HACK
238 sendSchedPolicy(data);
239#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800240 data.writeInt32(keyCode);
241 remote()->transact(EXTRACT_METADATA, data, &reply);
242 status_t ret = reply.readInt32();
243 if (ret != NO_ERROR) {
244 return NULL;
245 }
Sangkyu Leed01c1482013-02-08 16:26:39 +0900246 const char* str = reply.readCString();
247 if (str != NULL) {
248 String8 value(str);
249 if (mMetadata.indexOfKey(keyCode) < 0) {
250 mMetadata.add(keyCode, value);
251 } else {
252 mMetadata.replaceValueFor(keyCode, value);
253 }
254 return mMetadata.valueFor(keyCode).string();
255 } else {
256 return NULL;
257 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800258 }
Sangkyu Leed01c1482013-02-08 16:26:39 +0900259
260private:
261 KeyedVector<int, String8> mMetadata;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800262};
263
niko56f0cc52009-06-22 08:49:52 -0700264IMPLEMENT_META_INTERFACE(MediaMetadataRetriever, "android.media.IMediaMetadataRetriever");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800265
266// ----------------------------------------------------------------------
267
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800268status_t BnMediaMetadataRetriever::onTransact(
269 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
270{
271 switch (code) {
272 case DISCONNECT: {
273 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
274 disconnect();
275 return NO_ERROR;
276 } break;
277 case SET_DATA_SOURCE_URL: {
278 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
Andreas Huber1b86fe02014-01-29 11:13:26 -0800279
280 sp<IMediaHTTPService> httpService;
281 if (data.readInt32()) {
282 httpService =
283 interface_cast<IMediaHTTPService>(data.readStrongBinder());
284 }
285
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800286 const char* srcUrl = data.readCString();
Andreas Huberaf8791e2011-03-21 10:25:44 -0700287
Wei Jia2afac0c2016-01-07 12:13:07 -0800288 if (httpService == NULL || srcUrl == NULL) {
289 reply->writeInt32(BAD_VALUE);
290 return NO_ERROR;
291 }
292
Andreas Huberaf8791e2011-03-21 10:25:44 -0700293 KeyedVector<String8, String8> headers;
Glenn Kastene03dd222014-01-28 11:04:39 -0800294 size_t numHeaders = (size_t) data.readInt64();
295 for (size_t i = 0; i < numHeaders; ++i) {
Andreas Huberaf8791e2011-03-21 10:25:44 -0700296 String8 key = data.readString8();
297 String8 value = data.readString8();
298 headers.add(key, value);
299 }
300
301 reply->writeInt32(
Andreas Huber1b86fe02014-01-29 11:13:26 -0800302 setDataSource(
303 httpService, srcUrl, numHeaders > 0 ? &headers : NULL));
Andreas Huberaf8791e2011-03-21 10:25:44 -0700304
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800305 return NO_ERROR;
306 } break;
307 case SET_DATA_SOURCE_FD: {
308 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
Taiju Tsuiki55203e22015-04-21 17:36:22 +0900309 int fd = data.readFileDescriptor();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800310 int64_t offset = data.readInt64();
311 int64_t length = data.readInt64();
312 reply->writeInt32(setDataSource(fd, offset, length));
313 return NO_ERROR;
314 } break;
Chris Watkins99f31602015-03-20 13:06:33 -0700315 case SET_DATA_SOURCE_CALLBACK: {
316 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
317 sp<IDataSource> source =
318 interface_cast<IDataSource>(data.readStrongBinder());
Wei Jia2afac0c2016-01-07 12:13:07 -0800319 if (source == NULL) {
320 reply->writeInt32(BAD_VALUE);
321 } else {
Chong Zhang24c15772017-07-26 16:25:28 -0700322 int32_t hasMime = data.readInt32();
323 const char *mime = NULL;
324 if (hasMime) {
325 mime = data.readCString();
326 }
327 reply->writeInt32(setDataSource(source, mime));
Wei Jia2afac0c2016-01-07 12:13:07 -0800328 }
Chris Watkins99f31602015-03-20 13:06:33 -0700329 return NO_ERROR;
330 } break;
James Dong16afe2f2010-12-02 17:42:08 -0800331 case GET_FRAME_AT_TIME: {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800332 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
James Dong16afe2f2010-12-02 17:42:08 -0800333 int64_t timeUs = data.readInt64();
334 int option = data.readInt32();
Chong Zhang24c15772017-07-26 16:25:28 -0700335 int colorFormat = data.readInt32();
336 bool metaOnly = (data.readInt32() != 0);
337 ALOGV("getTimeAtTime: time(%" PRId64 " us), option(%d), colorFormat(%d), metaOnly(%d)",
338 timeUs, option, colorFormat, metaOnly);
Dave Sparksec4dde72009-11-23 16:51:15 -0800339#ifndef DISABLE_GROUP_SCHEDULE_HACK
340 setSchedPolicy(data);
341#endif
Chong Zhang24c15772017-07-26 16:25:28 -0700342 sp<IMemory> bitmap = getFrameAtTime(timeUs, option, colorFormat, metaOnly);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800343 if (bitmap != 0) { // Don't send NULL across the binder interface
344 reply->writeInt32(NO_ERROR);
Marco Nelissen06b46062014-11-14 07:58:25 -0800345 reply->writeStrongBinder(IInterface::asBinder(bitmap));
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800346 } else {
347 reply->writeInt32(UNKNOWN_ERROR);
348 }
Dave Sparksec4dde72009-11-23 16:51:15 -0800349#ifndef DISABLE_GROUP_SCHEDULE_HACK
350 restoreSchedPolicy();
351#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800352 return NO_ERROR;
353 } break;
Chong Zhangd3e0d862017-10-03 13:17:13 -0700354 case GET_IMAGE_AT_INDEX: {
355 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
356 int index = data.readInt32();
357 int colorFormat = data.readInt32();
358 bool metaOnly = (data.readInt32() != 0);
359 ALOGV("getImageAtIndex: index(%d), colorFormat(%d), metaOnly(%d)",
360 index, colorFormat, metaOnly);
361#ifndef DISABLE_GROUP_SCHEDULE_HACK
362 setSchedPolicy(data);
363#endif
364 sp<IMemory> bitmap = getImageAtIndex(index, colorFormat, metaOnly);
365 if (bitmap != 0) { // Don't send NULL across the binder interface
366 reply->writeInt32(NO_ERROR);
367 reply->writeStrongBinder(IInterface::asBinder(bitmap));
368 } else {
369 reply->writeInt32(UNKNOWN_ERROR);
370 }
371#ifndef DISABLE_GROUP_SCHEDULE_HACK
372 restoreSchedPolicy();
373#endif
374 return NO_ERROR;
375 } break;
376 case GET_FRAME_AT_INDEX: {
377 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
378 int frameIndex = data.readInt32();
379 int numFrames = data.readInt32();
380 int colorFormat = data.readInt32();
381 bool metaOnly = (data.readInt32() != 0);
382 ALOGV("getFrameAtIndex: frameIndex(%d), numFrames(%d), colorFormat(%d), metaOnly(%d)",
383 frameIndex, numFrames, colorFormat, metaOnly);
384#ifndef DISABLE_GROUP_SCHEDULE_HACK
385 setSchedPolicy(data);
386#endif
387 std::vector<sp<IMemory> > frames;
388 status_t err = getFrameAtIndex(
389 &frames, frameIndex, numFrames, colorFormat, metaOnly);
390 reply->writeInt32(err);
391 if (OK == err) {
392 reply->writeInt32(frames.size());
393 for (size_t i = 0; i < frames.size(); i++) {
394 reply->writeStrongBinder(IInterface::asBinder(frames[i]));
395 }
396 }
397#ifndef DISABLE_GROUP_SCHEDULE_HACK
398 restoreSchedPolicy();
399#endif
400 return NO_ERROR;
401 } break;
Dave Sparksec4dde72009-11-23 16:51:15 -0800402 case EXTRACT_ALBUM_ART: {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800403 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
Dave Sparksec4dde72009-11-23 16:51:15 -0800404#ifndef DISABLE_GROUP_SCHEDULE_HACK
405 setSchedPolicy(data);
406#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800407 sp<IMemory> albumArt = extractAlbumArt();
408 if (albumArt != 0) { // Don't send NULL across the binder interface
409 reply->writeInt32(NO_ERROR);
Marco Nelissen06b46062014-11-14 07:58:25 -0800410 reply->writeStrongBinder(IInterface::asBinder(albumArt));
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800411 } else {
412 reply->writeInt32(UNKNOWN_ERROR);
413 }
Dave Sparksec4dde72009-11-23 16:51:15 -0800414#ifndef DISABLE_GROUP_SCHEDULE_HACK
415 restoreSchedPolicy();
416#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800417 return NO_ERROR;
418 } break;
419 case EXTRACT_METADATA: {
420 CHECK_INTERFACE(IMediaMetadataRetriever, data, reply);
Dave Sparksec4dde72009-11-23 16:51:15 -0800421#ifndef DISABLE_GROUP_SCHEDULE_HACK
422 setSchedPolicy(data);
423#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800424 int keyCode = data.readInt32();
425 const char* value = extractMetadata(keyCode);
426 if (value != NULL) { // Don't send NULL across the binder interface
427 reply->writeInt32(NO_ERROR);
428 reply->writeCString(value);
429 } else {
430 reply->writeInt32(UNKNOWN_ERROR);
431 }
Dave Sparksec4dde72009-11-23 16:51:15 -0800432#ifndef DISABLE_GROUP_SCHEDULE_HACK
433 restoreSchedPolicy();
434#endif
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800435 return NO_ERROR;
436 } break;
437 default:
438 return BBinder::onTransact(code, data, reply, flags);
439 }
440}
441
442// ----------------------------------------------------------------------------
443
Glenn Kasten40bc9062015-03-20 09:09:33 -0700444} // namespace android