1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * PCM common functions 3cabdff1aSopenharmony_ci * Copyright (c) 2003 Fabrice Bellard 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 "libavutil/mathematics.h" 23cabdff1aSopenharmony_ci#include "avformat.h" 24cabdff1aSopenharmony_ci#include "internal.h" 25cabdff1aSopenharmony_ci#include "pcm.h" 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_ci#define RAW_SAMPLES 1024 28cabdff1aSopenharmony_ci 29cabdff1aSopenharmony_ciint ff_pcm_read_packet(AVFormatContext *s, AVPacket *pkt) 30cabdff1aSopenharmony_ci{ 31cabdff1aSopenharmony_ci AVCodecParameters *par = s->streams[0]->codecpar; 32cabdff1aSopenharmony_ci int ret, size; 33cabdff1aSopenharmony_ci 34cabdff1aSopenharmony_ci if (par->block_align <= 0) 35cabdff1aSopenharmony_ci return AVERROR(EINVAL); 36cabdff1aSopenharmony_ci 37cabdff1aSopenharmony_ci /* 38cabdff1aSopenharmony_ci * Compute read size to complete a read every 62ms. 39cabdff1aSopenharmony_ci * Clamp to RAW_SAMPLES if larger. 40cabdff1aSopenharmony_ci */ 41cabdff1aSopenharmony_ci size = FFMAX(par->sample_rate/25, 1); 42cabdff1aSopenharmony_ci if (par->block_align <= INT_MAX / RAW_SAMPLES) { 43cabdff1aSopenharmony_ci size = FFMIN(size, RAW_SAMPLES) * par->block_align; 44cabdff1aSopenharmony_ci } else { 45cabdff1aSopenharmony_ci size = par->block_align; 46cabdff1aSopenharmony_ci } 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_ci ret = av_get_packet(s->pb, pkt, size); 49cabdff1aSopenharmony_ci 50cabdff1aSopenharmony_ci pkt->flags &= ~AV_PKT_FLAG_CORRUPT; 51cabdff1aSopenharmony_ci pkt->stream_index = 0; 52cabdff1aSopenharmony_ci 53cabdff1aSopenharmony_ci return ret; 54cabdff1aSopenharmony_ci} 55cabdff1aSopenharmony_ci 56cabdff1aSopenharmony_ciint ff_pcm_read_seek(AVFormatContext *s, 57cabdff1aSopenharmony_ci int stream_index, int64_t timestamp, int flags) 58cabdff1aSopenharmony_ci{ 59cabdff1aSopenharmony_ci AVStream *st; 60cabdff1aSopenharmony_ci int block_align, byte_rate; 61cabdff1aSopenharmony_ci int64_t pos, ret; 62cabdff1aSopenharmony_ci 63cabdff1aSopenharmony_ci st = s->streams[0]; 64cabdff1aSopenharmony_ci 65cabdff1aSopenharmony_ci block_align = st->codecpar->block_align ? st->codecpar->block_align : 66cabdff1aSopenharmony_ci (av_get_bits_per_sample(st->codecpar->codec_id) * st->codecpar->ch_layout.nb_channels) >> 3; 67cabdff1aSopenharmony_ci byte_rate = st->codecpar->bit_rate ? st->codecpar->bit_rate >> 3 : 68cabdff1aSopenharmony_ci block_align * st->codecpar->sample_rate; 69cabdff1aSopenharmony_ci 70cabdff1aSopenharmony_ci if (block_align <= 0 || byte_rate <= 0) 71cabdff1aSopenharmony_ci return -1; 72cabdff1aSopenharmony_ci if (timestamp < 0) timestamp = 0; 73cabdff1aSopenharmony_ci 74cabdff1aSopenharmony_ci /* compute the position by aligning it to block_align */ 75cabdff1aSopenharmony_ci pos = av_rescale_rnd(timestamp * byte_rate, 76cabdff1aSopenharmony_ci st->time_base.num, 77cabdff1aSopenharmony_ci st->time_base.den * (int64_t)block_align, 78cabdff1aSopenharmony_ci (flags & AVSEEK_FLAG_BACKWARD) ? AV_ROUND_DOWN : AV_ROUND_UP); 79cabdff1aSopenharmony_ci pos *= block_align; 80cabdff1aSopenharmony_ci 81cabdff1aSopenharmony_ci /* recompute exact position */ 82cabdff1aSopenharmony_ci ffstream(st)->cur_dts = av_rescale(pos, st->time_base.den, byte_rate * (int64_t)st->time_base.num); 83cabdff1aSopenharmony_ci if ((ret = avio_seek(s->pb, pos + ffformatcontext(s)->data_offset, SEEK_SET)) < 0) 84cabdff1aSopenharmony_ci return ret; 85cabdff1aSopenharmony_ci return 0; 86cabdff1aSopenharmony_ci} 87