diff options
author | Mark Thompson <sw@jkqxz.net> | 2018-09-18 23:30:57 +0100 |
---|---|---|
committer | Mark Thompson <sw@jkqxz.net> | 2018-09-23 14:42:34 +0100 |
commit | cc549149d3bcb653976193eb56aedb3589e7347e (patch) | |
tree | 22934d63c6a1fb09f6fbdfbc8e3e2560f291fd32 /libavcodec/tests | |
parent | 40724026b7055c9951b579393e2c4178598f1194 (diff) | |
download | ffmpeg-cc549149d3bcb653976193eb56aedb3589e7347e.tar.gz |
lavc/h264: Add common code for level handling
Including a unit test.
Diffstat (limited to 'libavcodec/tests')
-rw-r--r-- | libavcodec/tests/.gitignore | 1 | ||||
-rw-r--r-- | libavcodec/tests/h264_levels.c | 183 |
2 files changed, 184 insertions, 0 deletions
diff --git a/libavcodec/tests/.gitignore b/libavcodec/tests/.gitignore index 350fb2990c..73945a7c82 100644 --- a/libavcodec/tests/.gitignore +++ b/libavcodec/tests/.gitignore @@ -7,6 +7,7 @@ /fft-fixed /fft-fixed32 /golomb +/h264_levels /htmlsubtitles /iirfilter /imgconvert diff --git a/libavcodec/tests/h264_levels.c b/libavcodec/tests/h264_levels.c new file mode 100644 index 0000000000..794517eb6c --- /dev/null +++ b/libavcodec/tests/h264_levels.c @@ -0,0 +1,183 @@ +/* + * This file is part of FFmpeg. + * + * FFmpeg is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2.1 of the License, or (at your option) any later version. + * + * FFmpeg is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with FFmpeg; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + */ + +#include "libavutil/common.h" +#include "libavcodec/h264_levels.h" + +static const struct { + int width; + int height; + int level_idc; +} test_sizes[] = { + // First level usable at some standard sizes. + // (From H.264 table A-6.) + { 176, 144, 10 }, // QCIF + { 352, 288, 11 }, // CIF + { 640, 480, 22 }, // VGA + { 720, 480, 22 }, // NTSC + { 720, 576, 22 }, // PAL + { 800, 600, 31 }, // SVGA + { 1280, 720, 31 }, // 720p + { 1280, 1024, 32 }, // SXGA + { 1920, 1080, 40 }, // 1080p + { 2048, 1080, 42 }, // 2Kx1080 + { 2048, 1536, 50 }, // 4XGA + { 3840, 2160, 51 }, // 4K + { 7680, 4320, 60 }, // 8K + + // Overly wide or tall sizes. + { 1, 256, 10 }, + { 1, 512, 11 }, + { 1, 1024, 21 }, + { 1, 1808, 22 }, + { 1, 1824, 31 }, + { 256, 1, 10 }, + { 512, 1, 11 }, + { 1024, 1, 21 }, + { 1808, 1, 22 }, + { 1824, 1, 31 }, + { 512, 4096, 40 }, + { 256, 4112, 42 }, + { 8688, 1024, 51 }, + { 8704, 512, 60 }, + { 16880, 1, 60 }, + { 16896, 1, 0 }, +}; + +static const struct { + int width; + int height; + int dpb_size; + int level_idc; +} test_dpb[] = { + // First level usable for some DPB sizes. + // (From H.264 table A-7.) + { 176, 144, 4, 10 }, + { 176, 144, 8, 11 }, + { 176, 144, 16, 12 }, + { 1280, 720, 1, 31 }, + { 1280, 720, 5, 31 }, + { 1280, 720, 9, 40 }, + { 1280, 720, 10, 50 }, + { 1920, 1080, 1, 40 }, + { 1920, 1080, 5, 50 }, + { 1920, 1080, 13, 50 }, + { 1920, 1080, 14, 51 }, + { 3840, 2160, 5, 51 }, + { 3840, 2160, 6, 60 }, + { 3840, 2160, 16, 60 }, + { 7680, 4320, 5, 60 }, + { 7680, 4320, 6, 0 }, +}; + +static const struct { + int64_t bitrate; + int profile_idc; + int level_idc; +} test_bitrate[] = { + // Values where profile affects level at a given bitrate. + { 2500000, 77, 21 }, + { 2500000, 100, 20 }, + { 2500000, 244, 13 }, + { 100000000, 77, 50 }, + { 100000000, 100, 50 }, + { 100000000, 244, 41 }, + { 999999999, 77, 0 }, + { 999999999, 100, 62 }, + // Check level 1b. + { 32 * 1200, 66, 10 }, + { 32 * 1500, 100, 10 }, + { 96 * 1200, 66, 10 }, + { 96 * 1500, 100, 9 }, + { 144 * 1200, 66, 11 }, + { 144 * 1500, 100, 11 }, +}; + +static const struct { + const char *name; + int profile_idc; + int64_t bitrate; + int width; + int height; + int dpb_frames; + int level_idc; +} test_all[] = { + { "Bluray 1080p 40Mb/s", 100, 40000000, 1920, 1080, 4, 41 }, + { "Bluray 1080p 24Mb/s", 100, 24000000, 1920, 1080, 4, 40 }, + { "Bluray 720p 40Mb/s", 100, 40000000, 1280, 720, 6, 41 }, + { "Bluray 720p 24Mb/s", 100, 24000000, 1280, 720, 6, 40 }, + { "Bluray PAL 40Mb/s", 100, 40000000, 720, 576, 6, 41 }, + { "Bluray PAL 24Mb/s", 100, 24000000, 720, 576, 6, 32 }, + { "Bluray PAL 16Mb/s", 100, 16800000, 720, 576, 6, 31 }, + { "Bluray PAL 12Mb/s", 100, 12000000, 720, 576, 5, 30 }, + { "Bluray NTSC 40Mb/s", 100, 40000000, 720, 480, 6, 41 }, + { "Bluray NTSC 24Mb/s", 100, 24000000, 720, 480, 6, 32 }, + { "Bluray NTSC 16Mb/s", 100, 16800000, 720, 480, 6, 31 }, + { "Bluray NTSC 12Mb/s", 100, 12000000, 720, 480, 6, 30 }, +}; + +int main(void) +{ + const H264LevelDescriptor *level; + int i; + +#define CHECK(expected, format, ...) do { \ + if (expected ? (!level || level->level_idc != expected) \ + : !!level) { \ + av_log(NULL, AV_LOG_ERROR, "Incorrect level for " \ + format ": expected %d, got %d.\n", __VA_ARGS__, \ + expected, level ? level->level_idc : -1); \ + return 1; \ + } \ + } while (0) + + for (i = 0; i < FF_ARRAY_ELEMS(test_sizes); i++) { + level = ff_h264_guess_level(0, 0, test_sizes[i].width, + test_sizes[i].height, 0); + CHECK(test_sizes[i].level_idc, "size %dx%d", + test_sizes[i].width, test_sizes[i].height); + } + + for (i = 0; i < FF_ARRAY_ELEMS(test_dpb); i++) { + level = ff_h264_guess_level(0, 0, test_dpb[i].width, + test_dpb[i].height, + test_dpb[i].dpb_size); + CHECK(test_dpb[i].level_idc, "size %dx%d dpb %d", + test_dpb[i].width, test_dpb[i].height, + test_dpb[i].dpb_size); + } + + for (i = 0; i < FF_ARRAY_ELEMS(test_bitrate); i++) { + level = ff_h264_guess_level(test_bitrate[i].profile_idc, + test_bitrate[i].bitrate, + 0, 0, 0); + CHECK(test_bitrate[i].level_idc, "bitrate %"PRId64" profile %d", + test_bitrate[i].bitrate, test_bitrate[i].profile_idc); + } + + for (i = 0; i < FF_ARRAY_ELEMS(test_all); i++) { + level = ff_h264_guess_level(test_all[i].profile_idc, + test_all[i].bitrate, + test_all[i].width, + test_all[i].height, + test_all[i].dpb_frames); + CHECK(test_all[i].level_idc, "%s", test_all[i].name); + } + + return 0; +} |