[FFmpeg-devel] [PATCH v3 2/2] fftools/ffmpeg: add exif orientation support per frame's metadata
Paul B Mahol
onemda at gmail.com
Tue May 28 17:24:22 EEST 2019
On 5/28/19, Jun Li <junli1026 at gmail.com> wrote:
> Fix #6945
> Rotate or/and flip frame according to frame's metadata orientation
> ---
> fftools/ffmpeg.c | 16 +++++++++++++++-
> fftools/ffmpeg.h | 3 ++-
> fftools/ffmpeg_filter.c | 28 +++++++++++++++++++++++-----
> 3 files changed, 40 insertions(+), 7 deletions(-)
>
> diff --git a/fftools/ffmpeg.c b/fftools/ffmpeg.c
> index 01f04103cf..2f4229a9d0 100644
> --- a/fftools/ffmpeg.c
> +++ b/fftools/ffmpeg.c
> @@ -2126,6 +2126,19 @@ static int ifilter_has_all_input_formats(FilterGraph
> *fg)
> return 1;
> }
>
> +static int orientation_need_update(InputFilter *ifilter, AVFrame *frame)
> +{
> + int orientaion = get_frame_orientation(frame);
> + int filterst = ifilter->orientation <= 1 ? 0 : // not set
> + ifilter->orientation <= 4 ? 1 : // auto flip
> + 2; // auto transpose
> + int framest = orientaion <= 1 ? 0 : // not set
> + orientaion <= 4 ? 1 : // auto flip
> + 2; // auto transpose
> +
> + return filterst != framest;
> +}
> +
> static int ifilter_send_frame(InputFilter *ifilter, AVFrame *frame)
> {
> FilterGraph *fg = ifilter->graph;
> @@ -2142,7 +2155,8 @@ static int ifilter_send_frame(InputFilter *ifilter,
> AVFrame *frame)
> break;
> case AVMEDIA_TYPE_VIDEO:
> need_reinit |= ifilter->width != frame->width ||
> - ifilter->height != frame->height;
> + ifilter->height != frame->height ||
> + orientation_need_update(ifilter, frame);
> break;
> }
>
> diff --git a/fftools/ffmpeg.h b/fftools/ffmpeg.h
> index eb1eaf6363..54532ef0eb 100644
> --- a/fftools/ffmpeg.h
> +++ b/fftools/ffmpeg.h
> @@ -244,7 +244,7 @@ typedef struct InputFilter {
> // parameters configured for this input
> int format;
>
> - int width, height;
> + int width, height, orientation;
> AVRational sample_aspect_ratio;
>
> int sample_rate;
> @@ -649,6 +649,7 @@ int init_complex_filtergraph(FilterGraph *fg);
> void sub2video_update(InputStream *ist, AVSubtitle *sub);
>
> int ifilter_parameters_from_frame(InputFilter *ifilter, const AVFrame
> *frame);
> +int get_frame_orientation(const AVFrame* frame);
>
> int ffmpeg_parse_options(int argc, char **argv);
>
> diff --git a/fftools/ffmpeg_filter.c b/fftools/ffmpeg_filter.c
> index 72838de1e2..ff63540906 100644
> --- a/fftools/ffmpeg_filter.c
> +++ b/fftools/ffmpeg_filter.c
> @@ -743,6 +743,18 @@ static int sub2video_prepare(InputStream *ist,
> InputFilter *ifilter)
> return 0;
> }
>
> +int get_frame_orientation(const AVFrame *frame)
> +{
> + AVDictionaryEntry *entry = NULL;
> + int orientation = 0;
> +
> + // read exif orientation data
> + entry = av_dict_get(frame->metadata, "Orientation", NULL, 0);
> + if (entry)
> + orientation = atoi(entry->value);
> + return orientation;
> +}
> +
> static int configure_input_video_filter(FilterGraph *fg, InputFilter
> *ifilter,
> AVFilterInOut *in)
> {
> @@ -809,13 +821,18 @@ static int configure_input_video_filter(FilterGraph
> *fg, InputFilter *ifilter,
> if (ist->autorotate) {
> double theta = get_rotation(ist->st);
>
> - if (fabs(theta - 90) < 1.0) {
> + if (fabs(theta) < 1.0) { // no rotation info in stream meta
> + if (ifilter->orientation < 0 || ifilter->orientation > 8) {
> + av_log(NULL, AV_LOG_ERROR, "Invalid frame orientation:
> %i\n", ifilter->orientation);
> + } else if (ifilter->orientation > 1 && ifilter->orientation <=
> 4) { // skip 0 (not set) and 1 (orientaion 'Normal')
> + ret = insert_filter(&last_filter, &pad_idx, "transpose",
> "orientation=-1");
> + } else if (ifilter->orientation > 4) {
> + ret = insert_filter(&last_filter, &pad_idx, "transpose",
> "orientation=-2");
Usning non-named option values is bad and not user-friendly.
> + }
> + } else if (fabs(theta - 90) < 1.0) {
> ret = insert_filter(&last_filter, &pad_idx, "transpose",
> "clock");
> } else if (fabs(theta - 180) < 1.0) {
> - ret = insert_filter(&last_filter, &pad_idx, "hflip", NULL);
> - if (ret < 0)
> - return ret;
> - ret = insert_filter(&last_filter, &pad_idx, "vflip", NULL);
> + ret = insert_filter(&last_filter, &pad_idx, "transpose",
> "orientation=3");
ditto
> } else if (fabs(theta - 270) < 1.0) {
> ret = insert_filter(&last_filter, &pad_idx, "transpose",
> "cclock");
> } else if (fabs(theta) > 1.0) {
> @@ -1191,6 +1208,7 @@ int ifilter_parameters_from_frame(InputFilter
> *ifilter, const AVFrame *frame)
> ifilter->width = frame->width;
> ifilter->height = frame->height;
> ifilter->sample_aspect_ratio = frame->sample_aspect_ratio;
> + ifilter->orientation = get_frame_orientation(frame);
>
> ifilter->sample_rate = frame->sample_rate;
> ifilter->channels = frame->channels;
> --
> 2.17.1
>
> _______________________________________________
> ffmpeg-devel mailing list
> ffmpeg-devel at ffmpeg.org
> https://ffmpeg.org/mailman/listinfo/ffmpeg-devel
>
> To unsubscribe, visit link above, or email
> ffmpeg-devel-request at ffmpeg.org with subject "unsubscribe".
More information about the ffmpeg-devel
mailing list