blob: 7ed76d8b2077e8963fca3786cd448748e77f61c0 [file] [log] [blame]
Marco Nelissenb2487f02015-09-01 13:23:23 -07001/*
2 * Copyright (C) 2009 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
Marco Nelissenb65990f2015-11-09 15:39:49 -080017//#define LOG_NDEBUG 0
Marco Nelissenb2487f02015-09-01 13:23:23 -070018#define LOG_TAG "BpMediaExtractor"
19#include <utils/Log.h>
20
21#include <stdint.h>
Marco Nelissenb5894372019-01-10 14:44:13 -080022#include <time.h>
Marco Nelissenb2487f02015-09-01 13:23:23 -070023#include <sys/types.h>
24
Marco Nelissen69d3d8a2016-03-07 13:20:01 -080025#include <binder/IPCThreadState.h>
Marco Nelissenb2487f02015-09-01 13:23:23 -070026#include <binder/Parcel.h>
ray-cy.leeb596c342017-03-07 18:25:15 +080027#include <binder/PermissionCache.h>
Marco Nelissendab79b32019-11-18 08:25:47 -080028#include <android/IMediaExtractor.h>
Marco Nelissenb2487f02015-09-01 13:23:23 -070029#include <media/stagefright/MetaData.h>
30
31namespace android {
32
33enum {
34 COUNTTRACKS = IBinder::FIRST_CALL_TRANSACTION,
35 GETTRACK,
36 GETTRACKMETADATA,
37 GETMETADATA,
38 FLAGS,
Chong Zhang9dbe9a52017-01-03 11:35:15 -080039 SETMEDIACAS,
Ray Essickba13b7b2017-02-07 10:03:18 -080040 NAME,
Santiago Seifert79c34f22020-11-11 23:49:28 +000041 GETMETRICS,
42 SETENTRYPOINT
Marco Nelissenb2487f02015-09-01 13:23:23 -070043};
44
45class BpMediaExtractor : public BpInterface<IMediaExtractor> {
46public:
Chih-Hung Hsieh64a28702016-05-03 09:52:51 -070047 explicit BpMediaExtractor(const sp<IBinder>& impl)
Marco Nelissenb2487f02015-09-01 13:23:23 -070048 : BpInterface<IMediaExtractor>(impl)
49 {
50 }
51
52 virtual size_t countTracks() {
53 ALOGV("countTracks");
54 Parcel data, reply;
55 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
56 status_t ret = remote()->transact(COUNTTRACKS, data, &reply);
57 size_t numTracks = 0;
58 if (ret == NO_ERROR) {
59 numTracks = reply.readUint32();
60 }
61 return numTracks;
62 }
63 virtual sp<IMediaSource> getTrack(size_t index) {
64 ALOGV("getTrack(%zu)", index);
65 Parcel data, reply;
66 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
67 data.writeUint32(index);
68 status_t ret = remote()->transact(GETTRACK, data, &reply);
69 if (ret == NO_ERROR) {
70 return interface_cast<IMediaSource>(reply.readStrongBinder());
71 }
72 return NULL;
73 }
74
75 virtual sp<MetaData> getTrackMetaData(
76 size_t index, uint32_t flags) {
77 ALOGV("getTrackMetaData(%zu, %u)", index, flags);
78 Parcel data, reply;
79 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
80 data.writeUint32(index);
81 data.writeUint32(flags);
82 status_t ret = remote()->transact(GETTRACKMETADATA, data, &reply);
83 if (ret == NO_ERROR) {
84 return MetaData::createFromParcel(reply);
85 }
86 return NULL;
87 }
88
89 virtual sp<MetaData> getMetaData() {
90 ALOGV("getMetaData");
91 Parcel data, reply;
92 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
93 status_t ret = remote()->transact(GETMETADATA, data, &reply);
94 if (ret == NO_ERROR) {
95 return MetaData::createFromParcel(reply);
96 }
97 return NULL;
98 }
99
Ray Essickba13b7b2017-02-07 10:03:18 -0800100 virtual status_t getMetrics(Parcel * reply) {
101 Parcel data;
102 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
103 status_t ret = remote()->transact(GETMETRICS, data, reply);
104 if (ret == NO_ERROR) {
105 return OK;
106 }
107 return UNKNOWN_ERROR;
108 }
109
Marco Nelissenb2487f02015-09-01 13:23:23 -0700110 virtual uint32_t flags() const {
Marco Nelissene4f4b792020-01-17 13:15:18 -0800111 ALOGV("flags");
112 Parcel data, reply;
113 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
114 status_t ret = remote()->transact(FLAGS, data, &reply);
115 int flgs = 0;
116 if (ret == NO_ERROR) {
117 flgs = reply.readUint32();
118 }
119 return flgs;
Marco Nelissenb2487f02015-09-01 13:23:23 -0700120 }
121
Chong Zhangd5a416a2017-05-16 11:16:34 -0700122 virtual status_t setMediaCas(const HInterfaceToken &casToken) {
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800123 ALOGV("setMediaCas");
124
125 Parcel data, reply;
126 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
Chong Zhangd5a416a2017-05-16 11:16:34 -0700127 data.writeByteVector(casToken);
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800128
129 status_t err = remote()->transact(SETMEDIACAS, data, &reply);
130 if (err != NO_ERROR) {
131 return err;
132 }
133 return reply.readInt32();
134 }
135
Marco Nelissene4f4b792020-01-17 13:15:18 -0800136 virtual String8 name() {
137 Parcel data, reply;
138 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
139 status_t ret = remote()->transact(NAME, data, &reply);
140 String8 nm;
141 if (ret == NO_ERROR) {
142 nm = reply.readString8();
143 }
144 return nm;
Marco Nelissenb2487f02015-09-01 13:23:23 -0700145 }
Santiago Seifert79c34f22020-11-11 23:49:28 +0000146
147 virtual status_t setEntryPoint(EntryPoint entryPoint) {
148 Parcel data, reply;
149 data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
150 data.writeInt32(static_cast<int32_t>(entryPoint));
151 return remote()->transact(SETENTRYPOINT, data, &reply);
152 }
Marco Nelissenb2487f02015-09-01 13:23:23 -0700153};
154
155IMPLEMENT_META_INTERFACE(MediaExtractor, "android.media.IMediaExtractor");
156
157#undef LOG_TAG
158#define LOG_TAG "BnMediaExtractor"
159
160status_t BnMediaExtractor::onTransact(
161 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
162{
163 switch (code) {
164 case COUNTTRACKS: {
165 ALOGV("countTracks");
166 CHECK_INTERFACE(IMediaExtractor, data, reply);
167 size_t numTracks = countTracks();
168 if (numTracks > INT32_MAX) {
169 numTracks = 0;
170 }
171 reply->writeUint32(uint32_t(numTracks));
172 return NO_ERROR;
173 }
174 case GETTRACK: {
175 ALOGV("getTrack()");
176 CHECK_INTERFACE(IMediaExtractor, data, reply);
177 uint32_t idx;
178 if (data.readUint32(&idx) == NO_ERROR) {
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800179 const sp<IMediaSource> track = getTrack(size_t(idx));
180 registerMediaSource(this, track);
181 return reply->writeStrongBinder(IInterface::asBinder(track));
Marco Nelissenb2487f02015-09-01 13:23:23 -0700182 }
183 return UNKNOWN_ERROR;
184 }
185 case GETTRACKMETADATA: {
186 ALOGV("getTrackMetaData");
187 CHECK_INTERFACE(IMediaExtractor, data, reply);
188 uint32_t idx;
189 uint32_t flags;
190 if (data.readUint32(&idx) == NO_ERROR &&
191 data.readUint32(&flags) == NO_ERROR) {
192 sp<MetaData> meta = getTrackMetaData(idx, flags);
Marco Nelissen0d138242016-10-06 15:31:52 -0700193 if (meta == NULL) {
194 return UNKNOWN_ERROR;
195 }
Marco Nelissenb2487f02015-09-01 13:23:23 -0700196 meta->writeToParcel(*reply);
197 return NO_ERROR;
198 }
199 return UNKNOWN_ERROR;
200 }
201 case GETMETADATA: {
202 ALOGV("getMetaData");
203 CHECK_INTERFACE(IMediaExtractor, data, reply);
204 sp<MetaData> meta = getMetaData();
205 if (meta != NULL) {
206 meta->writeToParcel(*reply);
207 return NO_ERROR;
208 }
209 return UNKNOWN_ERROR;
210 }
Ray Essickba13b7b2017-02-07 10:03:18 -0800211 case GETMETRICS: {
212 CHECK_INTERFACE(IMediaExtractor, data, reply);
213 status_t ret = getMetrics(reply);
214 return ret;
215 }
Marco Nelissene4f4b792020-01-17 13:15:18 -0800216 case FLAGS: {
217 ALOGV("flags");
218 CHECK_INTERFACE(IMediaExtractor, data, reply);
219 reply->writeUint32(this->flags());
220 return NO_ERROR;
221 }
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800222 case SETMEDIACAS: {
223 ALOGV("setMediaCas");
224 CHECK_INTERFACE(IMediaExtractor, data, reply);
225
Chong Zhangd5a416a2017-05-16 11:16:34 -0700226 HInterfaceToken casToken;
227 status_t err = data.readByteVector(&casToken);
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800228 if (err != NO_ERROR) {
Chong Zhangd5a416a2017-05-16 11:16:34 -0700229 ALOGE("Error reading casToken from parcel");
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800230 return err;
231 }
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800232
Chong Zhangd5a416a2017-05-16 11:16:34 -0700233 reply->writeInt32(setMediaCas(casToken));
Chong Zhang9dbe9a52017-01-03 11:35:15 -0800234 return OK;
235 }
Marco Nelissene4f4b792020-01-17 13:15:18 -0800236 case NAME: {
237 ALOGV("name");
238 CHECK_INTERFACE(IMediaExtractor, data, reply);
239 String8 nm = name();
240 reply->writeString8(nm);
241 return NO_ERROR;
242 }
Santiago Seifert79c34f22020-11-11 23:49:28 +0000243 case SETENTRYPOINT: {
244 ALOGV("setEntryPoint");
245 CHECK_INTERFACE(IMediaExtractor, data, reply);
246 int32_t entryPoint;
247 status_t err = data.readInt32(&entryPoint);
248 if (err == OK) {
249 setEntryPoint(EntryPoint(entryPoint));
250 }
251 return err;
252 }
Marco Nelissenb2487f02015-09-01 13:23:23 -0700253 default:
254 return BBinder::onTransact(code, data, reply, flags);
255 }
256}
257
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800258typedef struct {
259 String8 mime;
260 String8 name;
261 String8 sourceDescription;
262 pid_t owner;
263 wp<IMediaExtractor> extractor;
264 Vector<wp<IMediaSource>> tracks;
265 Vector<String8> trackDescriptions;
266 String8 toString() const;
Marco Nelissenb5894372019-01-10 14:44:13 -0800267 time_t when;
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800268} ExtractorInstance;
269
270String8 ExtractorInstance::toString() const {
Marco Nelissenb5894372019-01-10 14:44:13 -0800271 String8 str;
272 char timeString[32];
273 strftime(timeString, sizeof(timeString), "%m-%d %T", localtime(&when));
274 str.append(timeString);
275 str.append(": ");
276 str.append(name);
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800277 str.append(" for mime ");
278 str.append(mime);
279 str.append(", source ");
280 str.append(sourceDescription);
281 str.append(String8::format(", pid %d: ", owner));
282 if (extractor.promote() == NULL) {
283 str.append("deleted\n");
284 } else {
285 str.append("active\n");
286 }
287 for (size_t i = 0; i < tracks.size(); i++) {
288 const String8 desc = trackDescriptions.itemAt(i);
289 str.appendFormat(" track {%s} ", desc.string());
Marco Nelissen460b7e82016-12-19 14:06:30 -0800290 wp<IMediaSource> wSource = tracks.itemAt(i);
291 if (wSource == NULL) {
292 str.append(": null\n");
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800293 } else {
Marco Nelissen460b7e82016-12-19 14:06:30 -0800294 const sp<IMediaSource> source = wSource.promote();
295 if (source == NULL) {
296 str.append(": deleted\n");
297 } else {
298 str.appendFormat(": active\n");
299 }
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800300 }
301 }
302 return str;
303}
304
Marco Nelissen8d27f5e2016-04-14 08:06:26 -0700305static Vector<ExtractorInstance> sExtractors;
306static Mutex sExtractorsLock;
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800307
308void registerMediaSource(
309 const sp<IMediaExtractor> &ex,
310 const sp<IMediaSource> &source) {
Marco Nelissen8d27f5e2016-04-14 08:06:26 -0700311 Mutex::Autolock lock(sExtractorsLock);
312 for (size_t i = 0; i < sExtractors.size(); i++) {
313 ExtractorInstance &instance = sExtractors.editItemAt(i);
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800314 sp<IMediaExtractor> extractor = instance.extractor.promote();
315 if (extractor != NULL && extractor == ex) {
316 if (instance.tracks.size() > 5) {
317 instance.tracks.resize(5);
Marco Nelissen460b7e82016-12-19 14:06:30 -0800318 instance.trackDescriptions.resize(5);
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800319 }
320 instance.tracks.push_front(source);
Marco Nelissen460b7e82016-12-19 14:06:30 -0800321 if (source != NULL) {
322 instance.trackDescriptions.push_front(source->getFormat()->toString());
323 } else {
324 instance.trackDescriptions.push_front(String8::empty());
325 }
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800326 break;
327 }
328 }
329}
330
331void registerMediaExtractor(
332 const sp<IMediaExtractor> &extractor,
Marco Nelissena3214692016-05-06 10:57:20 -0700333 const sp<DataSource> &source,
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800334 const char *mime) {
335 ExtractorInstance ex;
336 ex.mime = mime == NULL ? "NULL" : mime;
337 ex.name = extractor->name();
338 ex.sourceDescription = source->toString();
339 ex.owner = IPCThreadState::self()->getCallingPid();
340 ex.extractor = extractor;
Marco Nelissenb5894372019-01-10 14:44:13 -0800341 ex.when = time(NULL);
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800342
Marco Nelissen8d27f5e2016-04-14 08:06:26 -0700343 {
344 Mutex::Autolock lock(sExtractorsLock);
345 if (sExtractors.size() > 10) {
346 sExtractors.resize(10);
347 }
348 sExtractors.push_front(ex);
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800349 }
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800350}
351
352status_t dumpExtractors(int fd, const Vector<String16>&) {
353 String8 out;
ray-cy.leeb596c342017-03-07 18:25:15 +0800354 const IPCThreadState* ipc = IPCThreadState::self();
355 const int pid = ipc->getCallingPid();
356 const int uid = ipc->getCallingUid();
357 if (!PermissionCache::checkPermission(String16("android.permission.DUMP"), pid, uid)) {
358 out.appendFormat("Permission Denial: "
359 "can't dump MediaExtractor from pid=%d, uid=%d\n", pid, uid);
360 } else {
361 out.append("Recent extractors, most recent first:\n");
362 {
363 Mutex::Autolock lock(sExtractorsLock);
364 for (size_t i = 0; i < sExtractors.size(); i++) {
365 const ExtractorInstance &instance = sExtractors.itemAt(i);
366 out.append(" ");
367 out.append(instance.toString());
368 }
Marco Nelissen8d27f5e2016-04-14 08:06:26 -0700369 }
Marco Nelissen69d3d8a2016-03-07 13:20:01 -0800370 }
371 write(fd, out.string(), out.size());
372 return OK;
373}
374
Marco Nelissenb2487f02015-09-01 13:23:23 -0700375
376} // namespace android
377