Mercurial > mplayer.hg
annotate aviheader.c @ 599:33f68d0f5fd1
broken frame warning moved to verbose level 1
author | arpi_esp |
---|---|
date | Mon, 23 Apr 2001 21:32:00 +0000 |
parents | 8511095c5283 |
children | ef932c18cb14 |
rev | line source |
---|---|
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
1 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
2 |
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> |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
5 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
6 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
|
7 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
8 #include "stream.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
9 #include "demuxer.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
10 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
11 #include "wine/mmreg.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
12 #include "wine/avifmt.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
13 #include "wine/vfw.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
14 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
15 #include "codec-cfg.h" |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
16 #include "stheader.h" |
1 | 17 |
18 #define MIN(a,b) (((a)<(b))?(a):(b)) | |
19 | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
20 static MainAVIHeader avih; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
21 |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
22 void read_avi_header(demuxer_t *demuxer,int index_mode){ |
426 | 23 sh_audio_t *sh_audio=NULL; |
24 sh_video_t *sh_video=NULL; | |
1 | 25 int stream_id=-1; |
568 | 26 int idxfix_videostream=0; |
27 int idxfix_divx=0; | |
1 | 28 |
29 //---- AVI header: | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
30 demuxer->idx_size=0; |
1 | 31 while(1){ |
32 int id=stream_read_dword_le(demuxer->stream); | |
33 int chunksize,size2; | |
34 static int last_fccType=0; | |
35 // | |
36 if(stream_eof(demuxer->stream)) break; | |
37 // | |
38 if(id==mmioFOURCC('L','I','S','T')){ | |
39 int len=stream_read_dword_le(demuxer->stream)-4; // list size | |
40 id=stream_read_dword_le(demuxer->stream); // list type | |
41 if(verbose>=2) printf("LIST %.4s len=%d\n",&id,len); | |
42 if(id==listtypeAVIMOVIE){ | |
43 // found MOVI header | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
44 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
|
45 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
|
46 if(verbose>=1) printf("Found movie at 0x%X - 0x%X\n",demuxer->movi_start,demuxer->movi_end); |
1 | 47 len=(len+1)&(~1); |
48 stream_skip(demuxer->stream,len); | |
49 } | |
50 continue; | |
51 } | |
52 size2=stream_read_dword_le(demuxer->stream); | |
53 if(verbose>=2) printf("CHUNK %.4s len=%d\n",&id,size2); | |
54 chunksize=(size2+1)&(~1); | |
55 switch(id){ | |
56 case ckidAVIMAINHDR: // read 'avih' | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
57 stream_read(demuxer->stream,(char*) &avih,MIN(size2,sizeof(avih))); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
58 chunksize-=MIN(size2,sizeof(avih)); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
59 if(verbose) print_avih(&avih); |
1 | 60 break; |
61 case ckidSTREAMHEADER: { // read 'strh' | |
62 AVIStreamHeader h; | |
63 stream_read(demuxer->stream,(char*) &h,MIN(size2,sizeof(h))); | |
64 chunksize-=MIN(size2,sizeof(h)); | |
426 | 65 ++stream_id; |
66 if(h.fccType==streamtypeVIDEO){ | |
67 sh_video=new_sh_video(stream_id); | |
68 memcpy(&sh_video->video,&h,sizeof(h)); | |
69 } else | |
70 if(h.fccType==streamtypeAUDIO){ | |
71 sh_audio=new_sh_audio(stream_id); | |
72 memcpy(&sh_audio->audio,&h,sizeof(h)); | |
73 } | |
1 | 74 last_fccType=h.fccType; |
75 if(verbose>=1) print_strh(&h); | |
76 break; } | |
77 case ckidSTREAMFORMAT: { // read 'strf' | |
78 if(last_fccType==streamtypeVIDEO){ | |
433 | 79 sh_video->bih=calloc((chunksize<sizeof(BITMAPINFOHEADER))?sizeof(BITMAPINFOHEADER):chunksize,1); |
80 // sh_video->bih=malloc(chunksize); memset(sh_video->bih,0,chunksize); | |
81 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
|
82 stream_read(demuxer->stream,(char*) sh_video->bih,chunksize); |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
83 chunksize=0; |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
84 // 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
|
85 // sh_video->frametime=(float)sh_video->video.dwScale/(float)sh_video->video.dwRate; |
426 | 86 // if(demuxer->video->id==-1) demuxer->video->id=stream_id; |
568 | 87 // IdxFix: |
88 idxfix_videostream=stream_id; | |
89 switch(sh_video->bih->biCompression){ | |
90 case mmioFOURCC('D', 'I', 'V', '3'): | |
91 case mmioFOURCC('d', 'i', 'v', '3'): | |
92 case mmioFOURCC('D', 'I', 'V', '4'): | |
93 case mmioFOURCC('d', 'i', 'v', '4'): | |
94 case mmioFOURCC('D', 'I', 'V', '5'): | |
95 case mmioFOURCC('d', 'i', 'v', '5'): | |
96 case mmioFOURCC('D', 'I', 'V', '6'): | |
97 case mmioFOURCC('d', 'i', 'v', '6'): | |
98 case mmioFOURCC('M', 'P', '4', '3'): | |
99 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
|
100 case mmioFOURCC('M', 'P', '4', '2'): |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
101 case mmioFOURCC('m', 'p', '4', '2'): |
568 | 102 case mmioFOURCC('A', 'P', '4', '1'): |
103 idxfix_divx=1; // we can fix keyframes only for divx coded files! | |
104 } | |
1 | 105 } else |
106 if(last_fccType==streamtypeAUDIO){ | |
433 | 107 sh_audio->wf=calloc((chunksize<sizeof(WAVEFORMATEX))?sizeof(WAVEFORMATEX):chunksize,1); |
108 // sh_audio->wf=malloc(chunksize); memset(sh_audio->wf,0,chunksize); | |
109 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
|
110 stream_read(demuxer->stream,(char*) sh_audio->wf,chunksize); |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
111 chunksize=0; |
5251b0c57e39
sh_audio->wf and sh_video->bih changed to dynamic (thanx to Jens Hoffmann)
arpi_esp
parents:
426
diff
changeset
|
112 if(verbose>=1) print_wave_header(sh_audio->wf); |
426 | 113 // if(demuxer->audio->id==-1) demuxer->audio->id=stream_id; |
1 | 114 } |
115 break; | |
116 } | |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
117 case ckidAVINEWINDEX: if(index_mode){ |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
118 demuxer->idx_size=size2>>4; |
1 | 119 if(verbose>=1) printf("Reading INDEX block, %d chunks for %d frames\n", |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
120 demuxer->idx_size,avih.dwTotalFrames); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
121 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
|
122 stream_read(demuxer->stream,(char*)demuxer->idx,demuxer->idx_size<<4); |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
123 chunksize-=demuxer->idx_size<<4; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
124 if(verbose>=2) print_index(demuxer->idx,demuxer->idx_size); |
1 | 125 break; |
126 } | |
127 } | |
128 if(chunksize>0) stream_skip(demuxer->stream,chunksize); else | |
129 if(chunksize<0) printf("WARNING!!! chunksize=%d (id=%.4s)\n",chunksize,&id); | |
130 | |
131 } | |
132 | |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
133 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
|
134 // build index for file: |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
135 stream_reset(demuxer->stream); |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
136 stream_seek(demuxer->stream,demuxer->movi_start); |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
137 |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
138 demuxer->idx_pos=0; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
139 demuxer->idx=NULL; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
140 |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
141 while(1){ |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
142 int id,len,skip; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
143 AVIINDEXENTRY* idx; |
569 | 144 unsigned char c; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
145 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
|
146 if(demuxer->filepos>=demuxer->movi_end) break; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
147 id=stream_read_dword_le(demuxer->stream); |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
148 len=stream_read_dword_le(demuxer->stream); |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
149 if(id==mmioFOURCC('L','I','S','T')){ |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
150 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
|
151 continue; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
152 } |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
153 if(stream_eof(demuxer->stream)) break; |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
154 if(demuxer->idx_pos<=demuxer->idx_size){ |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
155 demuxer->idx_size+=32; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
156 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
|
157 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
|
158 } |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
159 idx=&((AVIINDEXENTRY *)demuxer->idx)[demuxer->idx_pos++]; |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
160 idx->ckid=id; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
161 idx->dwFlags=AVIIF_KEYFRAME; // FIXME |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
162 idx->dwChunkOffset=demuxer->filepos; |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
163 idx->dwChunkLength=len; |
569 | 164 |
165 c=stream_read_char(demuxer->stream); | |
568 | 166 |
167 // Fix keyframes for DivX files: | |
168 if(idxfix_divx) | |
169 if(avi_stream_id(id)==idxfix_videostream){ | |
569 | 170 if(c&0x40) idx->dwFlags=0; |
568 | 171 } |
172 | |
569 | 173 if(verbose>=2) printf("%08X %08X %.4s %02X %X\n",demuxer->filepos,id,&id,c,idx->dwFlags); |
568 | 174 #if 0 |
175 { unsigned char tmp[64]; | |
176 int i; | |
177 stream_read(demuxer->stream,tmp,64); | |
178 printf("%.4s",&id); | |
179 for(i=0;i<64;i++) printf(" %02X",tmp[i]); | |
180 printf("\n"); | |
181 } | |
182 #endif | |
564
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
183 skip=(len+1)&(~1); // total bytes in this chunk |
568 | 184 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
|
185 } |
587
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
186 demuxer->idx_size=demuxer->idx_pos; |
8511095c5283
stage#1 completed: c files no more included from mplayer.c
arpi_esp
parents:
569
diff
changeset
|
187 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
|
188 } |
747759a4a28f
seeking in raw/broken avi files (rebuilding index chunk)
arpi_esp
parents:
433
diff
changeset
|
189 |
1 | 190 } |
191 | |
192 #undef MIN | |
193 |