Mercurial > mplayer.hg
annotate aviheader.c @ 1447:e82fbd67ae60
better local display connection patch by Adam Tla/lka atlka@pg.gda.pl
author | arpi |
---|---|
date | Mon, 06 Aug 2001 00:22:44 +0000 |
parents | 1728d249c783 |
children | 8c57a5a3c645 |
rev | line source |
---|---|
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
1 |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
2 #include "config.h" |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
3 #include <stdio.h> |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
4 #include <stdlib.h> |
1430 | 5 #include <unistd.h> |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
6 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
7 extern int verbose; // defined in mplayer.c |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
8 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
9 #include "stream.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
10 #include "demuxer.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
11 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
12 #include "wine/mmreg.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
13 #include "wine/avifmt.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
14 #include "wine/vfw.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
15 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
16 #include "codec-cfg.h" |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
17 #include "bswap.h" |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
18 #include "stheader.h" |
1342 | 19 #include "aviheader.h" |
1 | 20 |
21 #define MIN(a,b) (((a)<(b))?(a):(b)) | |
22 | |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
23 |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
24 static MainAVIHeader avih; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
25 |
601 | 26 extern void print_avih(MainAVIHeader *h); |
27 extern void print_strh(AVIStreamHeader *h); | |
28 extern void print_wave_header(WAVEFORMATEX *h); | |
29 extern void print_index(AVIINDEXENTRY *idx,int idx_size); | |
30 | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
31 void read_avi_header(demuxer_t *demuxer,int index_mode){ |
426 | 32 sh_audio_t *sh_audio=NULL; |
33 sh_video_t *sh_video=NULL; | |
1 | 34 int stream_id=-1; |
568 | 35 int idxfix_videostream=0; |
36 int idxfix_divx=0; | |
1 | 37 |
38 //---- AVI header: | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
39 demuxer->idx_size=0; |
1 | 40 while(1){ |
41 int id=stream_read_dword_le(demuxer->stream); | |
42 int chunksize,size2; | |
43 static int last_fccType=0; | |
44 // | |
45 if(stream_eof(demuxer->stream)) break; | |
46 // | |
47 if(id==mmioFOURCC('L','I','S','T')){ | |
48 int len=stream_read_dword_le(demuxer->stream)-4; // list size | |
49 id=stream_read_dword_le(demuxer->stream); // list type | |
600 | 50 if(verbose>=2) printf("LIST %.4s len=%d\n",(char *) &id,len); |
1 | 51 if(id==listtypeAVIMOVIE){ |
52 // found MOVI header | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
53 demuxer->movi_start=stream_tell(demuxer->stream); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
54 demuxer->movi_end=demuxer->movi_start+len; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
55 if(verbose>=1) printf("Found movie at 0x%X - 0x%X\n",demuxer->movi_start,demuxer->movi_end); |
692 | 56 if(index_mode==-2) break; // reading from non-seekable source (stdin) |
1 | 57 len=(len+1)&(~1); |
58 stream_skip(demuxer->stream,len); | |
59 } | |
60 continue; | |
61 } | |
62 size2=stream_read_dword_le(demuxer->stream); | |
600 | 63 if(verbose>=2) printf("CHUNK %.4s len=%d\n",(char *) &id,size2); |
1 | 64 chunksize=(size2+1)&(~1); |
65 switch(id){ | |
66 case ckidAVIMAINHDR: // read 'avih' | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
67 stream_read(demuxer->stream,(char*) &avih,MIN(size2,sizeof(avih))); |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
68 le2me_MainAVIHeader(&avih); // swap to machine endian |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
69 chunksize-=MIN(size2,sizeof(avih)); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
70 if(verbose) print_avih(&avih); |
1 | 71 break; |
72 case ckidSTREAMHEADER: { // read 'strh' | |
73 AVIStreamHeader h; | |
74 stream_read(demuxer->stream,(char*) &h,MIN(size2,sizeof(h))); | |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
75 le2me_AVIStreamHeader(&h); // swap to machine endian |
1 | 76 chunksize-=MIN(size2,sizeof(h)); |
426 | 77 ++stream_id; |
78 if(h.fccType==streamtypeVIDEO){ | |
1289 | 79 sh_video=new_sh_video(demuxer,stream_id); |
426 | 80 memcpy(&sh_video->video,&h,sizeof(h)); |
81 } else | |
82 if(h.fccType==streamtypeAUDIO){ | |
1289 | 83 sh_audio=new_sh_audio(demuxer,stream_id); |
426 | 84 memcpy(&sh_audio->audio,&h,sizeof(h)); |
85 } | |
1 | 86 last_fccType=h.fccType; |
87 if(verbose>=1) print_strh(&h); | |
88 break; } | |
89 case ckidSTREAMFORMAT: { // read 'strf' | |
90 if(last_fccType==streamtypeVIDEO){ | |
433 | 91 sh_video->bih=calloc((chunksize<sizeof(BITMAPINFOHEADER))?sizeof(BITMAPINFOHEADER):chunksize,1); |
92 // sh_video->bih=malloc(chunksize); memset(sh_video->bih,0,chunksize); | |
93 if(verbose>=1) printf("found 'bih', %d bytes of %d\n",chunksize,sizeof(BITMAPINFOHEADER)); | |
432
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
94 stream_read(demuxer->stream,(char*) sh_video->bih,chunksize); |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
95 le2me_BITMAPINFOHEADER(sh_video->bih); // swap to machine endian |
432
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
96 chunksize=0; |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
97 // sh_video->fps=(float)sh_video->video.dwRate/(float)sh_video->video.dwScale; |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
98 // sh_video->frametime=(float)sh_video->video.dwScale/(float)sh_video->video.dwRate; |
426 | 99 // if(demuxer->video->id==-1) demuxer->video->id=stream_id; |
568 | 100 // IdxFix: |
101 idxfix_videostream=stream_id; | |
102 switch(sh_video->bih->biCompression){ | |
103 case mmioFOURCC('D', 'I', 'V', '3'): | |
104 case mmioFOURCC('d', 'i', 'v', '3'): | |
105 case mmioFOURCC('D', 'I', 'V', '4'): | |
106 case mmioFOURCC('d', 'i', 'v', '4'): | |
107 case mmioFOURCC('D', 'I', 'V', '5'): | |
108 case mmioFOURCC('d', 'i', 'v', '5'): | |
109 case mmioFOURCC('D', 'I', 'V', '6'): | |
110 case mmioFOURCC('d', 'i', 'v', '6'): | |
111 case mmioFOURCC('M', 'P', '4', '3'): | |
112 case mmioFOURCC('m', 'p', '4', '3'): | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
113 case mmioFOURCC('M', 'P', '4', '2'): |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
114 case mmioFOURCC('m', 'p', '4', '2'): |
773 | 115 case mmioFOURCC('D', 'I', 'V', '2'): |
568 | 116 case mmioFOURCC('A', 'P', '4', '1'): |
117 idxfix_divx=1; // we can fix keyframes only for divx coded files! | |
118 } | |
1 | 119 } else |
120 if(last_fccType==streamtypeAUDIO){ | |
1038
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
121 int wf_size = chunksize<sizeof(WAVEFORMATEX)?sizeof(WAVEFORMATEX):chunksize; |
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
122 sh_audio->wf=calloc(wf_size,1); |
433 | 123 // sh_audio->wf=malloc(chunksize); memset(sh_audio->wf,0,chunksize); |
124 if(verbose>=1) printf("found 'wf', %d bytes of %d\n",chunksize,sizeof(WAVEFORMATEX)); | |
432
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
125 stream_read(demuxer->stream,(char*) sh_audio->wf,chunksize); |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
126 le2me_WAVEFORMATEX(sh_audio->wf); |
1038
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
127 if (sh_audio->wf->cbSize != 0 && |
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
128 wf_size < sizeof(WAVEFORMATEX)+sh_audio->wf->cbSize) { |
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
129 sh_audio->wf=realloc(sh_audio->wf, sizeof(WAVEFORMATEX)+sh_audio->wf->cbSize); |
b36fb1ae4b53
applied solaris8/netbsd/other fixes patch by J¸«ärgen Keil <jk@tools.de>
arpi_esp
parents:
773
diff
changeset
|
130 } |
432
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
131 chunksize=0; |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
132 if(verbose>=1) print_wave_header(sh_audio->wf); |
426 | 133 // if(demuxer->audio->id==-1) demuxer->audio->id=stream_id; |
1 | 134 } |
135 break; | |
136 } | |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
137 case ckidAVINEWINDEX: if(index_mode){ |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
138 int i; |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
139 demuxer->idx_size=size2>>4; |
600 | 140 if(verbose>=1) printf("Reading INDEX block, %d chunks for %ld frames\n", |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
141 demuxer->idx_size,avih.dwTotalFrames); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
142 demuxer->idx=malloc(demuxer->idx_size<<4); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
143 stream_read(demuxer->stream,(char*)demuxer->idx,demuxer->idx_size<<4); |
1309
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
144 for (i = 0; i < demuxer->idx_size; i++) // swap index to machine endian |
598e3047ce13
Add some preliminary support for non-x86 architectures to mplayer
jkeil
parents:
1289
diff
changeset
|
145 le2me_AVIINDEXENTRY((AVIINDEXENTRY*)demuxer->idx + i); |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
146 chunksize-=demuxer->idx_size<<4; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
147 if(verbose>=2) print_index(demuxer->idx,demuxer->idx_size); |
1 | 148 break; |
149 } | |
150 } | |
151 if(chunksize>0) stream_skip(demuxer->stream,chunksize); else | |
600 | 152 if(chunksize<0) printf("WARNING!!! chunksize=%d (id=%.4s)\n",chunksize,(char *) &id); |
1 | 153 |
154 } | |
155 | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
156 if(index_mode>=2 || (demuxer->idx_size==0 && index_mode==1)){ |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
157 // build index for file: |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
158 stream_reset(demuxer->stream); |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
159 stream_seek(demuxer->stream,demuxer->movi_start); |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
160 |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
161 demuxer->idx_pos=0; |
1392 | 162 demuxer->idx_size=0; |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
163 demuxer->idx=NULL; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
164 |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
165 while(1){ |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
166 int id,len,skip; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
167 AVIINDEXENTRY* idx; |
569 | 168 unsigned char c; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
169 demuxer->filepos=stream_tell(demuxer->stream); |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
170 if(demuxer->filepos>=demuxer->movi_end) break; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
171 id=stream_read_dword_le(demuxer->stream); |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
172 len=stream_read_dword_le(demuxer->stream); |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
173 if(id==mmioFOURCC('L','I','S','T')){ |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
174 id=stream_read_dword_le(demuxer->stream); // list type |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
175 continue; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
176 } |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
177 if(stream_eof(demuxer->stream)) break; |
1392 | 178 if(!id || avi_stream_id(id)==100) goto skip_chunk; // bad ID (or padding?) |
179 | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
180 if(demuxer->idx_pos<=demuxer->idx_size){ |
1392 | 181 // demuxer->idx_size+=32; |
182 demuxer->idx_size+=1024; // +16kB | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
183 demuxer->idx=realloc(demuxer->idx,demuxer->idx_size*sizeof(AVIINDEXENTRY)); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
184 if(!demuxer->idx){demuxer->idx_pos=0; break;} // error! |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
185 } |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
186 idx=&((AVIINDEXENTRY *)demuxer->idx)[demuxer->idx_pos++]; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
187 idx->ckid=id; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
188 idx->dwFlags=AVIIF_KEYFRAME; // FIXME |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
189 idx->dwChunkOffset=demuxer->filepos; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
190 idx->dwChunkLength=len; |
569 | 191 |
192 c=stream_read_char(demuxer->stream); | |
568 | 193 |
194 // Fix keyframes for DivX files: | |
195 if(idxfix_divx) | |
196 if(avi_stream_id(id)==idxfix_videostream){ | |
569 | 197 if(c&0x40) idx->dwFlags=0; |
568 | 198 } |
199 | |
600 | 200 if(verbose>=2) printf("%08X %08X %.4s %02X %X\n",demuxer->filepos,id,(char *) &id,c,(unsigned int) idx->dwFlags); |
568 | 201 #if 0 |
202 { unsigned char tmp[64]; | |
203 int i; | |
204 stream_read(demuxer->stream,tmp,64); | |
205 printf("%.4s",&id); | |
206 for(i=0;i<64;i++) printf(" %02X",tmp[i]); | |
207 printf("\n"); | |
208 } | |
209 #endif | |
1392 | 210 skip_chunk: |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
211 skip=(len+1)&(~1); // total bytes in this chunk |
568 | 212 stream_seek(demuxer->stream,8+demuxer->filepos+skip); |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
213 } |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
214 demuxer->idx_size=demuxer->idx_pos; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
215 printf("AVI: Generated index table for %d chunks!\n",demuxer->idx_size); |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
216 } |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
217 |
1 | 218 } |
219 | |
220 #undef MIN | |
221 |