MediaPlayerService.cpp revision 67295b543c16ed7c77b67f859d598fe7b4571660
189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/*
289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Copyright 2008, The Android Open Source Project
489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Licensed under the Apache License, Version 2.0 (the "License");
689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** you may not use this file except in compliance with the License.
789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** You may obtain a copy of the License at
889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**     http://www.apache.org/licenses/LICENSE-2.0
1089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project**
1189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** Unless required by applicable law or agreed to in writing, software
1289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** distributed under the License is distributed on an "AS IS" BASIS,
1389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** See the License for the specific language governing permissions and
1589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project** limitations under the License.
1689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project*/
1789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// Proxy for media player implementations
1989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
2089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project//#define LOG_NDEBUG 0
2189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#define LOG_TAG "MediaPlayerService"
2289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <utils/Log.h>
2389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
2489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/types.h>
2589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <sys/stat.h>
267cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang#include <sys/time.h>
2789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <dirent.h>
2889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <unistd.h>
2989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
3089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <string.h>
316f74b0cc490a3b8523252ded00f7ca55160effd1Mathias Agopian
3289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <cutils/atomic.h>
3314d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania#include <cutils/properties.h> // for property_get
346f74b0cc490a3b8523252ded00f7ca55160effd1Mathias Agopian
356f74b0cc490a3b8523252ded00f7ca55160effd1Mathias Agopian#include <utils/misc.h>
3689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
377562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/IPCThreadState.h>
387562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/IServiceManager.h>
397562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/MemoryHeapBase.h>
407562408b2261d38415453378b6188f74fda99d88Mathias Agopian#include <binder/MemoryBase.h>
417dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis#include <gui/SurfaceTextureClient.h>
421d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania#include <utils/Errors.h>  // for status_t
431d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania#include <utils/String8.h>
4410dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen#include <utils/SystemClock.h>
451d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania#include <utils/Vector.h>
4689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <cutils/properties.h>
4789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/MediaPlayerInterface.h>
4989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/mediarecorder.h>
5089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/MediaMetadataRetrieverInterface.h>
51a64c8c79af1a15911c55306d83a797fa50969f77niko#include <media/Metadata.h>
5289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include <media/AudioTrack.h>
538635b7b095fbf7ffc63d3ce791891a9116ace1f6James Dong#include <media/MemoryLeakTrackUtil.h>
549cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent#include <media/stagefright/MediaErrors.h>
5589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
5664760240f931714858a59c1579f07264d7182ba2Dima Zavin#include <system/audio.h>
57fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin
587cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang#include <private/android_filesystem_config.h>
597cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
60559bf2836f5da25b75bfb229fec0d20d540ee426James Dong#include "ActivityManager.h"
6189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include "MediaRecorderClient.h"
6289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include "MediaPlayerService.h"
6389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include "MetadataRetrieverClient.h"
6489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
6589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#include "MidiFile.h"
6614d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania#include "TestPlayerStub.h"
6720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#include "StagefrightPlayer.h"
68f933441648ef6a71dee783d733aac17b9508b452Andreas Huber#include "nuplayer/NuPlayerDriver.h"
6920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
7020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber#include <OMX.h>
7114d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania
72ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber#include "Crypto.h"
73ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
74761defc341c5ce9019a42919c441f035f665ec0dJohn Grossmannamespace android {
75761defc341c5ce9019a42919c441f035f665ec0dJohn Grossmansp<MediaPlayerBase> createAAH_TXPlayer();
76761defc341c5ce9019a42919c441f035f665ec0dJohn Grossmansp<MediaPlayerBase> createAAH_RXPlayer();
77761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman}
78761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman
79a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catanianamespace {
80a64c8c79af1a15911c55306d83a797fa50969f77nikousing android::media::Metadata;
81a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniausing android::status_t;
82a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniausing android::OK;
83a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniausing android::BAD_VALUE;
84a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniausing android::NOT_ENOUGH_DATA;
85a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniausing android::Parcel;
86a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
87a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// Max number of entries in the filter.
88a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniaconst int kMaxFilterSize = 64;  // I pulled that out of thin air.
89a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
90a64c8c79af1a15911c55306d83a797fa50969f77niko// FIXME: Move all the metadata related function in the Metadata.cpp
91d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko
92a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
93a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// Unmarshall a filter from a Parcel.
94a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// Filter format in a parcel:
95a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania//
96a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania//  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
97a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
98a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// |                       number of entries (n)                   |
99a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
100a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// |                       metadata type 1                         |
101a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
102a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// |                       metadata type 2                         |
103a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
104a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania//  ....
105a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
106a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// |                       metadata type n                         |
107a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
108a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania//
109a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @param p Parcel that should start with a filter.
110a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @param[out] filter On exit contains the list of metadata type to be
111a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania//                    filtered.
112a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @param[out] status On exit contains the status code to be returned.
113a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @return true if the parcel starts with a valid filter.
114a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniabool unmarshallFilter(const Parcel& p,
115a64c8c79af1a15911c55306d83a797fa50969f77niko                      Metadata::Filter *filter,
116a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania                      status_t *status)
117a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania{
1184829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    int32_t val;
1194829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (p.readInt32(&val) != OK)
120a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    {
12129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Failed to read filter's length");
122a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        *status = NOT_ENOUGH_DATA;
123a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return false;
124a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
125a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1264829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if( val > kMaxFilterSize || val < 0)
127a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    {
12829357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Invalid filter len %d", val);
129a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        *status = BAD_VALUE;
130a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return false;
131a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
132a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1334829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    const size_t num = val;
134a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
135a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    filter->clear();
1364829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    filter->setCapacity(num);
137a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
138a64c8c79af1a15911c55306d83a797fa50969f77niko    size_t size = num * sizeof(Metadata::Type);
139a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1404829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
1414829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (p.dataAvail() < size)
142a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    {
14329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Filter too short expected %d but got %d", size, p.dataAvail());
144a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        *status = NOT_ENOUGH_DATA;
145a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return false;
146a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
147a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
148a64c8c79af1a15911c55306d83a797fa50969f77niko    const Metadata::Type *data =
149a64c8c79af1a15911c55306d83a797fa50969f77niko            static_cast<const Metadata::Type*>(p.readInplace(size));
150a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1514829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (NULL == data)
152a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    {
15329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Filter had no data");
154a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        *status = BAD_VALUE;
155a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return false;
156a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
157a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
158a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    // TODO: The stl impl of vector would be more efficient here
159a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    // because it degenerates into a memcpy on pod types. Try to
160a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    // replace later or use stl::set.
1614829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    for (size_t i = 0; i < num; ++i)
162a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    {
1634829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        filter->add(*data);
164a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        ++data;
165a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
166a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    *status = OK;
167a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    return true;
168a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}
169a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1704829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania// @param filter Of metadata type.
171a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @param val To be searched.
172a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// @return true if a match was found.
173a64c8c79af1a15911c55306d83a797fa50969f77nikobool findMetadata(const Metadata::Filter& filter, const int32_t val)
174a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania{
175a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    // Deal with empty and ANY right away
176a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    if (filter.isEmpty()) return false;
177a64c8c79af1a15911c55306d83a797fa50969f77niko    if (filter[0] == Metadata::kAny) return true;
178a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
1794829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    return filter.indexOf(val) >= 0;
180a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}
181a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
182a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}  // anonymous namespace
183a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
184a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
18589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectnamespace android {
18689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
187d681bbb1767bed09415e050ba78975df214bcd68Dave Burkestatic bool checkPermission(const char* permissionString) {
188d681bbb1767bed09415e050ba78975df214bcd68Dave Burke#ifndef HAVE_ANDROID_OS
189d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    return true;
190d681bbb1767bed09415e050ba78975df214bcd68Dave Burke#endif
191d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    if (getpid() == IPCThreadState::self()->getCallingPid()) return true;
192d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    bool ok = checkCallingPermission(String16(permissionString));
19329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block    if (!ok) ALOGE("Request requires %s", permissionString);
194d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    return ok;
195d681bbb1767bed09415e050ba78975df214bcd68Dave Burke}
196d681bbb1767bed09415e050ba78975df214bcd68Dave Burke
19789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// TODO: Temp hack until we can register players
19889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projecttypedef struct {
19989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const char *extension;
20089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const player_type playertype;
20189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project} extmap;
20289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectextmap FILE_EXTS [] =  {
20389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".mid", SONIVOX_PLAYER},
20489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".midi", SONIVOX_PLAYER},
20589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".smf", SONIVOX_PLAYER},
20689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".xmf", SONIVOX_PLAYER},
20789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".imy", SONIVOX_PLAYER},
20889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".rtttl", SONIVOX_PLAYER},
20989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".rtx", SONIVOX_PLAYER},
21089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {".ota", SONIVOX_PLAYER},
21189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project};
21289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
21389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// TODO: Find real cause of Audio/Video delay in PV framework and remove this workaround
21489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/* static */ int MediaPlayerService::AudioOutput::mMinBufferCount = 4;
21589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project/* static */ bool MediaPlayerService::AudioOutput::mIsOnEmulator = false;
21689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
21789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::instantiate() {
21889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    defaultServiceManager()->addService(
21989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            String16("media.player"), new MediaPlayerService());
22089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
22189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
22289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayerService::MediaPlayerService()
22389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
2243856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("MediaPlayerService created");
22589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mNextConnId = 1;
2269ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
2279ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    mBatteryAudio.refCount = 0;
2289ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
2299ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.deviceOn[i] = 0;
2309ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.lastTime[i] = 0;
2319ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.totalTime[i] = 0;
2329ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    }
2339ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    // speaker is on by default
2349ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    mBatteryAudio.deviceOn[SPEAKER] = 1;
23589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
23689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
23789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayerService::~MediaPlayerService()
23889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
2393856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("MediaPlayerService destroyed");
24089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
24189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
24289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<IMediaRecorder> MediaPlayerService::createMediaRecorder(pid_t pid)
24389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
244dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    sp<MediaRecorderClient> recorder = new MediaRecorderClient(this, pid);
245dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    wp<MediaRecorderClient> w = recorder;
246dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    Mutex::Autolock lock(mLock);
247dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    mMediaRecorderClients.add(w);
2483856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Create new media recorder client from pid %d", pid);
24989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return recorder;
25089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
25189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
252dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wangvoid MediaPlayerService::removeMediaRecorderClient(wp<MediaRecorderClient> client)
253dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang{
254dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    Mutex::Autolock lock(mLock);
255dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang    mMediaRecorderClients.remove(client);
2563856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Delete media recorder client");
257dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang}
258dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang
25989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<IMediaMetadataRetriever> MediaPlayerService::createMetadataRetriever(pid_t pid)
26089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
26189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MetadataRetrieverClient> retriever = new MetadataRetrieverClient(pid);
2623856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Create new media retriever from pid %d", pid);
26389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return retriever;
26489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
26589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
26689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<IMediaPlayer> MediaPlayerService::create(pid_t pid, const sp<IMediaPlayerClient>& client,
267d681bbb1767bed09415e050ba78975df214bcd68Dave Burke        int audioSessionId)
26889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
26989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    int32_t connId = android_atomic_inc(&mNextConnId);
2709b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber
2719b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber    sp<Client> c = new Client(
2729b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber            this, pid, connId, client, audioSessionId,
2739b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber            IPCThreadState::self()->getCallingUid());
2749b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber
2753856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Create new client(%d) from pid %d, uid %d, ", connId, pid,
276d681bbb1767bed09415e050ba78975df214bcd68Dave Burke         IPCThreadState::self()->getCallingUid());
277e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber
278d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    wp<Client> w = c;
279d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    {
280e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber        Mutex::Autolock lock(mLock);
281e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber        mClients.add(w);
282e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    }
283e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    return c;
284e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber}
285e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber
286318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Hubersp<IOMX> MediaPlayerService::getOMX() {
287318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    Mutex::Autolock autoLock(mLock);
288318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
289318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    if (mOMX.get() == NULL) {
290318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber        mOMX = new OMX;
291318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    }
292318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber
293318ad9c1d9d6515026dfc2c021359d27decaa7a1Andreas Huber    return mOMX;
29420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber}
29520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
296ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Hubersp<ICrypto> MediaPlayerService::makeCrypto() {
2971bd139a2a68690e80398b70b27ca59550fea0e65Andreas Huber    return new Crypto;
298ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber}
299ed3e3e046840d5bf1ca84a8c0cc097425e89d6d6Andreas Huber
30089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::AudioCache::dump(int fd, const Vector<String16>& args) const
30189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
30289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const size_t SIZE = 256;
30389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char buffer[SIZE];
30489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    String8 result;
30589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
30689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(" AudioCache\n");
30789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mHeap != 0) {
30889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        snprintf(buffer, 255, "  heap base(%p), size(%d), flags(%d), device(%s)\n",
30989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                mHeap->getBase(), mHeap->getSize(), mHeap->getFlags(), mHeap->getDevice());
31089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        result.append(buffer);
31189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
31289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    snprintf(buffer, 255, "  msec per frame(%f), channel count(%d), format(%d), frame count(%ld)\n",
31389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mMsecsPerFrame, mChannelCount, mFormat, mFrameCount);
31489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(buffer);
31589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    snprintf(buffer, 255, "  sample rate(%d), size(%d), error(%d), command complete(%s)\n",
31689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mSampleRate, mSize, mError, mCommandComplete?"true":"false");
31789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(buffer);
31889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    ::write(fd, result.string(), result.size());
31989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
32089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
32189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
32289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::AudioOutput::dump(int fd, const Vector<String16>& args) const
32389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
32489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const size_t SIZE = 256;
32589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char buffer[SIZE];
32689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    String8 result;
32789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
32889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(" AudioOutput\n");
32989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    snprintf(buffer, 255, "  stream type(%d), left - right volume(%f, %f)\n",
33089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mStreamType, mLeftVolume, mRightVolume);
33189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(buffer);
33289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    snprintf(buffer, 255, "  msec per frame(%f), latency (%d)\n",
333db354e58e65592777aa17caa47933e14838b8b35Eric Laurent            mMsecsPerFrame, (mTrack != 0) ? mTrack->latency() : -1);
33489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(buffer);
3352beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    snprintf(buffer, 255, "  aux effect id(%d), send level (%f)\n",
3362beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent            mAuxEffectId, mSendLevel);
3372beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    result.append(buffer);
3382beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
33989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    ::write(fd, result.string(), result.size());
34089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack != 0) {
34189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mTrack->dump(fd, args);
34289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
34389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
34489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
34589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
34689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::dump(int fd, const Vector<String16>& args) const
34789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
34889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const size_t SIZE = 256;
34989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char buffer[SIZE];
35089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    String8 result;
35189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(" Client\n");
35289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    snprintf(buffer, 255, "  pid(%d), connId(%d), status(%d), looping(%s)\n",
35389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            mPid, mConnId, mStatus, mLoop?"true": "false");
35489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    result.append(buffer);
35589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    write(fd, result.string(), result.size());
356a0b1d4b161599c2bb2a47119e50c51e75bbe980eAndreas Huber    if (mPlayer != NULL) {
357a0b1d4b161599c2bb2a47119e50c51e75bbe980eAndreas Huber        mPlayer->dump(fd, args);
358a0b1d4b161599c2bb2a47119e50c51e75bbe980eAndreas Huber    }
35989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mAudioOutput != 0) {
36089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mAudioOutput->dump(fd, args);
36189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
36289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    write(fd, "\n", 1);
36389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
36489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
36589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
36689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::dump(int fd, const Vector<String16>& args)
36789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
36889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    const size_t SIZE = 256;
36989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char buffer[SIZE];
37089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    String8 result;
37189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (checkCallingPermission(String16("android.permission.DUMP")) == false) {
37289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        snprintf(buffer, SIZE, "Permission Denial: "
37389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                "can't dump MediaPlayerService from pid=%d, uid=%d\n",
37489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                IPCThreadState::self()->getCallingPid(),
37589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                IPCThreadState::self()->getCallingUid());
37689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        result.append(buffer);
37789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
37889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock lock(mLock);
37989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        for (int i = 0, n = mClients.size(); i < n; ++i) {
38089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            sp<Client> c = mClients[i].promote();
38189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (c != 0) c->dump(fd, args);
38289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
383b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong        if (mMediaRecorderClients.size() == 0) {
384b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong                result.append(" No media recorder client\n\n");
385b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong        } else {
386b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong            for (int i = 0, n = mMediaRecorderClients.size(); i < n; ++i) {
387b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong                sp<MediaRecorderClient> c = mMediaRecorderClients[i].promote();
388e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                if (c != 0) {
389e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                    snprintf(buffer, 255, " MediaRecorderClient pid(%d)\n", c->mPid);
390e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                    result.append(buffer);
391e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                    write(fd, result.string(), result.size());
392e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                    result = "\n";
393e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                    c->dump(fd, args);
394e579e28089c6b86f56d257030b778c5e22b1a5a5James Dong                }
395b914122eb9cb54bbeae4ec03bfebb194aecdccbdJames Dong            }
396dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang        }
397dac6a31a33ba53fb93850670cdddd1e6515dadceGloria Wang
39889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        result.append(" Files opened and/or mapped:\n");
3990512ab559d4670c2204078470d7ef5d376811c57Glenn Kasten        snprintf(buffer, SIZE, "/proc/%d/maps", gettid());
40089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        FILE *f = fopen(buffer, "r");
40189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (f) {
40289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            while (!feof(f)) {
40389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                fgets(buffer, SIZE, f);
40473ac1eebff620714c6977f7342ff8dcc4858b3cfMarco Nelissen                if (strstr(buffer, " /storage/") ||
40589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    strstr(buffer, " /system/sounds/") ||
40602fa834249a01144d6f23137951538600aa5d611Dave Sparks                    strstr(buffer, " /data/") ||
40789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    strstr(buffer, " /system/media/")) {
40889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    result.append("  ");
40989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    result.append(buffer);
41089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                }
41189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
41289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            fclose(f);
41389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
41489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append("couldn't open ");
41589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append(buffer);
41689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append("\n");
41789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
41889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4190512ab559d4670c2204078470d7ef5d376811c57Glenn Kasten        snprintf(buffer, SIZE, "/proc/%d/fd", gettid());
42089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        DIR *d = opendir(buffer);
42189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (d) {
42289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            struct dirent *ent;
42389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            while((ent = readdir(d)) != NULL) {
42489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                if (strcmp(ent->d_name,".") && strcmp(ent->d_name,"..")) {
4250512ab559d4670c2204078470d7ef5d376811c57Glenn Kasten                    snprintf(buffer, SIZE, "/proc/%d/fd/%s", gettid(), ent->d_name);
42689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    struct stat s;
42789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    if (lstat(buffer, &s) == 0) {
42889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                        if ((s.st_mode & S_IFMT) == S_IFLNK) {
42989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            char linkto[256];
43089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            int len = readlink(buffer, linkto, sizeof(linkto));
43189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            if(len > 0) {
43289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                if(len > 255) {
43389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    linkto[252] = '.';
43489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    linkto[253] = '.';
43589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    linkto[254] = '.';
43689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    linkto[255] = 0;
43789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                } else {
43889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    linkto[len] = 0;
43989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                }
44073ac1eebff620714c6977f7342ff8dcc4858b3cfMarco Nelissen                                if (strstr(linkto, "/storage/") == linkto ||
44189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    strstr(linkto, "/system/sounds/") == linkto ||
44202fa834249a01144d6f23137951538600aa5d611Dave Sparks                                    strstr(linkto, "/data/") == linkto ||
44389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    strstr(linkto, "/system/media/") == linkto) {
44489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    result.append("  ");
44589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    result.append(buffer);
44689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    result.append(" -> ");
44789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    result.append(linkto);
44889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                    result.append("\n");
44989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                                }
45089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            }
45189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                        } else {
45289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            result.append("  unexpected type for ");
45389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            result.append(buffer);
45489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                            result.append("\n");
45589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                        }
45689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                    }
45789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                }
45889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
45989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            closedir(d);
46089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
46189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append("couldn't open ");
46289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append(buffer);
46389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            result.append("\n");
46489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
46589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
46689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        bool dumpMem = false;
46789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        for (size_t i = 0; i < args.size(); i++) {
46889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            if (args[i] == String16("-m")) {
46989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project                dumpMem = true;
47089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
47189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
47289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (dumpMem) {
4738635b7b095fbf7ffc63d3ce791891a9116ace1f6James Dong            dumpMemoryAddresses(fd);
47489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
47589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
47689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    write(fd, result.string(), result.size());
47789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
47889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
47989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
48089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::removeClient(wp<Client> client)
48189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
48289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock lock(mLock);
48389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mClients.remove(client);
48489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
48589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
4869b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas HuberMediaPlayerService::Client::Client(
4879b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber        const sp<MediaPlayerService>& service, pid_t pid,
4889b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber        int32_t connId, const sp<IMediaPlayerClient>& client,
4899b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber        int audioSessionId, uid_t uid)
49089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
4913856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Client(%d) constructor", connId);
49289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPid = pid;
49389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mConnId = connId;
49489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mService = service;
49589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mClient = client;
49689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = false;
49789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mStatus = NO_INIT;
498a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent    mAudioSessionId = audioSessionId;
4999b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber    mUID = uid;
500c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    mRetransmitEndpointValid = false;
501a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent
50289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#if CALLBACK_ANTAGONIZER
503b8a805261bf0282e992d3608035e47d05a898710Steve Block    ALOGD("create Antagonizer");
50489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mAntagonizer = new Antagonizer(notify, this);
50589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#endif
50689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
50789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
50889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayerService::Client::~Client()
50989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
5103856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("Client(%d) destructor pid = %d", mConnId, mPid);
51189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mAudioOutput.clear();
51289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    wp<Client> client(this);
51389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    disconnect();
51489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mService->removeClient(client);
51589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
51689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
51789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::Client::disconnect()
51889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
5193856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("disconnect(%d) from pid %d", mConnId, mPid);
52089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // grab local reference and clear main reference to prevent future
52189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // access to object
52289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p;
52389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    {
52489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        Mutex::Autolock l(mLock);
52589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = mPlayer;
52689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
527795fa5848276bab87445fa64fd0941c13ef62398Dave Sparks    mClient.clear();
52820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
52989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mPlayer.clear();
53089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
53189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // clear the notification to prevent callbacks to dead client
53289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // and reset the player. We assume the player will serialize
53389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // access to itself if necessary.
53489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) {
53589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->setNotifyCallback(0, 0);
53689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#if CALLBACK_ANTAGONIZER
537b8a805261bf0282e992d3608035e47d05a898710Steve Block        ALOGD("kill Antagonizer");
53889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mAntagonizer->kill();
53989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#endif
54089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->reset();
54189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
54289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
5437dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    disconnectNativeWindow();
5447dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
54589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    IPCThreadState::self()->flushCommands();
54689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
54789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
54847f59cfe40b0ebb3afd6c39f97dcd1eac2e966f7Andreas Huberstatic player_type getDefaultPlayerType() {
549afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber    char value[PROPERTY_VALUE_MAX];
550afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber    if (property_get("media.stagefright.use-nuplayer", value, NULL)
551afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber            && (!strcmp("1", value) || !strcasecmp("true", value))) {
552afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber        return NU_PLAYER;
553afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber    }
554afed0e1fa37473a4cd30018577b560acc79d9a3fAndreas Huber
555608d77b1cf4fb9f63dc861e4e1fa3e80a732f626Andreas Huber    return STAGEFRIGHT_PLAYER;
5563cf4e43cdd391502690c07e08e805aa8ff5db0c0Andreas Huber}
5573cf4e43cdd391502690c07e08e805aa8ff5db0c0Andreas Huber
558148c1a2a96774517407717b61e5bc9cb08be8806James Dongplayer_type getPlayerType(int fd, int64_t offset, int64_t length)
55989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
56089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char buf[20];
56189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    lseek(fd, offset, SEEK_SET);
56289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    read(fd, buf, sizeof(buf));
56389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    lseek(fd, offset, SEEK_SET);
56489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
56589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    long ident = *((long*)buf);
56689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
56789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Ogg vorbis?
56889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ident == 0x5367674f) // 'OggS'
569608d77b1cf4fb9f63dc861e4e1fa3e80a732f626Andreas Huber        return STAGEFRIGHT_PLAYER;
57089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
57189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Some kind of MIDI?
57289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    EAS_DATA_HANDLE easdata;
57389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (EAS_Init(&easdata) == EAS_SUCCESS) {
57489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        EAS_FILE locator;
57589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        locator.path = NULL;
57689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        locator.fd = fd;
57789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        locator.offset = offset;
57889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        locator.length = length;
57989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        EAS_HANDLE  eashandle;
58089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (EAS_OpenFile(easdata, &locator, &eashandle) == EAS_SUCCESS) {
58189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            EAS_CloseFile(easdata, eashandle);
58289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            EAS_Shutdown(easdata);
58389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return SONIVOX_PLAYER;
58489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
58589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        EAS_Shutdown(easdata);
58689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
58789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
58847f59cfe40b0ebb3afd6c39f97dcd1eac2e966f7Andreas Huber    return getDefaultPlayerType();
58989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
59089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
591148c1a2a96774517407717b61e5bc9cb08be8806James Dongplayer_type getPlayerType(const char* url)
59289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
59314d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania    if (TestPlayerStub::canBeUsed(url)) {
59414d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania        return TEST_PLAYER;
59514d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania    }
59614d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania
5970a73d81f213fb2d6f2fdd59c4dda047bf453bb1cAndreas Huber    if (!strncasecmp("http://", url, 7)
5980a73d81f213fb2d6f2fdd59c4dda047bf453bb1cAndreas Huber            || !strncasecmp("https://", url, 8)) {
599ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber        size_t len = strlen(url);
600ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber        if (len >= 5 && !strcasecmp(".m3u8", &url[len - 5])) {
601ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber            return NU_PLAYER;
602ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber        }
6035bc087c573c70c84c6a39946457590b42d392a33Andreas Huber
604ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber        if (strstr(url,"m3u8")) {
605ed8d14f6a934072cd012992c4ef16990a54baa9aAndreas Huber            return NU_PLAYER;
6065bc087c573c70c84c6a39946457590b42d392a33Andreas Huber        }
6075bc087c573c70c84c6a39946457590b42d392a33Andreas Huber    }
6085bc087c573c70c84c6a39946457590b42d392a33Andreas Huber
6092bfdd428c56c7524d1a11979f200a1762866032dAndreas Huber    if (!strncasecmp("rtsp://", url, 7)) {
6102bfdd428c56c7524d1a11979f200a1762866032dAndreas Huber        return NU_PLAYER;
6112bfdd428c56c7524d1a11979f200a1762866032dAndreas Huber    }
6122bfdd428c56c7524d1a11979f200a1762866032dAndreas Huber
613761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    if (!strncasecmp("aahRX://", url, 8)) {
614761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman        return AAH_RX_PLAYER;
615761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    }
616761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman
61789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // use MidiFile for MIDI extensions
61889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    int lenURL = strlen(url);
61989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    for (int i = 0; i < NELEM(FILE_EXTS); ++i) {
62089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        int len = strlen(FILE_EXTS[i].extension);
62189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        int start = lenURL - len;
62289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (start > 0) {
623fc1c7b940c84793ad262a345af117c482a86acecAtsushi Eno            if (!strncasecmp(url + start, FILE_EXTS[i].extension, len)) {
624608d77b1cf4fb9f63dc861e4e1fa3e80a732f626Andreas Huber                return FILE_EXTS[i].playertype;
62589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            }
62689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
62789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
62889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
62947f59cfe40b0ebb3afd6c39f97dcd1eac2e966f7Andreas Huber    return getDefaultPlayerType();
63089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
63189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
632c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmanplayer_type MediaPlayerService::Client::getPlayerType(int fd,
633c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman                                                      int64_t offset,
634c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman                                                      int64_t length)
635c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman{
636c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Until re-transmit functionality is added to the existing core android
637c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // players, we use the special AAH TX player whenever we were configured
638c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // for retransmission.
639c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mRetransmitEndpointValid) {
640c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return AAH_TX_PLAYER;
641c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
642c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
643c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    return android::getPlayerType(fd, offset, length);
644c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
645c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
646c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmanplayer_type MediaPlayerService::Client::getPlayerType(const char* url)
647c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman{
648c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Until re-transmit functionality is added to the existing core android
649c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // players, we use the special AAH TX player whenever we were configured
650c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // for retransmission.
651c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mRetransmitEndpointValid) {
652c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return AAH_TX_PLAYER;
653c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
654c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
655c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    return android::getPlayerType(url);
656c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
657c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
658c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmanplayer_type MediaPlayerService::Client::getPlayerType(
659c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        const sp<IStreamSource> &source) {
660c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Until re-transmit functionality is added to the existing core android
661c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // players, we use the special AAH TX player whenever we were configured
662c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // for retransmission.
663c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mRetransmitEndpointValid) {
664c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return AAH_TX_PLAYER;
665c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
666c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
667c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    return NU_PLAYER;
668c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
669c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
67089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatic sp<MediaPlayerBase> createPlayer(player_type playerType, void* cookie,
67189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        notify_callback_f notifyFunc)
67289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
67389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p;
67489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    switch (playerType) {
67589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        case SONIVOX_PLAYER:
6763856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV(" create MidiFile");
67789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            p = new MidiFile();
67889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            break;
67920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber        case STAGEFRIGHT_PLAYER:
6803856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV(" create StagefrightPlayer");
68120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber            p = new StagefrightPlayer;
68220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber            break;
683f933441648ef6a71dee783d733aac17b9508b452Andreas Huber        case NU_PLAYER:
6843856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV(" create NuPlayer");
685f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            p = new NuPlayerDriver;
686f933441648ef6a71dee783d733aac17b9508b452Andreas Huber            break;
68714d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania        case TEST_PLAYER:
6883856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV("Create Test Player stub");
68914d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania            p = new TestPlayerStub();
69014d2747c7e54037e267bcff78b29e65b2181f0faNicolas Catania            break;
691761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman        case AAH_RX_PLAYER:
692761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            ALOGV(" create A@H RX Player");
693761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            p = createAAH_RXPlayer();
694761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            break;
695761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman        case AAH_TX_PLAYER:
696761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            ALOGV(" create A@H TX Player");
697761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            p = createAAH_TXPlayer();
698761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman            break;
69930d713a1c18a5ff892a7f13b2524ba624b70890aJames Dong        default:
70029357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Unknown player type: %d", playerType);
70130d713a1c18a5ff892a7f13b2524ba624b70890aJames Dong            return NULL;
70289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
70389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != NULL) {
70489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (p->initCheck() == NO_ERROR) {
70589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            p->setNotifyCallback(cookie, notifyFunc);
70689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        } else {
70789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            p.clear();
70889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
70989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
71089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == NULL) {
71129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Failed to create player object");
71289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
71389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
71489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
71589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
71689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectsp<MediaPlayerBase> MediaPlayerService::Client::createPlayer(player_type playerType)
71789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
71889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // determine if we have the right player type
71989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = mPlayer;
72089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((p != NULL) && (p->playerType() != playerType)) {
7213856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("delete player");
72289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p.clear();
72389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
72489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == NULL) {
72589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p = android::createPlayer(playerType, this, notify);
72689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
7279b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber
728db29e5238e28d59978755a2ff2e7e0f05393abdfJason Simmons    if (p != NULL) {
729db29e5238e28d59978755a2ff2e7e0f05393abdfJason Simmons        p->setUID(mUID);
730db29e5238e28d59978755a2ff2e7e0f05393abdfJason Simmons    }
7319b80c2bdb205bc143104f54d0743b6eedd67b14eAndreas Huber
73289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p;
73389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
73489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
735c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmansp<MediaPlayerBase> MediaPlayerService::Client::setDataSource_pre(
736c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        player_type playerType)
737c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman{
738c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    ALOGV("player type = %d", playerType);
739c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
740c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // create the right type of player
741c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    sp<MediaPlayerBase> p = createPlayer(playerType);
742c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (p == NULL) {
743c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return p;
744c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
745c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
746c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (!p->hardwareOutput()) {
747c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mAudioOutput = new AudioOutput(mAudioSessionId);
748c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        static_cast<MediaPlayerInterface*>(p.get())->setAudioSink(mAudioOutput);
749c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
750c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
751c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    return p;
752c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
753c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
754c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmanvoid MediaPlayerService::Client::setDataSource_post(
755c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        const sp<MediaPlayerBase>& p,
756c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        status_t status)
757c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman{
758c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    ALOGV(" setDataSource");
759c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    mStatus = status;
760c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mStatus != OK) {
761c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        ALOGE("  error: %d", mStatus);
762c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return;
763c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
764c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
765c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Set the re-transmission endpoint if one was chosen.
766c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mRetransmitEndpointValid) {
767c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mStatus = p->setRetransmitEndpoint(&mRetransmitEndpoint);
768c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        if (mStatus != NO_ERROR) {
769c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman            ALOGE("setRetransmitEndpoint error: %d", mStatus);
770c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        }
771c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
772c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
773c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (mStatus == OK) {
774c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mPlayer = p;
775c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
776c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
777c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
7782db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huberstatus_t MediaPlayerService::Client::setDataSource(
7792db8455d8f4468a637109d31f319ce02d9d743ecAndreas Huber        const char *url, const KeyedVector<String8, String8> *headers)
78089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
7813856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("setDataSource(%s)", url);
78289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (url == NULL)
78389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return UNKNOWN_ERROR;
78489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
785d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    if ((strncmp(url, "http://", 7) == 0) ||
786d681bbb1767bed09415e050ba78975df214bcd68Dave Burke        (strncmp(url, "https://", 8) == 0) ||
787d681bbb1767bed09415e050ba78975df214bcd68Dave Burke        (strncmp(url, "rtsp://", 7) == 0)) {
788d681bbb1767bed09415e050ba78975df214bcd68Dave Burke        if (!checkPermission("android.permission.INTERNET")) {
789d681bbb1767bed09415e050ba78975df214bcd68Dave Burke            return PERMISSION_DENIED;
790d681bbb1767bed09415e050ba78975df214bcd68Dave Burke        }
791d681bbb1767bed09415e050ba78975df214bcd68Dave Burke    }
792d681bbb1767bed09415e050ba78975df214bcd68Dave Burke
79389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (strncmp(url, "content://", 10) == 0) {
79489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // get a filedescriptor for the content Uri and
79589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        // pass it to the setDataSource(fd) method
79689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
79789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        String16 url16(url);
79889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        int fd = android::openContentProviderFile(url16);
79989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (fd < 0)
80089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        {
80129357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Couldn't open fd for %s", url);
80289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            return UNKNOWN_ERROR;
80389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
80489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        setDataSource(fd, 0, 0x7fffffffffLL); // this sets mStatus
80589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        close(fd);
80689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mStatus;
80789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
80889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        player_type playerType = getPlayerType(url);
809c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        sp<MediaPlayerBase> p = setDataSource_pre(playerType);
810c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        if (p == NULL) {
811c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman            return NO_INIT;
81289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
81389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
814c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        setDataSource_post(p, p->setDataSource(url, headers));
81589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mStatus;
81689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
81789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
81889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
81989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::setDataSource(int fd, int64_t offset, int64_t length)
82089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
8213856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("setDataSource fd=%d, offset=%lld, length=%lld", fd, offset, length);
82289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    struct stat sb;
82389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    int ret = fstat(fd, &sb);
82489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ret != 0) {
82529357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("fstat(%d) failed: %d, %s", fd, ret, strerror(errno));
82689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return UNKNOWN_ERROR;
82789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
82889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
8293856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("st_dev  = %llu", sb.st_dev);
8303856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("st_mode = %u", sb.st_mode);
8313856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("st_uid  = %lu", sb.st_uid);
8323856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("st_gid  = %lu", sb.st_gid);
8333856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("st_size = %llu", sb.st_size);
83489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
83589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (offset >= sb.st_size) {
83629357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("offset error");
83789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        ::close(fd);
83889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return UNKNOWN_ERROR;
83989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
84089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (offset + length > sb.st_size) {
84189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        length = sb.st_size - offset;
8423856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("calculated length = %lld", length);
84389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
84489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
845c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Until re-transmit functionality is added to the existing core android
846c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // players, we use the special AAH TX player whenever we were configured for
847c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // retransmission.
84889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player_type playerType = getPlayerType(fd, offset, length);
849c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    sp<MediaPlayerBase> p = setDataSource_pre(playerType);
850c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (p == NULL) {
851c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        return NO_INIT;
85289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
85389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
85489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // now set data source
855c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    setDataSource_post(p, p->setDataSource(fd, offset, length));
85689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mStatus;
85789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
85889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
859e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huberstatus_t MediaPlayerService::Client::setDataSource(
860e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber        const sp<IStreamSource> &source) {
861e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    // create the right type of player
862c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Until re-transmit functionality is added to the existing core android
863c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // players, we use the special AAH TX player whenever we were configured for
864c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // retransmission.
865c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    player_type playerType = getPlayerType(source);
866c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    sp<MediaPlayerBase> p = setDataSource_pre(playerType);
867e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    if (p == NULL) {
868e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber        return NO_INIT;
869e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    }
870e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber
871e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    // now set data source
872c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    setDataSource_post(p, p->setDataSource(source));
873e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber    return mStatus;
874e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber}
875e2b1028852120bcfded33b8f06f66b780437fe92Andreas Huber
8767dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennisvoid MediaPlayerService::Client::disconnectNativeWindow() {
8777dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    if (mConnectedWindow != NULL) {
8787dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        status_t err = native_window_api_disconnect(mConnectedWindow.get(),
8797dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis                NATIVE_WINDOW_API_MEDIA);
8807dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
8817dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        if (err != OK) {
8825ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("native_window_api_disconnect returned an error: %s (%d)",
8837dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis                    strerror(-err), err);
8847dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        }
8857dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    }
8867dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    mConnectedWindow.clear();
8877dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis}
8887dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
8891173118eace0e9e347cb007f0da817cee87579edGlenn Kastenstatus_t MediaPlayerService::Client::setVideoSurfaceTexture(
8901173118eace0e9e347cb007f0da817cee87579edGlenn Kasten        const sp<ISurfaceTexture>& surfaceTexture)
8911173118eace0e9e347cb007f0da817cee87579edGlenn Kasten{
8923856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setVideoSurfaceTexture(%p)", mConnId, surfaceTexture.get());
8931173118eace0e9e347cb007f0da817cee87579edGlenn Kasten    sp<MediaPlayerBase> p = getPlayer();
8941173118eace0e9e347cb007f0da817cee87579edGlenn Kasten    if (p == 0) return UNKNOWN_ERROR;
8957dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
8967dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    sp<IBinder> binder(surfaceTexture == NULL ? NULL :
8977dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            surfaceTexture->asBinder());
8987dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    if (mConnectedWindowBinder == binder) {
8997dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        return OK;
9007dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    }
9017dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9027dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    sp<ANativeWindow> anw;
9037dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    if (surfaceTexture != NULL) {
9047dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        anw = new SurfaceTextureClient(surfaceTexture);
9057dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        status_t err = native_window_api_connect(anw.get(),
9067dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis                NATIVE_WINDOW_API_MEDIA);
9077dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9087dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        if (err != OK) {
90929357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("setVideoSurfaceTexture failed: %d", err);
9107dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            // Note that we must do the reset before disconnecting from the ANW.
9117dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            // Otherwise queue/dequeue calls could be made on the disconnected
9127dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            // ANW, which may result in errors.
9137dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            reset();
9147dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9157dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            disconnectNativeWindow();
9167dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9177dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis            return err;
9187dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        }
9197dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    }
9207dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9217dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    // Note that we must set the player's new SurfaceTexture before
9227dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    // disconnecting the old one.  Otherwise queue/dequeue calls could be made
9237dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    // on the disconnected ANW, which may result in errors.
9247dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    status_t err = p->setVideoSurfaceTexture(surfaceTexture);
9257dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9267dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    disconnectNativeWindow();
9277dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9287dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    mConnectedWindow = anw;
9297dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9307dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    if (err == OK) {
9317dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        mConnectedWindowBinder = binder;
9327dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    } else {
9337dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis        disconnectNativeWindow();
9347dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    }
9357dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis
9367dae00baa6e8957be15523c46bb948bd1dde64c3Jamie Gennis    return err;
9371173118eace0e9e347cb007f0da817cee87579edGlenn Kasten}
9381173118eace0e9e347cb007f0da817cee87579edGlenn Kasten
9391d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Cataniastatus_t MediaPlayerService::Client::invoke(const Parcel& request,
9401d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania                                            Parcel *reply)
9411d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania{
9421d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    sp<MediaPlayerBase> p = getPlayer();
9431d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    if (p == NULL) return UNKNOWN_ERROR;
9441d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania    return p->invoke(request, reply);
9451d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania}
9461d187f1a86855f5f0694d7ec30efc9833bf7c589Nicolas Catania
947a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania// This call doesn't need to access the native player.
948a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Cataniastatus_t MediaPlayerService::Client::setMetadataFilter(const Parcel& filter)
949a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania{
950a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    status_t status;
951a64c8c79af1a15911c55306d83a797fa50969f77niko    media::Metadata::Filter allow, drop;
952a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
9534829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (unmarshallFilter(filter, &allow, &status) &&
9544829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        unmarshallFilter(filter, &drop, &status)) {
9554829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        Mutex::Autolock lock(mLock);
956a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
957a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        mMetadataAllow = allow;
958a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        mMetadataDrop = drop;
959a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
960a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    return status;
961a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}
962a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
9634829038419910aa6e75ce8992d45a223452d5c67Nicolas Cataniastatus_t MediaPlayerService::Client::getMetadata(
9644829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        bool update_only, bool apply_filter, Parcel *reply)
9658e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania{
966a64c8c79af1a15911c55306d83a797fa50969f77niko    sp<MediaPlayerBase> player = getPlayer();
967a64c8c79af1a15911c55306d83a797fa50969f77niko    if (player == 0) return UNKNOWN_ERROR;
968d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko
9698e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania    status_t status;
970d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    // Placeholder for the return code, updated by the caller.
971d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    reply->writeInt32(-1);
9724829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
973a64c8c79af1a15911c55306d83a797fa50969f77niko    media::Metadata::Filter ids;
9744829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
9754829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    // We don't block notifications while we fetch the data. We clear
9764829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    // mMetadataUpdated first so we don't lose notifications happening
9774829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    // during the rest of this call.
9784829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    {
9794829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        Mutex::Autolock lock(mLock);
9804829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        if (update_only) {
981d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko            ids = mMetadataUpdated;
9824829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        }
9834829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        mMetadataUpdated.clear();
9844829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    }
9858e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania
986a64c8c79af1a15911c55306d83a797fa50969f77niko    media::Metadata metadata(reply);
9874829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
988a64c8c79af1a15911c55306d83a797fa50969f77niko    metadata.appendHeader();
989a64c8c79af1a15911c55306d83a797fa50969f77niko    status = player->getMetadata(ids, reply);
990d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko
991d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    if (status != OK) {
992a64c8c79af1a15911c55306d83a797fa50969f77niko        metadata.resetParcel();
99329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("getMetadata failed %d", status);
994d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko        return status;
995d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    }
996d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko
997d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    // FIXME: Implement filtering on the result. Not critical since
998d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    // filtering takes place on the update notifications already. This
999d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    // would be when all the metadata are fetch and a filter is set.
1000d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko
1001d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    // Everything is fine, update the metadata length.
1002a64c8c79af1a15911c55306d83a797fa50969f77niko    metadata.updateLength();
1003d608a813a9d2cbc6e2a5ea81d78d4a9044090c4cniko    return OK;
10048e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania}
10058e1b6cce24574b9ecd5b0300155776bd0b4ef756Nicolas Catania
100689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::prepareAsync()
100789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10083856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] prepareAsync", mConnId);
100989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
101089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
101189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t ret = p->prepareAsync();
101289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#if CALLBACK_ANTAGONIZER
1013b8a805261bf0282e992d3608035e47d05a898710Steve Block    ALOGD("start Antagonizer");
101489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ret == NO_ERROR) mAntagonizer->start();
101589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#endif
101689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return ret;
101789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
101889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
101989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::start()
102089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10213856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] start", mConnId);
102289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
102389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
102489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    p->setLooping(mLoop);
102589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p->start();
102689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
102789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
102889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::stop()
102989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10303856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] stop", mConnId);
103189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
103289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
103389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p->stop();
103489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
103589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
103689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::pause()
103789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10383856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] pause", mConnId);
103989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
104089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
104189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p->pause();
104289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
104389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
104489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::isPlaying(bool* state)
104589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
104689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *state = false;
104789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
104889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
104989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *state = p->isPlaying();
10503856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] isPlaying: %d", mConnId, *state);
105189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
105289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
105389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
105489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::getCurrentPosition(int *msec)
105589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10563856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("getCurrentPosition");
105789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
105889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
105989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t ret = p->getCurrentPosition(msec);
106089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ret == NO_ERROR) {
10613856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("[%d] getCurrentPosition = %d", mConnId, *msec);
106289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
106329357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("getCurrentPosition returned %d", ret);
106489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
106589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return ret;
106689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
106789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
106889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::getDuration(int *msec)
106989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10703856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("getDuration");
107189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
107289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
107389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    status_t ret = p->getDuration(msec);
107489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (ret == NO_ERROR) {
10753856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("[%d] getDuration = %d", mConnId, *msec);
107689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
107729357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("getDuration returned %d", ret);
107889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
107989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return ret;
108089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
108189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
10826b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissenstatus_t MediaPlayerService::Client::setNextPlayer(const sp<IMediaPlayer>& player) {
10836b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    ALOGV("setNextPlayer");
10846b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    Mutex::Autolock l(mLock);
10856b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    sp<Client> c = static_cast<Client*>(player.get());
10866b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    mNextClient = c;
10876b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (mAudioOutput != NULL && c != NULL) {
10886b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mAudioOutput->setNextOutput(c->mAudioOutput);
10896b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    } else {
10906b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        ALOGE("no current audio output");
10916b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
10926b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    return OK;
10936b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen}
10946b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
10956b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
109689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::seekTo(int msec)
109789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
10983856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] seekTo(%d)", mConnId, msec);
109989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
110089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
110189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p->seekTo(msec);
110289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
110389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
110489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::reset()
110589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
11063856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] reset", mConnId);
1107c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    mRetransmitEndpointValid = false;
110889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
110989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == 0) return UNKNOWN_ERROR;
111089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return p->reset();
111189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
111289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1113fff6d715a8db0daf08a50634f242c40268de3d49Glenn Kastenstatus_t MediaPlayerService::Client::setAudioStreamType(audio_stream_type_t type)
111489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
11153856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setAudioStreamType(%d)", mConnId, type);
111689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // TODO: for hardware output, call player instead
111789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock l(mLock);
111889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mAudioOutput != 0) mAudioOutput->setAudioStreamType(type);
111989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
112089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
112189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
112289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::setLooping(int loop)
112389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
11243856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setLooping(%d)", mConnId, loop);
112589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLoop = loop;
112689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> p = getPlayer();
112789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p != 0) return p->setLooping(loop);
112889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
112989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
113089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
113189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::Client::setVolume(float leftVolume, float rightVolume)
113289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
11333856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setVolume(%f, %f)", mConnId, leftVolume, rightVolume);
1134761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman
1135761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    // for hardware output, call player instead
1136761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    sp<MediaPlayerBase> p = getPlayer();
1137761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    {
1138761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman      Mutex::Autolock l(mLock);
1139761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman      if (p != 0 && p->hardwareOutput()) {
1140761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman          MediaPlayerHWInterface* hwp =
1141761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman                  reinterpret_cast<MediaPlayerHWInterface*>(p.get());
1142761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman          return hwp->setVolume(leftVolume, rightVolume);
1143761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman      } else {
1144761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman          if (mAudioOutput != 0) mAudioOutput->setVolume(leftVolume, rightVolume);
1145761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman          return NO_ERROR;
1146761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman      }
1147761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman    }
1148761defc341c5ce9019a42919c441f035f665ec0dJohn Grossman
114989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
115089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
115189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
11522beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayerService::Client::setAuxEffectSendLevel(float level)
11532beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
11543856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setAuxEffectSendLevel(%f)", mConnId, level);
11552beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    Mutex::Autolock l(mLock);
11562beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mAudioOutput != 0) return mAudioOutput->setAuxEffectSendLevel(level);
11572beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return NO_ERROR;
11582beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
11592beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
11602beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayerService::Client::attachAuxEffect(int effectId)
11612beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
11623856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] attachAuxEffect(%d)", mConnId, effectId);
11632beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    Mutex::Autolock l(mLock);
11642beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mAudioOutput != 0) return mAudioOutput->attachAuxEffect(effectId);
11652beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return NO_ERROR;
11662beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
11674829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
11684f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wangstatus_t MediaPlayerService::Client::setParameter(int key, const Parcel &request) {
11693856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] setParameter(%d)", mConnId, key);
11704f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    sp<MediaPlayerBase> p = getPlayer();
11714f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    if (p == 0) return UNKNOWN_ERROR;
11724f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    return p->setParameter(key, request);
11734f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang}
11744f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang
11754f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wangstatus_t MediaPlayerService::Client::getParameter(int key, Parcel *reply) {
11763856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] getParameter(%d)", mConnId, key);
11774f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    sp<MediaPlayerBase> p = getPlayer();
11784f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    if (p == 0) return UNKNOWN_ERROR;
11794f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang    return p->getParameter(key, reply);
11804f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang}
11814f9e47f2c03ce36261c4717cd7e131d7940bb068Gloria Wang
1182c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossmanstatus_t MediaPlayerService::Client::setRetransmitEndpoint(
1183c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        const struct sockaddr_in* endpoint) {
1184c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1185c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (NULL != endpoint) {
1186c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        uint32_t a = ntohl(endpoint->sin_addr.s_addr);
1187c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        uint16_t p = ntohs(endpoint->sin_port);
1188c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        ALOGV("[%d] setRetransmitEndpoint(%u.%u.%u.%u:%hu)", mConnId,
1189c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman                (a >> 24), (a >> 16) & 0xFF, (a >> 8) & 0xFF, (a & 0xFF), p);
1190c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    } else {
1191c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        ALOGV("[%d] setRetransmitEndpoint = <none>", mConnId);
1192c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
1193c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1194c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    sp<MediaPlayerBase> p = getPlayer();
1195c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1196c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // Right now, the only valid time to set a retransmit endpoint is before
1197c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // player selection has been made (since the presence or absence of a
1198c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // retransmit endpoint is going to determine which player is selected during
1199c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    // setDataSource).
1200c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (p != 0) return INVALID_OPERATION;
1201c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1202c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    if (NULL != endpoint) {
1203c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mRetransmitEndpoint = *endpoint;
1204c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mRetransmitEndpointValid = true;
1205c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    } else {
1206c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman        mRetransmitEndpointValid = false;
1207c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    }
1208c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1209c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman    return NO_ERROR;
1210c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman}
1211c795b64060c3af9d7961fc1371e4ccfa8ee3e450John Grossman
1212b483c4724846c0b8d4e82afcbb7c17f671bae81cGloria Wangvoid MediaPlayerService::Client::notify(
1213b483c4724846c0b8d4e82afcbb7c17f671bae81cGloria Wang        void* cookie, int msg, int ext1, int ext2, const Parcel *obj)
121489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
121589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Client* client = static_cast<Client*>(cookie);
1216a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
12176b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    {
12186b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        Mutex::Autolock l(client->mLock);
12196b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        if (msg == MEDIA_PLAYBACK_COMPLETE && client->mNextClient != NULL) {
12206b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            client->mAudioOutput->switchToNextOutput();
12216b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            client->mNextClient->start();
12226b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            client->mNextClient->mClient->notify(MEDIA_INFO, MEDIA_INFO_STARTED_AS_NEXT, 0, obj);
12236b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        }
12246b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
12256b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
1226a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    if (MEDIA_INFO == msg &&
12274829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        MEDIA_INFO_METADATA_UPDATE == ext1) {
1228a64c8c79af1a15911c55306d83a797fa50969f77niko        const media::Metadata::Type metadata_type = ext2;
12294829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
12304829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        if(client->shouldDropMetadata(metadata_type)) {
12314829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania            return;
12324829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        }
12334829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
12344829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        // Update the list of metadata that have changed. getMetadata
12354829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        // also access mMetadataUpdated and clears it.
12364829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        client->addNewMetadataUpdate(metadata_type);
1237a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
12383856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("[%d] notify (%p, %d, %d, %d)", client->mConnId, cookie, msg, ext1, ext2);
1239b483c4724846c0b8d4e82afcbb7c17f671bae81cGloria Wang    client->mClient->notify(msg, ext1, ext2, obj);
124089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
124189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
12424829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
1243a64c8c79af1a15911c55306d83a797fa50969f77nikobool MediaPlayerService::Client::shouldDropMetadata(media::Metadata::Type code) const
1244a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania{
12454829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    Mutex::Autolock lock(mLock);
1246a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
12474829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (findMetadata(mMetadataDrop, code)) {
1248a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return true;
1249a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
1250a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
12514829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (mMetadataAllow.isEmpty() || findMetadata(mMetadataAllow, code)) {
1252a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return false;
12534829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    } else {
1254a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania        return true;
1255a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania    }
1256a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania}
1257a7e0e8b4c429fc68eb1bd5b5a30f5b91352288f9Nicolas Catania
12584829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
1259a64c8c79af1a15911c55306d83a797fa50969f77nikovoid MediaPlayerService::Client::addNewMetadataUpdate(media::Metadata::Type metadata_type) {
12604829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    Mutex::Autolock lock(mLock);
12614829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    if (mMetadataUpdated.indexOf(metadata_type) < 0) {
12624829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania        mMetadataUpdated.add(metadata_type);
12634829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania    }
12644829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania}
12654829038419910aa6e75ce8992d45a223452d5c67Nicolas Catania
126689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#if CALLBACK_ANTAGONIZER
126789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectconst int Antagonizer::interval = 10000; // 10 msecs
126889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
126989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectAntagonizer::Antagonizer(notify_callback_f cb, void* client) :
127089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mExit(false), mActive(false), mClient(client), mCb(cb)
127189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
127289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    createThread(callbackThread, this);
127389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
127489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
127589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid Antagonizer::kill()
127689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
127789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(mLock);
127889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mActive = false;
127989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mExit = true;
128089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCondition.wait(mLock);
128189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
128289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
128389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectint Antagonizer::callbackThread(void* user)
128489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
1285b8a805261bf0282e992d3608035e47d05a898710Steve Block    ALOGD("Antagonizer started");
128689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Antagonizer* p = reinterpret_cast<Antagonizer*>(user);
128789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    while (!p->mExit) {
128889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        if (p->mActive) {
12893856b090cd04ba5dd4a59a12430ed724d5995909Steve Block            ALOGV("send event");
129089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project            p->mCb(p->mClient, 0, 0, 0);
129189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        }
129289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        usleep(interval);
129389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
129489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock _l(p->mLock);
129589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    p->mCondition.signal();
1296b8a805261bf0282e992d3608035e47d05a898710Steve Block    ALOGD("Antagonizer stopped");
129789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return 0;
129889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
129989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#endif
130089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
130189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatic size_t kDefaultHeapSize = 1024 * 1024; // 1MB
130289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1303e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kastensp<IMemory> MediaPlayerService::decode(const char* url, uint32_t *pSampleRate, int* pNumChannels, audio_format_t* pFormat)
130489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
13053856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("decode(%s)", url);
130689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MemoryBase> mem;
130789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> player;
130889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
130989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Protect our precious, precious DRMd ringtones by only allowing
131089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // decoding of http, but not filesystem paths or content Uris.
131189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // If the application wants to decode those, it should open a
131289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // filedescriptor for them and use that.
131389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (url != NULL && strncmp(url, "http://", 7) != 0) {
1314b8a805261bf0282e992d3608035e47d05a898710Steve Block        ALOGD("Can't decode %s by path, use filedescriptor instead", url);
131589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return mem;
131689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
131789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
131889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player_type playerType = getPlayerType(url);
13193856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("player type = %d", playerType);
132089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
132189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // create the right type of player
132289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<AudioCache> cache = new AudioCache(url);
132389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player = android::createPlayer(playerType, cache.get(), cache->notify);
132489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player == NULL) goto Exit;
132589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player->hardwareOutput()) goto Exit;
132689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
132789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    static_cast<MediaPlayerInterface*>(player.get())->setAudioSink(cache);
132889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
132989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // set data source
133089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player->setDataSource(url) != NO_ERROR) goto Exit;
133189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13323856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("prepare");
133389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player->prepareAsync();
133489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13353856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("wait for prepare");
133689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (cache->wait() != NO_ERROR) goto Exit;
133789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13383856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("start");
133989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player->start();
134089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13413856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("wait for playback complete");
13429cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    cache->wait();
13439cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    // in case of error, return what was successfully decoded.
13449cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    if (cache->size() == 0) {
13459cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent        goto Exit;
13469cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    }
134789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
134889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mem = new MemoryBase(cache->getHeap(), 0, cache->size());
134989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *pSampleRate = cache->sampleRate();
135089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *pNumChannels = cache->channelCount();
1351e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten    *pFormat = cache->format();
13523856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("return memory @ %p, sampleRate=%u, channelCount = %d, format = %d", mem->pointer(), *pSampleRate, *pNumChannels, *pFormat);
135389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
135489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectExit:
135589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player != 0) player->reset();
135689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mem;
135789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
135889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1359e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kastensp<IMemory> MediaPlayerService::decode(int fd, int64_t offset, int64_t length, uint32_t *pSampleRate, int* pNumChannels, audio_format_t* pFormat)
136089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
13613856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("decode(%d, %lld, %lld)", fd, offset, length);
136289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MemoryBase> mem;
136389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<MediaPlayerBase> player;
136489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
136589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player_type playerType = getPlayerType(fd, offset, length);
13663856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("player type = %d", playerType);
136789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
136889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // create the right type of player
136989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    sp<AudioCache> cache = new AudioCache("decode_fd");
137089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player = android::createPlayer(playerType, cache.get(), cache->notify);
137189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player == NULL) goto Exit;
137289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player->hardwareOutput()) goto Exit;
137389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
137489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    static_cast<MediaPlayerInterface*>(player.get())->setAudioSink(cache);
137589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
137689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // set data source
137789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player->setDataSource(fd, offset, length) != NO_ERROR) goto Exit;
137889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13793856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("prepare");
138089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player->prepareAsync();
138189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13823856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("wait for prepare");
138389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (cache->wait() != NO_ERROR) goto Exit;
138489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13853856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("start");
138689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    player->start();
138789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
13883856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("wait for playback complete");
13899cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    cache->wait();
13909cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    // in case of error, return what was successfully decoded.
13919cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    if (cache->size() == 0) {
13929cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent        goto Exit;
13939cb839a0fcc98fe4278b39afb8b6d664c04f1673Eric Laurent    }
139489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
139589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mem = new MemoryBase(cache->getHeap(), 0, cache->size());
139689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *pSampleRate = cache->sampleRate();
139789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *pNumChannels = cache->channelCount();
139889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    *pFormat = cache->format();
13993856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("return memory @ %p, sampleRate=%u, channelCount = %d, format = %d", mem->pointer(), *pSampleRate, *pNumChannels, *pFormat);
140089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
140189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectExit:
140289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (player != 0) player->reset();
140389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    ::close(fd);
140489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mem;
140589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
140689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
140710dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen
140889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#undef LOG_TAG
140989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#define LOG_TAG "AudioSink"
1410a514bdb58b5de4986679f72b7204b4764f7a2778Eric LaurentMediaPlayerService::AudioOutput::AudioOutput(int sessionId)
141120111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    : mCallback(NULL),
1412a514bdb58b5de4986679f72b7204b4764f7a2778Eric Laurent      mCallbackCookie(NULL),
14136b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen      mCallbackData(NULL),
14144110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen      mBytesWritten(0),
14151948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent      mSessionId(sessionId),
14161948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent      mFlags(AUDIO_OUTPUT_FLAG_NONE) {
14173856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("AudioOutput(%d)", sessionId);
141889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mTrack = 0;
14196b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    mRecycledTrack = 0;
1420fce7a473248381cc83a01855f92581077d3c9ee2Dima Zavin    mStreamType = AUDIO_STREAM_MUSIC;
142189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = 1.0;
142289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mRightVolume = 1.0;
14237a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    mPlaybackRatePermille = 1000;
14247a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    mSampleRateHz = 0;
142589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mMsecsPerFrame = 0;
14262beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    mAuxEffectId = 0;
14272beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    mSendLevel = 0.0;
142889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    setMinBufferCount();
142989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
143089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
143189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayerService::AudioOutput::~AudioOutput()
143289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
143389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    close();
14346b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    delete mRecycledTrack;
14356b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    delete mCallbackData;
143689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
143789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
143889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::setMinBufferCount()
143989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
144089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    char value[PROPERTY_VALUE_MAX];
144189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (property_get("ro.kernel.qemu", value, 0)) {
144289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mIsOnEmulator = true;
144389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mMinBufferCount = 12;  // to prevent systematic buffer underrun for emulator
144489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
144589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
144689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
144789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectbool MediaPlayerService::AudioOutput::isOnEmulator()
144889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
144989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    setMinBufferCount();
145089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mIsOnEmulator;
145189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
145289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
145389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectint MediaPlayerService::AudioOutput::getMinBufferCount()
145489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
145589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    setMinBufferCount();
145689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mMinBufferCount;
145789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
145889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
145989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioOutput::bufferSize() const
146089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
146189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack == 0) return NO_INIT;
146289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mTrack->frameCount() * frameSize();
146389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
146489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
146589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioOutput::frameCount() const
146689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
146789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack == 0) return NO_INIT;
146889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mTrack->frameCount();
146989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
147089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
147189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioOutput::channelCount() const
147289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
147389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack == 0) return NO_INIT;
147489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mTrack->channelCount();
147589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
147689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
147789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioOutput::frameSize() const
147889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
147989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack == 0) return NO_INIT;
148089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mTrack->frameSize();
148189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
148289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
148389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t MediaPlayerService::AudioOutput::latency () const
148489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
1485db354e58e65592777aa17caa47933e14838b8b35Eric Laurent    if (mTrack == 0) return 0;
1486db354e58e65592777aa17caa47933e14838b8b35Eric Laurent    return mTrack->latency();
148789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
148889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
148989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectfloat MediaPlayerService::AudioOutput::msecsPerFrame() const
149089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
149189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mMsecsPerFrame;
149289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
149389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
14944110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenstatus_t MediaPlayerService::AudioOutput::getPosition(uint32_t *position) const
1495342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent{
1496342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent    if (mTrack == 0) return NO_INIT;
1497342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent    return mTrack->getPosition(position);
1498342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent}
1499342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent
15004110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenstatus_t MediaPlayerService::AudioOutput::getFramesWritten(uint32_t *frameswritten) const
15014110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen{
15024110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    if (mTrack == 0) return NO_INIT;
15034110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    *frameswritten = mBytesWritten / frameSize();
15044110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    return OK;
15054110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen}
15064110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen
150720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberstatus_t MediaPlayerService::AudioOutput::open(
1508786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        uint32_t sampleRate, int channelCount, audio_channel_mask_t channelMask,
1509786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        audio_format_t format, int bufferCount,
15101948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent        AudioCallback cb, void *cookie,
15111948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent        audio_output_flags_t flags)
151289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
151320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    mCallback = cb;
151420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    mCallbackCookie = cookie;
151520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
151689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // Check argument "bufferCount" against the mininum buffer count
151789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (bufferCount < mMinBufferCount) {
1518b8a805261bf0282e992d3608035e47d05a898710Steve Block        ALOGD("bufferCount (%d) is too small and increased to %d", bufferCount, mMinBufferCount);
151989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        bufferCount = mMinBufferCount;
152089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
152189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
1522786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi    ALOGV("open(%u, %d, 0x%x, %d, %d, %d)", sampleRate, channelCount, channelMask,
1523786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi            format, bufferCount, mSessionId);
152489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    int afSampleRate;
152589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    int afFrameCount;
15261948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent    uint32_t frameCount;
152789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
152889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (AudioSystem::getOutputFrameCount(&afFrameCount, mStreamType) != NO_ERROR) {
152989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_INIT;
153089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
153189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (AudioSystem::getOutputSamplingRate(&afSampleRate, mStreamType) != NO_ERROR) {
153289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        return NO_INIT;
153389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
153489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
153589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    frameCount = (sampleRate*afFrameCount*bufferCount)/afSampleRate;
153620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
1537786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi    if (channelMask == CHANNEL_MASK_USE_CHANNEL_ORDER) {
1538ab334fd351ae5a0e18903da123d63e565b536874Glenn Kasten        channelMask = audio_channel_out_mask_from_count(channelCount);
1539786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        if (0 == channelMask) {
1540786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi            ALOGE("open() error, can\'t derive mask for %d audio channels", channelCount);
1541786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi            return NO_INIT;
1542786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        }
1543786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi    }
15441948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent
154567295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    AudioTrack *t;
154667295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    CallbackData *newcbd = NULL;
154767295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    if (mCallback != NULL) {
154867295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        newcbd = new CallbackData(this);
154967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        t = new AudioTrack(
155067295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                mStreamType,
155167295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                sampleRate,
155267295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                format,
155367295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                channelMask,
155467295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                frameCount,
155567295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                flags,
155667295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                CallbackWrapper,
155767295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                newcbd,
155867295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                0,  // notification frames
155967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                mSessionId);
156067295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    } else {
156167295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        t = new AudioTrack(
156267295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                mStreamType,
156367295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                sampleRate,
156467295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                format,
156567295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                channelMask,
156667295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                frameCount,
156767295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                flags,
156867295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                NULL,
156967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                NULL,
157067295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                0,
157167295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                mSessionId);
157267295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    }
157367295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen
157467295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    if ((t == 0) || (t->initCheck() != NO_ERROR)) {
157567295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        ALOGE("Unable to create audio track");
157667295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        delete t;
157767295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        delete newcbd;
157867295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        return NO_INIT;
157967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    }
158067295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen
158167295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen
15826b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (mRecycledTrack) {
15836b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        // check if the existing track can be reused as-is, or if a new track needs to be created.
15846b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
15856b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        bool reuse = true;
15866b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        if ((mCallbackData == NULL && mCallback != NULL) ||
15876b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen                (mCallbackData != NULL && mCallback == NULL)) {
15886b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            // recycled track uses callbacks but the caller wants to use writes, or vice versa
15896b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            ALOGV("can't chain callback and write");
15906b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            reuse = false;
15916b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        } else if ((mRecycledTrack->getSampleRate() != sampleRate) ||
15926b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen                (mRecycledTrack->channelCount() != channelCount) ||
159367295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                (mRecycledTrack->frameCount() != t->frameCount())) {
159467295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen            ALOGV("samplerate, channelcount or framecount differ: %d/%d Hz, %d/%d ch, %d/%d frames",
159567295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                  mRecycledTrack->getSampleRate(), sampleRate,
159667295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                  mRecycledTrack->channelCount(), channelCount,
159767295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen                  mRecycledTrack->frameCount(), t->frameCount());
15986b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            reuse = false;
159967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen        } else if (flags != mFlags) {
160067295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen            ALOGV("output flags differ %08x/%08x", flags, mFlags);
16011948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent            reuse = false;
16026b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        }
16036b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        if (reuse) {
16046b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            ALOGV("chaining to next output");
16056b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            close();
16066b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            mTrack = mRecycledTrack;
16076b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            mRecycledTrack = NULL;
16086b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            if (mCallbackData != NULL) {
16096b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen                mCallbackData->setOutput(this);
16106b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            }
161167295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen            delete t;
161267295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen            delete newcbd;
16136b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            return OK;
16146b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        }
16156b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
16166b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        // if we're not going to reuse the track, unblock and flush it
16176b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        if (mCallbackData != NULL) {
16186b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            mCallbackData->setOutput(NULL);
16196b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            mCallbackData->endTrackSwitch();
16206b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        }
16216b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mRecycledTrack->flush();
16226b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        delete mRecycledTrack;
16236b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mRecycledTrack = NULL;
16246b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        delete mCallbackData;
16256b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mCallbackData = NULL;
16266b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        close();
16276b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
1628786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi
162967295b543c16ed7c77b67f859d598fe7b4571660Marco Nelissen    mCallbackData = newcbd;
16303856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("setVolume");
163189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    t->setVolume(mLeftVolume, mRightVolume);
16322beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
16337a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    mSampleRateHz = sampleRate;
16341948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent    mFlags = flags;
16357a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    mMsecsPerFrame = mPlaybackRatePermille / (float) sampleRate;
163699448608f375a7191c068c3bfd296f204b58a219Marco Nelissen    uint32_t pos;
163799448608f375a7191c068c3bfd296f204b58a219Marco Nelissen    if (t->getPosition(&pos) == OK) {
163899448608f375a7191c068c3bfd296f204b58a219Marco Nelissen        mBytesWritten = uint64_t(pos) * t->frameSize();
163999448608f375a7191c068c3bfd296f204b58a219Marco Nelissen    }
164089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mTrack = t;
16412beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
16427a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    status_t res = t->setSampleRate(mPlaybackRatePermille * mSampleRateHz / 1000);
16437a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    if (res != NO_ERROR) {
16447a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi        return res;
16457a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    }
16462beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    t->setAuxEffectSendLevel(mSendLevel);
16472beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return t->attachAuxEffect(mAuxEffectId);;
164889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
164989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
165089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::start()
165189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
16523856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("start");
16536b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (mCallbackData != NULL) {
16546b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mCallbackData->endTrackSwitch();
16556b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
165689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack) {
165789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mTrack->setVolume(mLeftVolume, mRightVolume);
16582beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        mTrack->setAuxEffectSendLevel(mSendLevel);
165989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mTrack->start();
166089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
166189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
166289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
16636b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissenvoid MediaPlayerService::AudioOutput::setNextOutput(const sp<AudioOutput>& nextOutput) {
16646b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    mNextOutput = nextOutput;
16656b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen}
16666b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen
16677ee8ac94bb1a724a481a7cddf10ce63d35df6296Marco Nelissen
16686b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissenvoid MediaPlayerService::AudioOutput::switchToNextOutput() {
16696b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    ALOGV("switchToNextOutput");
16706b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (mNextOutput != NULL) {
16716b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        if (mCallbackData != NULL) {
16726b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen            mCallbackData->beginTrackSwitch();
16736b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        }
16746b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        delete mNextOutput->mCallbackData;
16756b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mNextOutput->mCallbackData = mCallbackData;
16766b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mCallbackData = NULL;
16776b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mNextOutput->mRecycledTrack = mTrack;
16786b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mTrack = NULL;
16796b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mNextOutput->mSampleRateHz = mSampleRateHz;
16806b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        mNextOutput->mMsecsPerFrame = mMsecsPerFrame;
16814110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen        mNextOutput->mBytesWritten = mBytesWritten;
16826b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
16836b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen}
16847ee8ac94bb1a724a481a7cddf10ce63d35df6296Marco Nelissen
168589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioOutput::write(const void* buffer, size_t size)
168689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
168720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    LOG_FATAL_IF(mCallback != NULL, "Don't call write if supplying a callback.");
168820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
16893856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    //ALOGV("write(%p, %u)", buffer, size);
169010dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen    if (mTrack) {
169110dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen        ssize_t ret = mTrack->write(buffer, size);
16924110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen        mBytesWritten += ret;
169310dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen        return ret;
169410dbb8e97e7a81ca4867663b5517f048820b3094Marco Nelissen    }
169589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_INIT;
169689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
169789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
169889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::stop()
169989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17003856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("stop");
170189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack) mTrack->stop();
170289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
170389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
170489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::flush()
170589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17063856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("flush");
170789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack) mTrack->flush();
170889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
170989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
171089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::pause()
171189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17123856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("pause");
171389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack) mTrack->pause();
171489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
171589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
171689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::close()
171789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17183856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("close");
171989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    delete mTrack;
172089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mTrack = 0;
172189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
172289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
172389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectvoid MediaPlayerService::AudioOutput::setVolume(float left, float right)
172489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
17253856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("setVolume(%f, %f)", left, right);
172689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mLeftVolume = left;
172789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mRightVolume = right;
172889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mTrack) {
172989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mTrack->setVolume(left, right);
173089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
173189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
173289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
17337a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivistatus_t MediaPlayerService::AudioOutput::setPlaybackRatePermille(int32_t ratePermille)
17347a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi{
17357a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    ALOGV("setPlaybackRatePermille(%d)", ratePermille);
17367a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    status_t res = NO_ERROR;
17377a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    if (mTrack) {
17387a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi        res = mTrack->setSampleRate(ratePermille * mSampleRateHz / 1000);
17397a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    } else {
17407a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi        res = NO_INIT;
17417a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    }
17427a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    mPlaybackRatePermille = ratePermille;
17437a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    if (mSampleRateHz != 0) {
17447a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi        mMsecsPerFrame = mPlaybackRatePermille / (float) mSampleRateHz;
17457a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    }
17467a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi    return res;
17477a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi}
17487a8b0ed6419e57dd8b41d3806893d63d3df91aabJean-Michel Trivi
17492beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayerService::AudioOutput::setAuxEffectSendLevel(float level)
17502beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
17513856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("setAuxEffectSendLevel(%f)", level);
17522beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    mSendLevel = level;
17532beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mTrack) {
17542beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        return mTrack->setAuxEffectSendLevel(level);
17552beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    }
17562beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return NO_ERROR;
17572beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
17582beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
17592beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurentstatus_t MediaPlayerService::AudioOutput::attachAuxEffect(int effectId)
17602beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent{
17613856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("attachAuxEffect(%d)", effectId);
17622beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    mAuxEffectId = effectId;
17632beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    if (mTrack) {
17642beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent        return mTrack->attachAuxEffect(effectId);
17652beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    }
17662beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent    return NO_ERROR;
17672beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent}
17682beeb50b1bba9e92f6cacfeca37fe9fa9d36ead1Eric Laurent
176920111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber// static
177020111aa043c5f404472bc63b90bc5aad906b1101Andreas Hubervoid MediaPlayerService::AudioOutput::CallbackWrapper(
1771d217a8c4632b3e3065f8c2a26b9ce4dc4c97171fGlenn Kasten        int event, void *cookie, void *info) {
17723856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    //ALOGV("callbackwrapper");
177320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    if (event != AudioTrack::EVENT_MORE_DATA) {
177420111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber        return;
177520111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    }
177620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
17776b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    CallbackData *data = (CallbackData*)cookie;
17786b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    data->lock();
17796b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    AudioOutput *me = data->getOutput();
178020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber    AudioTrack::Buffer *buffer = (AudioTrack::Buffer *)info;
17816b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (me == NULL) {
17826b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        // no output set, likely because the track was scheduled to be reused
17836b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        // by another player, but the format turned out to be incompatible.
17846b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        data->unlock();
17856b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        buffer->size = 0;
17866b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen        return;
17876b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    }
178820111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
17897d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    size_t actualSize = (*me->mCallback)(
179020111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber            me, buffer->raw, buffer->size, me->mCallbackCookie);
17917d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
17926b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    if (actualSize == 0 && buffer->size > 0 && me->mNextOutput == NULL) {
179351c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber        // We've reached EOS but the audio track is not stopped yet,
179451c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber        // keep playing silence.
179551c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber
179651c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber        memset(buffer->raw, 0, buffer->size);
179751c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber        actualSize = buffer->size;
179851c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber    }
17992e8ffaf95bd35358ecd14ddf00cddaf3abefe0a9Andreas Huber
180051c1e0e86a0ad95bf3d890a9a2f51e54b8ef9444Andreas Huber    buffer->size = actualSize;
18016b74d671a1321a6ecc4a40b6c87beedfecc1ec44Marco Nelissen    data->unlock();
180220111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber}
180320111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
18044110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenint MediaPlayerService::AudioOutput::getSessionId() const
18058c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent{
18068c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent    return mSessionId;
18078c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent}
18088c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent
180989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#undef LOG_TAG
181089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project#define LOG_TAG "AudioCache"
181189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source ProjectMediaPlayerService::AudioCache::AudioCache(const char* name) :
181289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mChannelCount(0), mFrameCount(1024), mSampleRate(0), mSize(0),
181389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mError(NO_ERROR), mCommandComplete(false)
181489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
181589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // create ashmem heap
181689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mHeap = new MemoryHeapBase(kDefaultHeapSize, 0, name);
181789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
181889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
181989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectuint32_t MediaPlayerService::AudioCache::latency () const
182089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
182189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return 0;
182289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
182389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
182489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectfloat MediaPlayerService::AudioCache::msecsPerFrame() const
182589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
182689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mMsecsPerFrame;
182789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
182889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
18294110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenstatus_t MediaPlayerService::AudioCache::getPosition(uint32_t *position) const
1830342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent{
1831342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent    if (position == 0) return BAD_VALUE;
1832342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent    *position = mSize;
1833342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent    return NO_ERROR;
1834342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent}
1835342e9cf388cceb807def720e40e8b0a217f4bcaaEric Laurent
18364110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenstatus_t MediaPlayerService::AudioCache::getFramesWritten(uint32_t *written) const
18374110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen{
18384110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    if (written == 0) return BAD_VALUE;
18394110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    *written = mSize;
18404110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen    return NO_ERROR;
18414110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen}
18424110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissen
18437d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber////////////////////////////////////////////////////////////////////////////////
18447d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18457d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huberstruct CallbackThread : public Thread {
18467d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    CallbackThread(const wp<MediaPlayerBase::AudioSink> &sink,
18477d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber                   MediaPlayerBase::AudioSink::AudioCallback cb,
18487d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber                   void *cookie);
18497d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18507d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huberprotected:
18517d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    virtual ~CallbackThread();
18527d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18537d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    virtual bool threadLoop();
18547d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18557d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huberprivate:
18567d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    wp<MediaPlayerBase::AudioSink> mSink;
18577d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    MediaPlayerBase::AudioSink::AudioCallback mCallback;
18587d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    void *mCookie;
18597d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    void *mBuffer;
18607d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    size_t mBufferSize;
18617d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18627d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    CallbackThread(const CallbackThread &);
18637d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    CallbackThread &operator=(const CallbackThread &);
18647d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber};
18657d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18667d5b8a70c28c0d5746a600467b2887822dbff88eAndreas HuberCallbackThread::CallbackThread(
18677d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        const wp<MediaPlayerBase::AudioSink> &sink,
18687d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        MediaPlayerBase::AudioSink::AudioCallback cb,
18697d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        void *cookie)
18707d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    : mSink(sink),
18717d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber      mCallback(cb),
18727d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber      mCookie(cookie),
18737d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber      mBuffer(NULL),
18747d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber      mBufferSize(0) {
18757d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber}
18767d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18777d5b8a70c28c0d5746a600467b2887822dbff88eAndreas HuberCallbackThread::~CallbackThread() {
18787d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (mBuffer) {
18797d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        free(mBuffer);
18807d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mBuffer = NULL;
18817d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
18827d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber}
18837d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18847d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huberbool CallbackThread::threadLoop() {
18857d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    sp<MediaPlayerBase::AudioSink> sink = mSink.promote();
18867d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (sink == NULL) {
18877d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        return false;
18887d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
18897d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18907d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (mBuffer == NULL) {
18917d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mBufferSize = sink->bufferSize();
18927d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mBuffer = malloc(mBufferSize);
18937d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
18947d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18957d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    size_t actualSize =
18967d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        (*mCallback)(sink.get(), mBuffer, mBufferSize, mCookie);
18977d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
18987d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (actualSize > 0) {
18997d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        sink->write(mBuffer, actualSize);
19007d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
19017d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
19027d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    return true;
19037d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber}
19047d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
19057d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber////////////////////////////////////////////////////////////////////////////////
19067d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
190720111aa043c5f404472bc63b90bc5aad906b1101Andreas Huberstatus_t MediaPlayerService::AudioCache::open(
1908786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        uint32_t sampleRate, int channelCount, audio_channel_mask_t channelMask,
1909786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi        audio_format_t format, int bufferCount,
19101948eb3ea6eee336e8cdab9b0c693f93f5f19993Eric Laurent        AudioCallback cb, void *cookie, audio_output_flags_t flags)
191189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
1912786618ffe881aceb64d65a6a2e2d76ede6e01ec0Jean-Michel Trivi    ALOGV("open(%u, %d, 0x%x, %d, %d)", sampleRate, channelCount, channelMask, format, bufferCount);
19138eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    if (mHeap->getHeapID() < 0) {
19148eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks        return NO_INIT;
19158eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    }
191620111aa043c5f404472bc63b90bc5aad906b1101Andreas Huber
191789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSampleRate = sampleRate;
191889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mChannelCount = (uint16_t)channelCount;
1919e1c3962e268ffc12bfd1bd9ea84da1f135f36960Glenn Kasten    mFormat = format;
192089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mMsecsPerFrame = 1.e3 / (float) sampleRate;
19217d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
19227d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (cb != NULL) {
19237d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mCallbackThread = new CallbackThread(this, cb, cookie);
19247d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
192589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return NO_ERROR;
192689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
192789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
19287d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Hubervoid MediaPlayerService::AudioCache::start() {
19297d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (mCallbackThread != NULL) {
19307d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mCallbackThread->run("AudioCache callback");
19317d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
19327d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber}
19337d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
19347d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Hubervoid MediaPlayerService::AudioCache::stop() {
19357d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    if (mCallbackThread != NULL) {
19367d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber        mCallbackThread->requestExitAndWait();
19377d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber    }
19387d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber}
19397d5b8a70c28c0d5746a600467b2887822dbff88eAndreas Huber
194089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectssize_t MediaPlayerService::AudioCache::write(const void* buffer, size_t size)
194189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
19423856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("write(%p, %u)", buffer, size);
194389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if ((buffer == 0) || (size == 0)) return size;
194489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
194589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    uint8_t* p = static_cast<uint8_t*>(mHeap->getBase());
194689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (p == NULL) return NO_INIT;
194789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    p += mSize;
19483856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("memcpy(%p, %p, %u)", p, buffer, size);
194989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mSize + size > mHeap->getSize()) {
195029357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Heap size overflow! req size: %d, max size: %d", (mSize + size), mHeap->getSize());
195189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        size = mHeap->getSize() - mSize;
195289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
195389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    memcpy(p, buffer, size);
195489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mSize += size;
195589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return size;
195689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
195789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
195889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project// call with lock held
195989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Projectstatus_t MediaPlayerService::AudioCache::wait()
196089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
196189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    Mutex::Autolock lock(mLock);
19624bbc0ba371c52951191eff1cba7c1ea5d27ee976Dave Sparks    while (!mCommandComplete) {
196389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        mSignal.wait(mLock);
196489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
196589fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    mCommandComplete = false;
196689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
196789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    if (mError == NO_ERROR) {
19683856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("wait - success");
196989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    } else {
19703856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("wait - error");
197189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
197289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    return mError;
197389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
197489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
1975b483c4724846c0b8d4e82afcbb7c17f671bae81cGloria Wangvoid MediaPlayerService::AudioCache::notify(
1976b483c4724846c0b8d4e82afcbb7c17f671bae81cGloria Wang        void* cookie, int msg, int ext1, int ext2, const Parcel *obj)
197789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project{
19783856b090cd04ba5dd4a59a12430ed724d5995909Steve Block    ALOGV("notify(%p, %d, %d, %d)", cookie, msg, ext1, ext2);
197989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    AudioCache* p = static_cast<AudioCache*>(cookie);
198089fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
198189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // ignore buffering messages
19828eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    switch (msg)
19838eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    {
19848eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    case MEDIA_ERROR:
198529357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block        ALOGE("Error %d, %d occurred", ext1, ext2);
198689fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project        p->mError = ext1;
19878eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks        break;
19888eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    case MEDIA_PREPARED:
19893856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("prepared");
19908eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks        break;
19918eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    case MEDIA_PLAYBACK_COMPLETE:
19923856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("playback complete");
19938eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks        break;
19948eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks    default:
19953856b090cd04ba5dd4a59a12430ed724d5995909Steve Block        ALOGV("ignored");
19968eb8011120a68a86199b9960cf1ae91a6ddc7e6dDave Sparks        return;
199789fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    }
199889fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
199989fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    // wake up thread
2000fe4c6f01646de47d41bbe3aa6d17c6a0daa2b730Dave Sparks    Mutex::Autolock lock(p->mLock);
200189fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    p->mCommandComplete = true;
200289fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project    p->mSignal.signal();
200389fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project}
200489fa4ad53f2f4d57adbc97ae1149fc00c9b6f3c5The Android Open Source Project
20054110c101c3d0dd8dbc44c8d2d0edd3e2e7d6652fMarco Nelissenint MediaPlayerService::AudioCache::getSessionId() const
20068c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent{
20078c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent    return 0;
20088c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent}
20098c563ed9ca8a863a66965330b5d14bb4b4ab59d4Eric Laurent
20107cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wangvoid MediaPlayerService::addBatteryData(uint32_t params)
20117cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang{
20127cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    Mutex::Autolock lock(mLock);
20139ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20149ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    int32_t time = systemTime() / 1000000L;
20159ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20169ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    // change audio output devices. This notification comes from AudioFlinger
20179ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    if ((params & kBatteryDataSpeakerOn)
20189ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            || (params & kBatteryDataOtherAudioDeviceOn)) {
20199ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20209ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        int deviceOn[NUM_AUDIO_DEVICES];
20219ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
20229ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            deviceOn[i] = 0;
20239ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20249ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20259ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if ((params & kBatteryDataSpeakerOn)
20269ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                && (params & kBatteryDataOtherAudioDeviceOn)) {
20279ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            deviceOn[SPEAKER_AND_OTHER] = 1;
20289ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        } else if (params & kBatteryDataSpeakerOn) {
20299ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            deviceOn[SPEAKER] = 1;
20309ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        } else {
20319ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            deviceOn[OTHER_AUDIO_DEVICE] = 1;
20329ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20339ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20349ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
20359ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            if (mBatteryAudio.deviceOn[i] != deviceOn[i]){
20369ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20379ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                if (mBatteryAudio.refCount > 0) { // if playing audio
20389ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    if (!deviceOn[i]) {
20399ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                        mBatteryAudio.lastTime[i] += time;
20409ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                        mBatteryAudio.totalTime[i] += mBatteryAudio.lastTime[i];
20419ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                        mBatteryAudio.lastTime[i] = 0;
20429ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    } else {
20439ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                        mBatteryAudio.lastTime[i] = 0 - time;
20449ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    }
20459ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                }
20469ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20479ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                mBatteryAudio.deviceOn[i] = deviceOn[i];
20489ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            }
20499ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20509ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        return;
20519ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    }
20529ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20539ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    // an sudio stream is started
20549ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    if (params & kBatteryDataAudioFlingerStart) {
20559ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        // record the start time only if currently no other audio
20569ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        // is being played
20579ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if (mBatteryAudio.refCount == 0) {
20589ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
20599ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                if (mBatteryAudio.deviceOn[i]) {
20609ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    mBatteryAudio.lastTime[i] -= time;
20619ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                }
20629ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            }
20639ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20649ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20659ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.refCount ++;
20669ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        return;
20679ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20689ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    } else if (params & kBatteryDataAudioFlingerStop) {
20699ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if (mBatteryAudio.refCount <= 0) {
20705ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Battery track warning: refCount is <= 0");
20719ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            return;
20729ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20739ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20749ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        // record the stop time only if currently this is the only
20759ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        // audio being played
20769ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if (mBatteryAudio.refCount == 1) {
20779ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
20789ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                if (mBatteryAudio.deviceOn[i]) {
20799ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    mBatteryAudio.lastTime[i] += time;
20809ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    mBatteryAudio.totalTime[i] += mBatteryAudio.lastTime[i];
20819ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                    mBatteryAudio.lastTime[i] = 0;
20829ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                }
20839ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            }
20849ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
20859ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20869ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.refCount --;
20879ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        return;
20889ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    }
20899ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
20907cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    int uid = IPCThreadState::self()->getCallingUid();
20917cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    if (uid == AID_MEDIA) {
20927cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        return;
20937cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    }
20947cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    int index = mBatteryData.indexOfKey(uid);
20957cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
20967cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    if (index < 0) { // create a new entry for this UID
20977cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        BatteryUsageInfo info;
20987cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.audioTotalTime = 0;
20997cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.videoTotalTime = 0;
21007cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.audioLastTime = 0;
21017cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.videoLastTime = 0;
21027cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.refCount = 0;
21037cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21049ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if (mBatteryData.add(uid, info) == NO_MEMORY) {
210529357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Battery track error: no memory for new app");
21069ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            return;
21079ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
21087cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    }
21097cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21107cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    BatteryUsageInfo &info = mBatteryData.editValueFor(uid);
21117cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21127cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    if (params & kBatteryDataCodecStarted) {
21137cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (params & kBatteryDataTrackAudio) {
21147cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.audioLastTime -= time;
21157cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.refCount ++;
21167cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21177cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (params & kBatteryDataTrackVideo) {
21187cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.videoLastTime -= time;
21197cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.refCount ++;
21207cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21217cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    } else {
21227cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (info.refCount == 0) {
21235ff1dd576bb93c45b44088a51544a18fc43ebf58Steve Block            ALOGW("Battery track warning: refCount is already 0");
21247cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            return;
21257cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        } else if (info.refCount < 0) {
212629357bc2c0dd7c43ad3bd0c8e3efa4e6fd9bfd47Steve Block            ALOGE("Battery track error: refCount < 0");
21277cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            mBatteryData.removeItem(uid);
21287cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            return;
21297cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21307cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21317cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (params & kBatteryDataTrackAudio) {
21327cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.audioLastTime += time;
21337cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.refCount --;
21347cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21357cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (params & kBatteryDataTrackVideo) {
21367cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.videoLastTime += time;
21377cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.refCount --;
21387cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21397cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21407cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        // no stream is being played by this UID
21417cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (info.refCount == 0) {
21427cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.audioTotalTime += info.audioLastTime;
21437cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.audioLastTime = 0;
21447cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.videoTotalTime += info.videoLastTime;
21457cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            info.videoLastTime = 0;
21467cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21477cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    }
21487cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang}
21497cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21507cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wangstatus_t MediaPlayerService::pullBatteryData(Parcel* reply) {
21517cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    Mutex::Autolock lock(mLock);
21529ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
21539ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    // audio output devices usage
21549ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    int32_t time = systemTime() / 1000000L; //in ms
21559ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    int32_t totalTime;
21569ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
21579ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    for (int i = 0; i < NUM_AUDIO_DEVICES; i++) {
21589ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        totalTime = mBatteryAudio.totalTime[i];
21599ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
21609ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        if (mBatteryAudio.deviceOn[i]
21619ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang            && (mBatteryAudio.lastTime[i] != 0)) {
21629ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                int32_t tmpTime = mBatteryAudio.lastTime[i] + time;
21639ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang                totalTime += tmpTime;
21649ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        }
21659ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
21669ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        reply->writeInt32(totalTime);
21679ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        // reset the total time
21689ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang        mBatteryAudio.totalTime[i] = 0;
21699ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang   }
21709ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang
21719ee159b79022b2e1a050acb3890ce948e99e9ccbGloria Wang    // codec usage
21727cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    BatteryUsageInfo info;
21737cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    int size = mBatteryData.size();
21747cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21757cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    reply->writeInt32(size);
21767cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    int i = 0;
21777cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21787cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    while (i < size) {
21797cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info = mBatteryData.valueAt(i);
21807cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21817cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        reply->writeInt32(mBatteryData.keyAt(i)); //UID
21827cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        reply->writeInt32(info.audioTotalTime);
21837cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        reply->writeInt32(info.videoTotalTime);
21847cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21857cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.audioTotalTime = 0;
21867cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        info.videoTotalTime = 0;
21877cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang
21887cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        // remove the UID entry where no stream is being played
21897cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        if (info.refCount <= 0) {
21907cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            mBatteryData.removeItemsAt(i);
21917cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            size --;
21927cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang            i --;
21937cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        }
21947cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang        i++;
21957cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    }
21967cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang    return NO_ERROR;
21977cf180c9bff69e5cc4a2f4e53b432db45ebbebabGloria Wang}
2198a64c8c79af1a15911c55306d83a797fa50969f77niko} // namespace android
2199