xref: /aosp_15_r20/frameworks/av/media/libmedia/IMediaExtractor.cpp (revision ec779b8e0859a360c3d303172224686826e6e0e1)
1*ec779b8eSAndroid Build Coastguard Worker /*
2*ec779b8eSAndroid Build Coastguard Worker  * Copyright (C) 2009 The Android Open Source Project
3*ec779b8eSAndroid Build Coastguard Worker  *
4*ec779b8eSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*ec779b8eSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*ec779b8eSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*ec779b8eSAndroid Build Coastguard Worker  *
8*ec779b8eSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*ec779b8eSAndroid Build Coastguard Worker  *
10*ec779b8eSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*ec779b8eSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*ec779b8eSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*ec779b8eSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*ec779b8eSAndroid Build Coastguard Worker  * limitations under the License.
15*ec779b8eSAndroid Build Coastguard Worker  */
16*ec779b8eSAndroid Build Coastguard Worker 
17*ec779b8eSAndroid Build Coastguard Worker //#define LOG_NDEBUG 0
18*ec779b8eSAndroid Build Coastguard Worker #define LOG_TAG "BpMediaExtractor"
19*ec779b8eSAndroid Build Coastguard Worker #include <utils/Log.h>
20*ec779b8eSAndroid Build Coastguard Worker 
21*ec779b8eSAndroid Build Coastguard Worker #include <stdint.h>
22*ec779b8eSAndroid Build Coastguard Worker #include <time.h>
23*ec779b8eSAndroid Build Coastguard Worker #include <sys/types.h>
24*ec779b8eSAndroid Build Coastguard Worker 
25*ec779b8eSAndroid Build Coastguard Worker #include <binder/IPCThreadState.h>
26*ec779b8eSAndroid Build Coastguard Worker #include <binder/Parcel.h>
27*ec779b8eSAndroid Build Coastguard Worker #include <binder/PermissionCache.h>
28*ec779b8eSAndroid Build Coastguard Worker #include <android/IMediaExtractor.h>
29*ec779b8eSAndroid Build Coastguard Worker #include <media/stagefright/MetaData.h>
30*ec779b8eSAndroid Build Coastguard Worker 
31*ec779b8eSAndroid Build Coastguard Worker namespace android {
32*ec779b8eSAndroid Build Coastguard Worker 
33*ec779b8eSAndroid Build Coastguard Worker enum {
34*ec779b8eSAndroid Build Coastguard Worker     COUNTTRACKS = IBinder::FIRST_CALL_TRANSACTION,
35*ec779b8eSAndroid Build Coastguard Worker     GETTRACK,
36*ec779b8eSAndroid Build Coastguard Worker     GETTRACKMETADATA,
37*ec779b8eSAndroid Build Coastguard Worker     GETMETADATA,
38*ec779b8eSAndroid Build Coastguard Worker     FLAGS,
39*ec779b8eSAndroid Build Coastguard Worker     SETMEDIACAS,
40*ec779b8eSAndroid Build Coastguard Worker     NAME,
41*ec779b8eSAndroid Build Coastguard Worker     GETMETRICS,
42*ec779b8eSAndroid Build Coastguard Worker     SETENTRYPOINT,
43*ec779b8eSAndroid Build Coastguard Worker     SETLOGSESSIONID
44*ec779b8eSAndroid Build Coastguard Worker };
45*ec779b8eSAndroid Build Coastguard Worker 
46*ec779b8eSAndroid Build Coastguard Worker class BpMediaExtractor : public BpInterface<IMediaExtractor> {
47*ec779b8eSAndroid Build Coastguard Worker public:
BpMediaExtractor(const sp<IBinder> & impl)48*ec779b8eSAndroid Build Coastguard Worker     explicit BpMediaExtractor(const sp<IBinder>& impl)
49*ec779b8eSAndroid Build Coastguard Worker         : BpInterface<IMediaExtractor>(impl)
50*ec779b8eSAndroid Build Coastguard Worker     {
51*ec779b8eSAndroid Build Coastguard Worker     }
52*ec779b8eSAndroid Build Coastguard Worker 
countTracks()53*ec779b8eSAndroid Build Coastguard Worker     virtual size_t countTracks() {
54*ec779b8eSAndroid Build Coastguard Worker         ALOGV("countTracks");
55*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
56*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
57*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(COUNTTRACKS, data, &reply);
58*ec779b8eSAndroid Build Coastguard Worker         size_t numTracks = 0;
59*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
60*ec779b8eSAndroid Build Coastguard Worker             numTracks = reply.readUint32();
61*ec779b8eSAndroid Build Coastguard Worker         }
62*ec779b8eSAndroid Build Coastguard Worker         return numTracks;
63*ec779b8eSAndroid Build Coastguard Worker     }
getTrack(size_t index)64*ec779b8eSAndroid Build Coastguard Worker     virtual sp<IMediaSource> getTrack(size_t index) {
65*ec779b8eSAndroid Build Coastguard Worker         ALOGV("getTrack(%zu)", index);
66*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
67*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
68*ec779b8eSAndroid Build Coastguard Worker         data.writeUint32(index);
69*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(GETTRACK, data, &reply);
70*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
71*ec779b8eSAndroid Build Coastguard Worker             return interface_cast<IMediaSource>(reply.readStrongBinder());
72*ec779b8eSAndroid Build Coastguard Worker         }
73*ec779b8eSAndroid Build Coastguard Worker         return NULL;
74*ec779b8eSAndroid Build Coastguard Worker     }
75*ec779b8eSAndroid Build Coastguard Worker 
getTrackMetaData(size_t index,uint32_t flags)76*ec779b8eSAndroid Build Coastguard Worker     virtual sp<MetaData> getTrackMetaData(
77*ec779b8eSAndroid Build Coastguard Worker             size_t index, uint32_t flags) {
78*ec779b8eSAndroid Build Coastguard Worker         ALOGV("getTrackMetaData(%zu, %u)", index, flags);
79*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
80*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
81*ec779b8eSAndroid Build Coastguard Worker         data.writeUint32(index);
82*ec779b8eSAndroid Build Coastguard Worker         data.writeUint32(flags);
83*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(GETTRACKMETADATA, data, &reply);
84*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
85*ec779b8eSAndroid Build Coastguard Worker             return MetaData::createFromParcel(reply);
86*ec779b8eSAndroid Build Coastguard Worker         }
87*ec779b8eSAndroid Build Coastguard Worker         return NULL;
88*ec779b8eSAndroid Build Coastguard Worker     }
89*ec779b8eSAndroid Build Coastguard Worker 
getMetaData()90*ec779b8eSAndroid Build Coastguard Worker     virtual sp<MetaData> getMetaData() {
91*ec779b8eSAndroid Build Coastguard Worker         ALOGV("getMetaData");
92*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
93*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
94*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(GETMETADATA, data, &reply);
95*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
96*ec779b8eSAndroid Build Coastguard Worker             return MetaData::createFromParcel(reply);
97*ec779b8eSAndroid Build Coastguard Worker         }
98*ec779b8eSAndroid Build Coastguard Worker         return NULL;
99*ec779b8eSAndroid Build Coastguard Worker     }
100*ec779b8eSAndroid Build Coastguard Worker 
getMetrics(Parcel * reply)101*ec779b8eSAndroid Build Coastguard Worker     virtual status_t getMetrics(Parcel * reply) {
102*ec779b8eSAndroid Build Coastguard Worker         Parcel data;
103*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
104*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(GETMETRICS, data, reply);
105*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
106*ec779b8eSAndroid Build Coastguard Worker             return OK;
107*ec779b8eSAndroid Build Coastguard Worker         }
108*ec779b8eSAndroid Build Coastguard Worker         return UNKNOWN_ERROR;
109*ec779b8eSAndroid Build Coastguard Worker     }
110*ec779b8eSAndroid Build Coastguard Worker 
flags() const111*ec779b8eSAndroid Build Coastguard Worker     virtual uint32_t flags() const {
112*ec779b8eSAndroid Build Coastguard Worker         ALOGV("flags");
113*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
114*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
115*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(FLAGS, data, &reply);
116*ec779b8eSAndroid Build Coastguard Worker         int flgs = 0;
117*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
118*ec779b8eSAndroid Build Coastguard Worker             flgs = reply.readUint32();
119*ec779b8eSAndroid Build Coastguard Worker         }
120*ec779b8eSAndroid Build Coastguard Worker         return flgs;
121*ec779b8eSAndroid Build Coastguard Worker     }
122*ec779b8eSAndroid Build Coastguard Worker 
setMediaCas(const HInterfaceToken & casToken)123*ec779b8eSAndroid Build Coastguard Worker     virtual status_t setMediaCas(const HInterfaceToken &casToken) {
124*ec779b8eSAndroid Build Coastguard Worker         ALOGV("setMediaCas");
125*ec779b8eSAndroid Build Coastguard Worker 
126*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
127*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
128*ec779b8eSAndroid Build Coastguard Worker         data.writeByteVector(casToken);
129*ec779b8eSAndroid Build Coastguard Worker 
130*ec779b8eSAndroid Build Coastguard Worker         status_t err = remote()->transact(SETMEDIACAS, data, &reply);
131*ec779b8eSAndroid Build Coastguard Worker         if (err != NO_ERROR) {
132*ec779b8eSAndroid Build Coastguard Worker             return err;
133*ec779b8eSAndroid Build Coastguard Worker         }
134*ec779b8eSAndroid Build Coastguard Worker         return reply.readInt32();
135*ec779b8eSAndroid Build Coastguard Worker     }
136*ec779b8eSAndroid Build Coastguard Worker 
name()137*ec779b8eSAndroid Build Coastguard Worker     virtual String8 name() {
138*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
139*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
140*ec779b8eSAndroid Build Coastguard Worker         status_t ret = remote()->transact(NAME, data, &reply);
141*ec779b8eSAndroid Build Coastguard Worker         String8 nm;
142*ec779b8eSAndroid Build Coastguard Worker         if (ret == NO_ERROR) {
143*ec779b8eSAndroid Build Coastguard Worker             nm = reply.readString8();
144*ec779b8eSAndroid Build Coastguard Worker         }
145*ec779b8eSAndroid Build Coastguard Worker         return nm;
146*ec779b8eSAndroid Build Coastguard Worker     }
147*ec779b8eSAndroid Build Coastguard Worker 
setEntryPoint(EntryPoint entryPoint)148*ec779b8eSAndroid Build Coastguard Worker     virtual status_t setEntryPoint(EntryPoint entryPoint) {
149*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
150*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
151*ec779b8eSAndroid Build Coastguard Worker         data.writeInt32(static_cast<int32_t>(entryPoint));
152*ec779b8eSAndroid Build Coastguard Worker         return remote()->transact(SETENTRYPOINT, data, &reply);
153*ec779b8eSAndroid Build Coastguard Worker     }
154*ec779b8eSAndroid Build Coastguard Worker 
setLogSessionId(const String8 & logSessionId)155*ec779b8eSAndroid Build Coastguard Worker     virtual status_t setLogSessionId(const String8& logSessionId) {
156*ec779b8eSAndroid Build Coastguard Worker         Parcel data, reply;
157*ec779b8eSAndroid Build Coastguard Worker         data.writeInterfaceToken(BpMediaExtractor::getInterfaceDescriptor());
158*ec779b8eSAndroid Build Coastguard Worker         data.writeString8(logSessionId);
159*ec779b8eSAndroid Build Coastguard Worker         return remote()->transact(SETLOGSESSIONID, data, &reply);
160*ec779b8eSAndroid Build Coastguard Worker     }
161*ec779b8eSAndroid Build Coastguard Worker };
162*ec779b8eSAndroid Build Coastguard Worker 
163*ec779b8eSAndroid Build Coastguard Worker IMPLEMENT_META_INTERFACE(MediaExtractor, "android.media.IMediaExtractor");
164*ec779b8eSAndroid Build Coastguard Worker 
165*ec779b8eSAndroid Build Coastguard Worker #undef LOG_TAG
166*ec779b8eSAndroid Build Coastguard Worker #define LOG_TAG "BnMediaExtractor"
167*ec779b8eSAndroid Build Coastguard Worker 
onTransact(uint32_t code,const Parcel & data,Parcel * reply,uint32_t flags)168*ec779b8eSAndroid Build Coastguard Worker status_t BnMediaExtractor::onTransact(
169*ec779b8eSAndroid Build Coastguard Worker     uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
170*ec779b8eSAndroid Build Coastguard Worker {
171*ec779b8eSAndroid Build Coastguard Worker     switch (code) {
172*ec779b8eSAndroid Build Coastguard Worker         case COUNTTRACKS: {
173*ec779b8eSAndroid Build Coastguard Worker             ALOGV("countTracks");
174*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
175*ec779b8eSAndroid Build Coastguard Worker             size_t numTracks = countTracks();
176*ec779b8eSAndroid Build Coastguard Worker             if (numTracks > INT32_MAX) {
177*ec779b8eSAndroid Build Coastguard Worker                 numTracks = 0;
178*ec779b8eSAndroid Build Coastguard Worker             }
179*ec779b8eSAndroid Build Coastguard Worker             reply->writeUint32(uint32_t(numTracks));
180*ec779b8eSAndroid Build Coastguard Worker             return NO_ERROR;
181*ec779b8eSAndroid Build Coastguard Worker         }
182*ec779b8eSAndroid Build Coastguard Worker         case GETTRACK: {
183*ec779b8eSAndroid Build Coastguard Worker             ALOGV("getTrack()");
184*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
185*ec779b8eSAndroid Build Coastguard Worker             uint32_t idx;
186*ec779b8eSAndroid Build Coastguard Worker             if (data.readUint32(&idx) == NO_ERROR) {
187*ec779b8eSAndroid Build Coastguard Worker                 const sp<IMediaSource> track = getTrack(size_t(idx));
188*ec779b8eSAndroid Build Coastguard Worker                 registerMediaSource(this, track);
189*ec779b8eSAndroid Build Coastguard Worker                 return reply->writeStrongBinder(IInterface::asBinder(track));
190*ec779b8eSAndroid Build Coastguard Worker             }
191*ec779b8eSAndroid Build Coastguard Worker             return UNKNOWN_ERROR;
192*ec779b8eSAndroid Build Coastguard Worker         }
193*ec779b8eSAndroid Build Coastguard Worker         case GETTRACKMETADATA: {
194*ec779b8eSAndroid Build Coastguard Worker             ALOGV("getTrackMetaData");
195*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
196*ec779b8eSAndroid Build Coastguard Worker             uint32_t idx;
197*ec779b8eSAndroid Build Coastguard Worker             uint32_t flags;
198*ec779b8eSAndroid Build Coastguard Worker             if (data.readUint32(&idx) == NO_ERROR &&
199*ec779b8eSAndroid Build Coastguard Worker                     data.readUint32(&flags) == NO_ERROR) {
200*ec779b8eSAndroid Build Coastguard Worker                 sp<MetaData> meta = getTrackMetaData(idx, flags);
201*ec779b8eSAndroid Build Coastguard Worker                 if (meta == NULL) {
202*ec779b8eSAndroid Build Coastguard Worker                     return UNKNOWN_ERROR;
203*ec779b8eSAndroid Build Coastguard Worker                 }
204*ec779b8eSAndroid Build Coastguard Worker                 meta->writeToParcel(*reply);
205*ec779b8eSAndroid Build Coastguard Worker                 return NO_ERROR;
206*ec779b8eSAndroid Build Coastguard Worker             }
207*ec779b8eSAndroid Build Coastguard Worker             return UNKNOWN_ERROR;
208*ec779b8eSAndroid Build Coastguard Worker         }
209*ec779b8eSAndroid Build Coastguard Worker         case GETMETADATA: {
210*ec779b8eSAndroid Build Coastguard Worker             ALOGV("getMetaData");
211*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
212*ec779b8eSAndroid Build Coastguard Worker             sp<MetaData> meta = getMetaData();
213*ec779b8eSAndroid Build Coastguard Worker             if (meta != NULL) {
214*ec779b8eSAndroid Build Coastguard Worker                 meta->writeToParcel(*reply);
215*ec779b8eSAndroid Build Coastguard Worker                 return NO_ERROR;
216*ec779b8eSAndroid Build Coastguard Worker             }
217*ec779b8eSAndroid Build Coastguard Worker             return UNKNOWN_ERROR;
218*ec779b8eSAndroid Build Coastguard Worker         }
219*ec779b8eSAndroid Build Coastguard Worker         case GETMETRICS: {
220*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
221*ec779b8eSAndroid Build Coastguard Worker             status_t ret = getMetrics(reply);
222*ec779b8eSAndroid Build Coastguard Worker             return ret;
223*ec779b8eSAndroid Build Coastguard Worker         }
224*ec779b8eSAndroid Build Coastguard Worker         case FLAGS: {
225*ec779b8eSAndroid Build Coastguard Worker             ALOGV("flags");
226*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
227*ec779b8eSAndroid Build Coastguard Worker             reply->writeUint32(this->flags());
228*ec779b8eSAndroid Build Coastguard Worker             return NO_ERROR;
229*ec779b8eSAndroid Build Coastguard Worker         }
230*ec779b8eSAndroid Build Coastguard Worker         case SETMEDIACAS: {
231*ec779b8eSAndroid Build Coastguard Worker             ALOGV("setMediaCas");
232*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
233*ec779b8eSAndroid Build Coastguard Worker 
234*ec779b8eSAndroid Build Coastguard Worker             HInterfaceToken casToken;
235*ec779b8eSAndroid Build Coastguard Worker             status_t err = data.readByteVector(&casToken);
236*ec779b8eSAndroid Build Coastguard Worker             if (err != NO_ERROR) {
237*ec779b8eSAndroid Build Coastguard Worker                 ALOGE("Error reading casToken from parcel");
238*ec779b8eSAndroid Build Coastguard Worker                 return err;
239*ec779b8eSAndroid Build Coastguard Worker             }
240*ec779b8eSAndroid Build Coastguard Worker 
241*ec779b8eSAndroid Build Coastguard Worker             reply->writeInt32(setMediaCas(casToken));
242*ec779b8eSAndroid Build Coastguard Worker             return OK;
243*ec779b8eSAndroid Build Coastguard Worker         }
244*ec779b8eSAndroid Build Coastguard Worker         case NAME: {
245*ec779b8eSAndroid Build Coastguard Worker             ALOGV("name");
246*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
247*ec779b8eSAndroid Build Coastguard Worker             String8 nm = name();
248*ec779b8eSAndroid Build Coastguard Worker             reply->writeString8(nm);
249*ec779b8eSAndroid Build Coastguard Worker             return NO_ERROR;
250*ec779b8eSAndroid Build Coastguard Worker         }
251*ec779b8eSAndroid Build Coastguard Worker         case SETENTRYPOINT: {
252*ec779b8eSAndroid Build Coastguard Worker             ALOGV("setEntryPoint");
253*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
254*ec779b8eSAndroid Build Coastguard Worker             int32_t entryPoint;
255*ec779b8eSAndroid Build Coastguard Worker             status_t err = data.readInt32(&entryPoint);
256*ec779b8eSAndroid Build Coastguard Worker             if (err == OK) {
257*ec779b8eSAndroid Build Coastguard Worker                 setEntryPoint(EntryPoint(entryPoint));
258*ec779b8eSAndroid Build Coastguard Worker             }
259*ec779b8eSAndroid Build Coastguard Worker             return err;
260*ec779b8eSAndroid Build Coastguard Worker         }
261*ec779b8eSAndroid Build Coastguard Worker         case SETLOGSESSIONID: {
262*ec779b8eSAndroid Build Coastguard Worker             ALOGV("setLogSessionId");
263*ec779b8eSAndroid Build Coastguard Worker             CHECK_INTERFACE(IMediaExtractor, data, reply);
264*ec779b8eSAndroid Build Coastguard Worker             String8 logSessionId;
265*ec779b8eSAndroid Build Coastguard Worker             status_t status = data.readString8(&logSessionId);
266*ec779b8eSAndroid Build Coastguard Worker             if (status == OK) {
267*ec779b8eSAndroid Build Coastguard Worker               setLogSessionId(logSessionId);
268*ec779b8eSAndroid Build Coastguard Worker             }
269*ec779b8eSAndroid Build Coastguard Worker             return status;
270*ec779b8eSAndroid Build Coastguard Worker         }
271*ec779b8eSAndroid Build Coastguard Worker         default:
272*ec779b8eSAndroid Build Coastguard Worker             return BBinder::onTransact(code, data, reply, flags);
273*ec779b8eSAndroid Build Coastguard Worker     }
274*ec779b8eSAndroid Build Coastguard Worker }
275*ec779b8eSAndroid Build Coastguard Worker 
276*ec779b8eSAndroid Build Coastguard Worker typedef struct {
277*ec779b8eSAndroid Build Coastguard Worker     String8 mime;
278*ec779b8eSAndroid Build Coastguard Worker     String8 name;
279*ec779b8eSAndroid Build Coastguard Worker     String8 sourceDescription;
280*ec779b8eSAndroid Build Coastguard Worker     pid_t owner;
281*ec779b8eSAndroid Build Coastguard Worker     wp<IMediaExtractor> extractor;
282*ec779b8eSAndroid Build Coastguard Worker     Vector<wp<IMediaSource>> tracks;
283*ec779b8eSAndroid Build Coastguard Worker     Vector<String8> trackDescriptions;
284*ec779b8eSAndroid Build Coastguard Worker     String8 toString() const;
285*ec779b8eSAndroid Build Coastguard Worker     time_t when;
286*ec779b8eSAndroid Build Coastguard Worker } ExtractorInstance;
287*ec779b8eSAndroid Build Coastguard Worker 
toString() const288*ec779b8eSAndroid Build Coastguard Worker String8 ExtractorInstance::toString() const {
289*ec779b8eSAndroid Build Coastguard Worker     String8 str;
290*ec779b8eSAndroid Build Coastguard Worker     char timeString[32];
291*ec779b8eSAndroid Build Coastguard Worker     strftime(timeString, sizeof(timeString), "%m-%d %T", localtime(&when));
292*ec779b8eSAndroid Build Coastguard Worker     str.append(timeString);
293*ec779b8eSAndroid Build Coastguard Worker     str.append(": ");
294*ec779b8eSAndroid Build Coastguard Worker     str.append(name);
295*ec779b8eSAndroid Build Coastguard Worker     str.append(" for mime ");
296*ec779b8eSAndroid Build Coastguard Worker     str.append(mime);
297*ec779b8eSAndroid Build Coastguard Worker     str.append(", source ");
298*ec779b8eSAndroid Build Coastguard Worker     str.append(sourceDescription);
299*ec779b8eSAndroid Build Coastguard Worker     str.append(String8::format(", pid %d: ", owner));
300*ec779b8eSAndroid Build Coastguard Worker     if (extractor.promote() == NULL) {
301*ec779b8eSAndroid Build Coastguard Worker         str.append("deleted\n");
302*ec779b8eSAndroid Build Coastguard Worker     } else {
303*ec779b8eSAndroid Build Coastguard Worker         str.append("active\n");
304*ec779b8eSAndroid Build Coastguard Worker     }
305*ec779b8eSAndroid Build Coastguard Worker     for (size_t i = 0; i < tracks.size(); i++) {
306*ec779b8eSAndroid Build Coastguard Worker         const String8 desc = trackDescriptions.itemAt(i);
307*ec779b8eSAndroid Build Coastguard Worker         str.appendFormat("    track {%s} ", desc.c_str());
308*ec779b8eSAndroid Build Coastguard Worker         wp<IMediaSource> wSource = tracks.itemAt(i);
309*ec779b8eSAndroid Build Coastguard Worker         if (wSource == NULL) {
310*ec779b8eSAndroid Build Coastguard Worker             str.append(": null\n");
311*ec779b8eSAndroid Build Coastguard Worker         } else {
312*ec779b8eSAndroid Build Coastguard Worker             const sp<IMediaSource> source = wSource.promote();
313*ec779b8eSAndroid Build Coastguard Worker             if (source == NULL) {
314*ec779b8eSAndroid Build Coastguard Worker                 str.append(": deleted\n");
315*ec779b8eSAndroid Build Coastguard Worker             } else {
316*ec779b8eSAndroid Build Coastguard Worker                 str.appendFormat(": active\n");
317*ec779b8eSAndroid Build Coastguard Worker             }
318*ec779b8eSAndroid Build Coastguard Worker         }
319*ec779b8eSAndroid Build Coastguard Worker     }
320*ec779b8eSAndroid Build Coastguard Worker     return str;
321*ec779b8eSAndroid Build Coastguard Worker }
322*ec779b8eSAndroid Build Coastguard Worker 
323*ec779b8eSAndroid Build Coastguard Worker static Vector<ExtractorInstance> sExtractors;
324*ec779b8eSAndroid Build Coastguard Worker static Mutex sExtractorsLock;
325*ec779b8eSAndroid Build Coastguard Worker 
registerMediaSource(const sp<IMediaExtractor> & ex,const sp<IMediaSource> & source)326*ec779b8eSAndroid Build Coastguard Worker void registerMediaSource(
327*ec779b8eSAndroid Build Coastguard Worker         const sp<IMediaExtractor> &ex,
328*ec779b8eSAndroid Build Coastguard Worker         const sp<IMediaSource> &source) {
329*ec779b8eSAndroid Build Coastguard Worker     Mutex::Autolock lock(sExtractorsLock);
330*ec779b8eSAndroid Build Coastguard Worker     for (size_t i = 0; i < sExtractors.size(); i++) {
331*ec779b8eSAndroid Build Coastguard Worker         ExtractorInstance &instance = sExtractors.editItemAt(i);
332*ec779b8eSAndroid Build Coastguard Worker         sp<IMediaExtractor> extractor = instance.extractor.promote();
333*ec779b8eSAndroid Build Coastguard Worker         if (extractor != NULL && extractor == ex) {
334*ec779b8eSAndroid Build Coastguard Worker             if (instance.tracks.size() > 5) {
335*ec779b8eSAndroid Build Coastguard Worker                 instance.tracks.resize(5);
336*ec779b8eSAndroid Build Coastguard Worker                 instance.trackDescriptions.resize(5);
337*ec779b8eSAndroid Build Coastguard Worker             }
338*ec779b8eSAndroid Build Coastguard Worker             instance.tracks.push_front(source);
339*ec779b8eSAndroid Build Coastguard Worker             if (source != NULL) {
340*ec779b8eSAndroid Build Coastguard Worker                 instance.trackDescriptions.push_front(source->getFormat()->toString());
341*ec779b8eSAndroid Build Coastguard Worker             } else {
342*ec779b8eSAndroid Build Coastguard Worker                 instance.trackDescriptions.push_front(String8());
343*ec779b8eSAndroid Build Coastguard Worker             }
344*ec779b8eSAndroid Build Coastguard Worker             break;
345*ec779b8eSAndroid Build Coastguard Worker         }
346*ec779b8eSAndroid Build Coastguard Worker     }
347*ec779b8eSAndroid Build Coastguard Worker }
348*ec779b8eSAndroid Build Coastguard Worker 
registerMediaExtractor(const sp<IMediaExtractor> & extractor,const sp<DataSource> & source,const char * mime)349*ec779b8eSAndroid Build Coastguard Worker void registerMediaExtractor(
350*ec779b8eSAndroid Build Coastguard Worker         const sp<IMediaExtractor> &extractor,
351*ec779b8eSAndroid Build Coastguard Worker         const sp<DataSource> &source,
352*ec779b8eSAndroid Build Coastguard Worker         const char *mime) {
353*ec779b8eSAndroid Build Coastguard Worker     ExtractorInstance ex;
354*ec779b8eSAndroid Build Coastguard Worker     ex.mime = mime == NULL ? "NULL" : mime;
355*ec779b8eSAndroid Build Coastguard Worker     ex.name = extractor->name();
356*ec779b8eSAndroid Build Coastguard Worker     ex.sourceDescription = source->toString();
357*ec779b8eSAndroid Build Coastguard Worker     ex.owner = IPCThreadState::self()->getCallingPid();
358*ec779b8eSAndroid Build Coastguard Worker     ex.extractor = extractor;
359*ec779b8eSAndroid Build Coastguard Worker     ex.when = time(NULL);
360*ec779b8eSAndroid Build Coastguard Worker 
361*ec779b8eSAndroid Build Coastguard Worker     {
362*ec779b8eSAndroid Build Coastguard Worker         Mutex::Autolock lock(sExtractorsLock);
363*ec779b8eSAndroid Build Coastguard Worker         if (sExtractors.size() > 10) {
364*ec779b8eSAndroid Build Coastguard Worker             sExtractors.resize(10);
365*ec779b8eSAndroid Build Coastguard Worker         }
366*ec779b8eSAndroid Build Coastguard Worker         sExtractors.push_front(ex);
367*ec779b8eSAndroid Build Coastguard Worker     }
368*ec779b8eSAndroid Build Coastguard Worker }
369*ec779b8eSAndroid Build Coastguard Worker 
dumpExtractors(int fd,const Vector<String16> &)370*ec779b8eSAndroid Build Coastguard Worker status_t dumpExtractors(int fd, const Vector<String16>&) {
371*ec779b8eSAndroid Build Coastguard Worker     String8 out;
372*ec779b8eSAndroid Build Coastguard Worker     const IPCThreadState* ipc = IPCThreadState::self();
373*ec779b8eSAndroid Build Coastguard Worker     const int pid = ipc->getCallingPid();
374*ec779b8eSAndroid Build Coastguard Worker     const int uid = ipc->getCallingUid();
375*ec779b8eSAndroid Build Coastguard Worker     if (!PermissionCache::checkPermission(String16("android.permission.DUMP"), pid, uid)) {
376*ec779b8eSAndroid Build Coastguard Worker         out.appendFormat("Permission Denial: "
377*ec779b8eSAndroid Build Coastguard Worker                 "can't dump MediaExtractor from pid=%d, uid=%d\n", pid, uid);
378*ec779b8eSAndroid Build Coastguard Worker     } else {
379*ec779b8eSAndroid Build Coastguard Worker         out.append("Recent extractors, most recent first:\n");
380*ec779b8eSAndroid Build Coastguard Worker         {
381*ec779b8eSAndroid Build Coastguard Worker             Mutex::Autolock lock(sExtractorsLock);
382*ec779b8eSAndroid Build Coastguard Worker             for (size_t i = 0; i < sExtractors.size(); i++) {
383*ec779b8eSAndroid Build Coastguard Worker                 const ExtractorInstance &instance = sExtractors.itemAt(i);
384*ec779b8eSAndroid Build Coastguard Worker                 out.append("  ");
385*ec779b8eSAndroid Build Coastguard Worker                 out.append(instance.toString());
386*ec779b8eSAndroid Build Coastguard Worker             }
387*ec779b8eSAndroid Build Coastguard Worker         }
388*ec779b8eSAndroid Build Coastguard Worker     }
389*ec779b8eSAndroid Build Coastguard Worker     write(fd, out.c_str(), out.size());
390*ec779b8eSAndroid Build Coastguard Worker     return OK;
391*ec779b8eSAndroid Build Coastguard Worker }
392*ec779b8eSAndroid Build Coastguard Worker 
393*ec779b8eSAndroid Build Coastguard Worker 
394*ec779b8eSAndroid Build Coastguard Worker }  // namespace android
395*ec779b8eSAndroid Build Coastguard Worker 
396