OLD | NEW |
---|---|
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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/dxva_video_decode_accelerator_win.h" | 5 #include "content/common/gpu/media/dxva_video_decode_accelerator_win.h" |
6 | 6 |
7 #if !defined(OS_WIN) | 7 #if !defined(OS_WIN) |
8 #error This file should only be built on Windows. | 8 #error This file should only be built on Windows. |
9 #endif // !defined(OS_WIN) | 9 #endif // !defined(OS_WIN) |
10 | 10 |
(...skipping 841 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
852 video_format_converter_mft_.ReceiveVoid()); | 852 video_format_converter_mft_.ReceiveVoid()); |
853 if (FAILED(hr)) { | 853 if (FAILED(hr)) { |
854 base::debug::Alias(&hr); | 854 base::debug::Alias(&hr); |
855 // TODO(ananta) | 855 // TODO(ananta) |
856 // Remove this CHECK when the change to use DX11 for H/W decoding | 856 // Remove this CHECK when the change to use DX11 for H/W decoding |
857 // stablizes. | 857 // stablizes. |
858 CHECK(false); | 858 CHECK(false); |
859 } | 859 } |
860 | 860 |
861 RETURN_ON_HR_FAILURE(hr, "Failed to create video format converter", false); | 861 RETURN_ON_HR_FAILURE(hr, "Failed to create video format converter", false); |
862 | |
863 base::win::ScopedComPtr<IMFAttributes> converter_attributes; | |
864 hr = video_format_converter_mft_->GetAttributes( | |
865 converter_attributes.Receive()); | |
866 RETURN_ON_HR_FAILURE(hr, "Failed to get converter attributes", false); | |
867 | |
868 hr = converter_attributes->SetUINT32(MF_XVP_PLAYBACK_MODE, TRUE); | |
869 RETURN_ON_HR_FAILURE( | |
870 hr, | |
871 "Failed to set MF_XVP_PLAYBACK_MODE attribute on converter", | |
872 false); | |
873 | |
874 hr = converter_attributes->SetUINT32(MF_LOW_LATENCY, FALSE); | |
875 RETURN_ON_HR_FAILURE( | |
876 hr, | |
877 "Failed to set MF_LOW_LATENCY attribute on converter", | |
878 false); | |
862 return true; | 879 return true; |
863 } | 880 } |
864 | 881 |
865 void DXVAVideoDecodeAccelerator::Decode( | 882 void DXVAVideoDecodeAccelerator::Decode( |
866 const media::BitstreamBuffer& bitstream_buffer) { | 883 const media::BitstreamBuffer& bitstream_buffer) { |
867 DCHECK(main_thread_task_runner_->BelongsToCurrentThread()); | 884 DCHECK(main_thread_task_runner_->BelongsToCurrentThread()); |
868 | 885 |
869 State state = GetState(); | 886 State state = GetState(); |
870 RETURN_AND_NOTIFY_ON_FAILURE((state == kNormal || state == kStopped || | 887 RETURN_AND_NOTIFY_ON_FAILURE((state == kNormal || state == kStopped || |
871 state == kFlushing), | 888 state == kFlushing), |
(...skipping 389 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1261 MFVideoInterlace_MixedInterlaceOrProgressive); | 1278 MFVideoInterlace_MixedInterlaceOrProgressive); |
1262 RETURN_ON_HR_FAILURE(hr, "Failed to set interlace mode", false); | 1279 RETURN_ON_HR_FAILURE(hr, "Failed to set interlace mode", false); |
1263 | 1280 |
1264 hr = decoder_->SetInputType(0, media_type.get(), 0); // No flags | 1281 hr = decoder_->SetInputType(0, media_type.get(), 0); // No flags |
1265 RETURN_ON_HR_FAILURE(hr, "Failed to set decoder input type", false); | 1282 RETURN_ON_HR_FAILURE(hr, "Failed to set decoder input type", false); |
1266 return true; | 1283 return true; |
1267 } | 1284 } |
1268 | 1285 |
1269 bool DXVAVideoDecodeAccelerator::SetDecoderOutputMediaType( | 1286 bool DXVAVideoDecodeAccelerator::SetDecoderOutputMediaType( |
1270 const GUID& subtype) { | 1287 const GUID& subtype) { |
1271 base::win::ScopedComPtr<IMFMediaType> out_media_type; | 1288 return SetTransformOutputType(decoder_.get(), subtype, 0, 0); |
1272 | |
1273 for (uint32_t i = 0; SUCCEEDED( | |
1274 decoder_->GetOutputAvailableType(0, i, out_media_type.Receive())); | |
1275 ++i) { | |
1276 GUID out_subtype = {0}; | |
1277 HRESULT hr = out_media_type->GetGUID(MF_MT_SUBTYPE, &out_subtype); | |
1278 RETURN_ON_HR_FAILURE(hr, "Failed to get output major type", false); | |
1279 | |
1280 if (out_subtype == subtype) { | |
1281 hr = decoder_->SetOutputType(0, out_media_type.get(), 0); // No flags | |
1282 RETURN_ON_HR_FAILURE(hr, "Failed to set decoder output type", false); | |
1283 return true; | |
1284 } | |
1285 out_media_type.Release(); | |
1286 } | |
1287 return false; | |
1288 } | 1289 } |
1289 | 1290 |
1290 bool DXVAVideoDecodeAccelerator::SendMFTMessage(MFT_MESSAGE_TYPE msg, | 1291 bool DXVAVideoDecodeAccelerator::SendMFTMessage(MFT_MESSAGE_TYPE msg, |
1291 int32_t param) { | 1292 int32_t param) { |
1292 HRESULT hr = decoder_->ProcessMessage(msg, param); | 1293 HRESULT hr = decoder_->ProcessMessage(msg, param); |
1293 return SUCCEEDED(hr); | 1294 return SUCCEEDED(hr); |
1294 } | 1295 } |
1295 | 1296 |
1296 // Gets the minimum buffer sizes for input and output samples. The MFT will not | 1297 // Gets the minimum buffer sizes for input and output samples. The MFT will not |
1297 // allocate buffer for input nor output, so we have to do it ourselves and make | 1298 // allocate buffer for input nor output, so we have to do it ourselves and make |
(...skipping 881 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
2179 PLATFORM_FAILURE, false); | 2180 PLATFORM_FAILURE, false); |
2180 | 2181 |
2181 hr = media_type->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Video); | 2182 hr = media_type->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Video); |
2182 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set major input type", | 2183 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set major input type", |
2183 PLATFORM_FAILURE, false); | 2184 PLATFORM_FAILURE, false); |
2184 | 2185 |
2185 hr = media_type->SetGUID(MF_MT_SUBTYPE, MFVideoFormat_NV12); | 2186 hr = media_type->SetGUID(MF_MT_SUBTYPE, MFVideoFormat_NV12); |
2186 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set input sub type", | 2187 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set input sub type", |
2187 PLATFORM_FAILURE, false); | 2188 PLATFORM_FAILURE, false); |
2188 | 2189 |
2189 hr = media_type->SetUINT32(MF_MT_ALL_SAMPLES_INDEPENDENT, TRUE); | |
2190 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2191 "Failed to set attributes on media type", PLATFORM_FAILURE, false); | |
2192 | |
2193 hr = media_type->SetUINT32(MF_MT_INTERLACE_MODE, | |
2194 MFVideoInterlace_Progressive); | |
2195 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2196 "Failed to set attributes on media type", PLATFORM_FAILURE, false); | |
2197 | |
2198 base::win::ScopedComPtr<IMFAttributes> converter_attributes; | |
2199 hr = video_format_converter_mft_->GetAttributes( | |
2200 converter_attributes.Receive()); | |
2201 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to get converter attributes", | |
2202 PLATFORM_FAILURE, false); | |
2203 | |
2204 hr = converter_attributes->SetUINT32(MF_XVP_PLAYBACK_MODE, TRUE); | |
2205 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set converter attributes", | |
2206 PLATFORM_FAILURE, false); | |
2207 | |
2208 hr = converter_attributes->SetUINT32(MF_LOW_LATENCY, FALSE); | |
2209 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set converter attributes", | |
2210 PLATFORM_FAILURE, false); | |
2211 | |
2212 hr = MFSetAttributeSize(media_type.get(), MF_MT_FRAME_SIZE, width, height); | 2190 hr = MFSetAttributeSize(media_type.get(), MF_MT_FRAME_SIZE, width, height); |
2213 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set media type attributes", | 2191 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set media type attributes", |
2214 PLATFORM_FAILURE, false); | 2192 PLATFORM_FAILURE, false); |
2215 | 2193 |
2216 hr = video_format_converter_mft_->SetInputType(0, media_type.get(), 0); | 2194 hr = video_format_converter_mft_->SetInputType(0, media_type.get(), 0); |
2217 if (FAILED(hr)) { | 2195 if (FAILED(hr)) { |
2218 base::debug::Alias(&hr); | 2196 base::debug::Alias(&hr); |
2219 // TODO(ananta) | 2197 // TODO(ananta) |
2220 // Remove this CHECK when the change to use DX11 for H/W decoding | 2198 // Remove this CHECK when the change to use DX11 for H/W decoding |
2221 // stablizes. | 2199 // stablizes. |
2222 CHECK(false); | 2200 CHECK(false); |
2223 } | 2201 } |
2224 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set converter input type", | 2202 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to set converter input type", |
2225 PLATFORM_FAILURE, false); | 2203 PLATFORM_FAILURE, false); |
2226 | 2204 |
2227 base::win::ScopedComPtr<IMFMediaType> out_media_type; | 2205 // It appears that we fail to set MFVideoFormat_ARGB32 as the output media |
2206 // type in certain configurations. Try to fallback to MFVideoFormat_RGB32 | |
2207 // in such cases. If both fail, then bail. | |
2208 bool media_type_set = | |
2209 SetTransformOutputType(video_format_converter_mft_.get(), | |
2210 MFVideoFormat_ARGB32, | |
2211 width, | |
2212 height); | |
2213 if (!media_type_set) { | |
2214 media_type_set = | |
2215 SetTransformOutputType(video_format_converter_mft_.get(), | |
2216 MFVideoFormat_RGB32, | |
2217 width, | |
2218 height); | |
2219 } | |
2228 | 2220 |
2229 for (uint32_t i = 0; | 2221 if (!media_type_set) { |
2230 SUCCEEDED(video_format_converter_mft_->GetOutputAvailableType( | 2222 // Remove this once this stabilizes in the field. |
2231 0, i, out_media_type.Receive())); | 2223 CHECK(false); |
sandersd (OOO until July 31)
2016/01/15 19:25:02
Nit: Add an error message here?
ananta
2016/01/15 20:23:41
Done.
| |
2232 ++i) { | 2224 return false; |
2233 GUID out_subtype = {0}; | 2225 } |
2234 hr = out_media_type->GetGUID(MF_MT_SUBTYPE, &out_subtype); | |
2235 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, "Failed to get output major type", | |
2236 PLATFORM_FAILURE, false); | |
2237 | 2226 |
2238 if (out_subtype == MFVideoFormat_ARGB32) { | 2227 dx11_video_format_converter_media_type_needs_init_ = false; |
2239 hr = out_media_type->SetUINT32(MF_MT_ALL_SAMPLES_INDEPENDENT, TRUE); | 2228 return true; |
2240 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2241 "Failed to set attributes on media type", PLATFORM_FAILURE, false); | |
2242 | |
2243 hr = out_media_type->SetUINT32(MF_MT_INTERLACE_MODE, | |
2244 MFVideoInterlace_Progressive); | |
2245 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2246 "Failed to set attributes on media type", PLATFORM_FAILURE, false); | |
2247 | |
2248 hr = MFSetAttributeSize(out_media_type.get(), MF_MT_FRAME_SIZE, width, | |
2249 height); | |
2250 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2251 "Failed to set media type attributes", PLATFORM_FAILURE, false); | |
2252 | |
2253 hr = video_format_converter_mft_->SetOutputType( | |
2254 0, out_media_type.get(), 0); // No flags | |
2255 if (FAILED(hr)) { | |
2256 base::debug::Alias(&hr); | |
2257 // TODO(ananta) | |
2258 // Remove this CHECK when the change to use DX11 for H/W decoding | |
2259 // stablizes. | |
2260 CHECK(false); | |
2261 } | |
2262 RETURN_AND_NOTIFY_ON_HR_FAILURE(hr, | |
2263 "Failed to set converter output type", PLATFORM_FAILURE, false); | |
2264 | |
2265 dx11_video_format_converter_media_type_needs_init_ = false; | |
2266 return true; | |
2267 } | |
2268 out_media_type.Release(); | |
2269 } | |
2270 return false; | |
2271 } | 2229 } |
2272 | 2230 |
2273 bool DXVAVideoDecodeAccelerator::GetVideoFrameDimensions( | 2231 bool DXVAVideoDecodeAccelerator::GetVideoFrameDimensions( |
2274 IMFSample* sample, | 2232 IMFSample* sample, |
2275 int* width, | 2233 int* width, |
2276 int* height) { | 2234 int* height) { |
2277 base::win::ScopedComPtr<IMFMediaBuffer> output_buffer; | 2235 base::win::ScopedComPtr<IMFMediaBuffer> output_buffer; |
2278 HRESULT hr = sample->GetBufferByIndex(0, output_buffer.Receive()); | 2236 HRESULT hr = sample->GetBufferByIndex(0, output_buffer.Receive()); |
2279 RETURN_ON_HR_FAILURE(hr, "Failed to get buffer from output sample", false); | 2237 RETURN_ON_HR_FAILURE(hr, "Failed to get buffer from output sample", false); |
2280 | 2238 |
(...skipping 20 matching lines...) Expand all Loading... | |
2301 false); | 2259 false); |
2302 D3DSURFACE_DESC surface_desc; | 2260 D3DSURFACE_DESC surface_desc; |
2303 hr = surface->GetDesc(&surface_desc); | 2261 hr = surface->GetDesc(&surface_desc); |
2304 RETURN_ON_HR_FAILURE(hr, "Failed to get surface description", false); | 2262 RETURN_ON_HR_FAILURE(hr, "Failed to get surface description", false); |
2305 *width = surface_desc.Width; | 2263 *width = surface_desc.Width; |
2306 *height = surface_desc.Height; | 2264 *height = surface_desc.Height; |
2307 } | 2265 } |
2308 return true; | 2266 return true; |
2309 } | 2267 } |
2310 | 2268 |
2269 bool DXVAVideoDecodeAccelerator::SetTransformOutputType( | |
2270 IMFTransform* transform, | |
2271 const GUID& output_type, | |
2272 int width, | |
2273 int height) { | |
2274 HRESULT hr = E_FAIL; | |
2275 base::win::ScopedComPtr<IMFMediaType> media_type; | |
2276 | |
2277 for (uint32_t i = 0; | |
2278 SUCCEEDED(transform->GetOutputAvailableType( | |
2279 0, i, media_type.Receive())); | |
2280 ++i) { | |
2281 GUID out_subtype = {0}; | |
2282 hr = media_type->GetGUID(MF_MT_SUBTYPE, &out_subtype); | |
2283 RETURN_ON_HR_FAILURE(hr, "Failed to get output major type", false); | |
2284 | |
2285 if (out_subtype == output_type) { | |
2286 if (width && height) { | |
2287 hr = MFSetAttributeSize(media_type.get(), MF_MT_FRAME_SIZE, width, | |
2288 height); | |
2289 RETURN_ON_HR_FAILURE(hr, "Failed to set media type attributes", false); | |
2290 } | |
2291 hr = transform->SetOutputType(0, media_type.get(), 0); // No flags | |
2292 if (FAILED(hr)) { | |
2293 base::debug::Alias(&hr); | |
2294 // TODO(ananta) | |
2295 // Remove this CHECK when this stabilizes in the field. | |
2296 CHECK(false); | |
sandersd (OOO until July 31)
2016/01/15 19:25:02
Should this just be returning false in case the fa
ananta
2016/01/15 20:23:41
If the fallback works then I think we are good. I
| |
2297 } | |
2298 RETURN_ON_HR_FAILURE(hr, "Failed to set output type", false); | |
2299 return true; | |
2300 } | |
2301 media_type.Release(); | |
2302 } | |
2303 return false; | |
2304 } | |
2305 | |
2311 } // namespace content | 2306 } // namespace content |
OLD | NEW |