1 // Copyright 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "content/common/gpu/media/gpu_video_encode_accelerator.h"
7 #include "base/callback.h"
8 #include "base/command_line.h"
9 #include "base/logging.h"
10 #include "base/memory/shared_memory.h"
11 #include "base/numerics/safe_math.h"
12 #include "base/sys_info.h"
13 #include "build/build_config.h"
14 #include "content/common/gpu/gpu_channel.h"
15 #include "content/common/gpu/gpu_messages.h"
16 #include "content/common/gpu/media/gpu_video_accelerator_util.h"
17 #include "content/public/common/content_switches.h"
18 #include "ipc/ipc_message_macros.h"
19 #include "media/base/bind_to_current_loop.h"
20 #include "media/base/limits.h"
21 #include "media/base/video_frame.h"
23 #if defined(OS_CHROMEOS)
24 #if defined(USE_V4L2_CODEC)
25 #include "content/common/gpu/media/v4l2_video_encode_accelerator.h"
27 #if defined(ARCH_CPU_X86_FAMILY)
28 #include "content/common/gpu/media/vaapi_video_encode_accelerator.h"
30 #elif defined(OS_ANDROID) && defined(ENABLE_WEBRTC)
31 #include "content/common/gpu/media/android_video_encode_accelerator.h"
36 static bool MakeDecoderContextCurrent(
37 const base::WeakPtr
<GpuCommandBufferStub
> stub
) {
39 DLOG(ERROR
) << "Stub is gone; won't MakeCurrent().";
43 if (!stub
->decoder()->MakeCurrent()) {
44 DLOG(ERROR
) << "Failed to MakeCurrent()";
51 GpuVideoEncodeAccelerator::GpuVideoEncodeAccelerator(int32 host_route_id
,
52 GpuCommandBufferStub
* stub
)
53 : host_route_id_(host_route_id
),
55 input_format_(media::PIXEL_FORMAT_UNKNOWN
),
56 output_buffer_size_(0),
57 weak_this_factory_(this) {
58 stub_
->AddDestructionObserver(this);
59 make_context_current_
=
60 base::Bind(&MakeDecoderContextCurrent
, stub_
->AsWeakPtr());
63 GpuVideoEncodeAccelerator::~GpuVideoEncodeAccelerator() {
64 // This class can only be self-deleted from OnWillDestroyStub(), which means
65 // the VEA has already been destroyed in there.
69 void GpuVideoEncodeAccelerator::Initialize(
70 media::VideoPixelFormat input_format
,
71 const gfx::Size
& input_visible_size
,
72 media::VideoCodecProfile output_profile
,
73 uint32 initial_bitrate
,
74 IPC::Message
* init_done_msg
) {
75 DVLOG(2) << "GpuVideoEncodeAccelerator::Initialize(): "
76 "input_format=" << input_format
77 << ", input_visible_size=" << input_visible_size
.ToString()
78 << ", output_profile=" << output_profile
79 << ", initial_bitrate=" << initial_bitrate
;
82 if (!stub_
->channel()->AddRoute(host_route_id_
, this)) {
83 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::Initialize(): "
84 "failed to add route";
85 SendCreateEncoderReply(init_done_msg
, false);
89 if (input_visible_size
.width() > media::limits::kMaxDimension
||
90 input_visible_size
.height() > media::limits::kMaxDimension
||
91 input_visible_size
.GetArea() > media::limits::kMaxCanvas
) {
92 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::Initialize(): "
93 "input_visible_size " << input_visible_size
.ToString()
95 SendCreateEncoderReply(init_done_msg
, false);
99 std::vector
<GpuVideoEncodeAccelerator::CreateVEAFp
>
100 create_vea_fps
= CreateVEAFps();
101 // Try all possible encoders and use the first successful encoder.
102 for (size_t i
= 0; i
< create_vea_fps
.size(); ++i
) {
103 encoder_
= (*create_vea_fps
[i
])();
104 if (encoder_
&& encoder_
->Initialize(input_format
,
109 input_format_
= input_format
;
110 input_visible_size_
= input_visible_size
;
111 SendCreateEncoderReply(init_done_msg
, true);
117 << "GpuVideoEncodeAccelerator::Initialize(): VEA initialization failed";
118 SendCreateEncoderReply(init_done_msg
, false);
121 bool GpuVideoEncodeAccelerator::OnMessageReceived(const IPC::Message
& message
) {
123 IPC_BEGIN_MESSAGE_MAP(GpuVideoEncodeAccelerator
, message
)
124 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_Encode
, OnEncode
)
125 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_UseOutputBitstreamBuffer
,
126 OnUseOutputBitstreamBuffer
)
128 AcceleratedVideoEncoderMsg_RequestEncodingParametersChange
,
129 OnRequestEncodingParametersChange
)
130 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_Destroy
, OnDestroy
)
131 IPC_MESSAGE_UNHANDLED(handled
= false)
132 IPC_END_MESSAGE_MAP()
136 void GpuVideoEncodeAccelerator::RequireBitstreamBuffers(
137 unsigned int input_count
,
138 const gfx::Size
& input_coded_size
,
139 size_t output_buffer_size
) {
140 Send(new AcceleratedVideoEncoderHostMsg_RequireBitstreamBuffers(
141 host_route_id_
, input_count
, input_coded_size
, output_buffer_size
));
142 input_coded_size_
= input_coded_size
;
143 output_buffer_size_
= output_buffer_size
;
146 void GpuVideoEncodeAccelerator::BitstreamBufferReady(int32 bitstream_buffer_id
,
149 Send(new AcceleratedVideoEncoderHostMsg_BitstreamBufferReady(
150 host_route_id_
, bitstream_buffer_id
, payload_size
, key_frame
));
153 void GpuVideoEncodeAccelerator::NotifyError(
154 media::VideoEncodeAccelerator::Error error
) {
155 Send(new AcceleratedVideoEncoderHostMsg_NotifyError(host_route_id_
, error
));
158 void GpuVideoEncodeAccelerator::OnWillDestroyStub() {
160 stub_
->channel()->RemoveRoute(host_route_id_
);
161 stub_
->RemoveDestructionObserver(this);
167 gpu::VideoEncodeAcceleratorSupportedProfiles
168 GpuVideoEncodeAccelerator::GetSupportedProfiles() {
169 media::VideoEncodeAccelerator::SupportedProfiles profiles
;
170 std::vector
<GpuVideoEncodeAccelerator::CreateVEAFp
>
171 create_vea_fps
= CreateVEAFps();
173 for (size_t i
= 0; i
< create_vea_fps
.size(); ++i
) {
174 scoped_ptr
<media::VideoEncodeAccelerator
>
175 encoder
= (*create_vea_fps
[i
])();
178 media::VideoEncodeAccelerator::SupportedProfiles vea_profiles
=
179 encoder
->GetSupportedProfiles();
180 GpuVideoAcceleratorUtil::InsertUniqueEncodeProfiles(
181 vea_profiles
, &profiles
);
183 return GpuVideoAcceleratorUtil::ConvertMediaToGpuEncodeProfiles(profiles
);
187 std::vector
<GpuVideoEncodeAccelerator::CreateVEAFp
>
188 GpuVideoEncodeAccelerator::CreateVEAFps() {
189 std::vector
<GpuVideoEncodeAccelerator::CreateVEAFp
> create_vea_fps
;
190 create_vea_fps
.push_back(&GpuVideoEncodeAccelerator::CreateV4L2VEA
);
191 create_vea_fps
.push_back(&GpuVideoEncodeAccelerator::CreateVaapiVEA
);
192 create_vea_fps
.push_back(&GpuVideoEncodeAccelerator::CreateAndroidVEA
);
193 return create_vea_fps
;
197 scoped_ptr
<media::VideoEncodeAccelerator
>
198 GpuVideoEncodeAccelerator::CreateV4L2VEA() {
199 scoped_ptr
<media::VideoEncodeAccelerator
> encoder
;
200 #if defined(OS_CHROMEOS) && defined(USE_V4L2_CODEC)
201 scoped_refptr
<V4L2Device
> device
= V4L2Device::Create(V4L2Device::kEncoder
);
203 encoder
.reset(new V4L2VideoEncodeAccelerator(device
));
205 return encoder
.Pass();
209 scoped_ptr
<media::VideoEncodeAccelerator
>
210 GpuVideoEncodeAccelerator::CreateVaapiVEA() {
211 scoped_ptr
<media::VideoEncodeAccelerator
> encoder
;
212 #if defined(OS_CHROMEOS) && defined(ARCH_CPU_X86_FAMILY)
213 const base::CommandLine
* cmd_line
= base::CommandLine::ForCurrentProcess();
214 if (!cmd_line
->HasSwitch(switches::kDisableVaapiAcceleratedVideoEncode
))
215 encoder
.reset(new VaapiVideoEncodeAccelerator());
217 return encoder
.Pass();
221 scoped_ptr
<media::VideoEncodeAccelerator
>
222 GpuVideoEncodeAccelerator::CreateAndroidVEA() {
223 scoped_ptr
<media::VideoEncodeAccelerator
> encoder
;
224 #if defined(OS_ANDROID) && defined(ENABLE_WEBRTC)
225 encoder
.reset(new AndroidVideoEncodeAccelerator());
227 return encoder
.Pass();
230 void GpuVideoEncodeAccelerator::OnEncode(
231 const AcceleratedVideoEncoderMsg_Encode_Params
& params
) {
232 DVLOG(3) << "GpuVideoEncodeAccelerator::OnEncode: frame_id = "
233 << params
.frame_id
<< ", buffer_size=" << params
.buffer_size
234 << ", force_keyframe=" << params
.force_keyframe
;
237 if (params
.frame_id
< 0) {
238 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnEncode(): invalid "
239 "frame_id=" << params
.frame_id
;
240 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
244 const uint32 aligned_offset
=
245 params
.buffer_offset
% base::SysInfo::VMAllocationGranularity();
246 base::CheckedNumeric
<off_t
> map_offset
= params
.buffer_offset
;
247 map_offset
-= aligned_offset
;
248 base::CheckedNumeric
<size_t> map_size
= params
.buffer_size
;
249 map_size
+= aligned_offset
;
251 if (!map_offset
.IsValid() || !map_size
.IsValid()) {
252 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnEncode():"
253 << " invalid (buffer_offset,buffer_size)";
254 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
258 scoped_ptr
<base::SharedMemory
> shm(
259 new base::SharedMemory(params
.buffer_handle
, true));
260 if (!shm
->MapAt(map_offset
.ValueOrDie(), map_size
.ValueOrDie())) {
261 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnEncode(): "
262 "could not map frame_id=" << params
.frame_id
;
263 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
267 uint8
* shm_memory
= reinterpret_cast<uint8
*>(shm
->memory()) + aligned_offset
;
268 scoped_refptr
<media::VideoFrame
> frame
=
269 media::VideoFrame::WrapExternalSharedMemory(
272 gfx::Rect(input_visible_size_
),
276 params
.buffer_handle
,
277 params
.buffer_offset
,
279 frame
->AddDestructionObserver(
280 media::BindToCurrentLoop(
281 base::Bind(&GpuVideoEncodeAccelerator::EncodeFrameFinished
,
282 weak_this_factory_
.GetWeakPtr(),
284 base::Passed(&shm
))));
287 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnEncode(): could not create "
288 "VideoFrame for frame_id=" << params
.frame_id
;
289 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
293 encoder_
->Encode(frame
, params
.force_keyframe
);
296 void GpuVideoEncodeAccelerator::OnUseOutputBitstreamBuffer(
298 base::SharedMemoryHandle buffer_handle
,
299 uint32 buffer_size
) {
300 DVLOG(3) << "GpuVideoEncodeAccelerator::OnUseOutputBitstreamBuffer(): "
301 "buffer_id=" << buffer_id
302 << ", buffer_size=" << buffer_size
;
306 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnUseOutputBitstreamBuffer(): "
307 "invalid buffer_id=" << buffer_id
;
308 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
311 if (buffer_size
< output_buffer_size_
) {
312 DLOG(ERROR
) << "GpuVideoEncodeAccelerator::OnUseOutputBitstreamBuffer(): "
313 "buffer too small for buffer_id=" << buffer_id
;
314 NotifyError(media::VideoEncodeAccelerator::kPlatformFailureError
);
317 encoder_
->UseOutputBitstreamBuffer(
318 media::BitstreamBuffer(buffer_id
, buffer_handle
, buffer_size
));
321 void GpuVideoEncodeAccelerator::OnDestroy() {
322 DVLOG(2) << "GpuVideoEncodeAccelerator::OnDestroy()";
326 void GpuVideoEncodeAccelerator::OnRequestEncodingParametersChange(
329 DVLOG(2) << "GpuVideoEncodeAccelerator::OnRequestEncodingParametersChange(): "
330 "bitrate=" << bitrate
331 << ", framerate=" << framerate
;
334 encoder_
->RequestEncodingParametersChange(bitrate
, framerate
);
337 void GpuVideoEncodeAccelerator::EncodeFrameFinished(
339 scoped_ptr
<base::SharedMemory
> shm
) {
340 Send(new AcceleratedVideoEncoderHostMsg_NotifyInputDone(host_route_id_
,
342 // Just let |shm| fall out of scope.
345 void GpuVideoEncodeAccelerator::Send(IPC::Message
* message
) {
346 stub_
->channel()->Send(message
);
349 void GpuVideoEncodeAccelerator::SendCreateEncoderReply(IPC::Message
* message
,
351 GpuCommandBufferMsg_CreateVideoEncoder::WriteReplyParams(message
, succeeded
);
355 } // namespace content