1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
|
#include "handbrake/handbrake.h"
#include "handbrake/hbffmpeg.h"
static int get_frame_type(int type)
{
switch (type)
{
case AV_PICTURE_TYPE_B:
return HB_FRAME_B;
case AV_PICTURE_TYPE_S:
case AV_PICTURE_TYPE_P:
case AV_PICTURE_TYPE_SP:
return HB_FRAME_P;
case AV_PICTURE_TYPE_BI:
case AV_PICTURE_TYPE_SI:
case AV_PICTURE_TYPE_I:
default:
return HB_FRAME_I;
}
}
void hb_video_buffer_to_avframe(AVFrame *frame, hb_buffer_t * buf)
{
frame->data[0] = buf->plane[0].data;
frame->data[1] = buf->plane[1].data;
frame->data[2] = buf->plane[2].data;
frame->linesize[0] = buf->plane[0].stride;
frame->linesize[1] = buf->plane[1].stride;
frame->linesize[2] = buf->plane[2].stride;
frame->pts = buf->s.start;
frame->reordered_opaque = buf->s.duration;
frame->width = buf->f.width;
frame->height = buf->f.height;
frame->format = buf->f.fmt;
frame->interlaced_frame = !!buf->s.combed;
frame->top_field_first = !!(buf->s.flags & PIC_FLAG_TOP_FIELD_FIRST);
frame->format = buf->f.fmt;
frame->color_primaries = hb_colr_pri_hb_to_ff(buf->f.color_prim);
frame->color_trc = hb_colr_tra_hb_to_ff(buf->f.color_transfer);
frame->colorspace = hb_colr_mat_hb_to_ff(buf->f.color_matrix);
frame->color_range = buf->f.color_range;
}
void hb_avframe_set_video_buffer_flags(hb_buffer_t * buf, AVFrame *frame,
AVRational time_base)
{
if (buf == NULL || frame == NULL)
{
return;
}
buf->s.start = av_rescale_q(frame->pts, time_base, (AVRational){1, 90000});
buf->s.duration = frame->reordered_opaque;
if (frame->top_field_first)
{
buf->s.flags |= PIC_FLAG_TOP_FIELD_FIRST;
}
if (!frame->interlaced_frame)
{
buf->s.flags |= PIC_FLAG_PROGRESSIVE_FRAME;
}
else
{
buf->s.combed = HB_COMB_HEAVY;
}
if (frame->repeat_pict == 1)
{
buf->s.flags |= PIC_FLAG_REPEAT_FIRST_FIELD;
}
if (frame->repeat_pict == 2)
{
buf->s.flags |= PIC_FLAG_REPEAT_FRAME;
}
buf->s.frametype = get_frame_type(frame->pict_type);
buf->f.fmt = frame->format;
buf->f.color_prim = hb_colr_pri_ff_to_hb(frame->color_primaries);
buf->f.color_transfer = hb_colr_tra_ff_to_hb(frame->color_trc);
buf->f.color_matrix = hb_colr_mat_ff_to_hb(frame->colorspace);
buf->f.color_range = frame->color_range;
}
hb_buffer_t * hb_avframe_to_video_buffer(AVFrame *frame, AVRational time_base)
{
hb_buffer_t * buf;
buf = hb_frame_buffer_init(frame->format, frame->width, frame->height);
if (buf == NULL)
{
return NULL;
}
hb_avframe_set_video_buffer_flags(buf, frame, time_base);
int pp;
for (pp = 0; pp <= buf->f.max_plane; pp++)
{
int yy;
int stride = buf->plane[pp].stride;
int height = buf->plane[pp].height;
int linesize = frame->linesize[pp];
int size = linesize < stride ? linesize : stride;
uint8_t * dst = buf->plane[pp].data;
uint8_t * src = frame->data[pp];
for (yy = 0; yy < height; yy++)
{
memcpy(dst, src, size);
dst += stride;
src += linesize;
}
}
return buf;
}
static int handle_jpeg(enum AVPixelFormat *format)
{
switch (*format)
{
case AV_PIX_FMT_YUVJ420P: *format = AV_PIX_FMT_YUV420P; return 1;
case AV_PIX_FMT_YUVJ422P: *format = AV_PIX_FMT_YUV422P; return 1;
case AV_PIX_FMT_YUVJ444P: *format = AV_PIX_FMT_YUV444P; return 1;
case AV_PIX_FMT_YUVJ440P: *format = AV_PIX_FMT_YUV440P; return 1;
default: return 0;
}
}
struct SwsContext*
hb_sws_get_context(int srcW, int srcH, enum AVPixelFormat srcFormat,
int dstW, int dstH, enum AVPixelFormat dstFormat,
int flags, int colorspace)
{
struct SwsContext * ctx;
ctx = sws_alloc_context();
if ( ctx )
{
int srcRange, dstRange;
srcRange = handle_jpeg(&srcFormat);
dstRange = handle_jpeg(&dstFormat);
flags |= SWS_FULL_CHR_H_INT | SWS_FULL_CHR_H_INP;
av_opt_set_int(ctx, "srcw", srcW, 0);
av_opt_set_int(ctx, "srch", srcH, 0);
av_opt_set_int(ctx, "src_range", srcRange, 0);
av_opt_set_int(ctx, "src_format", srcFormat, 0);
av_opt_set_int(ctx, "dstw", dstW, 0);
av_opt_set_int(ctx, "dsth", dstH, 0);
av_opt_set_int(ctx, "dst_range", dstRange, 0);
av_opt_set_int(ctx, "dst_format", dstFormat, 0);
av_opt_set_int(ctx, "sws_flags", flags, 0);
sws_setColorspaceDetails( ctx,
sws_getCoefficients( colorspace ), // src colorspace
srcRange, // src range 0 = MPG, 1 = JPG
sws_getCoefficients( colorspace ), // dst colorspace
dstRange, // dst range 0 = MPG, 1 = JPG
0, // brightness
1 << 16, // contrast
1 << 16 ); // saturation
if (sws_init_context(ctx, NULL, NULL) < 0) {
hb_error("Cannot initialize resampling context");
sws_freeContext(ctx);
ctx = NULL;
}
}
return ctx;
}
int hb_sws_get_colorspace(int color_matrix)
{
int color_space = SWS_CS_DEFAULT;
switch (color_matrix)
{
case HB_COLR_MAT_SMPTE170M:
color_space = SWS_CS_ITU601;
break;
case HB_COLR_MAT_SMPTE240M:
color_space = SWS_CS_SMPTE240M;
break;
case HB_COLR_MAT_BT709:
color_space = SWS_CS_ITU709;
break;
case HB_COLR_MAT_BT2020_CL:
case HB_COLR_MAT_BT2020_NCL:
color_space = SWS_CS_BT2020;
break;
default:
break;
}
return color_space;
}
int hb_colr_pri_hb_to_ff(int colr_prim)
{
switch (colr_prim)
{
case HB_COLR_PRI_BT709:
return AVCOL_PRI_BT709;
case HB_COLR_PRI_EBUTECH:
return AVCOL_PRI_BT470BG;
case HB_COLR_PRI_BT470M:
return AVCOL_PRI_BT470M;
case HB_COLR_PRI_SMPTEC:
return AVCOL_PRI_SMPTE170M;
case HB_COLR_PRI_SMPTE240M:
return AVCOL_PRI_SMPTE240M;
case HB_COLR_PRI_BT2020:
return AVCOL_PRI_BT2020;
case HB_COLR_PRI_SMPTE428:
return AVCOL_PRI_SMPTE428;
case HB_COLR_PRI_SMPTE431:
return AVCOL_PRI_SMPTE431;
case HB_COLR_PRI_SMPTE432:
return AVCOL_PRI_SMPTE432;
case HB_COLR_PRI_JEDEC_P22:
return AVCOL_PRI_JEDEC_P22;
default:
case HB_COLR_PRI_UNDEF:
return AVCOL_PRI_UNSPECIFIED;
}
}
int hb_colr_tra_hb_to_ff(int colr_tra)
{
switch (colr_tra)
{
case HB_COLR_TRA_BT709:
return AVCOL_TRC_BT709;
case HB_COLR_TRA_GAMMA22:
return AVCOL_TRC_GAMMA22;
case HB_COLR_TRA_GAMMA28:
return AVCOL_TRC_GAMMA28;
case HB_COLR_TRA_SMPTE170M:
return AVCOL_TRC_SMPTE170M;
case HB_COLR_TRA_SMPTE240M:
return AVCOL_TRC_SMPTE240M;
case HB_COLR_TRA_LINEAR:
return AVCOL_TRC_LINEAR;
case HB_COLR_TRA_LOG:
return AVCOL_TRC_LOG;
case HB_COLR_TRA_LOG_SQRT:
return AVCOL_TRC_LOG_SQRT;
case HB_COLR_TRA_IEC61966_2_4:
return AVCOL_TRC_IEC61966_2_4;
case HB_COLR_TRA_BT1361_ECG:
return AVCOL_TRC_BT1361_ECG;
case HB_COLR_TRA_IEC61966_2_1:
return AVCOL_TRC_IEC61966_2_1;
case HB_COLR_TRA_BT2020_10:
return AVCOL_TRC_BT2020_10;
case HB_COLR_TRA_BT2020_12:
return AVCOL_TRC_BT2020_12;
case HB_COLR_TRA_SMPTEST2084:
return AVCOL_TRC_SMPTE2084;
case HB_COLR_TRA_SMPTE428:
return AVCOL_TRC_SMPTE428;
case HB_COLR_TRA_ARIB_STD_B67:
return AVCOL_TRC_ARIB_STD_B67;
default:
case HB_COLR_TRA_UNDEF:
return AVCOL_TRC_UNSPECIFIED;
}
}
int hb_colr_mat_hb_to_ff(int colr_mat)
{
switch (colr_mat)
{
case HB_COLR_MAT_RGB:
return AVCOL_SPC_RGB;
case HB_COLR_MAT_BT709:
return AVCOL_SPC_BT709;
case HB_COLR_MAT_FCC:
return AVCOL_SPC_FCC;
case HB_COLR_MAT_BT470BG:
return AVCOL_SPC_BT470BG;
case HB_COLR_MAT_SMPTE170M:
return AVCOL_SPC_SMPTE170M;
case HB_COLR_MAT_SMPTE240M:
return AVCOL_SPC_SMPTE240M;
case HB_COLR_MAT_YCGCO:
return AVCOL_SPC_YCGCO;
case HB_COLR_MAT_BT2020_NCL:
return AVCOL_SPC_BT2020_NCL;
case HB_COLR_MAT_BT2020_CL:
return AVCOL_SPC_BT2020_CL;
case HB_COLR_MAT_SMPTE2085:
return AVCOL_SPC_SMPTE2085;
case HB_COLR_MAT_CD_NCL:
return AVCOL_SPC_CHROMA_DERIVED_NCL;
case HB_COLR_MAT_CD_CL:
return AVCOL_SPC_CHROMA_DERIVED_CL;
case HB_COLR_MAT_ICTCP:
return AVCOL_SPC_ICTCP;
default:
case HB_COLR_MAT_UNDEF:
return AVCOL_SPC_UNSPECIFIED;
}
}
int hb_colr_pri_ff_to_hb(int colr_prim)
{
switch (colr_prim)
{
case AVCOL_PRI_BT709:
return HB_COLR_PRI_BT709;
case AVCOL_PRI_BT470M:
return HB_COLR_PRI_BT470M;
case AVCOL_PRI_BT470BG:
return HB_COLR_PRI_EBUTECH;
case AVCOL_PRI_SMPTE170M:
return HB_COLR_PRI_SMPTEC;
case AVCOL_PRI_SMPTE240M:
return HB_COLR_PRI_SMPTE240M;
case AVCOL_PRI_FILM:
return HB_COLR_PRI_FILM;
case AVCOL_PRI_BT2020:
return HB_COLR_PRI_BT2020;
case AVCOL_PRI_SMPTE428:
return HB_COLR_PRI_SMPTE428;
case AVCOL_PRI_SMPTE431:
return HB_COLR_PRI_SMPTE431;
case AVCOL_PRI_SMPTE432:
return HB_COLR_PRI_SMPTE432;
case AVCOL_PRI_JEDEC_P22:
return HB_COLR_PRI_JEDEC_P22;
default:
case AVCOL_PRI_RESERVED:
case AVCOL_PRI_RESERVED0:
case AVCOL_PRI_UNSPECIFIED:
return HB_COLR_PRI_UNDEF;
}
}
int hb_colr_tra_ff_to_hb(int colr_tra)
{
switch (colr_tra)
{
case AVCOL_TRC_BT709:
return HB_COLR_TRA_BT709;
case AVCOL_TRC_GAMMA22:
return HB_COLR_TRA_GAMMA22;
case AVCOL_TRC_GAMMA28:
return HB_COLR_TRA_GAMMA28;
case AVCOL_TRC_SMPTE170M:
return HB_COLR_TRA_SMPTE170M;
case AVCOL_TRC_SMPTE240M:
return HB_COLR_TRA_SMPTE240M;
case AVCOL_TRC_LINEAR:
return HB_COLR_TRA_LINEAR;
case AVCOL_TRC_LOG:
return HB_COLR_TRA_LOG;
case AVCOL_TRC_LOG_SQRT:
return HB_COLR_TRA_LOG_SQRT;
case AVCOL_TRC_IEC61966_2_4:
return HB_COLR_TRA_IEC61966_2_4;
case AVCOL_TRC_BT1361_ECG:
return HB_COLR_TRA_BT1361_ECG;
case AVCOL_TRC_IEC61966_2_1:
return HB_COLR_TRA_IEC61966_2_1;
case AVCOL_TRC_BT2020_10:
return HB_COLR_TRA_BT2020_10;
case AVCOL_TRC_BT2020_12:
return HB_COLR_TRA_BT2020_12;
case AVCOL_TRC_SMPTE2084:
return HB_COLR_TRA_SMPTEST2084;
case AVCOL_TRC_SMPTE428:
return HB_COLR_TRA_SMPTE428;
case AVCOL_TRC_ARIB_STD_B67:
return HB_COLR_TRA_ARIB_STD_B67;
default:
case AVCOL_TRC_UNSPECIFIED:
case AVCOL_TRC_RESERVED:
case AVCOL_TRC_RESERVED0:
return HB_COLR_TRA_UNDEF;
}
}
int hb_colr_mat_ff_to_hb(int colr_mat)
{
switch (colr_mat)
{
case AVCOL_SPC_RGB:
return HB_COLR_MAT_RGB;
case AVCOL_SPC_BT709:
return HB_COLR_MAT_BT709;
case AVCOL_SPC_FCC:
return HB_COLR_MAT_FCC;
case AVCOL_SPC_BT470BG:
return HB_COLR_MAT_BT470BG;
case AVCOL_SPC_SMPTE170M:
return HB_COLR_MAT_SMPTE170M;
case AVCOL_SPC_SMPTE240M:
return HB_COLR_MAT_SMPTE240M;
case AVCOL_SPC_YCGCO:
return HB_COLR_MAT_YCGCO;
case AVCOL_SPC_BT2020_NCL:
return HB_COLR_MAT_BT2020_NCL;
case AVCOL_SPC_BT2020_CL:
return HB_COLR_MAT_BT2020_CL;
case AVCOL_SPC_SMPTE2085:
return HB_COLR_MAT_SMPTE2085;
case AVCOL_SPC_CHROMA_DERIVED_NCL:
return HB_COLR_MAT_CD_NCL;
case AVCOL_SPC_CHROMA_DERIVED_CL:
return HB_COLR_MAT_CD_CL;
case AVCOL_SPC_ICTCP:
return HB_COLR_MAT_ICTCP;
default:
case AVCOL_SPC_UNSPECIFIED:
case AVCOL_SPC_RESERVED:
return HB_COLR_MAT_UNDEF;
}
}
uint64_t hb_ff_mixdown_xlat(int hb_mixdown, int *downmix_mode)
{
uint64_t ff_layout = 0;
int mode = AV_MATRIX_ENCODING_NONE;
switch (hb_mixdown)
{
// Passthru
case HB_AMIXDOWN_NONE:
break;
case HB_AMIXDOWN_MONO:
case HB_AMIXDOWN_LEFT:
case HB_AMIXDOWN_RIGHT:
ff_layout = AV_CH_LAYOUT_MONO;
break;
case HB_AMIXDOWN_DOLBY:
ff_layout = AV_CH_LAYOUT_STEREO;
mode = AV_MATRIX_ENCODING_DOLBY;
break;
case HB_AMIXDOWN_DOLBYPLII:
ff_layout = AV_CH_LAYOUT_STEREO;
mode = AV_MATRIX_ENCODING_DPLII;
break;
case HB_AMIXDOWN_STEREO:
ff_layout = AV_CH_LAYOUT_STEREO;
break;
case HB_AMIXDOWN_5POINT1:
ff_layout = AV_CH_LAYOUT_5POINT1;
break;
case HB_AMIXDOWN_6POINT1:
ff_layout = AV_CH_LAYOUT_6POINT1;
break;
case HB_AMIXDOWN_7POINT1:
ff_layout = AV_CH_LAYOUT_7POINT1;
break;
case HB_AMIXDOWN_5_2_LFE:
ff_layout = (AV_CH_LAYOUT_5POINT1_BACK|
AV_CH_FRONT_LEFT_OF_CENTER|
AV_CH_FRONT_RIGHT_OF_CENTER);
break;
default:
ff_layout = AV_CH_LAYOUT_STEREO;
hb_log("hb_ff_mixdown_xlat: unsupported mixdown %d", hb_mixdown);
break;
}
if (downmix_mode != NULL)
*downmix_mode = mode;
return ff_layout;
}
/*
* Set sample format to the request format if supported by the codec.
* The planar/packed variant of the requested format is the next best thing.
*/
void hb_ff_set_sample_fmt(AVCodecContext *context, AVCodec *codec,
enum AVSampleFormat request_sample_fmt)
{
if (context != NULL && codec != NULL &&
codec->type == AVMEDIA_TYPE_AUDIO && codec->sample_fmts != NULL)
{
const enum AVSampleFormat *fmt;
enum AVSampleFormat next_best_fmt;
next_best_fmt = (av_sample_fmt_is_planar(request_sample_fmt) ?
av_get_packed_sample_fmt(request_sample_fmt) :
av_get_planar_sample_fmt(request_sample_fmt));
context->request_sample_fmt = AV_SAMPLE_FMT_NONE;
for (fmt = codec->sample_fmts; *fmt != AV_SAMPLE_FMT_NONE; fmt++)
{
if (*fmt == request_sample_fmt)
{
context->request_sample_fmt = request_sample_fmt;
break;
}
else if (*fmt == next_best_fmt)
{
context->request_sample_fmt = next_best_fmt;
}
}
/*
* When encoding and AVCodec.sample_fmts exists, avcodec_open2()
* will error out if AVCodecContext.sample_fmt isn't set.
*/
if (context->request_sample_fmt == AV_SAMPLE_FMT_NONE)
{
context->request_sample_fmt = codec->sample_fmts[0];
}
context->sample_fmt = context->request_sample_fmt;
}
}
|