--- ray/src/rt/ambcomp.c 1991/06/07 17:08:34 1.3 +++ ray/src/rt/ambcomp.c 1991/06/13 12:00:29 1.7 @@ -54,7 +54,6 @@ AMBSAMP *d1, *d2; } -double divsample(dp, h, r) /* sample a division */ register AMBSAMP *dp; AMBHEMI *h; @@ -68,7 +67,7 @@ RAY *r; register int i; if (rayorigin(&ar, r, AMBIENT, 0.5) < 0) - return(0.0); + return(-1); hlist[0] = r->rno; hlist[1] = dp->t; hlist[2] = dp->p; @@ -87,8 +86,8 @@ RAY *r; rayvalue(&ar); ndims--; addcolor(dp->v, ar.rcol); - if (ar.rot < FHUGE) - dp->r += 1.0/ar.rot; + if (ar.rt < FHUGE) + dp->r += 1.0/ar.rt; /* (re)initialize error */ if (dp->n++) { b2 = bright(dp->v)/dp->n - bright(ar.rcol); @@ -96,7 +95,7 @@ RAY *r; dp->k = b2/(dp->n*dp->n); } else dp->k = 0.0; - return(ar.rot); + return(0); } @@ -139,7 +138,7 @@ FVECT pg, dg; setcolor(dp->v, 0.0, 0.0, 0.0); dp->r = 0.0; dp->n = 0; - if ((d = divsample(dp, &hemi, r)) == 0.0) + if (divsample(dp, &hemi, r) < 0) goto oopsy; if (div != NULL) dp++; @@ -149,15 +148,13 @@ FVECT pg, dg; } } if (ns > 0) { /* perform super-sampling */ - comperrs(div, hemi); /* compute errors */ + comperrs(div, &hemi); /* compute errors */ qsort(div, ndivs, sizeof(AMBSAMP), ambcmp); /* sort divs */ /* super-sample */ for (i = ns; i > 0; i--) { copystruct(&dnew, div); - if ((d = divsample(&dnew, &hemi)) == 0.0) + if (divsample(&dnew, &hemi, r) < 0) goto oopsy; - if (d < FHUGE) - arad += 1.0 / d; /* reinsert */ dp = div; j = ndivs < i ? ndivs : i; @@ -182,10 +179,27 @@ FVECT pg, dg; } addcolor(acol, dp->v); } - if (pg != NULL) - posgradient(pg, div, &hemi); - if (dg != NULL) - dirgradient(dg, div, &hemi); + b = bright(acol); + if (b > FTINY) { + b = ndivs/b; + if (pg != NULL) { + posgradient(pg, div, &hemi); + for (i = 0; i < 3; i++) + pg[i] *= b; + } + if (dg != NULL) { + dirgradient(dg, div, &hemi); + for (i = 0; i < 3; i++) + dg[i] *= b; + } + } else { + if (pg != NULL) + for (i = 0; i < 3; i++) + pg[i] = 0.0; + if (dg != NULL) + for (i = 0; i < 3; i++) + dg[i] = 0.0; + } free((char *)div); } b = 1.0/ndivs; @@ -199,6 +213,14 @@ FVECT pg, dg; else if (arad < minarad) arad = minarad; arad /= sqrt(r->rweight); + if (pg != NULL) { /* clip pos. gradient if too large */ + d = 4.0*DOT(pg,pg)*arad*arad; + if (d > 1.0) { + d = 1.0/sqrt(d); + for (i = 0; i < 3; i++) + pg[i] *= d; + } + } return(arad); oopsy: if (div != NULL) @@ -241,6 +263,11 @@ register AMBHEMI *hp; dp = da; for (i = 0; i < hp->nt; i++) for (j = 0; j < hp->np; j++) { +#ifdef DEBUG + if (dp->t != i || dp->p != j) + error(CONSISTENCY, + "division order in comperrs"); +#endif b = bright(dp[0].v); if (i > 0) { /* from above */ b2 = bright(dp[-hp->np].v) - b; @@ -253,12 +280,11 @@ register AMBHEMI *hp; b2 *= b2 * 0.25; dp[0].k += b2; dp[-1].k += b2; - } - if (j == hp->np-1) { /* around */ - b2 = bright(dp[-(hp->np-1)].v) - b; + } else { /* around */ + b2 = bright(dp[hp->np-1].v) - b; b2 *= b2 * 0.25; dp[0].k += b2; - dp[-(hp->np-1)].k += b2; + dp[hp->np-1].k += b2; } dp++; } @@ -316,7 +342,7 @@ AMBHEMI *hp; dp += hp->np; } if (hp->nt > 1) { - mag0 /= (double)(hp->nt-1); + mag0 /= (double)hp->np; mag1 /= (double)hp->nt; } phi = 2.0*PI * (double)j/hp->np; @@ -325,7 +351,7 @@ AMBHEMI *hp; yd += mag0*sinp + mag1*cosp; } for (i = 0; i < 3; i++) - gv[i] = (xd*hp->ux[i] + yd*hp->uy[i])/hp->np; + gv[i] = (xd*hp->ux[i] + yd*hp->uy[i])/PI; } @@ -357,5 +383,5 @@ AMBHEMI *hp; yd += mag * sin(phi); } for (i = 0; i < 3; i++) - gv[i] = (xd*hp->ux[i] + yd*hp->uy[i])/(hp->nt*hp->np); + gv[i] = (xd*hp->ux[i] + yd*hp->uy[i])*PI/(hp->nt*hp->np); }