| 5 |  | *  program to convert between RADIANCE and Windows BMP file | 
| 6 |  | */ | 
| 7 |  |  | 
| 8 | < | #include  <stdio.h> | 
| 9 | < | #include  <string.h> | 
| 8 | > | #include  <math.h> | 
| 9 |  |  | 
| 10 | + | #include  "rtio.h" | 
| 11 |  | #include  "platform.h" | 
| 12 |  | #include  "color.h" | 
| 13 |  | #include  "tonemap.h" | 
| 14 |  | #include  "resolu.h" | 
| 15 |  | #include  "bmpfile.h" | 
| 16 |  |  | 
| 17 | < | int  bradj = 0;                         /* brightness adjustment */ | 
| 17 | > | int             bradj = 0;              /* brightness adjustment */ | 
| 18 |  |  | 
| 19 | < | double  gamcor = 2.2;                   /* gamma correction value */ | 
| 19 | > | double          gamcor = 2.2;           /* gamma correction value */ | 
| 20 |  |  | 
| 21 | < | char  *progname; | 
| 21 | > | char            *progname; | 
| 22 |  |  | 
| 23 |  | static void quiterr(const char *err); | 
| 24 |  | static void tmap2bmp(char *fnin, char *fnout, char *expec, | 
| 25 |  | RGBPRIMP monpri, double gamval); | 
| 26 | < | static void rad2bmp(FILE *rfp, BMPWriter *bwr, int inv, int gry); | 
| 26 | > | static void rad2bmp(FILE *rfp, BMPWriter *bwr, int inv, RGBPRIMP monpri); | 
| 27 |  | static void bmp2rad(BMPReader *brd, FILE *rfp, int inv); | 
| 28 |  |  | 
| 29 | + | static RGBPRIMP rgbinp = stdprims;      /* RGB input primitives */ | 
| 30 | + | static RGBPRIMS myinprims;              /* custom primitives holder */ | 
| 31 |  |  | 
| 32 | + | static gethfunc headline; | 
| 33 | + |  | 
| 34 | + |  | 
| 35 |  | int | 
| 36 |  | main(int argc, char *argv[]) | 
| 37 |  | { | 
| 152 |  | exit(1); | 
| 153 |  | } | 
| 154 |  | /* get header info. */ | 
| 155 | < | if (checkheader(stdin, COLRFMT, NULL) < 0 || | 
| 155 | > | if (getheader(stdin, headline, NULL) < 0 || | 
| 156 |  | !fgetsresolu(&rs, stdin)) | 
| 157 |  | quiterr("bad Radiance picture format"); | 
| 158 |  | /* initialize BMP header */ | 
| 159 |  | if (rgbp == NULL) { | 
| 160 |  | hdr = BMPmappedHeader(scanlen(&rs), | 
| 161 |  | numscans(&rs), 0, 256); | 
| 162 | + | /* | 
| 163 |  | if (outfile != NULL) | 
| 164 |  | hdr->compr = BI_RLE8; | 
| 165 | + | */ | 
| 166 |  | } else | 
| 167 |  | hdr = BMPtruecolorHeader(scanlen(&rs), | 
| 168 |  | numscans(&rs), 0); | 
| 178 |  | if (wtr == NULL) | 
| 179 |  | quiterr("cannot allocate writer structure"); | 
| 180 |  | /* convert file */ | 
| 181 | < | rad2bmp(stdin, wtr, !hdr->yIsDown, rgbp==NULL); | 
| 181 | > | rad2bmp(stdin, wtr, !hdr->yIsDown, rgbp); | 
| 182 |  | /* flush output */ | 
| 183 |  | if (fflush((FILE *)wtr->c_data) < 0) | 
| 184 |  | quiterr("error writing BMP output"); | 
| 206 |  | exit(0); | 
| 207 |  | } | 
| 208 |  |  | 
| 209 | + | /* process header line (don't echo) */ | 
| 210 | + | static int | 
| 211 | + | headline(char *s, void *p) | 
| 212 | + | { | 
| 213 | + | char    fmt[MAXFMTLEN]; | 
| 214 | + |  | 
| 215 | + | if (formatval(fmt, s)) {        /* check if format string */ | 
| 216 | + | if (!strcmp(fmt,COLRFMT)) | 
| 217 | + | return(0); | 
| 218 | + | if (!strcmp(fmt,CIEFMT)) { | 
| 219 | + | rgbinp = TM_XYZPRIM; | 
| 220 | + | return(0); | 
| 221 | + | } | 
| 222 | + | return(-1); | 
| 223 | + | } | 
| 224 | + | if (isprims(s)) {               /* get input primaries */ | 
| 225 | + | primsval(myinprims, s); | 
| 226 | + | rgbinp = myinprims; | 
| 227 | + | return(0); | 
| 228 | + | } | 
| 229 | + | /* should I grok colcorr also? */ | 
| 230 | + | return(0); | 
| 231 | + | } | 
| 232 | + |  | 
| 233 | + |  | 
| 234 |  | /* convert Radiance picture to BMP */ | 
| 235 |  | static void | 
| 236 | < | rad2bmp(FILE *rfp, BMPWriter *bwr, int inv, int gry) | 
| 236 | > | rad2bmp(FILE *rfp, BMPWriter *bwr, int inv, RGBPRIMP monpri) | 
| 237 |  | { | 
| 238 | + | int     usexfm = 0; | 
| 239 | + | COLORMAT        xfm; | 
| 240 |  | COLR    *scanin; | 
| 241 | + | COLOR   cval; | 
| 242 |  | int     y, yend, ystp; | 
| 243 |  | int     x; | 
| 244 |  | /* allocate scanline */ | 
| 245 |  | scanin = (COLR *)malloc(bwr->hdr->width*sizeof(COLR)); | 
| 246 |  | if (scanin == NULL) | 
| 247 |  | quiterr("out of memory in rad2bmp"); | 
| 248 | + | /* set up color conversion */ | 
| 249 | + | usexfm = (monpri != NULL ? rgbinp != monpri : | 
| 250 | + | rgbinp != TM_XYZPRIM && rgbinp != stdprims); | 
| 251 | + | if (usexfm) { | 
| 252 | + | RGBPRIMP        destpri = monpri != NULL ? monpri : stdprims; | 
| 253 | + | double          expcomp = pow(2.0, (double)bradj); | 
| 254 | + | if (rgbinp == TM_XYZPRIM) | 
| 255 | + | compxyz2rgbWBmat(xfm, destpri); | 
| 256 | + | else | 
| 257 | + | comprgb2rgbWBmat(xfm, rgbinp, destpri); | 
| 258 | + | for (y = 0; y < 3; y++) | 
| 259 | + | for (x = 0; x < 3; x++) | 
| 260 | + | xfm[y][x] *= expcomp; | 
| 261 | + | } | 
| 262 |  | /* convert image */ | 
| 263 |  | if (inv) { | 
| 264 |  | y = bwr->hdr->height - 1; | 
| 271 |  | for ( ; y != yend; y += ystp) { | 
| 272 |  | if (freadcolrs(scanin, bwr->hdr->width, rfp) < 0) | 
| 273 |  | quiterr("error reading Radiance picture"); | 
| 274 | < | if (bradj) | 
| 274 | > | if (usexfm) | 
| 275 | > | for (x = bwr->hdr->width; x--; ) { | 
| 276 | > | colr_color(cval, scanin[x]); | 
| 277 | > | colortrans(cval, xfm, cval); | 
| 278 | > | setcolr(scanin[x], colval(cval,RED), | 
| 279 | > | colval(cval,GRN), | 
| 280 | > | colval(cval,BLU)); | 
| 281 | > | } | 
| 282 | > | else if (bradj) | 
| 283 |  | shiftcolrs(scanin, bwr->hdr->width, bradj); | 
| 284 | < | for (x = gry ? bwr->hdr->width : 0; x--; ) | 
| 285 | < | scanin[x][GRN] = normbright(scanin[x]); | 
| 284 | > | if (monpri == NULL && rgbinp != TM_XYZPRIM) | 
| 285 | > | for (x = bwr->hdr->width; x--; ) | 
| 286 | > | scanin[x][GRN] = normbright(scanin[x]); | 
| 287 |  | colrs_gambs(scanin, bwr->hdr->width); | 
| 288 | < | if (gry) | 
| 288 | > | if (monpri == NULL) | 
| 289 |  | for (x = bwr->hdr->width; x--; ) | 
| 290 |  | bwr->scanline[x] = scanin[x][GRN]; | 
| 291 |  | else | 
| 350 |  | int             tmflags; | 
| 351 |  | BMPHeader       *hdr; | 
| 352 |  | BMPWriter       *wtr; | 
| 295 | – | RESOLU          rs; | 
| 353 |  | FILE            *fp; | 
| 354 |  | int             xr, yr; | 
| 355 | < | BYTE            *pa; | 
| 355 | > | uby8            *pa; | 
| 356 |  | int             i; | 
| 357 |  | /* check tone-mapping spec */ | 
| 358 |  | i = strlen(expec); | 
| 375 |  | fprintf(stderr, "%s: cannot open\n", fnin); | 
| 376 |  | exit(1); | 
| 377 |  | } | 
| 321 | – | /* get picture orientation */ | 
| 322 | – | if (fnin != NULL) { | 
| 323 | – | if (getheader(fp, NULL, NULL) < 0 || !fgetsresolu(&rs, fp)) | 
| 324 | – | quiterr("bad Radiance picture format"); | 
| 325 | – | rewind(fp); | 
| 326 | – | } else                          /* assume stdin has normal orient */ | 
| 327 | – | rs.rt = PIXSTANDARD; | 
| 378 |  | /* tone-map picture */ | 
| 379 |  | if (tmMapPicture(&pa, &xr, &yr, tmflags, monpri, gamval, | 
| 380 |  | 0., 0., fnin, fp) != TM_E_OK) | 
| 397 |  | quiterr("cannot allocate writer structure"); | 
| 398 |  | /* write to BMP file */ | 
| 399 |  | while (wtr->yscan < yr) { | 
| 400 | < | BYTE    *scn = pa + xr*((tmflags & TM_F_BW) ? 1 : 3)* | 
| 400 | > | uby8    *scn = pa + xr*((tmflags & TM_F_BW) ? 1 : 3)* | 
| 401 |  | (yr-1 - wtr->yscan); | 
| 402 |  | if (tmflags & TM_F_BW) | 
| 403 |  | memcpy((void *)wtr->scanline, (void *)scn, xr); | 
| 414 |  | if (fflush((FILE *)wtr->c_data) < 0) | 
| 415 |  | quiterr("error writing BMP output"); | 
| 416 |  | /* clean up */ | 
| 417 | + | if (fnin != NULL) | 
| 418 | + | fclose(fp); | 
| 419 |  | free((void *)pa); | 
| 420 |  | BMPcloseOutput(wtr); | 
| 421 |  | } |