ViewVC Help
View File | Revision Log | Show Annotations | Download File | Root Listing
root/radiance/ray/src/rt/colortab.c
(Generate patch)

Comparing ray/src/rt/colortab.c (file contents):
Revision 1.3 by greg, Tue Oct 3 11:10:10 1989 UTC vs.
Revision 2.3 by greg, Mon Mar 8 12:37:21 1993 UTC

# Line 1 | Line 1
1 < /* Copyright (c) 1989 Regents of the University of California */
1 > /* Copyright (c) 1992 Regents of the University of California */
2  
3   #ifndef lint
4   static char SCCSid[] = "$SunId$ LBL";
# Line 16 | Line 16 | static char SCCSid[] = "$SunId$ LBL";
16   *      distribution is correlated to the last.
17   */
18  
19 < #include "color.h"
19 > #include "standard.h"
20  
21 < #define NULL            0
21 > #include "color.h"
22                                  /* histogram resolution */
23   #define NRED            24
24   #define NGRN            32
# Line 27 | Line 27 | static char SCCSid[] = "$SunId$ LBL";
27                                  /* minimum box count for adaptive partition */
28   #define MINSAMP         7
29                                  /* maximum distance^2 before color reassign */
30 < #define MAXDST2         5
31 <                                /* maximum frame buffer depth */
32 < #define FBDEPTH         8
33 <                                /* color map resolution */
34 < #define MAPSIZ          256
30 > #define MAXDST2         12
31                                  /* map a color */
32 < #define map_col(c,p)    clrmap[ colval(c,p)<1. ? \
33 <                                (int)(colval(c,p)*MAPSIZ) : MAPSIZ-1 ]
32 > #define map_col(c,p)    clrmap[p][ colval(c,p)<1. ? \
33 >                                (int)(colval(c,p)*256.) : 255 ]
34                                  /* color partition tree */
35   #define CNODE           short
36 < #define set_branch(p,c) ((c)<<2|(p))
36 > #define set_branch(p,c) ((c)<<2|(p))
37   #define set_pval(pv)    ((pv)<<2|3)
38 + #define is_branch(cn)   (((cn)&3)!=3)
39   #define is_pval(cn)     (((cn)&3)==3)
40   #define part(cn)        ((cn)>>2)
41   #define prim(cn)        ((cn)&3)
42   #define pval(cn)        ((cn)>>2)
43                                  /* our color table */
44 < struct tabent {
44 > static struct tabent {
45          long    sum[3];         /* sum of colors using this entry */
46 <        long    n;              /* number of colors */
47 <        short   ent[3];         /* current table value */
48 < }       clrtab[1<<FBDEPTH];
46 >        int     n;              /* number of colors */
47 >        BYTE    ent[3];         /* current table value */
48 > }       *clrtab = NULL;
49 >                                /* color cube partition */
50 > static CNODE    *ctree = NULL;
51                                  /* our color correction map */
52 < static BYTE     clrmap[MAPSIZ];
52 > static BYTE     clrmap[3][256];
53                                  /* histogram of colors used */
54 < static unsigned histo[NRED][NGRN][NBLU];
55 <                                /* initial color cube boundaries */
56 < static int      CLRCUBE[3][2] = {0,NRED,0,NGRN,0,NBLU};
58 <                                /* color cube partition */
59 < static CNODE    ctree[1<<(FBDEPTH+1)];
60 <                                /* callback for pixel assignment */
61 < static int      (*set_pixel)();
54 > static unsigned short   histo[NRED][NGRN][NBLU];
55 >                                /* initial color cube boundary */
56 > static int      CLRCUBE[3][2] = {{0,NRED},{0,NGRN},{0,NBLU}};
57  
58 + static int      split(), cut();
59  
60 +
61   int
62 < new_ctab(ncolors, cset)         /* start new color table with max ncolors */
62 > new_ctab(ncolors)               /* start new color table with max ncolors */
63   int     ncolors;
67 int     (*cset)();
64   {
65 <        if (ncolors < 1 || ncolors > 1<<FBDEPTH || cset == NULL)
65 >        int     treesize;
66 >
67 >        if (ncolors < 1)
68                  return(0);
69 <                                /* assign pixel callback routine */
70 <        set_pixel = cset;
71 <                                /* clear color table */
72 <        bzero(clrtab, sizeof(clrtab));
69 >                                /* free old tables */
70 >        if (clrtab != NULL)
71 >                free((char *)clrtab);
72 >        if (ctree != NULL)
73 >                free((char *)ctree);
74 >                                /* get new tables */
75 >        for (treesize = 1; treesize < ncolors; treesize <<= 1)
76 >                ;
77 >        treesize <<= 1;
78 >        clrtab = (struct tabent *)calloc(ncolors, sizeof(struct tabent));
79 >        ctree = (CNODE *)malloc(treesize*sizeof(CNODE));
80 >        if (clrtab == NULL || ctree == NULL)
81 >                return(0);
82                                  /* partition color space */
83 <        cut(ctree, FBDEPTH, CLRCUBE, 0, ncolors);
83 >        cut(ctree, 0, CLRCUBE, 0, ncolors);
84                                  /* clear histogram */
85 <        bzero(histo, sizeof(histo));
85 >        bzero((char *)histo, sizeof(histo));
86                                  /* return number of colors used */
87          return(ncolors);
88   }
89  
90  
91   int
92 < get_pixel(col)                  /* get pixel for color */
92 > get_pixel(col, set_pixel)       /* get pixel for color */
93   COLOR   col;
94 + int     (*set_pixel)();
95   {
96 +        extern char     errmsg[];
97          int     r, g, b;
98          int     cv[3];
99 <        register union { CNODE *t; struct tabent *e; }  p;
99 >        register CNODE  *tp;
100          register int    h;
101                                                  /* map color */
102          r = map_col(col,RED);
# Line 100 | Line 109 | COLOR  col;
109                                                  /* add to histogram */
110          histo[cv[RED]][cv[GRN]][cv[BLU]]++;
111                                                  /* find pixel in tree */
112 <        p.t = ctree;
113 <        for (h = FBDEPTH; h > 0; h--) {
114 <                if (is_pval(*p.t))
106 <                        break;
107 <                if (cv[prim(*p.t)] < part(*p.t))
108 <                        p.t++;          /* left branch */
112 >        for (tp = ctree, h = 0; is_branch(*tp); h++)
113 >                if (cv[prim(*tp)] < part(*tp))
114 >                        tp += 1<<h;             /* left branch */
115                  else
116 <                        p.t += 1<<h;    /* right branch */
117 <        }
112 <        h = pval(*p.t);
116 >                        tp += 1<<(h+1);         /* right branch */
117 >        h = pval(*tp);
118                                                  /* add to color table */
119 <        p.e = clrtab + h;
120 <                                        /* add to sum */
121 <        p.e->sum[RED] += r;
122 <        p.e->sum[GRN] += g;
118 <        p.e->sum[BLU] += b;
119 <        p.e->n++;
119 >        clrtab[h].sum[RED] += r;
120 >        clrtab[h].sum[GRN] += g;
121 >        clrtab[h].sum[BLU] += b;
122 >        clrtab[h].n++;
123                                          /* recompute average */
124 <        r = p.e->sum[RED] / p.e->n;
125 <        g = p.e->sum[GRN] / p.e->n;
126 <        b = p.e->sum[BLU] / p.e->n;
124 >        r = clrtab[h].sum[RED] / clrtab[h].n;
125 >        g = clrtab[h].sum[GRN] / clrtab[h].n;
126 >        b = clrtab[h].sum[BLU] / clrtab[h].n;
127                                          /* check for movement */
128 <        if (p.e->n == 1 ||
129 <                        (r-p.e->ent[RED])*(r-p.e->ent[RED]) +
130 <                        (g-p.e->ent[GRN])*(g-p.e->ent[GRN]) +
131 <                        (b-p.e->ent[BLU])*(b-p.e->ent[BLU]) > MAXDST2) {
132 <                p.e->ent[RED] = r;
133 <                p.e->ent[GRN] = g;      /* reassign pixel */
134 <                p.e->ent[BLU] = b;
135 < #ifdef notdef
136 <                printf("pixel %d = (%d,%d,%d) (%d refs)\n",
137 <                                h, r, g, b, p.e->n);
128 >        if (clrtab[h].n == 1 ||
129 >                        (r-clrtab[h].ent[RED])*(r-clrtab[h].ent[RED]) +
130 >                        (g-clrtab[h].ent[GRN])*(g-clrtab[h].ent[GRN]) +
131 >                        (b-clrtab[h].ent[BLU])*(b-clrtab[h].ent[BLU]) > MAXDST2) {
132 >                clrtab[h].ent[RED] = r;
133 >                clrtab[h].ent[GRN] = g; /* reassign pixel */
134 >                clrtab[h].ent[BLU] = b;
135 > #ifdef DEBUG
136 >                sprintf(errmsg, "pixel %d = (%d,%d,%d) (%d refs)\n",
137 >                                h, r, g, b, clrtab[h].n);
138 >                eputs(errmsg);
139   #endif
140                  (*set_pixel)(h, r, g, b);
141          }
# Line 139 | Line 143 | COLOR  col;
143   }
144  
145  
146 < make_cmap(gam)                  /* make gamma correction map */
147 < double  gam;
146 > make_gmap(gam)                  /* make gamma correction map */
147 > double  gam;
148   {
145        extern double   pow();
149          register int    i;
150          
151 <        for (i = 0; i < MAPSIZ; i++)
152 <                clrmap[i] = 256.0 * pow(i/(double)MAPSIZ, 1.0/gam);
151 >        for (i = 0; i < 256; i++)
152 >                clrmap[RED][i] =
153 >                clrmap[GRN][i] =
154 >                clrmap[BLU][i] = 256.0 * pow((i+0.5)/256.0, 1.0/gam);
155   }
156  
157  
158 + set_cmap(rmap, gmap, bmap)      /* set custom color correction map */
159 + BYTE    *rmap, *gmap, *bmap;
160 + {
161 +        bcopy((char *)rmap, (char *)clrmap[RED], 256);
162 +        bcopy((char *)gmap, (char *)clrmap[GRN], 256);
163 +        bcopy((char *)bmap, (char *)clrmap[BLU], 256);
164 + }
165 +
166 +
167 + map_color(rgb, col)             /* map a color to a byte triplet */
168 + BYTE    rgb[3];
169 + COLOR   col;
170 + {
171 +        rgb[RED] = map_col(col,RED);
172 +        rgb[GRN] = map_col(col,GRN);
173 +        rgb[BLU] = map_col(col,BLU);
174 + }
175 +
176 +
177   static
178 < cut(tree, height, box, c0, c1)          /* partition color space */
178 > cut(tree, level, box, c0, c1)           /* partition color space */
179   register CNODE  *tree;
180 < int     height;
180 > int     level;
181   register int    box[3][2];
182   int     c0, c1;
183   {
# Line 165 | Line 189 | int    c0, c1;
189          }
190                                          /* split box */
191          *tree = split(box);
192 <        bcopy(box, kb, sizeof(kb));
192 >        bcopy((char *)box, (char *)kb, sizeof(kb));
193                                                  /* do left (lesser) branch */
194          kb[prim(*tree)][1] = part(*tree);
195 <        cut(tree+1, height-1, kb, c0, (c0+c1)>>1);
195 >        cut(tree+(1<<level), level+1, kb, c0, (c0+c1)>>1);
196                                                  /* do right branch */
197          kb[prim(*tree)][0] = part(*tree);
198          kb[prim(*tree)][1] = box[prim(*tree)][1];
199 <        cut(tree+(1<<height), height-1, kb, (c0+c1)>>1, c1);
199 >        cut(tree+(1<<(level+1)), level+1, kb, (c0+c1)>>1, c1);
200   }
201  
202  

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines