10034
|
1 /*
|
|
2 Realaudio demuxer for MPlayer
|
|
3 (c) 2003 Roberto Togni
|
|
4 */
|
|
5
|
|
6 #include <stdio.h>
|
|
7 #include <stdlib.h>
|
|
8 #include <unistd.h>
|
|
9
|
|
10 #include "config.h"
|
|
11 #include "mp_msg.h"
|
|
12 #include "help_mp.h"
|
|
13
|
|
14 #include "stream.h"
|
|
15 #include "demuxer.h"
|
|
16 #include "stheader.h"
|
|
17 #include "bswap.h"
|
|
18
|
|
19
|
|
20 #define FOURCC_DOTRA mmioFOURCC('.','r','a', 0xfd)
|
|
21 #define FOURCC_144 mmioFOURCC('1','4','_','4')
|
|
22 #define FOURCC_288 mmioFOURCC('2','8','_','8')
|
|
23 #define FOURCC_DNET mmioFOURCC('d','n','e','t')
|
|
24 #define FOURCC_LPCJ mmioFOURCC('l','p','c','J')
|
|
25
|
|
26
|
|
27 typedef struct {
|
|
28 unsigned short version;
|
|
29 unsigned int dotranum;
|
|
30 unsigned int data_size;
|
|
31 unsigned short version2;
|
|
32 unsigned int hdr_size;
|
|
33 unsigned short codec_flavor;
|
|
34 unsigned int coded_framesize;
|
|
35 unsigned short sub_packet_h;
|
|
36 unsigned short frame_size;
|
|
37 unsigned short sub_packet_size;
|
|
38 char genr[4];
|
|
39 } ra_priv_t;
|
|
40
|
|
41
|
|
42
|
|
43 int ra_check_file(demuxer_t* demuxer)
|
|
44 {
|
|
45 unsigned int chunk_id;
|
|
46
|
|
47 stream_reset(demuxer->stream);
|
|
48 stream_seek(demuxer->stream, 0);
|
|
49
|
|
50 chunk_id = stream_read_dword_le(demuxer->stream);
|
|
51 if (chunk_id == FOURCC_DOTRA)
|
|
52 return 1;
|
|
53 else
|
|
54 return 0;
|
|
55 }
|
|
56
|
|
57
|
|
58
|
|
59 void hexdump(char *, unsigned long);
|
|
60
|
|
61 // return value:
|
|
62 // 0 = EOF or no stream found
|
|
63 // 1 = successfully read a packet
|
|
64 int demux_ra_fill_buffer(demuxer_t *demuxer)
|
|
65 {
|
|
66 ra_priv_t *ra_priv = demuxer->priv;
|
|
67 int len;
|
|
68 int timestamp;
|
|
69 int flags;
|
|
70 demux_stream_t *ds = demuxer->audio;
|
|
71 sh_audio_t *sh = ds->sh;
|
|
72 WAVEFORMATEX *wf = sh->wf;
|
|
73 demux_packet_t *dp;
|
|
74
|
|
75 if (demuxer->stream->eof)
|
|
76 return 0;
|
|
77
|
|
78 len = wf->nBlockAlign;
|
|
79 demuxer->filepos = stream_tell(demuxer->stream);
|
|
80
|
|
81 dp = new_demux_packet(len);
|
|
82 stream_read(demuxer->stream, dp->buffer, len);
|
|
83
|
|
84 if (sh->format == 0x2000) {
|
|
85 // if DNET, swap bytes, as DNET is byte-swapped AC3:
|
|
86 char *ptr = dp->buffer;
|
|
87 int i;
|
|
88 for (i = 0; i < len; i += 2) {
|
|
89 const char tmp = ptr[0];
|
|
90 ptr[0] = ptr[1];
|
|
91 ptr[1] = tmp;
|
|
92 ptr += 2;
|
|
93 }
|
|
94 }
|
|
95 dp->pts = demuxer->filepos / ra_priv->data_size;;
|
|
96 dp->pos = demuxer->filepos;
|
|
97 dp->flags = 0;
|
|
98 ds_add_packet(ds, dp);
|
|
99
|
|
100 return 1;
|
|
101 }
|
|
102
|
|
103
|
|
104
|
|
105 extern void print_wave_header(WAVEFORMATEX *h);
|
|
106
|
|
107
|
|
108
|
|
109 void demux_open_ra(demuxer_t* demuxer)
|
|
110 {
|
|
111 ra_priv_t* ra_priv = demuxer->priv;
|
|
112 sh_audio_t *sh;
|
|
113 int i;
|
|
114 char *buf;
|
|
115
|
|
116 if ((ra_priv = (ra_priv_t *)malloc(sizeof(ra_priv_t))) == NULL) {
|
|
117 mp_msg(MSGT_DEMUX, MSGL_ERR, "[RealAudio] Can't allocate memory for private data.\n");
|
|
118 return NULL;
|
|
119 }
|
|
120 memset(ra_priv, 0, sizeof(ra_priv_t));
|
|
121
|
|
122 demuxer->priv = ra_priv;
|
|
123 sh = new_sh_audio(demuxer, 0);
|
|
124 demuxer->audio->id = 0;
|
|
125 sh->ds=demuxer->audio;
|
|
126 demuxer->audio->sh = sh;
|
|
127
|
|
128 ra_priv->version = stream_read_word(demuxer->stream);
|
|
129 mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] File version: %d\n", ra_priv->version);
|
|
130 if ((ra_priv->version < 3) || (ra_priv->version > 4)) {
|
|
131 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] ra version %d is not supported yet, please "
|
|
132 "contact MPlayer developers\n", ra_priv->version);
|
|
133 return NULL;
|
|
134 }
|
|
135 if (ra_priv->version == 3) {
|
|
136 ra_priv->hdr_size = stream_read_word(demuxer->stream);
|
|
137 stream_skip(demuxer->stream, 10);
|
|
138 ra_priv->data_size = stream_read_dword(demuxer->stream);
|
|
139 } else {
|
|
140 stream_skip(demuxer->stream, 2);
|
|
141 ra_priv->dotranum = stream_read_dword(demuxer->stream);
|
|
142 ra_priv->data_size = stream_read_dword(demuxer->stream);
|
|
143 ra_priv->version2 = stream_read_word(demuxer->stream);
|
|
144 ra_priv->hdr_size = stream_read_dword(demuxer->stream);
|
|
145 ra_priv->codec_flavor = stream_read_word(demuxer->stream);
|
|
146 ra_priv->coded_framesize = stream_read_dword(demuxer->stream);
|
|
147 stream_skip(demuxer->stream, 4); // data size?
|
|
148 stream_skip(demuxer->stream, 8);
|
|
149 ra_priv->sub_packet_h = stream_read_word(demuxer->stream);
|
|
150 ra_priv->frame_size = stream_read_word(demuxer->stream);
|
|
151 mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Frame size: %d\n", ra_priv->frame_size);
|
|
152 ra_priv->sub_packet_size = stream_read_word(demuxer->stream);
|
|
153 mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] Sub packet size: %d\n", ra_priv->sub_packet_size);
|
|
154 stream_skip(demuxer->stream, 2);
|
|
155 sh->samplerate = stream_read_word(demuxer->stream);
|
|
156 stream_skip(demuxer->stream, 2);
|
|
157 sh->samplesize = stream_read_word(demuxer->stream);
|
|
158 sh->channels = stream_read_word(demuxer->stream);
|
|
159 mp_msg(MSGT_DEMUX,MSGL_V,"[RealAudio] %d channel, %d bit, %dHz\n", sh->channels,
|
|
160 sh->samplesize, sh->samplerate);
|
|
161 i = stream_read_char(demuxer->stream);
|
|
162 *((unsigned int *)(ra_priv->genr)) = stream_read_dword(demuxer->stream);
|
|
163 if (i != 4) {
|
|
164 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] Genr size is not 4 (%d), please report to "
|
|
165 "MPlayer developers\n", i);
|
|
166 stream_skip(demuxer->stream, i - 4);
|
|
167 }
|
|
168 i = stream_read_char(demuxer->stream);
|
|
169 sh->format = stream_read_dword_le(demuxer->stream);
|
|
170 if (i != 4) {
|
|
171 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] FourCC size is not 4 (%d), please report to "
|
|
172 "MPlayer developers\n", i);
|
|
173 stream_skip(demuxer->stream, i - 4);
|
|
174 }
|
|
175 stream_skip(demuxer->stream, 3);
|
|
176 }
|
|
177
|
|
178 if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
179 buf = malloc(i+1);
|
|
180 stream_read(demuxer->stream, buf, i);
|
|
181 buf[i] = 0;
|
|
182 demux_info_add(demuxer, "Title", buf);
|
|
183 free(buf);
|
|
184 }
|
|
185 if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
186 buf = malloc(i+1);
|
|
187 stream_read(demuxer->stream, buf, i);
|
|
188 buf[i] = 0;
|
|
189 demux_info_add(demuxer, "Author", buf);
|
|
190 free(buf);
|
|
191 }
|
|
192 if ((i = stream_read_char(demuxer->stream)) != 0) {
|
|
193 buf = malloc(i+1);
|
|
194 stream_read(demuxer->stream, buf, i);
|
|
195 buf[i] = 0;
|
|
196 demux_info_add(demuxer, "Copyright", buf);
|
|
197 free(buf);
|
|
198 }
|
|
199
|
|
200 if ((i = stream_read_char(demuxer->stream)) != 0)
|
|
201 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] Last header byte is not zero!\n");
|
|
202
|
|
203 if (ra_priv->version == 3) {
|
|
204 stream_skip(demuxer->stream, 1);
|
|
205 i = stream_read_char(demuxer->stream);
|
|
206 sh->format = stream_read_dword_le(demuxer->stream);
|
|
207 if (i != 4) {
|
|
208 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] FourCC size is not 4 (%d), please report to "
|
|
209 "MPlayer developers\n", i);
|
|
210 stream_skip(demuxer->stream, i - 4);
|
|
211 }
|
|
212 // stream_skip(demuxer->stream, 3);
|
|
213
|
|
214 if (sh->format != FOURCC_LPCJ) {
|
|
215 mp_msg(MSGT_DEMUX,MSGL_WARN,"[RealAudio] Version 3 with FourCC %8x, please report to "
|
|
216 "MPlayer developers\n", sh->format);
|
|
217 }
|
|
218
|
|
219 sh->channels = 1;
|
|
220 sh->samplesize = 16;
|
|
221 sh->samplerate = 8000;
|
|
222 ra_priv->frame_size = 240;
|
|
223 sh->format = FOURCC_144;
|
|
224 }
|
|
225
|
|
226 /* Fill WAVEFORMATEX */
|
|
227 sh->wf = malloc(sizeof(WAVEFORMATEX));
|
|
228 memset(sh->wf, 0, sizeof(WAVEFORMATEX));
|
|
229 sh->wf->nChannels = sh->channels;
|
|
230 sh->wf->wBitsPerSample = sh->samplesize;
|
|
231 sh->wf->nSamplesPerSec = sh->samplerate;
|
|
232 sh->wf->nAvgBytesPerSec = sh->samplerate*sh->samplesize/8;
|
|
233 sh->wf->nBlockAlign = ra_priv->frame_size;
|
|
234 sh->wf->cbSize = 0;
|
|
235 sh->wf->wFormatTag = sh->format;
|
|
236
|
|
237 switch (sh->format) {
|
|
238 case FOURCC_144:
|
|
239 mp_msg(MSGT_DEMUX,MSGL_V,"Audio: 14_4\n");
|
|
240 sh->wf->cbSize = 10/*+codecdata_length*/;
|
|
241 sh->wf = realloc(sh->wf, sizeof(WAVEFORMATEX)+sh->wf->cbSize);
|
|
242 ((short*)(sh->wf+1))[0]=0;
|
|
243 ((short*)(sh->wf+1))[1]=240;
|
|
244 ((short*)(sh->wf+1))[2]=0;
|
|
245 ((short*)(sh->wf+1))[3]=0x14;
|
|
246 ((short*)(sh->wf+1))[4]=0;
|
|
247 break;
|
|
248 case FOURCC_288:
|
|
249 mp_msg(MSGT_DEMUX,MSGL_V,"Audio: 28_8\n");
|
|
250 sh->wf->cbSize = 8/*+codecdata_length*/;
|
|
251 sh->wf = realloc(sh->wf, sizeof(WAVEFORMATEX)+sh->wf->cbSize);
|
|
252 ((short*)(sh->wf+1))[0]=0;
|
|
253 ((short*)(sh->wf+1))[1]=ra_priv->sub_packet_h;
|
|
254 ((short*)(sh->wf+1))[2]=ra_priv->codec_flavor;
|
|
255 ((short*)(sh->wf+1))[3]=ra_priv->coded_framesize;
|
|
256 ((short*)(sh->wf+1))[4]=0;
|
|
257 break;
|
|
258 case FOURCC_DNET:
|
|
259 mp_msg(MSGT_DEMUX,MSGL_V,"Audio: DNET -> AC3\n");
|
|
260 sh->format = 0x2000;
|
|
261 break;
|
|
262 default:
|
|
263 mp_msg(MSGT_DEMUX,MSGL_V,"Audio: Unknown (%d)\n", sh->format);
|
|
264 }
|
|
265
|
|
266 print_wave_header(sh->wf);
|
|
267
|
|
268 /* disable seeking */
|
|
269 demuxer->seekable = 0;
|
|
270
|
|
271 if(!ds_fill_buffer(demuxer->audio))
|
|
272 mp_msg(MSGT_DEMUXER,MSGL_INFO,"[RealAudio] No data.\n");
|
|
273 }
|
|
274
|
|
275
|
|
276
|
|
277 void demux_close_ra(demuxer_t *demuxer)
|
|
278 {
|
|
279 ra_priv_t* ra_priv = demuxer->priv;
|
|
280
|
|
281 if (ra_priv)
|
|
282 free(ra_priv);
|
|
283
|
|
284 return;
|
|
285 }
|
|
286
|
|
287
|
|
288 #if 0
|
|
289 /* please upload RV10 samples WITH INDEX CHUNK */
|
|
290 int demux_seek_ra(demuxer_t *demuxer, float rel_seek_secs, int flags)
|
|
291 {
|
|
292 real_priv_t *priv = demuxer->priv;
|
|
293 demux_stream_t *d_audio = demuxer->audio;
|
|
294 sh_audio_t *sh_audio = d_audio->sh;
|
|
295 int aid = d_audio->id;
|
|
296 int next_offset = 0;
|
|
297 int rel_seek_frames = 0;
|
|
298 int streams = 0;
|
|
299
|
|
300 return stream_seek(demuxer->stream, next_offset);
|
|
301 }
|
|
302 #endif
|