| Index: content/common/gpu/media/v4l2_device.cc
|
| diff --git a/content/common/gpu/media/v4l2_device.cc b/content/common/gpu/media/v4l2_device.cc
|
| index 7629bac5e19109bc35c7eb065b72950a5b4582e0..891da344d003118d4db4edadc5a33ac36e867d2e 100644
|
| --- a/content/common/gpu/media/v4l2_device.cc
|
| +++ b/content/common/gpu/media/v4l2_device.cc
|
| @@ -13,6 +13,9 @@
|
|
|
| namespace content {
|
|
|
| +V4L2Device::V4L2Device(Type type) : type_(type) {
|
| +}
|
| +
|
| V4L2Device::~V4L2Device() {
|
| }
|
|
|
| @@ -197,4 +200,98 @@ gfx::Size V4L2Device::CodedSizeFromV4L2Format(struct v4l2_format format) {
|
| return coded_size;
|
| }
|
|
|
| +void V4L2Device::GetSupportedResolution(uint32_t pixelformat,
|
| + gfx::Size* min_resolution,
|
| + gfx::Size* max_resolution) {
|
| + max_resolution->SetSize(0, 0);
|
| + min_resolution->SetSize(0, 0);
|
| + v4l2_frmsizeenum frame_size;
|
| + memset(&frame_size, 0, sizeof(frame_size));
|
| + frame_size.pixel_format = pixelformat;
|
| + for (; Ioctl(VIDIOC_ENUM_FRAMESIZES, &frame_size) == 0; ++frame_size.index) {
|
| + if (frame_size.type == V4L2_FRMSIZE_TYPE_DISCRETE) {
|
| + if (frame_size.discrete.width >=
|
| + base::checked_cast<uint32_t>(max_resolution->width()) &&
|
| + frame_size.discrete.height >=
|
| + base::checked_cast<uint32_t>(max_resolution->height())) {
|
| + max_resolution->SetSize(frame_size.discrete.width,
|
| + frame_size.discrete.height);
|
| + }
|
| + if (min_resolution->IsEmpty() ||
|
| + (frame_size.discrete.width <=
|
| + base::checked_cast<uint32_t>(min_resolution->width()) &&
|
| + frame_size.discrete.height <=
|
| + base::checked_cast<uint32_t>(min_resolution->height()))) {
|
| + min_resolution->SetSize(frame_size.discrete.width,
|
| + frame_size.discrete.height);
|
| + }
|
| + } else if (frame_size.type == V4L2_FRMSIZE_TYPE_STEPWISE ||
|
| + frame_size.type == V4L2_FRMSIZE_TYPE_CONTINUOUS) {
|
| + max_resolution->SetSize(frame_size.stepwise.max_width,
|
| + frame_size.stepwise.max_height);
|
| + min_resolution->SetSize(frame_size.stepwise.min_width,
|
| + frame_size.stepwise.min_height);
|
| + break;
|
| + }
|
| + }
|
| + if (max_resolution->IsEmpty()) {
|
| + max_resolution->SetSize(1920, 1088);
|
| + LOG(ERROR) << "GetSupportedResolution failed to get maximum resolution for "
|
| + << "fourcc " << std::hex << pixelformat
|
| + << ", fall back to " << max_resolution->ToString();
|
| + }
|
| + if (min_resolution->IsEmpty()) {
|
| + min_resolution->SetSize(16, 16);
|
| + LOG(ERROR) << "GetSupportedResolution failed to get minimum resolution for "
|
| + << "fourcc " << std::hex << pixelformat
|
| + << ", fall back to " << min_resolution->ToString();
|
| + }
|
| +}
|
| +
|
| +media::VideoDecodeAccelerator::SupportedProfiles
|
| +V4L2Device::GetSupportedDecodeProfiles(const size_t num_formats,
|
| + const uint32_t pixelformats[]) {
|
| + DCHECK_EQ(type_, kDecoder);
|
| + media::VideoDecodeAccelerator::SupportedProfiles profiles;
|
| + media::VideoDecodeAccelerator::SupportedProfile profile;
|
| + v4l2_fmtdesc fmtdesc;
|
| + memset(&fmtdesc, 0, sizeof(fmtdesc));
|
| + fmtdesc.type = V4L2_BUF_TYPE_VIDEO_OUTPUT_MPLANE;
|
| +
|
| + for (; Ioctl(VIDIOC_ENUM_FMT, &fmtdesc) == 0; ++fmtdesc.index) {
|
| + if (std::find(pixelformats, pixelformats + num_formats,
|
| + fmtdesc.pixelformat) == pixelformats + num_formats)
|
| + continue;
|
| + int min_profile, max_profile;
|
| + switch (fmtdesc.pixelformat) {
|
| + case V4L2_PIX_FMT_H264:
|
| + case V4L2_PIX_FMT_H264_SLICE:
|
| + min_profile = media::H264PROFILE_MIN;
|
| + max_profile = media::H264PROFILE_MAX;
|
| + break;
|
| + case V4L2_PIX_FMT_VP8:
|
| + case V4L2_PIX_FMT_VP8_FRAME:
|
| + min_profile = media::VP8PROFILE_MIN;
|
| + max_profile = media::VP8PROFILE_MAX;
|
| + break;
|
| + case V4L2_PIX_FMT_VP9:
|
| + min_profile = media::VP9PROFILE_MIN;
|
| + max_profile = media::VP9PROFILE_MAX;
|
| + break;
|
| + default:
|
| + NOTREACHED() << "Unhandled pixelformat " << std::hex
|
| + << fmtdesc.pixelformat;
|
| + return profiles;
|
| + }
|
| + GetSupportedResolution(fmtdesc.pixelformat, &profile.min_resolution,
|
| + &profile.max_resolution);
|
| + for (int media_profile = min_profile; media_profile <= max_profile;
|
| + ++media_profile) {
|
| + profile.profile = static_cast<media::VideoCodecProfile>(media_profile);
|
| + profiles.push_back(profile);
|
| + }
|
| + }
|
| + return profiles;
|
| +}
|
| +
|
| } // namespace content
|
|
|