mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-13 22:00:47 +01:00

Bug: webrtc:342905193 No-Try: True Change-Id: Icc968be43b8830038ea9a1f5f604307220457807 Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/361021 Auto-Submit: Florent Castelli <orphis@webrtc.org> Reviewed-by: Harald Alvestrand <hta@webrtc.org> Commit-Queue: Florent Castelli <orphis@webrtc.org> Cr-Commit-Position: refs/heads/main@{#42911}
226 lines
7.8 KiB
C++
226 lines
7.8 KiB
C++
/*
|
|
* Copyright (c) 2020 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "audio/channel_send_frame_transformer_delegate.h"
|
|
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
|
|
using IfaceFrameType = TransformableAudioFrameInterface::FrameType;
|
|
|
|
IfaceFrameType InternalFrameTypeToInterfaceFrameType(
|
|
const AudioFrameType frame_type) {
|
|
switch (frame_type) {
|
|
case AudioFrameType::kEmptyFrame:
|
|
return IfaceFrameType::kEmptyFrame;
|
|
case AudioFrameType::kAudioFrameSpeech:
|
|
return IfaceFrameType::kAudioFrameSpeech;
|
|
case AudioFrameType::kAudioFrameCN:
|
|
return IfaceFrameType::kAudioFrameCN;
|
|
}
|
|
RTC_DCHECK_NOTREACHED();
|
|
return IfaceFrameType::kEmptyFrame;
|
|
}
|
|
|
|
AudioFrameType InterfaceFrameTypeToInternalFrameType(
|
|
const IfaceFrameType frame_type) {
|
|
switch (frame_type) {
|
|
case IfaceFrameType::kEmptyFrame:
|
|
return AudioFrameType::kEmptyFrame;
|
|
case IfaceFrameType::kAudioFrameSpeech:
|
|
return AudioFrameType::kAudioFrameSpeech;
|
|
case IfaceFrameType::kAudioFrameCN:
|
|
return AudioFrameType::kAudioFrameCN;
|
|
}
|
|
RTC_DCHECK_NOTREACHED();
|
|
return AudioFrameType::kEmptyFrame;
|
|
}
|
|
} // namespace
|
|
|
|
class TransformableOutgoingAudioFrame
|
|
: public TransformableAudioFrameInterface {
|
|
public:
|
|
TransformableOutgoingAudioFrame(
|
|
AudioFrameType frame_type,
|
|
uint8_t payload_type,
|
|
uint32_t rtp_timestamp_with_offset,
|
|
const uint8_t* payload_data,
|
|
size_t payload_size,
|
|
std::optional<uint64_t> absolute_capture_timestamp_ms,
|
|
uint32_t ssrc,
|
|
std::vector<uint32_t> csrcs,
|
|
const std::string& codec_mime_type,
|
|
std::optional<uint16_t> sequence_number,
|
|
std::optional<uint8_t> audio_level_dbov)
|
|
: TransformableAudioFrameInterface(Passkey()),
|
|
frame_type_(frame_type),
|
|
payload_type_(payload_type),
|
|
rtp_timestamp_with_offset_(rtp_timestamp_with_offset),
|
|
payload_(payload_data, payload_size),
|
|
absolute_capture_timestamp_ms_(absolute_capture_timestamp_ms),
|
|
ssrc_(ssrc),
|
|
csrcs_(std::move(csrcs)),
|
|
codec_mime_type_(codec_mime_type),
|
|
sequence_number_(sequence_number),
|
|
audio_level_dbov_(audio_level_dbov) {}
|
|
~TransformableOutgoingAudioFrame() override = default;
|
|
rtc::ArrayView<const uint8_t> GetData() const override { return payload_; }
|
|
void SetData(rtc::ArrayView<const uint8_t> data) override {
|
|
payload_.SetData(data.data(), data.size());
|
|
}
|
|
uint32_t GetTimestamp() const override { return rtp_timestamp_with_offset_; }
|
|
uint32_t GetSsrc() const override { return ssrc_; }
|
|
|
|
IfaceFrameType Type() const override {
|
|
return InternalFrameTypeToInterfaceFrameType(frame_type_);
|
|
}
|
|
|
|
uint8_t GetPayloadType() const override { return payload_type_; }
|
|
Direction GetDirection() const override { return Direction::kSender; }
|
|
std::string GetMimeType() const override { return codec_mime_type_; }
|
|
|
|
rtc::ArrayView<const uint32_t> GetContributingSources() const override {
|
|
return csrcs_;
|
|
}
|
|
|
|
const std::optional<uint16_t> SequenceNumber() const override {
|
|
return sequence_number_;
|
|
}
|
|
|
|
void SetRTPTimestamp(uint32_t rtp_timestamp_with_offset) override {
|
|
rtp_timestamp_with_offset_ = rtp_timestamp_with_offset;
|
|
}
|
|
|
|
std::optional<uint64_t> AbsoluteCaptureTimestamp() const override {
|
|
return absolute_capture_timestamp_ms_;
|
|
}
|
|
|
|
std::optional<uint8_t> AudioLevel() const override {
|
|
return audio_level_dbov_;
|
|
}
|
|
|
|
std::optional<Timestamp> ReceiveTime() const override { return std::nullopt; }
|
|
|
|
private:
|
|
AudioFrameType frame_type_;
|
|
uint8_t payload_type_;
|
|
uint32_t rtp_timestamp_with_offset_;
|
|
rtc::Buffer payload_;
|
|
std::optional<uint64_t> absolute_capture_timestamp_ms_;
|
|
uint32_t ssrc_;
|
|
std::vector<uint32_t> csrcs_;
|
|
std::string codec_mime_type_;
|
|
std::optional<uint16_t> sequence_number_;
|
|
std::optional<uint8_t> audio_level_dbov_;
|
|
};
|
|
|
|
ChannelSendFrameTransformerDelegate::ChannelSendFrameTransformerDelegate(
|
|
SendFrameCallback send_frame_callback,
|
|
rtc::scoped_refptr<FrameTransformerInterface> frame_transformer,
|
|
TaskQueueBase* encoder_queue)
|
|
: send_frame_callback_(send_frame_callback),
|
|
frame_transformer_(std::move(frame_transformer)),
|
|
encoder_queue_(encoder_queue) {}
|
|
|
|
void ChannelSendFrameTransformerDelegate::Init() {
|
|
frame_transformer_->RegisterTransformedFrameCallback(
|
|
rtc::scoped_refptr<TransformedFrameCallback>(this));
|
|
}
|
|
|
|
void ChannelSendFrameTransformerDelegate::Reset() {
|
|
frame_transformer_->UnregisterTransformedFrameCallback();
|
|
frame_transformer_ = nullptr;
|
|
|
|
MutexLock lock(&send_lock_);
|
|
send_frame_callback_ = SendFrameCallback();
|
|
}
|
|
|
|
void ChannelSendFrameTransformerDelegate::Transform(
|
|
AudioFrameType frame_type,
|
|
uint8_t payload_type,
|
|
uint32_t rtp_timestamp,
|
|
const uint8_t* payload_data,
|
|
size_t payload_size,
|
|
int64_t absolute_capture_timestamp_ms,
|
|
uint32_t ssrc,
|
|
const std::string& codec_mimetype,
|
|
std::optional<uint8_t> audio_level_dbov) {
|
|
{
|
|
MutexLock lock(&send_lock_);
|
|
if (short_circuit_) {
|
|
send_frame_callback_(
|
|
frame_type, payload_type, rtp_timestamp,
|
|
rtc::ArrayView<const uint8_t>(payload_data, payload_size),
|
|
absolute_capture_timestamp_ms, /*csrcs=*/{}, audio_level_dbov);
|
|
return;
|
|
}
|
|
}
|
|
frame_transformer_->Transform(
|
|
std::make_unique<TransformableOutgoingAudioFrame>(
|
|
frame_type, payload_type, rtp_timestamp, payload_data, payload_size,
|
|
absolute_capture_timestamp_ms, ssrc,
|
|
/*csrcs=*/std::vector<uint32_t>(), codec_mimetype,
|
|
/*sequence_number=*/std::nullopt, audio_level_dbov));
|
|
}
|
|
|
|
void ChannelSendFrameTransformerDelegate::OnTransformedFrame(
|
|
std::unique_ptr<TransformableFrameInterface> frame) {
|
|
MutexLock lock(&send_lock_);
|
|
if (!send_frame_callback_)
|
|
return;
|
|
rtc::scoped_refptr<ChannelSendFrameTransformerDelegate> delegate(this);
|
|
encoder_queue_->PostTask(
|
|
[delegate = std::move(delegate), frame = std::move(frame)]() mutable {
|
|
delegate->SendFrame(std::move(frame));
|
|
});
|
|
}
|
|
|
|
void ChannelSendFrameTransformerDelegate::StartShortCircuiting() {
|
|
MutexLock lock(&send_lock_);
|
|
short_circuit_ = true;
|
|
}
|
|
|
|
void ChannelSendFrameTransformerDelegate::SendFrame(
|
|
std::unique_ptr<TransformableFrameInterface> frame) const {
|
|
MutexLock lock(&send_lock_);
|
|
RTC_DCHECK_RUN_ON(encoder_queue_);
|
|
if (!send_frame_callback_)
|
|
return;
|
|
auto* transformed_frame =
|
|
static_cast<TransformableAudioFrameInterface*>(frame.get());
|
|
send_frame_callback_(
|
|
InterfaceFrameTypeToInternalFrameType(transformed_frame->Type()),
|
|
transformed_frame->GetPayloadType(), transformed_frame->GetTimestamp(),
|
|
transformed_frame->GetData(),
|
|
transformed_frame->AbsoluteCaptureTimestamp()
|
|
? *transformed_frame->AbsoluteCaptureTimestamp()
|
|
: 0,
|
|
transformed_frame->GetContributingSources(),
|
|
transformed_frame->AudioLevel());
|
|
}
|
|
|
|
std::unique_ptr<TransformableAudioFrameInterface> CloneSenderAudioFrame(
|
|
TransformableAudioFrameInterface* original) {
|
|
std::vector<uint32_t> csrcs;
|
|
csrcs.assign(original->GetContributingSources().begin(),
|
|
original->GetContributingSources().end());
|
|
return std::make_unique<TransformableOutgoingAudioFrame>(
|
|
InterfaceFrameTypeToInternalFrameType(original->Type()),
|
|
original->GetPayloadType(), original->GetTimestamp(),
|
|
original->GetData().data(), original->GetData().size(),
|
|
original->AbsoluteCaptureTimestamp(), original->GetSsrc(),
|
|
std::move(csrcs), original->GetMimeType(), original->SequenceNumber(),
|
|
original->AudioLevel());
|
|
}
|
|
|
|
} // namespace webrtc
|