/* * 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 #define LOG_TAG "AHAL_Stream" #include #include #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 { DriverStub::DriverStub(const StreamContext& context, bool isInput) : mFrameSizeBytes(context.getFrameSize()), mSampleRate(context.getSampleRate()), mIsAsynchronous(!!context.getAsyncCallback()), mIsInput(isInput) {} ::android::status_t DriverStub::init() { usleep(500); return ::android::OK; } ::android::status_t DriverStub::drain(StreamDescriptor::DrainMode) { usleep(500); return ::android::OK; } ::android::status_t DriverStub::flush() { usleep(500); return ::android::OK; } ::android::status_t DriverStub::pause() { usleep(500); return ::android::OK; } ::android::status_t DriverStub::transfer(void* buffer, size_t frameCount, size_t* actualFrameCount, int32_t* latencyMs) { static constexpr float kMicrosPerSecond = MICROS_PER_SECOND; static constexpr float kScaleFactor = .8f; if (mIsAsynchronous) { usleep(500); } else { const size_t delayUs = static_cast( std::roundf(kScaleFactor * frameCount * kMicrosPerSecond / mSampleRate)); usleep(delayUs); } if (mIsInput) { uint8_t* byteBuffer = static_cast(buffer); for (size_t i = 0; i < frameCount * mFrameSizeBytes; ++i) { byteBuffer[i] = std::rand() % 255; } } *actualFrameCount = frameCount; *latencyMs = Module::kLatencyMs; return ::android::OK; } ::android::status_t DriverStub::standby() { usleep(500); return ::android::OK; } ::android::status_t DriverStub::setConnectedDevices( const std::vector& connectedDevices __unused) { usleep(500); return ::android::OK; } // static ndk::ScopedAStatus StreamInStub::createInstance(const SinkMetadata& sinkMetadata, StreamContext&& context, const std::vector& microphones, std::shared_ptr* result) { std::shared_ptr stream = ndk::SharedRefBase::make(sinkMetadata, std::move(context), microphones); if (auto status = initInstance(stream); !status.isOk()) { return status; } *result = std::move(stream); return ndk::ScopedAStatus::ok(); } StreamInStub::StreamInStub(const SinkMetadata& sinkMetadata, StreamContext&& context, const std::vector& microphones) : StreamIn( sinkMetadata, std::move(context), [](const StreamContext& ctx) -> DriverInterface* { return new DriverStub(ctx, true /*isInput*/); }, [](const StreamContext& ctx, DriverInterface* driver) -> StreamWorkerInterface* { // The default worker implementation is used. return new StreamInWorker(ctx, driver); }, microphones) {} // static ndk::ScopedAStatus StreamOutStub::createInstance(const SourceMetadata& sourceMetadata, StreamContext&& context, const std::optional& offloadInfo, std::shared_ptr* result) { std::shared_ptr stream = ndk::SharedRefBase::make( sourceMetadata, std::move(context), offloadInfo); if (auto status = initInstance(stream); !status.isOk()) { return status; } *result = std::move(stream); return ndk::ScopedAStatus::ok(); } StreamOutStub::StreamOutStub(const SourceMetadata& sourceMetadata, StreamContext&& context, const std::optional& offloadInfo) : StreamOut( sourceMetadata, std::move(context), [](const StreamContext& ctx) -> DriverInterface* { return new DriverStub(ctx, false /*isInput*/); }, [](const StreamContext& ctx, DriverInterface* driver) -> StreamWorkerInterface* { // The default worker implementation is used. return new StreamOutWorker(ctx, driver); }, offloadInfo) {} } // namespace aidl::android::hardware::audio::core