/* * ARIB STD-B24 caption decoder using the libaribb24 library * Copyright (c) 2019 Jan Ekström * * This file is part of FFmpeg. * * FFmpeg is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * FFmpeg is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with FFmpeg; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ #include "avcodec.h" #include "libavcodec/ass.h" #include "codec_internal.h" #include "libavutil/log.h" #include "libavutil/opt.h" #include #include #include typedef struct Libaribb24Context { AVClass *class; arib_instance_t *lib_instance; arib_parser_t *parser; arib_decoder_t *decoder; int read_order; char *aribb24_base_path; unsigned int aribb24_skip_ruby; int default_profile; } Libaribb24Context; static unsigned int get_profile_font_size(AVCodecContext *avctx) { Libaribb24Context *b24 = avctx->priv_data; int profile = avctx->profile; if (profile == AV_PROFILE_UNKNOWN) profile = b24->default_profile; switch (profile) { case AV_PROFILE_ARIB_PROFILE_A: return 36; case AV_PROFILE_ARIB_PROFILE_C: return 18; default: return 0; } } static void libaribb24_log(void *p, const char *msg) { av_log((AVCodecContext *)p, AV_LOG_INFO, "%s\n", msg); } static int libaribb24_generate_ass_header(AVCodecContext *avctx) { Libaribb24Context *b24 = avctx->priv_data; unsigned int plane_width = 0; unsigned int plane_height = 0; unsigned int font_size = 0; int profile = avctx->profile; if (profile == AV_PROFILE_UNKNOWN) profile = b24->default_profile; switch (profile) { case AV_PROFILE_ARIB_PROFILE_A: plane_width = 960; plane_height = 540; break; case AV_PROFILE_ARIB_PROFILE_C: plane_width = 320; plane_height = 180; break; default: av_log(avctx, AV_LOG_ERROR, "Unknown or unsupported profile set!\n"); return AVERROR(EINVAL); } font_size = get_profile_font_size(avctx); avctx->subtitle_header = av_asprintf( "[Script Info]\r\n" "; Script generated by FFmpeg/Lavc%s\r\n" "ScriptType: v4.00+\r\n" "PlayResX: %d\r\n" "PlayResY: %d\r\n" "\r\n" "[V4+ Styles]\r\n" /* ASSv4 header */ "Format: Name, " "Fontname, Fontsize, " "PrimaryColour, SecondaryColour, OutlineColour, BackColour, " "Bold, Italic, Underline, StrikeOut, " "ScaleX, ScaleY, " "Spacing, Angle, " "BorderStyle, Outline, Shadow, " "Alignment, MarginL, MarginR, MarginV, " "Encoding\r\n" "Style: " "Default," /* Name */ "%s,%d," /* Font{name,size} */ "&H%x,&H%x,&H%x,&H%x," /* {Primary,Secondary,Outline,Back}Colour */ "%d,%d,%d,0," /* Bold, Italic, Underline, StrikeOut */ "100,100," /* Scale{X,Y} */ "0,0," /* Spacing, Angle */ "%d,1,0," /* BorderStyle, Outline, Shadow */ "%d,10,10,10," /* Alignment, Margin[LRV] */ "0\r\n" /* Encoding */ "\r\n" "[Events]\r\n" "Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text\r\n", !(avctx->flags & AV_CODEC_FLAG_BITEXACT) ? AV_STRINGIFY(LIBAVCODEC_VERSION) : "", plane_width, plane_height, ASS_DEFAULT_FONT, font_size, ASS_DEFAULT_COLOR, ASS_DEFAULT_COLOR, ASS_DEFAULT_BACK_COLOR, ASS_DEFAULT_BACK_COLOR, -ASS_DEFAULT_BOLD, -ASS_DEFAULT_ITALIC, -ASS_DEFAULT_UNDERLINE, ASS_DEFAULT_BORDERSTYLE, ASS_DEFAULT_ALIGNMENT); if (!avctx->subtitle_header) return AVERROR(ENOMEM); avctx->subtitle_header_size = strlen(avctx->subtitle_header); return 0; } static int libaribb24_init(AVCodecContext *avctx) { Libaribb24Context *b24 = avctx->priv_data; void(* arib_dec_init)(arib_decoder_t* decoder) = NULL; int ret; int profile = avctx->profile; if (!(b24->lib_instance = arib_instance_new(avctx))) { av_log(avctx, AV_LOG_ERROR, "Failed to initialize libaribb24!\n"); return AVERROR_EXTERNAL; } if (b24->aribb24_base_path) { av_log(avctx, AV_LOG_INFO, "Setting the libaribb24 base path to '%s'\n", b24->aribb24_base_path); arib_set_base_path(b24->lib_instance, b24->aribb24_base_path); } arib_register_messages_callback(b24->lib_instance, libaribb24_log); if (!(b24->parser = arib_get_parser(b24->lib_instance))) { av_log(avctx, AV_LOG_ERROR, "Failed to initialize libaribb24 PES parser!\n"); return AVERROR_EXTERNAL; } if (!(b24->decoder = arib_get_decoder(b24->lib_instance))) { av_log(avctx, AV_LOG_ERROR, "Failed to initialize libaribb24 decoder!\n"); return AVERROR_EXTERNAL; } if (profile == AV_PROFILE_UNKNOWN) profile = b24->default_profile; switch (profile) { case AV_PROFILE_ARIB_PROFILE_A: arib_dec_init = arib_initialize_decoder_a_profile; break; case AV_PROFILE_ARIB_PROFILE_C: arib_dec_init = arib_initialize_decoder_c_profile; break; default: av_log(avctx, AV_LOG_ERROR, "Unknown or unsupported profile set!\n"); return AVERROR(EINVAL); } arib_dec_init(b24->decoder); ret = libaribb24_generate_ass_header(avctx); if (ret < 0) return ret; return 0; } static int libaribb24_close(AVCodecContext *avctx) { Libaribb24Context *b24 = avctx->priv_data; if (b24->decoder) arib_finalize_decoder(b24->decoder); if (b24->lib_instance) arib_instance_destroy(b24->lib_instance); return 0; } #define RGB_TO_BGR(c) (((c) & 0xff) << 16 | ((c) & 0xff00) | (((c) >> 16) & 0xff)) static int libaribb24_handle_regions(AVCodecContext *avctx, AVSubtitle *sub) { Libaribb24Context *b24 = avctx->priv_data; const arib_buf_region_t *region = arib_decoder_get_regions(b24->decoder); unsigned int profile_font_size = get_profile_font_size(avctx); AVBPrint buf; int ret = 0; av_bprint_init(&buf, 0, AV_BPRINT_SIZE_UNLIMITED); while (region) { ptrdiff_t region_length = region->p_end - region->p_start; unsigned int ruby_region = region->i_fontheight == (profile_font_size / 2); // ASS requires us to make the colors BGR, so we convert here int foreground_bgr_color = RGB_TO_BGR(region->i_foreground_color); int background_bgr_color = RGB_TO_BGR(region->i_background_color); if (region_length < 0) { av_log(avctx, AV_LOG_ERROR, "Invalid negative region length!\n"); ret = AVERROR_INVALIDDATA; break; } if (region_length == 0 || (ruby_region && b24->aribb24_skip_ruby)) { goto next_region; } // color and alpha if (foreground_bgr_color != ASS_DEFAULT_COLOR) av_bprintf(&buf, "{\\1c&H%06x&}", foreground_bgr_color); if (region->i_foreground_alpha != 0) av_bprintf(&buf, "{\\1a&H%02x&}", region->i_foreground_alpha); if (background_bgr_color != ASS_DEFAULT_BACK_COLOR) av_bprintf(&buf, "{\\3c&H%06x&}", background_bgr_color); if (region->i_background_alpha != 0) av_bprintf(&buf, "{\\3a&H%02x&}", region->i_background_alpha); // font size if (region->i_fontwidth != profile_font_size || region->i_fontheight != profile_font_size) { av_bprintf(&buf, "{\\fscx%"PRId64"\\fscy%"PRId64"}", av_rescale(region->i_fontwidth, 100, profile_font_size), av_rescale(region->i_fontheight, 100, profile_font_size)); } // TODO: positioning av_bprint_append_data(&buf, region->p_start, region_length); av_bprintf(&buf, "{\\r}"); next_region: region = region->p_next; } if (!av_bprint_is_complete(&buf)) ret = AVERROR(ENOMEM); if (ret == 0) { av_log(avctx, AV_LOG_DEBUG, "Styled ASS line: %s\n", buf.str); sub->format = 1; /* text */ ret = ff_ass_add_rect(sub, buf.str, b24->read_order++, 0, NULL, NULL); } av_bprint_finalize(&buf, NULL); return ret; } static int libaribb24_decode(AVCodecContext *avctx, AVSubtitle *sub, int *got_sub_ptr, const AVPacket *pkt) { Libaribb24Context *b24 = avctx->priv_data; size_t parsed_data_size = 0; size_t decoded_subtitle_size = 0; const unsigned char *parsed_data = NULL; char *decoded_subtitle = NULL; time_t subtitle_duration = 0; int ret = 0; if (pkt->size <= 0) return pkt->size; arib_parse_pes(b24->parser, pkt->data, pkt->size); parsed_data = arib_parser_get_data(b24->parser, &parsed_data_size); if (!parsed_data || !parsed_data_size) { av_log(avctx, AV_LOG_DEBUG, "No decode'able data was received from " "packet (dts: %"PRId64", pts: %"PRId64").\n", pkt->dts, pkt->pts); return pkt->size; } decoded_subtitle_size = parsed_data_size * 4; if (!(decoded_subtitle = av_mallocz(decoded_subtitle_size + 1))) { av_log(avctx, AV_LOG_ERROR, "Failed to allocate buffer for decoded subtitle!\n"); return AVERROR(ENOMEM); } decoded_subtitle_size = arib_decode_buffer(b24->decoder, parsed_data, parsed_data_size, decoded_subtitle, decoded_subtitle_size); subtitle_duration = arib_decoder_get_time(b24->decoder); if (avctx->pkt_timebase.num && pkt->pts != AV_NOPTS_VALUE) sub->pts = av_rescale_q(pkt->pts, avctx->pkt_timebase, AV_TIME_BASE_Q); sub->end_display_time = subtitle_duration ? av_rescale_q(subtitle_duration, AV_TIME_BASE_Q, (AVRational){1, 1000}) : UINT32_MAX; av_log(avctx, AV_LOG_DEBUG, "Result: '%s' (size: %zu, pkt_pts: %"PRId64", sub_pts: %"PRId64" " "duration: %"PRIu32", pkt_timebase: %d/%d, time_base: %d/%d')\n", decoded_subtitle ? decoded_subtitle : "", decoded_subtitle_size, pkt->pts, sub->pts, sub->end_display_time, avctx->pkt_timebase.num, avctx->pkt_timebase.den, avctx->time_base.num, avctx->time_base.den); if (decoded_subtitle) ret = libaribb24_handle_regions(avctx, sub); *got_sub_ptr = sub->num_rects > 0; av_free(decoded_subtitle); // flush the region buffers, otherwise the linked list keeps getting // longer and longer... arib_finalize_decoder(b24->decoder); return ret < 0 ? ret : pkt->size; } static void libaribb24_flush(AVCodecContext *avctx) { Libaribb24Context *b24 = avctx->priv_data; if (!(avctx->flags2 & AV_CODEC_FLAG2_RO_FLUSH_NOOP)) b24->read_order = 0; } #define OFFSET(x) offsetof(Libaribb24Context, x) #define SD AV_OPT_FLAG_SUBTITLE_PARAM | AV_OPT_FLAG_DECODING_PARAM static const AVOption options[] = { { "aribb24-base-path", "set the base path for the libaribb24 library", OFFSET(aribb24_base_path), AV_OPT_TYPE_STRING, { .str = NULL }, 0, 0, SD }, { "aribb24-skip-ruby-text", "skip ruby text blocks during decoding", OFFSET(aribb24_skip_ruby), AV_OPT_TYPE_BOOL, { .i64 = 1 }, 0, 1, SD }, { "default_profile", "default profile to use if not specified in the stream parameters", OFFSET(default_profile), AV_OPT_TYPE_INT, { .i64 = AV_PROFILE_UNKNOWN }, AV_PROFILE_UNKNOWN, AV_PROFILE_ARIB_PROFILE_C, SD, "profile" }, {"a", "Profile A", 0, AV_OPT_TYPE_CONST, {.i64 = AV_PROFILE_ARIB_PROFILE_A}, INT_MIN, INT_MAX, SD, "profile"}, {"c", "Profile C", 0, AV_OPT_TYPE_CONST, {.i64 = AV_PROFILE_ARIB_PROFILE_C}, INT_MIN, INT_MAX, SD, "profile"}, { NULL } }; static const AVClass aribb24_class = { .class_name = "libaribb24 decoder", .option = options, .version = LIBAVUTIL_VERSION_INT, }; const FFCodec ff_libaribb24_decoder = { .p.name = "libaribb24", CODEC_LONG_NAME("libaribb24 ARIB STD-B24 caption decoder"), .p.type = AVMEDIA_TYPE_SUBTITLE, .p.id = AV_CODEC_ID_ARIB_CAPTION, .p.priv_class = &aribb24_class, .p.wrapper_name = "libaribb24", .caps_internal = FF_CODEC_CAP_INIT_CLEANUP | FF_CODEC_CAP_NOT_INIT_THREADSAFE, .priv_data_size = sizeof(Libaribb24Context), .init = libaribb24_init, .close = libaribb24_close, FF_CODEC_DECODE_SUB_CB(libaribb24_decode), .flush = libaribb24_flush, };