aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c
diff options
context:
space:
mode:
Diffstat (limited to 'libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c')
-rw-r--r--libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c1002
1 files changed, 0 insertions, 1002 deletions
diff --git a/libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c b/libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c
deleted file mode 100644
index 4449db0..0000000
--- a/libraries/evas/src/modules/loaders/psd/evas_image_load_psd.c
+++ /dev/null
@@ -1,1002 +0,0 @@
1#define _XOPEN_SOURCE
2
3#ifdef HAVE_CONFIG_H
4# include <config.h>
5#endif
6
7#include <stdlib.h>
8#include <stdio.h>
9#include <string.h>
10#include <errno.h>
11
12#include <math.h>
13#include <sys/types.h>
14#include <sys/stat.h>
15#include <fcntl.h>
16
17#ifdef HAVE_NETINET_IN_H
18# include <netinet/in.h>
19#endif
20
21#ifdef HAVE_UNISTD_H
22# include <unistd.h>
23#endif
24
25#ifdef HAVE_EVIL
26# include <Evil.h>
27#endif
28
29#include "evas_common.h"
30#include "evas_private.h"
31
32typedef struct _PSD_Header PSD_Header;
33
34typedef enum _PSD_Mode
35 {
36 PSD_GREYSCALE = 1,
37 PSD_INDEXED = 2,
38 PSD_RGB = 3,
39 PSD_CMYK = 4
40 } PSD_Mode;
41
42struct _PSD_Header
43{
44 unsigned char signature[4];
45 unsigned short version;
46 unsigned char reserved[9];
47 unsigned short channels;
48 unsigned int height;
49 unsigned int width;
50 unsigned short depth;
51
52 unsigned short channel_num;
53
54 PSD_Mode mode;
55};
56
57enum {
58 READ_COMPRESSED_SUCCESS,
59 READ_COMPRESSED_ERROR_FILE_CORRUPT,
60 READ_COMPRESSED_ERROR_FILE_READ_ERROR
61};
62
63static Eina_Bool get_compressed_channels_length(PSD_Header *Head,
64 const unsigned char *map, size_t length, size_t *position,
65 unsigned short *rle_table,
66 unsigned int *chanlen);
67
68static int
69read_ushort(const unsigned char *map, size_t length, size_t *position, unsigned short *ret)
70{
71 if (*position + 2 > length) return 0;
72 // FIXME: need to check order
73 *ret = (map[(*position) + 0] << 8) | map[(*position) + 1];
74 *position += 2;
75 return 1;
76}
77
78static int
79read_uint(const unsigned char *map, size_t length, size_t *position, unsigned int *ret)
80{
81 if (*position + 4 > length) return 0;
82 // FIXME: need to check order
83 *ret = ARGB_JOIN(map[(*position) + 0], map[(*position) + 1], map[(*position) + 2], map[(*position) + 3]);
84 *position += 4;
85 return 1;
86}
87
88static int
89read_block(const unsigned char *map, size_t length, size_t *position, void *target, size_t size)
90{
91 if (*position + size > length) return 0;
92 memcpy(target, map + *position, size);
93 *position += size;
94 return 1;
95}
96
97// Internal function used to get the Psd header from the current file.
98Eina_Bool
99psd_get_header(PSD_Header *header, const unsigned char *map, size_t length, size_t *position)
100{
101 unsigned short tmp;
102
103#define CHECK_RET(Call) \
104 if (!Call) return EINA_FALSE;
105
106 CHECK_RET(read_block(map, length, position, header->signature, 4));
107 CHECK_RET(read_ushort(map, length, position, &header->version));
108 CHECK_RET(read_block(map, length, position, header->reserved, 6));
109 CHECK_RET(read_ushort(map, length, position, &header->channels));
110 CHECK_RET(read_uint(map, length, position, &header->height));
111 CHECK_RET(read_uint(map, length, position, &header->width));
112 CHECK_RET(read_ushort(map, length, position, &header->depth));
113
114 CHECK_RET(read_ushort(map, length, position, &tmp));
115 header->mode = tmp;
116
117#undef CHECK_RET
118
119 /* fprintf(stderr, "<<<<<<<<<<<\nsignature : %c%c%c%c\n", */
120 /* header->signature[0], */
121 /* header->signature[1], */
122 /* header->signature[2], */
123 /* header->signature[3]); */
124 /* fprintf(stderr, "version : %i\n", header->version); */
125 /* fprintf(stderr, "channels : %i\n", header->channels); */
126 /* fprintf(stderr, "width x height : %dx%d\n", header->width, header->height); */
127 /* fprintf(stderr, "depth : %i\n", header->depth); */
128 /* fprintf(stderr, "mode : %i\n>>>>>>>>>>>>\n", header->mode); */
129
130 return EINA_TRUE;
131}
132
133
134// Internal function used to check if the HEADER is a valid Psd header.
135Eina_Bool
136is_psd(PSD_Header *header)
137{
138 if (strncmp((char*)header->signature, "8BPS", 4))
139 return EINA_FALSE;
140 if (header->version != 1)
141 return EINA_FALSE;
142 if (header->channels < 1 || header->channels > 24)
143 return EINA_FALSE;
144 if (header->height < 1 || header->width < 1)
145 return EINA_FALSE;
146 if (header->depth != 1 && header->depth != 8 && header->depth != 16)
147 return EINA_FALSE;
148
149 return EINA_TRUE;
150}
151
152static Eina_Bool
153evas_image_load_file_head_psd(Image_Entry *ie, const char *FileName,
154 const char *key __UNUSED__, int *error)
155{
156 Eina_File *f;
157 void *map;
158 size_t length;
159 size_t position;
160 PSD_Header header;
161 Eina_Bool correct;
162
163 *error = EVAS_LOAD_ERROR_NONE;
164
165 f = eina_file_open(FileName, 0);
166 if (f == NULL)
167 {
168 *error = EVAS_LOAD_ERROR_DOES_NOT_EXIST;
169 return EINA_FALSE;
170 }
171
172 map = eina_file_map_all(f, EINA_FILE_SEQUENTIAL);
173 length = eina_file_size_get(f);
174 position = 0;
175 if (!map || length < 1)
176 {
177 eina_file_close(f);
178 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
179 return EINA_FALSE;
180 }
181 correct = psd_get_header(&header, map, length, &position);
182
183 eina_file_map_free(f, map);
184 eina_file_close(f);
185
186 if (!correct || !is_psd(&header))
187 {
188 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
189 return EINA_FALSE;
190 }
191
192 ie->w = header.width;
193 ie->h = header.height;
194 if (header.channels == 3) ie->flags.alpha = 0;
195 else ie->flags.alpha = 1;
196
197 return EINA_TRUE;
198}
199
200static unsigned int
201read_compressed_channel(const unsigned char *map, size_t length, size_t *position,
202 const unsigned int channel_length __UNUSED__,
203 unsigned int size,
204 unsigned char* channel)
205{
206 // FIXME: what does channel_length means, and why is it not used
207 unsigned int i;
208 char headbyte, c;
209
210#define CHECK_RET(Call) \
211 if (!Call) return READ_COMPRESSED_ERROR_FILE_READ_ERROR; \
212
213 for (i = 0; i < size; )
214 {
215 CHECK_RET(read_block(map, length, position, &headbyte, 1));
216
217 if (headbyte >= 0)
218 {
219 if (i + headbyte > size)
220 return READ_COMPRESSED_ERROR_FILE_CORRUPT;
221 CHECK_RET(read_block(map, length, position, channel + i, headbyte + 1));
222
223 i += headbyte + 1;
224 }
225 else if (headbyte >= -127 && headbyte <= -1)
226 {
227 int run;
228
229 CHECK_RET(read_block(map, length, position, &c, 1));
230
231 run = c;
232 /* if (run == -1) */
233 /* return READ_COMPRESSED_ERROR_FILE_READ_ERROR; */
234
235 if (i + (-headbyte + 1) > size)
236 return READ_COMPRESSED_ERROR_FILE_CORRUPT;
237
238 memset(channel + i, run, -headbyte + 1);
239 i += -headbyte + 1;
240 }
241 }
242
243#undef CHECK_RET
244
245 return READ_COMPRESSED_SUCCESS;
246}
247
248
249Eina_Bool
250psd_get_data(Image_Entry *ie __UNUSED__,
251 PSD_Header *head,
252 const unsigned char *map, size_t length, size_t *position,
253 unsigned char *buffer, Eina_Bool compressed,
254 int *error)
255{
256 unsigned int c, x, y, numchan, bps, bpc, bpp;
257 unsigned int pixels_count;
258 unsigned char *channel = NULL;
259 unsigned char *data = NULL;
260
261 // Added 01-07-2009: This is needed to correctly load greyscale and
262 // paletted images.
263 switch (head->mode)
264 {
265 case PSD_GREYSCALE:
266 case PSD_INDEXED:
267 numchan = 1;
268 break;
269 default:
270 numchan = 3;
271 }
272
273 bpp = head->channels;
274 bpc = head->depth / 8;
275 pixels_count = head->width * head->height;
276
277 data = malloc(sizeof (unsigned char) * pixels_count * bpp);
278 if (!data) return EINA_FALSE;
279
280 channel = malloc(sizeof (unsigned char) * pixels_count * bpc);
281 if (!channel)
282 {
283 free(data);
284 return EINA_FALSE;
285 }
286
287 bps = head->width * head->channels * bpc;
288 // @TODO: Add support for this in, though I have yet to run across a .psd
289 // file that uses this.
290 if (compressed && bpc == 2)
291 {
292 free(data);
293 free(channel);
294 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
295 return EINA_FALSE;
296 }
297
298#define CHECK_RET(Call) \
299 if (!Call) \
300 { \
301 free(data); \
302 free(channel); \
303 return EINA_FALSE; \
304 }
305
306 if (!compressed)
307 {
308 if (bpc == 1)
309 {
310 for (c = 0; c < numchan; c++)
311 {
312 unsigned char *tmp = channel;
313
314 CHECK_RET(read_block(map, length, position, tmp, pixels_count));
315
316 for (y = 0; y < head->height * bps; y += bps)
317 {
318 for (x = 0; x < bps; x += bpp, tmp++)
319 {
320 data[y + x + c] = *tmp;
321 }
322 }
323 }
324
325 // Accumulate any remaining channels into a single alpha channel
326 //@TODO: This needs to be changed for greyscale images.
327 for (; c < head->channels; c++)
328 {
329 unsigned char *tmp = channel;
330
331 CHECK_RET(read_block(map, length, position, channel, pixels_count));
332
333 for (y = 0; y < head->height * bps; y += bps)
334 {
335 for (x = 0; x < bps; x += bpp, tmp++)
336 {
337 unsigned short newval;
338
339 // previous formula was : (old / 255 * new / 255) * 255
340 newval = (*tmp) * data[y + x + 3];
341
342 data[y + x + 3] = newval >> 8;
343 }
344 }
345 }
346 }
347 else
348 {
349 int bps2;
350
351 bps2 = bps / 2;
352
353 // iCurImage->Bpc == 2
354 for (c = 0; c < numchan; c++)
355 {
356 unsigned short *shortptr = (unsigned short*) channel;
357
358 CHECK_RET(read_block(map, length, position, channel, pixels_count * 2));
359
360 for (y = 0; y < head->height * bps2; y += bps2)
361 {
362 for (x = 0; x < (unsigned int)bps2; x += bpp, shortptr++)
363 {
364 ((unsigned short*)data)[y + x + c] = *shortptr;
365 }
366 }
367 }
368
369 // Accumulate any remaining channels into a single alpha channel
370 //@TODO: This needs to be changed for greyscale images.
371 for (; c < head->channels; c++)
372 {
373 unsigned short *shortptr = (unsigned short*) channel;
374
375 CHECK_RET(read_block(map, length, position, channel, pixels_count * 2));
376
377 for (y = 0; y < head->height * bps2; y += bps2)
378 {
379 for (x = 0; x < (unsigned int)bps2; x += bpp, shortptr)
380 {
381 unsigned int newval;
382
383 newval = *shortptr * ((unsigned short*)data)[y + x + 3];
384
385 ((unsigned short*)data)[y + x + 3] = newval >> 16;
386 }
387 }
388 }
389 }
390 }
391 else
392 {
393 unsigned short *rle_table;
394 unsigned int *chanlen;
395
396 rle_table = alloca(head->height * head->channel_num * sizeof (unsigned short));
397 chanlen = alloca(head->channel_num * sizeof (unsigned int));
398 if (!get_compressed_channels_length(head, map, length, position, rle_table, chanlen))
399 goto file_read_error;
400
401 for (c = 0; c < numchan; c++)
402 {
403 unsigned char *tmp = channel;
404 int err;
405
406 err = read_compressed_channel(map, length, position,
407 chanlen[c],
408 pixels_count,
409 channel);
410 if (err == READ_COMPRESSED_ERROR_FILE_CORRUPT)
411 goto file_corrupt;
412 else if (err == READ_COMPRESSED_ERROR_FILE_READ_ERROR)
413 goto file_read_error;
414
415 for (y = 0; y < head->height * bps; y += bps)
416 {
417 for (x = 0; x < bps; x += bpp, tmp++)
418 {
419 data[y + x + c] = *tmp;
420 }
421 }
422 }
423
424 // Initialize the alpha channel to solid
425 //@TODO: This needs to be changed for greyscale images.
426 if (head->channels >= 4)
427 {
428 for (y = 0; y < head->height * bps; y += bps)
429 {
430 for (x = 0; x < bps; x += bpp)
431 {
432 data[y + x + 3] = 255;
433 }
434 }
435
436 for (; c < head->channels; c++)
437 {
438 unsigned char *tmp = channel;
439 int err;
440
441 err = read_compressed_channel(map, length, position,
442 chanlen[c],
443 pixels_count,
444 channel);
445 if (err == READ_COMPRESSED_ERROR_FILE_CORRUPT)
446 goto file_corrupt;
447 else if (err == READ_COMPRESSED_ERROR_FILE_READ_ERROR)
448 goto file_read_error;
449
450 for (y = 0; y < head->height * bps; y += bps)
451 {
452 for (x = 0; x < bps; x += bpp, tmp++)
453 {
454 unsigned short newval;
455
456 newval = *tmp * data[y + x + 3];
457
458 data[y + x + 3] = newval >> 8;
459 }
460 }
461 }
462 }
463 }
464
465 if (bpp == 3)
466 {
467 for (x = 0; x < pixels_count; x++)
468 {
469 buffer[x * 4 + 0] = data[x * 3 + 2];
470 buffer[x * 4 + 1] = data[x * 3 + 1];
471 buffer[x * 4 + 2] = data[x * 3 + 0];
472 buffer[x * 4 + 3] = 255;
473 }
474 }
475 else
476 {
477 // BRGA to RGBA
478 for (x= 0; x < pixels_count; x++)
479 {
480 buffer[x * 4 + 0] = data[x * 4 + 2];
481 buffer[x * 4 + 1] = data[x * 4 + 1];
482 buffer[x * 4 + 2] = data[x * 4 + 0];
483 buffer[x * 4 + 3] = data[x * 4 + 3];
484 }
485 }
486
487 free(channel);
488 free(data);
489 return EINA_TRUE;
490
491#undef CHECK_RET
492
493 file_corrupt:
494 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
495
496 file_read_error:
497 free(channel);
498 free(data);
499
500 return EINA_FALSE;
501}
502
503
504Eina_Bool
505get_single_channel(Image_Entry *ie __UNUSED__,
506 PSD_Header *head,
507 const unsigned char *map, size_t length, size_t *position,
508 unsigned char *buffer,
509 Eina_Bool compressed)
510{
511 unsigned int i, bpc;
512 char headbyte;
513 int c;
514 int pixels_count;
515
516 bpc = (head->depth / 8);
517 pixels_count = head->width * head->height;
518
519#define CHECK_RET(Call) \
520 if (!Call) return EINA_FALSE;
521
522 if (!compressed)
523 {
524 if (bpc == 1)
525 {
526 CHECK_RET(read_block(map, length, position, buffer, pixels_count));
527 }
528 else
529 { // Bpc == 2
530 CHECK_RET(read_block(map, length, position, buffer, pixels_count * 2));
531 }
532 }
533 else
534 {
535 for (i = 0; i < (unsigned int)pixels_count; )
536 {
537 CHECK_RET(read_block(map, length, position, &headbyte, 1));
538
539 if (headbyte >= 0)
540 { // && HeadByte <= 127
541 CHECK_RET(read_block(map, length, position, buffer + i, headbyte + 1));
542
543 i += headbyte + 1;
544 }
545 if (headbyte >= -127 && headbyte <= -1)
546 {
547 int run;
548
549 CHECK_RET(read_block(map, length, position, &c, 1));
550
551 run = c;
552 if (run == -1) return EINA_FALSE;
553
554 memset(buffer + i, run, -headbyte + 1);
555 i += -headbyte + 1;
556 }
557 }
558 }
559
560#undef CHECK_RET
561
562 return EINA_TRUE;
563}
564
565Eina_Bool
566read_psd_grey(Image_Entry *ie, PSD_Header *head, const unsigned char *map, size_t length, size_t *position, int *error)
567{
568 unsigned int color_mode, resource_size, misc_info;
569 unsigned short compressed;
570 void *surface = NULL;
571
572 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
573
574#define CHECK_RET(Call) \
575 if (!Call) return EINA_FALSE;
576
577 CHECK_RET(read_uint(map, length, position, &color_mode));
578 // Skip over the 'color mode data section'
579 *position += color_mode;
580
581 CHECK_RET(read_uint(map, length, position, &resource_size));
582 // Read the 'image resources section'
583 *position += resource_size;
584
585 CHECK_RET(read_uint(map, length, position, &misc_info));
586 *position += misc_info;
587
588 CHECK_RET(read_ushort(map, length, position, &compressed));
589
590 ie->w = head->width;
591 ie->h = head->height;
592 if (head->channels == 3) ie->flags.alpha = 0;
593 else ie->flags.alpha = 1;
594
595 head->channel_num = head->channels;
596 // Temporary to read only one channel...some greyscale .psd files have 2.
597 head->channels = 1;
598
599 switch (head->depth)
600 {
601 case 8:
602 case 16:
603 break;
604 default:
605 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
606 return EINA_FALSE;
607 }
608
609 evas_cache_image_surface_alloc(ie, ie->w, ie->h);
610 surface = evas_cache_image_pixels(ie);
611 if (!surface)
612 {
613 *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED;
614 goto cleanup_error;
615 }
616
617 if (!psd_get_data(ie, head, map, length, position, surface, compressed, error))
618 goto cleanup_error;
619
620 return EINA_TRUE;
621
622#undef CHECK_RET
623
624 cleanup_error:
625 return EINA_FALSE;
626}
627
628
629Eina_Bool
630read_psd_indexed(Image_Entry *ie, PSD_Header *head, const unsigned char *map, size_t length, size_t *position, int *error)
631{
632 unsigned int color_mode, resource_size, misc_info;
633 unsigned short compressed;
634 void *surface;
635
636 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
637
638#define CHECK_RET(Call) \
639 if (!(Call)) return EINA_FALSE;
640
641 CHECK_RET(read_uint(map, length, position, &color_mode));
642 CHECK_RET(!(color_mode % 3));
643 /*
644 Palette = (unsigned char*)malloc(Colormode);
645 if (Palette == NULL)
646 return EINA_FALSE;
647 if (fread(&Palette, 1, Colormode, file) != Colormode)
648 goto cleanup_error;
649 */
650 // Skip over the 'color mode data section'
651 *position += color_mode;
652
653 // Read the 'image resources section'
654 CHECK_RET(read_uint(map, length, position, &resource_size));
655 *position += resource_size;
656
657 CHECK_RET(read_uint(map, length, position, &misc_info));
658 *position += misc_info;
659
660 CHECK_RET(read_ushort(map, length, position, &compressed));
661
662 if (head->channels != 1 || head->depth != 8)
663 {
664 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
665 return EINA_FALSE;
666 }
667 head->channel_num = head->channels;
668
669 ie->w = head->width;
670 ie->h = head->height;
671 if (head->channels == 3) ie->flags.alpha = 0;
672 else ie->flags.alpha = 1;
673
674 evas_cache_image_surface_alloc(ie, ie->w, ie->h);
675 surface = evas_cache_image_pixels(ie);
676 if (!surface)
677 {
678 *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED;
679 return EINA_FALSE;
680 }
681
682 if (!psd_get_data(ie, head, map, length, position, surface, compressed, error))
683 return EINA_FALSE;
684 return EINA_TRUE;
685
686#undef CHECK_RET
687}
688
689Eina_Bool
690read_psd_rgb(Image_Entry *ie, PSD_Header *head, const unsigned char *map, size_t length, size_t *position, int *error)
691{
692 unsigned int color_mode, resource_size, misc_info;
693 unsigned short compressed;
694 void *surface;
695
696#define CHECK_RET(Call) \
697 if (!Call) return EINA_FALSE;
698
699 CHECK_RET(read_uint(map, length, position, &color_mode));
700 // Skip over the 'color mode data section'
701 *position += color_mode;
702
703 // Read the 'image resources section'
704 CHECK_RET(read_uint(map, length, position, &resource_size));
705 *position += resource_size;
706
707 CHECK_RET(read_uint(map, length, position, &misc_info));
708 *position += misc_info;
709
710 CHECK_RET(read_ushort(map, length, position, &compressed));
711
712 head->channel_num = head->channels;
713
714 switch (head->depth)
715 {
716 case 8:
717 case 16:
718 break;
719 default:
720 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
721 return EINA_FALSE;
722 }
723 ie->w = head->width;
724 ie->h = head->height;
725 if (head->channels == 3) ie->flags.alpha = 0;
726 else ie->flags.alpha = 1;
727
728 evas_cache_image_surface_alloc(ie, ie->w, ie->h);
729 surface = evas_cache_image_pixels(ie);
730 if (!surface)
731 {
732 *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED;
733 goto cleanup_error;
734 }
735
736 if (!psd_get_data(ie, head, map, length, position, surface, compressed, error))
737 goto cleanup_error;
738
739 evas_common_image_premul(ie);
740 return EINA_TRUE;
741
742#undef CHECK_RET
743
744 cleanup_error:
745 return EINA_FALSE;
746}
747
748Eina_Bool
749read_psd_cmyk(Image_Entry *ie, PSD_Header *head, const unsigned char *map, size_t length, size_t *position, int *error)
750{
751 unsigned int color_mode, resource_size, misc_info, size, j, data_size;
752 unsigned short compressed;
753 unsigned int format, type;
754 unsigned char *kchannel = NULL;
755 void *surface;
756
757 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
758
759#define CHECK_RET(Call) \
760 if (!Call) return EINA_FALSE;
761
762 CHECK_RET(read_uint(map, length, position, &color_mode));
763 // Skip over the 'color mode data section'
764 *position += color_mode;
765
766 CHECK_RET(read_uint(map, length, position, &resource_size));
767 // Read the 'image resources section'
768 *position += resource_size;
769
770 CHECK_RET(read_uint(map, length, position, &misc_info));
771 *position += misc_info;
772
773 CHECK_RET(read_ushort(map, length, position, &compressed));
774
775 switch (head->channels)
776 {
777 case 4:
778 format = 0x1907;
779 head->channel_num = 4;
780 head->channels = 3;
781 break;
782 case 5:
783 format = 0x1908;
784 head->channel_num = 5;
785 head->channels = 4;
786 break;
787 default:
788 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
789 return EINA_FALSE;
790 }
791
792 switch (head->depth)
793 {
794 case 8:
795 type = 1;
796 break;
797 case 16:
798 type = 2;
799 break;
800 default:
801 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
802 return EINA_FALSE;
803 }
804
805 ie->w = head->width;
806 ie->h = head->height;
807 if (head->channels == 3) ie->flags.alpha = 0;
808 else ie->flags.alpha = 1;
809
810 evas_cache_image_surface_alloc(ie, ie->w, ie->h);
811 surface = evas_cache_image_pixels(ie);
812 if (!surface)
813 {
814 *error = EVAS_LOAD_ERROR_RESOURCE_ALLOCATION_FAILED;
815 goto cleanup_error;
816 }
817
818 if (!psd_get_data(ie, head, map, length, position, surface, compressed, error))
819 goto cleanup_error;
820
821 size = type * ie->w * ie->h;
822 kchannel = malloc(size);
823 if (kchannel == NULL)
824 goto cleanup_error;
825 if (!get_single_channel(ie, head, map, length, position, kchannel, compressed))
826 goto cleanup_error;
827
828 data_size = head->channels * type * ie->w * ie->h;
829 if (format == 0x1907)
830 {
831 unsigned char *tmp = surface;
832 const unsigned char *limit = tmp + data_size;
833
834 for (j = 0; tmp < limit; tmp++, j++)
835 {
836 int k;
837
838 for (k = 0; k < 3; k++)
839 *tmp = (*tmp * kchannel[j]) >> 8;
840
841 // FIXME: tmp[i+3] = 255;
842 }
843 }
844 else
845 { // RGBA
846 unsigned char *tmp = surface;
847 const unsigned char *limit = tmp + data_size;
848
849 // The KChannel array really holds the alpha channel on this one.
850 for (j = 0; tmp < limit; tmp += 4, j++)
851 {
852 tmp[0] = (tmp[0] * tmp[3]) >> 8;
853 tmp[1] = (tmp[1] * tmp[3]) >> 8;
854 tmp[2] = (tmp[2] * tmp[3]) >> 8;
855 tmp[3] = kchannel[j]; // Swap 'K' with alpha channel.
856 }
857 }
858
859 free(kchannel);
860
861 evas_common_image_premul(ie);
862 return EINA_TRUE;
863
864 cleanup_error:
865 free(kchannel);
866 return EINA_FALSE;
867}
868
869static Eina_Bool
870evas_image_load_file_data_psd(Image_Entry *ie,
871 const char *file,
872 const char *key __UNUSED__,
873 int *error)
874{
875 Eina_File *f;
876 void *map;
877 size_t length;
878 size_t position;
879 PSD_Header header;
880 Eina_Bool bpsd = EINA_FALSE;
881
882 f = eina_file_open(file, 0);
883 if (f == NULL)
884 {
885 *error = EVAS_LOAD_ERROR_DOES_NOT_EXIST;
886 return bpsd;
887 }
888
889 map = eina_file_map_all(f, EINA_FILE_SEQUENTIAL);
890 length = eina_file_size_get(f);
891 position = 0;
892 if (!map || length < 1)
893 {
894 eina_file_close(f);
895 *error = EVAS_LOAD_ERROR_CORRUPT_FILE;
896 return EINA_FALSE;
897 }
898
899 if (!psd_get_header(&header, map, length, &position) || !is_psd(&header))
900 {
901 eina_file_map_free(f, map);
902 eina_file_close(f);
903 *error = EVAS_LOAD_ERROR_GENERIC;
904 return EINA_FALSE;
905 }
906
907 ie->w = header.width;
908 ie->h = header.height;
909
910 *error = EVAS_LOAD_ERROR_NONE;
911
912 switch (header.mode)
913 {
914 case PSD_GREYSCALE: // Greyscale
915 bpsd = read_psd_grey(ie, &header, map, length, &position, error);
916 break;
917 case PSD_INDEXED: // Indexed
918 bpsd = read_psd_indexed(ie, &header, map, length, &position, error);
919 break;
920 case PSD_RGB: // RGB
921 bpsd = read_psd_rgb(ie, &header, map, length, &position, error);
922 break;
923 case PSD_CMYK: // CMYK
924 bpsd = read_psd_cmyk(ie, &header, map, length, &position, error);
925 break;
926 default :
927 *error = EVAS_LOAD_ERROR_UNKNOWN_FORMAT;
928 bpsd = EINA_FALSE;
929 }
930
931 eina_file_map_free(f, map);
932 eina_file_close(f);
933
934 return bpsd;
935}
936
937static Eina_Bool
938get_compressed_channels_length(PSD_Header *head,
939 const unsigned char *map, size_t length, size_t *position,
940 unsigned short *rle_table,
941 unsigned int *chanlen)
942{
943 unsigned int j;
944 unsigned int c;
945
946 if (!read_block(map, length, position, rle_table,
947 sizeof (unsigned short) * head->height * head->channel_num))
948 return EINA_FALSE;
949
950 memset(chanlen, 0, head->channel_num * sizeof(unsigned int));
951 for (c = 0; c < head->channel_num; c++)
952 {
953 unsigned int i;
954
955 j = c * head->height;
956 for (i = 0; i < head->height; i++)
957 {
958 chanlen[c] += rle_table[i + j];
959 }
960 }
961
962 return EINA_TRUE;
963}
964
965static const Evas_Image_Load_Func evas_image_load_psd_func = {
966 EINA_TRUE,
967 evas_image_load_file_head_psd,
968 evas_image_load_file_data_psd,
969 NULL,
970 EINA_FALSE
971};
972
973static int
974module_open(Evas_Module *em)
975{
976 if (!em) return 0;
977 em->functions = (void *)(&evas_image_load_psd_func);
978 return 1;
979}
980
981static void
982module_close(Evas_Module *em __UNUSED__)
983{
984}
985
986static Evas_Module_Api evas_modapi =
987 {
988 EVAS_MODULE_API_VERSION,
989 "psd",
990 "none",
991 {
992 module_open,
993 module_close
994 }
995 };
996
997EVAS_MODULE_DEFINE(EVAS_MODULE_TYPE_IMAGE_LOADER, image_loader, psd);
998
999
1000#ifndef EVAS_STATIC_BUILD_PSD
1001EVAS_EINA_MODULE_DEFINE(image_loader, psd);
1002#endif