mirror of
https://github.com/Evolution-X/hardware_interfaces
synced 2026-02-01 11:36:00 +00:00
Stub output stream is like a /dev/null audio device, however
for synchronous transfers it must block for the duration of
the audio chunk being transferred. Implement more accurate
accounting for the blocking time using the same approach as
employed by the remote submix stream implementation.
Bug: 356719263
Test: atest CtsNativeMediaAAudioTestCases
on aosp_cf_x86_64_auto target
Change-Id: I137aed397246bcf2b5ef6789aa4d2d27ead64467
148 lines
5.1 KiB
C++
148 lines
5.1 KiB
C++
/*
|
|
* Copyright (C) 2023 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.
|
|
*/
|
|
|
|
#include <cmath>
|
|
|
|
#define LOG_TAG "AHAL_Stream"
|
|
#include <android-base/logging.h>
|
|
#include <audio_utils/clock.h>
|
|
|
|
#include "core-impl/Module.h"
|
|
#include "core-impl/StreamStub.h"
|
|
|
|
using aidl::android::hardware::audio::common::SinkMetadata;
|
|
using aidl::android::hardware::audio::common::SourceMetadata;
|
|
using aidl::android::media::audio::common::AudioDevice;
|
|
using aidl::android::media::audio::common::AudioOffloadInfo;
|
|
using aidl::android::media::audio::common::MicrophoneInfo;
|
|
|
|
namespace aidl::android::hardware::audio::core {
|
|
|
|
StreamStub::StreamStub(StreamContext* context, const Metadata& metadata)
|
|
: StreamCommonImpl(context, metadata),
|
|
mBufferSizeFrames(getContext().getBufferSizeInFrames()),
|
|
mFrameSizeBytes(getContext().getFrameSize()),
|
|
mSampleRate(getContext().getSampleRate()),
|
|
mIsAsynchronous(!!getContext().getAsyncCallback()),
|
|
mIsInput(isInput(metadata)) {}
|
|
|
|
StreamStub::~StreamStub() {
|
|
cleanupWorker();
|
|
}
|
|
|
|
::android::status_t StreamStub::init() {
|
|
mIsInitialized = true;
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::drain(StreamDescriptor::DrainMode) {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
if (!mIsInput) {
|
|
if (!mIsAsynchronous) {
|
|
static constexpr float kMicrosPerSecond = MICROS_PER_SECOND;
|
|
const size_t delayUs = static_cast<size_t>(
|
|
std::roundf(mBufferSizeFrames * kMicrosPerSecond / mSampleRate));
|
|
usleep(delayUs);
|
|
} else {
|
|
usleep(500);
|
|
}
|
|
}
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::flush() {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::pause() {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::standby() {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
mIsStandby = true;
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::start() {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
mIsStandby = false;
|
|
mStartTimeNs = ::android::uptimeNanos();
|
|
mFramesSinceStart = 0;
|
|
return ::android::OK;
|
|
}
|
|
|
|
::android::status_t StreamStub::transfer(void* buffer, size_t frameCount, size_t* actualFrameCount,
|
|
int32_t*) {
|
|
if (!mIsInitialized) {
|
|
LOG(FATAL) << __func__ << ": must not happen for an uninitialized driver";
|
|
}
|
|
if (mIsStandby) {
|
|
LOG(FATAL) << __func__ << ": must not happen while in standby";
|
|
}
|
|
*actualFrameCount = frameCount;
|
|
if (mIsAsynchronous) {
|
|
usleep(500);
|
|
} else {
|
|
mFramesSinceStart += *actualFrameCount;
|
|
const long bufferDurationUs =
|
|
(*actualFrameCount) * MICROS_PER_SECOND / mContext.getSampleRate();
|
|
const auto totalDurationUs =
|
|
(::android::uptimeNanos() - mStartTimeNs) / NANOS_PER_MICROSECOND;
|
|
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);
|
|
}
|
|
}
|
|
if (mIsInput) {
|
|
uint8_t* byteBuffer = static_cast<uint8_t*>(buffer);
|
|
for (size_t i = 0; i < frameCount * mFrameSizeBytes; ++i) {
|
|
byteBuffer[i] = std::rand() % 255;
|
|
}
|
|
}
|
|
return ::android::OK;
|
|
}
|
|
|
|
void StreamStub::shutdown() {
|
|
mIsInitialized = false;
|
|
}
|
|
|
|
StreamInStub::StreamInStub(StreamContext&& context, const SinkMetadata& sinkMetadata,
|
|
const std::vector<MicrophoneInfo>& microphones)
|
|
: StreamIn(std::move(context), microphones), StreamStub(&mContextInstance, sinkMetadata) {}
|
|
|
|
StreamOutStub::StreamOutStub(StreamContext&& context, const SourceMetadata& sourceMetadata,
|
|
const std::optional<AudioOffloadInfo>& offloadInfo)
|
|
: StreamOut(std::move(context), offloadInfo), StreamStub(&mContextInstance, sourceMetadata) {}
|
|
|
|
} // namespace aidl::android::hardware::audio::core
|