blob: 858e5aab057a3ad3bb74b1002106e66464fb1fcf [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/* //device/servers/AudioFlinger/AudioDumpInterface.cpp
2**
3** Copyright 2008, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
18#define LOG_TAG "AudioFlingerDump"
Eric Laurenta553c252009-07-17 12:17:14 -070019//#define LOG_NDEBUG 0
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080020
21#include <stdint.h>
22#include <sys/types.h>
23#include <utils/Log.h>
24
25#include <stdlib.h>
26#include <unistd.h>
27
28#include "AudioDumpInterface.h"
29
30namespace android {
31
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080032// ----------------------------------------------------------------------------
33
34AudioDumpInterface::AudioDumpInterface(AudioHardwareInterface* hw)
Eric Laurenta553c252009-07-17 12:17:14 -070035 : mFirstHwOutput(true), mPolicyCommands(String8("")), mFileName(String8(""))
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080036{
37 if(hw == 0) {
38 LOGE("Dump construct hw = 0");
39 }
40 mFinalInterface = hw;
Eric Laurenta553c252009-07-17 12:17:14 -070041 LOGV("Constructor %p, mFinalInterface %p", this, mFinalInterface);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080042}
43
44
45AudioDumpInterface::~AudioDumpInterface()
46{
Eric Laurenta553c252009-07-17 12:17:14 -070047 for (size_t i = 0; i < mOutputs.size(); i++) {
48 closeOutputStream((AudioStreamOut *)mOutputs[i]);
49 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080050 if(mFinalInterface) delete mFinalInterface;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080051}
52
53
54AudioStreamOut* AudioDumpInterface::openOutputStream(
Eric Laurenta553c252009-07-17 12:17:14 -070055 uint32_t devices, int *format, uint32_t *channels, uint32_t *sampleRate, status_t *status)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080056{
Eric Laurenta553c252009-07-17 12:17:14 -070057 AudioStreamOut* outFinal = NULL;
58 int lFormat = AudioSystem::PCM_16_BIT;
59 uint32_t lChannels = AudioSystem::CHANNEL_OUT_STEREO;
60 uint32_t lRate = 44100;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080061
Eric Laurenta553c252009-07-17 12:17:14 -070062
63 if (AudioSystem::isA2dpDevice((AudioSystem::audio_devices)devices) || mFirstHwOutput) {
64 outFinal = mFinalInterface->openOutputStream(devices, format, channels, sampleRate, status);
65 if (outFinal != 0) {
66 lFormat = outFinal->format();
67 lChannels = outFinal->channels();
68 lRate = outFinal->sampleRate();
69 if (!AudioSystem::isA2dpDevice((AudioSystem::audio_devices)devices)) {
70 mFirstHwOutput = false;
71 }
72 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080073 } else {
Eric Laurent69682b42009-08-04 07:29:18 -070074 if (format != 0 && *format != 0) {
75 lFormat = *format;
76 } else {
77 lFormat = AudioSystem::PCM_16_BIT;
78 }
79 if (channels != 0 && *channels != 0) {
80 lChannels = *channels;
81 } else {
82 lChannels = AudioSystem::CHANNEL_OUT_STEREO;
83 }
84 if (sampleRate != 0 && *sampleRate != 0) {
85 lRate = *sampleRate;
86 } else {
87 lRate = 44100;
88 }
Eric Laurenta553c252009-07-17 12:17:14 -070089 if (status) *status = NO_ERROR;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080090 }
Eric Laurenta553c252009-07-17 12:17:14 -070091 LOGV("openOutputStream(), outFinal %p", outFinal);
92
93 AudioStreamOutDump *dumOutput = new AudioStreamOutDump(this, mOutputs.size(), outFinal,
94 devices, lFormat, lChannels, lRate);
95 mOutputs.add(dumOutput);
96
97 return dumOutput;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080098}
99
Eric Laurenta553c252009-07-17 12:17:14 -0700100void AudioDumpInterface::closeOutputStream(AudioStreamOut* out)
101{
102 AudioStreamOutDump *dumpOut = (AudioStreamOutDump *)out;
103
104 if (mOutputs.indexOf(dumpOut) < 0) {
105 LOGW("Attempt to close invalid output stream");
106 return;
107 }
Eric Laurent69682b42009-08-04 07:29:18 -0700108
109 LOGV("closeOutputStream() output %p", out);
110
Eric Laurenta553c252009-07-17 12:17:14 -0700111 dumpOut->standby();
112 if (dumpOut->finalStream() != NULL) {
113 mFinalInterface->closeOutputStream(dumpOut->finalStream());
Eric Laurent69682b42009-08-04 07:29:18 -0700114 mFirstHwOutput = true;
Eric Laurenta553c252009-07-17 12:17:14 -0700115 }
116
117 mOutputs.remove(dumpOut);
118 delete dumpOut;
119}
120
121AudioStreamIn* AudioDumpInterface::openInputStream(uint32_t devices, int *format, uint32_t *channels,
122 uint32_t *sampleRate, status_t *status, AudioSystem::audio_in_acoustics acoustics)
123{
124 AudioStreamIn* inFinal = NULL;
125 int lFormat = AudioSystem::PCM_16_BIT;
126 uint32_t lChannels = AudioSystem::CHANNEL_IN_MONO;
127 uint32_t lRate = 8000;
128
129
130 if (mInputs.size() == 0) {
131 inFinal = mFinalInterface->openInputStream(devices, format, channels, sampleRate, status, acoustics);
132 if (inFinal == 0) return 0;
133
134 lFormat = inFinal->format();
135 lChannels = inFinal->channels();
136 lRate = inFinal->sampleRate();
137 } else {
138 if (format != 0 && *format != 0) lFormat = *format;
139 if (channels != 0 && *channels != 0) lChannels = *channels;
140 if (sampleRate != 0 && *sampleRate != 0) lRate = *sampleRate;
141 if (status) *status = NO_ERROR;
142 }
143 LOGV("openInputStream(), inFinal %p", inFinal);
144
145 AudioStreamInDump *dumInput = new AudioStreamInDump(this, mInputs.size(), inFinal,
146 devices, lFormat, lChannels, lRate);
147 mInputs.add(dumInput);
148
149 return dumInput;
150}
151void AudioDumpInterface::closeInputStream(AudioStreamIn* in)
152{
153 AudioStreamInDump *dumpIn = (AudioStreamInDump *)in;
154
155 if (mInputs.indexOf(dumpIn) < 0) {
156 LOGW("Attempt to close invalid input stream");
157 return;
158 }
159 dumpIn->standby();
160 if (dumpIn->finalStream() != NULL) {
161 mFinalInterface->closeInputStream(dumpIn->finalStream());
162 }
163
164 mInputs.remove(dumpIn);
165 delete dumpIn;
166}
167
168
169status_t AudioDumpInterface::setParameters(const String8& keyValuePairs)
170{
171 AudioParameter param = AudioParameter(keyValuePairs);
172 String8 value;
173 int valueInt;
174 LOGV("setParameters %s", keyValuePairs.string());
175
176 if (param.get(String8("test_cmd_file_name"), value) == NO_ERROR) {
177 mFileName = value;
Eric Laurent69682b42009-08-04 07:29:18 -0700178 param.remove(String8("test_cmd_file_name"));
Eric Laurenta553c252009-07-17 12:17:14 -0700179 }
180 if (param.get(String8("test_cmd_policy"), value) == NO_ERROR) {
181 Mutex::Autolock _l(mLock);
182 param.remove(String8("test_cmd_policy"));
183 mPolicyCommands = param.toString();
184 LOGV("test_cmd_policy command %s written", mPolicyCommands.string());
185 return NO_ERROR;
186 }
187
188 if (mFinalInterface != 0 ) return mFinalInterface->setParameters(keyValuePairs);
189 return NO_ERROR;
190}
191
192String8 AudioDumpInterface::getParameters(const String8& keys)
193{
194 AudioParameter param = AudioParameter(keys);
Eric Laurent69682b42009-08-04 07:29:18 -0700195 AudioParameter response;
Eric Laurenta553c252009-07-17 12:17:14 -0700196 String8 value;
197
198// LOGV("getParameters %s", keys.string());
Eric Laurenta553c252009-07-17 12:17:14 -0700199 if (param.get(String8("test_cmd_policy"), value) == NO_ERROR) {
200 Mutex::Autolock _l(mLock);
Eric Laurent69682b42009-08-04 07:29:18 -0700201 if (mPolicyCommands.length() != 0) {
202 response = AudioParameter(mPolicyCommands);
203 response.addInt(String8("test_cmd_policy"), 1);
204 } else {
205 response.addInt(String8("test_cmd_policy"), 0);
206 }
207 param.remove(String8("test_cmd_policy"));
Eric Laurenta553c252009-07-17 12:17:14 -0700208// LOGV("test_cmd_policy command %s read", mPolicyCommands.string());
Eric Laurenta553c252009-07-17 12:17:14 -0700209 }
210
Eric Laurent69682b42009-08-04 07:29:18 -0700211 if (param.get(String8("test_cmd_file_name"), value) == NO_ERROR) {
212 response.add(String8("test_cmd_file_name"), mFileName);
213 param.remove(String8("test_cmd_file_name"));
214 }
215
216 String8 keyValuePairs = response.toString();
217
218 if (param.size() && mFinalInterface != 0 ) {
219 keyValuePairs += ";";
220 keyValuePairs += mFinalInterface->getParameters(param.toString());
221 }
222
223 return keyValuePairs;
Eric Laurenta553c252009-07-17 12:17:14 -0700224}
225
226
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800227// ----------------------------------------------------------------------------
228
Eric Laurenta553c252009-07-17 12:17:14 -0700229AudioStreamOutDump::AudioStreamOutDump(AudioDumpInterface *interface,
230 int id,
231 AudioStreamOut* finalStream,
232 uint32_t devices,
233 int format,
234 uint32_t channels,
235 uint32_t sampleRate)
236 : mInterface(interface), mId(id),
237 mSampleRate(sampleRate), mFormat(format), mChannels(channels), mLatency(0), mDevice(devices),
238 mBufferSize(1024), mFinalStream(finalStream), mOutFile(0), mFileCount(0)
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800239{
Eric Laurenta553c252009-07-17 12:17:14 -0700240 LOGV("AudioStreamOutDump Constructor %p, mInterface %p, mFinalStream %p", this, mInterface, mFinalStream);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800241}
242
243
244AudioStreamOutDump::~AudioStreamOutDump()
245{
Eric Laurent69682b42009-08-04 07:29:18 -0700246 LOGV("AudioStreamOutDump destructor");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800247 Close();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800248}
249
250ssize_t AudioStreamOutDump::write(const void* buffer, size_t bytes)
251{
252 ssize_t ret;
253
Eric Laurenta553c252009-07-17 12:17:14 -0700254 if (mFinalStream) {
255 ret = mFinalStream->write(buffer, bytes);
256 } else {
257 usleep((bytes * 1000000) / frameSize() / sampleRate());
258 ret = bytes;
259 }
260 if(!mOutFile) {
261 if (mInterface->fileName() != "") {
262 char name[255];
263 sprintf(name, "%s_%d_%d.pcm", mInterface->fileName().string(), mId, ++mFileCount);
264 mOutFile = fopen(name, "wb");
265 LOGV("Opening dump file %s, fh %p", name, mOutFile);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800266 }
267 }
268 if (mOutFile) {
269 fwrite(buffer, bytes, 1, mOutFile);
270 }
271 return ret;
272}
273
274status_t AudioStreamOutDump::standby()
275{
Eric Laurenta553c252009-07-17 12:17:14 -0700276 LOGV("AudioStreamOutDump standby(), mOutFile %p, mFinalStream %p", mOutFile, mFinalStream);
277
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800278 Close();
Eric Laurenta553c252009-07-17 12:17:14 -0700279 if (mFinalStream != 0 ) return mFinalStream->standby();
280 return NO_ERROR;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800281}
282
Eric Laurenta553c252009-07-17 12:17:14 -0700283uint32_t AudioStreamOutDump::sampleRate() const
284{
285 if (mFinalStream != 0 ) return mFinalStream->sampleRate();
286 return mSampleRate;
287}
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800288
Eric Laurenta553c252009-07-17 12:17:14 -0700289size_t AudioStreamOutDump::bufferSize() const
290{
291 if (mFinalStream != 0 ) return mFinalStream->bufferSize();
292 return mBufferSize;
293}
294
295uint32_t AudioStreamOutDump::channels() const
296{
297 if (mFinalStream != 0 ) return mFinalStream->channels();
298 return mChannels;
299}
300int AudioStreamOutDump::format() const
301{
302 if (mFinalStream != 0 ) return mFinalStream->format();
303 return mFormat;
304}
305uint32_t AudioStreamOutDump::latency() const
306{
307 if (mFinalStream != 0 ) return mFinalStream->latency();
308 return 0;
309}
310status_t AudioStreamOutDump::setVolume(float left, float right)
311{
312 if (mFinalStream != 0 ) return mFinalStream->setVolume(left, right);
313 return NO_ERROR;
314}
315status_t AudioStreamOutDump::setParameters(const String8& keyValuePairs)
316{
Eric Laurent69682b42009-08-04 07:29:18 -0700317 LOGV("AudioStreamOutDump::setParameters %s", keyValuePairs.string());
318
319 if (mFinalStream != 0 ) {
320 return mFinalStream->setParameters(keyValuePairs);
321 }
322
323 AudioParameter param = AudioParameter(keyValuePairs);
324 String8 value;
325 int valueInt;
326 status_t status = NO_ERROR;
327
328 if (param.getInt(String8("set_id"), valueInt) == NO_ERROR) {
329 mId = valueInt;
330 }
331
332 if (param.getInt(String8("format"), valueInt) == NO_ERROR) {
333 if (mOutFile == 0) {
334 mFormat = valueInt;
335 } else {
336 status = INVALID_OPERATION;
337 }
338 }
339 if (param.getInt(String8("channels"), valueInt) == NO_ERROR) {
340 if (valueInt == AudioSystem::CHANNEL_OUT_STEREO || valueInt == AudioSystem::CHANNEL_OUT_MONO) {
341 mChannels = valueInt;
342 } else {
343 status = BAD_VALUE;
344 }
345 }
346 if (param.getInt(String8("sampling_rate"), valueInt) == NO_ERROR) {
347 if (valueInt > 0 && valueInt <= 48000) {
348 if (mOutFile == 0) {
349 mSampleRate = valueInt;
350 } else {
351 status = INVALID_OPERATION;
352 }
353 } else {
354 status = BAD_VALUE;
355 }
356 }
357 return status;
Eric Laurenta553c252009-07-17 12:17:14 -0700358}
Eric Laurent69682b42009-08-04 07:29:18 -0700359
Eric Laurenta553c252009-07-17 12:17:14 -0700360String8 AudioStreamOutDump::getParameters(const String8& keys)
361{
Eric Laurent69682b42009-08-04 07:29:18 -0700362 if (mFinalStream != 0 ) return mFinalStream->getParameters(keys);
363
364 AudioParameter param = AudioParameter(keys);
365 return param.toString();
Eric Laurenta553c252009-07-17 12:17:14 -0700366}
367
368status_t AudioStreamOutDump::dump(int fd, const Vector<String16>& args)
369{
370 if (mFinalStream != 0 ) return mFinalStream->dump(fd, args);
371 return NO_ERROR;
372}
373
374void AudioStreamOutDump::Close()
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800375{
376 if(mOutFile) {
377 fclose(mOutFile);
378 mOutFile = 0;
379 }
380}
381
Eric Laurenta553c252009-07-17 12:17:14 -0700382// ----------------------------------------------------------------------------
383
384AudioStreamInDump::AudioStreamInDump(AudioDumpInterface *interface,
385 int id,
386 AudioStreamIn* finalStream,
387 uint32_t devices,
388 int format,
389 uint32_t channels,
390 uint32_t sampleRate)
391 : mInterface(interface), mId(id),
392 mSampleRate(sampleRate), mFormat(format), mChannels(channels), mDevice(devices),
393 mBufferSize(1024), mFinalStream(finalStream), mInFile(0)
394{
395 LOGV("AudioStreamInDump Constructor %p, mInterface %p, mFinalStream %p", this, mInterface, mFinalStream);
396}
397
398
399AudioStreamInDump::~AudioStreamInDump()
400{
401 Close();
402}
403
404ssize_t AudioStreamInDump::read(void* buffer, ssize_t bytes)
405{
406 if (mFinalStream) {
407 return mFinalStream->read(buffer, bytes);
408 }
409
410 usleep((bytes * 1000000) / frameSize() / sampleRate());
411
412 if(!mInFile) {
413 char name[255];
414 strcpy(name, "/sdcard/music/sine440");
415 if (channels() == AudioSystem::CHANNEL_IN_MONO) {
416 strcat(name, "_mo");
417 } else {
418 strcat(name, "_st");
419 }
420 if (format() == AudioSystem::PCM_16_BIT) {
421 strcat(name, "_16b");
422 } else {
423 strcat(name, "_8b");
424 }
425 if (sampleRate() < 16000) {
426 strcat(name, "_8k");
427 } else if (sampleRate() < 32000) {
428 strcat(name, "_22k");
429 } else if (sampleRate() < 48000) {
430 strcat(name, "_44k");
431 } else {
432 strcat(name, "_48k");
433 }
434 strcat(name, ".wav");
435 mInFile = fopen(name, "rb");
436 LOGV("Opening dump file %s, fh %p", name, mInFile);
437 if (mInFile) {
438 fseek(mInFile, AUDIO_DUMP_WAVE_HDR_SIZE, SEEK_SET);
439 }
440
441 }
442 if (mInFile) {
443 ssize_t bytesRead = fread(buffer, bytes, 1, mInFile);
444 if (bytesRead != bytes) {
445 fseek(mInFile, AUDIO_DUMP_WAVE_HDR_SIZE, SEEK_SET);
446 fread((uint8_t *)buffer+bytesRead, bytes-bytesRead, 1, mInFile);
447 }
448 }
449 return bytes;
450}
451
452status_t AudioStreamInDump::standby()
453{
454 LOGV("AudioStreamInDump standby(), mInFile %p, mFinalStream %p", mInFile, mFinalStream);
455
456 Close();
457 if (mFinalStream != 0 ) return mFinalStream->standby();
458 return NO_ERROR;
459}
460
461status_t AudioStreamInDump::setGain(float gain)
462{
463 if (mFinalStream != 0 ) return mFinalStream->setGain(gain);
464 return NO_ERROR;
465}
466
467uint32_t AudioStreamInDump::sampleRate() const
468{
469 if (mFinalStream != 0 ) return mFinalStream->sampleRate();
470 return mSampleRate;
471}
472
473size_t AudioStreamInDump::bufferSize() const
474{
475 if (mFinalStream != 0 ) return mFinalStream->bufferSize();
476 return mBufferSize;
477}
478
479uint32_t AudioStreamInDump::channels() const
480{
481 if (mFinalStream != 0 ) return mFinalStream->channels();
482 return mChannels;
483}
484
485int AudioStreamInDump::format() const
486{
487 if (mFinalStream != 0 ) return mFinalStream->format();
488 return mFormat;
489}
490
491status_t AudioStreamInDump::setParameters(const String8& keyValuePairs)
492{
493 LOGV("AudioStreamInDump::setParameters()");
494 if (mFinalStream != 0 ) return mFinalStream->setParameters(keyValuePairs);
495 return NO_ERROR;
496}
497
498String8 AudioStreamInDump::getParameters(const String8& keys)
499{
Eric Laurent69682b42009-08-04 07:29:18 -0700500 if (mFinalStream != 0 ) return mFinalStream->getParameters(keys);
501
502 AudioParameter param = AudioParameter(keys);
503 return param.toString();
Eric Laurenta553c252009-07-17 12:17:14 -0700504}
505
506status_t AudioStreamInDump::dump(int fd, const Vector<String16>& args)
507{
508 if (mFinalStream != 0 ) return mFinalStream->dump(fd, args);
509 return NO_ERROR;
510}
511
512void AudioStreamInDump::Close()
513{
514 if(mInFile) {
515 fclose(mInFile);
516 mInFile = 0;
517 }
518}
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800519}; // namespace android