--- ray/src/rt/rv2.c 1990/01/11 09:03:39 1.15 +++ ray/src/rt/rv2.c 1991/10/29 12:06:44 1.32 @@ -1,4 +1,4 @@ -/* Copyright (c) 1987 Regents of the University of California */ +/* Copyright (c) 1991 Regents of the University of California */ #ifndef lint static char SCCSid[] = "$SunId$ LBL"; @@ -22,7 +22,9 @@ static char SCCSid[] = "$SunId$ LBL"; #define CTRL(c) ('c'-'@') +extern char VersionID[]; extern char *progname; +extern char *octname; getframe(s) /* get a new frame */ @@ -63,13 +65,14 @@ char *s; } fputs(progname, fp); fprintview(&ourview, fp); + fputs(sskip(s), fp); fputs("\n", fp); fclose(fp); return; } sprintf(buf, "view type (%c): ", ourview.type); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (buf[0] && buf[0] != ourview.type) { nv.type = buf[0]; @@ -79,7 +82,7 @@ char *s; sprintf(buf, "view point (%.6g %.6g %.6g): ", ourview.vp[0], ourview.vp[1], ourview.vp[2]); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (sscanf(buf, "%lf %lf %lf", &nv.vp[0], &nv.vp[1], &nv.vp[2]) == 3) change++; @@ -88,7 +91,7 @@ char *s; sprintf(buf, "view direction (%.6g %.6g %.6g): ", ourview.vdir[0], ourview.vdir[1], ourview.vdir[2]); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (sscanf(buf,"%lf %lf %lf",&nv.vdir[0],&nv.vdir[1],&nv.vdir[2]) == 3) change++; @@ -97,7 +100,7 @@ char *s; sprintf(buf, "view up (%.6g %.6g %.6g): ", ourview.vup[0], ourview.vup[1], ourview.vup[2]); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (sscanf(buf,"%lf %lf %lf",&nv.vup[0],&nv.vup[1],&nv.vup[2]) == 3) change++; @@ -106,7 +109,7 @@ char *s; sprintf(buf, "view horiz and vert size (%.6g %.6g): ", ourview.horiz, ourview.vert); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (sscanf(buf, "%lf %lf", &nv.horiz, &nv.vert) == 2) change++; @@ -116,7 +119,7 @@ char *s; sprintf(buf, "view shift and lift (%.6g %.6g): ", ourview.hoff, ourview.voff); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (buf[0] == CTRL(C)) return; if (sscanf(buf, "%lf %lf", &nv.hoff, &nv.voff) == 2) change++; @@ -137,9 +140,9 @@ char *s; VIEW nv; if (sscanf(s, "%s", buf) == 1) { /* get parameters from a file */ - copyview(&nv, &stdview); + copystruct(&nv, &stdview); if ((fname = getpath(buf, NULL, 0)) == NULL || - (success = viewfile(fname, &nv)) == -1) { + (success = viewfile(fname, &nv, 0, 0)) == -1) { sprintf(errmsg, "cannot open \"%s\"", buf); error(COMMAND, errmsg); return; @@ -150,13 +153,13 @@ char *s; newview(&nv); return; } - if (oldview.horiz == 0) { /* no old view! */ + if (oldview.type == 0) { /* no old view! */ error(COMMAND, "no previous view"); return; } - copyview(&nv, &ourview); - copyview(&ourview, &oldview); - copyview(&oldview, &nv); + copystruct(&nv, &ourview); + copystruct(&ourview, &oldview); + copystruct(&oldview, &nv); newimage(); } @@ -170,18 +173,12 @@ char *s; if (getinterest(s, 1, nv.vdir, &zfact) < 0) return; + nv.type = ourview.type; VCOPY(nv.vp, ourview.vp); VCOPY(nv.vup, ourview.vup); nv.hoff = ourview.hoff; nv.voff = ourview.voff; - if ((nv.type = ourview.type) == VT_PAR) { - nv.horiz = ourview.horiz / zfact; - nv.vert = ourview.vert / zfact; - } else { - nv.horiz = atan(tan(ourview.horiz*(PI/180./2.))/zfact) / - (PI/180./2.); - nv.vert = atan(tan(ourview.vert*(PI/180./2.))/zfact) / - (PI/180./2.); - } + nv.horiz = ourview.horiz; nv.vert = ourview.vert; + zoomview(&nv, zfact); newview(&nv); } @@ -211,6 +208,7 @@ char *s; error(COMMAND, "missing angle"); return; } + nv.type = ourview.type; VCOPY(nv.vp, ourview.vp); VCOPY(nv.vup, ourview.vup); nv.hoff = ourview.hoff; nv.voff = ourview.voff; @@ -220,15 +218,8 @@ char *s; normalize(v1); spinvector(nv.vdir, nv.vdir, v1, elev*(PI/180.)); } - if ((nv.type = ourview.type) == VT_PAR) { - nv.horiz = ourview.horiz / zfact; - nv.vert = ourview.vert / zfact; - } else { - nv.horiz = atan(tan(ourview.horiz*(PI/180./2.))/zfact) / - (PI/180./2.); - nv.vert = atan(tan(ourview.vert*(PI/180./2.))/zfact) / - (PI/180./2.); - } + nv.horiz = ourview.horiz; nv.vert = ourview.vert; + zoomview(&nv, zfact); newview(&nv); } @@ -282,7 +273,7 @@ char *s; if (*cp == '\0') { /* interactive */ sprintf(buf, "exposure (%lf): ", exposure); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); for (cp = buf; isspace(*cp); cp++) ; if (*cp == '\0') @@ -312,6 +303,69 @@ char *s; } +getparam(str, dsc, typ, ptr) /* get variable from user */ +char *str, *dsc; +int typ; +register union {int i; double d; COLOR C;} *ptr; +{ + extern char *index(); + int i0; + double d0, d1, d2; + char buf[48]; + + switch (typ) { + case 'i': /* integer */ + if (sscanf(str, "%d", &i0) != 1) { + (*dev->comout)(dsc); + sprintf(buf, " (%d): ", ptr->i); + (*dev->comout)(buf); + (*dev->comin)(buf, NULL); + if (sscanf(buf, "%d", &i0) != 1) + return(0); + } + ptr->i = i0; + return(1); + case 'r': /* real */ + if (sscanf(str, "%lf", &d0) != 1) { + (*dev->comout)(dsc); + sprintf(buf, " (%.6g): ", ptr->d); + (*dev->comout)(buf); + (*dev->comin)(buf, NULL); + if (sscanf(buf, "%lf", &d0) != 1) + return(0); + } + ptr->d = d0; + return(1); + case 'b': /* boolean */ + if (sscanf(str, "%1s", buf) != 1) { + (*dev->comout)(dsc); + sprintf(buf, " (%c): ", ptr->i ? 'y' : 'n'); + (*dev->comout)(buf); + (*dev->comin)(buf, NULL); + if (buf[0] == '\0' || + index("yY+1tTnN-0fF", buf[0]) == NULL) + return(0); + } + ptr->i = index("yY+1tT", buf[0]) != NULL; + return(1); + case 'C': /* color */ + if (sscanf(str, "%lf %lf %lf", &d0, &d1, &d2) != 3) { + (*dev->comout)(dsc); + sprintf(buf, " (%.6g %.6g %.6g): ", + colval(ptr->C,RED), + colval(ptr->C,GRN), + colval(ptr->C,BLU)); + (*dev->comout)(buf); + (*dev->comin)(buf, NULL); + if (sscanf(buf, "%lf %lf %lf", &d0, &d1, &d2) != 3) + return(0); + } + setcolor(ptr->C, d0, d1, d2); + return(1); + } +} + + setparam(s) /* get/set program parameter */ register char *s; { @@ -329,39 +383,25 @@ register char *s; extern int ambdiv; extern int ambssamp; extern int ambounce; - int i0; - double d0, d1, d2; + extern int directinvis; + extern double srcsizerat; + extern int do_irrad; char buf[128]; if (s[0] == '\0') { - (*dev->comout)("aa ab ad ar as av dc dj dt lr lw sp st: "); - (*dev->comin)(buf); + (*dev->comout)( + "aa ab ad ar as av b dc di dj ds dt i lr lw sp st: "); + (*dev->comin)(buf, NULL); s = buf; } switch (s[0]) { case 'l': /* limit */ switch (s[1]) { case 'w': /* weight */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "limit weight (%.6g): ", - minweight); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - minweight = d0; + getparam(s+2, "limit weight", 'r', &minweight); break; case 'r': /* reflection */ - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "limit reflection (%d): ", - maxdepth); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - maxdepth = i0; + getparam(s+2, "limit reflection", 'i', &maxdepth); break; default: goto badparam; @@ -370,114 +410,52 @@ register char *s; case 'd': /* direct */ switch (s[1]) { case 'j': /* jitter */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "direct jitter (%.6g): ", - dstrsrc); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - dstrsrc = d0; + getparam(s+2, "direct jitter", 'r', &dstrsrc); break; case 'c': /* certainty */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "direct certainty (%.6g): ", - shadcert); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - shadcert = d0; + getparam(s+2, "direct certainty", 'r', &shadcert); break; case 't': /* threshold */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "direct threshold (%.6g): ", - shadthresh); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - shadthresh = d0; + getparam(s+2, "direct threshold", 'r', &shadthresh); break; + case 'i': /* invisibility */ + getparam(s+2, "direct invisibility", + 'b', &directinvis); + break; + case 's': /* sampling */ + getparam(s+2, "direct sampling", 'r', &srcsizerat); + break; default: goto badparam; } break; + case 'b': /* black and white */ + getparam(s+1, "black and white", 'b', &greyscale); + break; + case 'i': /* irradiance */ + getparam(s+1, "irradiance", 'b', &do_irrad); + break; case 'a': /* ambient */ switch (s[1]) { case 'v': /* value */ - if (sscanf(s+2, "%lf %lf %lf", &d0, &d1, &d2) != 3) { - sprintf(buf, - "ambient value (%.6g %.6g %.6g): ", - colval(ambval,RED), - colval(ambval,GRN), - colval(ambval,BLU)); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf %lf %lf", - &d0, &d1, &d2) != 3) - break; - } - setcolor(ambval, d0, d1, d2); + getparam(s+2, "ambient value", 'C', ambval); break; case 'a': /* accuracy */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "ambient accuracy (%.6g): ", - ambacc); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - ambacc = d0; + getparam(s+2, "ambient accuracy", 'r', &ambacc); break; case 'd': /* divisions */ - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "ambient divisions (%d): ", - ambdiv); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - ambdiv = i0; + getparam(s+2, "ambient divisions", 'i', &ambdiv); break; case 's': /* samples */ - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "ambient super-samples (%d): ", - ambssamp); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - ambssamp = i0; + getparam(s+2, "ambient super-samples", 'i', &ambssamp); break; case 'b': /* bounces */ - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "ambient bounces (%d): ", - ambounce); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - ambounce = i0; + getparam(s+2, "ambient bounces", 'i', &ambounce); break; case 'r': - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "ambient resolution (%d): ", - ambres); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - ambres = i0; - minarad = ambres > 0 ? thescene.cusize/ambres : 0.0; + if (getparam(s+2, "ambient resolution", 'i', &ambres)) + minarad = ambres > 0 ? + thescene.cusize/ambres : 0.0; break; default: goto badparam; @@ -486,27 +464,12 @@ register char *s; case 's': /* sample */ switch (s[1]) { case 'p': /* pixel */ - if (sscanf(s+2, "%d", &i0) != 1) { - sprintf(buf, "sample pixel (%d): ", psample); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%d", &i0) != 1) - break; - } - psample = i0; - pdepth = 0; + if (getparam(s+2, "sample pixel", 'i', &psample)) + pdepth = 0; break; case 't': /* threshold */ - if (sscanf(s+2, "%lf", &d0) != 1) { - sprintf(buf, "sample threshold (%.6g): ", - maxdiff); - (*dev->comout)(buf); - (*dev->comin)(buf); - if (sscanf(buf, "%lf", &d0) != 1) - break; - } - maxdiff = d0; - pdepth = 0; + if (getparam(s+2, "sample threshold", 'r', &maxdiff)) + pdepth = 0; break; default: goto badparam; @@ -516,6 +479,7 @@ register char *s; break; default:; badparam: + *sskip(s) = '\0'; sprintf(errmsg, "%s: unknown variable", s); error(COMMAND, errmsg); break; @@ -540,9 +504,12 @@ char *s; if ((*dev->getcur)(&x, &y) == ABORT) return; - viewray(thisray.rorg, thisray.rdir, &ourview, - (x+.5)/hresolu, (y+.5)/vresolu); - + if (viewray(thisray.rorg, thisray.rdir, &ourview, + (x+.5)/hresolu, (y+.5)/vresolu) < 0) { + error(COMMAND, "not on image"); + return; + } + } else if (normalize(thisray.rdir) == 0.0) { error(COMMAND, "zero ray direction"); return; @@ -560,7 +527,7 @@ char *s; ofun[thisray.ro->otype].funame, thisray.ro->oname); (*dev->comout)(buf); - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); if (thisray.rot >= FHUGE) (*dev->comout)("at infinity"); else { @@ -568,14 +535,15 @@ char *s; thisray.rop[1], thisray.rop[2]); (*dev->comout)(buf); } - (*dev->comin)(buf); - sprintf(buf, "with value (%.6g %.6g %.6g)", + (*dev->comin)(buf, NULL); + sprintf(buf, "value (%.5g %.5g %.5g) (%.1fL)", colval(thisray.rcol,RED), colval(thisray.rcol,GRN), - colval(thisray.rcol,BLU)); + colval(thisray.rcol,BLU), + luminance(thisray.rcol)); (*dev->comout)(buf); } - (*dev->comin)(buf); + (*dev->comin)(buf, NULL); } @@ -604,23 +572,32 @@ char *s; /* write header */ fputs(progname, fp); fprintview(&ourview, fp); - putc('\n', fp); + if (octname != NULL) + fprintf(fp, " %s\n", octname); + else + putc('\n', fp); + fprintf(fp, "SOFTWARE= %s\n", VersionID); if (exposure != 1.0) fputexpos(exposure, fp); if (dev->pixaspect != 1.0) fputaspect(dev->pixaspect, fp); + fputformat(COLRFMT, fp); putc('\n', fp); fputresolu(YMAJOR|YDECR, hresolu, vresolu, fp); scanline = (COLR *)malloc(hresolu*sizeof(COLR)); - if (scanline == NULL) - error(SYSTEM, "out of memory in writepict"); + if (scanline == NULL) { + error(COMMAND, "not enough memory!"); + fclose(fp); + unlink(fname); + return; + } for (y = vresolu-1; y >= 0; y--) { getpictcolrs(y, scanline, &ptrunk, hresolu, vresolu); if (fwritecolrs(scanline, hresolu, fp) < 0) break; } + free((char *)scanline); if (fclose(fp) < 0) error(COMMAND, "write error"); - free((char *)scanline); }