mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-03-03 14:32:16 +02:00
h264_metadata: Add support for display orientation SEI messages
This commit is contained in:
parent
c4eeea7633
commit
84c3c766d8
@ -17,6 +17,7 @@
|
||||
*/
|
||||
|
||||
#include "libavutil/avstring.h"
|
||||
#include "libavutil/display.h"
|
||||
#include "libavutil/common.h"
|
||||
#include "libavutil/opt.h"
|
||||
|
||||
@ -30,6 +31,12 @@ enum {
|
||||
PASS,
|
||||
INSERT,
|
||||
REMOVE,
|
||||
EXTRACT,
|
||||
};
|
||||
|
||||
enum {
|
||||
FLIP_HORIZONTAL = 1,
|
||||
FLIP_VERTICAL = 2,
|
||||
};
|
||||
|
||||
typedef struct H264MetadataContext {
|
||||
@ -38,6 +45,8 @@ typedef struct H264MetadataContext {
|
||||
CodedBitstreamContext *cbc;
|
||||
CodedBitstreamFragment access_unit;
|
||||
|
||||
int done_first_au;
|
||||
|
||||
H264RawAUD aud_nal;
|
||||
H264RawSEI sei_nal;
|
||||
|
||||
@ -62,9 +71,12 @@ typedef struct H264MetadataContext {
|
||||
int crop_bottom;
|
||||
|
||||
const char *sei_user_data;
|
||||
int sei_first_au;
|
||||
|
||||
int delete_filler;
|
||||
|
||||
int display_orientation;
|
||||
double rotate;
|
||||
int flip;
|
||||
} H264MetadataContext;
|
||||
|
||||
|
||||
@ -211,6 +223,8 @@ static int h264_metadata_filter(AVBSFContext *bsf, AVPacket *out)
|
||||
AVPacket *in = NULL;
|
||||
CodedBitstreamFragment *au = &ctx->access_unit;
|
||||
int err, i, j, has_sps;
|
||||
uint8_t *displaymatrix_side_data = NULL;
|
||||
size_t displaymatrix_side_data_size = 0;
|
||||
|
||||
err = ff_bsf_get_packet(bsf, &in);
|
||||
if (err < 0)
|
||||
@ -292,15 +306,13 @@ static int h264_metadata_filter(AVBSFContext *bsf, AVPacket *out)
|
||||
|
||||
// Only insert the SEI in access units containing SPSs, and also
|
||||
// unconditionally in the first access unit we ever see.
|
||||
if (ctx->sei_user_data && (has_sps || !ctx->sei_first_au)) {
|
||||
if (ctx->sei_user_data && (has_sps || !ctx->done_first_au)) {
|
||||
H264RawSEIPayload payload = {
|
||||
.payload_type = H264_SEI_TYPE_USER_DATA_UNREGISTERED,
|
||||
};
|
||||
H264RawSEIUserDataUnregistered *udu =
|
||||
&payload.payload.user_data_unregistered;
|
||||
|
||||
ctx->sei_first_au = 1;
|
||||
|
||||
for (i = j = 0; j < 32 && ctx->sei_user_data[i]; i++) {
|
||||
int c, v;
|
||||
c = ctx->sei_user_data[i];
|
||||
@ -387,6 +399,121 @@ static int h264_metadata_filter(AVBSFContext *bsf, AVPacket *out)
|
||||
}
|
||||
}
|
||||
|
||||
if (ctx->display_orientation != PASS) {
|
||||
for (i = 0; i < au->nb_units; i++) {
|
||||
H264RawSEI *sei;
|
||||
if (au->units[i].type != H264_NAL_SEI)
|
||||
continue;
|
||||
sei = au->units[i].content;
|
||||
|
||||
for (j = 0; j < sei->payload_count; j++) {
|
||||
H264RawSEIDisplayOrientation *disp;
|
||||
int32_t *matrix;
|
||||
|
||||
if (sei->payload[j].payload_type !=
|
||||
H264_SEI_TYPE_DISPLAY_ORIENTATION)
|
||||
continue;
|
||||
disp = &sei->payload[j].payload.display_orientation;
|
||||
|
||||
if (ctx->display_orientation == REMOVE ||
|
||||
ctx->display_orientation == INSERT) {
|
||||
err = ff_cbs_h264_delete_sei_message(ctx->cbc, au,
|
||||
&au->units[i], j);
|
||||
if (err < 0) {
|
||||
av_log(bsf, AV_LOG_ERROR, "Failed to delete "
|
||||
"display orientation SEI message.\n");
|
||||
goto fail;
|
||||
}
|
||||
--i;
|
||||
break;
|
||||
}
|
||||
|
||||
matrix = av_mallocz(9 * sizeof(int32_t));
|
||||
if (!matrix) {
|
||||
err = AVERROR(ENOMEM);
|
||||
goto fail;
|
||||
}
|
||||
|
||||
av_display_rotation_set(matrix,
|
||||
disp->anticlockwise_rotation *
|
||||
180.0 / 65536.0);
|
||||
av_display_matrix_flip(matrix, disp->hor_flip, disp->ver_flip);
|
||||
|
||||
displaymatrix_side_data = (uint8_t*)matrix;
|
||||
displaymatrix_side_data_size = 9 * sizeof(int32_t);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (ctx->display_orientation == INSERT) {
|
||||
H264RawSEIPayload payload = {
|
||||
.payload_type = H264_SEI_TYPE_DISPLAY_ORIENTATION,
|
||||
};
|
||||
H264RawSEIDisplayOrientation *disp =
|
||||
&payload.payload.display_orientation;
|
||||
uint8_t *data;
|
||||
int size;
|
||||
int write = 0;
|
||||
|
||||
data = av_packet_get_side_data(in, AV_PKT_DATA_DISPLAYMATRIX, &size);
|
||||
if (data && size >= 9 * sizeof(int32_t)) {
|
||||
int32_t matrix[9];
|
||||
int hflip, vflip;
|
||||
double angle;
|
||||
|
||||
memcpy(matrix, data, sizeof(matrix));
|
||||
|
||||
hflip = vflip = 0;
|
||||
if (matrix[0] < 0 && matrix[4] > 0)
|
||||
hflip = 1;
|
||||
else if (matrix[0] > 0 && matrix[4] < 0)
|
||||
vflip = 1;
|
||||
av_display_matrix_flip(matrix, hflip, vflip);
|
||||
|
||||
angle = av_display_rotation_get(matrix);
|
||||
|
||||
if (!(angle >= -180.0 && angle <= 180.0 /* also excludes NaN */) ||
|
||||
matrix[2] != 0 || matrix[5] != 0 ||
|
||||
matrix[6] != 0 || matrix[7] != 0) {
|
||||
av_log(bsf, AV_LOG_WARNING, "Input display matrix is not "
|
||||
"representable in H.264 parameters.\n");
|
||||
} else {
|
||||
disp->hor_flip = hflip;
|
||||
disp->ver_flip = vflip;
|
||||
disp->anticlockwise_rotation =
|
||||
(uint16_t)rint((angle >= 0.0 ? angle
|
||||
: angle + 360.0) *
|
||||
65536.0 / 360.0);
|
||||
write = 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (has_sps || !ctx->done_first_au) {
|
||||
if (!isnan(ctx->rotate)) {
|
||||
disp->anticlockwise_rotation =
|
||||
(uint16_t)rint((ctx->rotate >= 0.0 ? ctx->rotate
|
||||
: ctx->rotate + 360.0) *
|
||||
65536.0 / 360.0);
|
||||
write = 1;
|
||||
}
|
||||
if (ctx->flip) {
|
||||
disp->hor_flip = !!(ctx->flip & FLIP_HORIZONTAL);
|
||||
disp->ver_flip = !!(ctx->flip & FLIP_VERTICAL);
|
||||
write = 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (write) {
|
||||
disp->display_orientation_repetition_period = 1;
|
||||
|
||||
err = ff_cbs_h264_add_sei_message(ctx->cbc, au, &payload);
|
||||
if (err < 0) {
|
||||
av_log(bsf, AV_LOG_ERROR, "Failed to add display orientation "
|
||||
"SEI message to access unit.\n");
|
||||
goto fail;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
err = ff_cbs_write_packet(ctx->cbc, out, au);
|
||||
if (err < 0) {
|
||||
av_log(bsf, AV_LOG_ERROR, "Failed to write packet.\n");
|
||||
@ -397,9 +524,24 @@ static int h264_metadata_filter(AVBSFContext *bsf, AVPacket *out)
|
||||
if (err < 0)
|
||||
goto fail;
|
||||
|
||||
if (displaymatrix_side_data) {
|
||||
err = av_packet_add_side_data(out, AV_PKT_DATA_DISPLAYMATRIX,
|
||||
displaymatrix_side_data,
|
||||
displaymatrix_side_data_size);
|
||||
if (err) {
|
||||
av_log(bsf, AV_LOG_ERROR, "Failed to attach extracted "
|
||||
"displaymatrix side data to packet.\n");
|
||||
goto fail;
|
||||
}
|
||||
displaymatrix_side_data = NULL;
|
||||
}
|
||||
|
||||
ctx->done_first_au = 1;
|
||||
|
||||
err = 0;
|
||||
fail:
|
||||
ff_cbs_fragment_uninit(ctx->cbc, au);
|
||||
av_freep(&displaymatrix_side_data);
|
||||
|
||||
av_packet_free(&in);
|
||||
|
||||
@ -509,6 +651,25 @@ static const AVOption h264_metadata_options[] = {
|
||||
{ "delete_filler", "Delete all filler (both NAL and SEI)",
|
||||
OFFSET(delete_filler), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, 1 },
|
||||
|
||||
{ "display_orientation", "Display orientation SEI",
|
||||
OFFSET(display_orientation), AV_OPT_TYPE_INT,
|
||||
{ .i64 = PASS }, PASS, EXTRACT, 0, "disp_or" },
|
||||
{ "pass", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = PASS }, .unit = "disp_or" },
|
||||
{ "insert", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = INSERT }, .unit = "disp_or" },
|
||||
{ "remove", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = REMOVE }, .unit = "disp_or" },
|
||||
{ "extract", NULL, 0, AV_OPT_TYPE_CONST, { .i64 = EXTRACT }, .unit = "disp_or" },
|
||||
|
||||
{ "rotate", "Set rotation in display orientation SEI (anticlockwise angle in degrees)",
|
||||
OFFSET(rotate), AV_OPT_TYPE_DOUBLE,
|
||||
{ .dbl = NAN }, -360.0, +360.0 },
|
||||
{ "flip", "Set flip in display orientation SEI",
|
||||
OFFSET(flip), AV_OPT_TYPE_FLAGS,
|
||||
{ .i64 = 0 }, 0, FLIP_HORIZONTAL | FLIP_VERTICAL, .unit ="flip" },
|
||||
{ "horizontal", "Set hor_flip",
|
||||
0, AV_OPT_TYPE_CONST, { .i64 = FLIP_HORIZONTAL }, .unit ="flip" },
|
||||
{ "vertical", "Set ver_flip",
|
||||
0, AV_OPT_TYPE_CONST, { .i64 = FLIP_VERTICAL }, .unit ="flip" },
|
||||
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user