123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135 |
- #include "pch.h"
- #include "../common/comm.h"
- #include "video_frame_observer.h"
- constexpr int kBufferAlignment = 64;
- ArgbBuffer::ArgbBuffer(int width, int height, int stride)
- : width_(width),
- height_(height),
- stride_(stride),
- data_(static_cast<uint8_t*>(
- webrtc::AlignedMalloc(ArgbDataSize(height, stride),
- kBufferAlignment))) {
- RTC_DCHECK_GT(width, 0);
- RTC_DCHECK_GT(height, 0);
- RTC_DCHECK_GE(stride, 4 * width);
- }
- rtc::scoped_refptr<webrtc::I420BufferInterface> ArgbBuffer::ToI420()
- {
- //创建了一个 webrtc::I420Buffer实例i420_buffer,并调用其静态方法 Create初始化一个I420格式的缓冲区
- rtc::scoped_refptr<webrtc::I420Buffer> i420_buffer =
- webrtc::I420Buffer::Create(width_, height_, stride_, stride_ / 2,
- stride_ / 2);
-
- //libyuv库中的函数,用于将ARGB格式的数据转换为I420格式
- libyuv::ARGBToI420(Data(), Stride(), i420_buffer->MutableDataY(),
- i420_buffer->StrideY(), i420_buffer->MutableDataU(),
- i420_buffer->StrideU(), i420_buffer->MutableDataV(),
- i420_buffer->StrideV(), width_, height_);
- return i420_buffer;
- }
- void VideoFrameObserver::SetCallback(I420FrameReadyCallback callback) {
- std::lock_guard<std::mutex> lock{ mutex_ };
- i420_callback_ = std::move(callback);
- }
- void VideoFrameObserver::SetCallback(ARGBFrameReadyCallback callback)
- {
- std::lock_guard<std::mutex> lock{ mutex_ };
- argb_callback_ = std::move(callback);
- }
- ArgbBuffer* VideoFrameObserver::GetArgbScratchBuffer(int width, int height)
- {
- //计算了所需的图像大小,并将结果保存在 needed_size 变量中
- const size_t needed_size = ArgbDataSize(width, height);
- //尝试从 argb_scratch_buffer_ 中获取已有的缓冲区,并将其存储在 buffer 变量中
- if (auto* buffer = argb_scratch_buffer_.get())
- {
- //检查已有的缓冲区是否足够大以容纳所需的图像数据
- if (buffer->Size() >= needed_size)
- {
- return buffer;
- }
- }
- //创建一个新的足够大的ArgbBuffer对象,并将其存储在argb_scratch_buffer_中
- argb_scratch_buffer_ = ArgbBuffer::Create(width, height);
- return argb_scratch_buffer_.get();
- }
- void VideoFrameObserver::OnFrame(const webrtc::VideoFrame& frame) {
-
- std::lock_guard<std::mutex> lock{ mutex_ };
- if (!i420_callback_ && !argb_callback_)
- return;
- //从 frame(一个 webrtc::VideoFrame 对象)中调用 video_frame_buffer() 方法获取视频帧缓冲区
- //将该缓冲区存储在一个 rtc::scoped_refptr<webrtc::VideoFrameBuffer> 类型的变量 buffer 中,以便自动管理其生命周期
- rtc::scoped_refptr<webrtc::VideoFrameBuffer> buffer(
- frame.video_frame_buffer());
- //获取当前视频帧的宽度和高度
- const int width = frame.width();
- const int height = frame.height();
- //检查缓冲区的类型是否为 kI420A(即带有 alpha 通道的 I420 格式)
- if (buffer->type() != webrtc::VideoFrameBuffer::Type::kI420A)
- {
- // The buffer is not encoded in I420 with alpha channel; use I420 without
- // alpha channel as interchange format for the callback, and convert the
- // buffer to that (or do nothing if already in I420).
- //如果不是,将缓冲区转换为 I420 格式,获取 Y、U、V 平面的数据指针,由于没有 alpha 通道,aptr 被设为 nullptr
- rtc::scoped_refptr<webrtc::I420BufferInterface> i420_buffer =
- buffer->ToI420();
- const uint8_t* yptr = i420_buffer->DataY();
- const uint8_t* uptr = i420_buffer->DataU();
- const uint8_t* vptr = i420_buffer->DataV();
- const uint8_t* aptr = nullptr;
- // if (i420_callback_) {
- // i420_callback_(yptr, uptr, vptr, aptr, i420_buffer->StrideY(),
- // i420_buffer->StrideU(), i420_buffer->StrideV(), 0, width,
- // height);
- // }
- //如果存在 argb_callback_ 回调函数,调用该函数
- if (argb_callback_)
- {
- // ArgbBuffer* const argb_buffer = GetArgbScratchBuffer(width, height);
- // libyuv::I420ToARGB(yptr, i420_buffer->StrideY(), uptr,
- // i420_buffer->StrideU(), vptr, i420_buffer->StrideV(),
- // argb_buffer->Data(), argb_buffer->Stride(), width,
- // height);
- argb_callback_(yptr,i420_buffer->StrideY(),uptr,i420_buffer->StrideU(),vptr,i420_buffer->StrideV(), width*4, width, height);
- }
- }
- //如果缓冲区类型是 kI420A,则直接处理带有 alpha 通道的 I420 缓冲区
- else
- {
- // The buffer is encoded in I420 with alpha channel, use it directly.
- const webrtc::I420ABufferInterface* i420a_buffer = buffer->GetI420A();
- const uint8_t* yptr = i420a_buffer->DataY();
- const uint8_t* uptr = i420a_buffer->DataU();
- const uint8_t* vptr = i420a_buffer->DataV();
- const uint8_t* aptr = i420a_buffer->DataA();
- // if (i420_callback_) {
- // i420_callback_(yptr, uptr, vptr, aptr, i420a_buffer->StrideY(),
- // i420a_buffer->StrideU(), i420a_buffer->StrideV(),
- // i420a_buffer->StrideA(), width, height);
- // }
- if (argb_callback_)
- {
- // ArgbBuffer* const argb_buffer = GetArgbScratchBuffer(width, height);
- // libyuv::I420AlphaToARGB(
- // yptr, i420a_buffer->StrideY(), uptr, i420a_buffer->StrideU(), vptr,
- // i420a_buffer->StrideV(), aptr, i420a_buffer->StrideA(),
- // argb_buffer->Data(), argb_buffer->Stride(), width, height, 0);
- argb_callback_(yptr, i420a_buffer->StrideY(), uptr, i420a_buffer->StrideU(), vptr, i420a_buffer->StrideV(), width * 4, width, height);
- }
- }
- }
|