]> git.sesse.net Git - ffmpeg/blob - libavformat/amr.c
Make avi and asf demuxer export palette in palctrl
[ffmpeg] / libavformat / amr.c
1 /* 
2  * amr file format
3  * Copyright (c) 2001 ffmpeg project
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18  */
19
20 /*
21 Write and read amr data according to RFC3267, http://www.ietf.org/rfc/rfc3267.txt?number=3267
22
23 Only mono files are supported.
24
25 */
26 #include "avformat.h"
27
28 static const unsigned char AMR_header [] = "#!AMR\n";
29 static const unsigned char AMRWB_header [] = "#!AMR-WB\n";
30
31 static int amr_write_header(AVFormatContext *s)
32 {
33     ByteIOContext *pb = &s->pb;
34     AVCodecContext *enc = &s->streams[0]->codec;
35
36     s->priv_data = NULL;
37
38     if (enc->codec_id == CODEC_ID_AMR_NB)
39     {
40         put_tag(pb, AMR_header);       /* magic number */
41     }
42     else if(enc->codec_id == CODEC_ID_AMR_WB)
43     {
44         put_tag(pb, AMRWB_header);       /* magic number */
45     }
46     else
47     {
48         //This is an error!
49     }
50     put_flush_packet(pb);
51     return 0;
52 }
53
54 static int amr_write_packet(AVFormatContext *s, int stream_index_ptr,
55                            uint8_t *buf, int size, int force_pts)
56 {
57     put_buffer(&s->pb, buf, size);
58     put_flush_packet(&s->pb);
59     return 0;
60 }
61
62 static int amr_write_trailer(AVFormatContext *s)
63 {
64     return 0;
65 }
66
67 static int amr_probe(AVProbeData *p)
68 {
69     //Only check for "#!AMR" which could be amr-wb, amr-nb. 
70     //This will also trigger multichannel files: "#!AMR_MC1.0\n" and 
71     //"#!AMR-WB_MC1.0\n" (not supported)
72
73     if (p->buf_size < 5)
74         return 0;
75     if(memcmp(p->buf,AMR_header,5)==0)
76         return AVPROBE_SCORE_MAX;
77     else
78         return 0;
79 }
80
81 /* amr input */
82 static int amr_read_header(AVFormatContext *s,
83                            AVFormatParameters *ap)
84 {
85     ByteIOContext *pb = &s->pb;
86     AVStream *st;
87     uint8_t header[9];
88
89     get_buffer(pb, header, 6);
90
91     if(memcmp(header,AMR_header,6)!=0)
92     {
93         get_buffer(pb, header+6, 3);
94         if(memcmp(header,AMRWB_header,9)!=0)
95         {
96             return -1;
97         }
98         st = av_new_stream(s, 0);
99         if (!st)
100         {
101             return AVERROR_NOMEM;
102         }
103     
104         st->codec.codec_type = CODEC_TYPE_AUDIO;
105         st->codec.codec_tag = CODEC_ID_AMR_WB;
106         st->codec.codec_id = CODEC_ID_AMR_WB;
107         st->codec.channels = 1;
108         st->codec.sample_rate = 16000;
109     }
110     else
111     {
112         st = av_new_stream(s, 0);
113         if (!st)
114         {
115             return AVERROR_NOMEM;
116         }
117     
118         st->codec.codec_type = CODEC_TYPE_AUDIO;
119         st->codec.codec_tag = CODEC_ID_AMR_NB;
120         st->codec.codec_id = CODEC_ID_AMR_NB;
121         st->codec.channels = 1;
122         st->codec.sample_rate = 8000;
123     }
124
125     return 0;
126 }
127
128 #define MAX_SIZE 32
129
130 static int amr_read_packet(AVFormatContext *s,
131                           AVPacket *pkt)
132 {
133     AVCodecContext *enc = &s->streams[0]->codec;
134
135     if (enc->codec_id == CODEC_ID_AMR_NB)
136     {
137         const static uint8_t packed_size[16] = {12, 13, 15, 17, 19, 20, 26, 31, 5, 0, 0, 0, 0, 0, 0, 0};
138         uint8_t toc, q, ft;
139         int read;
140         int size;
141     
142         if (url_feof(&s->pb))
143         {
144             return -EIO;
145         }
146     
147         toc=get_byte(&s->pb);
148         q  = (toc >> 2) & 0x01;
149         ft = (toc >> 3) & 0x0F;
150     
151         size=packed_size[ft];
152     
153         if (av_new_packet(pkt, size+1))
154         {
155             return -EIO;
156         }
157         pkt->stream_index = 0;
158         
159         pkt->data[0]=toc;
160     
161         read = get_buffer(&s->pb, pkt->data+1, size);
162     
163         if (read != size)
164         {
165             av_free_packet(pkt);
166             return -EIO;
167         }
168     
169         return 0;
170     }
171     else if(enc->codec_id == CODEC_ID_AMR_WB)
172     {
173         static uint8_t packed_size[16] = {18, 24, 33, 37, 41, 47, 51, 59, 61, 6, 6, 0, 0, 0, 1, 1};
174         uint8_t toc, mode;
175         int read;
176         int size;
177     
178         if (url_feof(&s->pb))
179         {
180             return -EIO;
181         }
182     
183         toc=get_byte(&s->pb);
184         mode = (uint8_t)((toc >> 3) & 0x0F);
185         size = packed_size[mode];
186     
187         if ( (size==0) || av_new_packet(pkt, size))
188         {
189             return -EIO;
190         }
191     
192         pkt->stream_index = 0;
193         pkt->data[0]=toc;
194     
195         read = get_buffer(&s->pb, pkt->data+1, size-1);
196     
197         if (read != (size-1))
198         {
199             av_free_packet(pkt);
200             return -EIO;
201         }
202     
203         return 0;
204     }
205     else
206     {
207         return -EIO;
208     }
209 }
210
211 static int amr_read_close(AVFormatContext *s)
212 {
213     return 0;
214 }
215
216 static AVInputFormat amr_iformat = {
217     "amr",
218     "3gpp amr file format",
219     0, /*priv_data_size*/
220     amr_probe,
221     amr_read_header,
222     amr_read_packet,
223     amr_read_close,
224 };
225
226 static AVOutputFormat amr_oformat = {
227     "amr",
228     "3gpp amr file format",
229     "audio/amr",
230     "amr",
231     0,
232     CODEC_ID_AMR_NB,
233     CODEC_ID_NONE,
234     amr_write_header,
235     amr_write_packet,
236     amr_write_trailer,
237 };
238
239 int amr_init(void)
240 {
241     av_register_input_format(&amr_iformat);
242     av_register_output_format(&amr_oformat);
243     return 0;
244 }