This repository has been archived on 2023-10-09. You can view files and clone it, but cannot push or open issues or pull requests.
Files
blender-archive/source/blender/imbuf/intern/cmap.c
Kent Mein f8ef881474 I initalized mask to 0 in IMB_converttocmap
This may not be correct but at least now its predictable.

Kent
2002-12-19 21:26:34 +00:00

594 lines
13 KiB
C

/**
* cmap.c
*
* $Id$
*
* ***** BEGIN GPL/BL DUAL LICENSE BLOCK *****
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version. The Blender
* Foundation also sells licenses for use in proprietary software under
* the Blender License. See http://www.blender.org/BL/ for information
* about this.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
* All rights reserved.
*
* The Original Code is: all of this file.
*
* Contributor(s): none yet.
*
* ***** END GPL/BL DUAL LICENSE BLOCK *****
*/
#include <ctype.h>
#include "BLI_blenlib.h"
#include "imbuf.h"
#include "imbuf_patch.h"
#include "IMB_imbuf_types.h"
#include "IMB_imbuf.h"
#include "IMB_cmap.h"
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
static short *lastcube = 0;
static uchar *lastcoltab = 0;
static short lastmaxcol;
static short lastmincol;
static short lastcbits;
short alpha_col0 = FALSE;
/*
* er zit nog ergens een bug/inconsequentie in het programma. Als je een plaatje om wilt zetten
* naar een colormap met 1 bit resolutie krijg je een zwart plaatje. Zowieso alles met minder
* dan 4 bits is te donker.
*/
void IMB_freeImBufdata(void)
{
if (lastcube) free(lastcube);
lastcube= 0;
if (lastcoltab) free(lastcoltab);
lastcoltab= 0;
}
int IMB_alpha_to_col0(int new)
{
int old;
old = alpha_col0;
alpha_col0 = new;
return (old);
}
void imb_losecmapbits(struct ImBuf *ibuf, unsigned int *coltab)
{
int i,bits;
unsigned int col, and1, and2, *rect;
if (ibuf == 0) return;
if (ibuf->rect == 0) return;
if (ibuf->cbits == 0) return;
if (ibuf->cbits >= 8) return;
/*
bij cbits = 5:
and1 = 11100000;
bij cbits = 6:
and1 = 11000000;
*/
bits = ibuf->cbits;
and1 = ((1 << (8-bits)) - 1) & 0xff;
and1 |= (and1 << 24) + (and1 << 16) + (and1 << 8);
and2 = ~and1;
and1 <<= bits;
rect = ibuf->rect;
for (i = ibuf->x * ibuf->y ; i > 0; i--) {
col = rect[0];
*rect++ = col - ((col & and1) >> bits);
}
if (coltab){
for (i = 0 ; i < ibuf->maxcol ; i++) {
col = coltab[i];
coltab[i] = (col - ((col & and1) >> bits)) & and2;
}
}
}
static void addcmapbits(struct ImBuf *ibuf)
/* struct ImBuf *ibuf; */
{
int i,bits;
int div,mul;
uchar * cmap;
if (ibuf == 0) return;
if (ibuf->cmap == 0) return;
if (ibuf->cbits == 0) return;
if (ibuf->cbits >= 8) return;
bits = ibuf->cbits;
/* bits = 4 -> div = 0xf0
* bits = 5 -> div = 0xf8
*/
div = ((1 << bits) - 1) << (8 - bits);
mul = 0xffff / div;
if (ibuf->cmap){
cmap = (uchar *) ibuf->cmap;
for (i = 0 ; i < ibuf->maxcol ; i++){
cmap[1] = (mul * cmap[1]) >> 8;
cmap[2] = (mul * cmap[2]) >> 8;
cmap[3] = (mul * cmap[3]) >> 8;
cmap += 4;
}
}
}
static short addplanetocube(short *cube, short *plane, int minx, int miny, int sizep, int addcx, int addcy, int sizec, int col)
{
short done = FALSE;
int x, numx, numy, skipc, skipp, temp;
/* eerst maar eens clippen */
numx = numy = sizep;
temp = minx + sizep - 1;
if (temp > sizec) numx -= temp - sizec;
temp = miny + sizep - 1;
if (temp > sizec) numy -= temp - sizec;
if (minx < 0){
plane -= minx;
cube -= minx * addcx;
numx += minx;
}
if (miny < 0){
plane -= miny * sizep;
cube -= miny * addcy;
numy += miny;
}
skipc = addcy - (numx * addcx);
skipp = sizep - numx;
for (; numy > 0 ; numy--){
for (x = numx ; x > 0; x--) {
if (plane[0] < cube[1]) {
cube[0] = col;
cube[1] = plane[0];
done = TRUE;
}
plane ++;
cube += addcx;
}
plane += skipp;
cube += skipc;
}
return (done);
}
short *imb_coldeltatab(unsigned char *coltab, short mincol, short maxcol, short cbits)
{
short max, *quadr, *_quadr, *_cube, *cube, *_plane, done, nocol;
unsigned int addcb, addcg, addcr, sizep;
uchar *_colp, *colp, *col;
int i, j, k, addcube;
int r, g, b;
max = (1 << cbits) - 1;
nocol = maxcol - mincol;
coltab += 4 * mincol;
/* kleuren terugbrengen tot juiste hoeveelheid bits */
{
unsigned int * lctab, and;
lctab = (unsigned int *) coltab;
and = max << (8 - cbits);
and = and + (and << 8) + (and << 16) + (and << 24);
for (i=nocol-1 ; i >= 0 ; i--) lctab[i] = (lctab[i] & and) >> (8 - cbits);
}
/* zijn deze gegevens hetzelfde als de vorige ? */
if (lastcube){
if (mincol == lastmincol && maxcol == lastmaxcol && cbits == lastcbits){
if (lastcoltab){
if (memcmp(lastcoltab, coltab, 4 * nocol) == 0) return(lastcube);
}
}
}
if (lastcube) free(lastcube);
if (lastcoltab) free(lastcoltab);
lastcube = 0;
lastcoltab = 0;
_cube = malloc(2 * (1 << (3 * cbits)) * sizeof(short));
_plane = malloc((2 * max + 1) * (2 * max + 1) * sizeof(short));
_quadr = malloc((2 * max + 1) * sizeof(short));
_colp = malloc(6 * nocol);
if (_cube == 0 || _plane == 0 || _quadr == 0 || _colp == 0){
if (_cube) free(_cube);
if (_plane) free(_plane);
if (_quadr) free(_quadr);
if (_colp) free(_colp);
return(0);
}
lastcoltab = malloc(4 * nocol);
if (lastcoltab) memcpy(lastcoltab, coltab, 4 * nocol);
lastcube = _cube;
lastmincol = mincol;
lastmaxcol = maxcol;
lastcbits = cbits;
/* cube initialiseren */
cube = _cube;
for (i = (1 << (3 * cbits)); i > 0 ; i--){
cube[0] = 0;
cube[1] = 32767;
cube += 2;
}
/* error look up table aan maken */
{
unsigned int delta;
quadr = _quadr + max + 1;
quadr[0] = 0;
delta = 3;
for (i = 1 ; i <= max ; i++){
quadr[i] = quadr[-i] = delta;
delta += i + 3;
}
}
/* colorplane initialiseren */
for (i = 6 * nocol - 1; i >= 0; i--) _colp[i] = 1;
addcr = 2;
addcg = (addcr << cbits);
addcb = (addcg << cbits);
/* eerste ronde invullen */
{
unsigned int ofs;
col = coltab;
cube = _cube;
for (i = 0 ; i < nocol ; i++){
ofs = (col[3] * addcr) + (col[2] * addcg) + (col[1] * addcb);
/* is deze kleur al ingevuld -> dan overslaan */
if (cube[ofs + 1]) cube[ofs] = i + mincol;
cube[ofs + 1] = 0;
col += 4;
}
}
for (i = 1; i <= max ; i++){
colp = _colp;
col = coltab;
done = FALSE;
sizep = 2*i +1;
/* plane initialiseren */
{
unsigned int delta;
short *plane;
plane = _plane;
for (j = -i ; j <= i; j++){
delta = quadr[i] + quadr[j];
for (k = -i; k <= i; k++){
*plane++ = delta + quadr[k];
}
}
}
for (j = mincol; j < maxcol; j++){
b = col[1] - i;
g = col[2] - i;
r = col[3] - i;
addcube= (addcr * r) + (addcg * g) + (addcb * b);
/* PRINT4(d, d, d, d, addcube, r, g, b); */
/* if(addcube >= 2 * (1 << (3 * cbits))) { */
/* printf("maxerror: %d %d\n", addcube, 2 * (1 << (3 * cbits))); */
/* add_cube= 2 * (1 << (3 * cbits)) -1; */
/* } */
cube = _cube + addcube;
if (colp[0]){
if (b < 0) colp[0] = 0;
else done |= colp[0] = addplanetocube(cube, _plane, r, g, sizep, addcr, addcg, max, j);
}
if (colp[1]){
if (g < 0) colp[1] = 0;
else done |= colp[1] = addplanetocube(cube, _plane, r, b, sizep, addcr, addcb, max, j);
}
if (colp[2]){
if (r < 0) colp[2] = 0;
else done |= colp[2] = addplanetocube(cube, _plane, b, g, sizep, addcb, addcg, max, j);
}
if (colp[3]){
if ((b + sizep - 1) > max) colp[3] = 0;
else done |= colp[3] = addplanetocube(cube + (sizep -1) * addcb, _plane, r, g, sizep, addcr,
addcg, max, j);
}
if (colp[4]){
if ((g + sizep - 1) > max) colp[4] = 0;
else done |= colp[4] = addplanetocube(cube + (sizep -1) * addcg, _plane, r, b, sizep, addcr,
addcb, max, j);
}
if (colp[5]){
if ((r + sizep - 1) > max) colp[5] = 0;
else done |= colp[5] = addplanetocube(cube + (sizep -1) * addcr, _plane, b, g, sizep, addcb,
addcg, max, j);
}
colp += 6;
col += 4;
}
if (done == 0) break;
}
free(_quadr);
free(_plane);
free(_colp);
return(_cube);
}
static void convcmap(struct ImBuf* ibuf, short *deltab, short cbits)
/* struct ImBuf* ibuf; */
/* short *deltab,cbits; */
{
unsigned int *rect;
short x,y;
unsigned int col;
unsigned int bbits,gbits,rbits;
unsigned int bmask,gmask,rmask;
bbits = 24 - 3 * cbits - 1;
gbits = 16 - 2 * cbits - 1;
rbits = 8 - cbits - 1;
rmask = ((1 << cbits) - 1) << (8 - cbits);
gmask = rmask << 8;
bmask = gmask << 8;
rect =(unsigned int *)ibuf->rect;
for(y=ibuf->y;y>0;y--){
for(x=ibuf->x;x>0;x--){
col = *rect;
col = ((col & bmask) >> bbits) + ((col & gmask) >> gbits) + ((col & rmask) >> rbits);
*rect++ = deltab[col];
}
}
}
short IMB_converttocmap(struct ImBuf *ibuf)
{
unsigned int *coltab;
short *deltab=0, cbits;
int i;
int mincol, mask = 0;
struct ImBuf * abuf = 0;
unsigned int * rect, * arect;
cbits = 5;
if (ibuf->cmap == 0) return(0);
if ((ibuf->cbits > 0) && (ibuf->cbits <8)) cbits = ibuf->cbits;
coltab = calloc(ibuf->maxcol, sizeof(unsigned int));
if (coltab == 0) return(0);
memcpy(coltab, ibuf->cmap, ibuf->maxcol * sizeof(unsigned int));
mincol = ibuf->mincol;
if (alpha_col0) {
if (mincol == 0) mincol = 1;
abuf = IMB_dupImBuf(ibuf);
}
imb_losecmapbits(ibuf, coltab);
deltab = imb_coldeltatab((uchar *) coltab, mincol ,ibuf->maxcol, cbits);
if (deltab == 0) {
free(coltab);
if (abuf) IMB_freeImBuf(abuf);
return(0);
}
IMB_dit0(ibuf,1,cbits);
IMB_dit0(ibuf,2,cbits);
IMB_dit0(ibuf,3,cbits);
convcmap(ibuf, deltab, cbits);
if (abuf) {
/* alpha omzetten naar kleur 0 */
rect = ibuf->rect;
arect = abuf->rect;
if (alpha_col0 == 1) mask = 0xff000000; /* alpha == 0 -> 0 */
if (alpha_col0 == 2) mask = 0x80000000; /* alpha < 128 -> 0 */
for (i = ibuf->x * ibuf->y; i > 0; i--) {
if ((*arect++ & mask) == 0) rect[0] = 0;
rect++;
}
IMB_freeImBuf(abuf);
}
free(coltab);
return (TRUE);
}
void imb_makecolarray(struct ImBuf *ibuf, unsigned char *mem, short nocols)
/* struct ImBuf *ibuf; */
/* uchar *mem; */
/* short nocols; */
{
short i,bits = 0;
uchar *cmap;
/* wat is hier de theorie achter */
nocols = ibuf->maxcol;
if (ibuf->cmap){
cmap = (uchar *) ibuf->cmap;
for (i = 0; i < nocols; i++){
cmap[3] = mem[0];
cmap[2] = mem[1];
cmap[1] = mem[2];
cmap[0] = 0;
bits |= mem[0] | mem[1] | mem[2];
mem += 3;
cmap += 4;
}
/* patch voor AdPro II */
if (IS_ham(ibuf)){
i = ibuf->depth - 2;
bits = ((1 << i) - 1) << (8 - i);
for (i=0 ; i<nocols ; i++) ibuf->cmap[i] &= (bits << 24) + (bits << 16) + (bits << 8) + bits;
}
if ((bits & 0x1f) == 0){
ibuf->cbits = 3;
} else if ((bits & 0x0f) == 0){
ibuf->cbits = 4;
} else if ((bits & 0x07) == 0){
ibuf->cbits = 5;
} else if ((bits & 0x03) == 0){
ibuf->cbits = 6;
} else ibuf->cbits = 8;
addcmapbits(ibuf);
if (IS_hbrite(ibuf)){
for (i=31;i>=0;i--){
ibuf->cmap[i+32] = (ibuf->cmap[i] & 0xfefefefe) >> 1;
}
}
if (IS_amiga(ibuf)){
cmap = (uchar * ) (ibuf->cmap + 1);
for (i = 1; i < nocols; i++){
cmap[0] = 0xff;
cmap += 4;
}
}
}
}
/* temporal... rects now are rgba, cmaps are abgr */
#define SWITCH_INT(a) {char s_i, *p_i; p_i= (char *)&(a); s_i= p_i[0]; p_i[0]= p_i[3]; p_i[3]= s_i; s_i= p_i[1]; p_i[1]= p_i[2]; p_i[2]= s_i; }
void IMB_applycmap(struct ImBuf *ibuf)
/* struct ImBuf *ibuf; */
{
unsigned int *rect, *cmap;
int x, y, i, col, code;
int *mask = 0;
if (ibuf == 0) return;
if (ibuf->rect == 0 || ibuf->cmap == 0) return;
rect = ibuf->rect;
cmap = ibuf->cmap;
if (IS_ham(ibuf)){
/* masker genereren maximaal (8 + 2) bits */
mask = malloc(1024 * 2 * sizeof(int));
x = 1 << (ibuf->depth - 2);
y = 65535 / (x - 1);
for (i = 0; i < x; i++){
mask[i] = 0;
mask[i + x] = 0x00ffff;
mask[i + x + x] = 0xffff00;
mask[i + x + x + x] = 0xff00ff;
col = (y * i) >> 8;
mask[i + 1024] = 0xff000000 | ibuf->cmap[i];
mask[i + x + 1024] = 0xff000000 | col << 16;
mask[i + x + x + 1024] = 0xff000000 | col;
mask[i + x + x + x + 1024] = 0xff000000 | col << 8;
}
/* alleen kleur 0 transparant */
mask[0+1024] =ibuf->cmap[0];
for (y = ibuf->y ; y>0 ; y--){
col = cmap[0];
for (x=ibuf->x ; x>0 ; x--){
code = *rect;
*rect++ = col = (col & mask[code]) | mask[code + 1024];
}
}
free(mask);
} else {
for(i = ibuf->x * ibuf->y; i>0; i--){
col = *rect;
if (col >= 0 && col < ibuf->maxcol) *rect = cmap[col];
rect++;
/* *(rect++) = cmap[*rect]; */
}
}
}