/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ //#define LOG_NDEBUG 0 #define LOG_TAG "AudioSource" #include #include #include #include #include #include #include #include #include #include namespace android { AudioSource::AudioSource( int inputSource, uint32_t sampleRate, uint32_t channels) : mStarted(false), mCollectStats(false), mTotalReadTimeUs(0), mTotalReadBytes(0), mTotalReads(0), mGroup(NULL) { LOGV("sampleRate: %d, channels: %d", sampleRate, channels); uint32_t flags = AudioRecord::RECORD_AGC_ENABLE | AudioRecord::RECORD_NS_ENABLE | AudioRecord::RECORD_IIR_ENABLE; mRecord = new AudioRecord( inputSource, sampleRate, AudioSystem::PCM_16_BIT, channels > 1? AudioSystem::CHANNEL_IN_STEREO: AudioSystem::CHANNEL_IN_MONO, 4 * kMaxBufferSize / sizeof(int16_t), /* Enable ping-pong buffers */ flags); mInitCheck = mRecord->initCheck(); } AudioSource::~AudioSource() { if (mStarted) { stop(); } delete mRecord; mRecord = NULL; } status_t AudioSource::initCheck() const { return mInitCheck; } status_t AudioSource::start(MetaData *params) { if (mStarted) { return UNKNOWN_ERROR; } char value[PROPERTY_VALUE_MAX]; if (property_get("media.stagefright.record-stats", value, NULL) && (!strcmp(value, "1") || !strcasecmp(value, "true"))) { mCollectStats = true; } mTrackMaxAmplitude = false; mMaxAmplitude = 0; mStartTimeUs = 0; int64_t startTimeUs; if (params && params->findInt64(kKeyTime, &startTimeUs)) { mStartTimeUs = startTimeUs; } status_t err = mRecord->start(); if (err == OK) { mGroup = new MediaBufferGroup; mGroup->add_buffer(new MediaBuffer(kMaxBufferSize)); mStarted = true; } return err; } status_t AudioSource::stop() { if (!mStarted) { return UNKNOWN_ERROR; } mRecord->stop(); delete mGroup; mGroup = NULL; mStarted = false; if (mCollectStats) { LOGI("%lld reads: %.2f bps in %lld us", mTotalReads, (mTotalReadBytes * 8000000.0) / mTotalReadTimeUs, mTotalReadTimeUs); } return OK; } sp AudioSource::getFormat() { sp meta = new MetaData; meta->setCString(kKeyMIMEType, MEDIA_MIMETYPE_AUDIO_RAW); meta->setInt32(kKeySampleRate, mRecord->getSampleRate()); meta->setInt32(kKeyChannelCount, mRecord->channelCount()); meta->setInt32(kKeyMaxInputSize, kMaxBufferSize); return meta; } status_t AudioSource::read( MediaBuffer **out, const ReadOptions *options) { *out = NULL; ++mTotalReads; MediaBuffer *buffer; CHECK_EQ(mGroup->acquire_buffer(&buffer), OK); uint32_t numFramesRecorded; mRecord->getPosition(&numFramesRecorded); int64_t latency = mRecord->latency() * 1000; int64_t readTime = systemTime() / 1000; if (numFramesRecorded == 0) { // Initial delay if (mStartTimeUs > 0) { mStartTimeUs = readTime - mStartTimeUs; } else { mStartTimeUs += latency; } } ssize_t n = 0; if (mCollectStats) { n = mRecord->read(buffer->data(), buffer->size()); int64_t endTime = systemTime() / 1000; mTotalReadTimeUs += (endTime - readTime); if (n >= 0) { mTotalReadBytes += n; } } else { n = mRecord->read(buffer->data(), buffer->size()); } if (n < 0) { buffer->release(); buffer = NULL; return (status_t)n; } if (mTrackMaxAmplitude) { trackMaxAmplitude((int16_t *) buffer->data(), n >> 1); } uint32_t sampleRate = mRecord->getSampleRate(); int64_t timestampUs = (1000000LL * numFramesRecorded) / sampleRate + mStartTimeUs; buffer->meta_data()->setInt64(kKeyTime, timestampUs); LOGV("initial delay: %lld, sample rate: %d, timestamp: %lld", mStartTimeUs, sampleRate, timestampUs); buffer->set_range(0, n); *out = buffer; return OK; } void AudioSource::trackMaxAmplitude(int16_t *data, int nSamples) { for (int i = nSamples; i > 0; --i) { int16_t value = *data++; if (value < 0) { value = -value; } if (mMaxAmplitude < value) { mMaxAmplitude = value; } } } int16_t AudioSource::getMaxAmplitude() { // First call activates the tracking. if (!mTrackMaxAmplitude) { mTrackMaxAmplitude = true; } int16_t value = mMaxAmplitude; mMaxAmplitude = 0; LOGV("max amplitude since last call: %d", value); return value; } } // namespace android