mirror of
				https://github.com/Rockbox/rockbox.git
				synced 2025-10-21 22:17:37 -04:00 
			
		
		
		
	There are some real bugs in here, but we're drowning in warnings. Change-Id: I7c2c0eafc8426327521bdd8a3ac2d3742ac16864
		
			
				
	
	
		
			1647 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1647 lines
		
	
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // WL_DRAW.C
 | |
| 
 | |
| #include "wl_def.h"
 | |
| #pragma hdrstop
 | |
| 
 | |
| #include "wl_cloudsky.h"
 | |
| #include "wl_atmos.h"
 | |
| #include "wl_shade.h"
 | |
| 
 | |
| /*
 | |
| =============================================================================
 | |
| 
 | |
|                                LOCAL CONSTANTS
 | |
| 
 | |
| =============================================================================
 | |
| */
 | |
| 
 | |
| // the door is the last picture before the sprites
 | |
| #define DOORWALL        (PMSpriteStart-8)
 | |
| 
 | |
| #define ACTORSIZE       0x4000
 | |
| 
 | |
| /*
 | |
| =============================================================================
 | |
| 
 | |
|                               GLOBAL VARIABLES
 | |
| 
 | |
| =============================================================================
 | |
| */
 | |
| 
 | |
| static byte *vbuf = NULL;
 | |
| unsigned vbufPitch = 0;
 | |
| 
 | |
| int32_t    lasttimecount;
 | |
| int32_t    frameon;
 | |
| boolean fpscounter;
 | |
| 
 | |
| int fps_frames=0, fps_time=0, fps=0;
 | |
| 
 | |
| int *wallheight;
 | |
| int min_wallheight;
 | |
| 
 | |
| //
 | |
| // math tables
 | |
| //
 | |
| short *pixelangle;
 | |
| int32_t finetangent[FINEANGLES/4];
 | |
| fixed sintable[ANGLES+ANGLES/4];
 | |
| fixed *costable = sintable+(ANGLES/4);
 | |
| 
 | |
| //
 | |
| // refresh variables
 | |
| //
 | |
| fixed   viewx,viewy;                    // the focal point
 | |
| short   viewangle;
 | |
| fixed   viewsin,viewcos;
 | |
| 
 | |
| void    TransformActor (objtype *ob);
 | |
| void    BuildTables (void);
 | |
| void    ClearScreen (void);
 | |
| int     CalcRotate (objtype *ob);
 | |
| void    DrawScaleds (void);
 | |
| void    CalcTics (void);
 | |
| void    ThreeDRefresh (void);
 | |
| 
 | |
| 
 | |
| 
 | |
| //
 | |
| // wall optimization variables
 | |
| //
 | |
| int     lastside;               // true for vertical
 | |
| int32_t    lastintercept;
 | |
| int     lasttilehit;
 | |
| int     lasttexture;
 | |
| 
 | |
| //
 | |
| // ray tracing variables
 | |
| //
 | |
| short    focaltx,focalty,viewtx,viewty;
 | |
| longword xpartialup,xpartialdown,ypartialup,ypartialdown;
 | |
| 
 | |
| short   midangle,angle;
 | |
| 
 | |
| word    tilehit;
 | |
| int     pixx;
 | |
| 
 | |
| short   xtile,ytile;
 | |
| short   xtilestep,ytilestep;
 | |
| int32_t    xintercept,yintercept;
 | |
| word    xstep,ystep;
 | |
| word    xspot,yspot;
 | |
| int     texdelta;
 | |
| 
 | |
| word horizwall[MAXWALLTILES],vertwall[MAXWALLTILES];
 | |
| 
 | |
| 
 | |
| /*
 | |
| ============================================================================
 | |
| 
 | |
|                            3 - D  DEFINITIONS
 | |
| 
 | |
| ============================================================================
 | |
| */
 | |
| 
 | |
| /*
 | |
| ========================
 | |
| =
 | |
| = TransformActor
 | |
| =
 | |
| = Takes paramaters:
 | |
| =   gx,gy               : globalx/globaly of point
 | |
| =
 | |
| = globals:
 | |
| =   viewx,viewy         : point of view
 | |
| =   viewcos,viewsin     : sin/cos of viewangle
 | |
| =   scale               : conversion from global value to screen value
 | |
| =
 | |
| = sets:
 | |
| =   screenx,transx,transy,screenheight: projected edge location and size
 | |
| =
 | |
| ========================
 | |
| */
 | |
| 
 | |
| 
 | |
| //
 | |
| // transform actor
 | |
| //
 | |
| void TransformActor (objtype *ob)
 | |
