diff options
Diffstat (limited to 'libraries/evas/src/modules/loaders/gif/evas_image_load_gif.c')
-rw-r--r-- | libraries/evas/src/modules/loaders/gif/evas_image_load_gif.c | 1038 |
1 files changed, 0 insertions, 1038 deletions
diff --git a/libraries/evas/src/modules/loaders/gif/evas_image_load_gif.c b/libraries/evas/src/modules/loaders/gif/evas_image_load_gif.c deleted file mode 100644 index dbb3584..0000000 --- a/libraries/evas/src/modules/loaders/gif/evas_image_load_gif.c +++ /dev/null | |||
@@ -1,1038 +0,0 @@ | |||
1 | #include "evas_common.h" | ||
2 | #include "evas_private.h" | ||
3 | |||
4 | #include <sys/types.h> | ||
5 | #include <sys/stat.h> | ||
6 | #include <fcntl.h> | ||
7 | #include <unistd.h> | ||
8 | |||
9 | #include <gif_lib.h> | ||
10 | |||
11 | typedef struct _Gif_Frame Gif_Frame; | ||
12 | |||
13 | typedef enum _Frame_Load_Type | ||
14 | { | ||
15 | LOAD_FRAME_NONE = 0, | ||
16 | LOAD_FRAME_INFO = 1, | ||
17 | LOAD_FRAME_DATA = 2, | ||
18 | LOAD_FRAME_DATA_INFO = 3 | ||
19 | } Frame_Load_Type; | ||
20 | |||
21 | struct _Gif_Frame | ||
22 | { | ||
23 | struct { | ||
24 | /* Image descriptor */ | ||
25 | int x; | ||
26 | int y; | ||
27 | int w; | ||
28 | int h; | ||
29 | int interlace; | ||
30 | } image_des; | ||
31 | |||
32 | struct { | ||
33 | /* Graphic Control*/ | ||
34 | int disposal; | ||
35 | int transparent; | ||
36 | int delay; | ||
37 | int input; | ||
38 | } frame_info; | ||
39 | }; | ||
40 | |||
41 | static Eina_Bool evas_image_load_file_data_gif_internal(Image_Entry *ie, Image_Entry_Frame *frame, int *error); | ||
42 | |||
43 | static Eina_Bool evas_image_load_file_head_gif(Image_Entry *ie, const char *file, const char *key, int *error) EINA_ARG_NONNULL(1, 2, 4); | ||
44 | static Eina_Bool evas_image_load_file_data_gif(Image_Entry *ie, const char *file, const char *key, int *error) EINA_ARG_NONNULL(1, 2, 4); | ||
45 | static double evas_image_load_frame_duration_gif(Image_Entry *ie, const char *file, int start_frame, int frame_num) ; | ||
46 | static Eina_Bool evas_image_load_specific_frame(Image_Entry *ie, const char *file, int frame_index, int *error); | ||
47 | |||
48 | static Evas_Image_Load_Func evas_image_load_gif_func = | ||
49 | { | ||
50 | EINA_TRUE, | ||
51 | evas_image_load_file_head_gif, | ||
52 | evas_image_load_file_data_gif, | ||
53 | evas_image_load_frame_duration_gif, | ||
54 | EINA_FALSE | ||
55 | }; | ||
56 | #define byte2_to_int(a,b) (((b)<<8)|(a)) | ||
57 | |||
58 | #define FRAME_MAX 1024 | ||
59 | |||
60 | /* find specific frame in image entry */ | ||
61 | static Eina_Bool | ||
62 | _find_frame(Image_Entry *ie, int frame_index, Image_Entry_Frame **frame) | ||
63 | { | ||
64 | Eina_List *l; | ||
65 | Image_Entry_Frame *hit_frame = NULL; | ||
66 | |||
67 | if (!ie) return EINA_FALSE; | ||
68 | if (!ie->frames) return EINA_FALSE; | ||
69 | |||
70 | EINA_LIST_FOREACH(ie->frames, l, hit_frame) | ||
71 | { | ||
72 | if (hit_frame->index == frame_index) | ||
73 | { | ||
74 | *frame = hit_frame; | ||
75 | return EINA_TRUE; | ||
76 | } | ||
77 | } | ||
78 | return EINA_FALSE; | ||
79 | } | ||
80 | |||
81 | static Eina_Bool | ||
82 | _find_close_frame(Image_Entry *ie, int frame_index, Image_Entry_Frame **frame) | ||
83 | { | ||
84 | int i; | ||
85 | Eina_Bool hit = EINA_FALSE; | ||
86 | i = frame_index -1; | ||
87 | |||
88 | if (!ie) return EINA_FALSE; | ||
89 | if (!ie->frames) return EINA_FALSE; | ||
90 | |||
91 | for (; i > 0; i--) | ||
92 | { | ||
93 | hit = _find_frame(ie, i, frame); | ||
94 | if (hit) | ||
95 | return EINA_TRUE; | ||
96 | } | ||
97 | return EINA_FALSE; | ||
98 | } | ||
99 | |||
100 | static Eina_Bool | ||
101 | _evas_image_skip_frame(GifFileType *gif, int frame) | ||
102 | { | ||
103 | int remain_frame = 0; | ||
104 | GifRecordType rec; | ||
105 | |||
106 | if (!gif) return EINA_FALSE; | ||
107 | if (frame == 0) return EINA_TRUE; /* no need to skip */ | ||
108 | if (frame < 0 || frame > FRAME_MAX) return EINA_FALSE; | ||
109 | |||
110 | remain_frame = frame; | ||
111 | |||
112 | do | ||
113 | { | ||
114 | if (DGifGetRecordType(gif, &rec) == GIF_ERROR) return EINA_FALSE; | ||
115 | |||
116 | if (rec == EXTENSION_RECORD_TYPE) | ||
117 | { | ||
118 | int ext_code; | ||
119 | GifByteType *ext; | ||
120 | |||
121 | ext = NULL; | ||
122 | DGifGetExtension(gif, &ext_code, &ext); | ||
123 | while (ext) | ||
124 | { /*skip extention */ | ||
125 | ext = NULL; | ||
126 | DGifGetExtensionNext(gif, &ext); | ||
127 | } | ||
128 | } | ||
129 | |||
130 | if (rec == IMAGE_DESC_RECORD_TYPE) | ||
131 | { | ||
132 | int img_code; | ||
133 | GifByteType *img; | ||
134 | |||
135 | if (DGifGetImageDesc(gif) == GIF_ERROR) return EINA_FALSE; | ||
136 | |||
137 | remain_frame --; | ||
138 | /* we have to count frame, so use DGifGetCode and skip decoding */ | ||
139 | if (DGifGetCode(gif, &img_code, &img) == GIF_ERROR) return EINA_FALSE; | ||
140 | |||
141 | while (img) | ||
142 | { | ||
143 | img = NULL; | ||
144 | DGifGetCodeNext(gif, &img); | ||
145 | } | ||
146 | if (remain_frame < 1) return EINA_TRUE; | ||
147 | } | ||
148 | if (rec == TERMINATE_RECORD_TYPE) return EINA_FALSE; /* end of file */ | ||
149 | |||
150 | } while ((rec != TERMINATE_RECORD_TYPE) && (remain_frame > 0)); | ||
151 | return EINA_FALSE; | ||
152 | } | ||
153 | |||
154 | static Eina_Bool | ||
155 | _evas_image_load_frame_graphic_info(Image_Entry_Frame *frame, GifByteType *ext) | ||
156 | { | ||
157 | Gif_Frame *gif_frame = NULL; | ||
158 | if ((!frame) || (!ext)) return EINA_FALSE; | ||
159 | |||
160 | gif_frame = (Gif_Frame *) frame->info; | ||
161 | |||
162 | /* transparent */ | ||
163 | if ((ext[1] & 0x1) != 0) | ||
164 | gif_frame->frame_info.transparent = ext[4]; | ||
165 | else | ||
166 | gif_frame->frame_info.transparent = -1; | ||
167 | |||
168 | gif_frame->frame_info.input = (ext[1] >>1) & 0x1; | ||
169 | gif_frame->frame_info.disposal = (ext[1] >>2) & 0x7; | ||
170 | gif_frame->frame_info.delay = byte2_to_int(ext[2], ext[3]); | ||
171 | return EINA_TRUE; | ||
172 | } | ||
173 | |||
174 | static Eina_Bool | ||
175 | _evas_image_load_frame_image_des_info(GifFileType *gif, Image_Entry_Frame *frame) | ||
176 | { | ||
177 | Gif_Frame *gif_frame = NULL; | ||
178 | if ((!gif) || (!frame)) return EINA_FALSE; | ||
179 | |||
180 | gif_frame = (Gif_Frame *) frame->info; | ||
181 | gif_frame->image_des.x = gif->Image.Left; | ||
182 | gif_frame->image_des.y = gif->Image.Top; | ||
183 | gif_frame->image_des.w = gif->Image.Width; | ||
184 | gif_frame->image_des.h = gif->Image.Height; | ||
185 | gif_frame->image_des.interlace = gif->Image.Interlace; | ||
186 | return EINA_TRUE; | ||
187 | } | ||
188 | |||
189 | static Eina_Bool | ||
190 | _evas_image_load_frame_image_data(Image_Entry *ie, GifFileType *gif, Image_Entry_Frame *frame, int *error) | ||
191 | { | ||
192 | int w; | ||
193 | int h; | ||
194 | int x; | ||
195 | int y; | ||
196 | int i,j; | ||
197 | int bg; | ||
198 | int r; | ||
199 | int g; | ||
200 | int b; | ||
201 | int alpha; | ||
202 | double per; | ||
203 | double per_inc; | ||
204 | ColorMapObject *cmap; | ||
205 | GifRowType *rows; | ||
206 | int intoffset[] = { 0, 4, 2, 1 }; | ||
207 | int intjump[] = { 8, 8, 4, 2 }; | ||
208 | size_t siz; | ||
209 | int cache_w; | ||
210 | int cache_h; | ||
211 | int cur_h; | ||
212 | int cur_w; | ||
213 | int disposal = 0; | ||
214 | int bg_val = 0; | ||
215 | DATA32 *ptr; | ||
216 | Gif_Frame *gif_frame = NULL; | ||
217 | |||
218 | if ((!gif) || (!frame)) return EINA_FALSE; | ||
219 | |||
220 | gif_frame = (Gif_Frame *) frame->info; | ||
221 | w = gif->Image.Width; | ||
222 | h = gif->Image.Height; | ||
223 | x = gif->Image.Left; | ||
224 | y = gif->Image.Top; | ||
225 | cache_w = ie->w; | ||
226 | cache_h = ie->h; | ||
227 | |||
228 | rows = malloc(h * sizeof(GifRowType *)); | ||
229 | if (!rows) | ||
230 | { | ||
231 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
232 | return EINA_FALSE; | ||
233 | } | ||
234 | for (i = 0; i < h; i++) | ||
235 | { | ||
236 | rows[i] = NULL; | ||
237 | } | ||
238 | for (i = 0; i < h; i++) | ||
239 | { | ||
240 | rows[i] = malloc(w * sizeof(GifPixelType)); | ||
241 | if (!rows[i]) | ||
242 | { | ||
243 | for (i = 0; i < h; i++) | ||
244 | { | ||
245 | if (rows[i]) | ||
246 | { | ||
247 | free(rows[i]); | ||
248 | } | ||
249 | } | ||
250 | free(rows); | ||
251 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
252 | return EINA_FALSE; | ||
253 | } | ||
254 | } | ||
255 | if (gif->Image.Interlace) | ||
256 | { | ||
257 | for (i = 0; i < 4; i++) | ||
258 | { | ||
259 | for (j = intoffset[i]; j < h; j += intjump[i]) | ||
260 | { | ||
261 | DGifGetLine(gif, rows[j], w); | ||
262 | } | ||
263 | } | ||
264 | } | ||
265 | else | ||
266 | { | ||
267 | for (i = 0; i < h; i++) | ||
268 | { | ||
269 | if (DGifGetLine(gif, rows[i], w) != GIF_OK) | ||
270 | { | ||
271 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
272 | goto error; | ||
273 | } | ||
274 | } | ||
275 | } | ||
276 | alpha = gif_frame->frame_info.transparent; | ||
277 | siz = cache_w *cache_h * sizeof(DATA32); | ||
278 | frame->data = malloc(siz); | ||
279 | if (!frame->data) | ||
280 | { | ||
281 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
282 | goto error; | ||
283 | } | ||
284 | ptr = frame->data; | ||
285 | bg = gif->SBackGroundColor; | ||
286 | cmap = (gif->Image.ColorMap ? gif->Image.ColorMap : gif->SColorMap); | ||
287 | |||
288 | if (!cmap) | ||
289 | { | ||
290 | DGifCloseFile(gif); | ||
291 | for (i = 0; i < h; i++) | ||
292 | { | ||
293 | free(rows[i]); | ||
294 | } | ||
295 | free(rows); | ||
296 | if (frame->data) free(frame->data); | ||
297 | *error = EVAS_LOAD_ERROR_CORRUPT_FILE; | ||
298 | return EINA_FALSE; | ||
299 | } | ||
300 | |||
301 | /* get the background value */ | ||
302 | r = cmap->Colors[bg].Red; | ||
303 | g = cmap->Colors[bg].Green; | ||
304 | b = cmap->Colors[bg].Blue; | ||
305 | bg_val = ARGB_JOIN(0xff, r, g, b); | ||
306 | |||
307 | per_inc = 100.0 / (((double)w) * h); | ||
308 | cur_h = h; | ||
309 | cur_w = w; | ||
310 | if (cur_h > cache_h) cur_h = cache_h; | ||
311 | if (cur_w > cache_w) cur_w = cache_w; | ||
312 | |||
313 | if (frame->index > 1) | ||
314 | { | ||
315 | /* get previous frame only frame index is bigger than 1 */ | ||
316 | DATA32 *ptr_src; | ||
317 | Image_Entry_Frame *new_frame = NULL; | ||
318 | int cur_frame = frame->index; | ||
319 | int start_frame = 1; | ||
320 | int j = 0; | ||
321 | |||
322 | if (_find_close_frame(ie, cur_frame, &new_frame)) | ||
323 | start_frame = new_frame->index + 1; | ||
324 | |||
325 | if ((start_frame < 1) || (start_frame > cur_frame)) | ||
326 | { | ||
327 | *error = EVAS_LOAD_ERROR_CORRUPT_FILE; | ||
328 | goto error; | ||
329 | } | ||
330 | /* load previous frame of cur_frame */ | ||
331 | for (j = start_frame; j < cur_frame ; j++) | ||
332 | { | ||
333 | if (!evas_image_load_specific_frame(ie, ie->file, j, error)) | ||
334 | { | ||
335 | *error = EVAS_LOAD_ERROR_CORRUPT_FILE; | ||
336 | goto error; | ||
337 | } | ||
338 | } | ||
339 | if (!_find_frame(ie, cur_frame - 1, &new_frame)) | ||
340 | { | ||
341 | *error = EVAS_LOAD_ERROR_CORRUPT_FILE; | ||
342 | goto error; | ||
343 | } | ||
344 | else | ||
345 | { | ||
346 | Gif_Frame *gif_frame = NULL; | ||
347 | ptr_src = new_frame->data; | ||
348 | if (new_frame->info) | ||
349 | { | ||
350 | gif_frame = (Gif_Frame *)(new_frame->info); | ||
351 | disposal = gif_frame->frame_info.disposal; | ||
352 | } | ||
353 | switch(disposal) /* we only support disposal flag 0,1,2 */ | ||
354 | { | ||
355 | case 1: /* Do not dispose. need previous frame*/ | ||
356 | memcpy(ptr, ptr_src, siz); | ||
357 | /* composite frames */ | ||
358 | ptr = ptr + cache_w * y; | ||
359 | |||
360 | for (i = 0; i < cur_h; i++) | ||
361 | { | ||
362 | ptr = ptr + x; | ||
363 | for (j = 0; j < cur_w; j++) | ||
364 | { | ||
365 | if (rows[i][j] == alpha) | ||
366 | { | ||
367 | ptr++ ; | ||
368 | } | ||
369 | else | ||
370 | { | ||
371 | r = cmap->Colors[rows[i][j]].Red; | ||
372 | g = cmap->Colors[rows[i][j]].Green; | ||
373 | b = cmap->Colors[rows[i][j]].Blue; | ||
374 | *ptr++ = ARGB_JOIN(0xff, r, g, b); | ||
375 | } | ||
376 | per += per_inc; | ||
377 | } | ||
378 | ptr = ptr + (cache_w - (x + cur_w)); | ||
379 | } | ||
380 | break; | ||
381 | case 2: /* Restore to background color */ | ||
382 | memcpy(ptr, ptr_src, siz); | ||
383 | /* composite frames */ | ||
384 | for (i = 0; i < cache_h; i++) | ||
385 | { | ||
386 | if ((i < y) || (i >= (y + cur_h))) | ||
387 | { | ||
388 | for (j = 0; j < cache_w; j++) | ||
389 | { | ||
390 | *ptr = bg_val; | ||
391 | ptr++; | ||
392 | } | ||
393 | } | ||
394 | else | ||
395 | { | ||
396 | int i1, j1; | ||
397 | i1 = i -y; | ||
398 | |||
399 | for (j = 0; j < cache_w; j++) | ||
400 | { | ||
401 | j1 = j - x; | ||
402 | if ((j < x) || (j >= (x + cur_w))) | ||
403 | { | ||
404 | *ptr = bg_val; | ||
405 | ptr++; | ||
406 | } | ||
407 | else | ||
408 | { | ||
409 | r = cmap->Colors[rows[i1][j1]].Red; | ||
410 | g = cmap->Colors[rows[i1][j1]].Green; | ||
411 | b = cmap->Colors[rows[i1][j1]].Blue; | ||
412 | *ptr++ = ARGB_JOIN(0xff, r, g, b); | ||
413 | } | ||
414 | } | ||
415 | } | ||
416 | } | ||
417 | break; | ||
418 | case 0: /* No disposal specified */ | ||
419 | default: | ||
420 | memset(ptr, 0, siz); | ||
421 | for (i = 0; i < cache_h; i++) | ||
422 | { | ||
423 | if ((i < y) || (i >= (y + cur_h))) | ||
424 | { | ||
425 | for (j = 0; j < cache_w; j++) | ||
426 | { | ||
427 | *ptr = bg_val; | ||
428 | ptr++; | ||
429 | } | ||
430 | } | ||
431 | else | ||
432 | { | ||
433 | int i1, j1; | ||
434 | i1 = i -y; | ||
435 | |||
436 | for (j = 0; j < cache_w; j++) | ||
437 | { | ||
438 | j1 = j - x; | ||
439 | if ((j < x) || (j >= (x + cur_w))) | ||
440 | { | ||
441 | *ptr = bg_val; | ||
442 | ptr++; | ||
443 | } | ||
444 | else | ||
445 | { | ||
446 | r = cmap->Colors[rows[i1][j1]].Red; | ||
447 | g = cmap->Colors[rows[i1][j1]].Green; | ||
448 | b = cmap->Colors[rows[i1][j1]].Blue; | ||
449 | *ptr++ = ARGB_JOIN(0xff, r, g, b); | ||
450 | } | ||
451 | } | ||
452 | } | ||
453 | } | ||
454 | break; | ||
455 | } | ||
456 | } | ||
457 | } | ||
458 | else /* first frame decoding */ | ||
459 | { | ||
460 | /* fill background color */ | ||
461 | for (i = 0; i < cache_h; i++) | ||
462 | { | ||
463 | if ((i < y) || (i >= (y + cur_h))) | ||
464 | { | ||
465 | for (j = 0; j < cache_w; j++) | ||
466 | { | ||
467 | *ptr = bg_val; | ||
468 | ptr++; | ||
469 | } | ||
470 | } | ||
471 | else | ||
472 | { | ||
473 | int i1, j1; | ||
474 | i1 = i -y; | ||
475 | |||
476 | for (j = 0; j < cache_w; j++) | ||
477 | { | ||
478 | j1 = j - x; | ||
479 | if ((j < x) || (j >= (x + cur_w))) | ||
480 | { | ||
481 | *ptr = bg_val; | ||
482 | ptr++; | ||
483 | } | ||
484 | else | ||
485 | { | ||
486 | if (rows[i1][j1] == alpha) | ||
487 | { | ||
488 | ptr++ ; | ||
489 | } | ||
490 | else | ||
491 | { | ||
492 | r = cmap->Colors[rows[i1][j1]].Red; | ||
493 | g = cmap->Colors[rows[i1][j1]].Green; | ||
494 | b = cmap->Colors[rows[i1][j1]].Blue; | ||
495 | *ptr++ = ARGB_JOIN(0xff, r, g, b); | ||
496 | } | ||
497 | } | ||
498 | } | ||
499 | } | ||
500 | } | ||
501 | } | ||
502 | |||
503 | for (i = 0; i < h; i++) | ||
504 | { | ||
505 | if (rows[i]) free(rows[i]); | ||
506 | } | ||
507 | if (rows) free(rows); | ||
508 | frame->loaded = EINA_TRUE; | ||
509 | return EINA_TRUE; | ||
510 | error: | ||
511 | for (i = 0; i < h; i++) | ||
512 | { | ||
513 | if (rows[i]) free(rows[i]); | ||
514 | } | ||
515 | if (rows) free(rows); | ||
516 | return EINA_FALSE; | ||
517 | } | ||
518 | |||
519 | static Eina_Bool | ||
520 | _evas_image_load_frame(Image_Entry *ie, GifFileType *gif, Image_Entry_Frame *frame, Frame_Load_Type type, int *error) | ||
521 | { | ||
522 | GifRecordType rec; | ||
523 | int gra_res = 0, img_res = 0; | ||
524 | Eina_Bool res = EINA_FALSE; | ||
525 | Gif_Frame *gif_frame = NULL; | ||
526 | |||
527 | if ((!gif) || (!frame)) return EINA_FALSE; | ||
528 | gif_frame = (Gif_Frame *) frame->info; | ||
529 | |||
530 | if (type > LOAD_FRAME_DATA_INFO) return EINA_FALSE; | ||
531 | |||
532 | do | ||
533 | { | ||
534 | if (DGifGetRecordType(gif, &rec) == GIF_ERROR) return EINA_FALSE; | ||
535 | if (rec == IMAGE_DESC_RECORD_TYPE) | ||
536 | { | ||
537 | img_res++; | ||
538 | break; | ||
539 | } | ||
540 | else if (rec == EXTENSION_RECORD_TYPE) | ||
541 | { | ||
542 | int ext_code; | ||
543 | GifByteType *ext; | ||
544 | |||
545 | ext = NULL; | ||
546 | DGifGetExtension(gif, &ext_code, &ext); | ||
547 | while (ext) | ||
548 | { | ||
549 | if (ext_code == 0xf9) /* Graphic Control Extension */ | ||
550 | { | ||
551 | gra_res++; | ||
552 | /* fill frame info */ | ||
553 | if ((type == LOAD_FRAME_INFO) || (type == LOAD_FRAME_DATA_INFO)) | ||
554 | _evas_image_load_frame_graphic_info(frame,ext); | ||
555 | } | ||
556 | ext = NULL; | ||
557 | DGifGetExtensionNext(gif, &ext); | ||
558 | } | ||
559 | } | ||
560 | } while ((rec != TERMINATE_RECORD_TYPE) && (img_res == 0)); | ||
561 | if (img_res != 1) return EINA_FALSE; | ||
562 | if (DGifGetImageDesc(gif) == GIF_ERROR) return EINA_FALSE; | ||
563 | if ((type == LOAD_FRAME_INFO) || (type == LOAD_FRAME_DATA_INFO)) | ||
564 | _evas_image_load_frame_image_des_info(gif, frame); | ||
565 | |||
566 | if ((type == LOAD_FRAME_DATA) || (type == LOAD_FRAME_DATA_INFO)) | ||
567 | { | ||
568 | res = _evas_image_load_frame_image_data(ie, gif,frame, error); | ||
569 | if (!res) return EINA_FALSE; | ||
570 | } | ||
571 | return EINA_TRUE; | ||
572 | } | ||
573 | |||
574 | |||
575 | /* set frame data to cache entry's data */ | ||
576 | static Eina_Bool | ||
577 | evas_image_load_file_data_gif_internal(Image_Entry *ie, Image_Entry_Frame *frame, int *error) | ||
578 | { | ||
579 | int w; | ||
580 | int h; | ||
581 | int dst_x; | ||
582 | int dst_y; | ||
583 | DATA32 *dst; | ||
584 | DATA32 *src; | ||
585 | int cache_w, cache_h; | ||
586 | size_t siz; | ||
587 | Gif_Frame *gif_frame = NULL; | ||
588 | |||
589 | gif_frame = (Gif_Frame *) frame->info; | ||
590 | cache_w = ie->w; | ||
591 | cache_h = ie->h; | ||
592 | w = gif_frame->image_des.w; | ||
593 | h = gif_frame->image_des.h; | ||
594 | dst_x = gif_frame->image_des.x; | ||
595 | dst_y = gif_frame->image_des.y; | ||
596 | |||
597 | src = frame->data; | ||
598 | |||
599 | if (!evas_cache_image_pixels(ie)) | ||
600 | { | ||
601 | evas_cache_image_surface_alloc(ie, cache_w, cache_h); | ||
602 | } | ||
603 | |||
604 | if (!evas_cache_image_pixels(ie)) | ||
605 | { | ||
606 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
607 | return EINA_FALSE; | ||
608 | } | ||
609 | |||
610 | /* only copy real frame part */ | ||
611 | siz = cache_w * cache_h * sizeof(DATA32); | ||
612 | dst = evas_cache_image_pixels(ie); | ||
613 | |||
614 | memcpy(dst, src, siz); | ||
615 | |||
616 | evas_common_image_premul(ie); | ||
617 | |||
618 | *error = EVAS_LOAD_ERROR_NONE; | ||
619 | return EINA_TRUE; | ||
620 | } | ||
621 | |||
622 | static Eina_Bool | ||
623 | evas_image_load_file_head_gif(Image_Entry *ie, const char *file, const char *key __UNUSED__, int *error) | ||
624 | { | ||
625 | int fd; | ||
626 | GifFileType *gif; | ||
627 | GifRecordType rec; | ||
628 | int w; | ||
629 | int h; | ||
630 | int alpha; | ||
631 | int loop_count = -1; | ||
632 | |||
633 | w = 0; | ||
634 | h = 0; | ||
635 | alpha = -1; | ||
636 | |||
637 | #ifndef __EMX__ | ||
638 | fd = open(file, O_RDONLY); | ||
639 | #else | ||
640 | fd = open(file, O_RDONLY | O_BINARY); | ||
641 | #endif | ||
642 | if (fd < 0) | ||
643 | { | ||
644 | *error = EVAS_LOAD_ERROR_DOES_NOT_EXIST; | ||
645 | return EINA_FALSE; | ||
646 | } | ||
647 | |||
648 | gif = DGifOpenFileHandle(fd); | ||
649 | if (!gif) | ||
650 | { | ||
651 | if (fd) close(fd); | ||
652 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
653 | return EINA_FALSE; | ||
654 | } | ||
655 | |||
656 | /* check logical screen size */ | ||
657 | w = gif->SWidth; | ||
658 | h = gif->SHeight; | ||
659 | |||
660 | if ((w < 1) || (h < 1) || (w > IMG_MAX_SIZE) || (h > IMG_MAX_SIZE) || | ||
661 | IMG_TOO_BIG(w, h)) | ||
662 | { | ||
663 | DGifCloseFile(gif); | ||
664 | if (IMG_TOO_BIG(w, h)) | ||
665 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
666 | else | ||
667 | *error = EVAS_LOAD_ERROR_GENERIC; | ||
668 | return EINA_FALSE; | ||
669 | } | ||
670 | ie->w = w; | ||
671 | ie->h = h; | ||
672 | |||
673 | do | ||
674 | { | ||
675 | if (DGifGetRecordType(gif, &rec) == GIF_ERROR) | ||
676 | { | ||
677 | /* PrintGifError(); */ | ||
678 | DGifCloseFile(gif); | ||
679 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
680 | return EINA_FALSE; | ||
681 | } | ||
682 | |||
683 | /* image descript info */ | ||
684 | if (rec == IMAGE_DESC_RECORD_TYPE) | ||
685 | { | ||
686 | int img_code; | ||
687 | GifByteType *img; | ||
688 | |||
689 | if (DGifGetImageDesc(gif) == GIF_ERROR) | ||
690 | { | ||
691 | /* PrintGifError(); */ | ||
692 | DGifCloseFile(gif); | ||
693 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
694 | return EINA_FALSE; | ||
695 | } | ||
696 | /* we have to count frame, so use DGifGetCode and skip decoding */ | ||
697 | if (DGifGetCode(gif, &img_code, &img) == GIF_ERROR) | ||
698 | { | ||
699 | /* PrintGifError(); */ | ||
700 | DGifCloseFile(gif); | ||
701 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
702 | return EINA_FALSE; | ||
703 | } | ||
704 | while (img) | ||
705 | { | ||
706 | img = NULL; | ||
707 | DGifGetCodeNext(gif, &img); | ||
708 | } | ||
709 | } | ||
710 | else if (rec == EXTENSION_RECORD_TYPE) | ||
711 | { | ||
712 | int ext_code; | ||
713 | GifByteType *ext; | ||
714 | |||
715 | ext = NULL; | ||
716 | DGifGetExtension(gif, &ext_code, &ext); | ||
717 | while (ext) | ||
718 | { | ||
719 | if (ext_code == 0xf9) /* Graphic Control Extension */ | ||
720 | { | ||
721 | if ((ext[1] & 1) && (alpha < 0)) alpha = (int)ext[4]; | ||
722 | } | ||
723 | else if (ext_code == 0xff) /* application extension */ | ||
724 | { | ||
725 | if (!strncmp ((char*)(&ext[1]), "NETSCAPE2.0", 11) || | ||
726 | !strncmp ((char*)(&ext[1]), "ANIMEXTS1.0", 11)) | ||
727 | { | ||
728 | ext=NULL; | ||
729 | DGifGetExtensionNext(gif, &ext); | ||
730 | |||
731 | if (ext[1] == 0x01) | ||
732 | { | ||
733 | loop_count = ext[2] + (ext[3] << 8); | ||
734 | if (loop_count > 0) loop_count++; | ||
735 | } | ||
736 | } | ||
737 | } | ||
738 | |||
739 | ext = NULL; | ||
740 | DGifGetExtensionNext(gif, &ext); | ||
741 | } | ||
742 | } | ||
743 | } while (rec != TERMINATE_RECORD_TYPE); | ||
744 | |||
745 | if (alpha >= 0) ie->flags.alpha = 1; | ||
746 | |||
747 | if (gif->ImageCount > 1) | ||
748 | { | ||
749 | ie->flags.animated = 1; | ||
750 | ie->loop_count = loop_count; | ||
751 | ie->loop_hint = EVAS_IMAGE_ANIMATED_HINT_LOOP; | ||
752 | ie->frame_count = gif->ImageCount; | ||
753 | ie->frames = NULL; | ||
754 | } | ||
755 | |||
756 | DGifCloseFile(gif); | ||
757 | *error = EVAS_LOAD_ERROR_NONE; | ||
758 | return EINA_TRUE; | ||
759 | } | ||
760 | |||
761 | static Eina_Bool | ||
762 | evas_image_load_specific_frame(Image_Entry *ie, const char *file, int frame_index, int *error) | ||
763 | { | ||
764 | int fd; | ||
765 | GifFileType *gif; | ||
766 | Image_Entry_Frame *frame = NULL; | ||
767 | Gif_Frame *gif_frame = NULL; | ||
768 | |||
769 | #ifndef __EMX__ | ||
770 | fd = open(file, O_RDONLY); | ||
771 | #else | ||
772 | fd = open(file, O_RDONLY | O_BINARY); | ||
773 | #endif | ||
774 | if (fd < 0) | ||
775 | { | ||
776 | *error = EVAS_LOAD_ERROR_DOES_NOT_EXIST; | ||
777 | return EINA_FALSE; | ||
778 | } | ||
779 | |||
780 | gif = DGifOpenFileHandle(fd); | ||
781 | if (!gif) | ||
782 | { | ||
783 | if (fd) close(fd); | ||
784 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
785 | return EINA_FALSE; | ||
786 | } | ||
787 | if (!_evas_image_skip_frame(gif, frame_index-1)) | ||
788 | { | ||
789 | if (fd) close(fd); | ||
790 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
791 | return EINA_FALSE; | ||
792 | } | ||
793 | |||
794 | frame = malloc(sizeof (Image_Entry_Frame)); | ||
795 | if (!frame) | ||
796 | { | ||
797 | if (fd) close(fd); | ||
798 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
799 | return EINA_FALSE; | ||
800 | } | ||
801 | |||
802 | gif_frame = malloc(sizeof (Gif_Frame)); | ||
803 | if (!gif_frame) | ||
804 | { | ||
805 | if (fd) close(fd); | ||
806 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
807 | return EINA_FALSE; | ||
808 | } | ||
809 | frame->info = gif_frame; | ||
810 | frame->index = frame_index; | ||
811 | if (!_evas_image_load_frame(ie,gif, frame, LOAD_FRAME_DATA_INFO,error)) | ||
812 | { | ||
813 | if (fd) close(fd); | ||
814 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
815 | return EINA_FALSE; | ||
816 | } | ||
817 | |||
818 | ie->frames = eina_list_append(ie->frames, frame); | ||
819 | DGifCloseFile(gif); | ||
820 | return EINA_TRUE; | ||
821 | } | ||
822 | |||
823 | static Eina_Bool | ||
824 | evas_image_load_file_data_gif(Image_Entry *ie, const char *file, const char *key __UNUSED__, int *error) | ||
825 | { | ||
826 | int cur_frame_index; | ||
827 | Image_Entry_Frame *frame = NULL; | ||
828 | Eina_Bool hit; | ||
829 | |||
830 | if(!ie->flags.animated) | ||
831 | cur_frame_index = 1; | ||
832 | else | ||
833 | cur_frame_index = ie->cur_frame; | ||
834 | |||
835 | if ((ie->flags.animated) && | ||
836 | ((cur_frame_index <0) || (cur_frame_index > FRAME_MAX) || (cur_frame_index > ie->frame_count))) | ||
837 | { | ||
838 | *error = EVAS_LOAD_ERROR_GENERIC; | ||
839 | return EINA_FALSE; | ||
840 | } | ||
841 | |||
842 | /* first time frame is set to be 0. so default is 1 */ | ||
843 | if (cur_frame_index == 0) cur_frame_index++; | ||
844 | |||
845 | /* Check current frame exists in hash table */ | ||
846 | hit = _find_frame(ie, cur_frame_index, &frame); | ||
847 | |||
848 | /* if current frame exist in has table, check load flag */ | ||
849 | if (hit) | ||
850 | { | ||
851 | if (frame->loaded) | ||
852 | evas_image_load_file_data_gif_internal(ie,frame,error); | ||
853 | else | ||
854 | { | ||
855 | int fd; | ||
856 | GifFileType *gif; | ||
857 | |||
858 | #ifndef __EMX__ | ||
859 | fd = open(file, O_RDONLY); | ||
860 | #else | ||
861 | fd = open(file, O_RDONLY | O_BINARY); | ||
862 | #endif | ||
863 | if (fd < 0) | ||
864 | { | ||
865 | *error = EVAS_LOAD_ERROR_DOES_NOT_EXIST; | ||
866 | return EINA_FALSE; | ||
867 | } | ||
868 | |||
869 | gif = DGifOpenFileHandle(fd); | ||
870 | if (!gif) | ||
871 | { | ||
872 | if (fd) close(fd); | ||
873 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
874 | return EINA_FALSE; | ||
875 | } | ||
876 | _evas_image_skip_frame(gif, cur_frame_index-1); | ||
877 | if (!_evas_image_load_frame(ie, gif, frame, LOAD_FRAME_DATA,error)) | ||
878 | { | ||
879 | if (fd) close(fd); | ||
880 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
881 | return EINA_FALSE; | ||
882 | } | ||
883 | if (!evas_image_load_file_data_gif_internal(ie, frame, error)) | ||
884 | { | ||
885 | if (fd) close(fd); | ||
886 | *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT; | ||
887 | return EINA_FALSE; | ||
888 | } | ||
889 | DGifCloseFile(gif); | ||
890 | *error = EVAS_LOAD_ERROR_NONE; | ||
891 | return EINA_TRUE; | ||
892 | } | ||
893 | } | ||
894 | /* current frame does is not exist */ | ||
895 | else | ||
896 | { | ||
897 | if (!evas_image_load_specific_frame(ie, file, cur_frame_index, error)) | ||
898 | { | ||
899 | return EINA_FALSE; | ||
900 | } | ||
901 | hit = EINA_FALSE; | ||
902 | frame = NULL; | ||
903 | hit = _find_frame(ie, cur_frame_index, &frame); | ||
904 | if (!hit) return EINA_FALSE; | ||
905 | if (!evas_image_load_file_data_gif_internal(ie, frame, error)) | ||
906 | { | ||
907 | *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED; | ||
908 | return EINA_FALSE; | ||
909 | } | ||
910 | return EINA_TRUE; | ||
911 | } | ||
912 | return EINA_FALSE; | ||
913 | } | ||
914 | |||
915 | static double | ||
916 | evas_image_load_frame_duration_gif(Image_Entry *ie, const char *file, const int start_frame, const int frame_num) | ||
917 | { | ||
918 | int fd; | ||
919 | GifFileType *gif; | ||
920 | GifRecordType rec; | ||
921 | int done; | ||
922 | int current_frame = 1; | ||
923 | int remain_frames = frame_num; | ||
924 | double duration = 0; | ||
925 | int frame_count = 0; | ||
926 | |||
927 | frame_count = ie->frame_count; | ||
928 | |||
929 | if (!ie->flags.animated) return -1; | ||
930 | if ((start_frame + frame_num) > frame_count) return -1; | ||
931 | if (frame_num < 0) return -1; | ||
932 | |||
933 | done = 0; | ||
934 | |||
935 | #ifndef __EMX__ | ||
936 | fd = open(file, O_RDONLY); | ||
937 | #else | ||
938 | fd = open(file, O_RDONLY | O_BINARY); | ||
939 | #endif | ||
940 | if (fd < 0) return -1; | ||
941 | |||
942 | gif = DGifOpenFileHandle(fd); | ||
943 | if (!gif) | ||
944 | { | ||
945 | if (fd) close(fd); | ||
946 | return -1; | ||
947 | } | ||
948 | |||
949 | do | ||
950 | { | ||
951 | if (DGifGetRecordType(gif, &rec) == GIF_ERROR) | ||
952 | { | ||
953 | rec = TERMINATE_RECORD_TYPE; | ||
954 | } | ||
955 | if (rec == IMAGE_DESC_RECORD_TYPE) | ||
956 | { | ||
957 | int img_code; | ||
958 | GifByteType *img; | ||
959 | |||
960 | if (DGifGetImageDesc(gif) == GIF_ERROR) | ||
961 | { | ||
962 | /* PrintGifError(); */ | ||
963 | rec = TERMINATE_RECORD_TYPE; | ||
964 | } | ||
965 | current_frame++; | ||
966 | /* we have to count frame, so use DGifGetCode and skip decoding */ | ||
967 | if (DGifGetCode(gif, &img_code, &img) == GIF_ERROR) | ||
968 | { | ||
969 | rec = TERMINATE_RECORD_TYPE; | ||
970 | } | ||
971 | while (img) | ||
972 | { | ||
973 | img = NULL; | ||
974 | DGifGetExtensionNext(gif, &img); | ||
975 | } | ||
976 | } | ||
977 | else if (rec == EXTENSION_RECORD_TYPE) | ||
978 | { | ||
979 | int ext_code; | ||
980 | GifByteType *ext; | ||
981 | |||
982 | ext = NULL; | ||
983 | DGifGetExtension(gif, &ext_code, &ext); | ||
984 | while (ext) | ||
985 | { | ||
986 | if (ext_code == 0xf9) /* Graphic Control Extension */ | ||
987 | { | ||
988 | if ((current_frame >= start_frame) && (current_frame <= frame_count)) | ||
989 | { | ||
990 | int frame_duration = 0; | ||
991 | if (remain_frames < 0) break; | ||
992 | frame_duration = byte2_to_int (ext[2], ext[3]); | ||
993 | if (frame_duration == 0) | ||
994 | duration += 0.1; | ||
995 | else | ||
996 | duration += (double)frame_duration/100; | ||
997 | remain_frames --; | ||
998 | } | ||
999 | } | ||
1000 | ext = NULL; | ||
1001 | DGifGetExtensionNext(gif, &ext); | ||
1002 | } | ||
1003 | } | ||
1004 | } while (rec != TERMINATE_RECORD_TYPE); | ||
1005 | |||
1006 | DGifCloseFile(gif); | ||
1007 | return duration; | ||
1008 | } | ||
1009 | |||
1010 | static int | ||
1011 | module_open(Evas_Module *em) | ||
1012 | { | ||
1013 | if (!em) return 0; | ||
1014 | em->functions = (void *)(&evas_image_load_gif_func); | ||
1015 | return 1; | ||
1016 | } | ||
1017 | |||
1018 | static void | ||
1019 | module_close(Evas_Module *em __UNUSED__) | ||
1020 | { | ||
1021 | } | ||
1022 | |||
1023 | static Evas_Module_Api evas_modapi = | ||
1024 | { | ||
1025 | EVAS_MODULE_API_VERSION, | ||
1026 | "gif", | ||
1027 | "none", | ||
1028 | { | ||
1029 | module_open, | ||
1030 | module_close | ||
1031 | } | ||
1032 | }; | ||
1033 | |||
1034 | EVAS_MODULE_DEFINE(EVAS_MODULE_TYPE_IMAGE_LOADER, image_loader, gif); | ||
1035 | |||
1036 | #ifndef EVAS_STATIC_BUILD_GIF | ||
1037 | EVAS_EINA_MODULE_DEFINE(image_loader, gif); | ||
1038 | #endif | ||