summaryrefslogtreecommitdiff
path: root/bluetooth/audio/aidl/default/LeAudioSoftwareAudioProvider.cpp
blob: f9962fd9d929a1b0f7bcc3e6e0b1333e556de336 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
/*
 * Copyright (C) 2022 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_TAG "BTAudioProviderLeAudioHW"

#include "LeAudioSoftwareAudioProvider.h"

#include <BluetoothAudioCodecs.h>
#include <BluetoothAudioSessionReport.h>
#include <android-base/logging.h>

#include <cstdint>

namespace aidl {
namespace android {
namespace hardware {
namespace bluetooth {
namespace audio {

static constexpr uint32_t kBufferOutCount = 2;  // two frame buffer
static constexpr uint32_t kBufferInCount = 2;   // two frame buffer

inline uint32_t channel_mode_to_channel_count(ChannelMode channel_mode) {
  switch (channel_mode) {
    case ChannelMode::MONO:
      return 1;
    case ChannelMode::STEREO:
      return 2;
    default:
      return 0;
  }
  return 0;
}

LeAudioSoftwareOutputAudioProvider::LeAudioSoftwareOutputAudioProvider()
    : LeAudioSoftwareAudioProvider() {
  session_type_ = SessionType::LE_AUDIO_SOFTWARE_ENCODING_DATAPATH;
}

LeAudioSoftwareInputAudioProvider::LeAudioSoftwareInputAudioProvider()
    : LeAudioSoftwareAudioProvider() {
  session_type_ = SessionType::LE_AUDIO_SOFTWARE_DECODING_DATAPATH;
}

LeAudioSoftwareAudioProvider::LeAudioSoftwareAudioProvider()
    : BluetoothAudioProvider(), data_mq_(nullptr) {}

bool LeAudioSoftwareAudioProvider::isValid(const SessionType& sessionType) {
  return (sessionType == session_type_);
}

ndk::ScopedAStatus LeAudioSoftwareAudioProvider::startSession(
    const std::shared_ptr<IBluetoothAudioPort>& host_if,
    const AudioConfiguration& audio_config, DataMQDesc* _aidl_return) {
  if (audio_config.getTag() != AudioConfiguration::pcmConfig) {
    LOG(WARNING) << __func__ << " - Invalid Audio Configuration="
                 << audio_config.toString();
    return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
  }
  const auto& pcm_config = audio_config.get<AudioConfiguration::pcmConfig>();
  if (!BluetoothAudioCodecs::IsSoftwarePcmConfigurationValid(pcm_config)) {
    LOG(WARNING) << __func__ << " - Unsupported PCM Configuration="
                 << pcm_config.toString();
    return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
  }

  uint32_t buffer_modifier = 0;
  if (session_type_ == SessionType::LE_AUDIO_SOFTWARE_ENCODING_DATAPATH)
    buffer_modifier = kBufferOutCount;
  else if (session_type_ == SessionType::LE_AUDIO_SOFTWARE_DECODING_DATAPATH)
    buffer_modifier = kBufferInCount;

  uint32_t data_mq_size =
      (ceil(pcm_config.sampleRateHz) / 1000) *
      channel_mode_to_channel_count(pcm_config.channelMode) *
      (pcm_config.bitsPerSample / 8) * (pcm_config.dataIntervalUs / 1000) *
      buffer_modifier;

  LOG(INFO) << __func__ << " - size of audio buffer " << data_mq_size
            << " byte(s)";

  std::unique_ptr<DataMQ> temp_data_mq(
      new DataMQ(data_mq_size, /* EventFlag */ true));
  if (temp_data_mq == nullptr || !temp_data_mq->isValid()) {
    ALOGE_IF(!temp_data_mq, "failed to allocate data MQ");
    ALOGE_IF(temp_data_mq && !temp_data_mq->isValid(), "data MQ is invalid");
    *_aidl_return = DataMQDesc();
    return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
  }
  data_mq_ = std::move(temp_data_mq);

  return BluetoothAudioProvider::startSession(host_if, audio_config,
                                              _aidl_return);
}

ndk::ScopedAStatus LeAudioSoftwareAudioProvider::onSessionReady(
    DataMQDesc* _aidl_return) {
  if (data_mq_ == nullptr || !data_mq_->isValid()) {
    *_aidl_return = DataMQDesc();
    return ndk::ScopedAStatus::fromExceptionCode(EX_ILLEGAL_ARGUMENT);
  }
  *_aidl_return = data_mq_->dupeDesc();
  BluetoothAudioSessionReport::OnSessionStarted(session_type_, stack_iface_,
                                                _aidl_return, *audio_config_);
  return ndk::ScopedAStatus::ok();
}

}  // namespace audio
}  // namespace bluetooth
}  // namespace hardware
}  // namespace android
}  // namespace aidl