nuclear@0: /* nuclear@0: libimago - a multi-format image file input/output library. nuclear@0: Copyright (C) 2010 John Tsiombikas nuclear@0: nuclear@0: This program is free software: you can redistribute it and/or modify nuclear@0: it under the terms of the GNU Lesser General Public License as published nuclear@0: by the Free Software Foundation, either version 3 of the License, or nuclear@0: (at your option) any later version. nuclear@0: nuclear@0: This program is distributed in the hope that it will be useful, nuclear@0: but WITHOUT ANY WARRANTY; without even the implied warranty of nuclear@0: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the nuclear@0: GNU Lesser General Public License for more details. nuclear@0: nuclear@0: You should have received a copy of the GNU Lesser General Public License nuclear@0: along with this program. If not, see . nuclear@0: */ nuclear@0: #include nuclear@0: #include "imago2.h" nuclear@0: nuclear@0: /* pixel-format conversions are sub-optimal at the moment to avoid nuclear@0: * writing a lot of code. optimize at some point ? nuclear@0: */ nuclear@0: nuclear@0: #define CLAMP(x, a, b) ((x) < (a) ? (a) : ((x) > (b) ? (b) : (x))) nuclear@0: nuclear@0: struct pixel { nuclear@0: float r, g, b, a; nuclear@0: }; nuclear@0: nuclear@0: static void unpack_grey8(struct pixel *unp, void *pptr, int count); nuclear@0: static void unpack_rgb24(struct pixel *unp, void *pptr, int count); nuclear@0: static void unpack_rgba32(struct pixel *unp, void *pptr, int count); nuclear@0: static void unpack_greyf(struct pixel *unp, void *pptr, int count); nuclear@0: static void unpack_rgbf(struct pixel *unp, void *pptr, int count); nuclear@0: static void unpack_rgbaf(struct pixel *unp, void *pptr, int count); nuclear@0: nuclear@0: static void pack_grey8(void *pptr, struct pixel *unp, int count); nuclear@0: static void pack_rgb24(void *pptr, struct pixel *unp, int count); nuclear@0: static void pack_rgba32(void *pptr, struct pixel *unp, int count); nuclear@0: static void pack_greyf(void *pptr, struct pixel *unp, int count); nuclear@0: static void pack_rgbf(void *pptr, struct pixel *unp, int count); nuclear@0: static void pack_rgbaf(void *pptr, struct pixel *unp, int count); nuclear@0: nuclear@0: /* XXX keep in sync with enum img_fmt at imago2.h */ nuclear@0: static void (*unpack[])(struct pixel*, void*, int) = { nuclear@0: unpack_grey8, nuclear@0: unpack_rgb24, nuclear@0: unpack_rgba32, nuclear@0: unpack_greyf, nuclear@0: unpack_rgbf, nuclear@0: unpack_rgbaf nuclear@0: }; nuclear@0: nuclear@0: /* XXX keep in sync with enum img_fmt at imago2.h */ nuclear@0: static void (*pack[])(void*, struct pixel*, int) = { nuclear@0: pack_grey8, nuclear@0: pack_rgb24, nuclear@0: pack_rgba32, nuclear@0: pack_greyf, nuclear@0: pack_rgbf, nuclear@0: pack_rgbaf nuclear@0: }; nuclear@0: nuclear@0: nuclear@0: int img_convert(struct img_pixmap *img, enum img_fmt tofmt) nuclear@0: { nuclear@0: struct pixel pbuf[8]; nuclear@0: int bufsz = (img->width & 7) == 0 ? 8 : ((img->width & 3) == 0 ? 4 : 1); nuclear@0: int i, num_pix = img->width * img->height; nuclear@0: int num_iter = num_pix / bufsz; nuclear@0: char *sptr, *dptr; nuclear@0: struct img_pixmap nimg; nuclear@0: nuclear@0: if(img->fmt == tofmt) { nuclear@0: return 0; /* nothing to do */ nuclear@0: } nuclear@0: nuclear@0: img_init(&nimg); nuclear@0: if(img_set_pixels(&nimg, img->width, img->height, tofmt, 0) == -1) { nuclear@0: img_destroy(&nimg); nuclear@0: return -1; nuclear@0: } nuclear@0: nuclear@0: sptr = img->pixels; nuclear@0: dptr = nimg.pixels; nuclear@0: nuclear@0: for(i=0; ifmt](pbuf, sptr, bufsz); nuclear@0: pack[tofmt](dptr, pbuf, bufsz); nuclear@0: nuclear@0: sptr += bufsz * img->pixelsz; nuclear@0: dptr += bufsz * nimg.pixelsz; nuclear@0: } nuclear@0: nuclear@0: img_copy(img, &nimg); nuclear@0: img_destroy(&nimg); nuclear@0: return 0; nuclear@0: } nuclear@0: nuclear@0: /* the following functions *could* benefit from SIMD */ nuclear@0: nuclear@0: static void unpack_grey8(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = unp->g = unp->b = (float)*pix++ / 255.0; nuclear@0: unp->a = 1.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void unpack_rgb24(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = (float)*pix++ / 255.0; nuclear@0: unp->g = (float)*pix++ / 255.0; nuclear@0: unp->b = (float)*pix++ / 255.0; nuclear@0: unp->a = 1.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void unpack_rgba32(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = (float)*pix++ / 255.0; nuclear@0: unp->g = (float)*pix++ / 255.0; nuclear@0: unp->b = (float)*pix++ / 255.0; nuclear@0: unp->a = (float)*pix++ / 255.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void unpack_greyf(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: float *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = unp->g = unp->b = *pix++; nuclear@0: unp->a = 1.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void unpack_rgbf(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: float *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = *pix++; nuclear@0: unp->g = *pix++; nuclear@0: unp->b = *pix++; nuclear@0: unp->a = 1.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void unpack_rgbaf(struct pixel *unp, void *pptr, int count) nuclear@0: { nuclear@0: int i; nuclear@0: float *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir = *pix++; nuclear@0: unp->g = *pix++; nuclear@0: unp->b = *pix++; nuclear@0: unp->a = *pix++; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: nuclear@0: static void pack_grey8(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir + unp->g + unp->b) / 3.0); nuclear@0: *pix++ = CLAMP(lum, 0, 255); nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void pack_rgb24(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir * 255.0); nuclear@0: int g = (int)(unp->g * 255.0); nuclear@0: int b = (int)(unp->b * 255.0); nuclear@0: nuclear@0: *pix++ = CLAMP(r, 0, 255); nuclear@0: *pix++ = CLAMP(g, 0, 255); nuclear@0: *pix++ = CLAMP(b, 0, 255); nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void pack_rgba32(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: int i; nuclear@0: unsigned char *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir * 255.0); nuclear@0: int g = (int)(unp->g * 255.0); nuclear@0: int b = (int)(unp->b * 255.0); nuclear@0: int a = (int)(unp->a * 255.0); nuclear@0: nuclear@0: *pix++ = CLAMP(r, 0, 255); nuclear@0: *pix++ = CLAMP(g, 0, 255); nuclear@0: *pix++ = CLAMP(b, 0, 255); nuclear@0: *pix++ = CLAMP(a, 0, 255); nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void pack_greyf(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: int i; nuclear@0: float *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir + unp->g + unp->b) / 3.0; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void pack_rgbf(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: int i; nuclear@0: float *pix = pptr; nuclear@0: nuclear@0: for(i=0; ir; nuclear@0: *pix++ = unp->g; nuclear@0: *pix++ = unp->b; nuclear@0: unp++; nuclear@0: } nuclear@0: } nuclear@0: nuclear@0: static void pack_rgbaf(void *pptr, struct pixel *unp, int count) nuclear@0: { nuclear@0: memcpy(pptr, unp, count * sizeof *unp); nuclear@0: } nuclear@0: