1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * PNG parser 3cabdff1aSopenharmony_ci * Copyright (c) 2009 Peter Holik 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/** 23cabdff1aSopenharmony_ci * @file 24cabdff1aSopenharmony_ci * PNG parser 25cabdff1aSopenharmony_ci */ 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_ci#include "parser.h" 28cabdff1aSopenharmony_ci#include "png.h" 29cabdff1aSopenharmony_ci 30cabdff1aSopenharmony_citypedef struct PNGParseContext { 31cabdff1aSopenharmony_ci ParseContext pc; 32cabdff1aSopenharmony_ci uint32_t chunk_pos; ///< position inside current chunk 33cabdff1aSopenharmony_ci uint32_t chunk_length; ///< length of the current chunk 34cabdff1aSopenharmony_ci uint32_t remaining_size; ///< remaining size of the current chunk 35cabdff1aSopenharmony_ci} PNGParseContext; 36cabdff1aSopenharmony_ci 37cabdff1aSopenharmony_cistatic int png_parse(AVCodecParserContext *s, AVCodecContext *avctx, 38cabdff1aSopenharmony_ci const uint8_t **poutbuf, int *poutbuf_size, 39cabdff1aSopenharmony_ci const uint8_t *buf, int buf_size) 40cabdff1aSopenharmony_ci{ 41cabdff1aSopenharmony_ci PNGParseContext *ppc = s->priv_data; 42cabdff1aSopenharmony_ci int next = END_NOT_FOUND; 43cabdff1aSopenharmony_ci int i = 0; 44cabdff1aSopenharmony_ci 45cabdff1aSopenharmony_ci s->pict_type = AV_PICTURE_TYPE_NONE; 46cabdff1aSopenharmony_ci 47cabdff1aSopenharmony_ci *poutbuf_size = 0; 48cabdff1aSopenharmony_ci *poutbuf = NULL; 49cabdff1aSopenharmony_ci 50cabdff1aSopenharmony_ci if (!ppc->pc.frame_start_found) { 51cabdff1aSopenharmony_ci uint64_t state64 = ppc->pc.state64; 52cabdff1aSopenharmony_ci for (; i < buf_size; i++) { 53cabdff1aSopenharmony_ci state64 = (state64 << 8) | buf[i]; 54cabdff1aSopenharmony_ci if (state64 == PNGSIG || state64 == MNGSIG) { 55cabdff1aSopenharmony_ci i++; 56cabdff1aSopenharmony_ci ppc->pc.frame_start_found = 1; 57cabdff1aSopenharmony_ci break; 58cabdff1aSopenharmony_ci } 59cabdff1aSopenharmony_ci } 60cabdff1aSopenharmony_ci ppc->pc.state64 = state64; 61cabdff1aSopenharmony_ci } else if (ppc->remaining_size) { 62cabdff1aSopenharmony_ci i = FFMIN(ppc->remaining_size, buf_size); 63cabdff1aSopenharmony_ci ppc->remaining_size -= i; 64cabdff1aSopenharmony_ci if (ppc->remaining_size) 65cabdff1aSopenharmony_ci goto flush; 66cabdff1aSopenharmony_ci if (ppc->chunk_pos == -1) { 67cabdff1aSopenharmony_ci next = i; 68cabdff1aSopenharmony_ci goto flush; 69cabdff1aSopenharmony_ci } 70cabdff1aSopenharmony_ci } 71cabdff1aSopenharmony_ci 72cabdff1aSopenharmony_ci for (; ppc->pc.frame_start_found && i < buf_size; i++) { 73cabdff1aSopenharmony_ci ppc->pc.state = (ppc->pc.state << 8) | buf[i]; 74cabdff1aSopenharmony_ci if (ppc->chunk_pos == 3) { 75cabdff1aSopenharmony_ci ppc->chunk_length = ppc->pc.state; 76cabdff1aSopenharmony_ci if (ppc->chunk_length > 0x7fffffff) { 77cabdff1aSopenharmony_ci ppc->chunk_pos = ppc->pc.frame_start_found = 0; 78cabdff1aSopenharmony_ci goto flush; 79cabdff1aSopenharmony_ci } 80cabdff1aSopenharmony_ci ppc->chunk_length += 4; 81cabdff1aSopenharmony_ci } else if (ppc->chunk_pos == 7) { 82cabdff1aSopenharmony_ci if (ppc->chunk_length >= buf_size - i) 83cabdff1aSopenharmony_ci ppc->remaining_size = ppc->chunk_length - buf_size + i + 1; 84cabdff1aSopenharmony_ci if (ppc->pc.state == MKBETAG('I', 'E', 'N', 'D')) { 85cabdff1aSopenharmony_ci if (ppc->remaining_size) 86cabdff1aSopenharmony_ci ppc->chunk_pos = -1; 87cabdff1aSopenharmony_ci else 88cabdff1aSopenharmony_ci next = ppc->chunk_length + i + 1; 89cabdff1aSopenharmony_ci break; 90cabdff1aSopenharmony_ci } else { 91cabdff1aSopenharmony_ci ppc->chunk_pos = 0; 92cabdff1aSopenharmony_ci if (ppc->remaining_size) 93cabdff1aSopenharmony_ci break; 94cabdff1aSopenharmony_ci else 95cabdff1aSopenharmony_ci i += ppc->chunk_length; 96cabdff1aSopenharmony_ci continue; 97cabdff1aSopenharmony_ci } 98cabdff1aSopenharmony_ci } 99cabdff1aSopenharmony_ci ppc->chunk_pos++; 100cabdff1aSopenharmony_ci } 101cabdff1aSopenharmony_ci 102cabdff1aSopenharmony_ciflush: 103cabdff1aSopenharmony_ci if (ff_combine_frame(&ppc->pc, next, &buf, &buf_size) < 0) 104cabdff1aSopenharmony_ci return buf_size; 105cabdff1aSopenharmony_ci 106cabdff1aSopenharmony_ci ppc->chunk_pos = ppc->pc.frame_start_found = 0; 107cabdff1aSopenharmony_ci 108cabdff1aSopenharmony_ci *poutbuf = buf; 109cabdff1aSopenharmony_ci *poutbuf_size = buf_size; 110cabdff1aSopenharmony_ci return next; 111cabdff1aSopenharmony_ci} 112cabdff1aSopenharmony_ci 113cabdff1aSopenharmony_ciconst AVCodecParser ff_png_parser = { 114cabdff1aSopenharmony_ci .codec_ids = { AV_CODEC_ID_PNG }, 115cabdff1aSopenharmony_ci .priv_data_size = sizeof(PNGParseContext), 116cabdff1aSopenharmony_ci .parser_parse = png_parse, 117cabdff1aSopenharmony_ci .parser_close = ff_parse_close, 118cabdff1aSopenharmony_ci}; 119