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