Mercurial > mplayer.hg
comparison libvo/vo_gif89a.c @ 6053:759c5a3314a2
-vo gif - by Joey Parrish <joey@yunamusic.com>
author | arpi |
---|---|
date | Sun, 12 May 2002 01:07:25 +0000 |
parents | |
children | 4a9c7041141d |
comparison
equal
deleted
inserted
replaced
6052:a5df6daa31fb | 6053:759c5a3314a2 |
---|---|
1 /* | |
2 * vo_gif89a.c Generate gif89a output in file out.gif | |
3 * | |
4 * Originally based on vo_png.c | |
5 * | |
6 * Stolen (C) 2002 by GifWhore <joey@yunamusic.com> | |
7 * | |
8 */ | |
9 | |
10 #include <stdio.h> | |
11 #include <stdlib.h> | |
12 #include <string.h> | |
13 #include <errno.h> | |
14 | |
15 #include "config.h" | |
16 #include "video_out.h" | |
17 #include "video_out_internal.h" | |
18 | |
19 #include "../postproc/rgb2rgb.h" | |
20 | |
21 #include <gif_lib.h> | |
22 | |
23 #define GIFWHORE_version 0.90 | |
24 | |
25 LIBVO_EXTERN (gif89a) | |
26 | |
27 static vo_info_t vo_info = | |
28 { | |
29 "GIF89a (out.gif)", | |
30 "gif89a", | |
31 "GifWhore <joey@yunamusic.com>", | |
32 "" | |
33 }; | |
34 | |
35 extern int verbose; | |
36 extern int vo_config_count; | |
37 | |
38 static int image_width; | |
39 static int image_height; | |
40 static int image_format; | |
41 static uint8_t *image_data=NULL; | |
42 static unsigned int scale_srcW = 0, scale_srcH = 0; | |
43 | |
44 static int reverse_map = 0; | |
45 static unsigned char framenum = 0; | |
46 static int gif_frameskip; | |
47 static int gif_framedelay; | |
48 static int target_fps = 0; | |
49 | |
50 GifFileType *newgif=NULL; | |
51 | |
52 /* | |
53 * TODO | |
54 * OSD!!! | |
55 */ | |
56 | |
57 static uint32_t config | |
58 (uint32_t width, uint32_t height, uint32_t d_width, uint32_t d_height, | |
59 uint32_t fullscreen, char *title, uint32_t format, const vo_tune_info_t *info) { | |
60 char filename[] = "out.gif"; | |
61 ColorMapObject *Cmap; | |
62 char LB[] = { | |
63 'N','E','T','S', | |
64 'C','A','P','E', | |
65 '2','.','0' }; | |
66 char LB2[] = { 1, 0x00, 0x00 }; | |
67 | |
68 if (target_fps == 0) target_fps = 5; | |
69 gif_frameskip = (vo_fps + 0.25) / target_fps; | |
70 gif_framedelay = 100 / target_fps; | |
71 | |
72 if ((width != d_width) || (height != d_height)) { | |
73 image_width = (d_width + 7) & ~7; | |
74 image_height = d_height; | |
75 scale_srcW = width; | |
76 scale_srcH = height; | |
77 SwScale_Init(); | |
78 } else { | |
79 image_width = width; | |
80 image_height = height; | |
81 } | |
82 image_format = format; | |
83 | |
84 Cmap = MakeMapObject(256, NULL); | |
85 | |
86 switch(format) { | |
87 case IMGFMT_BGR32: | |
88 case IMGFMT_BGR24: | |
89 reverse_map = 1; | |
90 break; | |
91 case IMGFMT_RGB32: | |
92 case IMGFMT_RGB24: | |
93 break; | |
94 case IMGFMT_IYUV: | |
95 case IMGFMT_I420: | |
96 case IMGFMT_YV12: | |
97 reverse_map = 1; | |
98 yuv2rgb_init(24, MODE_RGB); | |
99 image_data = malloc(image_width*image_height*3); | |
100 break; | |
101 default: | |
102 return 1; | |
103 } | |
104 | |
105 if (vo_config_count > 0) | |
106 return 0; | |
107 | |
108 EGifSetGifVersion("89a"); | |
109 newgif = EGifOpenFileName(filename, 0); | |
110 if (newgif == NULL) | |
111 { | |
112 fprintf(stderr, "error opening file for output.\n"); | |
113 return(1); | |
114 } | |
115 EGifPutScreenDesc(newgif, image_width, image_height, 256, 0, Cmap); | |
116 EGifPutExtensionFirst(newgif, 0xFF, 11, LB); | |
117 EGifPutExtensionLast(newgif, 0, 3, LB2); | |
118 | |
119 return 0; | |
120 } | |
121 | |
122 static const vo_info_t* get_info(void) | |
123 { | |
124 return &vo_info; | |
125 } | |
126 | |
127 static uint32_t draw_frame(uint8_t * src[]) | |
128 { | |
129 uint8_t *use_data; | |
130 ColorMapObject *Cmap; | |
131 uint8_t Colors[256 * 3]; | |
132 int z; | |
133 char CB[] = { (char)(gif_framedelay >> 8), (char)(gif_framedelay & 0xff), 0, 0}; | |
134 | |
135 if ((framenum++ % gif_frameskip)) return(0); | |
136 | |
137 if ((image_format == IMGFMT_BGR32) || (image_format == IMGFMT_RGB32)) | |
138 { | |
139 rgb32to24(src[0], image_data, image_width * image_height * 4); | |
140 src[0] = image_data; | |
141 } | |
142 | |
143 Cmap = MakeMapObject(256, NULL); | |
144 use_data = (uint8_t *)malloc(image_width * image_height); | |
145 if (gif_reduce(image_width, image_height, src[0], use_data, Colors)) return(0); | |
146 | |
147 if (reverse_map) | |
148 { | |
149 for (z = 0; z < 256; z++) { | |
150 Cmap->Colors[z].Blue = Colors[(z * 3) + 0]; | |
151 Cmap->Colors[z].Green = Colors[(z * 3) + 1]; | |
152 Cmap->Colors[z].Red = Colors[(z * 3) + 2]; | |
153 } | |
154 } | |
155 else | |
156 { | |
157 for (z = 0; z < 256; z++) { | |
158 Cmap->Colors[z].Red = Colors[(z * 3) + 0]; | |
159 Cmap->Colors[z].Green = Colors[(z * 3) + 1]; | |
160 Cmap->Colors[z].Blue = Colors[(z * 3) + 2]; | |
161 } | |
162 } | |
163 | |
164 EGifPutExtension(newgif, 0xF9, 0x04, CB); | |
165 EGifPutImageDesc(newgif, 0, 0, image_width, image_height, 0, Cmap); | |
166 EGifPutLine(newgif, use_data, image_width * image_height); | |
167 FreeMapObject(Cmap); | |
168 free(use_data); | |
169 | |
170 return (0); | |
171 } | |
172 | |
173 static void draw_osd(void) | |
174 { | |
175 } | |
176 | |
177 static void flip_page (void) | |
178 { | |
179 uint8_t *use_data; | |
180 ColorMapObject *Cmap; | |
181 uint8_t Colors[256 * 3]; | |
182 int z; | |
183 char CB[] = { (char)(gif_framedelay >> 8), (char)(gif_framedelay & 0xff), 0, 0}; | |
184 | |
185 if ((image_format == IMGFMT_YV12) || (image_format == IMGFMT_IYUV) || (image_format == IMGFMT_I420)) { | |
186 | |
187 if ((framenum++ % gif_frameskip)) return; | |
188 | |
189 Cmap = MakeMapObject(256, NULL); | |
190 use_data = (uint8_t *)malloc(image_width * image_height); | |
191 if (gif_reduce(image_width, image_height, image_data, use_data, Colors)) return; | |
192 | |
193 if (reverse_map) | |
194 { | |
195 for (z = 0; z < 256; z++) { | |
196 Cmap->Colors[z].Blue = Colors[(z * 3) + 0]; | |
197 Cmap->Colors[z].Green = Colors[(z * 3) + 1]; | |
198 Cmap->Colors[z].Red = Colors[(z * 3) + 2]; | |
199 } | |
200 } | |
201 else | |
202 { | |
203 for (z = 0; z < 256; z++) { | |
204 Cmap->Colors[z].Red = Colors[(z * 3) + 0]; | |
205 Cmap->Colors[z].Green = Colors[(z * 3) + 1]; | |
206 Cmap->Colors[z].Blue = Colors[(z * 3) + 2]; | |
207 } | |
208 } | |
209 | |
210 EGifPutExtension(newgif, 0xF9, 0x04, CB); | |
211 EGifPutImageDesc(newgif, 0, 0, image_width, image_height, 0, Cmap); | |
212 EGifPutLine(newgif, use_data, image_width * image_height); | |
213 FreeMapObject(Cmap); | |
214 free(use_data); | |
215 } | |
216 } | |
217 | |
218 static uint32_t draw_slice( uint8_t *src[],int stride[],int w,int h,int x,int y ) | |
219 { | |
220 /* hack: swap planes for I420 ;) -- alex */ | |
221 if ((image_format == IMGFMT_IYUV) || (image_format == IMGFMT_I420)) | |
222 { | |
223 uint8_t *src_i420[3]; | |
224 | |
225 src_i420[0] = src[0]; | |
226 src_i420[1] = src[2]; | |
227 src_i420[2] = src[1]; | |
228 src = src_i420; | |
229 } | |
230 | |
231 if (scale_srcW) { | |
232 uint8_t *dst[3] = {image_data, NULL, NULL}; | |
233 SwScale_YV12slice(src,stride,y,h, | |
234 dst, image_width*3, 24, | |
235 scale_srcW, scale_srcH, image_width, image_height); | |
236 } else { | |
237 uint8_t *dst = image_data + (image_width * y + x) * 3; | |
238 yuv2rgb(dst,src[0],src[1],src[2],w,h,image_width*3,stride[0],stride[1]); | |
239 } | |
240 return 0; | |
241 } | |
242 | |
243 static uint32_t | |
244 query_format(uint32_t format) | |
245 { | |
246 switch(format){ | |
247 case IMGFMT_IYUV: | |
248 case IMGFMT_I420: | |
249 case IMGFMT_YV12: | |
250 case IMGFMT_RGB|32: | |
251 case IMGFMT_BGR|32: | |
252 case IMGFMT_RGB|24: | |
253 case IMGFMT_BGR|24: | |
254 return 1 | VFCAP_SWSCALE | VFCAP_TIMER | VFCAP_ACCEPT_STRIDE; | |
255 } | |
256 return 0; | |
257 } | |
258 | |
259 static void | |
260 uninit(void) | |
261 { | |
262 char temp[256]; | |
263 | |
264 if (image_data) { free(image_data); image_data=NULL; } | |
265 | |
266 if (vo_config_count > 0) { | |
267 sprintf(temp, "gifwhore v%2.2f (c) %s\r\n", | |
268 GIFWHORE_version, "joey@yunamusic.com"); | |
269 EGifPutComment(newgif, temp); | |
270 EGifCloseFile(newgif); | |
271 } | |
272 } | |
273 | |
274 | |
275 static void check_events(void) | |
276 { | |
277 } | |
278 | |
279 int gif_reduce(int width, int height, | |
280 unsigned char *source, | |
281 unsigned char *destination, | |
282 unsigned char *palette) | |
283 { | |
284 GifColorType cmap[256]; | |
285 unsigned char Ra[width * height]; | |
286 unsigned char Ga[width * height]; | |
287 unsigned char Ba[width * height]; | |
288 unsigned char *R, *G, *B; | |
289 int Size = 256; | |
290 int i; | |
291 | |
292 R = Ra; G = Ga; B = Ba; | |
293 for (i = 0; i < width * height; i++) | |
294 { | |
295 *R++ = *source++; | |
296 *G++ = *source++; | |
297 *B++ = *source++; | |
298 } | |
299 | |
300 R = Ra; G = Ga; B = Ba; | |
301 if (QuantizeBuffer(width, height, &Size, | |
302 R, G, B, | |
303 destination, cmap) == GIF_ERROR) | |
304 { | |
305 fprintf(stderr, "vo_gif89a: Quantize failed!\n"); | |
306 return(-1); | |
307 } | |
308 | |
309 for (i = 0; i < Size; i++) | |
310 { | |
311 *palette++ = cmap[i].Red; | |
312 *palette++ = cmap[i].Green; | |
313 *palette++ = cmap[i].Blue; | |
314 } | |
315 | |
316 return(0); | |
317 } | |
318 | |
319 static uint32_t preinit(const char *arg) | |
320 { | |
321 int i = 0; | |
322 if (arg) i = atoi(arg); | |
323 if (i > vo_fps) i = vo_fps; | |
324 if (i < 1) i = 5; | |
325 target_fps = i; | |
326 return 0; | |
327 } | |
328 | |
329 static uint32_t control(uint32_t request, void *data, ...) | |
330 { | |
331 switch (request) { | |
332 case VOCTRL_QUERY_FORMAT: | |
333 return query_format(*((uint32_t*)data)); | |
334 } | |
335 return VO_NOTIMPL; | |
336 } | |
337 |