Mercurial > mplayer.hg
annotate libmpcodecs/vd_libmpeg2.c @ 26009:40fd58b7e8ab
sync w/r25308, patch by mesecam %mesecam A gmail P com %
author | gpoirier |
---|---|
date | Wed, 20 Feb 2008 22:40:14 +0000 |
parents | e3be2de66969 |
children | 2506f1b0bdbe |
rev | line source |
---|---|
4998 | 1 #include <stdio.h> |
2 #include <stdlib.h> | |
3 | |
4 #include "config.h" | |
8026
b465ba5897a3
usage of libmpeg2, liba52, mp3lib & svq1 can be disabled
arpi
parents:
7957
diff
changeset
|
5 |
4998 | 6 #include "mp_msg.h" |
7 | |
8 #include "vd_internal.h" | |
9 | |
9859 | 10 //#undef MPEG12_POSTPROC |
11 | |
4998 | 12 static vd_info_t info = |
13 { | |
12932 | 14 "MPEG 1/2 Video decoder libmpeg2-v0.4.0b", |
4998 | 15 "libmpeg2", |
9859 | 16 "A'rpi & Fabian Franz", |
4998 | 17 "Aaron & Walken", |
18 "native" | |
19 }; | |
20 | |
21 LIBVD_EXTERN(libmpeg2) | |
22 | |
9859 | 23 //#include "libvo/video_out.h" // FIXME!!! |
5465 | 24 |
4998 | 25 #include "libmpeg2/mpeg2.h" |
12932 | 26 #include "libmpeg2/attributes.h" |
4998 | 27 #include "libmpeg2/mpeg2_internal.h" |
28 | |
17012 | 29 #include "cpudetect.h" |
5465 | 30 |
18301 | 31 typedef struct { |
32 mpeg2dec_t *mpeg2dec; | |
33 int quant_store_idx; | |
34 char *quant_store[3]; | |
25977
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
35 int imgfmt; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
36 int width; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
37 int height; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
38 double aspect; |
18301 | 39 } vd_libmpeg2_ctx_t; |
40 | |
4998 | 41 // to set/get/query special features/parameters |
42 static int control(sh_video_t *sh,int cmd,void* arg,...){ | |
18301 | 43 vd_libmpeg2_ctx_t *context = sh->context; |
44 mpeg2dec_t * mpeg2dec = context->mpeg2dec; | |
14012 | 45 const mpeg2_info_t * info = mpeg2_info (mpeg2dec); |
13995 | 46 |
47 switch(cmd) { | |
48 case VDCTRL_QUERY_FORMAT: | |
14012 | 49 if (info->sequence->width >> 1 == info->sequence->chroma_width && |
50 info->sequence->height >> 1 == info->sequence->chroma_height && | |
51 (*((int*)arg)) == IMGFMT_YV12) | |
13995 | 52 return CONTROL_TRUE; |
14012 | 53 if (info->sequence->width >> 1 == info->sequence->chroma_width && |
54 info->sequence->height == info->sequence->chroma_height && | |
55 (*((int*)arg)) == IMGFMT_422P) | |
13995 | 56 return CONTROL_TRUE; |
57 return CONTROL_FALSE; | |
58 } | |
59 | |
4998 | 60 return CONTROL_UNKNOWN; |
61 } | |
62 | |
63 // init driver | |
64 static int init(sh_video_t *sh){ | |
18301 | 65 vd_libmpeg2_ctx_t *context; |
9859 | 66 mpeg2dec_t * mpeg2dec; |
13995 | 67 // const mpeg2_info_t * info; |
9859 | 68 int accel; |
5465 | 69 |
9859 | 70 accel = 0; |
71 if(gCpuCaps.hasMMX) | |
72 accel |= MPEG2_ACCEL_X86_MMX; | |
73 if(gCpuCaps.hasMMX2) | |
74 accel |= MPEG2_ACCEL_X86_MMXEXT; | |
75 if(gCpuCaps.has3DNow) | |
76 accel |= MPEG2_ACCEL_X86_3DNOW; | |
10267
d953763cc555
libmpeg2-altivec patch by Magnus Damm <damm@opensource.se>:
arpi
parents:
10250
diff
changeset
|
77 if(gCpuCaps.hasAltiVec) |
d953763cc555
libmpeg2-altivec patch by Magnus Damm <damm@opensource.se>:
arpi
parents:
10250
diff
changeset
|
78 accel |= MPEG2_ACCEL_PPC_ALTIVEC; |
13117 | 79 #ifdef HAVE_VIS |
80 accel |= MPEG2_ACCEL_SPARC_VIS; | |
9859 | 81 #endif |
82 mpeg2_accel(accel); | |
5465 | 83 |
9859 | 84 mpeg2dec = mpeg2_init (); |
85 | |
86 if(!mpeg2dec) return 0; | |
87 | |
88 mpeg2_custom_fbuf(mpeg2dec,1); // enable DR1 | |
18301 | 89 |
90 context = calloc(1, sizeof(vd_libmpeg2_ctx_t)); | |
91 context->mpeg2dec = mpeg2dec; | |
92 sh->context = context; | |
9859 | 93 |
13112 | 94 mpeg2dec->pending_buffer = 0; |
95 mpeg2dec->pending_length = 0; | |
96 | |
9859 | 97 return 1; |
4998 | 98 } |
99 | |
100 // uninit driver | |
101 static void uninit(sh_video_t *sh){ | |
18301 | 102 int i; |
103 vd_libmpeg2_ctx_t *context = sh->context; | |
104 mpeg2dec_t * mpeg2dec = context->mpeg2dec; | |
13112 | 105 if (mpeg2dec->pending_buffer) free(mpeg2dec->pending_buffer); |
14012 | 106 mpeg2dec->decoder.convert=NULL; |
107 mpeg2dec->decoder.convert_id=NULL; | |
9859 | 108 mpeg2_close (mpeg2dec); |
18301 | 109 for (i=0; i < 3; i++) |
110 free(context->quant_store[i]); | |
111 free(sh->context); | |
4998 | 112 } |
113 | |
13112 | 114 static void draw_slice (void * _sh, uint8_t * const * src, unsigned int y){ |
9859 | 115 sh_video_t* sh = (sh_video_t*) _sh; |
18301 | 116 vd_libmpeg2_ctx_t *context = sh->context; |
117 mpeg2dec_t* mpeg2dec = context->mpeg2dec; | |
9859 | 118 const mpeg2_info_t * info = mpeg2_info (mpeg2dec); |
119 int stride[3]; | |
5465 | 120 |
13112 | 121 // printf("draw_slice() y=%d \n",y); |
5465 | 122 |
9859 | 123 stride[0]=mpeg2dec->decoder.stride; |
124 stride[1]=stride[2]=mpeg2dec->decoder.uv_stride; | |
5465 | 125 |
9859 | 126 mpcodecs_draw_slice(sh, (uint8_t **)src, |
12572
7d681d8ebab8
display height may be a lot smaller or larger than picture height, sample provided by winnicki
iive
parents:
11080
diff
changeset
|
127 stride, info->sequence->picture_width, |
7d681d8ebab8
display height may be a lot smaller or larger than picture height, sample provided by winnicki
iive
parents:
11080
diff
changeset
|
128 (y+16<=info->sequence->picture_height) ? 16 : |
7d681d8ebab8
display height may be a lot smaller or larger than picture height, sample provided by winnicki
iive
parents:
11080
diff
changeset
|
129 info->sequence->picture_height-y, |
9859 | 130 0, y); |
4998 | 131 } |
132 | |
5465 | 133 // decode a frame |
134 static mp_image_t* decode(sh_video_t *sh,void* data,int len,int flags){ | |
18301 | 135 vd_libmpeg2_ctx_t *context = sh->context; |
136 mpeg2dec_t * mpeg2dec = context->mpeg2dec; | |
9859 | 137 const mpeg2_info_t * info = mpeg2_info (mpeg2dec); |
138 int drop_frame, framedrop=flags&3; | |
139 | |
13995 | 140 // MPlayer registers its own draw_slice callback, prevent libmpeg2 from freeing the context |
14012 | 141 mpeg2dec->decoder.convert=NULL; |
142 mpeg2dec->decoder.convert_id=NULL; | |
13150 | 143 |
11080
26f1b3ad4a77
skip null frames in mpeg files, patch by Zoltan Hidvegi <mplayer@hzoli.2y.net>
attila
parents:
10663
diff
changeset
|
144 if(len<=0) return NULL; // skipped null frame |
26f1b3ad4a77
skip null frames in mpeg files, patch by Zoltan Hidvegi <mplayer@hzoli.2y.net>
attila
parents:
10663
diff
changeset
|
145 |
9859 | 146 // append extra 'end of frame' code: |
147 ((char*)data+len)[0]=0; | |
148 ((char*)data+len)[1]=0; | |
149 ((char*)data+len)[2]=1; | |
150 ((char*)data+len)[3]=0xff; | |
151 len+=4; | |
5465 | 152 |
13112 | 153 if (mpeg2dec->pending_length) { |
154 mpeg2_buffer (mpeg2dec, mpeg2dec->pending_buffer, mpeg2dec->pending_buffer + mpeg2dec->pending_length); | |
155 } else { | |
23916 | 156 mpeg2_buffer (mpeg2dec, data, (uint8_t *)data+len); |
13112 | 157 } |
9859 | 158 |
159 while(1){ | |
160 int state=mpeg2_parse (mpeg2dec); | |
13995 | 161 int type, use_callback; |
162 mp_image_t* mpi_new; | |
18428
22805699b7b1
moved code to set aspect ratio for mpeg12 away from video.c and into decoder files; A/R changes work correctly with -vc mpeg12
nicodvb
parents:
18301
diff
changeset
|
163 unsigned long pw, ph; |
25977
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
164 int imgfmt; |
13995 | 165 |
9859 | 166 switch(state){ |
12932 | 167 case STATE_BUFFER: |
13112 | 168 if (mpeg2dec->pending_length) { |
13152 | 169 // just finished the pending data, continue with processing of the passed buffer |
13112 | 170 mpeg2dec->pending_length = 0; |
23916 | 171 mpeg2_buffer (mpeg2dec, data, (uint8_t *)data+len); |
13112 | 172 } else { |
173 // parsing of the passed buffer finished, return. | |
174 return 0; | |
175 } | |
176 break; | |
9859 | 177 case STATE_SEQUENCE: |
18428
22805699b7b1
moved code to set aspect ratio for mpeg12 away from video.c and into decoder files; A/R changes work correctly with -vc mpeg12
nicodvb
parents:
18301
diff
changeset
|
178 pw = info->sequence->display_width * info->sequence->pixel_width; |
22805699b7b1
moved code to set aspect ratio for mpeg12 away from video.c and into decoder files; A/R changes work correctly with -vc mpeg12
nicodvb
parents:
18301
diff
changeset
|
179 ph = info->sequence->display_height * info->sequence->pixel_height; |
22805699b7b1
moved code to set aspect ratio for mpeg12 away from video.c and into decoder files; A/R changes work correctly with -vc mpeg12
nicodvb
parents:
18301
diff
changeset
|
180 if(ph) sh->aspect = (float) pw / (float) ph; |
25962 | 181 // video parameters initialized/changed, (re)init libvo: |
13995 | 182 if (info->sequence->width >> 1 == info->sequence->chroma_width && |
183 info->sequence->height >> 1 == info->sequence->chroma_height) { | |
25977
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
184 imgfmt = IMGFMT_YV12; |
13995 | 185 } else if (info->sequence->width >> 1 == info->sequence->chroma_width && |
186 info->sequence->height == info->sequence->chroma_height) { | |
25977
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
187 imgfmt = IMGFMT_422P; |
13995 | 188 } else return 0; |
25977
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
189 if (imgfmt == context->imgfmt && |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
190 info->sequence->picture_width == context->width && |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
191 info->sequence->picture_height == context->height && |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
192 sh->aspect == context->aspect) |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
193 break; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
194 if(!mpcodecs_config_vo(sh, |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
195 info->sequence->picture_width, |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
196 info->sequence->picture_height, imgfmt)) |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
197 return 0; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
198 context->imgfmt = imgfmt; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
199 context->width = info->sequence->picture_width; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
200 context->height = info->sequence->picture_height; |
e3be2de66969
Avoid reinit of vo with the exactly same parameters over and over.
reimar
parents:
25962
diff
changeset
|
201 context->aspect = sh->aspect; |
9859 | 202 break; |
13995 | 203 case STATE_PICTURE: |
204 type=info->current_picture->flags&PIC_MASK_CODING_TYPE; | |
9859 | 205 |
206 drop_frame = framedrop && (mpeg2dec->decoder.coding_type == B_TYPE); | |
207 drop_frame |= framedrop>=2; // hard drop | |
208 if (drop_frame) { | |
209 mpeg2_skip(mpeg2dec, 1); | |
210 //printf("Dropping Frame ...\n"); | |
211 break; | |
212 } | |
213 mpeg2_skip(mpeg2dec, 0); //mpeg2skip skips frames until set again to 0 | |
214 | |
14012 | 215 use_callback = (!framedrop && vd_use_slices && |
216 (info->current_picture->flags&PIC_FLAG_PROGRESSIVE_FRAME)) ? | |
217 MP_IMGFLAG_DRAW_CALLBACK:0; | |
13995 | 218 |
9859 | 219 // get_buffer "callback": |
13995 | 220 mpi_new=mpcodecs_get_image(sh,MP_IMGTYPE_IPB, |
14012 | 221 (type==PIC_FLAG_CODING_TYPE_B) ? |
222 use_callback : (MP_IMGFLAG_PRESERVE|MP_IMGFLAG_READABLE), | |
14075 | 223 info->sequence->width, |
224 info->sequence->height); | |
13995 | 225 |
13112 | 226 if(!mpi_new) return 0; // VO ERROR!!!!!!!! |
227 mpeg2_set_buf(mpeg2dec, mpi_new->planes, mpi_new); | |
20591
ca1d5d9fe51c
Set mpi stride, mostly fixes http://samples.mplayerhq.hu/MPEG2/res_change_ffmpeg_aspect.ts
reimar
parents:
18771
diff
changeset
|
228 mpi_new->stride[0] = info->sequence->width; |
ca1d5d9fe51c
Set mpi stride, mostly fixes http://samples.mplayerhq.hu/MPEG2/res_change_ffmpeg_aspect.ts
reimar
parents:
18771
diff
changeset
|
229 mpi_new->stride[1] = info->sequence->chroma_width; |
ca1d5d9fe51c
Set mpi stride, mostly fixes http://samples.mplayerhq.hu/MPEG2/res_change_ffmpeg_aspect.ts
reimar
parents:
18771
diff
changeset
|
230 mpi_new->stride[2] = info->sequence->chroma_width; |
10510
73b3e4336cd4
Add mpeg2_flags to mp_image_t, copy flags in vd_libmpeg2.c,
ranma
parents:
10267
diff
changeset
|
231 if (info->current_picture->flags&PIC_FLAG_TOP_FIELD_FIRST) |
13112 | 232 mpi_new->fields |= MP_IMGFIELD_TOP_FIRST; |
233 else mpi_new->fields &= ~MP_IMGFIELD_TOP_FIRST; | |
10510
73b3e4336cd4
Add mpeg2_flags to mp_image_t, copy flags in vd_libmpeg2.c,
ranma
parents:
10267
diff
changeset
|
234 if (info->current_picture->flags&PIC_FLAG_REPEAT_FIRST_FIELD) |
13112 | 235 mpi_new->fields |= MP_IMGFIELD_REPEAT_FIRST; |
236 else mpi_new->fields &= ~MP_IMGFIELD_REPEAT_FIRST; | |
237 mpi_new->fields |= MP_IMGFIELD_ORDERED; | |
20639 | 238 if (!(info->current_picture->flags&PIC_FLAG_PROGRESSIVE_FRAME)) |
239 mpi_new->fields |= MP_IMGFIELD_INTERLACED; | |
9859 | 240 |
12935 | 241 #ifdef MPEG12_POSTPROC |
18301 | 242 mpi_new->qstride=info->sequence->width>>4; |
243 { | |
244 char **p = &context->quant_store[type==PIC_FLAG_CODING_TYPE_B ? | |
245 2 : (context->quant_store_idx ^= 1)]; | |
246 *p = realloc(*p, mpi_new->qstride*(info->sequence->height>>4)); | |
247 mpi_new->qscale = *p; | |
12935 | 248 } |
13112 | 249 mpeg2dec->decoder.quant_store=mpi_new->qscale; |
250 mpeg2dec->decoder.quant_stride=mpi_new->qstride; | |
251 mpi_new->pict_type=type; // 1->I, 2->P, 3->B | |
252 mpi_new->qscale_type= 1; | |
12935 | 253 #endif |
254 | |
14012 | 255 if (mpi_new->flags&MP_IMGFLAG_DRAW_CALLBACK |
256 && !(mpi_new->flags&MP_IMGFLAG_DIRECT)) { | |
257 // nice, filter/vo likes draw_callback :) | |
258 mpeg2dec->decoder.convert=draw_slice; | |
259 mpeg2dec->decoder.convert_id=sh; | |
260 } else { | |
261 mpeg2dec->decoder.convert=NULL; | |
262 mpeg2dec->decoder.convert_id=NULL; | |
13995 | 263 } |
264 | |
9859 | 265 break; |
266 case STATE_SLICE: | |
267 case STATE_END: | |
12932 | 268 case STATE_INVALID_END: |
9859 | 269 // decoding done: |
13112 | 270 if(info->display_fbuf) { |
13152 | 271 mp_image_t* mpi = info->display_fbuf->id; |
272 if (mpeg2dec->pending_length == 0) { | |
13995 | 273 mpeg2dec->pending_length = mpeg2dec->buf_end - mpeg2dec->buf_start; |
274 mpeg2dec->pending_buffer = realloc(mpeg2dec->pending_buffer, mpeg2dec->pending_length); | |
275 memcpy(mpeg2dec->pending_buffer, mpeg2dec->buf_start, mpeg2dec->pending_length); | |
13152 | 276 } else { |
277 // still some data in the pending buffer, shouldn't happen | |
278 mpeg2dec->pending_length = mpeg2dec->buf_end - mpeg2dec->buf_start; | |
279 memmove(mpeg2dec->pending_buffer, mpeg2dec->buf_start, mpeg2dec->pending_length); | |
280 mpeg2dec->pending_buffer = realloc(mpeg2dec->pending_buffer, mpeg2dec->pending_length + len); | |
281 memcpy(mpeg2dec->pending_buffer+mpeg2dec->pending_length, data, len); | |
282 mpeg2dec->pending_length += len; | |
283 } | |
284 // fprintf(stderr, "pending = %d\n", mpeg2dec->pending_length); | |
13112 | 285 return mpi; |
286 } | |
7957
31fd09cc9ba2
passing picture_type (might be usefull for postprocessing)
michael
parents:
7756
diff
changeset
|
287 } |
31fd09cc9ba2
passing picture_type (might be usefull for postprocessing)
michael
parents:
7756
diff
changeset
|
288 } |
5465 | 289 } |