mirror of
https://github.com/Evolution-X/hardware_interfaces
synced 2026-02-02 10:05:19 +00:00
Merge "audio: Fix remote submix module I/O timing and atomicity" into main am: 5260337cbb am: 73ddfafe34
Original change: https://android-review.googlesource.com/c/platform/hardware/interfaces/+/2855474 Change-Id: Id98a19caa437cc29a9ff9a948b26f7ef146cb7a2 Signed-off-by: Automerger Merge Worker <android-build-automerger-merge-worker@system.gserviceaccount.com>
This commit is contained in:
@@ -320,9 +320,9 @@ std::unique_ptr<Configuration> getPrimaryConfiguration() {
|
|||||||
// - no profiles specified
|
// - no profiles specified
|
||||||
//
|
//
|
||||||
// Mix ports:
|
// Mix ports:
|
||||||
// * "r_submix output", maximum 20 opened streams, maximum 10 active streams
|
// * "r_submix output", maximum 10 opened streams, maximum 10 active streams
|
||||||
// - profile PCM 16-bit; MONO, STEREO; 8000, 11025, 16000, 32000, 44100, 48000
|
// - profile PCM 16-bit; MONO, STEREO; 8000, 11025, 16000, 32000, 44100, 48000
|
||||||
// * "r_submix input", maximum 20 opened streams, maximum 10 active streams
|
// * "r_submix input", maximum 10 opened streams, maximum 10 active streams
|
||||||
// - profile PCM 16-bit; MONO, STEREO; 8000, 11025, 16000, 32000, 44100, 48000
|
// - profile PCM 16-bit; MONO, STEREO; 8000, 11025, 16000, 32000, 44100, 48000
|
||||||
//
|
//
|
||||||
// Routes:
|
// Routes:
|
||||||
@@ -355,12 +355,12 @@ std::unique_ptr<Configuration> getRSubmixConfiguration() {
|
|||||||
// Mix ports
|
// Mix ports
|
||||||
|
|
||||||
AudioPort rsubmixOutMix =
|
AudioPort rsubmixOutMix =
|
||||||
createPort(c.nextPortId++, "r_submix output", 0, false, createPortMixExt(20, 10));
|
createPort(c.nextPortId++, "r_submix output", 0, false, createPortMixExt(10, 10));
|
||||||
rsubmixOutMix.profiles = remoteSubmixPcmAudioProfiles;
|
rsubmixOutMix.profiles = remoteSubmixPcmAudioProfiles;
|
||||||
c.ports.push_back(rsubmixOutMix);
|
c.ports.push_back(rsubmixOutMix);
|
||||||
|
|
||||||
AudioPort rsubmixInMix =
|
AudioPort rsubmixInMix =
|
||||||
createPort(c.nextPortId++, "r_submix input", 0, true, createPortMixExt(20, 10));
|
createPort(c.nextPortId++, "r_submix input", 0, true, createPortMixExt(10, 10));
|
||||||
rsubmixInMix.profiles = remoteSubmixPcmAudioProfiles;
|
rsubmixInMix.profiles = remoteSubmixPcmAudioProfiles;
|
||||||
c.ports.push_back(rsubmixInMix);
|
c.ports.push_back(rsubmixInMix);
|
||||||
|
|
||||||
|
|||||||
@@ -71,6 +71,10 @@ class StreamRemoteSubmix : public StreamCommonImpl {
|
|||||||
static constexpr int kMaxReadFailureAttempts = 3;
|
static constexpr int kMaxReadFailureAttempts = 3;
|
||||||
// 5ms between two read attempts when pipe is empty
|
// 5ms between two read attempts when pipe is empty
|
||||||
static constexpr int kReadAttemptSleepUs = 5000;
|
static constexpr int kReadAttemptSleepUs = 5000;
|
||||||
|
|
||||||
|
long mStartTimeNs = 0;
|
||||||
|
long mFramesSinceStart = 0;
|
||||||
|
int mReadErrorCount = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class StreamInRemoteSubmix final : public StreamIn, public StreamSwitcher {
|
class StreamInRemoteSubmix final : public StreamIn, public StreamSwitcher {
|
||||||
|
|||||||
@@ -16,6 +16,9 @@
|
|||||||
|
|
||||||
#define LOG_TAG "AHAL_StreamRemoteSubmix"
|
#define LOG_TAG "AHAL_StreamRemoteSubmix"
|
||||||
#include <android-base/logging.h>
|
#include <android-base/logging.h>
|
||||||
|
#include <audio_utils/clock.h>
|
||||||
|
#include <error/Result.h>
|
||||||
|
#include <error/expected_utils.h>
|
||||||
|
|
||||||
#include "core-impl/StreamRemoteSubmix.h"
|
#include "core-impl/StreamRemoteSubmix.h"
|
||||||
|
|
||||||
@@ -50,37 +53,33 @@ std::map<AudioDeviceAddress, std::shared_ptr<SubmixRoute>> StreamRemoteSubmix::s
|
|||||||
if (routeItr != sSubmixRoutes.end()) {
|
if (routeItr != sSubmixRoutes.end()) {
|
||||||
mCurrentRoute = routeItr->second;
|
mCurrentRoute = routeItr->second;
|
||||||
}
|
}
|
||||||
}
|
// If route is not available for this port, add it.
|
||||||
// If route is not available for this port, add it.
|
if (mCurrentRoute == nullptr) {
|
||||||
if (mCurrentRoute == nullptr) {
|
// Initialize the pipe.
|
||||||
// Initialize the pipe.
|
mCurrentRoute = std::make_shared<SubmixRoute>();
|
||||||
mCurrentRoute = std::make_shared<SubmixRoute>();
|
if (::android::OK != mCurrentRoute->createPipe(mStreamConfig)) {
|
||||||
if (::android::OK != mCurrentRoute->createPipe(mStreamConfig)) {
|
LOG(ERROR) << __func__ << ": create pipe failed";
|
||||||
LOG(ERROR) << __func__ << ": create pipe failed";
|
|
||||||
return ::android::NO_INIT;
|
|
||||||
}
|
|
||||||
{
|
|
||||||
std::lock_guard guard(sSubmixRoutesLock);
|
|
||||||
sSubmixRoutes.emplace(mDeviceAddress, mCurrentRoute);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (!mCurrentRoute->isStreamConfigValid(mIsInput, mStreamConfig)) {
|
|
||||||
LOG(ERROR) << __func__ << ": invalid stream config";
|
|
||||||
return ::android::NO_INIT;
|
|
||||||
}
|
|
||||||
sp<MonoPipe> sink = mCurrentRoute->getSink();
|
|
||||||
if (sink == nullptr) {
|
|
||||||
LOG(ERROR) << __func__ << ": nullptr sink when opening stream";
|
|
||||||
return ::android::NO_INIT;
|
|
||||||
}
|
|
||||||
// If the sink has been shutdown or pipe recreation is forced, delete the pipe and
|
|
||||||
// recreate it.
|
|
||||||
if (sink->isShutdown()) {
|
|
||||||
LOG(DEBUG) << __func__ << ": Non-nullptr shut down sink when opening stream";
|
|
||||||
if (::android::OK != mCurrentRoute->resetPipe()) {
|
|
||||||
LOG(ERROR) << __func__ << ": reset pipe failed";
|
|
||||||
return ::android::NO_INIT;
|
return ::android::NO_INIT;
|
||||||
}
|
}
|
||||||
|
sSubmixRoutes.emplace(mDeviceAddress, mCurrentRoute);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!mCurrentRoute->isStreamConfigValid(mIsInput, mStreamConfig)) {
|
||||||
|
LOG(ERROR) << __func__ << ": invalid stream config";
|
||||||
|
return ::android::NO_INIT;
|
||||||
|
}
|
||||||
|
sp<MonoPipe> sink = mCurrentRoute->getSink();
|
||||||
|
if (sink == nullptr) {
|
||||||
|
LOG(ERROR) << __func__ << ": nullptr sink when opening stream";
|
||||||
|
return ::android::NO_INIT;
|
||||||
|
}
|
||||||
|
// If the sink has been shutdown or pipe recreation is forced, delete the pipe and
|
||||||
|
// recreate it.
|
||||||
|
if (sink->isShutdown()) {
|
||||||
|
LOG(DEBUG) << __func__ << ": Non-nullptr shut down sink when opening stream";
|
||||||
|
if (::android::OK != mCurrentRoute->resetPipe()) {
|
||||||
|
LOG(ERROR) << __func__ << ": reset pipe failed";
|
||||||
|
return ::android::NO_INIT;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -110,6 +109,8 @@ std::map<AudioDeviceAddress, std::shared_ptr<SubmixRoute>> StreamRemoteSubmix::s
|
|||||||
|
|
||||||
::android::status_t StreamRemoteSubmix::start() {
|
::android::status_t StreamRemoteSubmix::start() {
|
||||||
mCurrentRoute->exitStandby(mIsInput);
|
mCurrentRoute->exitStandby(mIsInput);
|
||||||
|
mStartTimeNs = ::android::uptimeNanos();
|
||||||
|
mFramesSinceStart = 0;
|
||||||
return ::android::OK;
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -161,8 +162,21 @@ void StreamRemoteSubmix::shutdown() {
|
|||||||
*latencyMs = getDelayInUsForFrameCount(getStreamPipeSizeInFrames()) / 1000;
|
*latencyMs = getDelayInUsForFrameCount(getStreamPipeSizeInFrames()) / 1000;
|
||||||
LOG(VERBOSE) << __func__ << ": Latency " << *latencyMs << "ms";
|
LOG(VERBOSE) << __func__ << ": Latency " << *latencyMs << "ms";
|
||||||
mCurrentRoute->exitStandby(mIsInput);
|
mCurrentRoute->exitStandby(mIsInput);
|
||||||
return (mIsInput ? inRead(buffer, frameCount, actualFrameCount)
|
RETURN_STATUS_IF_ERROR(mIsInput ? inRead(buffer, frameCount, actualFrameCount)
|
||||||
: outWrite(buffer, frameCount, actualFrameCount));
|
: outWrite(buffer, frameCount, actualFrameCount));
|
||||||
|
const long bufferDurationUs =
|
||||||
|
(*actualFrameCount) * MICROS_PER_SECOND / mContext.getSampleRate();
|
||||||
|
const long totalDurationUs = (::android::uptimeNanos() - mStartTimeNs) / NANOS_PER_MICROSECOND;
|
||||||
|
mFramesSinceStart += *actualFrameCount;
|
||||||
|
const long totalOffsetUs =
|
||||||
|
mFramesSinceStart * MICROS_PER_SECOND / mContext.getSampleRate() - totalDurationUs;
|
||||||
|
LOG(VERBOSE) << __func__ << ": totalOffsetUs " << totalOffsetUs;
|
||||||
|
if (totalOffsetUs > 0) {
|
||||||
|
const long sleepTimeUs = std::min(totalOffsetUs, bufferDurationUs);
|
||||||
|
LOG(VERBOSE) << __func__ << ": sleeping for " << sleepTimeUs << " us";
|
||||||
|
usleep(sleepTimeUs);
|
||||||
|
}
|
||||||
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
::android::status_t StreamRemoteSubmix::refinePosition(StreamDescriptor::Position* position) {
|
::android::status_t StreamRemoteSubmix::refinePosition(StreamDescriptor::Position* position) {
|
||||||
@@ -200,12 +214,7 @@ size_t StreamRemoteSubmix::getStreamPipeSizeInFrames() {
|
|||||||
if (sink != nullptr) {
|
if (sink != nullptr) {
|
||||||
if (sink->isShutdown()) {
|
if (sink->isShutdown()) {
|
||||||
sink.clear();
|
sink.clear();
|
||||||
const auto delayUs = getDelayInUsForFrameCount(frameCount);
|
LOG(DEBUG) << __func__ << ": pipe shutdown, ignoring the write";
|
||||||
LOG(DEBUG) << __func__ << ": pipe shutdown, ignoring the write, sleeping for "
|
|
||||||
<< delayUs << " us";
|
|
||||||
// the pipe has already been shutdown, this buffer will be lost but we must
|
|
||||||
// simulate timing so we don't drain the output faster than realtime
|
|
||||||
usleep(delayUs);
|
|
||||||
*actualFrameCount = frameCount;
|
*actualFrameCount = frameCount;
|
||||||
return ::android::OK;
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
@@ -214,6 +223,9 @@ size_t StreamRemoteSubmix::getStreamPipeSizeInFrames() {
|
|||||||
return ::android::UNKNOWN_ERROR;
|
return ::android::UNKNOWN_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LOG(VERBOSE) << __func__ << ": " << mDeviceAddress.toString() << ", " << frameCount
|
||||||
|
<< " frames";
|
||||||
|
|
||||||
const bool shouldBlockWrite = mCurrentRoute->shouldBlockWrite();
|
const bool shouldBlockWrite = mCurrentRoute->shouldBlockWrite();
|
||||||
size_t availableToWrite = sink->availableToWrite();
|
size_t availableToWrite = sink->availableToWrite();
|
||||||
// NOTE: sink has been checked above and sink and source life cycles are synchronized
|
// NOTE: sink has been checked above and sink and source life cycles are synchronized
|
||||||
@@ -236,6 +248,8 @@ size_t StreamRemoteSubmix::getStreamPipeSizeInFrames() {
|
|||||||
availableToWrite = sink->availableToWrite();
|
availableToWrite = sink->availableToWrite();
|
||||||
|
|
||||||
if (!shouldBlockWrite && frameCount > availableToWrite) {
|
if (!shouldBlockWrite && frameCount > availableToWrite) {
|
||||||
|
LOG(WARNING) << __func__ << ": writing " << availableToWrite << " vs. requested "
|
||||||
|
<< frameCount;
|
||||||
// Truncate the request to avoid blocking.
|
// Truncate the request to avoid blocking.
|
||||||
frameCount = availableToWrite;
|
frameCount = availableToWrite;
|
||||||
}
|
}
|
||||||
@@ -258,92 +272,59 @@ size_t StreamRemoteSubmix::getStreamPipeSizeInFrames() {
|
|||||||
*actualFrameCount = 0;
|
*actualFrameCount = 0;
|
||||||
return ::android::UNKNOWN_ERROR;
|
return ::android::UNKNOWN_ERROR;
|
||||||
}
|
}
|
||||||
LOG(VERBOSE) << __func__ << ": wrote " << writtenFrames << "frames";
|
if (writtenFrames > 0 && frameCount > (size_t)writtenFrames) {
|
||||||
|
LOG(WARNING) << __func__ << ": wrote " << writtenFrames << " vs. requested " << frameCount;
|
||||||
|
}
|
||||||
*actualFrameCount = writtenFrames;
|
*actualFrameCount = writtenFrames;
|
||||||
return ::android::OK;
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
::android::status_t StreamRemoteSubmix::inRead(void* buffer, size_t frameCount,
|
::android::status_t StreamRemoteSubmix::inRead(void* buffer, size_t frameCount,
|
||||||
size_t* actualFrameCount) {
|
size_t* actualFrameCount) {
|
||||||
|
// in any case, it is emulated that data for the entire buffer was available
|
||||||
|
memset(buffer, 0, mStreamConfig.frameSize * frameCount);
|
||||||
|
*actualFrameCount = frameCount;
|
||||||
|
|
||||||
// about to read from audio source
|
// about to read from audio source
|
||||||
sp<MonoPipeReader> source = mCurrentRoute->getSource();
|
sp<MonoPipeReader> source = mCurrentRoute->getSource();
|
||||||
if (source == nullptr) {
|
if (source == nullptr) {
|
||||||
int readErrorCount = mCurrentRoute->notifyReadError();
|
if (++mReadErrorCount < kMaxReadErrorLogs) {
|
||||||
if (readErrorCount < kMaxReadErrorLogs) {
|
|
||||||
LOG(ERROR) << __func__
|
LOG(ERROR) << __func__
|
||||||
<< ": no audio pipe yet we're trying to read! (not all errors will be "
|
<< ": no audio pipe yet we're trying to read! (not all errors will be "
|
||||||
"logged)";
|
"logged)";
|
||||||
} else {
|
|
||||||
LOG(ERROR) << __func__ << ": Read errors " << readErrorCount;
|
|
||||||
}
|
}
|
||||||
const auto delayUs = getDelayInUsForFrameCount(frameCount);
|
|
||||||
LOG(DEBUG) << __func__ << ": no source, ignoring the read, sleeping for " << delayUs
|
|
||||||
<< " us";
|
|
||||||
usleep(delayUs);
|
|
||||||
memset(buffer, 0, mStreamConfig.frameSize * frameCount);
|
|
||||||
*actualFrameCount = frameCount;
|
|
||||||
return ::android::OK;
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LOG(VERBOSE) << __func__ << ": " << mDeviceAddress.toString() << ", " << frameCount
|
||||||
|
<< " frames";
|
||||||
// read the data from the pipe
|
// read the data from the pipe
|
||||||
int attempts = 0;
|
|
||||||
const long delayUs = kReadAttemptSleepUs;
|
|
||||||
char* buff = (char*)buffer;
|
char* buff = (char*)buffer;
|
||||||
size_t remainingFrames = frameCount;
|
size_t actuallyRead = 0;
|
||||||
int availableToRead = source->availableToRead();
|
long remainingFrames = frameCount;
|
||||||
|
const long deadlineTimeNs = ::android::uptimeNanos() +
|
||||||
while ((remainingFrames > 0) && (availableToRead > 0) && (attempts < kMaxReadFailureAttempts)) {
|
getDelayInUsForFrameCount(frameCount) * NANOS_PER_MICROSECOND;
|
||||||
LOG(VERBOSE) << __func__ << ": frames available to read " << availableToRead;
|
while (remainingFrames > 0) {
|
||||||
|
|
||||||
ssize_t framesRead = source->read(buff, remainingFrames);
|
ssize_t framesRead = source->read(buff, remainingFrames);
|
||||||
|
|
||||||
LOG(VERBOSE) << __func__ << ": frames read " << framesRead;
|
LOG(VERBOSE) << __func__ << ": frames read " << framesRead;
|
||||||
|
|
||||||
if (framesRead > 0) {
|
if (framesRead > 0) {
|
||||||
remainingFrames -= framesRead;
|
remainingFrames -= framesRead;
|
||||||
buff += framesRead * mStreamConfig.frameSize;
|
buff += framesRead * mStreamConfig.frameSize;
|
||||||
availableToRead -= framesRead;
|
LOG(VERBOSE) << __func__ << ": got " << framesRead
|
||||||
LOG(VERBOSE) << __func__ << ": (attempts = " << attempts << ") got " << framesRead
|
|
||||||
<< " frames, remaining =" << remainingFrames;
|
<< " frames, remaining =" << remainingFrames;
|
||||||
} else {
|
actuallyRead += framesRead;
|
||||||
attempts++;
|
}
|
||||||
LOG(WARNING) << __func__ << ": read returned " << framesRead
|
if (::android::uptimeNanos() >= deadlineTimeNs) break;
|
||||||
<< " , read failure attempts = " << attempts << ", sleeping for "
|
if (framesRead <= 0) {
|
||||||
<< delayUs << " us";
|
LOG(VERBOSE) << __func__ << ": read returned " << framesRead
|
||||||
usleep(delayUs);
|
<< ", read failure, sleeping for " << kReadAttemptSleepUs << " us";
|
||||||
|
usleep(kReadAttemptSleepUs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// done using the source
|
if (actuallyRead < frameCount) {
|
||||||
source.clear();
|
LOG(WARNING) << __func__ << ": read " << actuallyRead << " vs. requested " << frameCount;
|
||||||
|
|
||||||
if (remainingFrames > 0) {
|
|
||||||
const size_t remainingBytes = remainingFrames * mStreamConfig.frameSize;
|
|
||||||
LOG(VERBOSE) << __func__ << ": clearing remaining_frames = " << remainingFrames;
|
|
||||||
memset(((char*)buffer) + (mStreamConfig.frameSize * frameCount) - remainingBytes, 0,
|
|
||||||
remainingBytes);
|
|
||||||
}
|
|
||||||
|
|
||||||
long readCounterFrames = mCurrentRoute->updateReadCounterFrames(frameCount);
|
|
||||||
*actualFrameCount = frameCount;
|
|
||||||
|
|
||||||
// compute how much we need to sleep after reading the data by comparing the wall clock with
|
|
||||||
// the projected time at which we should return.
|
|
||||||
// wall clock after reading from the pipe
|
|
||||||
auto recordDurationUs = std::chrono::duration_cast<std::chrono::microseconds>(
|
|
||||||
std::chrono::steady_clock::now() - mCurrentRoute->getRecordStartTime());
|
|
||||||
|
|
||||||
// readCounterFrames contains the number of frames that have been read since the beginning of
|
|
||||||
// recording (including this call): it's converted to usec and compared to how long we've been
|
|
||||||
// recording for, which gives us how long we must wait to sync the projected recording time, and
|
|
||||||
// the observed recording time.
|
|
||||||
const long projectedVsObservedOffsetUs =
|
|
||||||
getDelayInUsForFrameCount(readCounterFrames) - recordDurationUs.count();
|
|
||||||
|
|
||||||
LOG(VERBOSE) << __func__ << ": record duration " << recordDurationUs.count()
|
|
||||||
<< " us, will wait: " << projectedVsObservedOffsetUs << " us";
|
|
||||||
if (projectedVsObservedOffsetUs > 0) {
|
|
||||||
usleep(projectedVsObservedOffsetUs);
|
|
||||||
}
|
}
|
||||||
|
mCurrentRoute->updateReadCounterFrames(*actualFrameCount);
|
||||||
return ::android::OK;
|
return ::android::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -81,11 +81,6 @@ bool SubmixRoute::shouldBlockWrite() {
|
|||||||
return (mStreamInOpen || (mStreamInStandby && (mReadCounterFrames != 0)));
|
return (mStreamInOpen || (mStreamInStandby && (mReadCounterFrames != 0)));
|
||||||
}
|
}
|
||||||
|
|
||||||
int SubmixRoute::notifyReadError() {
|
|
||||||
std::lock_guard guard(mLock);
|
|
||||||
return ++mReadErrorCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
long SubmixRoute::updateReadCounterFrames(size_t frameCount) {
|
long SubmixRoute::updateReadCounterFrames(size_t frameCount) {
|
||||||
std::lock_guard guard(mLock);
|
std::lock_guard guard(mLock);
|
||||||
mReadCounterFrames += frameCount;
|
mReadCounterFrames += frameCount;
|
||||||
@@ -103,7 +98,6 @@ void SubmixRoute::openStream(bool isInput) {
|
|||||||
}
|
}
|
||||||
mStreamInStandby = true;
|
mStreamInStandby = true;
|
||||||
mReadCounterFrames = 0;
|
mReadCounterFrames = 0;
|
||||||
mReadErrorCount = 0;
|
|
||||||
} else {
|
} else {
|
||||||
mStreamOutOpen = true;
|
mStreamOutOpen = true;
|
||||||
}
|
}
|
||||||
@@ -214,9 +208,6 @@ void SubmixRoute::exitStandby(bool isInput) {
|
|||||||
if (mStreamInStandby || mStreamOutStandbyTransition) {
|
if (mStreamInStandby || mStreamOutStandbyTransition) {
|
||||||
mStreamInStandby = false;
|
mStreamInStandby = false;
|
||||||
mStreamOutStandbyTransition = false;
|
mStreamOutStandbyTransition = false;
|
||||||
// keep track of when we exit input standby (== first read == start "real recording")
|
|
||||||
// or when we start recording silence, and reset projected time
|
|
||||||
mRecordStartTime = std::chrono::steady_clock::now();
|
|
||||||
mReadCounterFrames = 0;
|
mReadCounterFrames = 0;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -16,7 +16,6 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
|
|
||||||
#include <android-base/thread_annotations.h>
|
#include <android-base/thread_annotations.h>
|
||||||
@@ -83,14 +82,6 @@ class SubmixRoute {
|
|||||||
std::lock_guard guard(mLock);
|
std::lock_guard guard(mLock);
|
||||||
return mReadCounterFrames;
|
return mReadCounterFrames;
|
||||||
}
|
}
|
||||||
int getReadErrorCount() {
|
|
||||||
std::lock_guard guard(mLock);
|
|
||||||
return mReadErrorCount;
|
|
||||||
}
|
|
||||||
std::chrono::time_point<std::chrono::steady_clock> getRecordStartTime() {
|
|
||||||
std::lock_guard guard(mLock);
|
|
||||||
return mRecordStartTime;
|
|
||||||
}
|
|
||||||
sp<MonoPipe> getSink() {
|
sp<MonoPipe> getSink() {
|
||||||
std::lock_guard guard(mLock);
|
std::lock_guard guard(mLock);
|
||||||
return mSink;
|
return mSink;
|
||||||
@@ -126,9 +117,6 @@ class SubmixRoute {
|
|||||||
bool mStreamOutStandby GUARDED_BY(mLock) = true;
|
bool mStreamOutStandby GUARDED_BY(mLock) = true;
|
||||||
// how many frames have been requested to be read since standby
|
// how many frames have been requested to be read since standby
|
||||||
long mReadCounterFrames GUARDED_BY(mLock) = 0;
|
long mReadCounterFrames GUARDED_BY(mLock) = 0;
|
||||||
int mReadErrorCount GUARDED_BY(mLock) = 0;
|
|
||||||
// wall clock when recording starts
|
|
||||||
std::chrono::time_point<std::chrono::steady_clock> mRecordStartTime GUARDED_BY(mLock);
|
|
||||||
|
|
||||||
// Pipe variables: they handle the ring buffer that "pipes" audio:
|
// Pipe variables: they handle the ring buffer that "pipes" audio:
|
||||||
// - from the submix virtual audio output == what needs to be played
|
// - from the submix virtual audio output == what needs to be played
|
||||||
|
|||||||
Reference in New Issue
Block a user