1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * Copyright (c) 2011 Justin Ruggles 3cabdff1aSopenharmony_ci * 4cabdff1aSopenharmony_ci * This file is part of FFmpeg. 5cabdff1aSopenharmony_ci * 6cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 7cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 8cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 9cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 10cabdff1aSopenharmony_ci * 11cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 12cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 13cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14cabdff1aSopenharmony_ci * Lesser General Public License for more details. 15cabdff1aSopenharmony_ci * 16cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 17cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 18cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 19cabdff1aSopenharmony_ci */ 20cabdff1aSopenharmony_ci 21cabdff1aSopenharmony_ci/** 22cabdff1aSopenharmony_ci * @file 23cabdff1aSopenharmony_ci * ADX audio parser 24cabdff1aSopenharmony_ci * 25cabdff1aSopenharmony_ci * Splits packets into individual blocks. 26cabdff1aSopenharmony_ci */ 27cabdff1aSopenharmony_ci 28cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 29cabdff1aSopenharmony_ci#include "parser.h" 30cabdff1aSopenharmony_ci#include "adx.h" 31cabdff1aSopenharmony_ci 32cabdff1aSopenharmony_citypedef struct ADXParseContext { 33cabdff1aSopenharmony_ci ParseContext pc; 34cabdff1aSopenharmony_ci int header_size; 35cabdff1aSopenharmony_ci int block_size; 36cabdff1aSopenharmony_ci int remaining; 37cabdff1aSopenharmony_ci} ADXParseContext; 38cabdff1aSopenharmony_ci 39cabdff1aSopenharmony_cistatic int adx_parse(AVCodecParserContext *s1, 40cabdff1aSopenharmony_ci AVCodecContext *avctx, 41cabdff1aSopenharmony_ci const uint8_t **poutbuf, int *poutbuf_size, 42cabdff1aSopenharmony_ci const uint8_t *buf, int buf_size) 43cabdff1aSopenharmony_ci{ 44cabdff1aSopenharmony_ci ADXParseContext *s = s1->priv_data; 45cabdff1aSopenharmony_ci ParseContext *pc = &s->pc; 46cabdff1aSopenharmony_ci int next = END_NOT_FOUND; 47cabdff1aSopenharmony_ci int i; 48cabdff1aSopenharmony_ci uint64_t state = pc->state64; 49cabdff1aSopenharmony_ci 50cabdff1aSopenharmony_ci if (!s->header_size) { 51cabdff1aSopenharmony_ci for (i = 0; i < buf_size; i++) { 52cabdff1aSopenharmony_ci state = (state << 8) | buf[i]; 53cabdff1aSopenharmony_ci /* check for fixed fields in ADX header for possible match */ 54cabdff1aSopenharmony_ci if ((state & 0xFFFF0000FFFFFF00) == 0x8000000003120400ULL) { 55cabdff1aSopenharmony_ci int channels = state & 0xFF; 56cabdff1aSopenharmony_ci int header_size = ((state >> 32) & 0xFFFF) + 4; 57cabdff1aSopenharmony_ci if (channels > 0 && header_size >= 8) { 58cabdff1aSopenharmony_ci s->header_size = header_size; 59cabdff1aSopenharmony_ci s->block_size = BLOCK_SIZE * channels; 60cabdff1aSopenharmony_ci s->remaining = i - 7 + s->header_size + s->block_size; 61cabdff1aSopenharmony_ci break; 62cabdff1aSopenharmony_ci } 63cabdff1aSopenharmony_ci } 64cabdff1aSopenharmony_ci } 65cabdff1aSopenharmony_ci pc->state64 = state; 66cabdff1aSopenharmony_ci } 67cabdff1aSopenharmony_ci 68cabdff1aSopenharmony_ci if (s->header_size) { 69cabdff1aSopenharmony_ci if (!s->remaining) 70cabdff1aSopenharmony_ci s->remaining = s->block_size; 71cabdff1aSopenharmony_ci if (s->remaining <= buf_size) { 72cabdff1aSopenharmony_ci next = s->remaining; 73cabdff1aSopenharmony_ci s->remaining = 0; 74cabdff1aSopenharmony_ci } else 75cabdff1aSopenharmony_ci s->remaining -= buf_size; 76cabdff1aSopenharmony_ci } 77cabdff1aSopenharmony_ci 78cabdff1aSopenharmony_ci if (ff_combine_frame(pc, next, &buf, &buf_size) < 0 || !buf_size) { 79cabdff1aSopenharmony_ci *poutbuf = NULL; 80cabdff1aSopenharmony_ci *poutbuf_size = 0; 81cabdff1aSopenharmony_ci return buf_size; 82cabdff1aSopenharmony_ci } 83cabdff1aSopenharmony_ci 84cabdff1aSopenharmony_ci s1->duration = BLOCK_SAMPLES; 85cabdff1aSopenharmony_ci 86cabdff1aSopenharmony_ci *poutbuf = buf; 87cabdff1aSopenharmony_ci *poutbuf_size = buf_size; 88cabdff1aSopenharmony_ci return next; 89cabdff1aSopenharmony_ci} 90cabdff1aSopenharmony_ci 91cabdff1aSopenharmony_ciconst AVCodecParser ff_adx_parser = { 92cabdff1aSopenharmony_ci .codec_ids = { AV_CODEC_ID_ADPCM_ADX }, 93cabdff1aSopenharmony_ci .priv_data_size = sizeof(ADXParseContext), 94cabdff1aSopenharmony_ci .parser_parse = adx_parse, 95cabdff1aSopenharmony_ci .parser_close = ff_parse_close, 96cabdff1aSopenharmony_ci}; 97