blob: 9ac91056f89d1c43d8e27858c6d45576625c213d [file] [log] [blame]
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001/*
2**
3** Copyright 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
18// Proxy for media player implementations
19
20//#define LOG_NDEBUG 0
21#define LOG_TAG "MediaPlayerService"
22#include <utils/Log.h>
23
24#include <sys/types.h>
25#include <sys/stat.h>
Gloria Wang7cf180c2011-02-19 18:37:57 -080026#include <sys/time.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080027#include <dirent.h>
28#include <unistd.h>
29
30#include <string.h>
Mathias Agopian6f74b0c2009-06-03 17:32:49 -070031
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080032#include <cutils/atomic.h>
Nicolas Catania14d27472009-07-13 14:37:49 -070033#include <cutils/properties.h> // for property_get
Mathias Agopian6f74b0c2009-06-03 17:32:49 -070034
35#include <utils/misc.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080036
Mathias Agopian75624082009-05-19 19:08:10 -070037#include <binder/IPCThreadState.h>
38#include <binder/IServiceManager.h>
39#include <binder/MemoryHeapBase.h>
40#include <binder/MemoryBase.h>
Mathias Agopianb1e7cd12013-02-14 17:11:27 -080041#include <gui/Surface.h>
Nicolas Catania1d187f12009-05-12 23:25:55 -070042#include <utils/Errors.h> // for status_t
43#include <utils/String8.h>
Marco Nelissen10dbb8e2009-09-20 10:42:13 -070044#include <utils/SystemClock.h>
Nicolas Catania1d187f12009-05-12 23:25:55 -070045#include <utils/Vector.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080046
Jeff Brown2013a542012-09-04 21:38:42 -070047#include <media/IRemoteDisplay.h>
48#include <media/IRemoteDisplayClient.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080049#include <media/MediaPlayerInterface.h>
50#include <media/mediarecorder.h>
51#include <media/MediaMetadataRetrieverInterface.h>
nikoa64c8c72009-07-20 15:07:26 -070052#include <media/Metadata.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080053#include <media/AudioTrack.h>
James Dong8635b7b2011-03-14 17:01:38 -070054#include <media/MemoryLeakTrackUtil.h>
Eric Laurent9cb839a2011-09-27 09:48:56 -070055#include <media/stagefright/MediaErrors.h>
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +010056#include <media/stagefright/AudioPlayer.h>
57#include <media/stagefright/foundation/ADebug.h>
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080058
Dima Zavin64760242011-05-11 14:15:23 -070059#include <system/audio.h>
Dima Zavinfce7a472011-04-19 22:30:36 -070060
Gloria Wang7cf180c2011-02-19 18:37:57 -080061#include <private/android_filesystem_config.h>
62
James Dong559bf282012-03-28 10:29:14 -070063#include "ActivityManager.h"
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080064#include "MediaRecorderClient.h"
65#include "MediaPlayerService.h"
66#include "MetadataRetrieverClient.h"
John Grossman44a7e422012-06-21 17:29:24 -070067#include "MediaPlayerFactory.h"
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -080068
69#include "MidiFile.h"
Nicolas Catania14d27472009-07-13 14:37:49 -070070#include "TestPlayerStub.h"
Andreas Huber20111aa2009-07-14 16:56:47 -070071#include "StagefrightPlayer.h"
Andreas Huberf9334412010-12-15 15:17:42 -080072#include "nuplayer/NuPlayerDriver.h"
Andreas Huber20111aa2009-07-14 16:56:47 -070073
Andreas Huber20111aa2009-07-14 16:56:47 -070074#include <OMX.h>
Nicolas Catania14d27472009-07-13 14:37:49 -070075
Andreas Hubered3e3e02012-03-26 11:13:27 -070076#include "Crypto.h"
Jeff Tinkercc82dc62013-02-08 10:18:35 -080077#include "Drm.h"
Andreas Huber59451f82012-09-18 10:36:32 -070078#include "HDCP.h"
Andreas Huberb7319a72013-05-29 14:20:52 -070079#include "HTTPBase.h"
Andreas Huber35213f12012-08-29 11:41:50 -070080#include "RemoteDisplay.h"
Andreas Hubered3e3e02012-03-26 11:13:27 -070081
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -070082namespace {
nikoa64c8c72009-07-20 15:07:26 -070083using android::media::Metadata;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -070084using android::status_t;
85using android::OK;
86using android::BAD_VALUE;
87using android::NOT_ENOUGH_DATA;
88using android::Parcel;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -070089
90// Max number of entries in the filter.
91const int kMaxFilterSize = 64; // I pulled that out of thin air.
92
nikoa64c8c72009-07-20 15:07:26 -070093// FIXME: Move all the metadata related function in the Metadata.cpp
nikod608a812009-07-16 16:39:53 -070094
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -070095
96// Unmarshall a filter from a Parcel.
97// Filter format in a parcel:
98//
99// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
100// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
101// | number of entries (n) |
102// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
103// | metadata type 1 |
104// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
105// | metadata type 2 |
106// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
107// ....
108// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
109// | metadata type n |
110// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
111//
112// @param p Parcel that should start with a filter.
113// @param[out] filter On exit contains the list of metadata type to be
114// filtered.
115// @param[out] status On exit contains the status code to be returned.
116// @return true if the parcel starts with a valid filter.
117bool unmarshallFilter(const Parcel& p,
nikoa64c8c72009-07-20 15:07:26 -0700118 Metadata::Filter *filter,
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700119 status_t *status)
120{
Nicolas Catania48290382009-07-10 13:53:06 -0700121 int32_t val;
122 if (p.readInt32(&val) != OK)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700123 {
Steve Block29357bc2012-01-06 19:20:56 +0000124 ALOGE("Failed to read filter's length");
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700125 *status = NOT_ENOUGH_DATA;
126 return false;
127 }
128
Nicolas Catania48290382009-07-10 13:53:06 -0700129 if( val > kMaxFilterSize || val < 0)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700130 {
Steve Block29357bc2012-01-06 19:20:56 +0000131 ALOGE("Invalid filter len %d", val);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700132 *status = BAD_VALUE;
133 return false;
134 }
135
Nicolas Catania48290382009-07-10 13:53:06 -0700136 const size_t num = val;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700137
138 filter->clear();
Nicolas Catania48290382009-07-10 13:53:06 -0700139 filter->setCapacity(num);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700140
nikoa64c8c72009-07-20 15:07:26 -0700141 size_t size = num * sizeof(Metadata::Type);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700142
Nicolas Catania48290382009-07-10 13:53:06 -0700143
144 if (p.dataAvail() < size)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700145 {
Steve Block29357bc2012-01-06 19:20:56 +0000146 ALOGE("Filter too short expected %d but got %d", size, p.dataAvail());
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700147 *status = NOT_ENOUGH_DATA;
148 return false;
149 }
150
nikoa64c8c72009-07-20 15:07:26 -0700151 const Metadata::Type *data =
152 static_cast<const Metadata::Type*>(p.readInplace(size));
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700153
Nicolas Catania48290382009-07-10 13:53:06 -0700154 if (NULL == data)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700155 {
Steve Block29357bc2012-01-06 19:20:56 +0000156 ALOGE("Filter had no data");
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700157 *status = BAD_VALUE;
158 return false;
159 }
160
161 // TODO: The stl impl of vector would be more efficient here
162 // because it degenerates into a memcpy on pod types. Try to
163 // replace later or use stl::set.
Nicolas Catania48290382009-07-10 13:53:06 -0700164 for (size_t i = 0; i < num; ++i)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700165 {
Nicolas Catania48290382009-07-10 13:53:06 -0700166 filter->add(*data);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700167 ++data;
168 }
169 *status = OK;
170 return true;
171}
172
Nicolas Catania48290382009-07-10 13:53:06 -0700173// @param filter Of metadata type.
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700174// @param val To be searched.
175// @return true if a match was found.
nikoa64c8c72009-07-20 15:07:26 -0700176bool findMetadata(const Metadata::Filter& filter, const int32_t val)
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700177{
178 // Deal with empty and ANY right away
179 if (filter.isEmpty()) return false;
nikoa64c8c72009-07-20 15:07:26 -0700180 if (filter[0] == Metadata::kAny) return true;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700181
Nicolas Catania48290382009-07-10 13:53:06 -0700182 return filter.indexOf(val) >= 0;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700183}
184
185} // anonymous namespace
186
187
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800188namespace android {
189
Dave Burked681bbb2011-08-30 14:39:17 +0100190static bool checkPermission(const char* permissionString) {
191#ifndef HAVE_ANDROID_OS
192 return true;
193#endif
194 if (getpid() == IPCThreadState::self()->getCallingPid()) return true;
195 bool ok = checkCallingPermission(String16(permissionString));
Steve Block29357bc2012-01-06 19:20:56 +0000196 if (!ok) ALOGE("Request requires %s", permissionString);
Dave Burked681bbb2011-08-30 14:39:17 +0100197 return ok;
198}
199
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800200// TODO: Find real cause of Audio/Video delay in PV framework and remove this workaround
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800201/* static */ int MediaPlayerService::AudioOutput::mMinBufferCount = 4;
202/* static */ bool MediaPlayerService::AudioOutput::mIsOnEmulator = false;
203
204void MediaPlayerService::instantiate() {
205 defaultServiceManager()->addService(
206 String16("media.player"), new MediaPlayerService());
207}
208
209MediaPlayerService::MediaPlayerService()
210{
Steve Block3856b092011-10-20 11:56:00 +0100211 ALOGV("MediaPlayerService created");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800212 mNextConnId = 1;
Gloria Wang9ee159b2011-02-24 14:51:45 -0800213
214 mBatteryAudio.refCount = 0;
215 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
216 mBatteryAudio.deviceOn[i] = 0;
217 mBatteryAudio.lastTime[i] = 0;
218 mBatteryAudio.totalTime[i] = 0;
219 }
220 // speaker is on by default
221 mBatteryAudio.deviceOn[SPEAKER] = 1;
John Grossman44a7e422012-06-21 17:29:24 -0700222
223 MediaPlayerFactory::registerBuiltinFactories();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800224}
225
226MediaPlayerService::~MediaPlayerService()
227{
Steve Block3856b092011-10-20 11:56:00 +0100228 ALOGV("MediaPlayerService destroyed");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800229}
230
Glenn Kastenf37971f2012-02-03 11:06:53 -0800231sp<IMediaRecorder> MediaPlayerService::createMediaRecorder()
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800232{
Glenn Kastenf37971f2012-02-03 11:06:53 -0800233 pid_t pid = IPCThreadState::self()->getCallingPid();
Gloria Wangdac6a312009-10-29 15:46:37 -0700234 sp<MediaRecorderClient> recorder = new MediaRecorderClient(this, pid);
235 wp<MediaRecorderClient> w = recorder;
236 Mutex::Autolock lock(mLock);
237 mMediaRecorderClients.add(w);
Steve Block3856b092011-10-20 11:56:00 +0100238 ALOGV("Create new media recorder client from pid %d", pid);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800239 return recorder;
240}
241
Gloria Wangdac6a312009-10-29 15:46:37 -0700242void MediaPlayerService::removeMediaRecorderClient(wp<MediaRecorderClient> client)
243{
244 Mutex::Autolock lock(mLock);
245 mMediaRecorderClients.remove(client);
Steve Block3856b092011-10-20 11:56:00 +0100246 ALOGV("Delete media recorder client");
Gloria Wangdac6a312009-10-29 15:46:37 -0700247}
248
Glenn Kastenf37971f2012-02-03 11:06:53 -0800249sp<IMediaMetadataRetriever> MediaPlayerService::createMetadataRetriever()
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800250{
Glenn Kastenf37971f2012-02-03 11:06:53 -0800251 pid_t pid = IPCThreadState::self()->getCallingPid();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800252 sp<MetadataRetrieverClient> retriever = new MetadataRetrieverClient(pid);
Steve Block3856b092011-10-20 11:56:00 +0100253 ALOGV("Create new media retriever from pid %d", pid);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800254 return retriever;
255}
256
Glenn Kastenf37971f2012-02-03 11:06:53 -0800257sp<IMediaPlayer> MediaPlayerService::create(const sp<IMediaPlayerClient>& client,
Dave Burked681bbb2011-08-30 14:39:17 +0100258 int audioSessionId)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800259{
Glenn Kastenf37971f2012-02-03 11:06:53 -0800260 pid_t pid = IPCThreadState::self()->getCallingPid();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800261 int32_t connId = android_atomic_inc(&mNextConnId);
Andreas Huber9b80c2b2011-06-30 15:47:02 -0700262
263 sp<Client> c = new Client(
264 this, pid, connId, client, audioSessionId,
265 IPCThreadState::self()->getCallingUid());
266
Steve Block3856b092011-10-20 11:56:00 +0100267 ALOGV("Create new client(%d) from pid %d, uid %d, ", connId, pid,
Dave Burked681bbb2011-08-30 14:39:17 +0100268 IPCThreadState::self()->getCallingUid());
269
270 wp<Client> w = c;
271 {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800272 Mutex::Autolock lock(mLock);
273 mClients.add(w);
274 }
Andreas Hubere2b10282010-11-23 11:41:34 -0800275 return c;
276}
277
Andreas Huber318ad9c2009-10-15 13:46:54 -0700278sp<IOMX> MediaPlayerService::getOMX() {
279 Mutex::Autolock autoLock(mLock);
280
281 if (mOMX.get() == NULL) {
282 mOMX = new OMX;
283 }
284
285 return mOMX;
Andreas Huber20111aa2009-07-14 16:56:47 -0700286}
287
Andreas Hubered3e3e02012-03-26 11:13:27 -0700288sp<ICrypto> MediaPlayerService::makeCrypto() {
Andreas Huber1bd139a2012-04-03 14:19:20 -0700289 return new Crypto;
Andreas Hubered3e3e02012-03-26 11:13:27 -0700290}
291
Jeff Tinkercc82dc62013-02-08 10:18:35 -0800292sp<IDrm> MediaPlayerService::makeDrm() {
293 return new Drm;
294}
295
Andreas Huber279dcd82013-01-30 10:41:25 -0800296sp<IHDCP> MediaPlayerService::makeHDCP(bool createEncryptionModule) {
297 return new HDCP(createEncryptionModule);
Andreas Huber59451f82012-09-18 10:36:32 -0700298}
299
Jeff Brown2013a542012-09-04 21:38:42 -0700300sp<IRemoteDisplay> MediaPlayerService::listenForRemoteDisplay(
301 const sp<IRemoteDisplayClient>& client, const String8& iface) {
Jeff Brownaba33d52012-09-07 17:38:58 -0700302 if (!checkPermission("android.permission.CONTROL_WIFI_DISPLAY")) {
303 return NULL;
304 }
305
Jeff Brownced24b32012-09-05 17:48:03 -0700306 return new RemoteDisplay(client, iface.string());
Jeff Brown2013a542012-09-04 21:38:42 -0700307}
308
Andreas Huberb7319a72013-05-29 14:20:52 -0700309status_t MediaPlayerService::updateProxyConfig(
310 const char *host, int32_t port, const char *exclusionList) {
311 return HTTPBase::UpdateProxyConfig(host, port, exclusionList);
312}
313
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800314status_t MediaPlayerService::AudioCache::dump(int fd, const Vector<String16>& args) const
315{
316 const size_t SIZE = 256;
317 char buffer[SIZE];
318 String8 result;
319
320 result.append(" AudioCache\n");
321 if (mHeap != 0) {
Eric Laurent3d00aa62013-09-24 09:53:27 -0700322 snprintf(buffer, 255, " heap base(%p), size(%d), flags(%d)\n",
323 mHeap->getBase(), mHeap->getSize(), mHeap->getFlags());
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800324 result.append(buffer);
325 }
Scott Fan7d409692013-04-28 10:13:54 +0800326 snprintf(buffer, 255, " msec per frame(%f), channel count(%d), format(%d), frame count(%zd)\n",
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800327 mMsecsPerFrame, mChannelCount, mFormat, mFrameCount);
328 result.append(buffer);
329 snprintf(buffer, 255, " sample rate(%d), size(%d), error(%d), command complete(%s)\n",
330 mSampleRate, mSize, mError, mCommandComplete?"true":"false");
331 result.append(buffer);
332 ::write(fd, result.string(), result.size());
333 return NO_ERROR;
334}
335
336status_t MediaPlayerService::AudioOutput::dump(int fd, const Vector<String16>& args) const
337{
338 const size_t SIZE = 256;
339 char buffer[SIZE];
340 String8 result;
341
342 result.append(" AudioOutput\n");
343 snprintf(buffer, 255, " stream type(%d), left - right volume(%f, %f)\n",
344 mStreamType, mLeftVolume, mRightVolume);
345 result.append(buffer);
346 snprintf(buffer, 255, " msec per frame(%f), latency (%d)\n",
Eric Laurentdb354e52012-03-05 17:27:11 -0800347 mMsecsPerFrame, (mTrack != 0) ? mTrack->latency() : -1);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800348 result.append(buffer);
Eric Laurent2beeb502010-07-16 07:43:46 -0700349 snprintf(buffer, 255, " aux effect id(%d), send level (%f)\n",
350 mAuxEffectId, mSendLevel);
351 result.append(buffer);
352
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800353 ::write(fd, result.string(), result.size());
354 if (mTrack != 0) {
355 mTrack->dump(fd, args);
356 }
357 return NO_ERROR;
358}
359
360status_t MediaPlayerService::Client::dump(int fd, const Vector<String16>& args) const
361{
362 const size_t SIZE = 256;
363 char buffer[SIZE];
364 String8 result;
365 result.append(" Client\n");
366 snprintf(buffer, 255, " pid(%d), connId(%d), status(%d), looping(%s)\n",
367 mPid, mConnId, mStatus, mLoop?"true": "false");
368 result.append(buffer);
369 write(fd, result.string(), result.size());
Andreas Hubera0b1d4b2011-06-07 15:52:25 -0700370 if (mPlayer != NULL) {
371 mPlayer->dump(fd, args);
372 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800373 if (mAudioOutput != 0) {
374 mAudioOutput->dump(fd, args);
375 }
376 write(fd, "\n", 1);
377 return NO_ERROR;
378}
379
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800380status_t MediaPlayerService::dump(int fd, const Vector<String16>& args)
381{
382 const size_t SIZE = 256;
383 char buffer[SIZE];
384 String8 result;
385 if (checkCallingPermission(String16("android.permission.DUMP")) == false) {
386 snprintf(buffer, SIZE, "Permission Denial: "
387 "can't dump MediaPlayerService from pid=%d, uid=%d\n",
388 IPCThreadState::self()->getCallingPid(),
389 IPCThreadState::self()->getCallingUid());
390 result.append(buffer);
391 } else {
392 Mutex::Autolock lock(mLock);
393 for (int i = 0, n = mClients.size(); i < n; ++i) {
394 sp<Client> c = mClients[i].promote();
395 if (c != 0) c->dump(fd, args);
396 }
James Dongb9141222010-07-08 11:16:11 -0700397 if (mMediaRecorderClients.size() == 0) {
398 result.append(" No media recorder client\n\n");
399 } else {
400 for (int i = 0, n = mMediaRecorderClients.size(); i < n; ++i) {
401 sp<MediaRecorderClient> c = mMediaRecorderClients[i].promote();
James Donge579e282011-10-18 22:29:20 -0700402 if (c != 0) {
403 snprintf(buffer, 255, " MediaRecorderClient pid(%d)\n", c->mPid);
404 result.append(buffer);
405 write(fd, result.string(), result.size());
406 result = "\n";
407 c->dump(fd, args);
408 }
James Dongb9141222010-07-08 11:16:11 -0700409 }
Gloria Wangdac6a312009-10-29 15:46:37 -0700410 }
411
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800412 result.append(" Files opened and/or mapped:\n");
Glenn Kasten0512ab52011-05-04 17:58:57 -0700413 snprintf(buffer, SIZE, "/proc/%d/maps", gettid());
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800414 FILE *f = fopen(buffer, "r");
415 if (f) {
416 while (!feof(f)) {
417 fgets(buffer, SIZE, f);
Marco Nelissen73ac1ee2012-05-07 15:36:32 -0700418 if (strstr(buffer, " /storage/") ||
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800419 strstr(buffer, " /system/sounds/") ||
Dave Sparks02fa8342010-09-27 16:55:18 -0700420 strstr(buffer, " /data/") ||
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800421 strstr(buffer, " /system/media/")) {
422 result.append(" ");
423 result.append(buffer);
424 }
425 }
426 fclose(f);
427 } else {
428 result.append("couldn't open ");
429 result.append(buffer);
430 result.append("\n");
431 }
432
Glenn Kasten0512ab52011-05-04 17:58:57 -0700433 snprintf(buffer, SIZE, "/proc/%d/fd", gettid());
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800434 DIR *d = opendir(buffer);
435 if (d) {
436 struct dirent *ent;
437 while((ent = readdir(d)) != NULL) {
438 if (strcmp(ent->d_name,".") && strcmp(ent->d_name,"..")) {
Glenn Kasten0512ab52011-05-04 17:58:57 -0700439 snprintf(buffer, SIZE, "/proc/%d/fd/%s", gettid(), ent->d_name);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800440 struct stat s;
441 if (lstat(buffer, &s) == 0) {
442 if ((s.st_mode & S_IFMT) == S_IFLNK) {
443 char linkto[256];
444 int len = readlink(buffer, linkto, sizeof(linkto));
445 if(len > 0) {
446 if(len > 255) {
447 linkto[252] = '.';
448 linkto[253] = '.';
449 linkto[254] = '.';
450 linkto[255] = 0;
451 } else {
452 linkto[len] = 0;
453 }
Marco Nelissen73ac1ee2012-05-07 15:36:32 -0700454 if (strstr(linkto, "/storage/") == linkto ||
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800455 strstr(linkto, "/system/sounds/") == linkto ||
Dave Sparks02fa8342010-09-27 16:55:18 -0700456 strstr(linkto, "/data/") == linkto ||
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800457 strstr(linkto, "/system/media/") == linkto) {
458 result.append(" ");
459 result.append(buffer);
460 result.append(" -> ");
461 result.append(linkto);
462 result.append("\n");
463 }
464 }
465 } else {
466 result.append(" unexpected type for ");
467 result.append(buffer);
468 result.append("\n");
469 }
470 }
471 }
472 }
473 closedir(d);
474 } else {
475 result.append("couldn't open ");
476 result.append(buffer);
477 result.append("\n");
478 }
479
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800480 bool dumpMem = false;
481 for (size_t i = 0; i < args.size(); i++) {
482 if (args[i] == String16("-m")) {
483 dumpMem = true;
484 }
485 }
486 if (dumpMem) {
James Dong8635b7b2011-03-14 17:01:38 -0700487 dumpMemoryAddresses(fd);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800488 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800489 }
490 write(fd, result.string(), result.size());
491 return NO_ERROR;
492}
493
494void MediaPlayerService::removeClient(wp<Client> client)
495{
496 Mutex::Autolock lock(mLock);
497 mClients.remove(client);
498}
499
Andreas Huber9b80c2b2011-06-30 15:47:02 -0700500MediaPlayerService::Client::Client(
501 const sp<MediaPlayerService>& service, pid_t pid,
502 int32_t connId, const sp<IMediaPlayerClient>& client,
503 int audioSessionId, uid_t uid)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800504{
Steve Block3856b092011-10-20 11:56:00 +0100505 ALOGV("Client(%d) constructor", connId);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800506 mPid = pid;
507 mConnId = connId;
508 mService = service;
509 mClient = client;
510 mLoop = false;
511 mStatus = NO_INIT;
Eric Laurenta514bdb2010-06-21 09:27:30 -0700512 mAudioSessionId = audioSessionId;
Andreas Huber9b80c2b2011-06-30 15:47:02 -0700513 mUID = uid;
John Grossmanc795b642012-02-22 15:38:35 -0800514 mRetransmitEndpointValid = false;
Eric Laurenta514bdb2010-06-21 09:27:30 -0700515
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800516#if CALLBACK_ANTAGONIZER
Steve Blockb8a80522011-12-20 16:23:08 +0000517 ALOGD("create Antagonizer");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800518 mAntagonizer = new Antagonizer(notify, this);
519#endif
520}
521
522MediaPlayerService::Client::~Client()
523{
Steve Block3856b092011-10-20 11:56:00 +0100524 ALOGV("Client(%d) destructor pid = %d", mConnId, mPid);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800525 mAudioOutput.clear();
526 wp<Client> client(this);
527 disconnect();
528 mService->removeClient(client);
529}
530
531void MediaPlayerService::Client::disconnect()
532{
Steve Block3856b092011-10-20 11:56:00 +0100533 ALOGV("disconnect(%d) from pid %d", mConnId, mPid);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800534 // grab local reference and clear main reference to prevent future
535 // access to object
536 sp<MediaPlayerBase> p;
537 {
538 Mutex::Autolock l(mLock);
539 p = mPlayer;
beanzdcfefde2012-11-05 09:51:43 +0800540 mClient.clear();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800541 }
Andreas Huber20111aa2009-07-14 16:56:47 -0700542
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800543 mPlayer.clear();
544
545 // clear the notification to prevent callbacks to dead client
546 // and reset the player. We assume the player will serialize
547 // access to itself if necessary.
548 if (p != 0) {
549 p->setNotifyCallback(0, 0);
550#if CALLBACK_ANTAGONIZER
Steve Blockb8a80522011-12-20 16:23:08 +0000551 ALOGD("kill Antagonizer");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800552 mAntagonizer->kill();
553#endif
554 p->reset();
555 }
556
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700557 disconnectNativeWindow();
558
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800559 IPCThreadState::self()->flushCommands();
560}
561
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800562sp<MediaPlayerBase> MediaPlayerService::Client::createPlayer(player_type playerType)
563{
564 // determine if we have the right player type
565 sp<MediaPlayerBase> p = mPlayer;
566 if ((p != NULL) && (p->playerType() != playerType)) {
Steve Block3856b092011-10-20 11:56:00 +0100567 ALOGV("delete player");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800568 p.clear();
569 }
570 if (p == NULL) {
John Grossman44a7e422012-06-21 17:29:24 -0700571 p = MediaPlayerFactory::createPlayer(playerType, this, notify);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800572 }
Andreas Huber9b80c2b2011-06-30 15:47:02 -0700573
Jason Simmonsdb29e522011-08-12 13:46:55 -0700574 if (p != NULL) {
575 p->setUID(mUID);
576 }
Andreas Huber9b80c2b2011-06-30 15:47:02 -0700577
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800578 return p;
579}
580
John Grossmanc795b642012-02-22 15:38:35 -0800581sp<MediaPlayerBase> MediaPlayerService::Client::setDataSource_pre(
582 player_type playerType)
583{
584 ALOGV("player type = %d", playerType);
585
586 // create the right type of player
587 sp<MediaPlayerBase> p = createPlayer(playerType);
588 if (p == NULL) {
589 return p;
590 }
591
592 if (!p->hardwareOutput()) {
Marco Nelissen462fd2f2013-01-14 14:12:05 -0800593 mAudioOutput = new AudioOutput(mAudioSessionId, IPCThreadState::self()->getCallingUid());
John Grossmanc795b642012-02-22 15:38:35 -0800594 static_cast<MediaPlayerInterface*>(p.get())->setAudioSink(mAudioOutput);
595 }
596
597 return p;
598}
599
600void MediaPlayerService::Client::setDataSource_post(
601 const sp<MediaPlayerBase>& p,
602 status_t status)
603{
604 ALOGV(" setDataSource");
605 mStatus = status;
606 if (mStatus != OK) {
607 ALOGE(" error: %d", mStatus);
608 return;
609 }
610
611 // Set the re-transmission endpoint if one was chosen.
612 if (mRetransmitEndpointValid) {
613 mStatus = p->setRetransmitEndpoint(&mRetransmitEndpoint);
614 if (mStatus != NO_ERROR) {
615 ALOGE("setRetransmitEndpoint error: %d", mStatus);
616 }
617 }
618
619 if (mStatus == OK) {
620 mPlayer = p;
621 }
622}
623
Andreas Huber2db84552010-01-28 11:19:57 -0800624status_t MediaPlayerService::Client::setDataSource(
625 const char *url, const KeyedVector<String8, String8> *headers)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800626{
Steve Block3856b092011-10-20 11:56:00 +0100627 ALOGV("setDataSource(%s)", url);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800628 if (url == NULL)
629 return UNKNOWN_ERROR;
630
Dave Burked681bbb2011-08-30 14:39:17 +0100631 if ((strncmp(url, "http://", 7) == 0) ||
632 (strncmp(url, "https://", 8) == 0) ||
633 (strncmp(url, "rtsp://", 7) == 0)) {
634 if (!checkPermission("android.permission.INTERNET")) {
635 return PERMISSION_DENIED;
636 }
637 }
638
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800639 if (strncmp(url, "content://", 10) == 0) {
640 // get a filedescriptor for the content Uri and
641 // pass it to the setDataSource(fd) method
642
643 String16 url16(url);
644 int fd = android::openContentProviderFile(url16);
645 if (fd < 0)
646 {
Steve Block29357bc2012-01-06 19:20:56 +0000647 ALOGE("Couldn't open fd for %s", url);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800648 return UNKNOWN_ERROR;
649 }
650 setDataSource(fd, 0, 0x7fffffffffLL); // this sets mStatus
651 close(fd);
652 return mStatus;
653 } else {
John Grossman44a7e422012-06-21 17:29:24 -0700654 player_type playerType = MediaPlayerFactory::getPlayerType(this, url);
John Grossmanc795b642012-02-22 15:38:35 -0800655 sp<MediaPlayerBase> p = setDataSource_pre(playerType);
656 if (p == NULL) {
657 return NO_INIT;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800658 }
659
John Grossmanc795b642012-02-22 15:38:35 -0800660 setDataSource_post(p, p->setDataSource(url, headers));
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800661 return mStatus;
662 }
663}
664
665status_t MediaPlayerService::Client::setDataSource(int fd, int64_t offset, int64_t length)
666{
Steve Block3856b092011-10-20 11:56:00 +0100667 ALOGV("setDataSource fd=%d, offset=%lld, length=%lld", fd, offset, length);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800668 struct stat sb;
669 int ret = fstat(fd, &sb);
670 if (ret != 0) {
Steve Block29357bc2012-01-06 19:20:56 +0000671 ALOGE("fstat(%d) failed: %d, %s", fd, ret, strerror(errno));
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800672 return UNKNOWN_ERROR;
673 }
674
Steve Block3856b092011-10-20 11:56:00 +0100675 ALOGV("st_dev = %llu", sb.st_dev);
676 ALOGV("st_mode = %u", sb.st_mode);
677 ALOGV("st_uid = %lu", sb.st_uid);
678 ALOGV("st_gid = %lu", sb.st_gid);
679 ALOGV("st_size = %llu", sb.st_size);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800680
681 if (offset >= sb.st_size) {
Steve Block29357bc2012-01-06 19:20:56 +0000682 ALOGE("offset error");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800683 ::close(fd);
684 return UNKNOWN_ERROR;
685 }
686 if (offset + length > sb.st_size) {
687 length = sb.st_size - offset;
Steve Block3856b092011-10-20 11:56:00 +0100688 ALOGV("calculated length = %lld", length);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800689 }
690
John Grossman44a7e422012-06-21 17:29:24 -0700691 player_type playerType = MediaPlayerFactory::getPlayerType(this,
692 fd,
693 offset,
694 length);
John Grossmanc795b642012-02-22 15:38:35 -0800695 sp<MediaPlayerBase> p = setDataSource_pre(playerType);
696 if (p == NULL) {
697 return NO_INIT;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800698 }
699
700 // now set data source
John Grossmanc795b642012-02-22 15:38:35 -0800701 setDataSource_post(p, p->setDataSource(fd, offset, length));
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800702 return mStatus;
703}
704
Andreas Hubere2b10282010-11-23 11:41:34 -0800705status_t MediaPlayerService::Client::setDataSource(
706 const sp<IStreamSource> &source) {
707 // create the right type of player
John Grossman44a7e422012-06-21 17:29:24 -0700708 player_type playerType = MediaPlayerFactory::getPlayerType(this, source);
John Grossmanc795b642012-02-22 15:38:35 -0800709 sp<MediaPlayerBase> p = setDataSource_pre(playerType);
Andreas Hubere2b10282010-11-23 11:41:34 -0800710 if (p == NULL) {
711 return NO_INIT;
712 }
713
Andreas Hubere2b10282010-11-23 11:41:34 -0800714 // now set data source
John Grossmanc795b642012-02-22 15:38:35 -0800715 setDataSource_post(p, p->setDataSource(source));
Andreas Hubere2b10282010-11-23 11:41:34 -0800716 return mStatus;
717}
718
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700719void MediaPlayerService::Client::disconnectNativeWindow() {
720 if (mConnectedWindow != NULL) {
721 status_t err = native_window_api_disconnect(mConnectedWindow.get(),
722 NATIVE_WINDOW_API_MEDIA);
723
724 if (err != OK) {
Steve Block5ff1dd52012-01-05 23:22:43 +0000725 ALOGW("native_window_api_disconnect returned an error: %s (%d)",
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700726 strerror(-err), err);
727 }
728 }
729 mConnectedWindow.clear();
730}
731
Glenn Kasten11731182011-02-08 17:26:17 -0800732status_t MediaPlayerService::Client::setVideoSurfaceTexture(
Andy McFadden484566c2012-12-18 09:46:54 -0800733 const sp<IGraphicBufferProducer>& bufferProducer)
Glenn Kasten11731182011-02-08 17:26:17 -0800734{
Andy McFadden484566c2012-12-18 09:46:54 -0800735 ALOGV("[%d] setVideoSurfaceTexture(%p)", mConnId, bufferProducer.get());
Glenn Kasten11731182011-02-08 17:26:17 -0800736 sp<MediaPlayerBase> p = getPlayer();
737 if (p == 0) return UNKNOWN_ERROR;
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700738
Andy McFadden484566c2012-12-18 09:46:54 -0800739 sp<IBinder> binder(bufferProducer == NULL ? NULL :
740 bufferProducer->asBinder());
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700741 if (mConnectedWindowBinder == binder) {
742 return OK;
743 }
744
745 sp<ANativeWindow> anw;
Andy McFadden484566c2012-12-18 09:46:54 -0800746 if (bufferProducer != NULL) {
Marco Nelissenee08f7e2013-09-16 13:30:01 -0700747 anw = new Surface(bufferProducer, true /* controlledByApp */);
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700748 status_t err = native_window_api_connect(anw.get(),
749 NATIVE_WINDOW_API_MEDIA);
750
751 if (err != OK) {
Steve Block29357bc2012-01-06 19:20:56 +0000752 ALOGE("setVideoSurfaceTexture failed: %d", err);
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700753 // Note that we must do the reset before disconnecting from the ANW.
754 // Otherwise queue/dequeue calls could be made on the disconnected
755 // ANW, which may result in errors.
756 reset();
757
758 disconnectNativeWindow();
759
760 return err;
761 }
762 }
763
Andy McFadden484566c2012-12-18 09:46:54 -0800764 // Note that we must set the player's new GraphicBufferProducer before
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700765 // disconnecting the old one. Otherwise queue/dequeue calls could be made
766 // on the disconnected ANW, which may result in errors.
Andy McFadden484566c2012-12-18 09:46:54 -0800767 status_t err = p->setVideoSurfaceTexture(bufferProducer);
Jamie Gennis7dae00b2011-10-26 18:36:31 -0700768
769 disconnectNativeWindow();
770
771 mConnectedWindow = anw;
772
773 if (err == OK) {
774 mConnectedWindowBinder = binder;
775 } else {
776 disconnectNativeWindow();
777 }
778
779 return err;
Glenn Kasten11731182011-02-08 17:26:17 -0800780}
781
Nicolas Catania1d187f12009-05-12 23:25:55 -0700782status_t MediaPlayerService::Client::invoke(const Parcel& request,
783 Parcel *reply)
784{
785 sp<MediaPlayerBase> p = getPlayer();
786 if (p == NULL) return UNKNOWN_ERROR;
787 return p->invoke(request, reply);
788}
789
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700790// This call doesn't need to access the native player.
791status_t MediaPlayerService::Client::setMetadataFilter(const Parcel& filter)
792{
793 status_t status;
nikoa64c8c72009-07-20 15:07:26 -0700794 media::Metadata::Filter allow, drop;
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700795
Nicolas Catania48290382009-07-10 13:53:06 -0700796 if (unmarshallFilter(filter, &allow, &status) &&
797 unmarshallFilter(filter, &drop, &status)) {
798 Mutex::Autolock lock(mLock);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -0700799
800 mMetadataAllow = allow;
801 mMetadataDrop = drop;
802 }
803 return status;
804}
805
Nicolas Catania48290382009-07-10 13:53:06 -0700806status_t MediaPlayerService::Client::getMetadata(
807 bool update_only, bool apply_filter, Parcel *reply)
Nicolas Catania8e1b6cc2009-07-09 09:21:33 -0700808{
nikoa64c8c72009-07-20 15:07:26 -0700809 sp<MediaPlayerBase> player = getPlayer();
810 if (player == 0) return UNKNOWN_ERROR;
Nicolas Catania48290382009-07-10 13:53:06 -0700811
nikod608a812009-07-16 16:39:53 -0700812 status_t status;
813 // Placeholder for the return code, updated by the caller.
814 reply->writeInt32(-1);
815
nikoa64c8c72009-07-20 15:07:26 -0700816 media::Metadata::Filter ids;
Nicolas Catania48290382009-07-10 13:53:06 -0700817
818 // We don't block notifications while we fetch the data. We clear
819 // mMetadataUpdated first so we don't lose notifications happening
820 // during the rest of this call.
821 {
822 Mutex::Autolock lock(mLock);
823 if (update_only) {
nikod608a812009-07-16 16:39:53 -0700824 ids = mMetadataUpdated;
Nicolas Catania48290382009-07-10 13:53:06 -0700825 }
826 mMetadataUpdated.clear();
827 }
Nicolas Catania8e1b6cc2009-07-09 09:21:33 -0700828
nikoa64c8c72009-07-20 15:07:26 -0700829 media::Metadata metadata(reply);
Nicolas Catania48290382009-07-10 13:53:06 -0700830
nikoa64c8c72009-07-20 15:07:26 -0700831 metadata.appendHeader();
832 status = player->getMetadata(ids, reply);
nikod608a812009-07-16 16:39:53 -0700833
834 if (status != OK) {
nikoa64c8c72009-07-20 15:07:26 -0700835 metadata.resetParcel();
Steve Block29357bc2012-01-06 19:20:56 +0000836 ALOGE("getMetadata failed %d", status);
nikod608a812009-07-16 16:39:53 -0700837 return status;
838 }
839
840 // FIXME: Implement filtering on the result. Not critical since
841 // filtering takes place on the update notifications already. This
842 // would be when all the metadata are fetch and a filter is set.
843
nikod608a812009-07-16 16:39:53 -0700844 // Everything is fine, update the metadata length.
nikoa64c8c72009-07-20 15:07:26 -0700845 metadata.updateLength();
nikod608a812009-07-16 16:39:53 -0700846 return OK;
Nicolas Catania8e1b6cc2009-07-09 09:21:33 -0700847}
848
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800849status_t MediaPlayerService::Client::prepareAsync()
850{
Steve Block3856b092011-10-20 11:56:00 +0100851 ALOGV("[%d] prepareAsync", mConnId);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800852 sp<MediaPlayerBase> p = getPlayer();
853 if (p == 0) return UNKNOWN_ERROR;
854 status_t ret = p->prepareAsync();
855#if CALLBACK_ANTAGONIZER
Steve Blockb8a80522011-12-20 16:23:08 +0000856 ALOGD("start Antagonizer");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800857 if (ret == NO_ERROR) mAntagonizer->start();
858#endif
859 return ret;
860}
861
862status_t MediaPlayerService::Client::start()
863{
Steve Block3856b092011-10-20 11:56:00 +0100864 ALOGV("[%d] start", mConnId);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800865 sp<MediaPlayerBase> p = getPlayer();
866 if (p == 0) return UNKNOWN_ERROR;
867 p->setLooping(mLoop);
868 return p->start();
869}
870
871status_t MediaPlayerService::Client::stop()
872{
Steve Block3856b092011-10-20 11:56:00 +0100873 ALOGV("[%d] stop", mConnId);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800874 sp<MediaPlayerBase> p = getPlayer();
875 if (p == 0) return UNKNOWN_ERROR;
876 return p->stop();
877}
878
879status_t MediaPlayerService::Client::pause()
880{
Steve Block3856b092011-10-20 11:56:00 +0100881 ALOGV("[%d] pause", mConnId);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800882 sp<MediaPlayerBase> p = getPlayer();
883 if (p == 0) return UNKNOWN_ERROR;
884 return p->pause();
885}
886
887status_t MediaPlayerService::Client::isPlaying(bool* state)
888{
889 *state = false;
890 sp<MediaPlayerBase> p = getPlayer();
891 if (p == 0) return UNKNOWN_ERROR;
892 *state = p->isPlaying();
Steve Block3856b092011-10-20 11:56:00 +0100893 ALOGV("[%d] isPlaying: %d", mConnId, *state);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800894 return NO_ERROR;
895}
896
897status_t MediaPlayerService::Client::getCurrentPosition(int *msec)
898{
Steve Block3856b092011-10-20 11:56:00 +0100899 ALOGV("getCurrentPosition");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800900 sp<MediaPlayerBase> p = getPlayer();
901 if (p == 0) return UNKNOWN_ERROR;
902 status_t ret = p->getCurrentPosition(msec);
903 if (ret == NO_ERROR) {
Steve Block3856b092011-10-20 11:56:00 +0100904 ALOGV("[%d] getCurrentPosition = %d", mConnId, *msec);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800905 } else {
Steve Block29357bc2012-01-06 19:20:56 +0000906 ALOGE("getCurrentPosition returned %d", ret);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800907 }
908 return ret;
909}
910
911status_t MediaPlayerService::Client::getDuration(int *msec)
912{
Steve Block3856b092011-10-20 11:56:00 +0100913 ALOGV("getDuration");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800914 sp<MediaPlayerBase> p = getPlayer();
915 if (p == 0) return UNKNOWN_ERROR;
916 status_t ret = p->getDuration(msec);
917 if (ret == NO_ERROR) {
Steve Block3856b092011-10-20 11:56:00 +0100918 ALOGV("[%d] getDuration = %d", mConnId, *msec);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800919 } else {
Steve Block29357bc2012-01-06 19:20:56 +0000920 ALOGE("getDuration returned %d", ret);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800921 }
922 return ret;
923}
924
Marco Nelissen6b74d672012-02-28 16:07:44 -0800925status_t MediaPlayerService::Client::setNextPlayer(const sp<IMediaPlayer>& player) {
926 ALOGV("setNextPlayer");
927 Mutex::Autolock l(mLock);
928 sp<Client> c = static_cast<Client*>(player.get());
929 mNextClient = c;
John Grossman5f7e55e2012-08-24 14:47:25 -0700930
931 if (c != NULL) {
932 if (mAudioOutput != NULL) {
933 mAudioOutput->setNextOutput(c->mAudioOutput);
934 } else if ((mPlayer != NULL) && !mPlayer->hardwareOutput()) {
935 ALOGE("no current audio output");
936 }
937
938 if ((mPlayer != NULL) && (mNextClient->getPlayer() != NULL)) {
939 mPlayer->setNextPlayer(mNextClient->getPlayer());
940 }
Marco Nelissen6b74d672012-02-28 16:07:44 -0800941 }
John Grossman5f7e55e2012-08-24 14:47:25 -0700942
Marco Nelissen6b74d672012-02-28 16:07:44 -0800943 return OK;
944}
945
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800946status_t MediaPlayerService::Client::seekTo(int msec)
947{
Steve Block3856b092011-10-20 11:56:00 +0100948 ALOGV("[%d] seekTo(%d)", mConnId, msec);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800949 sp<MediaPlayerBase> p = getPlayer();
950 if (p == 0) return UNKNOWN_ERROR;
951 return p->seekTo(msec);
952}
953
954status_t MediaPlayerService::Client::reset()
955{
Steve Block3856b092011-10-20 11:56:00 +0100956 ALOGV("[%d] reset", mConnId);
John Grossmanc795b642012-02-22 15:38:35 -0800957 mRetransmitEndpointValid = false;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800958 sp<MediaPlayerBase> p = getPlayer();
959 if (p == 0) return UNKNOWN_ERROR;
960 return p->reset();
961}
962
Glenn Kastenfff6d712012-01-12 16:38:12 -0800963status_t MediaPlayerService::Client::setAudioStreamType(audio_stream_type_t type)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800964{
Steve Block3856b092011-10-20 11:56:00 +0100965 ALOGV("[%d] setAudioStreamType(%d)", mConnId, type);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800966 // TODO: for hardware output, call player instead
967 Mutex::Autolock l(mLock);
968 if (mAudioOutput != 0) mAudioOutput->setAudioStreamType(type);
969 return NO_ERROR;
970}
971
972status_t MediaPlayerService::Client::setLooping(int loop)
973{
Steve Block3856b092011-10-20 11:56:00 +0100974 ALOGV("[%d] setLooping(%d)", mConnId, loop);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800975 mLoop = loop;
976 sp<MediaPlayerBase> p = getPlayer();
977 if (p != 0) return p->setLooping(loop);
978 return NO_ERROR;
979}
980
981status_t MediaPlayerService::Client::setVolume(float leftVolume, float rightVolume)
982{
Steve Block3856b092011-10-20 11:56:00 +0100983 ALOGV("[%d] setVolume(%f, %f)", mConnId, leftVolume, rightVolume);
John Grossman761defc2012-02-09 15:09:05 -0800984
985 // for hardware output, call player instead
986 sp<MediaPlayerBase> p = getPlayer();
987 {
988 Mutex::Autolock l(mLock);
989 if (p != 0 && p->hardwareOutput()) {
990 MediaPlayerHWInterface* hwp =
991 reinterpret_cast<MediaPlayerHWInterface*>(p.get());
992 return hwp->setVolume(leftVolume, rightVolume);
993 } else {
994 if (mAudioOutput != 0) mAudioOutput->setVolume(leftVolume, rightVolume);
995 return NO_ERROR;
996 }
997 }
998
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -0800999 return NO_ERROR;
1000}
1001
Eric Laurent2beeb502010-07-16 07:43:46 -07001002status_t MediaPlayerService::Client::setAuxEffectSendLevel(float level)
1003{
Steve Block3856b092011-10-20 11:56:00 +01001004 ALOGV("[%d] setAuxEffectSendLevel(%f)", mConnId, level);
Eric Laurent2beeb502010-07-16 07:43:46 -07001005 Mutex::Autolock l(mLock);
1006 if (mAudioOutput != 0) return mAudioOutput->setAuxEffectSendLevel(level);
1007 return NO_ERROR;
1008}
1009
1010status_t MediaPlayerService::Client::attachAuxEffect(int effectId)
1011{
Steve Block3856b092011-10-20 11:56:00 +01001012 ALOGV("[%d] attachAuxEffect(%d)", mConnId, effectId);
Eric Laurent2beeb502010-07-16 07:43:46 -07001013 Mutex::Autolock l(mLock);
1014 if (mAudioOutput != 0) return mAudioOutput->attachAuxEffect(effectId);
1015 return NO_ERROR;
1016}
Nicolas Catania48290382009-07-10 13:53:06 -07001017
Gloria Wang4f9e47f2011-04-25 17:28:22 -07001018status_t MediaPlayerService::Client::setParameter(int key, const Parcel &request) {
Steve Block3856b092011-10-20 11:56:00 +01001019 ALOGV("[%d] setParameter(%d)", mConnId, key);
Gloria Wang4f9e47f2011-04-25 17:28:22 -07001020 sp<MediaPlayerBase> p = getPlayer();
1021 if (p == 0) return UNKNOWN_ERROR;
1022 return p->setParameter(key, request);
1023}
1024
1025status_t MediaPlayerService::Client::getParameter(int key, Parcel *reply) {
Steve Block3856b092011-10-20 11:56:00 +01001026 ALOGV("[%d] getParameter(%d)", mConnId, key);
Gloria Wang4f9e47f2011-04-25 17:28:22 -07001027 sp<MediaPlayerBase> p = getPlayer();
1028 if (p == 0) return UNKNOWN_ERROR;
1029 return p->getParameter(key, reply);
1030}
1031
John Grossmanc795b642012-02-22 15:38:35 -08001032status_t MediaPlayerService::Client::setRetransmitEndpoint(
1033 const struct sockaddr_in* endpoint) {
1034
1035 if (NULL != endpoint) {
1036 uint32_t a = ntohl(endpoint->sin_addr.s_addr);
1037 uint16_t p = ntohs(endpoint->sin_port);
1038 ALOGV("[%d] setRetransmitEndpoint(%u.%u.%u.%u:%hu)", mConnId,
1039 (a >> 24), (a >> 16) & 0xFF, (a >> 8) & 0xFF, (a & 0xFF), p);
1040 } else {
1041 ALOGV("[%d] setRetransmitEndpoint = <none>", mConnId);
1042 }
1043
1044 sp<MediaPlayerBase> p = getPlayer();
1045
1046 // Right now, the only valid time to set a retransmit endpoint is before
1047 // player selection has been made (since the presence or absence of a
1048 // retransmit endpoint is going to determine which player is selected during
1049 // setDataSource).
1050 if (p != 0) return INVALID_OPERATION;
1051
1052 if (NULL != endpoint) {
1053 mRetransmitEndpoint = *endpoint;
1054 mRetransmitEndpointValid = true;
1055 } else {
1056 mRetransmitEndpointValid = false;
1057 }
1058
1059 return NO_ERROR;
1060}
1061
John Grossman44a7e422012-06-21 17:29:24 -07001062status_t MediaPlayerService::Client::getRetransmitEndpoint(
1063 struct sockaddr_in* endpoint)
1064{
1065 if (NULL == endpoint)
1066 return BAD_VALUE;
1067
1068 sp<MediaPlayerBase> p = getPlayer();
1069
1070 if (p != NULL)
1071 return p->getRetransmitEndpoint(endpoint);
1072
1073 if (!mRetransmitEndpointValid)
1074 return NO_INIT;
1075
1076 *endpoint = mRetransmitEndpoint;
1077
1078 return NO_ERROR;
1079}
1080
Gloria Wangb483c472011-04-11 17:23:27 -07001081void MediaPlayerService::Client::notify(
1082 void* cookie, int msg, int ext1, int ext2, const Parcel *obj)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001083{
1084 Client* client = static_cast<Client*>(cookie);
James Dongb8a98252012-08-26 16:13:03 -07001085 if (client == NULL) {
1086 return;
1087 }
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001088
James Dongb8a98252012-08-26 16:13:03 -07001089 sp<IMediaPlayerClient> c;
Marco Nelissen6b74d672012-02-28 16:07:44 -08001090 {
1091 Mutex::Autolock l(client->mLock);
James Dongb8a98252012-08-26 16:13:03 -07001092 c = client->mClient;
Marco Nelissen6b74d672012-02-28 16:07:44 -08001093 if (msg == MEDIA_PLAYBACK_COMPLETE && client->mNextClient != NULL) {
John Grossmancb0b7552012-08-23 17:47:31 -07001094 if (client->mAudioOutput != NULL)
1095 client->mAudioOutput->switchToNextOutput();
Marco Nelissen6b74d672012-02-28 16:07:44 -08001096 client->mNextClient->start();
1097 client->mNextClient->mClient->notify(MEDIA_INFO, MEDIA_INFO_STARTED_AS_NEXT, 0, obj);
1098 }
1099 }
1100
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001101 if (MEDIA_INFO == msg &&
Nicolas Catania48290382009-07-10 13:53:06 -07001102 MEDIA_INFO_METADATA_UPDATE == ext1) {
nikoa64c8c72009-07-20 15:07:26 -07001103 const media::Metadata::Type metadata_type = ext2;
Nicolas Catania48290382009-07-10 13:53:06 -07001104
1105 if(client->shouldDropMetadata(metadata_type)) {
1106 return;
1107 }
1108
1109 // Update the list of metadata that have changed. getMetadata
1110 // also access mMetadataUpdated and clears it.
1111 client->addNewMetadataUpdate(metadata_type);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001112 }
James Dongb8a98252012-08-26 16:13:03 -07001113
1114 if (c != NULL) {
1115 ALOGV("[%d] notify (%p, %d, %d, %d)", client->mConnId, cookie, msg, ext1, ext2);
1116 c->notify(msg, ext1, ext2, obj);
1117 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001118}
1119
Nicolas Catania48290382009-07-10 13:53:06 -07001120
nikoa64c8c72009-07-20 15:07:26 -07001121bool MediaPlayerService::Client::shouldDropMetadata(media::Metadata::Type code) const
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001122{
Nicolas Catania48290382009-07-10 13:53:06 -07001123 Mutex::Autolock lock(mLock);
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001124
Nicolas Catania48290382009-07-10 13:53:06 -07001125 if (findMetadata(mMetadataDrop, code)) {
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001126 return true;
1127 }
1128
Nicolas Catania48290382009-07-10 13:53:06 -07001129 if (mMetadataAllow.isEmpty() || findMetadata(mMetadataAllow, code)) {
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001130 return false;
Nicolas Catania48290382009-07-10 13:53:06 -07001131 } else {
Nicolas Cataniaa7e0e8b2009-07-08 08:57:42 -07001132 return true;
1133 }
1134}
1135
Nicolas Catania48290382009-07-10 13:53:06 -07001136
nikoa64c8c72009-07-20 15:07:26 -07001137void MediaPlayerService::Client::addNewMetadataUpdate(media::Metadata::Type metadata_type) {
Nicolas Catania48290382009-07-10 13:53:06 -07001138 Mutex::Autolock lock(mLock);
1139 if (mMetadataUpdated.indexOf(metadata_type) < 0) {
1140 mMetadataUpdated.add(metadata_type);
1141 }
1142}
1143
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001144#if CALLBACK_ANTAGONIZER
1145const int Antagonizer::interval = 10000; // 10 msecs
1146
1147Antagonizer::Antagonizer(notify_callback_f cb, void* client) :
1148 mExit(false), mActive(false), mClient(client), mCb(cb)
1149{
1150 createThread(callbackThread, this);
1151}
1152
1153void Antagonizer::kill()
1154{
1155 Mutex::Autolock _l(mLock);
1156 mActive = false;
1157 mExit = true;
1158 mCondition.wait(mLock);
1159}
1160
1161int Antagonizer::callbackThread(void* user)
1162{
Steve Blockb8a80522011-12-20 16:23:08 +00001163 ALOGD("Antagonizer started");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001164 Antagonizer* p = reinterpret_cast<Antagonizer*>(user);
1165 while (!p->mExit) {
1166 if (p->mActive) {
Steve Block3856b092011-10-20 11:56:00 +01001167 ALOGV("send event");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001168 p->mCb(p->mClient, 0, 0, 0);
1169 }
1170 usleep(interval);
1171 }
1172 Mutex::Autolock _l(p->mLock);
1173 p->mCondition.signal();
Steve Blockb8a80522011-12-20 16:23:08 +00001174 ALOGD("Antagonizer stopped");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001175 return 0;
1176}
1177#endif
1178
Eric Laurent3d00aa62013-09-24 09:53:27 -07001179status_t MediaPlayerService::decode(const char* url, uint32_t *pSampleRate, int* pNumChannels,
1180 audio_format_t* pFormat,
1181 const sp<IMemoryHeap>& heap, size_t *pSize)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001182{
Steve Block3856b092011-10-20 11:56:00 +01001183 ALOGV("decode(%s)", url);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001184 sp<MediaPlayerBase> player;
Eric Laurent3d00aa62013-09-24 09:53:27 -07001185 status_t status = BAD_VALUE;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001186
1187 // Protect our precious, precious DRMd ringtones by only allowing
1188 // decoding of http, but not filesystem paths or content Uris.
1189 // If the application wants to decode those, it should open a
1190 // filedescriptor for them and use that.
1191 if (url != NULL && strncmp(url, "http://", 7) != 0) {
Steve Blockb8a80522011-12-20 16:23:08 +00001192 ALOGD("Can't decode %s by path, use filedescriptor instead", url);
Eric Laurent3d00aa62013-09-24 09:53:27 -07001193 return BAD_VALUE;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001194 }
1195
John Grossman44a7e422012-06-21 17:29:24 -07001196 player_type playerType =
1197 MediaPlayerFactory::getPlayerType(NULL /* client */, url);
Steve Block3856b092011-10-20 11:56:00 +01001198 ALOGV("player type = %d", playerType);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001199
1200 // create the right type of player
Eric Laurent3d00aa62013-09-24 09:53:27 -07001201 sp<AudioCache> cache = new AudioCache(heap);
John Grossman44a7e422012-06-21 17:29:24 -07001202 player = MediaPlayerFactory::createPlayer(playerType, cache.get(), cache->notify);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001203 if (player == NULL) goto Exit;
1204 if (player->hardwareOutput()) goto Exit;
1205
1206 static_cast<MediaPlayerInterface*>(player.get())->setAudioSink(cache);
1207
1208 // set data source
1209 if (player->setDataSource(url) != NO_ERROR) goto Exit;
1210
Steve Block3856b092011-10-20 11:56:00 +01001211 ALOGV("prepare");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001212 player->prepareAsync();
1213
Steve Block3856b092011-10-20 11:56:00 +01001214 ALOGV("wait for prepare");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001215 if (cache->wait() != NO_ERROR) goto Exit;
1216
Steve Block3856b092011-10-20 11:56:00 +01001217 ALOGV("start");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001218 player->start();
1219
Steve Block3856b092011-10-20 11:56:00 +01001220 ALOGV("wait for playback complete");
Eric Laurent9cb839a2011-09-27 09:48:56 -07001221 cache->wait();
1222 // in case of error, return what was successfully decoded.
1223 if (cache->size() == 0) {
1224 goto Exit;
1225 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001226
Eric Laurent3d00aa62013-09-24 09:53:27 -07001227 *pSize = cache->size();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001228 *pSampleRate = cache->sampleRate();
1229 *pNumChannels = cache->channelCount();
Glenn Kastene1c39622012-01-04 09:36:37 -08001230 *pFormat = cache->format();
Eric Laurent3d00aa62013-09-24 09:53:27 -07001231 ALOGV("return size %d sampleRate=%u, channelCount = %d, format = %d",
1232 *pSize, *pSampleRate, *pNumChannels, *pFormat);
1233 status = NO_ERROR;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001234
1235Exit:
1236 if (player != 0) player->reset();
Eric Laurent3d00aa62013-09-24 09:53:27 -07001237 return status;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001238}
1239
Eric Laurent3d00aa62013-09-24 09:53:27 -07001240status_t MediaPlayerService::decode(int fd, int64_t offset, int64_t length,
1241 uint32_t *pSampleRate, int* pNumChannels,
1242 audio_format_t* pFormat,
1243 const sp<IMemoryHeap>& heap, size_t *pSize)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001244{
Steve Block3856b092011-10-20 11:56:00 +01001245 ALOGV("decode(%d, %lld, %lld)", fd, offset, length);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001246 sp<MediaPlayerBase> player;
Eric Laurent3d00aa62013-09-24 09:53:27 -07001247 status_t status = BAD_VALUE;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001248
John Grossman44a7e422012-06-21 17:29:24 -07001249 player_type playerType = MediaPlayerFactory::getPlayerType(NULL /* client */,
1250 fd,
1251 offset,
1252 length);
Steve Block3856b092011-10-20 11:56:00 +01001253 ALOGV("player type = %d", playerType);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001254
1255 // create the right type of player
Eric Laurent3d00aa62013-09-24 09:53:27 -07001256 sp<AudioCache> cache = new AudioCache(heap);
John Grossman44a7e422012-06-21 17:29:24 -07001257 player = MediaPlayerFactory::createPlayer(playerType, cache.get(), cache->notify);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001258 if (player == NULL) goto Exit;
1259 if (player->hardwareOutput()) goto Exit;
1260
1261 static_cast<MediaPlayerInterface*>(player.get())->setAudioSink(cache);
1262
1263 // set data source
1264 if (player->setDataSource(fd, offset, length) != NO_ERROR) goto Exit;
1265
Steve Block3856b092011-10-20 11:56:00 +01001266 ALOGV("prepare");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001267 player->prepareAsync();
1268
Steve Block3856b092011-10-20 11:56:00 +01001269 ALOGV("wait for prepare");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001270 if (cache->wait() != NO_ERROR) goto Exit;
1271
Steve Block3856b092011-10-20 11:56:00 +01001272 ALOGV("start");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001273 player->start();
1274
Steve Block3856b092011-10-20 11:56:00 +01001275 ALOGV("wait for playback complete");
Eric Laurent9cb839a2011-09-27 09:48:56 -07001276 cache->wait();
1277 // in case of error, return what was successfully decoded.
1278 if (cache->size() == 0) {
1279 goto Exit;
1280 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001281
Eric Laurent3d00aa62013-09-24 09:53:27 -07001282 *pSize = cache->size();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001283 *pSampleRate = cache->sampleRate();
1284 *pNumChannels = cache->channelCount();
1285 *pFormat = cache->format();
Eric Laurent3d00aa62013-09-24 09:53:27 -07001286 ALOGV("return size %d, sampleRate=%u, channelCount = %d, format = %d",
1287 *pSize, *pSampleRate, *pNumChannels, *pFormat);
1288 status = NO_ERROR;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001289
1290Exit:
1291 if (player != 0) player->reset();
1292 ::close(fd);
Eric Laurent3d00aa62013-09-24 09:53:27 -07001293 return status;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001294}
1295
Marco Nelissen10dbb8e2009-09-20 10:42:13 -07001296
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001297#undef LOG_TAG
1298#define LOG_TAG "AudioSink"
Marco Nelissen462fd2f2013-01-14 14:12:05 -08001299MediaPlayerService::AudioOutput::AudioOutput(int sessionId, int uid)
Andreas Huber20111aa2009-07-14 16:56:47 -07001300 : mCallback(NULL),
Eric Laurenta514bdb2010-06-21 09:27:30 -07001301 mCallbackCookie(NULL),
Marco Nelissen6b74d672012-02-28 16:07:44 -08001302 mCallbackData(NULL),
Marco Nelissen4110c102012-03-29 09:31:28 -07001303 mBytesWritten(0),
Eric Laurent1948eb32012-04-13 16:50:19 -07001304 mSessionId(sessionId),
Marco Nelissen462fd2f2013-01-14 14:12:05 -08001305 mUid(uid),
Eric Laurent1948eb32012-04-13 16:50:19 -07001306 mFlags(AUDIO_OUTPUT_FLAG_NONE) {
Steve Block3856b092011-10-20 11:56:00 +01001307 ALOGV("AudioOutput(%d)", sessionId);
Dima Zavinfce7a472011-04-19 22:30:36 -07001308 mStreamType = AUDIO_STREAM_MUSIC;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001309 mLeftVolume = 1.0;
1310 mRightVolume = 1.0;
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001311 mPlaybackRatePermille = 1000;
1312 mSampleRateHz = 0;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001313 mMsecsPerFrame = 0;
Eric Laurent2beeb502010-07-16 07:43:46 -07001314 mAuxEffectId = 0;
1315 mSendLevel = 0.0;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001316 setMinBufferCount();
1317}
1318
1319MediaPlayerService::AudioOutput::~AudioOutput()
1320{
1321 close();
Marco Nelissen6b74d672012-02-28 16:07:44 -08001322 delete mCallbackData;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001323}
1324
1325void MediaPlayerService::AudioOutput::setMinBufferCount()
1326{
1327 char value[PROPERTY_VALUE_MAX];
1328 if (property_get("ro.kernel.qemu", value, 0)) {
1329 mIsOnEmulator = true;
1330 mMinBufferCount = 12; // to prevent systematic buffer underrun for emulator
1331 }
1332}
1333
1334bool MediaPlayerService::AudioOutput::isOnEmulator()
1335{
1336 setMinBufferCount();
1337 return mIsOnEmulator;
1338}
1339
1340int MediaPlayerService::AudioOutput::getMinBufferCount()
1341{
1342 setMinBufferCount();
1343 return mMinBufferCount;
1344}
1345
1346ssize_t MediaPlayerService::AudioOutput::bufferSize() const
1347{
1348 if (mTrack == 0) return NO_INIT;
1349 return mTrack->frameCount() * frameSize();
1350}
1351
1352ssize_t MediaPlayerService::AudioOutput::frameCount() const
1353{
1354 if (mTrack == 0) return NO_INIT;
1355 return mTrack->frameCount();
1356}
1357
1358ssize_t MediaPlayerService::AudioOutput::channelCount() const
1359{
1360 if (mTrack == 0) return NO_INIT;
1361 return mTrack->channelCount();
1362}
1363
1364ssize_t MediaPlayerService::AudioOutput::frameSize() const
1365{
1366 if (mTrack == 0) return NO_INIT;
1367 return mTrack->frameSize();
1368}
1369
1370uint32_t MediaPlayerService::AudioOutput::latency () const
1371{
Eric Laurentdb354e52012-03-05 17:27:11 -08001372 if (mTrack == 0) return 0;
1373 return mTrack->latency();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001374}
1375
1376float MediaPlayerService::AudioOutput::msecsPerFrame() const
1377{
1378 return mMsecsPerFrame;
1379}
1380
Marco Nelissen4110c102012-03-29 09:31:28 -07001381status_t MediaPlayerService::AudioOutput::getPosition(uint32_t *position) const
Eric Laurent342e9cf2010-01-19 17:37:09 -08001382{
1383 if (mTrack == 0) return NO_INIT;
1384 return mTrack->getPosition(position);
1385}
1386
Marco Nelissen4110c102012-03-29 09:31:28 -07001387status_t MediaPlayerService::AudioOutput::getFramesWritten(uint32_t *frameswritten) const
1388{
1389 if (mTrack == 0) return NO_INIT;
1390 *frameswritten = mBytesWritten / frameSize();
1391 return OK;
1392}
1393
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001394status_t MediaPlayerService::AudioOutput::setParameters(const String8& keyValuePairs)
1395{
1396 if (mTrack == 0) return NO_INIT;
1397 return mTrack->setParameters(keyValuePairs);
1398}
1399
1400String8 MediaPlayerService::AudioOutput::getParameters(const String8& keys)
1401{
1402 if (mTrack == 0) return String8::empty();
1403 return mTrack->getParameters(keys);
1404}
1405
1406void MediaPlayerService::AudioOutput::deleteRecycledTrack()
1407{
1408 ALOGV("deleteRecycledTrack");
1409
1410 if (mRecycledTrack != 0) {
1411
1412 if (mCallbackData != NULL) {
1413 mCallbackData->setOutput(NULL);
1414 mCallbackData->endTrackSwitch();
1415 }
1416
1417 if ((mRecycledTrack->getFlags() & AUDIO_OUTPUT_FLAG_COMPRESS_OFFLOAD) == 0) {
1418 mRecycledTrack->flush();
1419 }
1420 // An offloaded track isn't flushed because the STREAM_END is reported
1421 // slightly prematurely to allow time for the gapless track switch
1422 // but this means that if we decide not to recycle the track there
1423 // could be a small amount of residual data still playing. We leave
1424 // AudioFlinger to drain the track.
1425
1426 mRecycledTrack.clear();
1427 delete mCallbackData;
1428 mCallbackData = NULL;
1429 close();
1430 }
1431}
1432
Andreas Huber20111aa2009-07-14 16:56:47 -07001433status_t MediaPlayerService::AudioOutput::open(
Jean-Michel Trivi786618f2012-03-02 14:54:07 -08001434 uint32_t sampleRate, int channelCount, audio_channel_mask_t channelMask,
1435 audio_format_t format, int bufferCount,
Eric Laurent1948eb32012-04-13 16:50:19 -07001436 AudioCallback cb, void *cookie,
Richard Fitzgeraldad3af332013-03-25 16:54:37 +00001437 audio_output_flags_t flags,
1438 const audio_offload_info_t *offloadInfo)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001439{
Andreas Huber20111aa2009-07-14 16:56:47 -07001440 mCallback = cb;
1441 mCallbackCookie = cookie;
1442
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001443 // Check argument "bufferCount" against the mininum buffer count
1444 if (bufferCount < mMinBufferCount) {
Steve Blockb8a80522011-12-20 16:23:08 +00001445 ALOGD("bufferCount (%d) is too small and increased to %d", bufferCount, mMinBufferCount);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001446 bufferCount = mMinBufferCount;
1447
1448 }
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001449 ALOGV("open(%u, %d, 0x%x, 0x%x, %d, %d 0x%x)", sampleRate, channelCount, channelMask,
1450 format, bufferCount, mSessionId, flags);
Glenn Kasten1127d652012-11-14 08:44:39 -08001451 uint32_t afSampleRate;
Glenn Kasten7da35f22012-11-14 12:54:39 -08001452 size_t afFrameCount;
Eric Laurent1948eb32012-04-13 16:50:19 -07001453 uint32_t frameCount;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001454
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001455 // offloading is only supported in callback mode for now.
1456 // offloadInfo must be present if offload flag is set
1457 if (((flags & AUDIO_OUTPUT_FLAG_COMPRESS_OFFLOAD) != 0) &&
1458 ((cb == NULL) || (offloadInfo == NULL))) {
1459 return BAD_VALUE;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001460 }
1461
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001462 if ((flags & AUDIO_OUTPUT_FLAG_COMPRESS_OFFLOAD) != 0) {
1463 frameCount = 0; // AudioTrack will get frame count from AudioFlinger
1464 } else {
1465 uint32_t afSampleRate;
1466 size_t afFrameCount;
1467
1468 if (AudioSystem::getOutputFrameCount(&afFrameCount, mStreamType) != NO_ERROR) {
1469 return NO_INIT;
1470 }
1471 if (AudioSystem::getOutputSamplingRate(&afSampleRate, mStreamType) != NO_ERROR) {
1472 return NO_INIT;
1473 }
1474
1475 frameCount = (sampleRate*afFrameCount*bufferCount)/afSampleRate;
1476 }
Andreas Huber20111aa2009-07-14 16:56:47 -07001477
Jean-Michel Trivi786618f2012-03-02 14:54:07 -08001478 if (channelMask == CHANNEL_MASK_USE_CHANNEL_ORDER) {
Glenn Kastenab334fd2012-03-14 12:56:06 -07001479 channelMask = audio_channel_out_mask_from_count(channelCount);
Jean-Michel Trivi786618f2012-03-02 14:54:07 -08001480 if (0 == channelMask) {
1481 ALOGE("open() error, can\'t derive mask for %d audio channels", channelCount);
1482 return NO_INIT;
1483 }
1484 }
Eric Laurent1948eb32012-04-13 16:50:19 -07001485
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001486 // Check whether we can recycle the track
1487 bool reuse = false;
1488 bool bothOffloaded = false;
Marco Nelissen67295b52012-06-11 14:52:53 -07001489
Glenn Kasten2799d742013-05-30 14:33:29 -07001490 if (mRecycledTrack != 0) {
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001491 // check whether we are switching between two offloaded tracks
1492 bothOffloaded = (flags & mRecycledTrack->getFlags()
1493 & AUDIO_OUTPUT_FLAG_COMPRESS_OFFLOAD) != 0;
Marco Nelissen67295b52012-06-11 14:52:53 -07001494
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001495 // check if the existing track can be reused as-is, or if a new track needs to be created.
1496 reuse = true;
1497
Marco Nelissen67295b52012-06-11 14:52:53 -07001498 if ((mCallbackData == NULL && mCallback != NULL) ||
1499 (mCallbackData != NULL && mCallback == NULL)) {
1500 // recycled track uses callbacks but the caller wants to use writes, or vice versa
1501 ALOGV("can't chain callback and write");
1502 reuse = false;
1503 } else if ((mRecycledTrack->getSampleRate() != sampleRate) ||
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001504 (mRecycledTrack->channelCount() != (uint32_t)channelCount) ) {
1505 ALOGV("samplerate, channelcount differ: %u/%u Hz, %u/%d ch",
Marco Nelissen67295b52012-06-11 14:52:53 -07001506 mRecycledTrack->getSampleRate(), sampleRate,
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001507 mRecycledTrack->channelCount(), channelCount);
Marco Nelissen67295b52012-06-11 14:52:53 -07001508 reuse = false;
1509 } else if (flags != mFlags) {
1510 ALOGV("output flags differ %08x/%08x", flags, mFlags);
1511 reuse = false;
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001512 } else if (mRecycledTrack->format() != format) {
1513 reuse = false;
Marco Nelissen67295b52012-06-11 14:52:53 -07001514 }
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001515 } else {
1516 ALOGV("no track available to recycle");
1517 }
1518
1519 ALOGV_IF(bothOffloaded, "both tracks offloaded");
1520
1521 // If we can't recycle and both tracks are offloaded
1522 // we must close the previous output before opening a new one
1523 if (bothOffloaded && !reuse) {
1524 ALOGV("both offloaded and not recycling");
1525 deleteRecycledTrack();
1526 }
1527
1528 sp<AudioTrack> t;
1529 CallbackData *newcbd = NULL;
1530
1531 // We don't attempt to create a new track if we are recycling an
1532 // offloaded track. But, if we are recycling a non-offloaded or we
1533 // are switching where one is offloaded and one isn't then we create
1534 // the new track in advance so that we can read additional stream info
1535
1536 if (!(reuse && bothOffloaded)) {
1537 ALOGV("creating new AudioTrack");
1538
1539 if (mCallback != NULL) {
1540 newcbd = new CallbackData(this);
1541 t = new AudioTrack(
1542 mStreamType,
1543 sampleRate,
1544 format,
1545 channelMask,
1546 frameCount,
1547 flags,
1548 CallbackWrapper,
1549 newcbd,
1550 0, // notification frames
1551 mSessionId,
1552 AudioTrack::TRANSFER_CALLBACK,
Marco Nelissen462fd2f2013-01-14 14:12:05 -08001553 offloadInfo,
1554 mUid);
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001555 } else {
1556 t = new AudioTrack(
1557 mStreamType,
1558 sampleRate,
1559 format,
1560 channelMask,
1561 frameCount,
1562 flags,
Marco Nelissen462fd2f2013-01-14 14:12:05 -08001563 NULL, // callback
1564 NULL, // user data
1565 0, // notification frames
1566 mSessionId,
1567 AudioTrack::TRANSFER_DEFAULT,
1568 NULL, // offload info
1569 mUid);
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001570 }
1571
1572 if ((t == 0) || (t->initCheck() != NO_ERROR)) {
1573 ALOGE("Unable to create audio track");
1574 delete newcbd;
1575 return NO_INIT;
1576 }
1577 }
1578
1579 if (reuse) {
1580 CHECK(mRecycledTrack != NULL);
1581
1582 if (!bothOffloaded) {
1583 if (mRecycledTrack->frameCount() != t->frameCount()) {
1584 ALOGV("framecount differs: %u/%u frames",
1585 mRecycledTrack->frameCount(), t->frameCount());
1586 reuse = false;
1587 }
1588 }
1589
Marco Nelissen67295b52012-06-11 14:52:53 -07001590 if (reuse) {
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001591 ALOGV("chaining to next output and recycling track");
Marco Nelissen67295b52012-06-11 14:52:53 -07001592 close();
1593 mTrack = mRecycledTrack;
Glenn Kasten2799d742013-05-30 14:33:29 -07001594 mRecycledTrack.clear();
Marco Nelissen67295b52012-06-11 14:52:53 -07001595 if (mCallbackData != NULL) {
1596 mCallbackData->setOutput(this);
1597 }
Marco Nelissen67295b52012-06-11 14:52:53 -07001598 delete newcbd;
1599 return OK;
1600 }
Marco Nelissen67295b52012-06-11 14:52:53 -07001601 }
1602
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001603 // we're not going to reuse the track, unblock and flush it
1604 // this was done earlier if both tracks are offloaded
1605 if (!bothOffloaded) {
1606 deleteRecycledTrack();
1607 }
1608
1609 CHECK((t != NULL) && ((mCallback == NULL) || (newcbd != NULL)));
1610
Marco Nelissen67295b52012-06-11 14:52:53 -07001611 mCallbackData = newcbd;
Steve Block3856b092011-10-20 11:56:00 +01001612 ALOGV("setVolume");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001613 t->setVolume(mLeftVolume, mRightVolume);
Eric Laurent2beeb502010-07-16 07:43:46 -07001614
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001615 mSampleRateHz = sampleRate;
Eric Laurent1948eb32012-04-13 16:50:19 -07001616 mFlags = flags;
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001617 mMsecsPerFrame = mPlaybackRatePermille / (float) sampleRate;
Marco Nelissen99448602012-04-02 12:16:49 -07001618 uint32_t pos;
1619 if (t->getPosition(&pos) == OK) {
1620 mBytesWritten = uint64_t(pos) * t->frameSize();
1621 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001622 mTrack = t;
Eric Laurent2beeb502010-07-16 07:43:46 -07001623
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001624 status_t res = NO_ERROR;
1625 if ((flags & AUDIO_OUTPUT_FLAG_COMPRESS_OFFLOAD) == 0) {
1626 res = t->setSampleRate(mPlaybackRatePermille * mSampleRateHz / 1000);
1627 if (res == NO_ERROR) {
1628 t->setAuxEffectSendLevel(mSendLevel);
1629 res = t->attachAuxEffect(mAuxEffectId);
1630 }
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001631 }
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001632 ALOGV("open() DONE status %d", res);
1633 return res;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001634}
1635
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001636status_t MediaPlayerService::AudioOutput::start()
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001637{
Steve Block3856b092011-10-20 11:56:00 +01001638 ALOGV("start");
Marco Nelissen6b74d672012-02-28 16:07:44 -08001639 if (mCallbackData != NULL) {
1640 mCallbackData->endTrackSwitch();
1641 }
Glenn Kasten2799d742013-05-30 14:33:29 -07001642 if (mTrack != 0) {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001643 mTrack->setVolume(mLeftVolume, mRightVolume);
Eric Laurent2beeb502010-07-16 07:43:46 -07001644 mTrack->setAuxEffectSendLevel(mSendLevel);
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001645 return mTrack->start();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001646 }
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001647 return NO_INIT;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001648}
1649
Marco Nelissen6b74d672012-02-28 16:07:44 -08001650void MediaPlayerService::AudioOutput::setNextOutput(const sp<AudioOutput>& nextOutput) {
1651 mNextOutput = nextOutput;
1652}
Marco Nelissen7ee8ac92010-01-12 09:23:54 -08001653
1654
Marco Nelissen6b74d672012-02-28 16:07:44 -08001655void MediaPlayerService::AudioOutput::switchToNextOutput() {
1656 ALOGV("switchToNextOutput");
1657 if (mNextOutput != NULL) {
1658 if (mCallbackData != NULL) {
1659 mCallbackData->beginTrackSwitch();
1660 }
1661 delete mNextOutput->mCallbackData;
1662 mNextOutput->mCallbackData = mCallbackData;
1663 mCallbackData = NULL;
1664 mNextOutput->mRecycledTrack = mTrack;
Glenn Kasten2799d742013-05-30 14:33:29 -07001665 mTrack.clear();
Marco Nelissen6b74d672012-02-28 16:07:44 -08001666 mNextOutput->mSampleRateHz = mSampleRateHz;
1667 mNextOutput->mMsecsPerFrame = mMsecsPerFrame;
Marco Nelissen4110c102012-03-29 09:31:28 -07001668 mNextOutput->mBytesWritten = mBytesWritten;
Marco Nelissend791e092012-06-11 17:00:59 -07001669 mNextOutput->mFlags = mFlags;
Marco Nelissen6b74d672012-02-28 16:07:44 -08001670 }
1671}
1672
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001673ssize_t MediaPlayerService::AudioOutput::write(const void* buffer, size_t size)
1674{
Andreas Huber20111aa2009-07-14 16:56:47 -07001675 LOG_FATAL_IF(mCallback != NULL, "Don't call write if supplying a callback.");
1676
Steve Block3856b092011-10-20 11:56:00 +01001677 //ALOGV("write(%p, %u)", buffer, size);
Glenn Kasten2799d742013-05-30 14:33:29 -07001678 if (mTrack != 0) {
Marco Nelissen10dbb8e2009-09-20 10:42:13 -07001679 ssize_t ret = mTrack->write(buffer, size);
Marco Nelissen4110c102012-03-29 09:31:28 -07001680 mBytesWritten += ret;
Marco Nelissen10dbb8e2009-09-20 10:42:13 -07001681 return ret;
1682 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001683 return NO_INIT;
1684}
1685
1686void MediaPlayerService::AudioOutput::stop()
1687{
Steve Block3856b092011-10-20 11:56:00 +01001688 ALOGV("stop");
Glenn Kasten2799d742013-05-30 14:33:29 -07001689 if (mTrack != 0) mTrack->stop();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001690}
1691
1692void MediaPlayerService::AudioOutput::flush()
1693{
Steve Block3856b092011-10-20 11:56:00 +01001694 ALOGV("flush");
Glenn Kasten2799d742013-05-30 14:33:29 -07001695 if (mTrack != 0) mTrack->flush();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001696}
1697
1698void MediaPlayerService::AudioOutput::pause()
1699{
Steve Block3856b092011-10-20 11:56:00 +01001700 ALOGV("pause");
Glenn Kasten2799d742013-05-30 14:33:29 -07001701 if (mTrack != 0) mTrack->pause();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001702}
1703
1704void MediaPlayerService::AudioOutput::close()
1705{
Steve Block3856b092011-10-20 11:56:00 +01001706 ALOGV("close");
Glenn Kasten2799d742013-05-30 14:33:29 -07001707 mTrack.clear();
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001708}
1709
1710void MediaPlayerService::AudioOutput::setVolume(float left, float right)
1711{
Steve Block3856b092011-10-20 11:56:00 +01001712 ALOGV("setVolume(%f, %f)", left, right);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001713 mLeftVolume = left;
1714 mRightVolume = right;
Glenn Kasten2799d742013-05-30 14:33:29 -07001715 if (mTrack != 0) {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001716 mTrack->setVolume(left, right);
1717 }
1718}
1719
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001720status_t MediaPlayerService::AudioOutput::setPlaybackRatePermille(int32_t ratePermille)
1721{
1722 ALOGV("setPlaybackRatePermille(%d)", ratePermille);
1723 status_t res = NO_ERROR;
Glenn Kasten2799d742013-05-30 14:33:29 -07001724 if (mTrack != 0) {
Jean-Michel Trivi7a8b0ed2012-02-02 09:06:31 -08001725 res = mTrack->setSampleRate(ratePermille * mSampleRateHz / 1000);
1726 } else {
1727 res = NO_INIT;
1728 }
1729 mPlaybackRatePermille = ratePermille;
1730 if (mSampleRateHz != 0) {
1731 mMsecsPerFrame = mPlaybackRatePermille / (float) mSampleRateHz;
1732 }
1733 return res;
1734}
1735
Eric Laurent2beeb502010-07-16 07:43:46 -07001736status_t MediaPlayerService::AudioOutput::setAuxEffectSendLevel(float level)
1737{
Steve Block3856b092011-10-20 11:56:00 +01001738 ALOGV("setAuxEffectSendLevel(%f)", level);
Eric Laurent2beeb502010-07-16 07:43:46 -07001739 mSendLevel = level;
Glenn Kasten2799d742013-05-30 14:33:29 -07001740 if (mTrack != 0) {
Eric Laurent2beeb502010-07-16 07:43:46 -07001741 return mTrack->setAuxEffectSendLevel(level);
1742 }
1743 return NO_ERROR;
1744}
1745
1746status_t MediaPlayerService::AudioOutput::attachAuxEffect(int effectId)
1747{
Steve Block3856b092011-10-20 11:56:00 +01001748 ALOGV("attachAuxEffect(%d)", effectId);
Eric Laurent2beeb502010-07-16 07:43:46 -07001749 mAuxEffectId = effectId;
Glenn Kasten2799d742013-05-30 14:33:29 -07001750 if (mTrack != 0) {
Eric Laurent2beeb502010-07-16 07:43:46 -07001751 return mTrack->attachAuxEffect(effectId);
1752 }
1753 return NO_ERROR;
1754}
1755
Andreas Huber20111aa2009-07-14 16:56:47 -07001756// static
1757void MediaPlayerService::AudioOutput::CallbackWrapper(
Glenn Kastend217a8c2011-06-01 15:20:35 -07001758 int event, void *cookie, void *info) {
Steve Block3856b092011-10-20 11:56:00 +01001759 //ALOGV("callbackwrapper");
Marco Nelissen6b74d672012-02-28 16:07:44 -08001760 CallbackData *data = (CallbackData*)cookie;
1761 data->lock();
1762 AudioOutput *me = data->getOutput();
Andreas Huber20111aa2009-07-14 16:56:47 -07001763 AudioTrack::Buffer *buffer = (AudioTrack::Buffer *)info;
Marco Nelissen6b74d672012-02-28 16:07:44 -08001764 if (me == NULL) {
1765 // no output set, likely because the track was scheduled to be reused
1766 // by another player, but the format turned out to be incompatible.
1767 data->unlock();
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001768 if (buffer != NULL) {
1769 buffer->size = 0;
1770 }
Marco Nelissen6b74d672012-02-28 16:07:44 -08001771 return;
1772 }
Andreas Huber20111aa2009-07-14 16:56:47 -07001773
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001774 switch(event) {
1775 case AudioTrack::EVENT_MORE_DATA: {
1776 size_t actualSize = (*me->mCallback)(
1777 me, buffer->raw, buffer->size, me->mCallbackCookie,
1778 CB_EVENT_FILL_BUFFER);
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001779
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001780 if (actualSize == 0 && buffer->size > 0 && me->mNextOutput == NULL) {
1781 // We've reached EOS but the audio track is not stopped yet,
1782 // keep playing silence.
Andreas Huber2e8ffaf2010-02-18 16:45:13 -08001783
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001784 memset(buffer->raw, 0, buffer->size);
1785 actualSize = buffer->size;
1786 }
1787
1788 buffer->size = actualSize;
1789 } break;
1790
1791
1792 case AudioTrack::EVENT_STREAM_END:
1793 ALOGV("callbackwrapper: deliver EVENT_STREAM_END");
1794 (*me->mCallback)(me, NULL /* buffer */, 0 /* size */,
1795 me->mCallbackCookie, CB_EVENT_STREAM_END);
1796 break;
1797
1798 case AudioTrack::EVENT_NEW_IAUDIOTRACK :
1799 ALOGV("callbackwrapper: deliver EVENT_TEAR_DOWN");
1800 (*me->mCallback)(me, NULL /* buffer */, 0 /* size */,
1801 me->mCallbackCookie, CB_EVENT_TEAR_DOWN);
1802 break;
1803
1804 default:
1805 ALOGE("received unknown event type: %d inside CallbackWrapper !", event);
Andreas Huber51c1e0e2011-04-04 11:43:40 -07001806 }
1807
Marco Nelissen6b74d672012-02-28 16:07:44 -08001808 data->unlock();
Andreas Huber20111aa2009-07-14 16:56:47 -07001809}
1810
Marco Nelissen4110c102012-03-29 09:31:28 -07001811int MediaPlayerService::AudioOutput::getSessionId() const
Eric Laurent8c563ed2010-10-07 18:23:03 -07001812{
1813 return mSessionId;
1814}
1815
Eric Laurent6f59db12013-07-26 17:16:50 -07001816uint32_t MediaPlayerService::AudioOutput::getSampleRate() const
1817{
1818 if (mTrack == 0) return 0;
1819 return mTrack->getSampleRate();
1820}
1821
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001822#undef LOG_TAG
1823#define LOG_TAG "AudioCache"
Eric Laurent3d00aa62013-09-24 09:53:27 -07001824MediaPlayerService::AudioCache::AudioCache(const sp<IMemoryHeap>& heap) :
1825 mHeap(heap), mChannelCount(0), mFrameCount(1024), mSampleRate(0), mSize(0),
1826 mError(NO_ERROR), mCommandComplete(false)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001827{
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001828}
1829
1830uint32_t MediaPlayerService::AudioCache::latency () const
1831{
1832 return 0;
1833}
1834
1835float MediaPlayerService::AudioCache::msecsPerFrame() const
1836{
1837 return mMsecsPerFrame;
1838}
1839
Marco Nelissen4110c102012-03-29 09:31:28 -07001840status_t MediaPlayerService::AudioCache::getPosition(uint32_t *position) const
Eric Laurent342e9cf2010-01-19 17:37:09 -08001841{
1842 if (position == 0) return BAD_VALUE;
1843 *position = mSize;
1844 return NO_ERROR;
1845}
1846
Marco Nelissen4110c102012-03-29 09:31:28 -07001847status_t MediaPlayerService::AudioCache::getFramesWritten(uint32_t *written) const
1848{
1849 if (written == 0) return BAD_VALUE;
1850 *written = mSize;
1851 return NO_ERROR;
1852}
1853
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001854////////////////////////////////////////////////////////////////////////////////
1855
1856struct CallbackThread : public Thread {
1857 CallbackThread(const wp<MediaPlayerBase::AudioSink> &sink,
1858 MediaPlayerBase::AudioSink::AudioCallback cb,
1859 void *cookie);
1860
1861protected:
1862 virtual ~CallbackThread();
1863
1864 virtual bool threadLoop();
1865
1866private:
1867 wp<MediaPlayerBase::AudioSink> mSink;
1868 MediaPlayerBase::AudioSink::AudioCallback mCallback;
1869 void *mCookie;
1870 void *mBuffer;
1871 size_t mBufferSize;
1872
1873 CallbackThread(const CallbackThread &);
1874 CallbackThread &operator=(const CallbackThread &);
1875};
1876
1877CallbackThread::CallbackThread(
1878 const wp<MediaPlayerBase::AudioSink> &sink,
1879 MediaPlayerBase::AudioSink::AudioCallback cb,
1880 void *cookie)
1881 : mSink(sink),
1882 mCallback(cb),
1883 mCookie(cookie),
1884 mBuffer(NULL),
1885 mBufferSize(0) {
1886}
1887
1888CallbackThread::~CallbackThread() {
1889 if (mBuffer) {
1890 free(mBuffer);
1891 mBuffer = NULL;
1892 }
1893}
1894
1895bool CallbackThread::threadLoop() {
1896 sp<MediaPlayerBase::AudioSink> sink = mSink.promote();
1897 if (sink == NULL) {
1898 return false;
1899 }
1900
1901 if (mBuffer == NULL) {
1902 mBufferSize = sink->bufferSize();
1903 mBuffer = malloc(mBufferSize);
1904 }
1905
1906 size_t actualSize =
Richard Fitzgeraldad3af332013-03-25 16:54:37 +00001907 (*mCallback)(sink.get(), mBuffer, mBufferSize, mCookie,
1908 MediaPlayerBase::AudioSink::CB_EVENT_FILL_BUFFER);
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001909
1910 if (actualSize > 0) {
1911 sink->write(mBuffer, actualSize);
1912 }
1913
1914 return true;
1915}
1916
1917////////////////////////////////////////////////////////////////////////////////
1918
Andreas Huber20111aa2009-07-14 16:56:47 -07001919status_t MediaPlayerService::AudioCache::open(
Jean-Michel Trivi786618f2012-03-02 14:54:07 -08001920 uint32_t sampleRate, int channelCount, audio_channel_mask_t channelMask,
1921 audio_format_t format, int bufferCount,
Richard Fitzgeraldad3af332013-03-25 16:54:37 +00001922 AudioCallback cb, void *cookie, audio_output_flags_t flags,
1923 const audio_offload_info_t *offloadInfo)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001924{
Jean-Michel Trivi786618f2012-03-02 14:54:07 -08001925 ALOGV("open(%u, %d, 0x%x, %d, %d)", sampleRate, channelCount, channelMask, format, bufferCount);
Dave Sparks8eb80112009-12-09 20:20:26 -08001926 if (mHeap->getHeapID() < 0) {
1927 return NO_INIT;
1928 }
Andreas Huber20111aa2009-07-14 16:56:47 -07001929
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001930 mSampleRate = sampleRate;
1931 mChannelCount = (uint16_t)channelCount;
Glenn Kastene1c39622012-01-04 09:36:37 -08001932 mFormat = format;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001933 mMsecsPerFrame = 1.e3 / (float) sampleRate;
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001934
1935 if (cb != NULL) {
1936 mCallbackThread = new CallbackThread(this, cb, cookie);
1937 }
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001938 return NO_ERROR;
1939}
1940
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001941status_t MediaPlayerService::AudioCache::start() {
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001942 if (mCallbackThread != NULL) {
1943 mCallbackThread->run("AudioCache callback");
1944 }
Richard Fitzgeraldd89532e2013-05-14 13:18:21 +01001945 return NO_ERROR;
Andreas Huber7d5b8a72010-02-09 16:59:18 -08001946}
1947
1948void MediaPlayerService::AudioCache::stop() {
1949 if (mCallbackThread != NULL) {
1950 mCallbackThread->requestExitAndWait();
1951 }
1952}
1953
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001954ssize_t MediaPlayerService::AudioCache::write(const void* buffer, size_t size)
1955{
Steve Block3856b092011-10-20 11:56:00 +01001956 ALOGV("write(%p, %u)", buffer, size);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001957 if ((buffer == 0) || (size == 0)) return size;
1958
1959 uint8_t* p = static_cast<uint8_t*>(mHeap->getBase());
1960 if (p == NULL) return NO_INIT;
1961 p += mSize;
Steve Block3856b092011-10-20 11:56:00 +01001962 ALOGV("memcpy(%p, %p, %u)", p, buffer, size);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001963 if (mSize + size > mHeap->getSize()) {
Steve Block29357bc2012-01-06 19:20:56 +00001964 ALOGE("Heap size overflow! req size: %d, max size: %d", (mSize + size), mHeap->getSize());
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001965 size = mHeap->getSize() - mSize;
1966 }
1967 memcpy(p, buffer, size);
1968 mSize += size;
1969 return size;
1970}
1971
1972// call with lock held
1973status_t MediaPlayerService::AudioCache::wait()
1974{
1975 Mutex::Autolock lock(mLock);
Dave Sparks4bbc0ba2010-03-01 19:29:58 -08001976 while (!mCommandComplete) {
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001977 mSignal.wait(mLock);
1978 }
1979 mCommandComplete = false;
1980
1981 if (mError == NO_ERROR) {
Steve Block3856b092011-10-20 11:56:00 +01001982 ALOGV("wait - success");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001983 } else {
Steve Block3856b092011-10-20 11:56:00 +01001984 ALOGV("wait - error");
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001985 }
1986 return mError;
1987}
1988
Gloria Wangb483c472011-04-11 17:23:27 -07001989void MediaPlayerService::AudioCache::notify(
1990 void* cookie, int msg, int ext1, int ext2, const Parcel *obj)
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001991{
Steve Block3856b092011-10-20 11:56:00 +01001992 ALOGV("notify(%p, %d, %d, %d)", cookie, msg, ext1, ext2);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08001993 AudioCache* p = static_cast<AudioCache*>(cookie);
1994
1995 // ignore buffering messages
Dave Sparks8eb80112009-12-09 20:20:26 -08001996 switch (msg)
1997 {
1998 case MEDIA_ERROR:
Steve Block29357bc2012-01-06 19:20:56 +00001999 ALOGE("Error %d, %d occurred", ext1, ext2);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08002000 p->mError = ext1;
Dave Sparks8eb80112009-12-09 20:20:26 -08002001 break;
2002 case MEDIA_PREPARED:
Steve Block3856b092011-10-20 11:56:00 +01002003 ALOGV("prepared");
Dave Sparks8eb80112009-12-09 20:20:26 -08002004 break;
2005 case MEDIA_PLAYBACK_COMPLETE:
Steve Block3856b092011-10-20 11:56:00 +01002006 ALOGV("playback complete");
Dave Sparks8eb80112009-12-09 20:20:26 -08002007 break;
2008 default:
Steve Block3856b092011-10-20 11:56:00 +01002009 ALOGV("ignored");
Dave Sparks8eb80112009-12-09 20:20:26 -08002010 return;
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08002011 }
2012
2013 // wake up thread
Dave Sparksfe4c6f02010-03-02 12:56:37 -08002014 Mutex::Autolock lock(p->mLock);
The Android Open Source Project89fa4ad2009-03-03 19:31:44 -08002015 p->mCommandComplete = true;
2016 p->mSignal.signal();
2017}
2018
Marco Nelissen4110c102012-03-29 09:31:28 -07002019int MediaPlayerService::AudioCache::getSessionId() const
Eric Laurent8c563ed2010-10-07 18:23:03 -07002020{
2021 return 0;
2022}
2023
Eric Laurent6f59db12013-07-26 17:16:50 -07002024uint32_t MediaPlayerService::AudioCache::getSampleRate() const
2025{
2026 if (mMsecsPerFrame == 0) {
2027 return 0;
2028 }
2029 return (uint32_t)(1.e3 / mMsecsPerFrame);
2030}
2031
Gloria Wang7cf180c2011-02-19 18:37:57 -08002032void MediaPlayerService::addBatteryData(uint32_t params)
2033{
2034 Mutex::Autolock lock(mLock);
Gloria Wang9ee159b2011-02-24 14:51:45 -08002035
2036 int32_t time = systemTime() / 1000000L;
2037
2038 // change audio output devices. This notification comes from AudioFlinger
2039 if ((params & kBatteryDataSpeakerOn)
2040 || (params & kBatteryDataOtherAudioDeviceOn)) {
2041
2042 int deviceOn[NUM_AUDIO_DEVICES];
2043 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2044 deviceOn[i] = 0;
2045 }
2046
2047 if ((params & kBatteryDataSpeakerOn)
2048 && (params & kBatteryDataOtherAudioDeviceOn)) {
2049 deviceOn[SPEAKER_AND_OTHER] = 1;
2050 } else if (params & kBatteryDataSpeakerOn) {
2051 deviceOn[SPEAKER] = 1;
2052 } else {
2053 deviceOn[OTHER_AUDIO_DEVICE] = 1;
2054 }
2055
2056 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2057 if (mBatteryAudio.deviceOn[i] != deviceOn[i]){
2058
2059 if (mBatteryAudio.refCount > 0) { // if playing audio
2060 if (!deviceOn[i]) {
2061 mBatteryAudio.lastTime[i] += time;
2062 mBatteryAudio.totalTime[i] += mBatteryAudio.lastTime[i];
2063 mBatteryAudio.lastTime[i] = 0;
2064 } else {
2065 mBatteryAudio.lastTime[i] = 0 - time;
2066 }
2067 }
2068
2069 mBatteryAudio.deviceOn[i] = deviceOn[i];
2070 }
2071 }
2072 return;
2073 }
2074
2075 // an sudio stream is started
2076 if (params & kBatteryDataAudioFlingerStart) {
2077 // record the start time only if currently no other audio
2078 // is being played
2079 if (mBatteryAudio.refCount == 0) {
2080 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2081 if (mBatteryAudio.deviceOn[i]) {
2082 mBatteryAudio.lastTime[i] -= time;
2083 }
2084 }
2085 }
2086
2087 mBatteryAudio.refCount ++;
2088 return;
2089
2090 } else if (params & kBatteryDataAudioFlingerStop) {
2091 if (mBatteryAudio.refCount <= 0) {
Steve Block5ff1dd52012-01-05 23:22:43 +00002092 ALOGW("Battery track warning: refCount is <= 0");
Gloria Wang9ee159b2011-02-24 14:51:45 -08002093 return;
2094 }
2095
2096 // record the stop time only if currently this is the only
2097 // audio being played
2098 if (mBatteryAudio.refCount == 1) {
2099 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2100 if (mBatteryAudio.deviceOn[i]) {
2101 mBatteryAudio.lastTime[i] += time;
2102 mBatteryAudio.totalTime[i] += mBatteryAudio.lastTime[i];
2103 mBatteryAudio.lastTime[i] = 0;
2104 }
2105 }
2106 }
2107
2108 mBatteryAudio.refCount --;
2109 return;
2110 }
2111
Gloria Wang7cf180c2011-02-19 18:37:57 -08002112 int uid = IPCThreadState::self()->getCallingUid();
2113 if (uid == AID_MEDIA) {
2114 return;
2115 }
2116 int index = mBatteryData.indexOfKey(uid);
Gloria Wang7cf180c2011-02-19 18:37:57 -08002117
2118 if (index < 0) { // create a new entry for this UID
2119 BatteryUsageInfo info;
2120 info.audioTotalTime = 0;
2121 info.videoTotalTime = 0;
2122 info.audioLastTime = 0;
2123 info.videoLastTime = 0;
2124 info.refCount = 0;
2125
Gloria Wang9ee159b2011-02-24 14:51:45 -08002126 if (mBatteryData.add(uid, info) == NO_MEMORY) {
Steve Block29357bc2012-01-06 19:20:56 +00002127 ALOGE("Battery track error: no memory for new app");
Gloria Wang9ee159b2011-02-24 14:51:45 -08002128 return;
2129 }
Gloria Wang7cf180c2011-02-19 18:37:57 -08002130 }
2131
2132 BatteryUsageInfo &info = mBatteryData.editValueFor(uid);
2133
2134 if (params & kBatteryDataCodecStarted) {
2135 if (params & kBatteryDataTrackAudio) {
2136 info.audioLastTime -= time;
2137 info.refCount ++;
2138 }
2139 if (params & kBatteryDataTrackVideo) {
2140 info.videoLastTime -= time;
2141 info.refCount ++;
2142 }
2143 } else {
2144 if (info.refCount == 0) {
Steve Block5ff1dd52012-01-05 23:22:43 +00002145 ALOGW("Battery track warning: refCount is already 0");
Gloria Wang7cf180c2011-02-19 18:37:57 -08002146 return;
2147 } else if (info.refCount < 0) {
Steve Block29357bc2012-01-06 19:20:56 +00002148 ALOGE("Battery track error: refCount < 0");
Gloria Wang7cf180c2011-02-19 18:37:57 -08002149 mBatteryData.removeItem(uid);
2150 return;
2151 }
2152
2153 if (params & kBatteryDataTrackAudio) {
2154 info.audioLastTime += time;
2155 info.refCount --;
2156 }
2157 if (params & kBatteryDataTrackVideo) {
2158 info.videoLastTime += time;
2159 info.refCount --;
2160 }
2161
2162 // no stream is being played by this UID
2163 if (info.refCount == 0) {
2164 info.audioTotalTime += info.audioLastTime;
2165 info.audioLastTime = 0;
2166 info.videoTotalTime += info.videoLastTime;
2167 info.videoLastTime = 0;
2168 }
2169 }
2170}
2171
2172status_t MediaPlayerService::pullBatteryData(Parcel* reply) {
2173 Mutex::Autolock lock(mLock);
Gloria Wang9ee159b2011-02-24 14:51:45 -08002174
2175 // audio output devices usage
2176 int32_t time = systemTime() / 1000000L; //in ms
2177 int32_t totalTime;
2178
2179 for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2180 totalTime = mBatteryAudio.totalTime[i];
2181
2182 if (mBatteryAudio.deviceOn[i]
2183 && (mBatteryAudio.lastTime[i] != 0)) {
2184 int32_t tmpTime = mBatteryAudio.lastTime[i] + time;
2185 totalTime += tmpTime;
2186 }
2187
2188 reply->writeInt32(totalTime);
2189 // reset the total time
2190 mBatteryAudio.totalTime[i] = 0;
2191 }
2192
2193 // codec usage
Gloria Wang7cf180c2011-02-19 18:37:57 -08002194 BatteryUsageInfo info;
2195 int size = mBatteryData.size();
2196
2197 reply->writeInt32(size);
2198 int i = 0;
2199
2200 while (i < size) {
2201 info = mBatteryData.valueAt(i);
2202
2203 reply->writeInt32(mBatteryData.keyAt(i)); //UID
2204 reply->writeInt32(info.audioTotalTime);
2205 reply->writeInt32(info.videoTotalTime);
2206
2207 info.audioTotalTime = 0;
2208 info.videoTotalTime = 0;
2209
2210 // remove the UID entry where no stream is being played
2211 if (info.refCount <= 0) {
2212 mBatteryData.removeItemsAt(i);
2213 size --;
2214 i --;
2215 }
2216 i++;
2217 }
2218 return NO_ERROR;
2219}
nikoa64c8c72009-07-20 15:07:26 -07002220} // namespace android