iphone_render.c 54 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752
  1. /*
  2. Copyright (C) 2009-2011 id Software LLC, a ZeniMax Media company.
  3. Copyright (C) 2009 Id Software, Inc.
  4. This program is free software; you can redistribute it and/or
  5. modify it under the terms of the GNU General Public License
  6. as published by the Free Software Foundation; either version 2
  7. of the License, or (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program; if not, write to the Free Software
  14. Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
  15. */
  16. #include <stdio.h>
  17. #include <string.h>
  18. #include <math.h>
  19. #include "SDL_opengl.h"
  20. #include "doomtype.h"
  21. #include "w_wad.h"
  22. #include "m_argv.h"
  23. #include "d_event.h"
  24. #include "v_video.h"
  25. #include "doomstat.h"
  26. #include "r_bsp.h"
  27. #include "r_main.h"
  28. #include "r_draw.h"
  29. #include "r_sky.h"
  30. #include "r_plane.h"
  31. #include "r_data.h"
  32. #include "r_things.h"
  33. #include "r_fps.h"
  34. #include "p_maputl.h"
  35. #include "m_bbox.h"
  36. #include "lprintf.h"
  37. #include "gl_intern.h"
  38. #include "gl_struct.h"
  39. // If the Doom levels had been built with realistic visibility
  40. // taken into account for the sky areas, we could just draw the
  41. // sky first and then the walls, but that gives artifacts where
  42. // you see some sectors floating in the sky now. This causes
  43. // the walls to draw extended top and bottom sections for skies.
  44. #define SKYWALLS
  45. #define MINZ (FRACUNIT*4)
  46. #define BASEYCENTER 100
  47. #define MINZ_FLOAT 4
  48. typedef struct {
  49. GLTexture *tex;
  50. side_t *side;
  51. int flag; // GLDWF_TOP, GLDWF_M1S, etc
  52. } sortLine_t;
  53. #define MAX_SORT_LINES 4096
  54. sortLine_t sortLines[MAX_SORT_LINES];
  55. int numSortLines;
  56. typedef struct {
  57. GLTexture *texture;
  58. sector_t *sector;
  59. boolean ceiling;
  60. } sortSectorPlane_t;
  61. #define MAX_SECTOR_PLANES 1024
  62. sortSectorPlane_t sectorPlanes[MAX_SECTOR_PLANES];
  63. int numSectorPlanes;
  64. typedef struct {
  65. GLTexture *tex;
  66. } sortSprite_t;
  67. // Cleared to 0 at frame start.
  68. // Individual columns will be set to 1 as occluding segments are processed.
  69. // An occluding segment is either a one-sided line, a line that has a back
  70. // sector with equal floor and ceiling heights, a line with a back ceiling
  71. // height lower than the fron floor height, or a line with a back floor height
  72. // higher than the front ceiling height.
  73. // Entire nodes are culled when their bounds does not include a 0 column.
  74. // Individual line segments are culled when their span does not include 0 columns.
  75. // Sprites could be checked against it, but it may not be worth it.
  76. char occlusion[MAX_SCREENWIDTH+2]; // +2 for guard columns to avoid clamping
  77. // when the iphone is upside down, the occlusion segments are reversed
  78. boolean reversedLandscape;
  79. // this matrix is exactly what GL uses, but there will still
  80. // be floating point differences between the GPU and CPU
  81. float glMVPmatrix[16];
  82. // if any sector textures are the sky texture, we will draw the sky and
  83. // ignore those sector geometries
  84. boolean skyIsVisible;
  85. // these should really be initialized based on viewwidth somewhere...
  86. float halfWidthFloat = 240.0f;
  87. // used during debugging to isolate incorrect culling
  88. int failCount;
  89. // Some of the two sided line segments in the original game don't have a valid
  90. // texture, so stick something there instead of leaving a gaping hole in the world.
  91. GLTexture *defaultTexture;
  92. // just for the sky texture setup
  93. float yaw;
  94. // counters
  95. int c_occludedSprites;
  96. int c_sectors;
  97. int c_subsectors;
  98. // test options
  99. int testClear = 0;
  100. int testNewRenderer = 1;
  101. int showRenderTime;
  102. int blendAll;
  103. void BuildIndexedTriangles();
  104. void BuildSideSegs();
  105. void IR_MergeSectors( int fromSector, int intoSector ) {
  106. // E3M8 (and possibly others somewhere) has a bad sector
  107. // classification with two stray lines in sector 2 that
  108. // should be a part of sector 1. This makes both of the
  109. // sectors "broken" and unable to be properly tesselated.
  110. assert( (unsigned)fromSector < numsectors );
  111. assert( (unsigned)intoSector < numsectors );
  112. sector_t *fromSectorPtr = &sectors[fromSector];
  113. sector_t *intoSectorPtr = &sectors[intoSector];
  114. int moveLines = 0;
  115. for ( int i = 0 ; i < numlines ; i++ ) {
  116. if ( lines[i].frontsector == fromSectorPtr ) {
  117. moveLines++;
  118. } else if ( lines[i].backsector == fromSectorPtr ) {
  119. moveLines++;
  120. }
  121. }
  122. // add these lines to intoSector
  123. // Unfortunately, the sector->lines list is not allocated per-sector, but
  124. // is a single block for the entire level, so we can't realloc it. I'm
  125. // going to just let the new table leak.
  126. line_t **newLines = Z_Malloc( ( intoSectorPtr->linecount + moveLines ) * sizeof( *intoSectorPtr->lines ),
  127. PU_LEVEL,0);
  128. memcpy( newLines, intoSectorPtr->lines, intoSectorPtr->linecount * sizeof( *newLines ) );
  129. intoSectorPtr->lines = newLines;
  130. for ( int i = 0 ; i < numlines ; i++ ) {
  131. if ( lines[i].frontsector == fromSectorPtr ) {
  132. intoSectorPtr->lines[intoSectorPtr->linecount++] = &lines[i];
  133. lines[i].frontsector = intoSectorPtr;
  134. } else if ( lines[i].backsector == fromSectorPtr ) {
  135. intoSectorPtr->lines[intoSectorPtr->linecount++] = &lines[i];
  136. lines[i].backsector = intoSectorPtr;
  137. }
  138. }
  139. // change all the segs
  140. for ( int i = 0 ; i < numsegs ; i++ ) {
  141. if ( segs[i].frontsector == fromSectorPtr ) {
  142. segs[i].frontsector = intoSectorPtr;
  143. }
  144. if ( segs[i].backsector == fromSectorPtr ) {
  145. segs[i].backsector = intoSectorPtr;
  146. }
  147. }
  148. // change all the sides to point to the new one
  149. for ( int i = 0 ; i < numsides ; i++ ) {
  150. if ( sides[i].sector == fromSectorPtr ) {
  151. sides[i].sector = intoSectorPtr;
  152. }
  153. }
  154. // change all the subsectors to point to the new one
  155. for ( int i = 0 ; i < numsubsectors ; i++ ) {
  156. if ( subsectors[i].sector == fromSectorPtr ) {
  157. subsectors[i].sector = intoSectorPtr;
  158. }
  159. }
  160. // make fromSector vestigial so it doesn't get tesselated
  161. fromSectorPtr->linecount = 0;
  162. }
  163. void IR_InitLevel() {
  164. BuildIndexedTriangles(); // convert the loops into indexed triangles
  165. BuildSideSegs(); // create a seg_t for each side_t so we can draw the
  166. // unclipped versions that fit perfectly with the sectors
  167. // find something else used in the level for a default texture
  168. for ( int i = 0 ; i < numsides ; i++ ) {
  169. if ( sides[i].toptexture ) {
  170. defaultTexture=gld_RegisterTexture(sides[i].toptexture, true, false);
  171. if ( defaultTexture ) {
  172. break;
  173. }
  174. }
  175. }
  176. assert( defaultTexture );
  177. }
  178. float lightDistance = 10.0f; // in prBoom MAP_SCALE units, increasing this makes things get dimmer faster
  179. #define MAX_LIGHT_DROP 96
  180. float lightingVector[3]; // transform and scale [ x y 1 ] to get color units to subtract
  181. static int FadedLighting( float x, float y, int sectorLightLevel ) {
  182. // Ramp down the lightover lightDistance world units.
  183. // Triangles that extend across behind the view origin and past
  184. // the lightDistance clamping boundary will not have completely linear fading,
  185. // but nobody should notice.
  186. // A proportional drop in lighting sounds like a better idea, but
  187. // this linear drop seems to look nicer. It's not like Doom's
  188. // lighting is realistic in any case...
  189. int idist = x * lightingVector[0] + y * lightingVector[1] + lightingVector[2];
  190. if ( idist < 0 ) {
  191. idist = 0;
  192. } else if ( idist > MAX_LIGHT_DROP ) {
  193. idist = MAX_LIGHT_DROP;
  194. }
  195. sectorLightLevel -= idist;
  196. if ( sectorLightLevel < 0 ) {
  197. sectorLightLevel = 0;
  198. }
  199. if ( sectorLightLevel > 255 ) {
  200. sectorLightLevel = 255;
  201. }
  202. return sectorLightLevel | (sectorLightLevel<<8) | (sectorLightLevel<<16) | (255<<24);
  203. }
  204. //
  205. // IR_ProjectSprite
  206. // Generates a vissprite for a thing if it might be visible.
  207. //
  208. static void IR_ProjectSprite (mobj_t* thing, int lightlevel)
  209. {
  210. fixed_t gzt; // killough 3/27/98
  211. fixed_t tx;
  212. fixed_t xscale;
  213. int x1;
  214. int x2;
  215. spritedef_t *sprdef;
  216. spriteframe_t *sprframe;
  217. int lump;
  218. boolean flip;
  219. // transform the origin point
  220. fixed_t tr_x, tr_y;
  221. fixed_t fx, fy, fz;
  222. fixed_t gxt, gyt;
  223. fixed_t tz;
  224. int width;
  225. fx = thing->x;
  226. fy = thing->y;
  227. fz = thing->z;
  228. tr_x = fx - viewx;
  229. tr_y = fy - viewy;
  230. gxt = FixedMul(tr_x,viewcos);
  231. gyt = -FixedMul(tr_y,viewsin);
  232. tz = gxt-gyt;
  233. // thing is behind view plane?
  234. if (tz < MINZ)
  235. return;
  236. xscale = FixedDiv(projection, tz);
  237. gxt = -FixedMul(tr_x,viewsin);
  238. gyt = FixedMul(tr_y,viewcos);
  239. tx = -(gyt+gxt);
  240. // too far off the side?
  241. if (D_abs(tx)>(tz<<2))
  242. return;
  243. // decide which patch to use for sprite relative to player
  244. #ifdef RANGECHECK
  245. if ((unsigned) thing->sprite >= (unsigned)numsprites)
  246. I_Error ("R_ProjectSprite: Invalid sprite number %i", thing->sprite);
  247. #endif
  248. sprdef = &sprites[thing->sprite];
  249. #ifdef RANGECHECK
  250. if ((thing->frame&FF_FRAMEMASK) >= sprdef->numframes)
  251. I_Error ("R_ProjectSprite: Invalid sprite frame %i : %i", thing->sprite,
  252. thing->frame);
  253. #endif
  254. if (!sprdef->spriteframes)
  255. I_Error ("R_ProjectSprite: Missing spriteframes %i : %i", thing->sprite,
  256. thing->frame);
  257. sprframe = &sprdef->spriteframes[thing->frame & FF_FRAMEMASK];
  258. if (sprframe->rotate)
  259. {
  260. // choose a different rotation based on player view
  261. // JDC: this could be better...
  262. angle_t ang = R_PointToAngle(fx, fy);
  263. unsigned rot = (ang-thing->angle+(unsigned)(ANG45/2)*9)>>29;
  264. lump = sprframe->lump[rot];
  265. flip = (boolean) sprframe->flip[rot];
  266. }
  267. else
  268. {
  269. // use single rotation for all views
  270. lump = sprframe->lump[0];
  271. flip = (boolean) sprframe->flip[0];
  272. }
  273. {
  274. const rpatch_t* patch = R_CachePatchNum(lump+firstspritelump);
  275. /* calculate edges of the shape
  276. * cph 2003/08/1 - fraggle points out that this offset must be flipped
  277. * if the sprite is flipped; e.g. FreeDoom imp is messed up by this. */
  278. if (flip) {
  279. tx -= (patch->width - patch->leftoffset) << FRACBITS;
  280. } else {
  281. tx -= patch->leftoffset << FRACBITS;
  282. }
  283. x1 = (centerxfrac + FixedMul(tx,xscale)) >> FRACBITS;
  284. tx += patch->width<<FRACBITS;
  285. x2 = ((centerxfrac + FixedMul (tx,xscale) ) >> FRACBITS) - 1;
  286. gzt = fz + (patch->topoffset << FRACBITS);
  287. width = patch->width;
  288. // JDC: we don't care if they never get freed,
  289. // so don't bother changing the zone tag status each time
  290. //R_UnlockPatchNum(lump+firstspritelump);
  291. }
  292. // off the side?
  293. if (x1 > viewwidth || x2 < 0)
  294. return;
  295. // killough 4/9/98: clip things which are out of view due to height
  296. // e6y: fix of hanging decoration disappearing in Batman Doom MAP02
  297. // centeryfrac -> viewheightfrac
  298. if (fz > viewz + FixedDiv(viewheightfrac, xscale) ||
  299. gzt < viewz - FixedDiv(viewheightfrac-viewheight, xscale))
  300. return;
  301. // JDC: clip to the occlusio buffer
  302. int testLow = x1 < 0 ? 0 : x1;
  303. int testHigh = x2 >= viewwidth ? viewwidth - 1 : x2;
  304. if ( reversedLandscape ) {
  305. testLow = viewwidth-1-testLow;
  306. testHigh = viewwidth-1-testHigh;
  307. }
  308. if ( !memchr( occlusion+testLow, 0, testHigh - testLow ) ) {
  309. c_occludedSprites++;
  310. return;
  311. }
  312. // ------------ gld_AddSprite ----------
  313. mobj_t *pSpr= thing;
  314. GLSprite sprite;
  315. float voff,hoff;
  316. sprite.scale= FixedDiv(projectiony, tz);
  317. if (pSpr->frame & FF_FULLBRIGHT)
  318. sprite.light = 255;
  319. else
  320. sprite.light = pSpr->subsector->sector->lightlevel+(extralight<<5);
  321. sprite.cm=CR_LIMIT+(int)((pSpr->flags & MF_TRANSLATION) >> (MF_TRANSSHIFT));
  322. sprite.gltexture=gld_RegisterPatch(lump+firstspritelump,sprite.cm);
  323. if (!sprite.gltexture)
  324. return;
  325. sprite.shadow = (pSpr->flags & MF_SHADOW) != 0;
  326. sprite.trans = (pSpr->flags & MF_TRANSLUCENT) != 0;
  327. if (movement_smooth)
  328. {
  329. sprite.x = (float)(-pSpr->PrevX + FixedMul (tic_vars.frac, -pSpr->x - (-pSpr->PrevX)))/MAP_SCALE;
  330. sprite.y = (float)(pSpr->PrevZ + FixedMul (tic_vars.frac, pSpr->z - pSpr->PrevZ))/MAP_SCALE;
  331. sprite.z = (float)(pSpr->PrevY + FixedMul (tic_vars.frac, pSpr->y - pSpr->PrevY))/MAP_SCALE;
  332. }
  333. else
  334. {
  335. sprite.x=-(float)pSpr->x/MAP_SCALE;
  336. sprite.y= (float)pSpr->z/MAP_SCALE;
  337. sprite.z= (float)pSpr->y/MAP_SCALE;
  338. }
  339. sprite.vt=0.0f;
  340. sprite.vb=(float)sprite.gltexture->height/(float)sprite.gltexture->tex_height;
  341. if (flip)
  342. {
  343. sprite.ul=0.0f;
  344. sprite.ur=(float)sprite.gltexture->width/(float)sprite.gltexture->tex_width;
  345. }
  346. else
  347. {
  348. sprite.ul=(float)sprite.gltexture->width/(float)sprite.gltexture->tex_width;
  349. sprite.ur=0.0f;
  350. }
  351. hoff=(float)sprite.gltexture->leftoffset/(float)(MAP_COEFF);
  352. voff=(float)sprite.gltexture->topoffset/(float)(MAP_COEFF);
  353. sprite.x1=hoff-((float)sprite.gltexture->realtexwidth/(float)(MAP_COEFF));
  354. sprite.x2=hoff;
  355. sprite.y1=voff;
  356. sprite.y2=voff-((float)sprite.gltexture->realtexheight/(float)(MAP_COEFF));
  357. // JDC: don't let sprites poke below the ground level.
  358. // Software rendering Doom didn't use depth buffering,
  359. // so sprites always got drawn on top of the flat they
  360. // were on, but in GL they tend to get a couple pixel
  361. // rows clipped off.
  362. if ( sprite.y2 < 0 ) {
  363. sprite.y1 -= sprite.y2;
  364. sprite.y2 = 0;
  365. }
  366. if (gld_drawinfo.num_sprites>=gld_drawinfo.max_sprites)
  367. {
  368. gld_drawinfo.max_sprites+=128;
  369. gld_drawinfo.sprites=Z_Realloc(gld_drawinfo.sprites,gld_drawinfo.max_sprites*sizeof(GLSprite),PU_LEVEL,0);
  370. }
  371. gld_drawinfo.sprites[gld_drawinfo.num_sprites++]=sprite;
  372. }
  373. // JDC: removed the 0.001f epsilons that were presumably added
  374. // to try to hide T-junction cracks, but now that we are drawing
  375. // source lines instead of clipped segs, it is a non-problem.
  376. #define LINE seg->linedef
  377. #define CALC_Y_VALUES(w, lineheight, floor_height, ceiling_height)\
  378. (w).ytop=((float)(ceiling_height)/(float)MAP_SCALE);\
  379. (w).ybottom=((float)(floor_height)/(float)MAP_SCALE);\
  380. lineheight=((float)fabs(((ceiling_height)/(float)FRACUNIT)-((floor_height)/(float)FRACUNIT)))
  381. #define OU(w,seg) (((float)((seg)->sidedef->textureoffset+(seg)->offset)/(float)FRACUNIT)/(float)(w).gltexture->buffer_width)
  382. #define OV(w,seg) (((float)((seg)->sidedef->rowoffset)/(float)FRACUNIT)/(float)(w).gltexture->buffer_height)
  383. #define OV_PEG(w,seg,v_offset) (OV((w),(seg))-(((float)(v_offset)/(float)FRACUNIT)/(float)(w).gltexture->buffer_height))
  384. #define CALC_TEX_VALUES_TOP(w, seg, peg, linelength, lineheight)\
  385. (w).flag=GLDWF_TOP;\
  386. (w).ul=OU((w),(seg))+(0.0f);\
  387. (w).ur=OU((w),(seg))+((linelength)/(float)(w).gltexture->buffer_width);\
  388. (peg)?\
  389. (\
  390. (w).vb=OV((w),(seg))+((float)(w).gltexture->height/(float)(w).gltexture->tex_height),\
  391. (w).vt=((w).vb-((float)(lineheight)/(float)(w).gltexture->buffer_height))\
  392. ):(\
  393. (w).vt=OV((w),(seg))+(0.0f),\
  394. (w).vb=OV((w),(seg))+((float)(lineheight)/(float)(w).gltexture->buffer_height)\
  395. )
  396. #define CALC_TEX_VALUES_MIDDLE1S(w, seg, peg, linelength, lineheight)\
  397. (w).flag=GLDWF_M1S;\
  398. (w).ul=OU((w),(seg))+(0.0f);\
  399. (w).ur=OU((w),(seg))+((linelength)/(float)(w).gltexture->buffer_width);\
  400. (peg)?\
  401. (\
  402. (w).vb=OV((w),(seg))+((float)(w).gltexture->height/(float)(w).gltexture->tex_height),\
  403. (w).vt=((w).vb-((float)(lineheight)/(float)(w).gltexture->buffer_height))\
  404. ):(\
  405. (w).vt=OV((w),(seg))+(0.0f),\
  406. (w).vb=OV((w),(seg))+((float)(lineheight)/(float)(w).gltexture->buffer_height)\
  407. )
  408. #define CALC_TEX_VALUES_MIDDLE2S(w, seg, peg, linelength, lineheight)\
  409. (w).flag=GLDWF_M2S;\
  410. (w).ul=OU((w),(seg))+(0.0f);\
  411. (w).ur=OU((w),(seg))+((linelength)/(float)(w).gltexture->buffer_width);\
  412. (peg)?\
  413. (\
  414. (w).vb=((float)(w).gltexture->height/(float)(w).gltexture->tex_height),\
  415. (w).vt=((w).vb-((float)(lineheight)/(float)(w).gltexture->buffer_height))\
  416. ):(\
  417. (w).vt=(0.0f),\
  418. (w).vb=((float)(lineheight)/(float)(w).gltexture->buffer_height)\
  419. )
  420. #define CALC_TEX_VALUES_BOTTOM(w, seg, peg, linelength, lineheight, v_offset)\
  421. (w).flag=GLDWF_BOT;\
  422. (w).ul=OU((w),(seg))+(0.0f);\
  423. (w).ur=OU((w),(seg))+((linelength)/(float)(w).gltexture->realtexwidth);\
  424. (peg)?\
  425. (\
  426. (w).vb=OV_PEG((w),(seg),(v_offset))+((float)(w).gltexture->height/(float)(w).gltexture->tex_height),\
  427. (w).vt=((w).vb-((float)(lineheight)/(float)(w).gltexture->buffer_height))\
  428. ):(\
  429. (w).vt=OV((w),(seg))+(0.0f),\
  430. (w).vb=OV((w),(seg))+((float)(lineheight)/(float)(w).gltexture->buffer_height)\
  431. )
  432. // e6y
  433. // Sky textures with a zero index should be forced
  434. // See third episode of requiem.wad
  435. #define SKYTEXTURE_PRBOOM(sky1,sky2)\
  436. if ((sky1) & PL_SKYFLAT)\
  437. {\
  438. const line_t *l = &lines[sky1 & ~PL_SKYFLAT];\
  439. const side_t *s = *l->sidenum + sides;\
  440. wall.gltexture=gld_RegisterTexture(texturetranslation[s->toptexture], false, texturetranslation[s->toptexture]==skytexture);\
  441. wall.skyyaw=-2.0f*((-(float)((viewangle+s->textureoffset)>>ANGLETOFINESHIFT)*360.0f/FINEANGLES)/90.0f);\
  442. wall.skyymid = 200.0f/319.5f*(((float)s->rowoffset/(float)FRACUNIT - 28.0f)/100.0f);\
  443. wall.flag = l->special==272 ? GLDWF_SKY : GLDWF_SKYFLIP;\
  444. }\
  445. else\
  446. if ((sky2) & PL_SKYFLAT)\
  447. {\
  448. const line_t *l = &lines[sky2 & ~PL_SKYFLAT];\
  449. const side_t *s = *l->sidenum + sides;\
  450. wall.gltexture=gld_RegisterTexture(texturetranslation[s->toptexture], false, texturetranslation[s->toptexture]==skytexture);\
  451. wall.skyyaw=-2.0f*((-(float)((viewangle+s->textureoffset)>>ANGLETOFINESHIFT)*360.0f/FINEANGLES)/90.0f);\
  452. wall.skyymid = 200.0f/319.5f*(((float)s->rowoffset/(float)FRACUNIT - 28.0f)/100.0f);\
  453. wall.flag = l->special==272 ? GLDWF_SKY : GLDWF_SKYFLIP;\
  454. }\
  455. else\
  456. {\
  457. wall.gltexture=gld_RegisterTexture(skytexture, false, true);\
  458. wall.skyyaw=-2.0f*((yaw+90.0f)/90.0f);\
  459. wall.skyymid = 200.0f/319.5f*((100.0f)/100.0f);\
  460. wall.flag = GLDWF_SKY;\
  461. };
  462. #define SKYTEXTURE(sky1,sky2)\
  463. wall.gltexture=NULL;\
  464. wall.flag = GLDWF_SKY;
  465. #define ADDWALL(wall)\
  466. {\
  467. if (gld_drawinfo.num_walls>=gld_drawinfo.max_walls)\
  468. {\
  469. gld_drawinfo.max_walls+=128;\
  470. gld_drawinfo.walls=Z_Realloc(gld_drawinfo.walls,gld_drawinfo.max_walls*sizeof(GLWall),PU_LEVEL,0);\
  471. }\
  472. gld_drawinfo.walls[gld_drawinfo.num_walls++]=*wall;\
  473. };
  474. extern GLSeg *gl_segs;
  475. extern byte rendermarker;
  476. extern byte *segrendered;
  477. void IR_AddWall(seg_t *seg)
  478. {
  479. GLWall wall;
  480. GLTexture *temptex;
  481. sector_t *frontsector;
  482. sector_t *backsector;
  483. float lineheight;
  484. int rellight = 0;
  485. wall.glseg=NULL;
  486. wall.side = seg->sidedef;
  487. frontsector = seg->frontsector;
  488. // JDC: improve this lighting tweak
  489. rellight = seg->linedef->dx==0? +8 : seg->linedef->dy==0 ? -8 : 0;
  490. int light = frontsector->lightlevel+rellight+(extralight<<5);
  491. wall.light = MAX(MIN((light),255),0);
  492. wall.alpha=1.0f;
  493. wall.gltexture=NULL;
  494. if (!seg->backsector) /* onesided */
  495. {
  496. #ifdef SKYWALLS
  497. if (frontsector->ceilingpic==skyflatnum)
  498. {
  499. wall.ytop=255.0f;
  500. wall.ybottom=(float)frontsector->ceilingheight/MAP_SCALE;
  501. SKYTEXTURE(frontsector->sky,frontsector->sky);
  502. ADDWALL(&wall);
  503. }
  504. if (frontsector->floorpic==skyflatnum)
  505. {
  506. wall.ytop=(float)frontsector->floorheight/MAP_SCALE;
  507. wall.ybottom=-255.0f;
  508. SKYTEXTURE(frontsector->sky,frontsector->sky);
  509. ADDWALL(&wall);
  510. }
  511. #endif
  512. temptex=gld_RegisterTexture(texturetranslation[seg->sidedef->midtexture], true, false);
  513. if (temptex)
  514. {
  515. wall.gltexture=temptex;
  516. CALC_Y_VALUES(wall, lineheight, frontsector->floorheight, frontsector->ceilingheight);
  517. CALC_TEX_VALUES_MIDDLE1S(
  518. wall, seg, (LINE->flags & ML_DONTPEGBOTTOM)>0,
  519. seg->length, lineheight
  520. );
  521. ADDWALL(&wall);
  522. }
  523. }
  524. else /* twosided */
  525. {
  526. int floor_height,ceiling_height;
  527. backsector=seg->backsector;
  528. /* toptexture */
  529. ceiling_height=frontsector->ceilingheight;
  530. floor_height=backsector->ceilingheight;
  531. #ifdef SKYWALLS
  532. if (frontsector->ceilingpic==skyflatnum)
  533. {
  534. wall.ytop=255.0f;
  535. if (
  536. // e6y
  537. // Fix for HOM in the starting area on Memento Mori map29 and on map30.
  538. // old code: (backsector->ceilingheight==backsector->floorheight) &&
  539. (backsector->ceilingheight==backsector->floorheight||(backsector->ceilingheight<=frontsector->floorheight)) &&
  540. (backsector->ceilingpic==skyflatnum)
  541. )
  542. {
  543. wall.ybottom=(float)backsector->floorheight/MAP_SCALE;
  544. SKYTEXTURE(frontsector->sky,backsector->sky);
  545. ADDWALL(&wall);
  546. }
  547. else
  548. {
  549. if ( (texturetranslation[seg->sidedef->toptexture]!=NO_TEXTURE) )
  550. {
  551. // e6y
  552. // It corrects some problem with sky, but I do not remember which one
  553. // old code: wall.ybottom=(float)frontsector->ceilingheight/MAP_SCALE;
  554. wall.ybottom=(float)MAX(frontsector->ceilingheight,backsector->ceilingheight)/MAP_SCALE;
  555. SKYTEXTURE(frontsector->sky,backsector->sky);
  556. ADDWALL(&wall);
  557. }
  558. else
  559. if ( (backsector->ceilingheight <= frontsector->floorheight) ||
  560. (backsector->ceilingpic != skyflatnum) )
  561. {
  562. wall.ybottom=(float)backsector->ceilingheight/MAP_SCALE;
  563. SKYTEXTURE(frontsector->sky,backsector->sky);
  564. ADDWALL(&wall);
  565. }
  566. }
  567. }
  568. #endif
  569. if (floor_height<ceiling_height)
  570. {
  571. if (!((frontsector->ceilingpic==skyflatnum) && (backsector->ceilingpic==skyflatnum)))
  572. {
  573. temptex=gld_RegisterTexture(texturetranslation[seg->sidedef->toptexture], true, false);
  574. if ( !temptex ) {
  575. temptex = defaultTexture; // it seems some line segments have bad data...
  576. }
  577. wall.gltexture=temptex;
  578. CALC_Y_VALUES(wall, lineheight, floor_height, ceiling_height);
  579. CALC_TEX_VALUES_TOP(
  580. wall, seg, (LINE->flags & (ML_DONTPEGBOTTOM | ML_DONTPEGTOP))==0,
  581. seg->length, lineheight
  582. );
  583. ADDWALL(&wall);
  584. }
  585. }
  586. /* midtexture */
  587. //e6y
  588. if (comp[comp_maskedanim])
  589. temptex=gld_RegisterTexture(seg->sidedef->midtexture, true, false);
  590. else
  591. // e6y
  592. // Animated middle textures with a zero index should be forced
  593. // See spacelab.wad (http://www.doomworld.com/idgames/index.php?id=6826)
  594. temptex=gld_RegisterTexture(texturetranslation[seg->sidedef->midtexture], true, true);
  595. if (temptex && seg->sidedef->midtexture != NO_TEXTURE)
  596. {
  597. wall.gltexture=temptex;
  598. if ( (LINE->flags & ML_DONTPEGBOTTOM) >0)
  599. {
  600. if (seg->backsector->ceilingheight<=seg->frontsector->floorheight)
  601. goto bottomtexture;
  602. floor_height=MAX(seg->frontsector->floorheight,seg->backsector->floorheight)+(seg->sidedef->rowoffset);
  603. ceiling_height=floor_height+(wall.gltexture->realtexheight<<FRACBITS);
  604. }
  605. else
  606. {
  607. if (seg->backsector->ceilingheight<=seg->frontsector->floorheight)
  608. goto bottomtexture;
  609. ceiling_height=MIN(seg->frontsector->ceilingheight,seg->backsector->ceilingheight)+(seg->sidedef->rowoffset);
  610. floor_height=ceiling_height-(wall.gltexture->realtexheight<<FRACBITS);
  611. }
  612. // e6y
  613. // The fix for wrong middle texture drawing
  614. // if it exceeds the boundaries of its floor and ceiling
  615. /*CALC_Y_VALUES(wall, lineheight, floor_height, ceiling_height);
  616. CALC_TEX_VALUES_MIDDLE2S(
  617. wall, seg, (LINE->flags & ML_DONTPEGBOTTOM)>0,
  618. segs[seg->iSegID].length, lineheight
  619. );*/
  620. {
  621. int floormax, ceilingmin, linelen;
  622. float mip;
  623. mip = (float)wall.gltexture->realtexheight/(float)wall.gltexture->buffer_height;
  624. // if ( (texturetranslation[seg->sidedef->bottomtexture]!=R_TextureNumForName("-")) )
  625. if (seg->sidedef->bottomtexture)
  626. floormax=MAX(seg->frontsector->floorheight,seg->backsector->floorheight);
  627. else
  628. floormax=floor_height;
  629. if (seg->sidedef->toptexture)
  630. ceilingmin=MIN(seg->frontsector->ceilingheight,seg->backsector->ceilingheight);
  631. else
  632. ceilingmin=ceiling_height;
  633. linelen=abs(ceiling_height-floor_height);
  634. wall.ytop=((float)MIN(ceilingmin, ceiling_height)/(float)MAP_SCALE);
  635. wall.ybottom=((float)MAX(floormax, floor_height)/(float)MAP_SCALE);
  636. wall.flag=GLDWF_M2S;
  637. wall.ul=OU((wall),(seg))+(0.0f);
  638. wall.ur=OU(wall,(seg))+((seg->length)/(float)wall.gltexture->buffer_width);
  639. if (floormax<=floor_height)
  640. wall.vb=mip*1.0f;
  641. else
  642. wall.vb=mip*((float)(ceiling_height - floormax))/linelen;
  643. if (ceilingmin>=ceiling_height)
  644. wall.vt=0.0f;
  645. else
  646. wall.vt=mip*((float)(ceiling_height - ceilingmin))/linelen;
  647. }
  648. wall.alpha=1.0f;
  649. ADDWALL(&wall);
  650. }
  651. bottomtexture:
  652. /* bottomtexture */
  653. ceiling_height=backsector->floorheight;
  654. floor_height=frontsector->floorheight;
  655. #ifdef SKYWALLS
  656. if (frontsector->floorpic==skyflatnum)
  657. {
  658. wall.ybottom=-255.0f;
  659. if (
  660. (backsector->ceilingheight==backsector->floorheight) &&
  661. (backsector->floorpic==skyflatnum)
  662. )
  663. {
  664. wall.ytop=(float)backsector->floorheight/MAP_SCALE;
  665. SKYTEXTURE(frontsector->sky,backsector->sky);
  666. ADDWALL(&wall);
  667. }
  668. else
  669. {
  670. if ( (texturetranslation[seg->sidedef->bottomtexture]!=NO_TEXTURE) )
  671. {
  672. wall.ytop=(float)frontsector->floorheight/MAP_SCALE;
  673. SKYTEXTURE(frontsector->sky,backsector->sky);
  674. ADDWALL(&wall);
  675. }
  676. else
  677. if ( (backsector->floorheight >= frontsector->ceilingheight) ||
  678. (backsector->floorpic != skyflatnum) )
  679. {
  680. wall.ytop=(float)backsector->floorheight/MAP_SCALE;
  681. SKYTEXTURE(frontsector->sky,backsector->sky);
  682. ADDWALL(&wall);
  683. }
  684. }
  685. }
  686. #endif
  687. if (floor_height<ceiling_height)
  688. {
  689. temptex=gld_RegisterTexture(texturetranslation[seg->sidedef->bottomtexture], true, false);
  690. if ( !temptex ) {
  691. temptex = defaultTexture; // it seems some line segments have bad data...
  692. }
  693. wall.gltexture=temptex;
  694. CALC_Y_VALUES(wall, lineheight, floor_height, ceiling_height);
  695. CALC_TEX_VALUES_BOTTOM(
  696. wall, seg, (LINE->flags & ML_DONTPEGBOTTOM)>0,
  697. seg->length, lineheight,
  698. floor_height-frontsector->ceilingheight
  699. );
  700. ADDWALL(&wall);
  701. }
  702. }
  703. }
  704. #undef LINE
  705. #undef CALC_Y_VALUES
  706. #undef OU
  707. #undef OV
  708. #undef OV_PEG
  709. #undef CALC_TEX_VALUES_TOP
  710. #undef CALC_TEX_VALUES_MIDDLE1S
  711. #undef CALC_TEX_VALUES_MIDDLE2S
  712. #undef CALC_TEX_VALUES_BOTTOM
  713. #undef SKYTEXTURE
  714. #undef ADDWALL
  715. /*
  716. TransformAndClipSegment
  717. Converts a world coordinate line segment to screen space.
  718. Returns false if the segment is off screen.
  719. There would be some savings if all the points in a subsector
  720. were transformed and clip tested as a unit, instead of as discrete segments.
  721. */
  722. boolean TransformAndClipSegment( float v[2][2], float ends[2] ) {
  723. float clip[2][4];
  724. // if we are in iphone reverse-landscape mode, we need
  725. // to flip the coordinates around
  726. float *v0, *v1;
  727. if ( reversedLandscape ) {
  728. v0 = v[1];
  729. v1 = v[0];
  730. } else {
  731. v0 = v[0];
  732. v1 = v[1];
  733. }
  734. // transform from model to clip space
  735. // because the iPhone screen hardware is portrait mode,
  736. // we need to look at the Y axis for the segment ends,
  737. // not the X axis.
  738. clip[0][1] = v0[0] * glMVPmatrix[1] + v0[1] * glMVPmatrix[2*4+1] + glMVPmatrix[3*4+1];
  739. clip[0][3] = v0[0] * glMVPmatrix[3] + v0[1] * glMVPmatrix[2*4+3] + glMVPmatrix[3*4+3];
  740. clip[1][1] = v1[0] * glMVPmatrix[1] + v1[1] * glMVPmatrix[2*4+1] + glMVPmatrix[3*4+1];
  741. clip[1][3] = v1[0] * glMVPmatrix[3] + v1[1] * glMVPmatrix[2*4+3] + glMVPmatrix[3*4+3];
  742. float d0, d1;
  743. // clip to the near plane
  744. float nearClip = 0.01f;
  745. d0 = clip[0][3] - nearClip;
  746. d1 = clip[1][3] - nearClip;
  747. if ( d0 < 0 && d1 < 0 ) {
  748. // near clipped
  749. return false;
  750. }
  751. if ( d0 < 0 ) {
  752. float f = d0 / ( d0 - d1 );
  753. clip[0][1] = clip[0][1] + f * ( clip[1][1] - clip[0][1] );
  754. clip[0][3] = nearClip;
  755. } else if ( d1 < 0 ) {
  756. float f = d1 / ( d1 - d0 );
  757. clip[1][1] = clip[1][1] + f * ( clip[0][1] - clip[1][1] );
  758. clip[1][3] = nearClip;
  759. }
  760. if ( clip[0][1] > clip[0][3] ) {
  761. // entire segment is off the right side of the screen
  762. return false;
  763. }
  764. if ( clip[1][1] < -clip[1][3] ) {
  765. // entire segment is off the left side of the screen
  766. return false;
  767. }
  768. // project
  769. for ( int i = 0 ; i < 2 ; i++ ) {
  770. float x = viewwidth * ( ( clip[i][1] / clip[i][3] ) * 0.5 + 0.5 );
  771. if ( x < 0 ) {
  772. x = 0;
  773. } else if ( x > viewwidth ) {
  774. x = viewwidth;
  775. }
  776. ends[i] = x;
  777. }
  778. // part of the segment is on screen
  779. return true;
  780. }
  781. /*
  782. IR_Subsector
  783. All possible culling should be performed here, but most calculations should be
  784. deferred until draw time, rather than storing intermediate values that are
  785. later referenced.
  786. Don't make this static, or the compiler inlines it in the recursive node
  787. function, which bloats the stack.
  788. */
  789. void IR_Subsector(int num)
  790. {
  791. subsector_t *sub = &subsectors[num];
  792. c_subsectors++;
  793. // at this point we know that at least part of the subsector is
  794. // not covered in the occlusion array
  795. // if the sector that this subsector is a part of has not already had its
  796. // planes and sprites added, add them now.
  797. sector_t *frontsector = sub->sector;
  798. int lightlevel = frontsector->lightlevel+(extralight<<5);
  799. // There can be several subsectors in each sector due to non-convex
  800. // sectors or BSP splits, but we draw the floors, ceilings and lines
  801. // with a single draw call for the entire thing, so ensure that they
  802. // are only added once per frame.
  803. if ( frontsector->validcount != validcount ) {
  804. frontsector->validcount = validcount;
  805. c_sectors++;
  806. GLFlat flat;
  807. flat.sectornum = frontsector->iSectorID;
  808. flat.light = lightlevel;
  809. flat.uoffs= 0; // no support in standard doom
  810. flat.voffs= 0;
  811. if ( frontsector->floorheight < viewz ) {
  812. if (frontsector->floorpic == skyflatnum) {
  813. skyIsVisible = true;
  814. } else {
  815. // get the texture. flattranslation is maintained by doom and
  816. // contains the number of the current animation frame
  817. GLTexture *tex = gld_RegisterFlat(flattranslation[frontsector->floorpic], true);
  818. if ( tex ) {
  819. sectorPlanes[numSectorPlanes].texture = tex;
  820. sectorPlanes[numSectorPlanes].ceiling = false;
  821. sectorPlanes[numSectorPlanes].sector = frontsector;
  822. numSectorPlanes++;
  823. }
  824. }
  825. }
  826. if ( frontsector->ceilingheight > viewz ) {
  827. if (frontsector->ceilingpic == skyflatnum) {
  828. skyIsVisible = true;
  829. } else {
  830. // get the texture. flattranslation is maintained by doom and
  831. // contains the number of the current animation frame
  832. GLTexture *tex = gld_RegisterFlat(flattranslation[frontsector->ceilingpic], true);
  833. if ( tex ) {
  834. sectorPlanes[numSectorPlanes].texture = tex;
  835. sectorPlanes[numSectorPlanes].ceiling = true;
  836. sectorPlanes[numSectorPlanes].sector = frontsector;
  837. numSectorPlanes++;
  838. }
  839. }
  840. }
  841. // Add all the sprites in this sector.
  842. // It would be better if they were linked into all the subsectors, because
  843. // we could do more accurate occlusion culling. With non-convex sectors,
  844. // occasionally a sprite will be added in a rear portion of the sector that
  845. // would have been occluded away if everything was done in BSP subsector order.
  846. for ( mobj_t *thing = frontsector->thinglist; thing; thing = thing->snext) {
  847. IR_ProjectSprite( thing, lightlevel );
  848. }
  849. }
  850. // If a segment in this subsector is not fully occluded, mark
  851. // the line that it is a part of as needing to be drawn. Because
  852. // we are using a depth buffer, we can draw complete line segments
  853. // instead of just segments.
  854. for ( int i = 0 ; i < sub->numlines ; i++ ) {
  855. seg_t *seg = &segs[sub->firstline+i];
  856. line_t *line = seg->linedef;
  857. // Determine if it will completely occlude farther objects.
  858. // Given that changing sector heights is much less common than
  859. // traversing lines during every render, it would be marginally better if
  860. // lines had an "occluder" flag on them that was updated as sectors
  861. // moved, but it hardly matters.
  862. boolean occluder;
  863. if ( seg->backsector == NULL ||
  864. seg->backsector->floorheight >= seg->backsector->ceilingheight ||
  865. seg->backsector->floorheight >= seg->frontsector->ceilingheight ||
  866. seg->backsector->ceilingheight <= seg->frontsector->floorheight ) {
  867. // this segment can't be seen past, so fill in the occlusion table
  868. occluder = true;
  869. } else {
  870. // If the line has already been made visible and we don't need to
  871. // update the occlusion buffer, we don't need to do anything else here.
  872. // This happens when a line is split into multiple segs, and also
  873. // when the line is reached from the backsector. In the backsector
  874. // case, it would be back-face culled, but this test throws it out
  875. // without having to transform and clip the ends.
  876. if ( line->validcount == validcount ) {
  877. continue;
  878. }
  879. // check to see if the seg won't draw any walls at all
  880. // we won't fill in the occlusion table for this
  881. occluder = false;
  882. }
  883. // transform and clip the two endpoints
  884. float v[2][2];
  885. float floatEnds[2];
  886. v[0][0] = -seg->v1->x/MAP_SCALE;
  887. v[0][1] = seg->v1->y/MAP_SCALE;
  888. v[1][0] = -seg->v2->x/MAP_SCALE;
  889. v[1][1] = seg->v2->y/MAP_SCALE;
  890. if ( !TransformAndClipSegment( v, floatEnds ) ) {
  891. // the line is off to the side or facing away
  892. continue;
  893. }
  894. // Allow segs that we consider to be slightly back
  895. // facing to still pass through, because GPU floating
  896. // point calculations may not see them exactly the same.
  897. if ( floatEnds[0] > floatEnds[1] + 1.0f ) {
  898. // back face
  899. continue;
  900. }
  901. // Check it against the occlusion buffer.
  902. // Because the perspective divide is not going to be bit-exact between
  903. // the CPU and GPU, we check an extra column here. That will result
  904. // in an occasional line being drawn that might not need to be, but
  905. // it avoids missing columns.
  906. int checkMin = floor( floatEnds[0] - 1 );
  907. int checkMax = ceil( floatEnds[1] + 1 );
  908. if ( checkMin < 0 ) {
  909. checkMin = 0;
  910. }
  911. if ( checkMax > viewwidth ) {
  912. checkMax = viewwidth;
  913. }
  914. if ( !memchr( occlusion + checkMin, 0, checkMax - checkMin ) ) {
  915. failCount++;
  916. // every column it would touch is already solid, so it isn't visible
  917. continue;
  918. }
  919. if ( occluder ) {
  920. // It is important to update the occlusion array as individual
  921. // segs are processed to maintain pure front to back order. If
  922. // the occlusion buffer was updated by complete lines, it would
  923. // result in some elements being incorrectly occlusion culled.
  924. // Use a consistant fill rule for the occlusion, which is only
  925. // referenced by the CPU, and should be water tight.
  926. int fillMin = (int) (floatEnds[0]+0.5);
  927. int fillMax = (int) (floatEnds[1]+0.5);
  928. if ( fillMax > fillMin ) {
  929. memset( occlusion + fillMin, 1, fillMax-fillMin );
  930. }
  931. }
  932. if ( line->validcount == validcount ) {
  933. continue;
  934. }
  935. line->validcount = validcount;
  936. // this line can show up on the automap now
  937. line->flags |= ML_MAPPED;
  938. // Adding a line may generate up to four drawn walls -- a top wall,
  939. // a bottom wall, a perforated middle wall, and a sky wall.
  940. // Use the complete, unclipped segment for the side
  941. IR_AddWall( &seg->sidedef->sideSeg );
  942. }
  943. }
  944. PUREFUNC static int IR_PointOnSide(fixed_t x, fixed_t y, const node_t *node)
  945. {
  946. // JDC: these early tests probably aren't worth it on iphone...
  947. if (!node->dx)
  948. return x <= node->x ? node->dy > 0 : node->dy < 0;
  949. if (!node->dy)
  950. return y <= node->y ? node->dx < 0 : node->dx > 0;
  951. x -= node->x;
  952. y -= node->y;
  953. // Try to quickly decide by looking at sign bits.
  954. if ((node->dy ^ node->dx ^ x ^ y) < 0)
  955. return (node->dy ^ x) < 0; // (left is negative)
  956. return (__int64_t)y * (__int64_t)node->dx >= (__int64_t)x * (__int64_t)node->dy;
  957. }
  958. static const int checkcoord[12][4] = // killough -- static const
  959. {
  960. {3,0,2,1},
  961. {3,0,2,0},
  962. {3,1,2,0},
  963. {0},
  964. {2,0,2,1},
  965. {0,0,0,0},
  966. {3,1,3,0},
  967. {0},
  968. {2,0,3,1},
  969. {2,1,3,1},
  970. {2,1,3,0}
  971. };
  972. static boolean IR_IsBBoxCompletelyOccluded(const fixed_t *bspcoord) {
  973. // conservatively accept if close to the box, so
  974. // we don't need to worry about the near clip plane
  975. // in TrnasformAndClipSegment. Mapscale is 128*fracunit
  976. // and nearclip is 0.1, so accepting 2 fracunits away works.
  977. if ( viewx > bspcoord[BOXLEFT]-2*FRACUNIT && viewx < bspcoord[BOXRIGHT] + 2*FRACUNIT
  978. && viewy > bspcoord[BOXBOTTOM]-2*FRACUNIT && viewy < bspcoord[BOXTOP] + 2*FRACUNIT ) {
  979. return false;
  980. }
  981. // Find the corners of the box
  982. // that define the edges from current viewpoint.
  983. int boxpos = (viewx <= bspcoord[BOXLEFT] ? 0 : viewx < bspcoord[BOXRIGHT ] ? 1 : 2) +
  984. (viewy >= bspcoord[BOXTOP ] ? 0 : viewy > bspcoord[BOXBOTTOM] ? 4 : 8);
  985. const int *check = checkcoord[boxpos];
  986. float v[2][2];
  987. v[0][0] = -bspcoord[check[0]]/MAP_SCALE;
  988. v[0][1] = bspcoord[check[1]]/MAP_SCALE;
  989. v[1][0] = -bspcoord[check[2]]/MAP_SCALE;
  990. v[1][1] = bspcoord[check[3]]/MAP_SCALE;
  991. float ends[2];
  992. if ( !TransformAndClipSegment( v, ends ) ) {
  993. // the line is off to the side or facing away
  994. return true;
  995. }
  996. if ( ends[0] >= ends[1] ) {
  997. return true;
  998. }
  999. assert( ends[0] >= 0 );
  1000. assert( ends[1] <= viewwidth );
  1001. // Check it against the occlusion buffer, with an extra column
  1002. // of slop to account for GPU / CPU floating point differences.
  1003. if ( !memchr( occlusion + (int)ends[0], 0, (int)ends[1]-(int)ends[0]+1 ) ) {
  1004. // every column it would touch is already solid, so it isn't visible
  1005. return true;
  1006. }
  1007. // there are gaps, so we may need to draw something
  1008. return false;
  1009. }
  1010. /*
  1011. RenderBSPNode
  1012. Renders all subsectors below a given node,
  1013. traversing subtree recursively.
  1014. Because this function is recursive, avoid doing work that
  1015. would give a large stack frame. Important that the compiler
  1016. doesn't inline big functions.
  1017. */
  1018. static void IR_RenderBSPNode( int bspnum ) {
  1019. while (!(bspnum & NF_SUBSECTOR)) {
  1020. // decision node
  1021. const node_t *bsp = &nodes[bspnum];
  1022. // Decide which side the view point is on.
  1023. int side = IR_PointOnSide(viewx, viewy, bsp);
  1024. // check the front space
  1025. if ( !IR_IsBBoxCompletelyOccluded(bsp->bbox[side]) ) {
  1026. IR_RenderBSPNode( bsp->children[side] );
  1027. }
  1028. // continue down the back space
  1029. if ( IR_IsBBoxCompletelyOccluded( bsp->bbox[side^1]) ) {
  1030. return;
  1031. }
  1032. bspnum = bsp->children[side^1];
  1033. }
  1034. // subsector with contents
  1035. // add all the drawable elements in the subsector
  1036. if ( bspnum == -1 ) {
  1037. bspnum = 0;
  1038. }
  1039. bspnum &= ~NF_SUBSECTOR;
  1040. IR_Subsector( bspnum );
  1041. }
  1042. typedef struct {
  1043. void *texture;
  1044. void *user;
  1045. } texSort_t;
  1046. static int TexSort( const void *a, const void *b ) {
  1047. if ( ((texSort_t *)a)->texture < ((texSort_t *)b)->texture ) {
  1048. return -1;
  1049. }
  1050. return 1;
  1051. }
  1052. int SysIphoneMicroseconds();
  1053. void SetImmediateModeGLVertexArrays();
  1054. extern float yaw;
  1055. extern GLTexture **gld_GLTextures;
  1056. extern GLTexture **gld_GLPatchTextures;
  1057. extern GLVertex *gld_vertexes;
  1058. extern GLTexcoord *gld_texcoords;
  1059. #define MAX_DRAW_INDEXES 0x10000
  1060. unsigned short drawIndexes[MAX_DRAW_INDEXES];
  1061. int numDrawIndexes;
  1062. drawVert_t drawVerts[MAX_DRAW_VERTS];
  1063. int numDrawVerts;
  1064. void NewDrawScene(player_t *player) // JDC: new version
  1065. {
  1066. int i,k;
  1067. glDisable( GL_ALPHA_TEST );
  1068. glDisable(GL_CULL_FACE);
  1069. glDisable(GL_FOG);
  1070. glEnableClientState(GL_TEXTURE_COORD_ARRAY);
  1071. glEnableClientState(GL_VERTEX_ARRAY);
  1072. glDisableClientState(GL_COLOR_ARRAY);
  1073. // Use the 2x texture combiner mode to allow the game to be brightened up
  1074. // above the normal maximum. All calculated color values for lighting will
  1075. // be multiplied by a value ranging from 0.5 for original brightness, up to
  1076. // 1.0 for 2x brightness. This combine mode will be canceled after all
  1077. // the 3D and view weapon drawing is completed, so the hud elements are
  1078. // drawn at normal brightness.
  1079. glTexEnvf( GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_COMBINE );
  1080. glTexEnvf( GL_TEXTURE_ENV, GL_RGB_SCALE, 2.0 );
  1081. // opaque skies, flats, and walls write to the depth buffer and don't blend
  1082. glDisable( GL_BLEND );
  1083. glDepthMask( 1 );
  1084. // debug tool to check for things being drawn that shouldn't be
  1085. if ( blendAll ) {
  1086. glClearColor( 0, 0, 0, 0 );
  1087. glClear( GL_COLOR_BUFFER_BIT );
  1088. glEnable( GL_BLEND );
  1089. glDisable( GL_DEPTH_TEST );
  1090. glBlendFunc( GL_ONE, GL_ONE );
  1091. skyIsVisible = false;
  1092. }
  1093. // debug tool to look for pixel cracks
  1094. if ( testClear ) {
  1095. glClearColor( 1, 0, 0, 0 );
  1096. glClear( GL_COLOR_BUFFER_BIT );
  1097. skyIsVisible = false;
  1098. }
  1099. //-----------------------------------------
  1100. // draw the sky if needed
  1101. //-----------------------------------------
  1102. if ( skyIsVisible ) {
  1103. float s;
  1104. float y;
  1105. // Note that these texcoords would have to be corrected
  1106. // for different screen aspect ratios or fields of view!
  1107. s = ((yaw+90.0f)/90.0f);
  1108. y = 1 - 2 * 128.0 / 200;
  1109. // With identity matricies, the vertex coordinates
  1110. // can just be in the 0-1 range.
  1111. glMatrixMode( GL_MODELVIEW );
  1112. glPushMatrix();
  1113. glLoadIdentity();
  1114. glMatrixMode( GL_PROJECTION );
  1115. glPushMatrix();
  1116. glLoadIdentity();
  1117. iphoneRotateForLandscape();
  1118. gld_BindTexture( gld_RegisterTexture( skytexture, true, true ) );
  1119. glColor4f( 0.5, 0.5, 0.5, 1.0 ); // native texture color, not double bright
  1120. glBegin(GL_TRIANGLE_STRIP);
  1121. glTexCoord2f( s, 1 ); glVertex3f(-1,y,0.999);
  1122. glTexCoord2f( s, 0 ); glVertex3f(-1,1,0.999);
  1123. glTexCoord2f( s+1, 1 ); glVertex3f(1,y,0.999);
  1124. glTexCoord2f( s+1, 0 ); glVertex3f(1,1,0.999);
  1125. glEnd();
  1126. // back to the normal drawing matrix
  1127. glPopMatrix();
  1128. glMatrixMode( GL_MODELVIEW );
  1129. glPopMatrix();
  1130. }
  1131. // walls and flats use the drawVerts array for everything
  1132. glTexCoordPointer(2,GL_FLOAT,sizeof(drawVert_t),drawVerts[0].st);
  1133. glVertexPointer(3,GL_FLOAT,sizeof(drawVert_t),drawVerts[0].xyz);
  1134. glColorPointer(4,GL_UNSIGNED_BYTE,sizeof(drawVert_t),drawVerts[0].rgba);
  1135. // everything will draw at full brightness in this case
  1136. if (player->fixedcolormap) {
  1137. glColor4f(1.0f, 1.0f, 1.0f, 1.0f );
  1138. glDisableClientState(GL_COLOR_ARRAY);
  1139. } else {
  1140. glEnableClientState( GL_COLOR_ARRAY );
  1141. }
  1142. int c_drawElements = 0;
  1143. numDrawIndexes = 0;
  1144. //-----------------------------------------
  1145. // draw all the walls, sky walls sorted first
  1146. //-----------------------------------------
  1147. // sort the walls by texture
  1148. texSort_t *wallSort = alloca( sizeof( wallSort[0] ) * gld_drawinfo.num_walls );
  1149. for (i=0 ; i < gld_drawinfo.num_walls ; i++ ) {
  1150. GLWall *wall = &gld_drawinfo.walls[i];
  1151. wallSort[i].texture = wall->gltexture;
  1152. wallSort[i].user = wall;
  1153. }
  1154. qsort( wallSort, gld_drawinfo.num_walls, sizeof( wallSort[0] ), TexSort );
  1155. // continue building drawverts after the floor and ceiling data
  1156. int tempDrawVert = numDrawVerts;
  1157. // alpha tested walls will use half alpha to get the best edging effects
  1158. glAlphaFunc( GL_GREATER, 0.5 );
  1159. for (i=0 ; i < gld_drawinfo.num_walls ; i++ ) {
  1160. texSort_t *sort = &wallSort[i];
  1161. GLWall *wall = sort->user;
  1162. rendered_segs++;
  1163. // add two tris to draw the wall
  1164. drawIndexes[numDrawIndexes+0] = tempDrawVert;
  1165. drawIndexes[numDrawIndexes+1] = tempDrawVert+1;
  1166. drawIndexes[numDrawIndexes+2] = tempDrawVert+2;
  1167. drawIndexes[numDrawIndexes+3] = tempDrawVert+1;
  1168. drawIndexes[numDrawIndexes+4] = tempDrawVert+2;
  1169. drawIndexes[numDrawIndexes+5] = tempDrawVert+3;
  1170. numDrawIndexes += 6;
  1171. unsigned char rgba[4];
  1172. rgba[0] = rgba[1] = rgba[2] = wall->light;
  1173. rgba[3] = 255;
  1174. int lightInt = *(int *)rgba;
  1175. drawVerts[tempDrawVert].st[0] = wall->ul;
  1176. drawVerts[tempDrawVert].st[1] = wall->vt;
  1177. drawVerts[tempDrawVert].xyz[0] = -wall->side->sideSeg.v1->x / MAP_SCALE;
  1178. drawVerts[tempDrawVert].xyz[1] = wall->ytop;
  1179. drawVerts[tempDrawVert].xyz[2] = wall->side->sideSeg.v1->y / MAP_SCALE;
  1180. lightInt = FadedLighting( drawVerts[tempDrawVert].xyz[0], drawVerts[tempDrawVert].xyz[2], wall->light );
  1181. *(int *)drawVerts[tempDrawVert].rgba = lightInt;
  1182. tempDrawVert++;
  1183. drawVerts[tempDrawVert].st[0] = wall->ul;
  1184. drawVerts[tempDrawVert].st[1] = wall->vb;
  1185. drawVerts[tempDrawVert].xyz[0] = -wall->side->sideSeg.v1->x / MAP_SCALE;
  1186. drawVerts[tempDrawVert].xyz[1] = wall->ybottom;
  1187. drawVerts[tempDrawVert].xyz[2] = wall->side->sideSeg.v1->y / MAP_SCALE;
  1188. *(int *)drawVerts[tempDrawVert].rgba = lightInt;
  1189. tempDrawVert++;
  1190. drawVerts[tempDrawVert].st[0] = wall->ur;
  1191. drawVerts[tempDrawVert].st[1] = wall->vt;
  1192. drawVerts[tempDrawVert].xyz[0] = -wall->side->sideSeg.v2->x / MAP_SCALE;
  1193. drawVerts[tempDrawVert].xyz[1] = wall->ytop;
  1194. drawVerts[tempDrawVert].xyz[2] = wall->side->sideSeg.v2->y / MAP_SCALE;
  1195. lightInt = FadedLighting( drawVerts[tempDrawVert].xyz[0], drawVerts[tempDrawVert].xyz[2], wall->light );
  1196. *(int *)drawVerts[tempDrawVert].rgba = lightInt;
  1197. tempDrawVert++;
  1198. drawVerts[tempDrawVert].st[0] = wall->ur;
  1199. drawVerts[tempDrawVert].st[1] = wall->vb;
  1200. drawVerts[tempDrawVert].xyz[0] = -wall->side->sideSeg.v2->x / MAP_SCALE;
  1201. drawVerts[tempDrawVert].xyz[1] = wall->ybottom;
  1202. drawVerts[tempDrawVert].xyz[2] = wall->side->sideSeg.v2->y / MAP_SCALE;
  1203. *(int *)drawVerts[tempDrawVert].rgba = lightInt;
  1204. tempDrawVert++;
  1205. // only draw when textures change
  1206. if ( i == gld_drawinfo.num_walls-1 || sort->texture != (sort+1)->texture ) {
  1207. if ( numDrawIndexes ) {
  1208. if ( wall->flag == GLDWF_SKY ) {
  1209. // we aren't actually drawing anything with this,
  1210. // we are just masking off areas in the depth
  1211. // buffer so nothing can overwrite the already
  1212. // drawn sky image
  1213. glColorMask( 0, 0, 0, 0 );
  1214. }
  1215. if ( wall->flag == GLDWF_M2S ) {
  1216. glEnable( GL_ALPHA_TEST );
  1217. }
  1218. if ( wall->gltexture ) { // skies are texture = NULL
  1219. gld_BindTexture( wall->gltexture );
  1220. }
  1221. glDrawElements( GL_TRIANGLES, numDrawIndexes, GL_UNSIGNED_SHORT,
  1222. drawIndexes );
  1223. if ( wall->flag == GLDWF_M1S ) {
  1224. glDisable( GL_ALPHA_TEST );
  1225. }
  1226. if ( wall->flag == GLDWF_SKY ) {
  1227. glColorMask( 1, 1, 1, 1 );
  1228. }
  1229. numDrawIndexes = 0;
  1230. tempDrawVert = numDrawVerts;
  1231. c_drawElements++;
  1232. }
  1233. }
  1234. }
  1235. //-----------------------------------------
  1236. // draw all the flats
  1237. //
  1238. // If we were able to directly fill the GPU command buffers,
  1239. // we would be using multiple DrawArrays instead of a single DrawElements,
  1240. // and we would fill the plane height and color in as we copied vertexes
  1241. // from a single set of verts per sector, but because the driver validation
  1242. // overhead is the main poison on the iPhone currently, it is better to
  1243. // duplicate the windings for the floors and ceilings, patch the
  1244. // vertex data, and generate new index lists to minimize the number of
  1245. // draw calls.
  1246. //-----------------------------------------
  1247. // sort the flats by texture
  1248. qsort( sectorPlanes, numSectorPlanes, sizeof( sectorPlanes[0] ), TexSort );
  1249. // draw them in texture order
  1250. for (i=0 ; i < numSectorPlanes ; i++ ) {
  1251. sortSectorPlane_t *sort = &sectorPlanes[i];
  1252. sector_t *sector = sort->sector;
  1253. drawVert_t *dv = sector->verts[(int)sort->ceiling];
  1254. // Patch the height values if they have changed since the last draw.
  1255. float y = sort->ceiling ? sector->ceilingheight : sector->floorheight;
  1256. y *= ( 1.0 / MAP_SCALE );
  1257. if ( y != dv->xyz[1] ) {
  1258. for ( int j = 0 ; j < sector->numVerts ; j++ ) {
  1259. (dv+j)->xyz[1] = y;
  1260. }
  1261. }
  1262. // per-vertex faded light color
  1263. int light = sector->lightlevel + (extralight<<5);
  1264. if ( light > 255 ) {
  1265. light = 255;
  1266. }
  1267. for ( int j = 0 ; j < sector->numVerts ; j++ ) {
  1268. *(int *)(dv+j)->rgba = FadedLighting( (dv+j)->xyz[0], (dv+j)->xyz[2], light );
  1269. }
  1270. // copy the indexes
  1271. assert( numDrawIndexes + sector->numIndexes < MAX_DRAW_INDEXES );
  1272. memcpy( drawIndexes + numDrawIndexes, sector->indexes[(int)sort->ceiling], sector->numIndexes*sizeof(drawIndexes[0]) );
  1273. numDrawIndexes += sector->numIndexes;
  1274. // only change textures when necessary
  1275. if ( i == numSectorPlanes - 1 || sort->texture != (sort+1)->texture ) {
  1276. if ( numDrawIndexes ) {
  1277. gld_BindFlat( sort->texture );
  1278. glDrawElements( GL_TRIANGLES, numDrawIndexes, GL_UNSIGNED_SHORT,
  1279. drawIndexes );
  1280. numDrawIndexes = 0;
  1281. c_drawElements++;
  1282. }
  1283. }
  1284. }
  1285. glDisableClientState( GL_COLOR_ARRAY );
  1286. // back to our immediate mode vertex arrays
  1287. SetImmediateModeGLVertexArrays();
  1288. //-----------------------------------------
  1289. // draw all the sprites
  1290. //-----------------------------------------
  1291. // transparent sprites blend and don't write to the depth buffer
  1292. glEnable( GL_BLEND );
  1293. glDepthMask( 0 );
  1294. glEnable( GL_ALPHA_TEST );
  1295. // get the screen space vector for sprites
  1296. float yaws = -sin( yaw * 3.141592657 / 180.0 );
  1297. float yawc = -cos( yaw * 3.141592657 / 180.0 );
  1298. int c_spriteBind = 0;
  1299. int c_spriteDraw = 0;
  1300. while( 1 )
  1301. {
  1302. // pick out the sprites from farthest to nearest
  1303. fixed_t max_scale=INT_MAX;
  1304. k=-1;
  1305. for (i=0 ; i < gld_drawinfo.num_sprites ; i++ ) {
  1306. GLSprite *sprite = &gld_drawinfo.sprites[i];
  1307. if (sprite->scale<max_scale)
  1308. {
  1309. max_scale=sprite->scale;
  1310. k=i;
  1311. }
  1312. }
  1313. if ( k == -1 ) {
  1314. break;
  1315. }
  1316. GLSprite *sprite = &gld_drawinfo.sprites[k];
  1317. sprite->scale=INT_MAX;
  1318. if ( sprite->gltexture != last_gltexture ) {
  1319. c_spriteBind++;
  1320. }
  1321. c_spriteDraw++;
  1322. gld_BindPatch(sprite->gltexture,sprite->cm);
  1323. if(sprite->shadow)
  1324. {
  1325. glBlendFunc(GL_DST_COLOR, GL_ONE_MINUS_SRC_ALPHA);
  1326. glColor4f(0.2f,0.2f,0.2f,0.33f);
  1327. glAlphaFunc( GL_GREATER, 0.1 ); // don't alpha test away the blended-down version
  1328. }
  1329. else
  1330. {
  1331. float flight = (float)sprite->light*(1.0f/255);
  1332. // We could do the distance-lighting here, but leaving the sprites
  1333. // brighter is a good accent in most cases. There are a few places
  1334. // where environmental sprites look a little wrong, but it is probably
  1335. // better in general.
  1336. if (player->fixedcolormap) {
  1337. flight = 1.0; // light amp goggles
  1338. }
  1339. glColor4f(flight, flight, flight, 1.0f );
  1340. }
  1341. glBegin(GL_TRIANGLE_STRIP);
  1342. glTexCoord2f(sprite->ul, sprite->vt);
  1343. glVertex3f(sprite->x + sprite->x1 * yawc, sprite->y + sprite->y1, sprite->z + sprite->x1 * yaws );
  1344. glTexCoord2f(sprite->ur, sprite->vt);
  1345. glVertex3f(sprite->x + sprite->x2 * yawc, sprite->y + sprite->y1, sprite->z + sprite->x2 * yaws );
  1346. glTexCoord2f(sprite->ul, sprite->vb);
  1347. glVertex3f(sprite->x + sprite->x1 * yawc, sprite->y + sprite->y2, sprite->z + sprite->x1 * yaws );
  1348. glTexCoord2f(sprite->ur, sprite->vb);
  1349. glVertex3f(sprite->x + sprite->x2 * yawc, sprite->y + sprite->y2, sprite->z + sprite->x2 * yaws );
  1350. glEnd();
  1351. if(sprite->shadow)
  1352. {
  1353. glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
  1354. glAlphaFunc( GL_GREATER, 0.5 );
  1355. }
  1356. }
  1357. glDisable( GL_ALPHA_TEST );
  1358. glDepthMask( 1 );
  1359. }
  1360. static float roll = 0.0f;
  1361. /* JDC static */ float yaw = 0.0f;
  1362. static float inv_yaw = 0.0f;
  1363. static float pitch = 0.0f;
  1364. #define __glPi 3.14159265358979323846
  1365. static void infinitePerspective(GLdouble fovy, GLdouble aspect, GLdouble znear)
  1366. {
  1367. GLfloat left, right, bottom, top; // JDC: was GLdouble
  1368. GLfloat m[16]; // JDC: was GLdouble
  1369. top = znear * tan(fovy * __glPi / 360.0);
  1370. bottom = -top;
  1371. left = bottom * aspect;
  1372. right = top * aspect;
  1373. //qglFrustum(left, right, bottom, top, znear, zfar);
  1374. m[ 0] = (2 * znear) / (right - left);
  1375. m[ 4] = 0;
  1376. m[ 8] = (right + left) / (right - left);
  1377. m[12] = 0;
  1378. m[ 1] = 0;
  1379. m[ 5] = (2 * znear) / (top - bottom);
  1380. m[ 9] = (top + bottom) / (top - bottom);
  1381. m[13] = 0;
  1382. m[ 2] = 0;
  1383. m[ 6] = 0;
  1384. //m[10] = - (zfar + znear) / (zfar - znear);
  1385. //m[14] = - (2 * zfar * znear) / (zfar - znear);
  1386. m[10] = -1;
  1387. m[14] = -2 * znear;
  1388. m[ 3] = 0;
  1389. m[ 7] = 0;
  1390. m[11] = -1;
  1391. m[15] = 0;
  1392. glMultMatrixf(m); // JDC: was glMultMatrixd
  1393. }
  1394. /*
  1395. IR_RenderPlayerView
  1396. Replace the prBoom rendering code with a higher performance
  1397. version. Most of the fancy new features are gone, because I
  1398. have no idea what the reight test cases would be for them.
  1399. */
  1400. void IR_RenderPlayerView (player_t* player) {
  1401. int start = 0;
  1402. if ( showRenderTime ) {
  1403. start = SysIphoneMicroseconds();
  1404. }
  1405. viewplayer = player;
  1406. tic_vars.frac = FRACUNIT;
  1407. viewx = player->mo->x;
  1408. viewy = player->mo->y;
  1409. viewz = player->viewz;
  1410. viewangle = player->mo->angle;
  1411. extralight = player->extralight; // gun flashes
  1412. yaw=270.0f-(float)(viewangle>>ANGLETOFINESHIFT)*360.0f/FINEANGLES;
  1413. viewsin = finesine[viewangle>>ANGLETOFINESHIFT];
  1414. viewcos = finecosine[viewangle>>ANGLETOFINESHIFT];
  1415. // IR goggles
  1416. if (player->fixedcolormap) {
  1417. fixedcolormap = fullcolormap + player->fixedcolormap*256*sizeof(lighttable_t);
  1418. } else {
  1419. fixedcolormap = 0;
  1420. }
  1421. // this is used to tell if a line, sector, or sprite is going to be drawn this frame
  1422. validcount++;
  1423. // clear the 1D occlusion buffer, set a final guard column to occluded so we can
  1424. // check an extra pixel to account for slight floating point differences between
  1425. // the GPU and CPU transformations
  1426. assert( viewwidth <= MAX_SCREENWIDTH );
  1427. memset( occlusion, 0, sizeof( occlusion ) );
  1428. occlusion[viewwidth] = 1;
  1429. float trY ;
  1430. float xCamera,yCamera;
  1431. extern int screenblocks;
  1432. int height;
  1433. gld_SetPalette(-1);
  1434. if (screenblocks == 11)
  1435. height = SCREENHEIGHT;
  1436. else if (screenblocks == 10)
  1437. height = SCREENHEIGHT;
  1438. else
  1439. height = (screenblocks*SCREENHEIGHT/10) & ~7;
  1440. glViewport(viewwindowx, SCREENHEIGHT-(height+viewwindowy-((height-viewheight)/2)), viewwidth, height);
  1441. glScissor(viewwindowx, SCREENHEIGHT-(viewheight+viewwindowy), viewwidth, viewheight);
  1442. glEnable(GL_SCISSOR_TEST);
  1443. glEnable(GL_DEPTH_TEST);
  1444. glDisable(GL_FOG);
  1445. glShadeModel(GL_SMOOTH);
  1446. // Player coordinates
  1447. xCamera=-(float)viewx/MAP_SCALE;
  1448. yCamera=(float)viewy/MAP_SCALE;
  1449. trY=(float)viewz/MAP_SCALE;
  1450. yaw=270.0f-(float)(viewangle>>ANGLETOFINESHIFT)*360.0f/FINEANGLES;
  1451. inv_yaw=-90.0f+(float)(viewangle>>ANGLETOFINESHIFT)*360.0f/FINEANGLES;
  1452. #ifdef _DEBUG
  1453. glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
  1454. #else
  1455. glClear(GL_DEPTH_BUFFER_BIT);
  1456. #endif
  1457. // To make it easier to accurately mimic the GL model to screen transformation,
  1458. // this is set up so that the projection transformation is also done in the
  1459. // modelview matrix, leaving the projection matrix as an identity. This means
  1460. // that things done in eye space, like lighting and fog, won't work, but
  1461. // we don't need them.
  1462. glMatrixMode(GL_PROJECTION);
  1463. glLoadIdentity();
  1464. glMatrixMode(GL_MODELVIEW);
  1465. glLoadIdentity();
  1466. // make the 320x480 hardware seem like 480x320 in two different orientations
  1467. // and note if the occlusion segmenrs need to be reversed
  1468. reversedLandscape = iphoneRotateForLandscape();
  1469. infinitePerspective(64.0f, 320.0f/200.0f, 5.0f/100.0f);
  1470. glRotatef(roll, 0.0f, 0.0f, 1.0f);
  1471. glRotatef(pitch, 1.0f, 0.0f, 0.0f);
  1472. glRotatef(yaw, 0.0f, 1.0f, 0.0f);
  1473. glTranslatef(-xCamera, -trY, -yCamera);
  1474. // read back the matrix so we can do exact calculations that match
  1475. // what GL is doing. It would probably be better to build the matricies
  1476. // ourselves and just do a loadMatrix...
  1477. glGetFloatv( GL_MODELVIEW_MATRIX, glMVPmatrix );
  1478. // setup the vector for calculating light fades, which is just a scale
  1479. // of the forward vector
  1480. lightingVector[0] = lightDistance * glMVPmatrix[2];
  1481. lightingVector[1] = lightDistance * glMVPmatrix[10];
  1482. lightingVector[2] = lightDistance * glMVPmatrix[14];
  1483. rendermarker++;
  1484. gld_drawinfo.num_walls=0;
  1485. gld_drawinfo.num_flats=0;
  1486. gld_drawinfo.num_sprites=0;
  1487. gld_drawinfo.num_drawitems=0;
  1488. c_occludedSprites = 0;
  1489. c_sectors = 0;
  1490. c_subsectors = 0;
  1491. numSectorPlanes = 0;
  1492. failCount = 0;
  1493. // Find everything we need to draw, but don't draw anything yet,
  1494. // because we want to sort by texture to reduce GL driver overhead.
  1495. IR_RenderBSPNode( numnodes-1 );
  1496. NewDrawScene(player);
  1497. gld_EndDrawScene();
  1498. if ( showRenderTime ) {
  1499. int end = SysIphoneMicroseconds();
  1500. printf( "%i usec\n", end - start );
  1501. }
  1502. }