ref: 26684157e0d0b5885f33e88c4d31805f48fa5d15
dir: /r_surf.c/
#include "quakedef.h" drawsurf_t r_drawsurf; static int sourcetstep, lightleftstep, lightrightstep; static int surfrowbytes; // used by ASM files static int r_stepback; static int r_numhblocks, r_numvblocks; static pixel_t *r_source, *r_sourcemax; static unsigned *r_lightptr; static int r_lightwidth; static pixel_t *pbasesource; static void *prowdestbase; static void R_DrawSurfaceBlock8_mip0 (void); static void R_DrawSurfaceBlock8_mip1 (void); static void R_DrawSurfaceBlock8_mip2 (void); static void R_DrawSurfaceBlock8_mip3 (void); static void (*surfmiptable[4])(void) = { R_DrawSurfaceBlock8_mip0, R_DrawSurfaceBlock8_mip1, R_DrawSurfaceBlock8_mip2, R_DrawSurfaceBlock8_mip3 }; static unsigned blocklights[18*18]; /* =============== R_AddDynamicLights =============== */ void R_AddDynamicLights (void) { msurface_t *surf; int lnum; int sd, td; float dist, rad, minlight; vec3_t impact, local, entorigin; int s, t; int i; int smax, tmax; mtexinfo_t *tex; surf = r_drawsurf.surf; smax = (surf->extents[0]>>4)+1; tmax = (surf->extents[1]>>4)+1; tex = surf->texinfo; for (lnum=0 ; lnum<MAX_DLIGHTS ; lnum++) { if ( !(surf->dlightbits & (1<<lnum) ) ) continue; // not lit by this light rad = cl_dlights[lnum].radius; VectorSubtract(cl_dlights[lnum].origin, currententity->origin, entorigin); dist = DotProduct (entorigin, surf->plane->normal) - surf->plane->dist; rad -= fabs(dist); minlight = cl_dlights[lnum].minlight; if (rad < minlight) continue; minlight = rad - minlight; for (i=0 ; i<3 ; i++) { impact[i] = entorigin[i] - surf->plane->normal[i]*dist; } local[0] = DotProduct (impact, tex->vecs[0]) + tex->vecs[0][3]; local[1] = DotProduct (impact, tex->vecs[1]) + tex->vecs[1][3]; local[0] -= surf->texturemins[0]; local[1] -= surf->texturemins[1]; for (t = 0 ; t<tmax ; t++) { td = local[1] - t*16; if (td < 0) td = -td; for (s=0 ; s<smax ; s++) { sd = local[0] - s*16; if (sd < 0) sd = -sd; if (sd > td) dist = sd + (td>>1); else dist = td + (sd>>1); if (dist < minlight) blocklights[t*smax + s] += (rad - dist)*256; } } } } /* =============== R_BuildLightMap Combine and scale multiple lightmaps into the 8.8 format in blocklights =============== */ void R_BuildLightMap (void) { int smax, tmax; int t; int i, size; byte *lightmap; unsigned scale; int maps; msurface_t *surf; surf = r_drawsurf.surf; smax = (surf->extents[0]>>4)+1; tmax = (surf->extents[1]>>4)+1; size = smax*tmax; lightmap = surf->samples; if (r_fullbright.value || !cl.worldmodel->lightdata) { for (i=0 ; i<size ; i++) blocklights[i] = 0; return; } // clear to ambient for (i=0 ; i<size ; i++) blocklights[i] = r_refdef.ambientlight<<8; // add all the lightmaps if (lightmap) for (maps = 0 ; maps < MAXLIGHTMAPS && surf->styles[maps] != 255 ; maps++) { scale = r_drawsurf.lightadj[maps]; // 8.8 fraction for (i=0 ; i<size ; i++) blocklights[i] += lightmap[i] * scale; lightmap += size; // skip to next lightmap } // add all the dynamic lights if (surf->dlightframe == r_framecount) R_AddDynamicLights (); // bound, invert, and shift for (i=0 ; i<size ; i++) { t = (255*256 - (int)blocklights[i]) >> (8 - VID_CBITS); if (t < (1 << 6)) t = (1 << 6); blocklights[i] = t; } } /* =============== R_TextureAnimation Returns the proper texture for a given time and base texture =============== */ texture_t *R_TextureAnimation (texture_t *base) { int reletive; int count; if (currententity->frame) { if (base->alternate_anims) base = base->alternate_anims; } if (!base->anim_total) return base; reletive = (int)(cl.time*10) % base->anim_total; count = 0; while (base->anim_min > reletive || base->anim_max <= reletive) { base = base->anim_next; if (!base) fatal ("R_TextureAnimation: broken cycle"); if (++count > 100) fatal ("R_TextureAnimation: infinite cycle"); } return base; } /* =============== R_DrawSurface =============== */ void R_DrawSurface (void) { pixel_t *basetptr; int smax, tmax, twidth; int u, blockdivshift, blocksize; int soffset, basetoffset, texwidth; int horzblockstep; pixel_t *pcolumndest; void (*pblockdrawer)(void); texture_t *mt; // calculate the lightings R_BuildLightMap (); surfrowbytes = r_drawsurf.rowbytes; mt = r_drawsurf.texture; r_source = (pixel_t*)((byte*)mt + mt->offsets[r_drawsurf.surfmip]); // the fractional light values should range from 0 to (VID_GRADES - 1) << 16 // from a source range of 0 - 255 texwidth = mt->width >> r_drawsurf.surfmip; blocksize = 16 >> r_drawsurf.surfmip; blockdivshift = 4 - r_drawsurf.surfmip; r_lightwidth = (r_drawsurf.surf->extents[0]>>4)+1; r_numhblocks = r_drawsurf.surfwidth >> blockdivshift; r_numvblocks = r_drawsurf.surfheight >> blockdivshift; pblockdrawer = surfmiptable[r_drawsurf.surfmip]; // TODO: only needs to be set when there is a display settings change horzblockstep = blocksize; smax = mt->width >> r_drawsurf.surfmip; twidth = texwidth; tmax = mt->height >> r_drawsurf.surfmip; sourcetstep = texwidth; r_stepback = tmax * twidth; r_sourcemax = r_source + (tmax * smax); soffset = r_drawsurf.surf->texturemins[0]; basetoffset = r_drawsurf.surf->texturemins[1]; // << 16 components are to guarantee positive values for % soffset = ((soffset >> r_drawsurf.surfmip) + (smax << 16)) % smax; basetptr = &r_source[((((basetoffset >> r_drawsurf.surfmip) + (tmax << 16)) % tmax) * twidth)]; pcolumndest = r_drawsurf.surfdat; for (u=0 ; u<r_numhblocks; u++) { r_lightptr = blocklights + u; prowdestbase = pcolumndest; pbasesource = basetptr + soffset; (*pblockdrawer)(); soffset = soffset + blocksize; if (soffset >= smax) soffset = 0; pcolumndest += horzblockstep; } } //============================================================================= pixel_t addlight(pixel_t x, int light) { int r, g, b, y; r = (x>>16) & 0xff; g = (x>>8) & 0xff; b = (x>>0) & 0xff; y = (light & 0xff00) >> 8; r = (r * (63-y)+16) >> 5; r = min(r, 255); g = (g * (63-y)+16) >> 5; g = min(g, 255); b = (b * (63-y)+16) >> 5; b = min(b, 255); x = (x & ~0xffffff) | r<<16 | g<<8 | b<<0; return x; } /* ================ R_DrawSurfaceBlock8_mip0 ================ */ static void R_DrawSurfaceBlock8_mip0 (void) { int b, v, i, lightstep, lighttemp, light, lightleft, lightright; pixel_t *psource, *prowdest; psource = pbasesource; prowdest = prowdestbase; for (v=0 ; v<r_numvblocks ; v++) { // FIXME: use delta rather than both right and left, like ASM? lightleft = r_lightptr[0]; lightright = r_lightptr[1]; r_lightptr += r_lightwidth; lightleftstep = (r_lightptr[0] - lightleft) >> 4; lightrightstep = (r_lightptr[1] - lightright) >> 4; for (i=0 ; i<16 ; i++) { lighttemp = lightleft - lightright; lightstep = lighttemp >> 4; light = lightright; for(b = 15; b >= 0; b--){ prowdest[b] = addlight(psource[b], light); light += lightstep; } psource += sourcetstep; lightright += lightrightstep; lightleft += lightleftstep; prowdest += surfrowbytes; } if (psource >= r_sourcemax) psource -= r_stepback; } } /* ================ R_DrawSurfaceBlock8_mip1 ================ */ static void R_DrawSurfaceBlock8_mip1 (void) { int v, i, b, lightstep, lighttemp, light, lightleft, lightright; pixel_t *psource, *prowdest; psource = pbasesource; prowdest = prowdestbase; for (v=0 ; v<r_numvblocks ; v++) { // FIXME: use delta rather than both right and left, like ASM? lightleft = r_lightptr[0]; lightright = r_lightptr[1]; r_lightptr += r_lightwidth; lightleftstep = (r_lightptr[0] - lightleft) >> 3; lightrightstep = (r_lightptr[1] - lightright) >> 3; for (i=0 ; i<8 ; i++) { lighttemp = lightleft - lightright; lightstep = lighttemp >> 3; light = lightright; for(b = 7; b >= 0; b--){ prowdest[b] = addlight(psource[b], light); light += lightstep; } psource += sourcetstep; lightright += lightrightstep; lightleft += lightleftstep; prowdest += surfrowbytes; } if (psource >= r_sourcemax) psource -= r_stepback; } } /* ================ R_DrawSurfaceBlock8_mip2 ================ */ static void R_DrawSurfaceBlock8_mip2 (void) { int v, i, b, lightstep, lighttemp, light, lightleft, lightright; pixel_t *psource, *prowdest; psource = pbasesource; prowdest = prowdestbase; for (v=0 ; v<r_numvblocks ; v++) { // FIXME: use delta rather than both right and left, like ASM? lightleft = r_lightptr[0]; lightright = r_lightptr[1]; r_lightptr += r_lightwidth; lightleftstep = (r_lightptr[0] - lightleft) >> 2; lightrightstep = (r_lightptr[1] - lightright) >> 2; for (i=0 ; i<4 ; i++) { lighttemp = lightleft - lightright; lightstep = lighttemp >> 2; light = lightright; for(b = 3; b >= 0; b--){ prowdest[b] = addlight(psource[b], light); light += lightstep; } psource += sourcetstep; lightright += lightrightstep; lightleft += lightleftstep; prowdest += surfrowbytes; } if (psource >= r_sourcemax) psource -= r_stepback; } } /* ================ R_DrawSurfaceBlock8_mip3 ================ */ static void R_DrawSurfaceBlock8_mip3 (void) { int v, i, b, lightstep, lighttemp, light, lightleft, lightright; pixel_t *psource, *prowdest; psource = pbasesource; prowdest = prowdestbase; for (v=0 ; v<r_numvblocks ; v++) { // FIXME: use delta rather than both right and left, like ASM? lightleft = r_lightptr[0]; lightright = r_lightptr[1]; r_lightptr += r_lightwidth; lightleftstep = (r_lightptr[0] - lightleft) >> 1; lightrightstep = (r_lightptr[1] - lightright) >> 1; for (i=0 ; i<2 ; i++) { lighttemp = lightleft - lightright; lightstep = lighttemp >> 1; light = lightright; for(b = 1; b >= 0; b--){ prowdest[b] = addlight(psource[b], light); light += lightstep; } psource += sourcetstep; lightright += lightrightstep; lightleft += lightleftstep; prowdest += surfrowbytes; } if (psource >= r_sourcemax) psource -= r_stepback; } }