Mercurial > libavformat.hg
comparison tta.c @ 948:1e766711e6c8 libavformat
tta demuxer, also usable for moving tta audio data into an other container
author | alex |
---|---|
date | Sun, 12 Feb 2006 02:18:37 +0000 |
parents | |
children | 99a7f76a8954 |
comparison
equal
deleted
inserted
replaced
947:929f84cc63c3 | 948:1e766711e6c8 |
---|---|
1 /* | |
2 * TTA demuxer | |
3 * Copyright (c) 2006 Alex Beregszaszi | |
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 #include "avformat.h" | |
20 #define ALT_BITSREAM_READER_LE | |
21 #include "bitstream.h" | |
22 | |
23 typedef struct { | |
24 int totalframes, currentframe; | |
25 uint32_t *seektable; | |
26 } TTAContext; | |
27 | |
28 static int tta_probe(AVProbeData *p) | |
29 { | |
30 const uint8_t *d = p->buf; | |
31 if (p->buf_size < 4) | |
32 return 0; | |
33 if (d[0] == 'T' && d[1] == 'T' && d[2] == 'A' && d[3] == '1') | |
34 return 80; | |
35 return 0; | |
36 } | |
37 | |
38 static int tta_read_header(AVFormatContext *s, AVFormatParameters *ap) | |
39 { | |
40 TTAContext *c = s->priv_data; | |
41 AVStream *st; | |
42 int i, channels, bps, samplerate, datalen, framelen, start; | |
43 | |
44 start = url_ftell(&s->pb); | |
45 | |
46 if (get_le32(&s->pb) != ff_get_fourcc("TTA1")) | |
47 return -1; // not tta file | |
48 | |
49 url_fskip(&s->pb, 2); // FIXME: flags | |
50 channels = get_le16(&s->pb); | |
51 bps = get_le16(&s->pb); | |
52 samplerate = get_le32(&s->pb); | |
53 datalen = get_le32(&s->pb); | |
54 url_fskip(&s->pb, 4); // header crc | |
55 | |
56 framelen = 1.04489795918367346939 * samplerate; | |
57 c->totalframes = datalen / framelen + ((datalen % framelen) ? 1 : 0); | |
58 c->currentframe = 0; | |
59 | |
60 c->seektable = av_mallocz(sizeof(uint32_t)*c->totalframes); | |
61 if (!c->seektable) | |
62 return AVERROR_NOMEM; | |
63 | |
64 for (i = 0; i < c->totalframes; i++) | |
65 c->seektable[i] = get_le32(&s->pb); | |
66 url_fskip(&s->pb, 4); // seektable crc | |
67 | |
68 st = av_new_stream(s, 0); | |
69 // av_set_pts_info(st, 32, 1, 1000); | |
70 if (!st) | |
71 return AVERROR_NOMEM; | |
72 st->codec->codec_type = CODEC_TYPE_AUDIO; | |
73 st->codec->codec_id = CODEC_ID_TTA; | |
74 st->codec->channels = channels; | |
75 st->codec->sample_rate = samplerate; | |
76 st->codec->bits_per_sample = bps; | |
77 | |
78 st->codec->extradata_size = url_ftell(&s->pb) - start; | |
79 st->codec->extradata = av_mallocz(st->codec->extradata_size); | |
80 url_fseek(&s->pb, start, SEEK_SET); // or SEEK_CUR and -size ? :) | |
81 get_buffer(&s->pb, st->codec->extradata, st->codec->extradata_size); | |
82 | |
83 return 0; | |
84 } | |
85 | |
86 static int tta_read_packet(AVFormatContext *s, AVPacket *pkt) | |
87 { | |
88 TTAContext *c = s->priv_data; | |
89 int ret, size; | |
90 | |
91 // FIXME! | |
92 if (c->currentframe > c->totalframes) | |
93 size = 0; | |
94 else | |
95 size = c->seektable[c->currentframe]; | |
96 | |
97 c->currentframe++; | |
98 | |
99 if (av_new_packet(pkt, size) < 0) | |
100 return AVERROR_IO; | |
101 | |
102 pkt->pos = url_ftell(&s->pb); | |
103 pkt->stream_index = 0; | |
104 ret = get_buffer(&s->pb, pkt->data, size); | |
105 if (ret <= 0) { | |
106 av_free_packet(pkt); | |
107 return AVERROR_IO; | |
108 } | |
109 pkt->size = ret; | |
110 // av_log(s, AV_LOG_INFO, "TTA packet #%d desired size: %d read size: %d at pos %d\n", | |
111 // c->currentframe, size, ret, pkt->pos); | |
112 return 0; //ret; | |
113 } | |
114 | |
115 static int tta_read_close(AVFormatContext *s) | |
116 { | |
117 TTAContext *c = s->priv_data; | |
118 if (c->seektable) | |
119 av_free(c->seektable); | |
120 return 0; | |
121 } | |
122 | |
123 AVInputFormat tta_iformat = { | |
124 "tta", | |
125 "true-audio", | |
126 sizeof(TTAContext), | |
127 tta_probe, | |
128 tta_read_header, | |
129 tta_read_packet, | |
130 tta_read_close, | |
131 .extensions = "tta", | |
132 }; | |
133 | |
134 int tta_init(void) | |
135 { | |
136 av_register_input_format(&tta_iformat); | |
137 return 0; | |
138 } |