IAudioTrack.cpp revision e4756fe3a387615acb63c6a05788c8db9b5786cb
1/* 2** 3** Copyright 2007, The Android Open Source Project 4** 5** Licensed under the Apache License, Version 2.0 (the "License"); 6** you may not use this file except in compliance with the License. 7** You may obtain a copy of the License at 8** 9** http://www.apache.org/licenses/LICENSE-2.0 10** 11** Unless required by applicable law or agreed to in writing, software 12** distributed under the License is distributed on an "AS IS" BASIS, 13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14** See the License for the specific language governing permissions and 15** limitations under the License. 16*/ 17 18#define LOG_TAG "IAudioTrack" 19//#define LOG_NDEBUG 0 20#include <utils/Log.h> 21 22#include <stdint.h> 23#include <sys/types.h> 24 25#include <binder/Parcel.h> 26 27#include <media/IAudioTrack.h> 28 29namespace android { 30 31enum { 32 GET_CBLK = IBinder::FIRST_CALL_TRANSACTION, 33 START, 34 STOP, 35 FLUSH, 36 RESERVED, // was MUTE 37 PAUSE, 38 ATTACH_AUX_EFFECT, 39 ALLOCATE_TIMED_BUFFER, 40 QUEUE_TIMED_BUFFER, 41 SET_MEDIA_TIME_TRANSFORM, 42}; 43 44class BpAudioTrack : public BpInterface<IAudioTrack> 45{ 46public: 47 BpAudioTrack(const sp<IBinder>& impl) 48 : BpInterface<IAudioTrack>(impl) 49 { 50 } 51 52 virtual sp<IMemory> getCblk() const 53 { 54 Parcel data, reply; 55 sp<IMemory> cblk; 56 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 57 status_t status = remote()->transact(GET_CBLK, data, &reply); 58 if (status == NO_ERROR) { 59 cblk = interface_cast<IMemory>(reply.readStrongBinder()); 60 } 61 return cblk; 62 } 63 64 virtual status_t start() 65 { 66 Parcel data, reply; 67 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 68 status_t status = remote()->transact(START, data, &reply); 69 if (status == NO_ERROR) { 70 status = reply.readInt32(); 71 } else { 72 ALOGW("start() error: %s", strerror(-status)); 73 } 74 return status; 75 } 76 77 virtual void stop() 78 { 79 Parcel data, reply; 80 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 81 remote()->transact(STOP, data, &reply); 82 } 83 84 virtual void flush() 85 { 86 Parcel data, reply; 87 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 88 remote()->transact(FLUSH, data, &reply); 89 } 90 91 virtual void pause() 92 { 93 Parcel data, reply; 94 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 95 remote()->transact(PAUSE, data, &reply); 96 } 97 98 virtual status_t attachAuxEffect(int effectId) 99 { 100 Parcel data, reply; 101 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 102 data.writeInt32(effectId); 103 status_t status = remote()->transact(ATTACH_AUX_EFFECT, data, &reply); 104 if (status == NO_ERROR) { 105 status = reply.readInt32(); 106 } else { 107 ALOGW("attachAuxEffect() error: %s", strerror(-status)); 108 } 109 return status; 110 } 111 112 virtual status_t allocateTimedBuffer(size_t size, sp<IMemory>* buffer) { 113 Parcel data, reply; 114 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 115 data.writeInt32(size); 116 status_t status = remote()->transact(ALLOCATE_TIMED_BUFFER, 117 data, &reply); 118 if (status == NO_ERROR) { 119 status = reply.readInt32(); 120 if (status == NO_ERROR) { 121 *buffer = interface_cast<IMemory>(reply.readStrongBinder()); 122 } 123 } 124 return status; 125 } 126 127 virtual status_t queueTimedBuffer(const sp<IMemory>& buffer, 128 int64_t pts) { 129 Parcel data, reply; 130 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 131 data.writeStrongBinder(buffer->asBinder()); 132 data.writeInt64(pts); 133 status_t status = remote()->transact(QUEUE_TIMED_BUFFER, 134 data, &reply); 135 if (status == NO_ERROR) { 136 status = reply.readInt32(); 137 } 138 return status; 139 } 140 141 virtual status_t setMediaTimeTransform(const LinearTransform& xform, 142 int target) { 143 Parcel data, reply; 144 data.writeInterfaceToken(IAudioTrack::getInterfaceDescriptor()); 145 data.writeInt64(xform.a_zero); 146 data.writeInt64(xform.b_zero); 147 data.writeInt32(xform.a_to_b_numer); 148 data.writeInt32(xform.a_to_b_denom); 149 data.writeInt32(target); 150 status_t status = remote()->transact(SET_MEDIA_TIME_TRANSFORM, 151 data, &reply); 152 if (status == NO_ERROR) { 153 status = reply.readInt32(); 154 } 155 return status; 156 } 157}; 158 159IMPLEMENT_META_INTERFACE(AudioTrack, "android.media.IAudioTrack"); 160 161// ---------------------------------------------------------------------- 162 163status_t BnAudioTrack::onTransact( 164 uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags) 165{ 166 switch (code) { 167 case GET_CBLK: { 168 CHECK_INTERFACE(IAudioTrack, data, reply); 169 reply->writeStrongBinder(getCblk()->asBinder()); 170 return NO_ERROR; 171 } break; 172 case START: { 173 CHECK_INTERFACE(IAudioTrack, data, reply); 174 reply->writeInt32(start()); 175 return NO_ERROR; 176 } break; 177 case STOP: { 178 CHECK_INTERFACE(IAudioTrack, data, reply); 179 stop(); 180 return NO_ERROR; 181 } break; 182 case FLUSH: { 183 CHECK_INTERFACE(IAudioTrack, data, reply); 184 flush(); 185 return NO_ERROR; 186 } break; 187 case PAUSE: { 188 CHECK_INTERFACE(IAudioTrack, data, reply); 189 pause(); 190 return NO_ERROR; 191 } 192 case ATTACH_AUX_EFFECT: { 193 CHECK_INTERFACE(IAudioTrack, data, reply); 194 reply->writeInt32(attachAuxEffect(data.readInt32())); 195 return NO_ERROR; 196 } break; 197 case ALLOCATE_TIMED_BUFFER: { 198 CHECK_INTERFACE(IAudioTrack, data, reply); 199 sp<IMemory> buffer; 200 status_t status = allocateTimedBuffer(data.readInt32(), &buffer); 201 reply->writeInt32(status); 202 if (status == NO_ERROR) { 203 reply->writeStrongBinder(buffer->asBinder()); 204 } 205 return NO_ERROR; 206 } break; 207 case QUEUE_TIMED_BUFFER: { 208 CHECK_INTERFACE(IAudioTrack, data, reply); 209 sp<IMemory> buffer = interface_cast<IMemory>( 210 data.readStrongBinder()); 211 uint64_t pts = data.readInt64(); 212 reply->writeInt32(queueTimedBuffer(buffer, pts)); 213 return NO_ERROR; 214 } break; 215 case SET_MEDIA_TIME_TRANSFORM: { 216 CHECK_INTERFACE(IAudioTrack, data, reply); 217 LinearTransform xform; 218 xform.a_zero = data.readInt64(); 219 xform.b_zero = data.readInt64(); 220 xform.a_to_b_numer = data.readInt32(); 221 xform.a_to_b_denom = data.readInt32(); 222 int target = data.readInt32(); 223 reply->writeInt32(setMediaTimeTransform(xform, target)); 224 return NO_ERROR; 225 } break; 226 default: 227 return BBinder::onTransact(code, data, reply, flags); 228 } 229} 230 231}; // namespace android 232