47 #define ZIMG_ALIGNMENT 64
48 #define MIN_TILESIZE 64
49 #define MAX_THREADS 64
139 zimg_image_format_default(&
s->src_format, ZIMG_API_VERSION);
140 zimg_image_format_default(&
s->dst_format, ZIMG_API_VERSION);
141 zimg_image_format_default(&
s->src_format_tmp, ZIMG_API_VERSION);
142 zimg_image_format_default(&
s->dst_format_tmp, ZIMG_API_VERSION);
144 zimg_graph_builder_params_default(&
s->params, ZIMG_API_VERSION);
145 zimg_graph_builder_params_default(&
s->params_tmp, ZIMG_API_VERSION);
147 if (
s->size_str && (
s->w_expr ||
s->h_expr)) {
149 "Size and width/height expressions cannot be set at the same time.\n");
153 if (
s->w_expr && !
s->h_expr)
154 FFSWAP(
char *,
s->w_expr,
s->size_str);
160 "Invalid size '%s'\n",
s->size_str);
163 snprintf(buf,
sizeof(buf)-1,
"%d",
s->w);
165 snprintf(buf,
sizeof(buf)-1,
"%d",
s->h);
222 formats =
s->colorspace != ZIMG_MATRIX_UNSPECIFIED &&
s->colorspace > 0
243 s->out_slice_start[0] = 0;
244 for (
int i = 1;
i <
s->nb_threads;
i++) {
248 s->out_slice_end[
s->nb_threads - 1] = out_h;
250 for (
int i = 0;
i <
s->nb_threads;
i++) {
251 s->in_slice_start[
i] =
s->out_slice_start[
i] * in_h / (
double)out_h;
252 s->in_slice_end[
i] =
s->out_slice_end[
i] * in_h / (
double)out_h;
264 double var_values[
VARS_NB], res;
275 (
double)
inlink->sample_aspect_ratio.num /
inlink->sample_aspect_ratio.den : 1;
291 if (!(res >= INT32_MIN && res <= INT32_MAX)) {
302 if (!(res >= INT32_MIN && res <= INT32_MAX)) {
340 if (
s->force_original_aspect_ratio) {
344 if (
s->force_original_aspect_ratio == 1) {
353 if (
w > INT_MAX ||
h > INT_MAX ||
363 if (
inlink->sample_aspect_ratio.num){
368 av_log(
ctx,
AV_LOG_DEBUG,
"w:%d h:%d fmt:%s csp:%s range:%s sar:%d/%d -> w:%d h:%d fmt:%s csp:%s range:%s sar:%d/%d\n",
371 inlink->sample_aspect_ratio.num,
inlink->sample_aspect_ratio.den,
379 "Error when evaluating the expression '%s'.\n"
380 "Maybe the expression for out_w:'%s' or for out_h:'%s' is self-referencing.\n",
381 expr,
s->w_expr,
s->h_expr);
388 int err_code = zimg_get_last_error(err_msg,
sizeof(err_msg));
397 switch (chroma_location) {
400 return ZIMG_CHROMA_LEFT;
402 return ZIMG_CHROMA_CENTER;
404 return ZIMG_CHROMA_TOP_LEFT;
406 return ZIMG_CHROMA_TOP;
408 return ZIMG_CHROMA_BOTTOM_LEFT;
410 return ZIMG_CHROMA_BOTTOM;
412 return ZIMG_CHROMA_LEFT;
417 switch (colorspace) {
419 return ZIMG_MATRIX_RGB;
421 return ZIMG_MATRIX_709;
423 return ZIMG_MATRIX_UNSPECIFIED;
425 return ZIMG_MATRIX_FCC;
427 return ZIMG_MATRIX_470BG;
429 return ZIMG_MATRIX_170M;
431 return ZIMG_MATRIX_240M;
433 return ZIMG_MATRIX_YCGCO;
435 return ZIMG_MATRIX_2020_NCL;
437 return ZIMG_MATRIX_2020_CL;
439 return ZIMG_MATRIX_CHROMATICITY_DERIVED_NCL;
441 return ZIMG_MATRIX_CHROMATICITY_DERIVED_CL;
443 return ZIMG_MATRIX_ICTCP;
445 return ZIMG_MATRIX_UNSPECIFIED;
452 return ZIMG_TRANSFER_UNSPECIFIED;
454 return ZIMG_TRANSFER_709;
456 return ZIMG_TRANSFER_470_M;
458 return ZIMG_TRANSFER_470_BG;
460 return ZIMG_TRANSFER_601;
462 return ZIMG_TRANSFER_240M;
464 return ZIMG_TRANSFER_LINEAR;
466 return ZIMG_TRANSFER_LOG_100;
468 return ZIMG_TRANSFER_LOG_316;
470 return ZIMG_TRANSFER_IEC_61966_2_4;
472 return ZIMG_TRANSFER_2020_10;
474 return ZIMG_TRANSFER_2020_12;
476 return ZIMG_TRANSFER_ST2084;
478 return ZIMG_TRANSFER_ARIB_B67;
480 return ZIMG_TRANSFER_IEC_61966_2_1;
482 return ZIMG_TRANSFER_UNSPECIFIED;
489 return ZIMG_PRIMARIES_UNSPECIFIED;
491 return ZIMG_PRIMARIES_709;
493 return ZIMG_PRIMARIES_470_M;
495 return ZIMG_PRIMARIES_470_BG;
497 return ZIMG_PRIMARIES_170M;
499 return ZIMG_PRIMARIES_240M;
501 return ZIMG_PRIMARIES_FILM;
503 return ZIMG_PRIMARIES_2020;
505 return ZIMG_PRIMARIES_ST428;
507 return ZIMG_PRIMARIES_ST431_2;
509 return ZIMG_PRIMARIES_ST432_1;
511 return ZIMG_PRIMARIES_EBU3213_E;
513 return ZIMG_PRIMARIES_UNSPECIFIED;
521 return ZIMG_RANGE_LIMITED;
523 return ZIMG_RANGE_FULL;
525 return ZIMG_RANGE_LIMITED;
531 case ZIMG_RANGE_LIMITED:
533 case ZIMG_RANGE_FULL:
542 return ((img_fmt0->chroma_location != img_fmt1->chroma_location) ||
543 #
if ZIMG_API_VERSION >= 0x204
544 (img_fmt0->alpha != img_fmt1->alpha) ||
546 (img_fmt0->color_family != img_fmt1->color_family) ||
547 (img_fmt0->color_primaries != img_fmt1->color_primaries) ||
548 (img_fmt0->depth != img_fmt1->depth) ||
549 (img_fmt0->field_parity != img_fmt1->field_parity) ||
550 (img_fmt0->height != img_fmt1->height) ||
551 (img_fmt0->matrix_coefficients != img_fmt1->matrix_coefficients) ||
552 (img_fmt0->pixel_range != img_fmt1->pixel_range) ||
553 (img_fmt0->pixel_type != img_fmt1->pixel_type) ||
554 (img_fmt0->subsample_h != img_fmt1->subsample_h) ||
555 (img_fmt0->subsample_w != img_fmt1->subsample_w) ||
556 (img_fmt0->transfer_characteristics != img_fmt1->transfer_characteristics) ||
557 (img_fmt0->width != img_fmt1->width));
565 int ret = (parm0->allow_approximate_gamma != parm1->allow_approximate_gamma) ||
566 (parm0->dither_type != parm1->dither_type) ||
567 (parm0->resample_filter != parm1->resample_filter) ||
568 (parm0->resample_filter_uv != parm1->resample_filter_uv);
570 if ((
isnan(parm0->nominal_peak_luminance) == 0) || (
isnan(parm1->nominal_peak_luminance) == 0))
571 ret =
ret || (parm0->nominal_peak_luminance != parm1->nominal_peak_luminance);
572 if ((
isnan(parm0->filter_param_a) == 0) || (
isnan(parm1->filter_param_a) == 0))
573 ret =
ret || (parm0->filter_param_a != parm1->filter_param_a);
574 if ((
isnan(parm0->filter_param_a_uv) == 0) || (
isnan(parm1->filter_param_a_uv) == 0))
575 ret =
ret || (parm0->filter_param_a_uv != parm1->filter_param_a_uv);
576 if ((
isnan(parm0->filter_param_b) == 0) || (
isnan(parm1->filter_param_b) == 0))
577 ret =
ret || (parm0->filter_param_b != parm1->filter_param_b);
578 if ((
isnan(parm0->filter_param_b_uv) == 0) || (
isnan(parm1->filter_param_b_uv) == 0))
579 ret =
ret || (parm0->filter_param_b_uv != parm1->filter_param_b_uv);
593 : (
desc->comp[0].depth > 8 ? ZIMG_PIXEL_WORD : ZIMG_PIXEL_BYTE);
595 : (
desc->nb_components > 1 ? ZIMG_COLOR_YUV : ZIMG_COLOR_GREY);
611 zimg_image_format src_format;
612 zimg_image_format dst_format;
613 const double in_slice_start =
s->in_slice_start[job_nr];
614 const double in_slice_end =
s->in_slice_end[job_nr];
615 const int out_slice_start =
s->out_slice_start[job_nr];
616 const int out_slice_end =
s->out_slice_end[job_nr];
618 src_format =
s->src_format;
619 dst_format =
s->dst_format;
623 src_format.active_region.width = in->
width;
624 src_format.active_region.height = in_slice_end - in_slice_start;
625 src_format.active_region.left = 0;
626 src_format.active_region.top = in_slice_start;
628 dst_format.width =
out->width;
629 dst_format.height = out_slice_end - out_slice_start;
631 if (
s->graph[job_nr]) {
632 zimg_filter_graph_free(
s->graph[job_nr]);
634 s->graph[job_nr] = zimg_filter_graph_build(&src_format, &dst_format, &
s->params);
635 if (!
s->graph[job_nr])
638 ret = zimg_filter_graph_get_tmp_size(
s->graph[job_nr], &
size);
661 for (plane = 0; plane <
planes; plane++) {
662 int p =
desc->comp[plane].plane;
687 if (
s->primaries != -1)
688 frame->color_primaries = (int)
s->dst_format.color_primaries;
691 frame->color_trc = (int)
s->dst_format.transfer_characteristics;
693 if (
s->chromal != -1)
694 frame->chroma_location = (int)
s->dst_format.chroma_location + 1;
704 zimg_image_buffer_const src_buf = { ZIMG_API_VERSION };
705 zimg_image_buffer dst_buf = { ZIMG_API_VERSION };
706 const int out_slice_start =
s->out_slice_start[job_nr];
719 for (
int i = 0;
i < 4;
i++) {
724 if (i < td->
desc->nb_components) {
725 src_buf.plane[
i].data = td->
in->
data[
p];
727 src_buf.plane[
i].mask = -1;
731 if (i < td->odesc->nb_components) {
734 dst_buf.plane[
i].mask = -1;
737 if (!
s->graph[job_nr])
739 ret = zimg_filter_graph_process(
s->graph[job_nr], &src_buf, &dst_buf,
756 int ret = 0, changed = 0;
763 (
link->w != outlink->
w) ||
764 (
link->h != outlink->
h) ||
768 (
s->src_format.chroma_location !=
s->dst_format.chroma_location) ||
769 (
s->src_format.color_family !=
s->dst_format.color_family) ||
770 (
s->src_format.color_primaries !=
s->dst_format.color_primaries) ||
771 (
s->src_format.depth !=
s->dst_format.depth) ||
772 (
s->src_format.matrix_coefficients !=
s->dst_format.matrix_coefficients) ||
773 (
s->src_format.field_parity !=
s->dst_format.field_parity) ||
774 (
s->src_format.pixel_range !=
s->dst_format.pixel_range) ||
775 (
s->src_format.pixel_type !=
s->dst_format.pixel_type) ||
776 (
s->src_format.transfer_characteristics !=
s->dst_format.transfer_characteristics)
791 snprintf(buf,
sizeof(buf)-1,
"%d", outlink->
w);
793 snprintf(buf,
sizeof(buf)-1,
"%d", outlink->
h);
805 zimg_image_format_default(&
s->src_format, ZIMG_API_VERSION);
806 zimg_image_format_default(&
s->dst_format, ZIMG_API_VERSION);
807 zimg_graph_builder_params_default(&
s->params, ZIMG_API_VERSION);
810 s->primaries_in,
s->trc_in,
s->range_in,
s->chromal_in);
812 s->primaries,
s->trc,
s->range,
s->chromal);
815 s->params.dither_type =
s->dither;
816 s->params.cpu_type = ZIMG_CPU_AUTO_64B;
817 s->params.resample_filter =
s->filter;
818 s->params.resample_filter_uv =
s->filter;
819 s->params.nominal_peak_luminance =
s->nominal_peak_luminance;
820 s->params.allow_approximate_gamma =
s->approximate_gamma;
821 s->params.filter_param_a =
s->params.filter_param_a_uv =
s->param_a;
822 s->params.filter_param_b =
s->params.filter_param_b_uv =
s->param_b;
825 av_reduce(&
out->sample_aspect_ratio.num, &
out->sample_aspect_ratio.den,
841 memset(
s->jobs_ret, 0,
s->nb_threads *
sizeof(*
s->jobs_ret));
843 for (
int i = 0;
ret >= 0 &&
i <
s->nb_threads;
i++)
844 if (
s->jobs_ret[
i] < 0)
845 ret =
s->jobs_ret[
i];
852 s->src_format_tmp =
s->src_format;
853 s->dst_format_tmp =
s->dst_format;
854 s->params_tmp =
s->params;
859 const uint16_t h_one = 0x3C00;
860 for (y = 0; y <
out->height; y++) {
861 const ptrdiff_t row = y *
out->linesize[3];
863 for (x = 0; x <
out->width; x++) {
867 for (x = 0; x <
out->width; x++) {
872 }
else if (
s->dst_format.depth == 8) {
873 for (y = 0; y < outlink->
h; y++)
874 memset(
out->data[3] + y *
out->linesize[3], 0xff, outlink->
w);
876 const uint16_t
max = (1 <<
s->dst_format.depth) - 1;
877 for (y = 0; y < outlink->
h; y++) {
878 const ptrdiff_t row = y *
out->linesize[3];
879 for (x = 0; x <
out->width; x++)
902 for (
int i = 0;
i <
s->nb_threads;
i++) {
905 zimg_filter_graph_free(
s->graph[
i]);
912 char *res,
int res_len,
int flags)
917 if ( !strcmp(cmd,
"width") || !strcmp(cmd,
"w")
918 || !strcmp(cmd,
"height") || !strcmp(cmd,
"h")) {
935 #define OFFSET(x) offsetof(ZScaleContext, x)
936 #define FLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM
937 #define TFLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_RUNTIME_PARAM
951 {
"error_diffusion", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_DITHER_ERROR_DIFFUSION}, 0, 0,
FLAGS, .unit =
"dither" },
974 {
"unspecified", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_UNSPECIFIED}, 0, 0,
FLAGS, .unit =
"primaries" },
978 {
"unknown", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_UNSPECIFIED}, 0, 0,
FLAGS, .unit =
"primaries" },
987 {
"smpte431", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_ST431_2}, 0, 0,
FLAGS, .unit =
"primaries" },
988 {
"smpte432", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_ST432_1}, 0, 0,
FLAGS, .unit =
"primaries" },
989 {
"jedec-p22", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_EBU3213_E}, 0, 0,
FLAGS, .unit =
"primaries" },
990 {
"ebu3213", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_PRIMARIES_EBU3213_E}, 0, 0,
FLAGS, .unit =
"primaries" },
991 {
"transfer",
"set transfer characteristic",
OFFSET(trc),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"transfer" },
995 {
"unspecified", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_UNSPECIFIED}, 0, 0,
FLAGS, .unit =
"transfer" },
1000 {
"unknown", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_UNSPECIFIED}, 0, 0,
FLAGS, .unit =
"transfer" },
1009 {
"bt2020-10", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_2020_10}, 0, 0,
FLAGS, .unit =
"transfer" },
1010 {
"bt2020-12", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_2020_12}, 0, 0,
FLAGS, .unit =
"transfer" },
1011 {
"smpte2084", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_ST2084}, 0, 0,
FLAGS, .unit =
"transfer" },
1012 {
"iec61966-2-4", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_IEC_61966_2_4},0, 0,
FLAGS, .unit =
"transfer" },
1013 {
"iec61966-2-1", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_IEC_61966_2_1},0, 0,
FLAGS, .unit =
"transfer" },
1014 {
"arib-std-b67", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_TRANSFER_ARIB_B67}, 0, 0,
FLAGS, .unit =
"transfer" },
1015 {
"matrix",
"set colorspace matrix",
OFFSET(colorspace),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"matrix" },
1019 {
"unspecified", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_MATRIX_UNSPECIFIED}, 0, 0,
FLAGS, .unit =
"matrix" },
1034 {
"chroma-derived-nc",0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_MATRIX_CHROMATICITY_DERIVED_NCL}, 0, 0,
FLAGS, .unit =
"matrix" },
1035 {
"chroma-derived-c", 0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_MATRIX_CHROMATICITY_DERIVED_CL}, 0, 0,
FLAGS, .unit =
"matrix" },
1037 {
"in_range",
"set input color range",
OFFSET(range_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_RANGE_FULL,
FLAGS, .unit =
"range" },
1038 {
"rangein",
"set input color range",
OFFSET(range_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_RANGE_FULL,
FLAGS, .unit =
"range" },
1039 {
"rin",
"set input color range",
OFFSET(range_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_RANGE_FULL,
FLAGS, .unit =
"range" },
1040 {
"primariesin",
"set input color primaries",
OFFSET(primaries_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"primaries" },
1041 {
"pin",
"set input color primaries",
OFFSET(primaries_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"primaries" },
1042 {
"transferin",
"set input transfer characteristic",
OFFSET(trc_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"transfer" },
1043 {
"tin",
"set input transfer characteristic",
OFFSET(trc_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"transfer" },
1044 {
"matrixin",
"set input colorspace matrix",
OFFSET(colorspace_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"matrix" },
1045 {
"min",
"set input colorspace matrix",
OFFSET(colorspace_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, INT_MAX,
FLAGS, .unit =
"matrix" },
1046 {
"chromal",
"set output chroma location",
OFFSET(chromal),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_CHROMA_BOTTOM,
FLAGS, .unit =
"chroma" },
1047 {
"c",
"set output chroma location",
OFFSET(chromal),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_CHROMA_BOTTOM,
FLAGS, .unit =
"chroma" },
1053 {
"bottomleft",0, 0,
AV_OPT_TYPE_CONST, {.i64 = ZIMG_CHROMA_BOTTOM_LEFT}, 0, 0,
FLAGS, .unit =
"chroma" },
1055 {
"chromalin",
"set input chroma location",
OFFSET(chromal_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_CHROMA_BOTTOM,
FLAGS, .unit =
"chroma" },
1056 {
"cin",
"set input chroma location",
OFFSET(chromal_in),
AV_OPT_TYPE_INT, {.i64 = -1}, -1, ZIMG_CHROMA_BOTTOM,
FLAGS, .unit =
"chroma" },
1059 {
"param_a",
"parameter A, which is parameter \"b\" for bicubic, "
1086 .p.priv_class = &zscale_class,