Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(175)

Side by Side Diff: content/common/gpu/media/gpu_video_encode_accelerator.cc

Issue 826663002: Support multiple video decoders and encoders (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Use callback function for decoders Created 5 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2013 The Chromium Authors. All rights reserved. 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 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "content/common/gpu/media/gpu_video_encode_accelerator.h" 5 #include "content/common/gpu/media/gpu_video_encode_accelerator.h"
6 6
7 #include "base/callback.h" 7 #include "base/callback.h"
8 #include "base/command_line.h" 8 #include "base/command_line.h"
9 #include "base/logging.h" 9 #include "base/logging.h"
10 #include "base/memory/shared_memory.h" 10 #include "base/memory/shared_memory.h"
11 #include "base/message_loop/message_loop_proxy.h" 11 #include "base/message_loop/message_loop_proxy.h"
12 #include "build/build_config.h" 12 #include "build/build_config.h"
13 #include "content/common/gpu/gpu_channel.h" 13 #include "content/common/gpu/gpu_channel.h"
14 #include "content/common/gpu/gpu_messages.h" 14 #include "content/common/gpu/gpu_messages.h"
15 #include "content/public/common/content_switches.h" 15 #include "content/public/common/content_switches.h"
16 #include "ipc/ipc_message_macros.h" 16 #include "ipc/ipc_message_macros.h"
17 #include "media/base/limits.h" 17 #include "media/base/limits.h"
18 #include "media/base/video_frame.h" 18 #include "media/base/video_frame.h"
19 19
20 #if defined(OS_CHROMEOS) && defined(USE_X11) 20 #if defined(OS_CHROMEOS) && defined(USE_X11)
21 21
22 #if defined(ARCH_CPU_ARMEL) 22 #if defined(ARCH_CPU_ARMEL)
23 #include "content/common/gpu/media/v4l2_video_encode_accelerator.h" 23 #include "content/common/gpu/media/v4l2_video_encode_accelerator.h"
24 #elif defined(ARCH_CPU_X86_FAMILY) 24 #elif defined(ARCH_CPU_X86_FAMILY)
25 #include "content/common/gpu/media/v4l2_video_encode_accelerator.h"
25 #include "content/common/gpu/media/vaapi_video_encode_accelerator.h" 26 #include "content/common/gpu/media/vaapi_video_encode_accelerator.h"
26 #include "ui/gfx/x/x11_types.h" 27 #include "ui/gfx/x/x11_types.h"
27 #endif 28 #endif
28 29
29 #elif defined(OS_ANDROID) && defined(ENABLE_WEBRTC) 30 #elif defined(OS_ANDROID) && defined(ENABLE_WEBRTC)
30 #include "content/common/gpu/media/android_video_encode_accelerator.h" 31 #include "content/common/gpu/media/android_video_encode_accelerator.h"
31 #endif 32 #endif
32 33
33 namespace content { 34 namespace content {
34 35
(...skipping 53 matching lines...) Expand 10 before | Expand all | Expand 10 after
88 if (input_visible_size.width() > media::limits::kMaxDimension || 89 if (input_visible_size.width() > media::limits::kMaxDimension ||
89 input_visible_size.height() > media::limits::kMaxDimension || 90 input_visible_size.height() > media::limits::kMaxDimension ||
90 input_visible_size.GetArea() > media::limits::kMaxCanvas) { 91 input_visible_size.GetArea() > media::limits::kMaxCanvas) {
91 DLOG(ERROR) << "GpuVideoEncodeAccelerator::Initialize(): " 92 DLOG(ERROR) << "GpuVideoEncodeAccelerator::Initialize(): "
92 "input_visible_size " << input_visible_size.ToString() 93 "input_visible_size " << input_visible_size.ToString()
93 << " too large"; 94 << " too large";
94 SendCreateEncoderReply(init_done_msg, false); 95 SendCreateEncoderReply(init_done_msg, false);
95 return; 96 return;
96 } 97 }
97 98
98 encoder_ = CreateEncoder(); 99 std::vector<GpuVideoEncodeAccelerator::CreateVEACb>
99 if (!encoder_) { 100 create_vea_cbs = CreateEncoderCbs();
101 if (create_vea_cbs.empty()) {
Pawel Osciak 2014/12/31 07:56:00 Not needed?
henryhsu 2014/12/31 08:47:53 Done.
100 DLOG(ERROR) 102 DLOG(ERROR)
101 << "GpuVideoEncodeAccelerator::Initialize(): VEA creation failed"; 103 << "GpuVideoEncodeAccelerator::Initialize(): VEA creation failed";
102 SendCreateEncoderReply(init_done_msg, false); 104 SendCreateEncoderReply(init_done_msg, false);
103 return; 105 return;
104 } 106 }
105 if (!encoder_->Initialize(input_format, 107 // Try all possible encoders and use the first successful encoder.
106 input_visible_size, 108 for (size_t i = 0; i < create_vea_cbs.size(); ++i) {
107 output_profile, 109 encoder_ = create_vea_cbs[i].Run();
108 initial_bitrate, 110 if (encoder_ && encoder_->Initialize(input_format,
109 this)) { 111 input_visible_size,
110 DLOG(ERROR) 112 output_profile,
111 << "GpuVideoEncodeAccelerator::Initialize(): VEA initialization failed"; 113 initial_bitrate,
112 SendCreateEncoderReply(init_done_msg, false); 114 this)) {
113 return; 115 input_format_ = input_format;
116 input_visible_size_ = input_visible_size;
117 SendCreateEncoderReply(init_done_msg, true);
118 return;
119 }
114 } 120 }
115 input_format_ = input_format; 121 encoder_.reset();
116 input_visible_size_ = input_visible_size; 122 DLOG(ERROR)
117 SendCreateEncoderReply(init_done_msg, true); 123 << "GpuVideoEncodeAccelerator::Initialize(): VEA initialization failed";
124 SendCreateEncoderReply(init_done_msg, false);
118 } 125 }
119 126
120 bool GpuVideoEncodeAccelerator::OnMessageReceived(const IPC::Message& message) { 127 bool GpuVideoEncodeAccelerator::OnMessageReceived(const IPC::Message& message) {
121 bool handled = true; 128 bool handled = true;
122 IPC_BEGIN_MESSAGE_MAP(GpuVideoEncodeAccelerator, message) 129 IPC_BEGIN_MESSAGE_MAP(GpuVideoEncodeAccelerator, message)
123 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_Encode, OnEncode) 130 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_Encode, OnEncode)
124 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_UseOutputBitstreamBuffer, 131 IPC_MESSAGE_HANDLER(AcceleratedVideoEncoderMsg_UseOutputBitstreamBuffer,
125 OnUseOutputBitstreamBuffer) 132 OnUseOutputBitstreamBuffer)
126 IPC_MESSAGE_HANDLER( 133 IPC_MESSAGE_HANDLER(
127 AcceleratedVideoEncoderMsg_RequestEncodingParametersChange, 134 AcceleratedVideoEncoderMsg_RequestEncodingParametersChange,
(...skipping 30 matching lines...) Expand all
158 DCHECK(stub_); 165 DCHECK(stub_);
159 stub_->channel()->RemoveRoute(host_route_id_); 166 stub_->channel()->RemoveRoute(host_route_id_);
160 stub_->RemoveDestructionObserver(this); 167 stub_->RemoveDestructionObserver(this);
161 encoder_.reset(); 168 encoder_.reset();
162 delete this; 169 delete this;
163 } 170 }
164 171
165 // static 172 // static
166 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile> 173 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile>
167 GpuVideoEncodeAccelerator::GetSupportedProfiles() { 174 GpuVideoEncodeAccelerator::GetSupportedProfiles() {
168 scoped_ptr<media::VideoEncodeAccelerator> encoder = CreateEncoder(); 175 std::vector<media::VideoEncodeAccelerator::SupportedProfile> profiles;
169 if (!encoder) 176 std::vector<GpuVideoEncodeAccelerator::CreateVEACb>
177 create_vea_cbs = CreateEncoderCbs();
178 if (create_vea_cbs.empty())
Pawel Osciak 2014/12/31 07:56:00 Not needed, for() handles this?
henryhsu 2014/12/31 08:47:53 Done.
170 return std::vector<gpu::VideoEncodeAcceleratorSupportedProfile>(); 179 return std::vector<gpu::VideoEncodeAcceleratorSupportedProfile>();
171 return ConvertMediaToGpuProfiles(encoder->GetSupportedProfiles()); 180
181 std::set<media::VideoEncodeAccelerator::SupportedProfile> profile_set;
182 for (size_t i = 0; i < create_vea_cbs.size(); ++i) {
183 scoped_ptr<media::VideoEncodeAccelerator>
184 encoder = create_vea_cbs[i].Run();
185 if (!encoder)
186 continue;
187 std::vector<media::VideoEncodeAccelerator::SupportedProfile>
188 vea_profiles = encoder->GetSupportedProfiles();
189 for (size_t j = 0; j < vea_profiles.size(); ++j)
190 profile_set.insert(vea_profiles[j]);
191 }
192 profiles.assign(profile_set.begin(), profile_set.end());
193 return ConvertMediaToGpuProfiles(profiles);
172 } 194 }
173 195
196 // static
174 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile> 197 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile>
175 GpuVideoEncodeAccelerator::ConvertMediaToGpuProfiles(const std::vector< 198 GpuVideoEncodeAccelerator::ConvertMediaToGpuProfiles(const std::vector<
176 media::VideoEncodeAccelerator::SupportedProfile>& media_profiles) { 199 media::VideoEncodeAccelerator::SupportedProfile>& media_profiles) {
177 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile> profiles; 200 std::vector<gpu::VideoEncodeAcceleratorSupportedProfile> profiles;
178 for (size_t i = 0; i < media_profiles.size(); i++) { 201 for (size_t i = 0; i < media_profiles.size(); i++) {
179 gpu::VideoEncodeAcceleratorSupportedProfile profile; 202 gpu::VideoEncodeAcceleratorSupportedProfile profile;
180 profile.profile = 203 profile.profile =
181 static_cast<gpu::VideoCodecProfile>(media_profiles[i].profile); 204 static_cast<gpu::VideoCodecProfile>(media_profiles[i].profile);
182 profile.max_resolution = media_profiles[i].max_resolution; 205 profile.max_resolution = media_profiles[i].max_resolution;
183 profile.max_framerate_numerator = media_profiles[i].max_framerate_numerator; 206 profile.max_framerate_numerator = media_profiles[i].max_framerate_numerator;
184 profile.max_framerate_denominator = 207 profile.max_framerate_denominator =
185 media_profiles[i].max_framerate_denominator; 208 media_profiles[i].max_framerate_denominator;
186 profiles.push_back(profile); 209 profiles.push_back(profile);
187 } 210 }
188 return profiles; 211 return profiles;
189 } 212 }
190 213
214 // static
215 std::vector<GpuVideoEncodeAccelerator::CreateVEACb>
216 GpuVideoEncodeAccelerator::CreateEncoderCbs() {
217 std::vector<GpuVideoEncodeAccelerator::CreateVEACb> create_vea_cbs;
218 #if defined(OS_CHROMEOS) && defined(USE_X11)
Pawel Osciak 2014/12/31 07:56:00 Same as in gvda.cc
henryhsu 2014/12/31 08:47:53 Done.
219 create_vea_cbs.push_back(base::Bind(&CreateV4L2Encoder));
220 #if defined(ARCH_CPU_X86_FAMILY)
221 create_vea_cbs.push_back(base::Bind(&CreateVaapiEncoder));
222 #endif
223 #elif defined(OS_ANDROID) && defined(ENABLE_WEBRTC)
224 create_vea_cbs.push_back(base::Bind(&CreateAndroidEncoder));
225 #endif
226 return create_vea_cbs;
227 }
228
229 // static
191 scoped_ptr<media::VideoEncodeAccelerator> 230 scoped_ptr<media::VideoEncodeAccelerator>
192 GpuVideoEncodeAccelerator::CreateEncoder() { 231 GpuVideoEncodeAccelerator::CreateV4L2Encoder() {
193 scoped_ptr<media::VideoEncodeAccelerator> encoder; 232 scoped_ptr<media::VideoEncodeAccelerator> encoder;
194 #if defined(OS_CHROMEOS) && defined(USE_X11) 233 #if defined(OS_CHROMEOS) && defined(USE_X11)
195 #if defined(ARCH_CPU_ARMEL)
196 scoped_ptr<V4L2Device> device = V4L2Device::Create(V4L2Device::kEncoder); 234 scoped_ptr<V4L2Device> device = V4L2Device::Create(V4L2Device::kEncoder);
197 if (device) 235 if (device)
198 encoder.reset(new V4L2VideoEncodeAccelerator(device.Pass())); 236 encoder.reset(new V4L2VideoEncodeAccelerator(device.Pass()));
199 #elif defined(ARCH_CPU_X86_FAMILY) 237 #endif
238 return encoder.Pass();
239 }
240
241 // static
242 scoped_ptr<media::VideoEncodeAccelerator>
243 GpuVideoEncodeAccelerator::CreateVaapiEncoder() {
244 scoped_ptr<media::VideoEncodeAccelerator> encoder;
245 #if defined(OS_CHROMEOS) && defined(USE_X11) && defined(ARCH_CPU_X86_FAMILY)
200 const base::CommandLine* cmd_line = base::CommandLine::ForCurrentProcess(); 246 const base::CommandLine* cmd_line = base::CommandLine::ForCurrentProcess();
201 if (!cmd_line->HasSwitch(switches::kDisableVaapiAcceleratedVideoEncode)) 247 if (!cmd_line->HasSwitch(switches::kDisableVaapiAcceleratedVideoEncode))
202 encoder.reset(new VaapiVideoEncodeAccelerator(gfx::GetXDisplay())); 248 encoder.reset(new VaapiVideoEncodeAccelerator(gfx::GetXDisplay()));
203 #endif 249 #endif
204 #elif defined(OS_ANDROID) && defined(ENABLE_WEBRTC) 250 return encoder.Pass();
251 }
252
253 // static
254 scoped_ptr<media::VideoEncodeAccelerator>
255 GpuVideoEncodeAccelerator::CreateAndroidEncoder() {
256 scoped_ptr<media::VideoEncodeAccelerator> encoder;
257 #if defined(OS_ANDROID) && defined(ENABLE_WEBRTC)
205 encoder.reset(new AndroidVideoEncodeAccelerator()); 258 encoder.reset(new AndroidVideoEncodeAccelerator());
206 #endif 259 #endif
207 return encoder.Pass(); 260 return encoder.Pass();
208 } 261 }
209 262
210 void GpuVideoEncodeAccelerator::OnEncode(int32 frame_id, 263 void GpuVideoEncodeAccelerator::OnEncode(int32 frame_id,
211 base::SharedMemoryHandle buffer_handle, 264 base::SharedMemoryHandle buffer_handle,
212 uint32 buffer_size, 265 uint32 buffer_size,
213 bool force_keyframe) { 266 bool force_keyframe) {
214 DVLOG(3) << "GpuVideoEncodeAccelerator::OnEncode(): frame_id=" << frame_id 267 DVLOG(3) << "GpuVideoEncodeAccelerator::OnEncode(): frame_id=" << frame_id
(...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after
315 stub_->channel()->Send(message); 368 stub_->channel()->Send(message);
316 } 369 }
317 370
318 void GpuVideoEncodeAccelerator::SendCreateEncoderReply(IPC::Message* message, 371 void GpuVideoEncodeAccelerator::SendCreateEncoderReply(IPC::Message* message,
319 bool succeeded) { 372 bool succeeded) {
320 GpuCommandBufferMsg_CreateVideoEncoder::WriteReplyParams(message, succeeded); 373 GpuCommandBufferMsg_CreateVideoEncoder::WriteReplyParams(message, succeeded);
321 Send(message); 374 Send(message);
322 } 375 }
323 376
324 } // namespace content 377 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698