Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2020 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
| 11 | #include "audio/channel_send_frame_transformer_delegate.h" |
| 12 | |
| 13 | #include <utility> |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 14 | #include <vector> |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 15 | |
| 16 | namespace webrtc { |
| 17 | namespace { |
| 18 | |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 19 | using IfaceFrameType = TransformableAudioFrameInterface::FrameType; |
| 20 | |
| 21 | IfaceFrameType InternalFrameTypeToInterfaceFrameType( |
| 22 | const AudioFrameType frame_type) { |
| 23 | switch (frame_type) { |
| 24 | case AudioFrameType::kEmptyFrame: |
| 25 | return IfaceFrameType::kEmptyFrame; |
| 26 | case AudioFrameType::kAudioFrameSpeech: |
| 27 | return IfaceFrameType::kAudioFrameSpeech; |
| 28 | case AudioFrameType::kAudioFrameCN: |
| 29 | return IfaceFrameType::kAudioFrameCN; |
| 30 | } |
| 31 | RTC_DCHECK_NOTREACHED(); |
| 32 | return IfaceFrameType::kEmptyFrame; |
| 33 | } |
| 34 | |
| 35 | AudioFrameType InterfaceFrameTypeToInternalFrameType( |
| 36 | const IfaceFrameType frame_type) { |
| 37 | switch (frame_type) { |
| 38 | case IfaceFrameType::kEmptyFrame: |
| 39 | return AudioFrameType::kEmptyFrame; |
| 40 | case IfaceFrameType::kAudioFrameSpeech: |
| 41 | return AudioFrameType::kAudioFrameSpeech; |
| 42 | case IfaceFrameType::kAudioFrameCN: |
| 43 | return AudioFrameType::kAudioFrameCN; |
| 44 | } |
| 45 | RTC_DCHECK_NOTREACHED(); |
| 46 | return AudioFrameType::kEmptyFrame; |
| 47 | } |
| 48 | |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 49 | class TransformableOutgoingAudioFrame |
| 50 | : public TransformableAudioFrameInterface { |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 51 | public: |
Palak Agarwal | 0505115 | 2023-09-20 08:01:18 | [diff] [blame] | 52 | TransformableOutgoingAudioFrame( |
| 53 | AudioFrameType frame_type, |
| 54 | uint8_t payload_type, |
| 55 | uint32_t rtp_timestamp_with_offset, |
| 56 | const uint8_t* payload_data, |
| 57 | size_t payload_size, |
| 58 | absl::optional<uint64_t> absolute_capture_timestamp_ms, |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 59 | uint32_t ssrc, |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 60 | std::vector<uint32_t> csrcs, |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 61 | const std::string& codec_mime_type, |
| 62 | absl::optional<uint16_t> sequence_number) |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 63 | : frame_type_(frame_type), |
| 64 | payload_type_(payload_type), |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 65 | rtp_timestamp_with_offset_(rtp_timestamp_with_offset), |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 66 | payload_(payload_data, payload_size), |
| 67 | absolute_capture_timestamp_ms_(absolute_capture_timestamp_ms), |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 68 | ssrc_(ssrc), |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 69 | csrcs_(std::move(csrcs)), |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 70 | codec_mime_type_(codec_mime_type), |
| 71 | sequence_number_(sequence_number) {} |
Tony Herre | 8fb41a3 | 2021-09-24 12:05:20 | [diff] [blame] | 72 | ~TransformableOutgoingAudioFrame() override = default; |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 73 | rtc::ArrayView<const uint8_t> GetData() const override { return payload_; } |
| 74 | void SetData(rtc::ArrayView<const uint8_t> data) override { |
| 75 | payload_.SetData(data.data(), data.size()); |
| 76 | } |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 77 | uint32_t GetTimestamp() const override { return rtp_timestamp_with_offset_; } |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 78 | uint32_t GetSsrc() const override { return ssrc_; } |
| 79 | |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 80 | IfaceFrameType Type() const override { |
| 81 | return InternalFrameTypeToInterfaceFrameType(frame_type_); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 82 | } |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 83 | |
| 84 | uint8_t GetPayloadType() const override { return payload_type_; } |
Tony Herre | 8fb41a3 | 2021-09-24 12:05:20 | [diff] [blame] | 85 | Direction GetDirection() const override { return Direction::kSender; } |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 86 | std::string GetMimeType() const override { return codec_mime_type_; } |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 87 | |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 88 | rtc::ArrayView<const uint32_t> GetContributingSources() const override { |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 89 | return csrcs_; |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 90 | } |
| 91 | |
| 92 | const absl::optional<uint16_t> SequenceNumber() const override { |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 93 | return sequence_number_; |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 94 | } |
| 95 | |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 96 | void SetRTPTimestamp(uint32_t rtp_timestamp_with_offset) override { |
| 97 | rtp_timestamp_with_offset_ = rtp_timestamp_with_offset; |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 98 | } |
| 99 | |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 100 | absl::optional<uint64_t> AbsoluteCaptureTimestamp() const override { |
| 101 | return absolute_capture_timestamp_ms_; |
| 102 | } |
| 103 | |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 104 | private: |
| 105 | AudioFrameType frame_type_; |
| 106 | uint8_t payload_type_; |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 107 | uint32_t rtp_timestamp_with_offset_; |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 108 | rtc::Buffer payload_; |
Palak Agarwal | 0505115 | 2023-09-20 08:01:18 | [diff] [blame] | 109 | absl::optional<uint64_t> absolute_capture_timestamp_ms_; |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 110 | uint32_t ssrc_; |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 111 | std::vector<uint32_t> csrcs_; |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 112 | std::string codec_mime_type_; |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 113 | absl::optional<uint16_t> sequence_number_; |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 114 | }; |
| 115 | } // namespace |
| 116 | |
| 117 | ChannelSendFrameTransformerDelegate::ChannelSendFrameTransformerDelegate( |
| 118 | SendFrameCallback send_frame_callback, |
| 119 | rtc::scoped_refptr<FrameTransformerInterface> frame_transformer, |
Danil Chapovalov | 0f1b9a9 | 2024-01-17 18:32:15 | [diff] [blame] | 120 | TaskQueueBase* encoder_queue) |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 121 | : send_frame_callback_(send_frame_callback), |
| 122 | frame_transformer_(std::move(frame_transformer)), |
| 123 | encoder_queue_(encoder_queue) {} |
| 124 | |
| 125 | void ChannelSendFrameTransformerDelegate::Init() { |
| 126 | frame_transformer_->RegisterTransformedFrameCallback( |
| 127 | rtc::scoped_refptr<TransformedFrameCallback>(this)); |
| 128 | } |
| 129 | |
| 130 | void ChannelSendFrameTransformerDelegate::Reset() { |
| 131 | frame_transformer_->UnregisterTransformedFrameCallback(); |
| 132 | frame_transformer_ = nullptr; |
| 133 | |
Markus Handell | 6287280 | 2020-07-06 13:15:07 | [diff] [blame] | 134 | MutexLock lock(&send_lock_); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 135 | send_frame_callback_ = SendFrameCallback(); |
| 136 | } |
| 137 | |
| 138 | void ChannelSendFrameTransformerDelegate::Transform( |
| 139 | AudioFrameType frame_type, |
| 140 | uint8_t payload_type, |
| 141 | uint32_t rtp_timestamp, |
| 142 | const uint8_t* payload_data, |
| 143 | size_t payload_size, |
| 144 | int64_t absolute_capture_timestamp_ms, |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 145 | uint32_t ssrc, |
| 146 | const std::string& codec_mimetype) { |
Tony Herre | 6e95605 | 2023-11-16 13:59:54 | [diff] [blame] | 147 | { |
| 148 | MutexLock lock(&send_lock_); |
| 149 | if (short_circuit_) { |
| 150 | send_frame_callback_( |
| 151 | frame_type, payload_type, rtp_timestamp, |
| 152 | rtc::ArrayView<const uint8_t>(payload_data, payload_size), |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 153 | absolute_capture_timestamp_ms, /*csrcs=*/{}); |
Tony Herre | 6e95605 | 2023-11-16 13:59:54 | [diff] [blame] | 154 | return; |
| 155 | } |
| 156 | } |
Tony Herre | 8fb41a3 | 2021-09-24 12:05:20 | [diff] [blame] | 157 | frame_transformer_->Transform( |
| 158 | std::make_unique<TransformableOutgoingAudioFrame>( |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 159 | frame_type, payload_type, rtp_timestamp, payload_data, payload_size, |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 160 | absolute_capture_timestamp_ms, ssrc, |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 161 | /*csrcs=*/std::vector<uint32_t>(), codec_mimetype, |
| 162 | /*sequence_number=*/absl::nullopt)); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 163 | } |
| 164 | |
| 165 | void ChannelSendFrameTransformerDelegate::OnTransformedFrame( |
| 166 | std::unique_ptr<TransformableFrameInterface> frame) { |
Markus Handell | 6287280 | 2020-07-06 13:15:07 | [diff] [blame] | 167 | MutexLock lock(&send_lock_); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 168 | if (!send_frame_callback_) |
| 169 | return; |
Niels Möller | a3361ff | 2022-01-13 13:05:38 | [diff] [blame] | 170 | rtc::scoped_refptr<ChannelSendFrameTransformerDelegate> delegate(this); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 171 | encoder_queue_->PostTask( |
| 172 | [delegate = std::move(delegate), frame = std::move(frame)]() mutable { |
| 173 | delegate->SendFrame(std::move(frame)); |
| 174 | }); |
| 175 | } |
| 176 | |
Tony Herre | 6e95605 | 2023-11-16 13:59:54 | [diff] [blame] | 177 | void ChannelSendFrameTransformerDelegate::StartShortCircuiting() { |
| 178 | MutexLock lock(&send_lock_); |
| 179 | short_circuit_ = true; |
| 180 | } |
| 181 | |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 182 | void ChannelSendFrameTransformerDelegate::SendFrame( |
| 183 | std::unique_ptr<TransformableFrameInterface> frame) const { |
Markus Handell | 6287280 | 2020-07-06 13:15:07 | [diff] [blame] | 184 | MutexLock lock(&send_lock_); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 185 | RTC_DCHECK_RUN_ON(encoder_queue_); |
| 186 | if (!send_frame_callback_) |
| 187 | return; |
Tony Herre | 8fb41a3 | 2021-09-24 12:05:20 | [diff] [blame] | 188 | auto* transformed_frame = |
Palak Agarwal | 14e5d4c | 2023-08-22 14:58:33 | [diff] [blame] | 189 | static_cast<TransformableAudioFrameInterface*>(frame.get()); |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 190 | send_frame_callback_( |
| 191 | InterfaceFrameTypeToInternalFrameType(transformed_frame->Type()), |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 192 | transformed_frame->GetPayloadType(), transformed_frame->GetTimestamp(), |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 193 | transformed_frame->GetData(), |
Palak Agarwal | 14e5d4c | 2023-08-22 14:58:33 | [diff] [blame] | 194 | transformed_frame->AbsoluteCaptureTimestamp() |
| 195 | ? *transformed_frame->AbsoluteCaptureTimestamp() |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 196 | : 0, |
| 197 | transformed_frame->GetContributingSources()); |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 198 | } |
| 199 | |
Tony Herre | 097a4de | 2023-06-19 15:13:08 | [diff] [blame] | 200 | std::unique_ptr<TransformableAudioFrameInterface> CloneSenderAudioFrame( |
Tove Petersson | 1e2d951 | 2023-03-05 11:03:06 | [diff] [blame] | 201 | TransformableAudioFrameInterface* original) { |
Tony Herre | 5f3ac43 | 2023-12-07 11:39:40 | [diff] [blame] | 202 | std::vector<uint32_t> csrcs; |
| 203 | csrcs.assign(original->GetContributingSources().begin(), |
| 204 | original->GetContributingSources().end()); |
Tove Petersson | 1e2d951 | 2023-03-05 11:03:06 | [diff] [blame] | 205 | return std::make_unique<TransformableOutgoingAudioFrame>( |
Tony Herre | fc68f1f | 2023-06-22 13:32:24 | [diff] [blame] | 206 | InterfaceFrameTypeToInternalFrameType(original->Type()), |
| 207 | original->GetPayloadType(), original->GetTimestamp(), |
Tony Herre | 36500ab | 2023-08-29 10:01:32 | [diff] [blame] | 208 | original->GetData().data(), original->GetData().size(), |
Philipp Hancke | d209893 | 2023-11-01 18:18:32 | [diff] [blame] | 209 | original->AbsoluteCaptureTimestamp(), original->GetSsrc(), |
Tony Herre | 7aa7972 | 2024-01-29 07:31:14 | [diff] [blame] | 210 | std::move(csrcs), original->GetMimeType(), original->SequenceNumber()); |
Tove Petersson | 1e2d951 | 2023-03-05 11:03:06 | [diff] [blame] | 211 | } |
| 212 | |
Marina Ciocea | 65674d8 | 2020-03-31 20:41:30 | [diff] [blame] | 213 | } // namespace webrtc |