]> git.imager.perl.org - imager.git/blob - raw.c
8b363d5a96b2d3c20a780fcccd97674c5c3c5c60
[imager.git] / raw.c
1 #include "imager.h"
2 #include <stdio.h>
3 #include "iolayer.h"
4 #ifndef _MSC_VER
5 #include <unistd.h>
6 #endif
7 #include <string.h>
8 #include <errno.h>
9
10
11
12 /*
13
14  Image loader for raw files.
15
16  This is a barebones raw loader...
17
18              fd: filedescriptor
19               x: xsize
20               y: ysize
21    datachannels: the number of channels the file contains
22   storechannels: the bitmap of channels we will read
23           intrl: interlace flag,
24                        0 = sample interleaving
25                        1 = line interleaving
26                        2 = image interleaving
27
28 */
29
30 static
31 void
32 interleave(unsigned char *inbuffer,unsigned char *outbuffer,int rowsize,int channels) {
33   int ch,ind,i;
34   i=0;
35   if (inbuffer == outbuffer) return; /* Check if data is already in interleaved format */
36   for (ind=0; ind<rowsize; ind++) 
37     for (ch=0; ch<channels; ch++) 
38       outbuffer[i++] = inbuffer[rowsize*ch+ind]; 
39 }
40
41 static
42 void
43 expandchannels(unsigned char *inbuffer, unsigned char *outbuffer, 
44                int chunks, int datachannels, int storechannels) {
45   int ch,i;
46   if (inbuffer == outbuffer) return; /* Check if data is already in expanded format */
47   for(ch=0; ch<chunks; ch++) 
48     for (i=0; i<storechannels; i++) 
49       outbuffer[ch*storechannels+i] = inbuffer[ch*datachannels+i];
50 }
51
52 i_img *
53 i_readraw_wiol(io_glue *ig, int x, int y, int datachannels, int storechannels, int intrl) {
54   i_img* im;
55   int rc,k;
56   
57   unsigned char *inbuffer;
58   unsigned char *ilbuffer;
59   unsigned char *exbuffer;
60   
61   int inbuflen,ilbuflen,exbuflen;
62
63   io_glue_commit_types(ig);
64   mm_log((1, "i_readraw(ig %p,x %d,y %d,datachannels %d,storechannels %d,intrl %d)\n",
65           ig, x, y, datachannels, storechannels, intrl));
66   
67   im = i_img_empty_ch(NULL,x,y,storechannels);
68   if (!im)
69     return NULL;
70   
71   inbuflen = im->xsize*datachannels;
72   ilbuflen = inbuflen;
73   exbuflen = im->xsize*storechannels;
74   inbuffer = (unsigned char*)mymalloc(inbuflen);
75   mm_log((1,"inbuflen: %d, ilbuflen: %d, exbuflen: %d.\n",inbuflen,ilbuflen,exbuflen));
76
77   if (intrl==0) ilbuffer = inbuffer; 
78   else ilbuffer=mymalloc(inbuflen);
79
80   if (datachannels==storechannels) exbuffer=ilbuffer; 
81   else exbuffer= mymalloc(exbuflen);
82   
83   k=0;
84   while( k<im->ysize ) {
85     rc = ig->readcb(ig, inbuffer, inbuflen);
86     if (rc != inbuflen) { fprintf(stderr,"Premature end of file.\n"); exit(2); }
87     interleave(inbuffer,ilbuffer,im->xsize,datachannels);
88     expandchannels(ilbuffer,exbuffer,im->xsize,datachannels,storechannels);
89     /* FIXME: Do we ever want to save to a virtual image? */
90     memcpy(&(im->idata[im->xsize*storechannels*k]),exbuffer,exbuflen);
91     k++;
92   }
93
94   myfree(inbuffer);
95   if (intrl != 0) myfree(ilbuffer);
96   if (datachannels != storechannels) myfree(exbuffer);
97
98   i_tags_add(&im->tags, "i_format", 0, "raw", -1, 0);
99
100   return im;
101 }
102
103
104
105 undef_int
106 i_writeraw_wiol(i_img* im, io_glue *ig) {
107   int rc;
108
109   io_glue_commit_types(ig);
110   i_clear_error();
111   mm_log((1,"writeraw(im %p,ig %p)\n", im, ig));
112   
113   if (im == NULL) { mm_log((1,"Image is empty\n")); return(0); }
114   if (!im->virtual) {
115     rc = ig->writecb(ig,im->idata,im->bytes);
116     if (rc != im->bytes) { 
117       i_push_error(errno, "Could not write to file");
118       mm_log((1,"i_writeraw: Couldn't write to file\n")); 
119       return(0);
120     }
121   } else {
122     if (im->type == i_direct_type) {
123       /* just save it as 8-bits, maybe support saving higher bit count
124          raw images later */
125       int line_size = im->xsize * im->channels;
126       unsigned char *data = mymalloc(line_size);
127
128       int y = 0;
129       rc = line_size;
130       while (rc == line_size && y < im->ysize) {
131         i_gsamp(im, 0, im->xsize, y, data, NULL, im->channels);
132         rc = ig->writecb(ig, data, line_size);
133         ++y;
134       }
135       if (rc != line_size) {
136         i_push_error(errno, "write error");
137         return 0;
138       }
139       myfree(data);
140     } else {
141       /* paletted image - assumes the caller puts the palette somewhere 
142          else
143       */
144       int line_size = sizeof(i_palidx) * im->xsize;
145       i_palidx *data = mymalloc(sizeof(i_palidx) * im->xsize);
146
147       int y = 0;
148       rc = line_size;
149       while (rc == line_size && y < im->ysize) {
150         i_gpal(im, 0, im->xsize, y, data);
151         rc = ig->writecb(ig, data, line_size);
152         ++y;
153       }
154       myfree(data);
155       if (rc != line_size) {
156         i_push_error(errno, "write error");
157         return 0;
158       }
159     }
160   }
161
162   ig->closecb(ig);
163
164   return(1);
165 }