blob: 52d66a150fff13b78fbc24b410c4b5445e0c58fc [file] [log] [blame]
Phil Burk062e67a2015-02-11 13:40:50 -08001/*
Dean Wheatley6c009512023-10-23 09:34:14 +11002 *
3 * Copyright 2015, 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 */
Phil Burk062e67a2015-02-11 13:40:50 -080017
18#define LOG_TAG "AudioFlinger"
19//#define LOG_NDEBUG 0
Dean Wheatleyace1eeb2023-11-06 21:58:14 +110020#include "Configuration.h"
Mikhail Naganovcbc8f612016-10-11 18:05:13 -070021#include <system/audio.h>
Phil Burk062e67a2015-02-11 13:40:50 -080022#include <utils/Log.h>
23
24#include <audio_utils/spdif/SPDIFEncoder.h>
25
26#include "AudioHwDevice.h"
Phil Burk062e67a2015-02-11 13:40:50 -080027#include "SpdifStreamOut.h"
28
29namespace android {
30
31/**
32 * If the AudioFlinger is processing encoded data and the HAL expects
33 * PCM then we need to wrap the data in an SPDIF wrapper.
34 */
Phil Burk23d89972015-04-06 16:22:23 -070035SpdifStreamOut::SpdifStreamOut(AudioHwDevice *dev,
Phil Burk23d89972015-04-06 16:22:23 -070036 audio_format_t format)
Dean Wheatleydfb67b82024-01-23 09:36:29 +110037 : AudioStreamOut(dev)
Phil Burk23d89972015-04-06 16:22:23 -070038 , mSpdifEncoder(this, format)
Phil Burk062e67a2015-02-11 13:40:50 -080039{
40}
41
42status_t SpdifStreamOut::open(
Dean Wheatley6c009512023-10-23 09:34:14 +110043 audio_io_handle_t handle,
44 audio_devices_t devices,
45 struct audio_config *config,
Dean Wheatleydfb67b82024-01-23 09:36:29 +110046 audio_output_flags_t *flags,
Haofan Wangb75aa6a2024-07-09 23:06:58 -070047 const char *address,
48 const std::vector<playback_track_metadata_v7_t>& sourceMetadata)
Phil Burk062e67a2015-02-11 13:40:50 -080049{
50 struct audio_config customConfig = *config;
51
Dean Wheatley60e87222023-11-15 20:56:14 +110052 mApplicationConfig.format = config->format;
53 mApplicationConfig.sample_rate = config->sample_rate;
54 mApplicationConfig.channel_mask = config->channel_mask;
Phil Burkca5e6142015-07-14 09:42:29 -070055
Dean Wheatley6c009512023-10-23 09:34:14 +110056 mRateMultiplier = spdif_rate_multiplier(config->format);
57 if (mRateMultiplier <= 0) {
58 ALOGE("ERROR SpdifStreamOut::open() unrecognized format 0x%08X\n", config->format);
59 return BAD_VALUE;
Phil Burk062e67a2015-02-11 13:40:50 -080060 }
61 customConfig.sample_rate = config->sample_rate * mRateMultiplier;
62
Phil Burk23d89972015-04-06 16:22:23 -070063 customConfig.format = AUDIO_FORMAT_PCM_16_BIT;
64 customConfig.channel_mask = AUDIO_CHANNEL_OUT_STEREO;
Dean Wheatleydfb67b82024-01-23 09:36:29 +110065 // Tell the HAL that the data will be compressed audio wrapped in a data burst.
66 *flags = (audio_output_flags_t)(*flags | AUDIO_OUTPUT_FLAG_IEC958_NONAUDIO);
Phil Burk23d89972015-04-06 16:22:23 -070067
Phil Burk062e67a2015-02-11 13:40:50 -080068 // Always print this because otherwise it could be very confusing if the
69 // HAL and AudioFlinger are using different formats.
70 // Print before open() because HAL may modify customConfig.
Dean Wheatley6c009512023-10-23 09:34:14 +110071 ALOGI("SpdifStreamOut::open() AudioFlinger requested sampleRate %d, format %#x,"
72 " channelMask %#x", config->sample_rate, config->format, config->channel_mask);
73 ALOGI("SpdifStreamOut::open() HAL configured for sampleRate %d, format %#x, channelMask %#x",
74 customConfig.sample_rate, customConfig.format, customConfig.channel_mask);
Phil Burk062e67a2015-02-11 13:40:50 -080075
Andy Hung1ef77382023-06-15 14:50:18 -070076 const status_t status = AudioStreamOut::open(
Phil Burk062e67a2015-02-11 13:40:50 -080077 handle,
78 devices,
79 &customConfig,
Dean Wheatleydfb67b82024-01-23 09:36:29 +110080 flags,
Haofan Wangb75aa6a2024-07-09 23:06:58 -070081 address,
82 sourceMetadata);
Phil Burk062e67a2015-02-11 13:40:50 -080083
84 ALOGI("SpdifStreamOut::open() status = %d", status);
85
Dean Wheatleyace1eeb2023-11-06 21:58:14 +110086#ifdef TEE_SINK
87 if (status == OK) {
88 // Don't use PCM 16-bit format to avoid WAV encoding IEC61937 data.
89 mTee.set(customConfig.sample_rate,
90 audio_channel_count_from_out_mask(customConfig.channel_mask),
91 AUDIO_FORMAT_IEC61937, NBAIO_Tee::TEE_FLAG_OUTPUT_THREAD);
92 mTee.setId(std::string("_") + std::to_string(handle) + "_D");
93 }
94#endif
95
Phil Burk062e67a2015-02-11 13:40:50 -080096 return status;
97}
98
Phil Burk062e67a2015-02-11 13:40:50 -080099int SpdifStreamOut::flush()
100{
Phil Burk48e6ea92015-06-18 15:37:08 -0700101 mSpdifEncoder.reset();
Phil Burk062e67a2015-02-11 13:40:50 -0800102 return AudioStreamOut::flush();
103}
104
105int SpdifStreamOut::standby()
106{
Phil Burk48e6ea92015-06-18 15:37:08 -0700107 mSpdifEncoder.reset();
Phil Burk062e67a2015-02-11 13:40:50 -0800108 return AudioStreamOut::standby();
109}
110
Phil Burk062e67a2015-02-11 13:40:50 -0800111ssize_t SpdifStreamOut::writeDataBurst(const void* buffer, size_t bytes)
112{
Dean Wheatleyace1eeb2023-11-06 21:58:14 +1100113 const ssize_t written = AudioStreamOut::write(buffer, bytes);
114
115#ifdef TEE_SINK
116 if (written > 0) {
117 mTee.write(reinterpret_cast<const char *>(buffer),
118 written / AudioStreamOut::getFrameSize());
119 }
120#endif
121 return written;
Phil Burk062e67a2015-02-11 13:40:50 -0800122}
123
Phil Burkca5e6142015-07-14 09:42:29 -0700124ssize_t SpdifStreamOut::write(const void* buffer, size_t numBytes)
Phil Burk062e67a2015-02-11 13:40:50 -0800125{
126 // Write to SPDIF wrapper. It will call back to writeDataBurst().
Phil Burkca5e6142015-07-14 09:42:29 -0700127 return mSpdifEncoder.write(buffer, numBytes);
Phil Burk062e67a2015-02-11 13:40:50 -0800128}
129
130} // namespace android