| {
 | |
|     fixed gx,gy,gxt,gyt,nx,ny;
 | |
| 
 | |
| //
 | |
| // translate point to view centered coordinates
 | |
| //
 | |
|     gx = ob->x-viewx;
 | |
|     gy = ob->y-viewy;
 | |
| 
 | |
| //
 | |
| // calculate newx
 | |
| //
 | |
|     gxt = FixedMul(gx,viewcos);
 | |
|     gyt = FixedMul(gy,viewsin);
 | |
|     nx = gxt-gyt-ACTORSIZE;         // fudge the shape forward a bit, because
 | |
|                                     // the midpoint could put parts of the shape
 | |
|                                     // into an adjacent wall
 | |
| 
 | |
| //
 | |
| // calculate newy
 | |
| //
 | |
|     gxt = FixedMul(gx,viewsin);
 | |
|     gyt = FixedMul(gy,viewcos);
 | |
|     ny = gyt+gxt;
 | |
| 
 | |
| //
 | |
| // calculate perspective ratio
 | |
| //
 | |
|     ob->transx = nx;
 | |
|     ob->transy = ny;
 | |
| 
 | |
|     if (nx<MINDIST)                 // too close, don't overflow the divide
 | |
|     {
 | |
|         ob->viewheight = 0;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     ob->viewx = (word)(centerx + ny*scale/nx);
 | |
| 
 | |
| //
 | |
| // calculate height (heightnumerator/(nx>>8))
 | |
| //
 | |
|     ob->viewheight = (word)(heightnumerator/(nx>>8));
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ========================
 | |
| =
 | |
| = TransformTile
 | |
| =
 | |
| = Takes paramaters:
 | |
| =   tx,ty               : tile the object is centered in
 | |
| =
 | |
| = globals:
 | |
| =   viewx,viewy         : point of view
 | |
| =   viewcos,viewsin     : sin/cos of viewangle
 | |
| =   scale               : conversion from global value to screen value
 | |
| =
 | |
| = sets:
 | |
| =   screenx,transx,transy,screenheight: projected edge location and size
 | |
| =
 | |
| = Returns true if the tile is withing getting distance
 | |
| =
 | |
| ========================
 | |
| */
 | |
| 
 | |
| boolean TransformTile (int tx, int ty, short *dispx, short *dispheight)
 | |
| {
 | |
|     fixed gx,gy,gxt,gyt,nx,ny;
 | |
| 
 | |
| //
 | |
| // translate point to view centered coordinates
 | |
| //
 | |
|     gx = ((int32_t)tx<<TILESHIFT)+0x8000-viewx;
 | |
|     gy = ((int32_t)ty<<TILESHIFT)+0x8000-viewy;
 | |
| 
 | |
| //
 | |
| // calculate newx
 | |
| //
 | |
|     gxt = FixedMul(gx,viewcos);
 | |
|     gyt = FixedMul(gy,viewsin);
 | |
|     nx = gxt-gyt-0x2000;            // 0x2000 is size of object
 | |
| 
 | |
| //
 | |
| // calculate newy
 | |
| //
 | |
|     gxt = FixedMul(gx,viewsin);
 | |
|     gyt = FixedMul(gy,viewcos);
 | |
|     ny = gyt+gxt;
 | |
| 
 | |
| 
 | |
| //
 | |
| // calculate height / perspective ratio
 | |
| //
 | |
|     if (nx<MINDIST)                 // too close, don't overflow the divide
 | |
|         *dispheight = 0;
 | |
|     else
 | |
|     {
 | |
|         *dispx = (short)(centerx + ny*scale/nx);
 | |
|         *dispheight = (short)(heightnumerator/(nx>>8));
 | |
|     }
 | |
| 
 | |
| //
 | |
| // see if it should be grabbed
 | |
| //
 | |
|     if (nx<TILEGLOBAL && ny>-TILEGLOBAL/2 && ny<TILEGLOBAL/2)
 | |
|         return true;
 | |
|     else
 | |
|         return false;
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = CalcHeight
 | |
| =
 | |
| = Calculates the height of xintercept,yintercept from viewx,viewy
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| int CalcHeight(void)
 | |
| {
 | |
|     fixed z = FixedMul(xintercept - viewx, viewcos)
 | |
|         - FixedMul(yintercept - viewy, viewsin);
 | |
|     if(z < MINDIST) z = MINDIST;
 | |
|     int height = heightnumerator / (z >> 8);
 | |
|     if(height < min_wallheight) min_wallheight = height;
 | |
|     return height;
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ===================
 | |
| =
 | |
| = ScalePost
 | |
| =
 | |
| ===================
 | |
| */
 | |
| 
 | |
| byte *postsource;
 | |
| int postx;
 | |
| int postwidth;
 | |
| 
 | |
| void ScalePost(void)
 | |
| {
 | |
|     int ywcount, yoffs, yw, yd, yendoffs;
 | |
|     byte col;
 | |
| 
 | |
| #ifdef USE_SHADING
 | |
|     byte *curshades = shadetable[GetShade(wallheight[postx])];
 | |
| #endif
 | |
| 
 | |
|     ywcount = yd = wallheight[postx] >> 3;
 | |
|     if(yd <= 0) yd = 100;
 | |
| 
 | |
|     yoffs = (viewheight / 2 - ywcount) * vbufPitch;
 | |
|     if(yoffs < 0) yoffs = 0;
 | |
|     yoffs += postx;
 | |
| 
 | |
|     yendoffs = viewheight / 2 + ywcount - 1;
 | |
|     yw=TEXTURESIZE-1;
 | |
| 
 | |
|     while(yendoffs >= viewheight)
 | |
|     {
 | |
|         ywcount -= TEXTURESIZE/2;
 | |
|         while(ywcount <= 0)
 | |
|         {
 | |
|             ywcount += yd;
 | |
|             yw--;
 | |
|         }
 | |
|         yendoffs--;
 | |
|     }
 | |
|     if(yw < 0) return;
 | |
| 
 | |
| #ifdef USE_SHADING
 | |
|     col = curshades[postsource[yw]];
 | |
| #else
 | |
|     col = postsource[yw];
 | |
| #endif
 | |
|     yendoffs = yendoffs * vbufPitch + postx;
 | |
|     while(yoffs <= yendoffs)
 | |
|     {
 | |
|         vbuf[yendoffs] = col;
 | |
|         ywcount -= TEXTURESIZE/2;
 | |
|         if(ywcount <= 0)
 | |
|         {
 | |
|             do
 | |
|             {
 | |
|                 ywcount += yd;
 | |
|                 yw--;
 | |
|             }
 | |
|             while(ywcount <= 0);
 | |
|             if(yw < 0) break;
 | |
| #ifdef USE_SHADING
 | |
|             col = curshades[postsource[yw]];
 | |
| #else
 | |
|             col = postsource[yw];
 | |
| #endif
 | |
|         }
 | |
|         yendoffs -= vbufPitch;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void GlobalScalePost(byte *vidbuf, unsigned pitch)
 | |
| {
 | |
|     vbuf = vidbuf;
 | |
|     vbufPitch = pitch;
 | |
|     ScalePost();
 | |
| }
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = HitVertWall
 | |
| =
 | |
| = tilehit bit 7 is 0, because it's not a door tile
 | |
| = if bit 6 is 1 and the adjacent tile is a door tile, use door side pic
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| void HitVertWall (void)
 | |
| {
 | |
|     int wallpic;
 | |
|     int texture;
 | |
| 
 | |
|     texture = ((yintercept+texdelta)>>TEXTUREFROMFIXEDSHIFT)&TEXTUREMASK;
 | |
|     if (xtilestep == -1)
 | |
|     {
 | |
|         texture = TEXTUREMASK-texture;
 | |
|         xintercept += TILEGLOBAL;
 | |
|     }
 | |
| 
 | |
|     if(lastside==1 && lastintercept==xtile && lasttilehit==tilehit && !(lasttilehit & 0x40))
 | |
|     {
 | |
|         if((pixx&3) && texture == lasttexture)
 | |
|         {
 | |
|             ScalePost();
 | |
|             postx = pixx;
 | |
|             wallheight[pixx] = wallheight[pixx-1];
 | |
|             return;
 | |
|         }
 | |
|         ScalePost();
 | |
|         wallheight[pixx] = CalcHeight();
 | |
|         postsource+=texture-lasttexture;
 | |
|         postwidth=1;
 | |
|         postx=pixx;
 | |
|         lasttexture=texture;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if(lastside!=-1) ScalePost();
 | |
| 
 | |
|     lastside=1;
 | |
|     lastintercept=xtile;
 | |
|     lasttilehit=tilehit;
 | |
|     lasttexture=texture;
 | |
|     wallheight[pixx] = CalcHeight();
 | |
|     postx = pixx;
 | |
|     postwidth = 1;
 | |
| 
 | |
|     if (tilehit & 0x40)
 | |
|     {                                                               // check for adjacent doors
 | |
|         ytile = (short)(yintercept>>TILESHIFT);
 | |
|         if ( tilemap[xtile-xtilestep][ytile]&0x80 )
 | |
|             wallpic = DOORWALL+3;
 | |
|         else
 | |
|             wallpic = vertwall[tilehit & ~0x40];
 | |
|     }
 | |
|     else
 | |
|         wallpic = vertwall[tilehit];
 | |
| 
 | |
|     postsource = PM_GetTexture(wallpic) + texture;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = HitHorizWall
 | |
| =
 | |
| = tilehit bit 7 is 0, because it's not a door tile
 | |
| = if bit 6 is 1 and the adjacent tile is a door tile, use door side pic
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| void HitHorizWall (void)
 | |
| {
 | |
|     int wallpic;
 | |
|     int texture;
 | |
| 
 | |
|     texture = ((xintercept+texdelta)>>TEXTUREFROMFIXEDSHIFT)&TEXTUREMASK;
 | |
|     if (ytilestep == -1)
 | |
|         yintercept += TILEGLOBAL;
 | |
|     else
 | |
|         texture = TEXTUREMASK-texture;
 | |
| 
 | |
|     if(lastside==0 && lastintercept==ytile && lasttilehit==tilehit && !(lasttilehit & 0x40))
 | |
|     {
 | |
|         if((pixx&3) && texture == lasttexture)
 | |
|         {
 | |
|             ScalePost();
 | |
|             postx=pixx;
 | |
|             wallheight[pixx] = wallheight[pixx-1];
 | |
|             return;
 | |
|         }
 | |
|         ScalePost();
 | |
|         wallheight[pixx] = CalcHeight();
 | |
|         postsource+=texture-lasttexture;
 | |
|         postwidth=1;
 | |
|         postx=pixx;
 | |
|         lasttexture=texture;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if(lastside!=-1) ScalePost();
 | |
| 
 | |
|     lastside=0;
 | |
|     lastintercept=ytile;
 | |
|     lasttilehit=tilehit;
 | |
|     lasttexture=texture;
 | |
|     wallheight[pixx] = CalcHeight();
 | |
|     postx = pixx;
 | |
|     postwidth = 1;
 | |
| 
 | |
|     if (tilehit & 0x40)
 | |
|     {                                                               // check for adjacent doors
 | |
|         xtile = (short)(xintercept>>TILESHIFT);
 | |
|         if ( tilemap[xtile][ytile-ytilestep]&0x80)
 | |
|             wallpic = DOORWALL+2;
 | |
|         else
 | |
|             wallpic = horizwall[tilehit & ~0x40];
 | |
|     }
 | |
|     else
 | |
|         wallpic = horizwall[tilehit];
 | |
| 
 | |
|     postsource = PM_GetTexture(wallpic) + texture;
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = HitHorizDoor
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| void HitHorizDoor (void)
 | |
| {
 | |
|     int doorpage;
 | |
|     int doornum;
 | |
|     int texture;
 | |
| 
 | |
|     doornum = tilehit&0x7f;
 | |
|     texture = ((xintercept-doorposition[doornum])>>TEXTUREFROMFIXEDSHIFT)&TEXTUREMASK;
 | |
| 
 | |
|     if(lasttilehit==tilehit)
 | |
|     {
 | |
|         if((pixx&3) && texture == lasttexture)
 | |
|         {
 | |
|             ScalePost();
 | |
|             postx=pixx;
 | |
|             wallheight[pixx] = wallheight[pixx-1];
 | |
|             return;
 | |
|         }
 | |
|         ScalePost();
 | |
|         wallheight[pixx] = CalcHeight();
 | |
|         postsource+=texture-lasttexture;
 | |
|         postwidth=1;
 | |
|         postx=pixx;
 | |
|         lasttexture=texture;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if(lastside!=-1) ScalePost();
 | |
| 
 | |
|     lastside=2;
 | |
|     lasttilehit=tilehit;
 | |
|     lasttexture=texture;
 | |
|     wallheight[pixx] = CalcHeight();
 | |
|     postx = pixx;
 | |
|     postwidth = 1;
 | |
| 
 | |
|     switch(doorobjlist[doornum].lock)
 | |
|     {
 | |
|         case dr_normal:
 | |
|             doorpage = DOORWALL;
 | |
|             break;
 | |
|         case dr_lock1:
 | |
|         case dr_lock2:
 | |
|         case dr_lock3:
 | |
|         case dr_lock4:
 | |
|             doorpage = DOORWALL+6;
 | |
|             break;
 | |
|         case dr_elevator:
 | |
|             doorpage = DOORWALL+4;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     postsource = PM_GetTexture(doorpage) + texture;
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = HitVertDoor
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| void HitVertDoor (void)
 | |
| {
 | |
|     int doorpage;
 | |
|     int doornum;
 | |
|     int texture;
 | |
| 
 | |
|     doornum = tilehit&0x7f;
 | |
|     texture = ((yintercept-doorposition[doornum])>>TEXTUREFROMFIXEDSHIFT)&TEXTUREMASK;
 | |
| 
 | |
|     if(lasttilehit==tilehit)
 | |
|     {
 | |
|         if((pixx&3) && texture == lasttexture)
 | |
|         {
 | |
|             ScalePost();
 | |
|             postx=pixx;
 | |
|             wallheight[pixx] = wallheight[pixx-1];
 | |
|             return;
 | |
|         }
 | |
|         ScalePost();
 | |
|         wallheight[pixx] = CalcHeight();
 | |
|         postsource+=texture-lasttexture;
 | |
|         postwidth=1;
 | |
|         postx=pixx;
 | |
|         lasttexture=texture;
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if(lastside!=-1) ScalePost();
 | |
| 
 | |
|     lastside=2;
 | |
|     lasttilehit=tilehit;
 | |
|     lasttexture=texture;
 | |
|     wallheight[pixx] = CalcHeight();
 | |
|     postx = pixx;
 | |
|     postwidth = 1;
 | |
| 
 | |
|     switch(doorobjlist[doornum].lock)
 | |
|     {
 | |
|         case dr_normal:
 | |
|             doorpage = DOORWALL+1;
 | |
|             break;
 | |
|         case dr_lock1:
 | |
|         case dr_lock2:
 | |
|         case dr_lock3:
 | |
|         case dr_lock4:
 | |
|             doorpage = DOORWALL+7;
 | |
|             break;
 | |
|         case dr_elevator:
 | |
|             doorpage = DOORWALL+5;
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     postsource = PM_GetTexture(doorpage) + texture;
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| #define HitHorizBorder HitHorizWall
 | |
| #define HitVertBorder HitVertWall
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| byte vgaCeiling[]=
 | |
| {
 | |
| #ifndef SPEAR
 | |
|  0x1d,0x1d,0x1d,0x1d,0x1d,0x1d,0x1d,0x1d,0x1d,0xbf,
 | |
|  0x4e,0x4e,0x4e,0x1d,0x8d,0x4e,0x1d,0x2d,0x1d,0x8d,
 | |
|  0x1d,0x1d,0x1d,0x1d,0x1d,0x2d,0xdd,0x1d,0x1d,0x98,
 | |
| 
 | |
|  0x1d,0x9d,0x2d,0xdd,0xdd,0x9d,0x2d,0x4d,0x1d,0xdd,
 | |
|  0x7d,0x1d,0x2d,0x2d,0xdd,0xd7,0x1d,0x1d,0x1d,0x2d,
 | |
|  0x1d,0x1d,0x1d,0x1d,0xdd,0xdd,0x7d,0xdd,0xdd,0xdd
 | |
| #else
 | |
|  0x6f,0x4f,0x1d,0xde,0xdf,0x2e,0x7f,0x9e,0xae,0x7f,
 | |
|  0x1d,0xde,0xdf,0xde,0xdf,0xde,0xe1,0xdc,0x2e,0x1d,0xdc
 | |
| #endif
 | |
| };
 | |
| 
 | |
| /*
 | |
| =====================
 | |
| =
 | |
| = VGAClearScreen
 | |
| =
 | |
| =====================
 | |
| */
 | |
| 
 | |
| void VGAClearScreen (void)
 | |
| {
 | |
|     byte ceiling=vgaCeiling[gamestate.episode*10+mapon];
 | |
| 
 | |
|     int y;
 | |
|     byte *ptr = vbuf;
 | |
| #ifdef USE_SHADING
 | |
|     for(y = 0; y < viewheight / 2; y++, ptr += vbufPitch)
 | |
|         memset(ptr, shadetable[GetShade((viewheight / 2 - y) << 3)][ceiling], viewwidth);
 | |
|     for(; y < viewheight; y++, ptr += vbufPitch)
 | |
|         memset(ptr, shadetable[GetShade((y - viewheight / 2) << 3)][0x19], viewwidth);
 | |
| #else
 | |
|     for(y = 0; y < viewheight / 2; y++, ptr += vbufPitch)
 | |
|         memset(ptr, ceiling, viewwidth);
 | |
|     for(; y < viewheight; y++, ptr += vbufPitch)
 | |
|         memset(ptr, 0x19, viewwidth);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| =====================
 | |
| =
 | |
| = CalcRotate
 | |
| =
 | |
| =====================
 | |
| */
 | |
| 
 | |
| int CalcRotate (objtype *ob)
 | |
| {
 | |
|     int angle, viewangle;
 | |
| 
 | |
|     // this isn't exactly correct, as it should vary by a trig value,
 | |
|     // but it is close enough with only eight rotations
 | |
| 
 | |
|     viewangle = player->angle + (centerx - ob->viewx)/8;
 | |
| 
 | |
|     if (ob->obclass == rocketobj || ob->obclass == hrocketobj)
 | |
|         angle = (viewangle-180) - ob->angle;
 | |
|     else
 | |
|         angle = (viewangle-180) - dirangle[ob->dir];
 | |
| 
 | |
|     angle+=ANGLES/16;
 | |
|     while (angle>=ANGLES)
 | |
|         angle-=ANGLES;
 | |
|     while (angle<0)
 | |
|         angle+=ANGLES;
 | |
| 
 | |
|     if (ob->state->rotate == 2)             // 2 rotation pain frame
 | |
|         return 0;               // pain with shooting frame bugfix
 | |
| 
 | |
|     return angle/(ANGLES/8);
 | |
| }
 | |
| 
 | |
| void ScaleShape (int xcenter, int shapenum, unsigned height, uint32_t flags)
 | |
| {
 | |
|     t_compshape *shape;
 | |
|     unsigned scale,pixheight;
 | |
|     unsigned starty,endy;
 | |
|     word *cmdptr;
 | |
|     byte *cline;
 | |
|     byte *line;
 | |
|     byte *vmem;
 | |
|     int actx,i,upperedge;
 | |
|     short newstart;
 | |
|     int scrstarty,screndy,lpix,rpix,pixcnt,ycnt;
 | |
|     unsigned j;
 | |
|     byte col;
 | |
| 
 | |
| #ifdef USE_SHADING
 | |
|     byte *curshades;
 | |
|     if(flags & FL_FULLBRIGHT)
 | |
|         curshades = shadetable[0];
 | |
|     else
 | |
|         curshades = shadetable[GetShade(height)];
 | |
| #endif
 | |
| 
 | |
|     shape = (t_compshape *) PM_GetSprite(shapenum);
 | |
| 
 | |
|     scale=height>>3;                 // low three bits are fractional
 | |
|     if(!scale) return;   // too close or far away
 | |
| 
 | |
|     pixheight=scale*SPRITESCALEFACTOR;
 | |
|     actx=xcenter-scale;
 | |
|     upperedge=viewheight/2-scale;
 | |
| 
 | |
|     cmdptr=(word *) shape->dataofs;
 | |
| 
 | |
|     for(i=shape->leftpix,pixcnt=i*pixheight,rpix=(pixcnt>>6)+actx;i<=shape->rightpix;i++,cmdptr++)
 | |
|     {
 | |
|         lpix=rpix;
 | |
|         if(lpix>=viewwidth) break;
 | |
|         pixcnt+=pixheight;
 | |
|         rpix=(pixcnt>>6)+actx;
 | |
|         if(lpix!=rpix && rpix>0)
 | |
|         {
 | |
|             if(lpix<0) lpix=0;
 | |
|             if(rpix>viewwidth) rpix=viewwidth,i=shape->rightpix+1;
 | |
|             cline=(byte *)shape + *cmdptr;
 | |
|             while(lpix<rpix)
 | |
|             {
 | |
|                 if(wallheight[lpix]<=(int)height)
 | |
|                 {
 | |
|                     line=cline;
 | |
|                     while((endy = READWORD(&line)) != 0)
 | |
|                     {
 | |
|                         endy >>= 1;
 | |
|                         newstart = READWORD(&line);
 | |
|                         starty = READWORD(&line) >> 1;
 | |
|                         j=starty;
 | |
|                         ycnt=j*pixheight;
 | |
|                         screndy=(ycnt>>6)+upperedge;
 | |
|                         if(screndy<0) vmem=vbuf+lpix;
 | |
|                         else vmem=vbuf+screndy*vbufPitch+lpix;
 | |
|                         for(;j<endy;j++)
 | |
|                         {
 | |
|                             scrstarty=screndy;
 | |
|                             ycnt+=pixheight;
 | |
|                             screndy=(ycnt>>6)+upperedge;
 | |
|                             if(scrstarty!=screndy && screndy>0)
 | |
|                             {
 | |
| #ifdef USE_SHADING
 | |
|                                 col=curshades[((byte *)shape)[newstart+j]];
 | |
| #else
 | |
|                                 col=((byte *)shape)[newstart+j];
 | |
| #endif
 | |
|                                 if(scrstarty<0) scrstarty=0;
 | |
|                                 if(screndy>viewheight) screndy=viewheight,j=endy;
 | |
| 
 | |
|                                 while(scrstarty<screndy)
 | |
|                                 {
 | |
|                                     *vmem=col;
 | |
|                                     vmem+=vbufPitch;
 | |
|                                     scrstarty++;
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 lpix++;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void SimpleScaleShape (int xcenter, int shapenum, unsigned height)
 | |
| {
 | |
|     t_compshape   *shape;
 | |
|     unsigned scale,pixheight;
 | |
|     unsigned starty,endy;
 | |
|     word *cmdptr;
 | |
|     byte *cline;
 | |
|     byte *line;
 | |
|     int actx,i,upperedge;
 | |
|     short newstart;
 | |
|     int scrstarty,screndy,lpix,rpix,pixcnt,ycnt;
 | |
|     unsigned j;
 | |
|     byte col;
 | |
|     byte *vmem;
 | |
| 
 | |
|     shape = (t_compshape *) PM_GetSprite(shapenum);
 | |
| 
 | |
|     scale=height>>1;
 | |
|     pixheight=scale*SPRITESCALEFACTOR;
 | |
|     actx=xcenter-scale;
 | |
|     upperedge=viewheight/2-scale;
 | |
| 
 | |
|     cmdptr=shape->dataofs;
 | |
| 
 | |
|     for(i=shape->leftpix,pixcnt=i*pixheight,rpix=(pixcnt>>6)+actx;i<=shape->rightpix;i++,cmdptr++)
 | |
|     {
 | |
|         lpix=rpix;
 | |
|         if(lpix>=viewwidth) break;
 | |
|         pixcnt+=pixheight;
 | |
|         rpix=(pixcnt>>6)+actx;
 | |
|         if(lpix!=rpix && rpix>0)
 | |
|         {
 | |
|             if(lpix<0) lpix=0;
 | |
|             if(rpix>viewwidth) rpix=viewwidth,i=shape->rightpix+1;
 | |
|             cline = (byte *)shape + *cmdptr;
 | |
|             while(lpix<rpix)
 | |
|             {
 | |
|                 line=cline;
 | |
|                 while((endy = READWORD(&line)) != 0)
 | |
|                 {
 | |
|                     endy >>= 1;
 | |
|                     newstart = READWORD(&line);
 | |
|                     starty = READWORD(&line) >> 1;
 | |
|                     j=starty;
 | |
|                     ycnt=j*pixheight;
 | |
|                     screndy=(ycnt>>6)+upperedge;
 | |
|                     if(screndy<0) vmem=vbuf+lpix;
 | |
|                     else vmem=vbuf+screndy*vbufPitch+lpix;
 | |
|                     for(;j<endy;j++)
 | |
|                     {
 | |
|                         scrstarty=screndy;
 | |
|                         ycnt+=pixheight;
 | |
|                         screndy=(ycnt>>6)+upperedge;
 | |
|                         if(scrstarty!=screndy && screndy>0)
 | |
|                         {
 | |
|                             col=((byte *)shape)[newstart+j];
 | |
|                             if(scrstarty<0) scrstarty=0;
 | |
|                             if(screndy>viewheight) screndy=viewheight,j=endy;
 | |
| 
 | |
|                             while(scrstarty<screndy)
 | |
|                             {
 | |
|                                 *vmem=col;
 | |
|                                 vmem+=vbufPitch;
 | |
|                                 scrstarty++;
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 lpix++;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*
 | |
| =====================
 | |
| =
 | |
| = DrawScaleds
 | |
| =
 | |
| = Draws all objects that are visable
 | |
| =
 | |
| =====================
 | |
| */
 | |
| 
 | |
| #define MAXVISABLE 250
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|     short      viewx,
 | |
|                viewheight,
 | |
|                shapenum;
 | |
|     short      flags;          // this must be changed to uint32_t, when you
 | |
|                                // you need more than 16-flags for drawing
 | |
| #ifdef USE_DIR3DSPR
 | |
|     statobj_t *transsprite;
 | |
| #endif
 | |
| } visobj_t;
 | |
| 
 | |
| visobj_t vislist[MAXVISABLE];
 | |
| visobj_t *visptr,*visstep,*farthest;
 | |
| 
 | |
| void DrawScaleds (void)
 | |
| {
 | |
|     int      i,least,numvisable,height;
 | |
|     byte     *tilespot,*visspot;
 | |
|     unsigned spotloc;
 | |
| 
 | |
|     statobj_t *statptr;
 | |
|     objtype   *obj;
 | |
| 
 | |
|     visptr = &vislist[0];
 | |
| 
 | |
| //
 | |
| // place static objects
 | |
| //
 | |
|     for (statptr = &statobjlist[0] ; statptr !=laststatobj ; statptr++)
 | |
|     {
 | |
|         if ((visptr->shapenum = statptr->shapenum) == -1)
 | |
|             continue;                                               // object has been deleted
 | |
| 
 | |
|         if (!*statptr->visspot)
 | |
|             continue;                                               // not visable
 | |
| 
 | |
|         if (TransformTile (statptr->tilex,statptr->tiley,
 | |
|             &visptr->viewx,&visptr->viewheight) && statptr->flags & FL_BONUS)
 | |
|         {
 | |
|             GetBonus (statptr);
 | |
|             if(statptr->shapenum == -1)
 | |
|                 continue;                                           // object has been taken
 | |
|         }
 | |
| 
 | |
|         if (!visptr->viewheight)
 | |
|             continue;                                               // to close to the object
 | |
| 
 | |
| #ifdef USE_DIR3DSPR
 | |
|         if(statptr->flags & FL_DIR_MASK)
 | |
|             visptr->transsprite=statptr;
 | |
|         else
 | |
|             visptr->transsprite=NULL;
 | |
| #endif
 | |
| 
 | |
|         if (visptr < &vislist[MAXVISABLE-1])    // don't let it overflow
 | |
|         {
 | |
|             visptr->flags = (short) statptr->flags;
 | |
|             visptr++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
| //
 | |
| // place active objects
 | |
| //
 | |
|     for (obj = player->next;obj;obj=obj->next)
 | |
|     {
 | |
|         if ((visptr->shapenum = obj->state->shapenum)==0)
 | |
|             continue;                                               // no shape
 | |
| 
 | |
|         spotloc = (obj->tilex<<mapshift)+obj->tiley;   // optimize: keep in struct?
 | |
|         visspot = &spotvis[0][0]+spotloc;
 | |
|         tilespot = &tilemap[0][0]+spotloc;
 | |
| 
 | |
|         //
 | |
|         // could be in any of the nine surrounding tiles
 | |
|         //
 | |
|         if (*visspot
 | |
|             || ( *(visspot-1) && !*(tilespot-1) )
 | |
|             || ( *(visspot+1) && !*(tilespot+1) )
 | |
|             || ( *(visspot-65) && !*(tilespot-65) )
 | |
|             || ( *(visspot-64) && !*(tilespot-64) )
 | |
|             || ( *(visspot-63) && !*(tilespot-63) )
 | |
|             || ( *(visspot+65) && !*(tilespot+65) )
 | |
|             || ( *(visspot+64) && !*(tilespot+64) )
 | |
|             || ( *(visspot+63) && !*(tilespot+63) ) )
 | |
|         {
 | |
|             obj->active = ac_yes;
 | |
|             TransformActor (obj);
 | |
|             if (!obj->viewheight)
 | |
|                 continue;                                               // too close or far away
 | |
| 
 | |
|             visptr->viewx = obj->viewx;
 | |
|             visptr->viewheight = obj->viewheight;
 | |
|             if (visptr->shapenum == -1)
 | |
|                 visptr->shapenum = obj->temp1;  // special shape
 | |
| 
 | |
|             if (obj->state->rotate)
 | |
|                 visptr->shapenum += CalcRotate (obj);
 | |
| 
 | |
|             if (visptr < &vislist[MAXVISABLE-1])    // don't let it overflow
 | |
|             {
 | |
|                 visptr->flags = (short) obj->flags;
 | |
| #ifdef USE_DIR3DSPR
 | |
|                 visptr->transsprite = NULL;
 | |
| #endif
 | |
|                 visptr++;
 | |
|             }
 | |
|             obj->flags |= FL_VISABLE;
 | |
|         }
 | |
|         else
 | |
|             obj->flags &= ~FL_VISABLE;
 | |
|     }
 | |
| 
 | |
| //
 | |
| // draw from back to front
 | |
| //
 | |
|     numvisable = (int) (visptr-&vislist[0]);
 | |
| 
 | |
|     if (!numvisable)
 | |
|         return;                                                                 // no visable objects
 | |
| 
 | |
|     for (i = 0; i<numvisable; i++)
 | |
|     {
 | |
|         least = 32000;
 | |
|         for (visstep=&vislist[0] ; visstep<visptr ; visstep++)
 | |
|         {
 | |
|             height = visstep->viewheight;
 | |
|             if (height < least)
 | |
|             {
 | |
|                 least = height;
 | |
|                 farthest = visstep;
 | |
|             }
 | |
|         }
 | |
|         //
 | |
|         // draw farthest
 | |
|         //
 | |
| #ifdef USE_DIR3DSPR
 | |
|         if(farthest->transsprite)
 | |
|             Scale3DShape(vbuf, vbufPitch, farthest->transsprite);
 | |
|         else
 | |
| #endif
 | |
|             ScaleShape(farthest->viewx, farthest->shapenum, farthest->viewheight, farthest->flags);
 | |
| 
 | |
|         farthest->viewheight = 32000;
 | |
|     }
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ==============
 | |
| =
 | |
| = DrawPlayerWeapon
 | |
| =
 | |
| = Draw the player's hands
 | |
| =
 | |
| ==============
 | |
| */
 | |
| 
 | |
| int weaponscale[NUMWEAPONS] = {SPR_KNIFEREADY, SPR_PISTOLREADY,
 | |
|     SPR_MACHINEGUNREADY, SPR_CHAINREADY};
 | |
| 
 | |
| void DrawPlayerWeapon (void)
 | |
| {
 | |
|     int shapenum;
 | |
| 
 | |
| #ifndef SPEAR
 | |
|     if (gamestate.victoryflag)
 | |
|     {
 | |
| #ifndef APOGEE_1_0
 | |
|         if (player->state == &s_deathcam && (GetTimeCount()&32) )
 | |
|             SimpleScaleShape(viewwidth/2,SPR_DEATHCAM,viewheight+1);
 | |
| #endif
 | |
|         return;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     if (gamestate.weapon != -1)
 | |
|     {
 | |
|         shapenum = weaponscale[gamestate.weapon]+gamestate.weaponframe;
 | |
|         SimpleScaleShape(viewwidth/2,shapenum,viewheight+1);
 | |
|     }
 | |
| 
 | |
|     if (demorecord || demoplayback)
 | |
|         SimpleScaleShape(viewwidth/2,SPR_DEMO,viewheight+1);
 | |
| }
 | |
| 
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| 
 | |
| /*
 | |
| =====================
 | |
| =
 | |
| = CalcTics
 | |
| =
 | |
| =====================
 | |
| */
 | |
| 
 | |
| void CalcTics (void)
 | |
| {
 | |
| //
 | |
| // calculate tics since last refresh for adaptive timing
 | |
| //
 | |
|     if (lasttimecount > (int32_t) GetTimeCount())
 | |
|         lasttimecount = GetTimeCount();    // if the game was paused a LONG time
 | |
| 
 | |
|     uint32_t curtime = SDL_GetTicks();
 | |
|     tics = (curtime * 7) / 100 - lasttimecount;
 | |
|     if(!tics)
 | |
|     {
 | |
|         // wait until end of current tic
 | |
|         SDL_Delay(((lasttimecount + 1) * 100) / 7 - curtime);
 | |
|         tics = 1;
 | |
|     }
 | |
| 
 | |
|     lasttimecount += tics;
 | |
| 
 | |
|     if (tics>MAXTICS)
 | |
|         tics = MAXTICS;
 | |
| }
 | |
| 
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| void AsmRefresh(void)
 | |
| {
 | |
|     int32_t xstep,ystep;
 | |
|     longword xpartial,ypartial;
 | |
|     boolean playerInPushwallBackTile = tilemap[focaltx][focalty] == 64;
 | |
| 
 | |
|     for(pixx=0;pixx<viewwidth;pixx++)
 | |
|     {
 | |
|         short angl=midangle+pixelangle[pixx];
 | |
|         if(angl<0) angl+=FINEANGLES;
 | |
|         if(angl>=3600) angl-=FINEANGLES;
 | |
|         if(angl<900)
 | |
|         {
 | |
|             xtilestep=1;
 | |
|             ytilestep=-1;
 | |
|             xstep=finetangent[900-1-angl];
 | |
|             ystep=-finetangent[angl];
 | |
|             xpartial=xpartialup;
 | |
|             ypartial=ypartialdown;
 | |
|         }
 | |
|         else if(angl<1800)
 | |
|         {
 | |
|             xtilestep=-1;
 | |
|             ytilestep=-1;
 | |
|             xstep=-finetangent[angl-900];
 | |
|             ystep=-finetangent[1800-1-angl];
 | |
|             xpartial=xpartialdown;
 | |
|             ypartial=ypartialdown;
 | |
|         }
 | |
|         else if(angl<2700)
 | |
|         {
 | |
|             xtilestep=-1;
 | |
|             ytilestep=1;
 | |
|             xstep=-finetangent[2700-1-angl];
 | |
|             ystep=finetangent[angl-1800];
 | |
|             xpartial=xpartialdown;
 | |
|             ypartial=ypartialup;
 | |
|         }
 | |
|         else if(angl<3600)
 | |
|         {
 | |
|             xtilestep=1;
 | |
|             ytilestep=1;
 | |
|             xstep=finetangent[angl-2700];
 | |
|             ystep=finetangent[3600-1-angl];
 | |
|             xpartial=xpartialup;
 | |
|             ypartial=ypartialup;
 | |
|         }
 | |
|         yintercept=FixedMul(ystep,xpartial)+viewy;
 | |
|         xtile=focaltx+xtilestep;
 | |
|         xspot=(word)((xtile<<mapshift)+((uint32_t)yintercept>>16));
 | |
|         xintercept=FixedMul(xstep,ypartial)+viewx;
 | |
|         ytile=focalty+ytilestep;
 | |
|         yspot=(word)((((uint32_t)xintercept>>16)<<mapshift)+ytile);
 | |
|         texdelta=0;
 | |
| 
 | |
|         // Special treatment when player is in back tile of pushwall
 | |
|         if(playerInPushwallBackTile)
 | |
|         {
 | |
|             if(    pwalldir == di_east && xtilestep ==  1
 | |
|                 || pwalldir == di_west && xtilestep == -1)
 | |
|             {
 | |
|                 int32_t yintbuf = yintercept - ((ystep * (64 - pwallpos)) >> 6);
 | |
|                 if((yintbuf >> 16) == focalty)   // ray hits pushwall back?
 | |
|                 {
 | |
|                     if(pwalldir == di_east)
 | |
|                         xintercept = (focaltx << TILESHIFT) + (pwallpos << 10);
 | |
|                     else
 | |
|                         xintercept = (focaltx << TILESHIFT) - TILEGLOBAL + ((64 - pwallpos) << 10);
 | |
|                     yintercept = yintbuf;
 | |
|                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                     tilehit = pwalltile;
 | |
|                     HitVertWall();
 | |
|                     continue;
 | |
|                 }
 | |
|             }
 | |
|             else if(pwalldir == di_south && ytilestep ==  1
 | |
|                 ||  pwalldir == di_north && ytilestep == -1)
 | |
|             {
 | |
|                 int32_t xintbuf = xintercept - ((xstep * (64 - pwallpos)) >> 6);
 | |
|                 if((xintbuf >> 16) == focaltx)   // ray hits pushwall back?
 | |
|                 {
 | |
|                     xintercept = xintbuf;
 | |
|                     if(pwalldir == di_south)
 | |
|                         yintercept = (focalty << TILESHIFT) + (pwallpos << 10);
 | |
|                     else
 | |
|                         yintercept = (focalty << TILESHIFT) - TILEGLOBAL + ((64 - pwallpos) << 10);
 | |
|                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                     tilehit = pwalltile;
 | |
|                     HitHorizWall();
 | |
|                     continue;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         do
 | |
|         {
 | |
|             if(ytilestep==-1 && (yintercept>>16)<=ytile) goto horizentry;
 | |
|             if(ytilestep==1 && (yintercept>>16)>=ytile) goto horizentry;
 | |
| vertentry:
 | |
|             if((uint32_t)yintercept>mapheight*65536-1 || (word)xtile>=mapwidth)
 | |
|             {
 | |
|                 if(xtile<0) xintercept=0, xtile=0;
 | |
|                 else if(xtile>=mapwidth) xintercept=mapwidth<<TILESHIFT, xtile=mapwidth-1;
 | |
|                 else xtile=(short) (xintercept >> TILESHIFT);
 | |
|                 if(yintercept<0) yintercept=0, ytile=0;
 | |
|                 else if(yintercept>=(mapheight<<TILESHIFT)) yintercept=mapheight<<TILESHIFT, ytile=mapheight-1;
 | |
|                 yspot=0xffff;
 | |
|                 tilehit=0;
 | |
|                 HitHorizBorder();
 | |
|                 break;
 | |
|             }
 | |
|             if(xspot>=maparea) break;
 | |
|             tilehit=((byte *)tilemap)[xspot];
 | |
|             if(tilehit)
 | |
|             {
 | |
|                 if(tilehit&0x80)
 | |
|                 {
 | |
|                     int32_t yintbuf=yintercept+(ystep>>1);
 | |
|                     if((yintbuf>>16)!=(yintercept>>16))
 | |
|                         goto passvert;
 | |
|                     if((word)yintbuf<doorposition[tilehit&0x7f])
 | |
|                         goto passvert;
 | |
|                     yintercept=yintbuf;
 | |
|                     xintercept=(xtile<<TILESHIFT)|0x8000;
 | |
|                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                     HitVertDoor();
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if(tilehit==64)
 | |
|                     {
 | |
|                         if(pwalldir==di_west || pwalldir==di_east)
 | |
|                         {
 | |
| 	                        int32_t yintbuf;
 | |
|                             int pwallposnorm;
 | |
|                             int pwallposinv;
 | |
|                             if(pwalldir==di_west)
 | |
|                             {
 | |
|                                 pwallposnorm = 64-pwallpos;
 | |
|                                 pwallposinv = pwallpos;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 pwallposnorm = pwallpos;
 | |
|                                 pwallposinv = 64-pwallpos;
 | |
|                             }
 | |
|                             if(pwalldir == di_east && xtile==pwallx && ((uint32_t)yintercept>>16)==pwally
 | |
|                                 || pwalldir == di_west && !(xtile==pwallx && ((uint32_t)yintercept>>16)==pwally))
 | |
|                             {
 | |
|                                 yintbuf=yintercept+((ystep*pwallposnorm)>>6);
 | |
|                                 if((yintbuf>>16)!=(yintercept>>16))
 | |
|                                     goto passvert;
 | |
| 
 | |
|                                 xintercept=(xtile<<TILESHIFT)+TILEGLOBAL-(pwallposinv<<10);
 | |
|                                 yintercept=yintbuf;
 | |
|                                 ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                 tilehit=pwalltile;
 | |
|                                 HitVertWall();
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 yintbuf=yintercept+((ystep*pwallposinv)>>6);
 | |
|                                 if((yintbuf>>16)!=(yintercept>>16))
 | |
|                                     goto passvert;
 | |
| 
 | |
|                                 xintercept=(xtile<<TILESHIFT)-(pwallposinv<<10);
 | |
|                                 yintercept=yintbuf;
 | |
|                                 ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                 tilehit=pwalltile;
 | |
|                                 HitVertWall();
 | |
|                             }
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             int pwallposi = pwallpos;
 | |
|                             if(pwalldir==di_north) pwallposi = 64-pwallpos;
 | |
|                             if(pwalldir==di_south && (word)yintercept<(pwallposi<<10)
 | |
|                                 || pwalldir==di_north && (word)yintercept>(pwallposi<<10))
 | |
|                             {
 | |
|                                 if(((uint32_t)yintercept>>16)==pwally && xtile==pwallx)
 | |
|                                 {
 | |
|                                     if(pwalldir==di_south && (int32_t)((word)yintercept)+ystep<(pwallposi<<10)
 | |
|                                             || pwalldir==di_north && (int32_t)((word)yintercept)+ystep>(pwallposi<<10))
 | |
|                                         goto passvert;
 | |
| 
 | |
|                                     if(pwalldir==di_south)
 | |
|                                         yintercept=(yintercept&0xffff0000)+(pwallposi<<10);
 | |
|                                     else
 | |
|                                         yintercept=(yintercept&0xffff0000)-TILEGLOBAL+(pwallposi<<10);
 | |
|                                     xintercept=xintercept-((xstep*(64-pwallpos))>>6);
 | |
|                                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitHorizWall();
 | |
|                                 }
 | |
|                                 else
 | |
|                                 {
 | |
|                                     texdelta = -(pwallposi<<10);
 | |
|                                     xintercept=xtile<<TILESHIFT;
 | |
|                                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitVertWall();
 | |
|                                 }
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 if(((uint32_t)yintercept>>16)==pwally && xtile==pwallx)
 | |
|                                 {
 | |
|                                     texdelta = -(pwallposi<<10);
 | |
|                                     xintercept=xtile<<TILESHIFT;
 | |
|                                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitVertWall();
 | |
|                                 }
 | |
|                                 else
 | |
|                                 {
 | |
|                                     if(pwalldir==di_south && (int32_t)((word)yintercept)+ystep>(pwallposi<<10)
 | |
|                                             || pwalldir==di_north && (int32_t)((word)yintercept)+ystep<(pwallposi<<10))
 | |
|                                         goto passvert;
 | |
| 
 | |
|                                     if(pwalldir==di_south)
 | |
|                                         yintercept=(yintercept&0xffff0000)-((64-pwallpos)<<10);
 | |
|                                     else
 | |
|                                         yintercept=(yintercept&0xffff0000)+((64-pwallpos)<<10);
 | |
|                                     xintercept=xintercept-((xstep*pwallpos)>>6);
 | |
|                                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitHorizWall();
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         xintercept=xtile<<TILESHIFT;
 | |
|                         ytile = (short) (yintercept >> TILESHIFT);
 | |
|                         HitVertWall();
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
| passvert:
 | |
|             *((byte *)spotvis+xspot)=1;
 | |
|             xtile+=xtilestep;
 | |
|             yintercept+=ystep;
 | |
|             xspot=(word)((xtile<<mapshift)+((uint32_t)yintercept>>16));
 | |
|         }
 | |
|         while(1);
 | |
|         continue;
 | |
| 
 | |
|         do
 | |
|         {
 | |
|             if(xtilestep==-1 && (xintercept>>16)<=xtile) goto vertentry;
 | |
|             if(xtilestep==1 && (xintercept>>16)>=xtile) goto vertentry;
 | |
| horizentry:
 | |
|             if((uint32_t)xintercept>mapwidth*65536-1 || (word)ytile>=mapheight)
 | |
|             {
 | |
|                 if(ytile<0) yintercept=0, ytile=0;
 | |
|                 else if(ytile>=mapheight) yintercept=mapheight<<TILESHIFT, ytile=mapheight-1;
 | |
|                 else ytile=(short) (yintercept >> TILESHIFT);
 | |
|                 if(xintercept<0) xintercept=0, xtile=0;
 | |
|                 else if(xintercept>=(mapwidth<<TILESHIFT)) xintercept=mapwidth<<TILESHIFT, xtile=mapwidth-1;
 | |
|                 xspot=0xffff;
 | |
|                 tilehit=0;
 | |
|                 HitVertBorder();
 | |
|                 break;
 | |
|             }
 | |
|             if(yspot>=maparea) break;
 | |
|             tilehit=((byte *)tilemap)[yspot];
 | |
|             if(tilehit)
 | |
|             {
 | |
|                 if(tilehit&0x80)
 | |
|                 {
 | |
|                     int32_t xintbuf=xintercept+(xstep>>1);
 | |
|                     if((xintbuf>>16)!=(xintercept>>16))
 | |
|                         goto passhoriz;
 | |
|                     if((word)xintbuf<doorposition[tilehit&0x7f])
 | |
|                         goto passhoriz;
 | |
|                     xintercept=xintbuf;
 | |
|                     yintercept=(ytile<<TILESHIFT)+0x8000;
 | |
|                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                     HitHorizDoor();
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if(tilehit==64)
 | |
|                     {
 | |
|                         if(pwalldir==di_north || pwalldir==di_south)
 | |
|                         {
 | |
|                             int32_t xintbuf;
 | |
|                             int pwallposnorm;
 | |
|                             int pwallposinv;
 | |
|                             if(pwalldir==di_north)
 | |
|                             {
 | |
|                                 pwallposnorm = 64-pwallpos;
 | |
|                                 pwallposinv = pwallpos;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 pwallposnorm = pwallpos;
 | |
|                                 pwallposinv = 64-pwallpos;
 | |
|                             }
 | |
|                             if(pwalldir == di_south && ytile==pwally && ((uint32_t)xintercept>>16)==pwallx
 | |
|                                 || pwalldir == di_north && !(ytile==pwally && ((uint32_t)xintercept>>16)==pwallx))
 | |
|                             {
 | |
|                                 xintbuf=xintercept+((xstep*pwallposnorm)>>6);
 | |
|                                 if((xintbuf>>16)!=(xintercept>>16))
 | |
|                                     goto passhoriz;
 | |
| 
 | |
|                                 yintercept=(ytile<<TILESHIFT)+TILEGLOBAL-(pwallposinv<<10);
 | |
|                                 xintercept=xintbuf;
 | |
|                                 xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                 tilehit=pwalltile;
 | |
|                                 HitHorizWall();
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 xintbuf=xintercept+((xstep*pwallposinv)>>6);
 | |
|                                 if((xintbuf>>16)!=(xintercept>>16))
 | |
|                                     goto passhoriz;
 | |
| 
 | |
|                                 yintercept=(ytile<<TILESHIFT)-(pwallposinv<<10);
 | |
|                                 xintercept=xintbuf;
 | |
|                                 xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                 tilehit=pwalltile;
 | |
|                                 HitHorizWall();
 | |
|                             }
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             int pwallposi = pwallpos;
 | |
|                             if(pwalldir==di_west) pwallposi = 64-pwallpos;
 | |
|                             if(pwalldir==di_east && (word)xintercept<(pwallposi<<10)
 | |
|                                     || pwalldir==di_west && (word)xintercept>(pwallposi<<10))
 | |
|                             {
 | |
|                                 if(((uint32_t)xintercept>>16)==pwallx && ytile==pwally)
 | |
|                                 {
 | |
|                                     if(pwalldir==di_east && (int32_t)((word)xintercept)+xstep<(pwallposi<<10)
 | |
|                                             || pwalldir==di_west && (int32_t)((word)xintercept)+xstep>(pwallposi<<10))
 | |
|                                         goto passhoriz;
 | |
| 
 | |
|                                     if(pwalldir==di_east)
 | |
|                                         xintercept=(xintercept&0xffff0000)+(pwallposi<<10);
 | |
|                                     else
 | |
|                                         xintercept=(xintercept&0xffff0000)-TILEGLOBAL+(pwallposi<<10);
 | |
|                                     yintercept=yintercept-((ystep*(64-pwallpos))>>6);
 | |
|                                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitVertWall();
 | |
|                                 }
 | |
|                                 else
 | |
|                                 {
 | |
|                                     texdelta = -(pwallposi<<10);
 | |
|                                     yintercept=ytile<<TILESHIFT;
 | |
|                                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitHorizWall();
 | |
|                                 }
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 if(((uint32_t)xintercept>>16)==pwallx && ytile==pwally)
 | |
|                                 {
 | |
|                                     texdelta = -(pwallposi<<10);
 | |
|                                     yintercept=ytile<<TILESHIFT;
 | |
|                                     xtile = (short) (xintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitHorizWall();
 | |
|                                 }
 | |
|                                 else
 | |
|                                 {
 | |
|                                     if(pwalldir==di_east && (int32_t)((word)xintercept)+xstep>(pwallposi<<10)
 | |
|                                             || pwalldir==di_west && (int32_t)((word)xintercept)+xstep<(pwallposi<<10))
 | |
|                                         goto passhoriz;
 | |
| 
 | |
|                                     if(pwalldir==di_east)
 | |
|                                         xintercept=(xintercept&0xffff0000)-((64-pwallpos)<<10);
 | |
|                                     else
 | |
|                                         xintercept=(xintercept&0xffff0000)+((64-pwallpos)<<10);
 | |
|                                     yintercept=yintercept-((ystep*pwallpos)>>6);
 | |
|                                     ytile = (short) (yintercept >> TILESHIFT);
 | |
|                                     tilehit=pwalltile;
 | |
|                                     HitVertWall();
 | |
|                                 }
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         yintercept=ytile<<TILESHIFT;
 | |
|                         xtile = (short) (xintercept >> TILESHIFT);
 | |
|                         HitHorizWall();
 | |
|                     }
 | |
|                 }
 | |
|                 break;
 | |
|             }
 | |
| passhoriz:
 | |
|             *((byte *)spotvis+yspot)=1;
 | |
|             ytile+=ytilestep;
 | |
|             xintercept+=xstep;
 | |
|             yspot=(word)((((uint32_t)xintercept>>16)<<mapshift)+ytile);
 | |
|         }
 | |
|         while(1);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*
 | |
| ====================
 | |
| =
 | |
| = WallRefresh
 | |
| =
 | |
| ====================
 | |
| */
 | |
| 
 | |
| void WallRefresh (void)
 | |
| {
 | |
|     xpartialdown = viewx&(TILEGLOBAL-1);
 | |
|     xpartialup = TILEGLOBAL-xpartialdown;
 | |
|     ypartialdown = viewy&(TILEGLOBAL-1);
 | |
|     ypartialup = TILEGLOBAL-ypartialdown;
 | |
| 
 | |
|     min_wallheight = viewheight;
 | |
|     lastside = -1;                  // the first pixel is on a new wall
 | |
|     AsmRefresh ();
 | |
|     ScalePost ();                   // no more optimization on last post
 | |
| }
 | |
| 
 | |
| void CalcViewVariables(void)
 | |
| {
 | |
|     viewangle = player->angle;
 | |
|     midangle = viewangle*(FINEANGLES/ANGLES);
 | |
|     viewsin = sintable[viewangle];
 | |
|     viewcos = costable[viewangle];
 | |
|     viewx = player->x - FixedMul(focallength,viewcos);
 | |
|     viewy = player->y + FixedMul(focallength,viewsin);
 | |
| 
 | |
|     focaltx = (short)(viewx>>TILESHIFT);
 | |
|     focalty = (short)(viewy>>TILESHIFT);
 | |
| 
 | |
|     viewtx = (short)(player->x >> TILESHIFT);
 | |
|     viewty = (short)(player->y >> TILESHIFT);
 | |
| }
 | |
| 
 | |
| //==========================================================================
 | |
| 
 | |
| /*
 | |
| ========================
 | |
| =
 | |
| = ThreeDRefresh
 | |
| =
 | |
| ========================
 | |
| */
 | |
| 
 | |
| void    ThreeDRefresh (void)
 | |
| {
 | |
| //
 | |
| // clear out the traced array
 | |
| //
 | |
|     memset(spotvis,0,maparea);
 | |
|     spotvis[player->tilex][player->tiley] = 1;       // Detect all sprites over player fix
 | |
| 
 | |
|     vbuf = VL_LockSurface(screenBuffer);
 | |
|     vbuf+=screenofs;
 | |
|     vbufPitch = bufferPitch;
 | |
| 
 | |
|     CalcViewVariables();
 | |
| 
 | |
| //
 | |
| // follow the walls from there to the right, drawing as we go
 | |
| //
 | |
|     VGAClearScreen ();
 | |
| #if defined(USE_FEATUREFLAGS) && defined(USE_STARSKY)
 | |
|     if(GetFeatureFlags() & FF_STARSKY)
 | |
|         DrawStarSky(vbuf, vbufPitch);
 | |
| #endif
 | |
| 
 | |
|     WallRefresh ();
 | |
| 
 | |
| #if defined(USE_FEATUREFLAGS) && defined(USE_PARALLAX)
 | |
|     if(GetFeatureFlags() & FF_PARALLAXSKY)
 | |
|         DrawParallax(vbuf, vbufPitch);
 | |
| #endif
 | |
| #if defined(USE_FEATUREFLAGS) && defined(USE_CLOUDSKY)
 | |
|     if(GetFeatureFlags() & FF_CLOUDSKY)
 | |
|         DrawClouds(vbuf, vbufPitch, min_wallheight);
 | |
| #endif
 | |
| #ifdef USE_FLOORCEILINGTEX
 | |
|     DrawFloorAndCeiling(vbuf, vbufPitch, min_wallheight);
 | |
| #endif
 | |
| 
 | |
| //
 | |
| // draw all the scaled images
 | |
| //
 | |
|     DrawScaleds();                  // draw scaled stuff
 | |
| 
 | |
| #if defined(USE_FEATUREFLAGS) && defined(USE_RAIN)
 | |
|     if(GetFeatureFlags() & FF_RAIN)
 | |
|         DrawRain(vbuf, vbufPitch);
 | |
| #endif
 | |
| #if defined(USE_FEATUREFLAGS) && defined(USE_SNOW)
 | |
|     if(GetFeatureFlags() & FF_SNOW)
 | |
|         DrawSnow(vbuf, vbufPitch);
 | |
| #endif
 | |
| 
 | |
|     DrawPlayerWeapon ();    // draw player's hands
 | |
| 
 | |
|     if(Keyboard[sc_Tab] && viewsize == 21 && gamestate.weapon != -1)
 | |
|         ShowActStatus();
 | |
| 
 | |
|     VL_UnlockSurface(screenBuffer);
 | |
|     vbuf = NULL;
 | |
| 
 | |
| //
 | |
| // show screen and time last cycle
 | |
| //
 | |
| 
 | |
|     if (fizzlein)
 | |
|     {
 | |
|         FizzleFade(screenBuffer, 0, 0, screenWidth, screenHeight, 20, false);
 | |
|         fizzlein = false;
 | |
| 
 | |
|         lasttimecount = GetTimeCount();          // don't make a big tic count
 | |
|     }
 | |
|     else
 | |
|     {
 | |
| #ifndef REMDEBUG
 | |
|         if (fpscounter)
 | |
|         {
 | |
|             fontnumber = 0;
 | |
|             SETFONTCOLOR(7,127);
 | |
|             PrintX=4; PrintY=1;
 | |
|             VWB_Bar(0,0,50,10,bordercol);
 | |
|             US_PrintSigned(fps);
 | |
|             US_Print(" fps");
 | |
|         }
 | |
| #endif
 | |
|         SDL_BlitSurface(screenBuffer, NULL, screen, NULL);
 | |
|         SDL_Flip(screen);
 | |
|     }
 | |
| 
 | |
| #ifndef REMDEBUG
 | |
|     if (fpscounter)
 | |
|     {
 | |
|         fps_frames++;
 | |
|         fps_time+=tics;
 | |
| 
 | |
|         if(fps_time>35)
 | |
|         {
 | |
|             fps_time-=35;
 | |
|             fps=fps_frames<<1;
 | |
|             fps_frames=0;
 | |
|         }
 | |
|     }
 | |
| #endif
 | |
| }
 |