[FFmpeg-devel] [PATCH 1/2] Decoding of Bold Italic and Underlined styles for 3GPP timed text subtitles

Michael Niedermayer michaelni at gmx.at
Mon Apr 13 17:53:43 CEST 2015


On Mon, Apr 13, 2015 at 08:26:59PM +0530, Niklesh Lalwani wrote:
> From: Niklesh <niklesh.lalwani at iitb.ac.in>
> 
> This patch is a part of my qualification task to implement support for Bold, Italic, and Underlined style records for 3GPP timed text subtitles. I am continuing Wesley's work. This patch supports decoding of no more than one style record. Patch[2/2] attempts to use dynamic arrays to support multiple style records.
> Signed-off-by: Niklesh <niklesh.lalwani at iitb.ac.in>
> ---
>  libavcodec/movtextdec.c | 81 +++++++++++++++++++++++++++++++++++++++++++++----
>  1 file changed, 75 insertions(+), 6 deletions(-)
> 
> diff --git a/libavcodec/movtextdec.c b/libavcodec/movtextdec.c
> index 1c7ffea..4e463ed 100644
> --- a/libavcodec/movtextdec.c
> +++ b/libavcodec/movtextdec.c
> @@ -26,9 +26,25 @@
>  #include "libavutil/bprint.h"
>  #include "libavutil/intreadwrite.h"
>  
> -static int text_to_ass(AVBPrint *buf, const char *text, const char *text_end)
> +#define STYLE_FLAG_BOLD         1
> +#define STYLE_FLAG_ITALIC       2
> +#define STYLE_FLAG_UNDERLINE    4
> +
> +static int text_to_ass(AVBPrint *buf, const char *text, const char *text_end, 
> +                        const char *style_start, const char *style_end, 
> +                        const int style_flags)
>  {
> -    while (text < text_end) {
> +        while (text < text_end) {
> +        if (style_flags && text == style_start)
> +        {
> +            if (style_flags & STYLE_FLAG_BOLD)
> +                av_bprintf(buf, "{\\b1}");
> +            if (style_flags & STYLE_FLAG_ITALIC)
> +                av_bprintf(buf, "{\\i1}");
> +            if (style_flags & STYLE_FLAG_UNDERLINE)
> +                av_bprintf(buf, "{\\u1}");
> +        }
> +
>          switch (*text) {
>          case '\r':
>              break;
> @@ -39,6 +55,16 @@ static int text_to_ass(AVBPrint *buf, const char *text, const char *text_end)
>              av_bprint_chars(buf, *text, 1);
>              break;
>          }
> +
> +        if (style_flags && text == style_end)
> +        {
> +            if (style_flags & STYLE_FLAG_BOLD)
> +                av_bprintf(buf, "{\\b0}");
> +            if (style_flags & STYLE_FLAG_ITALIC)
> +                av_bprintf(buf, "{\\i0}");
> +            if (style_flags & STYLE_FLAG_UNDERLINE)
> +                av_bprintf(buf, "{\\u0}");
> +        }
>          text++;
>      }
>  
> @@ -63,6 +89,9 @@ static int mov_text_decode_frame(AVCodecContext *avctx,
>      AVBPrint buf;
>      const char *ptr = avpkt->data;
>      const char *end;
> +    int text_length, tsmb_type, style_entries, style_flags, tsmb_size;
> +    const char *style_start, *style_end;
> +    const uint8_t *tsmb;
>  
>      if (!ptr || avpkt->size < 2)
>          return AVERROR_INVALIDDATA;

> @@ -74,6 +103,7 @@ static int mov_text_decode_frame(AVCodecContext *avctx,
>       * already. If the value is non-zero, then it's technically a
>       * bad packet.
>       */
> +
>      if (avpkt->size == 2)
>          return AV_RB16(ptr) == 0 ? 0 : AVERROR_INVALIDDATA;
>  

doesnt belong in this patch


> @@ -82,7 +112,10 @@ static int mov_text_decode_frame(AVCodecContext *avctx,
>       * In complex cases, there are style descriptors appended to the string
>       * so we can't just assume the packet size is the string size.
>       */

> -    end = ptr + FFMIN(2 + AV_RB16(ptr), avpkt->size);
> +    //end = ptr + FFMIN(2 + AV_RB16(ptr), avpkt->size);

either remove code or leave it, dont just comment it out


> +    text_length = AV_RB16(ptr);
> +
> +    end = ptr + FFMIN(2 + text_length, avpkt->size); 
>      ptr += 2;
>  
>      ts_start = av_rescale_q(avpkt->pts,
> @@ -91,11 +124,47 @@ static int mov_text_decode_frame(AVCodecContext *avctx,
>      ts_end   = av_rescale_q(avpkt->pts + avpkt->duration,
>                              avctx->time_base,
>                              (AVRational){1,100});
> -
> +    tsmb_size=0;
>      // Note that the spec recommends lines be no longer than 2048 characters.
>      av_bprint_init(&buf, 0, AV_BPRINT_SIZE_UNLIMITED);
> -    text_to_ass(&buf, ptr, end);
> -    ret = ff_ass_add_rect_bprint(sub, &buf, ts_start, ts_end-ts_start);

> +    if (text_length + 2 != avpkt->size)
> +    {
> +    while (text_length + 2 + tsmb_size < avpkt->size)
> +    {

please indent the code somehow


> +        tsmb = ptr + text_length+tsmb_size;
> +        tsmb_size = AV_RB32(tsmb);  
> +        tsmb += 4;
> +        tsmb_type = AV_RB32(tsmb);
> +        tsmb += 4;
> +
> +        if (tsmb_type == MKBETAG('s','t','y','l'))
> +        {
> +            style_entries = AV_RB16(tsmb);
> +            tsmb += 2;
> +

> +            for(int i = 0; i < style_entries;i++)

the int i should be at the begin of a block with other declarations
some compilers have problems with this style


[...]

> +    }
> +    }


[...]

-- 
Michael     GnuPG fingerprint: 9FF2128B147EF6730BADF133611EC787040B0FAB

Complexity theory is the science of finding the exact solution to an
approximation. Benchmarking OTOH is finding an approximation of the exact
-------------- next part --------------
A non-text attachment was scrubbed...
Name: not available
Type: application/pgp-signature
Size: 181 bytes
Desc: Digital signature
URL: <https://ffmpeg.org/pipermail/ffmpeg-devel/attachments/20150413/14141f2a/attachment.asc>


More information about the ffmpeg-devel mailing list