]> git.sesse.net Git - ffmpeg/blob - libavformat/voc_packet.c
avformat/aadec: Replace strncpy by av_strlcpy
[ffmpeg] / libavformat / voc_packet.c
1 /*
2  * Copyright (c) 2006  Aurelien Jacobs <aurel@gnuage.org>
3  *
4  * This file is part of FFmpeg.
5  *
6  * FFmpeg is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * FFmpeg is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with FFmpeg; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19  */
20
21 #include "libavutil/intreadwrite.h"
22 #include "avformat.h"
23 #include "internal.h"
24 #include "voc.h"
25
26 int
27 ff_voc_get_packet(AVFormatContext *s, AVPacket *pkt, AVStream *st, int max_size)
28 {
29     VocDecContext *voc = s->priv_data;
30     AVCodecParameters *par = st->codecpar;
31     AVIOContext *pb = s->pb;
32     VocType type;
33     int size, tmp_codec=-1;
34     int sample_rate = 0;
35     int channels = 1;
36     int64_t duration;
37     int ret;
38
39     av_add_index_entry(st,
40                        avio_tell(pb),
41                        voc->pts,
42                        voc->remaining_size,
43                        0,
44                        AVINDEX_KEYFRAME);
45
46     while (!voc->remaining_size) {
47         if (max_size < 4)
48             max_size = 0;
49         type = avio_r8(pb);
50         if (type == VOC_TYPE_EOF)
51             return AVERROR_EOF;
52         voc->remaining_size = avio_rl24(pb);
53         if (!voc->remaining_size) {
54             if (!(s->pb->seekable & AVIO_SEEKABLE_NORMAL))
55                 return AVERROR(EIO);
56             voc->remaining_size = avio_size(pb) - avio_tell(pb);
57         }
58         max_size -= 4;
59
60         switch (type) {
61         case VOC_TYPE_VOICE_DATA:
62             if (!par->sample_rate) {
63                 par->sample_rate = 1000000 / (256 - avio_r8(pb));
64                 if (sample_rate)
65                     par->sample_rate = sample_rate;
66                 avpriv_set_pts_info(st, 64, 1, par->sample_rate);
67                 par->channels = channels;
68                 par->bits_per_coded_sample = av_get_bits_per_sample(par->codec_id);
69             } else
70                 avio_skip(pb, 1);
71             tmp_codec = avio_r8(pb);
72             voc->remaining_size -= 2;
73             max_size -= 2;
74             channels = 1;
75             break;
76
77         case VOC_TYPE_VOICE_DATA_CONT:
78             break;
79
80         case VOC_TYPE_EXTENDED:
81             sample_rate = avio_rl16(pb);
82             avio_r8(pb);
83             channels = avio_r8(pb) + 1;
84             sample_rate = 256000000 / (channels * (65536 - sample_rate));
85             voc->remaining_size = 0;
86             max_size -= 4;
87             break;
88
89         case VOC_TYPE_NEW_VOICE_DATA:
90             if (!par->sample_rate) {
91                 par->sample_rate = avio_rl32(pb);
92                 avpriv_set_pts_info(st, 64, 1, par->sample_rate);
93                 par->bits_per_coded_sample = avio_r8(pb);
94                 par->channels = avio_r8(pb);
95             } else
96                 avio_skip(pb, 6);
97             tmp_codec = avio_rl16(pb);
98             avio_skip(pb, 4);
99             voc->remaining_size -= 12;
100             max_size -= 12;
101             break;
102
103         default:
104             avio_skip(pb, voc->remaining_size);
105             max_size -= voc->remaining_size;
106             voc->remaining_size = 0;
107             break;
108         }
109     }
110
111     if (par->sample_rate <= 0) {
112         av_log(s, AV_LOG_ERROR, "Invalid sample rate %d\n", par->sample_rate);
113         return AVERROR_INVALIDDATA;
114     }
115
116     if (tmp_codec >= 0) {
117         tmp_codec = ff_codec_get_id(ff_voc_codec_tags, tmp_codec);
118         if (par->codec_id == AV_CODEC_ID_NONE)
119             par->codec_id = tmp_codec;
120         else if (par->codec_id != tmp_codec)
121             av_log(s, AV_LOG_WARNING, "Ignoring mid-stream change in audio codec\n");
122         if (par->codec_id == AV_CODEC_ID_NONE) {
123             if (s->audio_codec_id == AV_CODEC_ID_NONE) {
124                 av_log(s, AV_LOG_ERROR, "unknown codec tag\n");
125                 return AVERROR(EINVAL);
126             }
127             av_log(s, AV_LOG_WARNING, "unknown codec tag\n");
128         }
129     }
130
131     par->bit_rate = (int64_t)par->sample_rate * par->channels * par->bits_per_coded_sample;
132
133     if (max_size <= 0)
134         max_size = 2048;
135     size = FFMIN(voc->remaining_size, max_size);
136     voc->remaining_size -= size;
137
138     ret = av_get_packet(pb, pkt, size);
139     pkt->dts = pkt->pts = voc->pts;
140
141     duration = av_get_audio_frame_duration2(st->codecpar, size);
142     if (duration > 0 && voc->pts != AV_NOPTS_VALUE)
143         voc->pts += duration;
144     else
145         voc->pts = AV_NOPTS_VALUE;
146
147     return ret;
148 }