1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * This file is part of FFmpeg.
3cabdff1aSopenharmony_ci *
4cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
5cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
6cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
7cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
8cabdff1aSopenharmony_ci *
9cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
10cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
11cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12cabdff1aSopenharmony_ci * Lesser General Public License for more details.
13cabdff1aSopenharmony_ci *
14cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
15cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
16cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17cabdff1aSopenharmony_ci */
18cabdff1aSopenharmony_ci
19cabdff1aSopenharmony_ci#include "h265_profile_level.h"
20cabdff1aSopenharmony_ci
21cabdff1aSopenharmony_ci
22cabdff1aSopenharmony_cistatic const H265LevelDescriptor h265_levels[] = {
23cabdff1aSopenharmony_ci    // Name             CpbFactor-Main    MaxSliceSegmentsPerPicture
24cabdff1aSopenharmony_ci    // |  level_idc            | CpbFactor-High           MaxLumaSr      BrFactor-High
25cabdff1aSopenharmony_ci    // |      |   MaxLumaPs    |       |      | MaxTileRows   |   BrFactor-Main | MinCr-Main
26cabdff1aSopenharmony_ci    // |      |      |         |       |      |   | MaxTileCols         |       |    |  MinCr-High
27cabdff1aSopenharmony_ci    { "1",    30,    36864,    350,      0,  16,  1,  1,     552960,    128,      0, 2, 2 },
28cabdff1aSopenharmony_ci    { "2",    60,   122880,   1500,      0,  16,  1,  1,    3686400,   1500,      0, 2, 2 },
29cabdff1aSopenharmony_ci    { "2.1",  63,   245760,   3000,      0,  20,  1,  1,    7372800,   3000,      0, 2, 2 },
30cabdff1aSopenharmony_ci    { "3",    90,   552960,   6000,      0,  30,  2,  2,   16588800,   6000,      0, 2, 2 },
31cabdff1aSopenharmony_ci    { "3.1",  93,   983040,  10000,      0,  40,  3,  3,   33177600,  10000,      0, 2, 2 },
32cabdff1aSopenharmony_ci    { "4",   120,  2228224,  12000,  30000,  75,  5,  5,   66846720,  12000,  30000, 4, 4 },
33cabdff1aSopenharmony_ci    { "4.1", 123,  2228224,  20000,  50000,  75,  5,  5,  133693440,  20000,  50000, 4, 4 },
34cabdff1aSopenharmony_ci    { "5",   150,  8912896,  25000, 100000, 200, 11, 10,  267386880,  25000, 100000, 6, 4 },
35cabdff1aSopenharmony_ci    { "5.1", 153,  8912896,  40000, 160000, 200, 11, 10,  534773760,  40000, 160000, 8, 4 },
36cabdff1aSopenharmony_ci    { "5.2", 156,  8912896,  60000, 240000, 200, 11, 10, 1069547520,  60000, 240000, 8, 4 },
37cabdff1aSopenharmony_ci    { "6",   180, 35651584,  60000, 240000, 600, 22, 20, 1069547520,  60000, 240000, 8, 4 },
38cabdff1aSopenharmony_ci    { "6.1", 183, 35651584, 120000, 480000, 600, 22, 20, 2139095040, 120000, 480000, 8, 4 },
39cabdff1aSopenharmony_ci    { "6.2", 186, 35651584, 240000, 800000, 600, 22, 20, 4278190080, 240000, 800000, 6, 4 },
40cabdff1aSopenharmony_ci};
41cabdff1aSopenharmony_ci
42cabdff1aSopenharmony_cistatic const H265ProfileDescriptor h265_profiles[] = {
43cabdff1aSopenharmony_ci    // profile_idc   8bit       one-picture
44cabdff1aSopenharmony_ci    //   HT-profile  | 422chroma    | lower-bit-rate
45cabdff1aSopenharmony_ci    //   |  14bit    |  | 420chroma |  | CpbVclFactor     MinCrScaleFactor
46cabdff1aSopenharmony_ci    //   |  |  12bit |  |  | monochrome|    | CpbNalFactor    | maxDpbPicBuf
47cabdff1aSopenharmony_ci    //   |  |  |  10bit |  |  | intra  |    |     | FormatCapabilityFactor
48cabdff1aSopenharmony_ci    { "Monochrome", //  |  |  |  |  |  |    |     |     |     |   |
49cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 1, 1, 1, 1, 0, 0, 1,  667,  733, 1.000, 1.0, 6 },
50cabdff1aSopenharmony_ci    { "Monochrome 10",
51cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 1, 1, 1, 0, 0, 1,  833,  917, 1.250, 1.0, 6 },
52cabdff1aSopenharmony_ci    { "Monochrome 12",
53cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 1, 1, 1, 0, 0, 1, 1000, 1100, 1.500, 1.0, 6 },
54cabdff1aSopenharmony_ci    { "Monochrome 16",
55cabdff1aSopenharmony_ci      4, 0, 2, 0, 0, 0, 1, 1, 1, 0, 0, 1, 1333, 1467, 2.000, 1.0, 6 },
56cabdff1aSopenharmony_ci    { "Main",
57cabdff1aSopenharmony_ci      1, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1000, 1100, 1.500, 1.0, 6 },
58cabdff1aSopenharmony_ci    { "Screen-Extended Main",
59cabdff1aSopenharmony_ci      9, 0, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1000, 1100, 1.500, 1.0, 7 },
60cabdff1aSopenharmony_ci    { "Main 10",
61cabdff1aSopenharmony_ci      2, 0, 2, 2, 2, 2, 2, 2, 2, 2, 0, 2, 1000, 1100, 1.875, 1.0, 6 },
62cabdff1aSopenharmony_ci    { "Screen-Extended Main 10",
63cabdff1aSopenharmony_ci      9, 0, 1, 1, 1, 0, 1, 1, 0, 0, 0, 1, 1000, 1100, 1.875, 1.0, 7 },
64cabdff1aSopenharmony_ci    { "Main 12",
65cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 1, 1, 0, 0, 0, 1, 1500, 1650, 2.250, 1.0, 6 },
66cabdff1aSopenharmony_ci    { "Main Still Picture",
67cabdff1aSopenharmony_ci      3, 0, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1000, 1100, 1.500, 1.0, 6 },
68cabdff1aSopenharmony_ci    { "Main 10 Still Picture",
69cabdff1aSopenharmony_ci      2, 0, 2, 2, 2, 2, 2, 2, 2, 2, 1, 2, 1000, 1100, 1.875, 1.0, 6 },
70cabdff1aSopenharmony_ci    { "Main 4:2:2 10",
71cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 1, 0, 0, 0, 0, 1, 1667, 1833, 2.500, 0.5, 6 },
72cabdff1aSopenharmony_ci    { "Main 4:2:2 12",
73cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 1, 0, 0, 0, 0, 1, 2000, 2200, 3.000, 0.5, 6 },
74cabdff1aSopenharmony_ci    { "Main 4:4:4",
75cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 1, 0, 0, 0, 0, 0, 1, 2000, 2200, 3.000, 0.5, 6 },
76cabdff1aSopenharmony_ci    { "High Throughput 4:4:4",
77cabdff1aSopenharmony_ci      5, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 2000, 2200, 3.000, 0.5, 6 },
78cabdff1aSopenharmony_ci    { "Screen-Extended Main 4:4:4",
79cabdff1aSopenharmony_ci      9, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 2000, 2200, 3.000, 0.5, 7 },
80cabdff1aSopenharmony_ci    { "Screen-Extended High Throughput 4:4:4",
81cabdff1aSopenharmony_ci      9, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 1, 2000, 2200, 3.000, 0.5, 7 },
82cabdff1aSopenharmony_ci    { "Main 4:4:4 10",
83cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 0, 0, 0, 0, 0, 1, 2500, 2750, 3.750, 0.5, 6 },
84cabdff1aSopenharmony_ci    { "High Throughput 4:4:4 10",
85cabdff1aSopenharmony_ci      5, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 1, 2500, 2750, 3.750, 0.5, 6 },
86cabdff1aSopenharmony_ci    { "Screen-Extended Main 4:4:4 10",
87cabdff1aSopenharmony_ci      9, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 1, 2500, 2750, 3.750, 0.5, 7 },
88cabdff1aSopenharmony_ci    { "Screen-Extended High Throughput 4:4:4 10",
89cabdff1aSopenharmony_ci      9, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 1, 2500, 2750, 3.750, 0.5, 7 },
90cabdff1aSopenharmony_ci    { "Main 4:4:4 12",
91cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 0, 0, 0, 0, 0, 1, 3000, 3300, 4.500, 0.5, 6 },
92cabdff1aSopenharmony_ci    { "High Throughput 4:4:4 14",
93cabdff1aSopenharmony_ci      5, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 3500, 3850, 5.250, 0.5, 6 },
94cabdff1aSopenharmony_ci    { "Screen-Extended High Throughput 4:4:4 14",
95cabdff1aSopenharmony_ci      9, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 1, 3500, 3850, 5.250, 0.5, 7 },
96cabdff1aSopenharmony_ci    { "Main Intra",
97cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 1, 1, 1, 0, 1, 0, 2, 1000, 1100, 1.500, 1.0, 6 },
98cabdff1aSopenharmony_ci    { "Main 10 Intra",
99cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 1, 1, 0, 1, 0, 2, 1000, 1100, 1.875, 1.0, 6 },
100cabdff1aSopenharmony_ci    { "Main 12 Intra",
101cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 1, 1, 0, 1, 0, 2, 1500, 1650, 2.250, 1.0, 6 },
102cabdff1aSopenharmony_ci    { "Main 4:2:2 10 Intra",
103cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 1, 0, 0, 1, 0, 2, 1667, 1833, 2.500, 0.5, 6 },
104cabdff1aSopenharmony_ci    { "Main 4:2:2 12 Intra",
105cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 1, 0, 0, 1, 0, 2, 2000, 2200, 3.000, 0.5, 6 },
106cabdff1aSopenharmony_ci    { "Main 4:4:4 Intra",
107cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 1, 0, 0, 0, 1, 0, 2, 2000, 2200, 3.000, 0.5, 6 },
108cabdff1aSopenharmony_ci    { "Main 4:4:4 10 Intra",
109cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 0, 0, 0, 0, 1, 0, 2, 2500, 2750, 3.750, 0.5, 6 },
110cabdff1aSopenharmony_ci    { "Main 4:4:4 12 Intra",
111cabdff1aSopenharmony_ci      4, 0, 2, 1, 0, 0, 0, 0, 0, 1, 0, 2, 3000, 3300, 4.500, 0.5, 6 },
112cabdff1aSopenharmony_ci    { "Main 4:4:4 16 Intra",
113cabdff1aSopenharmony_ci      4, 0, 2, 0, 0, 0, 0, 0, 0, 1, 0, 2, 4000, 4400, 6.000, 0.5, 6 },
114cabdff1aSopenharmony_ci    { "Main 4:4:4 Still Picture",
115cabdff1aSopenharmony_ci      4, 0, 2, 1, 1, 1, 0, 0, 0, 1, 1, 2, 2000, 2200, 3.000, 0.5, 6 },
116cabdff1aSopenharmony_ci    { "Main 4:4:4 16 Still Picture",
117cabdff1aSopenharmony_ci      4, 0, 2, 0, 0, 0, 0, 0, 0, 1, 1, 2, 4000, 4400, 6.000, 0.5, 6 },
118cabdff1aSopenharmony_ci    { "High Throughput 4:4:4 16 Intra",
119cabdff1aSopenharmony_ci      5, 1, 0, 0, 0, 0, 0, 0, 0, 1, 0, 2, 4000, 4400, 6.000, 0.5, 6 },
120cabdff1aSopenharmony_ci};
121cabdff1aSopenharmony_ci
122cabdff1aSopenharmony_ci
123cabdff1aSopenharmony_ciconst H265ProfileDescriptor *ff_h265_get_profile(const H265RawProfileTierLevel *ptl)
124cabdff1aSopenharmony_ci{
125cabdff1aSopenharmony_ci    int i;
126cabdff1aSopenharmony_ci
127cabdff1aSopenharmony_ci    if (ptl->general_profile_space)
128cabdff1aSopenharmony_ci        return NULL;
129cabdff1aSopenharmony_ci
130cabdff1aSopenharmony_ci    for (i = 0; i < FF_ARRAY_ELEMS(h265_profiles); i++) {
131cabdff1aSopenharmony_ci        const H265ProfileDescriptor *profile = &h265_profiles[i];
132cabdff1aSopenharmony_ci
133cabdff1aSopenharmony_ci        if (ptl->general_profile_idc &&
134cabdff1aSopenharmony_ci            ptl->general_profile_idc != profile->profile_idc)
135cabdff1aSopenharmony_ci            continue;
136cabdff1aSopenharmony_ci        if (!ptl->general_profile_compatibility_flag[profile->profile_idc])
137cabdff1aSopenharmony_ci            continue;
138cabdff1aSopenharmony_ci
139cabdff1aSopenharmony_ci#define check_flag(name) \
140cabdff1aSopenharmony_ci        if (profile->name < 2) { \
141cabdff1aSopenharmony_ci            if (profile->name != ptl->general_ ## name ## _constraint_flag) \
142cabdff1aSopenharmony_ci                continue; \
143cabdff1aSopenharmony_ci        }
144cabdff1aSopenharmony_ci        check_flag(max_14bit);
145cabdff1aSopenharmony_ci        check_flag(max_12bit);
146cabdff1aSopenharmony_ci        check_flag(max_10bit);
147cabdff1aSopenharmony_ci        check_flag(max_8bit);
148cabdff1aSopenharmony_ci        check_flag(max_422chroma);
149cabdff1aSopenharmony_ci        check_flag(max_420chroma);
150cabdff1aSopenharmony_ci        check_flag(max_monochrome);
151cabdff1aSopenharmony_ci        check_flag(intra);
152cabdff1aSopenharmony_ci        check_flag(one_picture_only);
153cabdff1aSopenharmony_ci        check_flag(lower_bit_rate);
154cabdff1aSopenharmony_ci#undef check_flag
155cabdff1aSopenharmony_ci
156cabdff1aSopenharmony_ci        return profile;
157cabdff1aSopenharmony_ci    }
158cabdff1aSopenharmony_ci
159cabdff1aSopenharmony_ci    return NULL;
160cabdff1aSopenharmony_ci}
161cabdff1aSopenharmony_ci
162cabdff1aSopenharmony_ciconst H265LevelDescriptor *ff_h265_guess_level(const H265RawProfileTierLevel *ptl,
163cabdff1aSopenharmony_ci                                               int64_t bitrate,
164cabdff1aSopenharmony_ci                                               int width, int height,
165cabdff1aSopenharmony_ci                                               int slice_segments,
166cabdff1aSopenharmony_ci                                               int tile_rows, int tile_cols,
167cabdff1aSopenharmony_ci                                               int max_dec_pic_buffering)
168cabdff1aSopenharmony_ci{
169cabdff1aSopenharmony_ci    const H265ProfileDescriptor *profile;
170cabdff1aSopenharmony_ci    int pic_size, tier_flag, lbr_flag, hbr_factor;
171cabdff1aSopenharmony_ci    int i;
172cabdff1aSopenharmony_ci
173cabdff1aSopenharmony_ci    if (ptl)
174cabdff1aSopenharmony_ci        profile = ff_h265_get_profile(ptl);
175cabdff1aSopenharmony_ci    else
176cabdff1aSopenharmony_ci        profile = NULL;
177cabdff1aSopenharmony_ci    if (!profile) {
178cabdff1aSopenharmony_ci        // Default to using multiplication factors for Main profile.
179cabdff1aSopenharmony_ci        profile = &h265_profiles[4];
180cabdff1aSopenharmony_ci    }
181cabdff1aSopenharmony_ci
182cabdff1aSopenharmony_ci    pic_size = width * height;
183cabdff1aSopenharmony_ci
184cabdff1aSopenharmony_ci    if (ptl) {
185cabdff1aSopenharmony_ci        tier_flag = ptl->general_tier_flag;
186cabdff1aSopenharmony_ci        lbr_flag  = ptl->general_lower_bit_rate_constraint_flag;
187cabdff1aSopenharmony_ci    } else {
188cabdff1aSopenharmony_ci        tier_flag = 0;
189cabdff1aSopenharmony_ci        lbr_flag  = profile->lower_bit_rate > 0;
190cabdff1aSopenharmony_ci    }
191cabdff1aSopenharmony_ci    if (profile->profile_idc == 1 || profile->profile_idc == 2) {
192cabdff1aSopenharmony_ci        hbr_factor = 1;
193cabdff1aSopenharmony_ci    } else if (profile->high_throughput) {
194cabdff1aSopenharmony_ci        if (profile->intra)
195cabdff1aSopenharmony_ci            hbr_factor = 24 - 12 * lbr_flag;
196cabdff1aSopenharmony_ci        else
197cabdff1aSopenharmony_ci            hbr_factor = 6;
198cabdff1aSopenharmony_ci    } else {
199cabdff1aSopenharmony_ci        hbr_factor = 2 - lbr_flag;
200cabdff1aSopenharmony_ci    }
201cabdff1aSopenharmony_ci
202cabdff1aSopenharmony_ci    for (i = 0; i < FF_ARRAY_ELEMS(h265_levels); i++) {
203cabdff1aSopenharmony_ci        const H265LevelDescriptor *level = &h265_levels[i];
204cabdff1aSopenharmony_ci        int max_br, max_dpb_size;
205cabdff1aSopenharmony_ci
206cabdff1aSopenharmony_ci        if (tier_flag && !level->max_br_high)
207cabdff1aSopenharmony_ci            continue;
208cabdff1aSopenharmony_ci
209cabdff1aSopenharmony_ci        if (pic_size > level->max_luma_ps)
210cabdff1aSopenharmony_ci            continue;
211cabdff1aSopenharmony_ci        if (width  * width  > 8 * level->max_luma_ps)
212cabdff1aSopenharmony_ci            continue;
213cabdff1aSopenharmony_ci        if (height * height > 8 * level->max_luma_ps)
214cabdff1aSopenharmony_ci            continue;
215cabdff1aSopenharmony_ci
216cabdff1aSopenharmony_ci        if (slice_segments > level->max_slice_segments_per_picture)
217cabdff1aSopenharmony_ci            continue;
218cabdff1aSopenharmony_ci        if (tile_rows > level->max_tile_rows)
219cabdff1aSopenharmony_ci            continue;
220cabdff1aSopenharmony_ci        if (tile_cols > level->max_tile_cols)
221cabdff1aSopenharmony_ci            continue;
222cabdff1aSopenharmony_ci
223cabdff1aSopenharmony_ci        if (tier_flag)
224cabdff1aSopenharmony_ci            max_br = level->max_br_high;
225cabdff1aSopenharmony_ci        else
226cabdff1aSopenharmony_ci            max_br = level->max_br_main;
227cabdff1aSopenharmony_ci        if (!max_br)
228cabdff1aSopenharmony_ci            continue;
229cabdff1aSopenharmony_ci        if (bitrate > (int64_t)profile->cpb_nal_factor * hbr_factor * max_br)
230cabdff1aSopenharmony_ci            continue;
231cabdff1aSopenharmony_ci
232cabdff1aSopenharmony_ci        if (pic_size <= (level->max_luma_ps >> 2))
233cabdff1aSopenharmony_ci            max_dpb_size = FFMIN(4 * profile->max_dpb_pic_buf, 16);
234cabdff1aSopenharmony_ci        else if (pic_size <= (level->max_luma_ps >> 1))
235cabdff1aSopenharmony_ci            max_dpb_size = FFMIN(2 * profile->max_dpb_pic_buf, 16);
236cabdff1aSopenharmony_ci        else if (pic_size <= (3 * level->max_luma_ps >> 2))
237cabdff1aSopenharmony_ci            max_dpb_size = FFMIN(4 * profile->max_dpb_pic_buf / 3, 16);
238cabdff1aSopenharmony_ci        else
239cabdff1aSopenharmony_ci            max_dpb_size = profile->max_dpb_pic_buf;
240cabdff1aSopenharmony_ci        if (max_dec_pic_buffering > max_dpb_size)
241cabdff1aSopenharmony_ci            continue;
242cabdff1aSopenharmony_ci
243cabdff1aSopenharmony_ci        return level;
244cabdff1aSopenharmony_ci    }
245cabdff1aSopenharmony_ci
246cabdff1aSopenharmony_ci    return NULL;
247cabdff1aSopenharmony_ci}
248