1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Librsvg rasterization wrapper 3cabdff1aSopenharmony_ci * Copyright (c) 2017 Rostislav Pehlivanov <atomnuker@gmail.com> 4cabdff1aSopenharmony_ci * 5cabdff1aSopenharmony_ci * This file is part of FFmpeg. 6cabdff1aSopenharmony_ci * 7cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 8cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 9cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 10cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 11cabdff1aSopenharmony_ci * 12cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 13cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 14cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15cabdff1aSopenharmony_ci * Lesser General Public License for more details. 16cabdff1aSopenharmony_ci * 17cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 18cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 19cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 20cabdff1aSopenharmony_ci */ 21cabdff1aSopenharmony_ci 22cabdff1aSopenharmony_ci#include "avcodec.h" 23cabdff1aSopenharmony_ci#include "codec_internal.h" 24cabdff1aSopenharmony_ci#include "internal.h" 25cabdff1aSopenharmony_ci#include "libavutil/opt.h" 26cabdff1aSopenharmony_ci#include "librsvg-2.0/librsvg/rsvg.h" 27cabdff1aSopenharmony_ci 28cabdff1aSopenharmony_citypedef struct LibRSVGContext { 29cabdff1aSopenharmony_ci AVClass *class; 30cabdff1aSopenharmony_ci 31cabdff1aSopenharmony_ci int width; 32cabdff1aSopenharmony_ci int height; 33cabdff1aSopenharmony_ci int keep_ar; 34cabdff1aSopenharmony_ci} LibRSVGContext; 35cabdff1aSopenharmony_ci 36cabdff1aSopenharmony_cistatic int librsvg_decode_frame(AVCodecContext *avctx, AVFrame *frame, 37cabdff1aSopenharmony_ci int *got_frame, AVPacket *pkt) 38cabdff1aSopenharmony_ci{ 39cabdff1aSopenharmony_ci int ret; 40cabdff1aSopenharmony_ci LibRSVGContext *s = avctx->priv_data; 41cabdff1aSopenharmony_ci 42cabdff1aSopenharmony_ci RsvgHandle *handle; 43cabdff1aSopenharmony_ci RsvgDimensionData unscaled_dimensions, dimensions; 44cabdff1aSopenharmony_ci cairo_surface_t *image; 45cabdff1aSopenharmony_ci cairo_t *crender = NULL; 46cabdff1aSopenharmony_ci GError *error = NULL; 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci *got_frame = 0; 49cabdff1aSopenharmony_ci 50cabdff1aSopenharmony_ci handle = rsvg_handle_new_from_data(pkt->data, pkt->size, &error); 51cabdff1aSopenharmony_ci if (error) { 52cabdff1aSopenharmony_ci av_log(avctx, AV_LOG_ERROR, "Error parsing svg!\n"); 53cabdff1aSopenharmony_ci g_error_free(error); 54cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 55cabdff1aSopenharmony_ci } 56cabdff1aSopenharmony_ci 57cabdff1aSopenharmony_ci rsvg_handle_get_dimensions(handle, &dimensions); 58cabdff1aSopenharmony_ci rsvg_handle_get_dimensions(handle, &unscaled_dimensions); 59cabdff1aSopenharmony_ci dimensions.width = s->width ? s->width : dimensions.width; 60cabdff1aSopenharmony_ci dimensions.height = s->height ? s->height : dimensions.height; 61cabdff1aSopenharmony_ci if (s->keep_ar && (s->width || s->height)) { 62cabdff1aSopenharmony_ci double default_ar = unscaled_dimensions.width/(double)unscaled_dimensions.height; 63cabdff1aSopenharmony_ci if (!s->width) 64cabdff1aSopenharmony_ci dimensions.width = lrintf(dimensions.height * default_ar); 65cabdff1aSopenharmony_ci else 66cabdff1aSopenharmony_ci dimensions.height = lrintf(dimensions.width / default_ar); 67cabdff1aSopenharmony_ci } 68cabdff1aSopenharmony_ci 69cabdff1aSopenharmony_ci if ((ret = ff_set_dimensions(avctx, dimensions.width, dimensions.height))) 70cabdff1aSopenharmony_ci return ret; 71cabdff1aSopenharmony_ci avctx->pix_fmt = AV_PIX_FMT_RGB32; 72cabdff1aSopenharmony_ci 73cabdff1aSopenharmony_ci if ((ret = ff_get_buffer(avctx, frame, 0))) 74cabdff1aSopenharmony_ci return ret; 75cabdff1aSopenharmony_ci frame->pict_type = AV_PICTURE_TYPE_I; 76cabdff1aSopenharmony_ci frame->key_frame = 1; 77cabdff1aSopenharmony_ci 78cabdff1aSopenharmony_ci image = cairo_image_surface_create_for_data(frame->data[0], CAIRO_FORMAT_ARGB32, 79cabdff1aSopenharmony_ci frame->width, frame->height, 80cabdff1aSopenharmony_ci frame->linesize[0]); 81cabdff1aSopenharmony_ci if (cairo_surface_status(image) != CAIRO_STATUS_SUCCESS) 82cabdff1aSopenharmony_ci return AVERROR_INVALIDDATA; 83cabdff1aSopenharmony_ci 84cabdff1aSopenharmony_ci crender = cairo_create(image); 85cabdff1aSopenharmony_ci 86cabdff1aSopenharmony_ci cairo_save(crender); 87cabdff1aSopenharmony_ci cairo_set_operator(crender, CAIRO_OPERATOR_CLEAR); 88cabdff1aSopenharmony_ci cairo_paint(crender); 89cabdff1aSopenharmony_ci cairo_restore(crender); 90cabdff1aSopenharmony_ci 91cabdff1aSopenharmony_ci cairo_scale(crender, dimensions.width / (double)unscaled_dimensions.width, 92cabdff1aSopenharmony_ci dimensions.height / (double)unscaled_dimensions.height); 93cabdff1aSopenharmony_ci 94cabdff1aSopenharmony_ci rsvg_handle_render_cairo(handle, crender); 95cabdff1aSopenharmony_ci 96cabdff1aSopenharmony_ci cairo_destroy(crender); 97cabdff1aSopenharmony_ci cairo_surface_destroy(image); 98cabdff1aSopenharmony_ci g_object_unref(handle); 99cabdff1aSopenharmony_ci 100cabdff1aSopenharmony_ci *got_frame = 1; 101cabdff1aSopenharmony_ci 102cabdff1aSopenharmony_ci return 0; 103cabdff1aSopenharmony_ci} 104cabdff1aSopenharmony_ci 105cabdff1aSopenharmony_ci#define OFFSET(x) offsetof(LibRSVGContext, x) 106cabdff1aSopenharmony_ci#define DEC (AV_OPT_FLAG_DECODING_PARAM | AV_OPT_FLAG_VIDEO_PARAM) 107cabdff1aSopenharmony_cistatic const AVOption options[] = { 108cabdff1aSopenharmony_ci { "width", "Width to render to (0 for default)", OFFSET(width), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, DEC }, 109cabdff1aSopenharmony_ci { "height", "Height to render to (0 for default)", OFFSET(height), AV_OPT_TYPE_INT, { .i64 = 0 }, 0, INT_MAX, DEC }, 110cabdff1aSopenharmony_ci { "keep_ar", "Keep aspect ratio with custom width/height", OFFSET(keep_ar), AV_OPT_TYPE_BOOL, { .i64 = 1 }, 0, 1, DEC }, 111cabdff1aSopenharmony_ci { NULL }, 112cabdff1aSopenharmony_ci}; 113cabdff1aSopenharmony_ci 114cabdff1aSopenharmony_cistatic const AVClass librsvg_decoder_class = { 115cabdff1aSopenharmony_ci .class_name = "Librsvg", 116cabdff1aSopenharmony_ci .item_name = av_default_item_name, 117cabdff1aSopenharmony_ci .option = options, 118cabdff1aSopenharmony_ci .version = LIBAVUTIL_VERSION_INT, 119cabdff1aSopenharmony_ci}; 120cabdff1aSopenharmony_ci 121cabdff1aSopenharmony_ciconst FFCodec ff_librsvg_decoder = { 122cabdff1aSopenharmony_ci .p.name = "librsvg", 123cabdff1aSopenharmony_ci .p.long_name = NULL_IF_CONFIG_SMALL("Librsvg rasterizer"), 124cabdff1aSopenharmony_ci .p.priv_class = &librsvg_decoder_class, 125cabdff1aSopenharmony_ci .p.type = AVMEDIA_TYPE_VIDEO, 126cabdff1aSopenharmony_ci .p.id = AV_CODEC_ID_SVG, 127cabdff1aSopenharmony_ci .p.capabilities = AV_CODEC_CAP_DR1, 128cabdff1aSopenharmony_ci .p.wrapper_name = "librsvg", 129cabdff1aSopenharmony_ci FF_CODEC_DECODE_CB(librsvg_decode_frame), 130cabdff1aSopenharmony_ci .priv_data_size = sizeof(LibRSVGContext), 131cabdff1aSopenharmony_ci}; 132