yquake2remaster/src/refresh/r_main.c

1661 lines
36 KiB
C
Raw Normal View History

/*
* Copyright (C) 1997-2001 Id Software, Inc.
*
2010-10-23 06:58:56 +00:00
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or (at
* your option) any later version.
*
2010-10-23 06:58:56 +00:00
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
2010-10-23 06:58:56 +00:00
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*
2010-10-23 06:58:56 +00:00
* =======================================================================
*
* Refresher setup and main part of the frame generation
*
* =======================================================================
*/
2009-03-05 12:42:43 +00:00
2009-03-05 11:03:08 +00:00
#include "header/local.h"
#define NUM_BEAM_SEGS 6
2012-07-21 12:09:45 +00:00
void R_Clear(void);
void R_BeginRegistration(char *map);
struct model_s *R_RegisterModel(char *name);
struct image_s *R_RegisterSkin(char *name);
void R_SetSky(char *name, float rotate, vec3_t axis);
void R_EndRegistration(void);
void R_RenderFrame(refdef_t *fd);
struct image_s *Draw_FindPic(char *name);
void Draw_Pic(int x, int y, char *name);
void Draw_Char(int x, int y, int c);
void Draw_TileClear(int x, int y, int w, int h, char *name);
void Draw_Fill(int x, int y, int w, int h, int c);
void Draw_FadeScreen(void);
viddef_t vid;
refimport_t ri;
int QGL_TEXTURE0, QGL_TEXTURE1;
2012-07-21 12:09:45 +00:00
model_t *r_worldmodel;
float gldepthmin, gldepthmax;
glconfig_t gl_config;
glstate_t gl_state;
2012-07-21 12:09:45 +00:00
image_t *r_notexture; /* use for bad textures */
image_t *r_particletexture; /* little dot for particles */
2012-07-21 12:09:45 +00:00
entity_t *currententity;
model_t *currentmodel;
2012-07-21 12:09:45 +00:00
cplane_t frustum[4];
2012-07-21 12:09:45 +00:00
int r_visframecount; /* bumped when going to a new PVS */
int r_framecount; /* used for dlight push checking */
int c_brush_polys, c_alias_polys;
2012-07-21 12:09:45 +00:00
float v_blend[4]; /* final blending color */
2012-07-21 12:09:45 +00:00
void R_Strings(void);
/* view origin */
vec3_t vup;
vec3_t vpn;
vec3_t vright;
vec3_t r_origin;
2012-07-21 12:09:45 +00:00
float r_world_matrix[16];
float r_base_world_matrix[16];
/* screen size info */
refdef_t r_newrefdef;
int r_viewcluster, r_viewcluster2, r_oldviewcluster, r_oldviewcluster2;
extern qboolean have_stencil;
2012-07-21 12:09:45 +00:00
unsigned r_rawpalette[256];
cvar_t *gl_norefresh;
cvar_t *gl_drawentities;
cvar_t *gl_drawworld;
cvar_t *gl_speeds;
cvar_t *gl_fullbright;
cvar_t *gl_novis;
cvar_t *gl_nocull;
cvar_t *gl_lerpmodels;
cvar_t *gl_lefthand;
cvar_t *gl_farsee;
cvar_t *gl_lightlevel;
cvar_t *gl_overbrightbits;
cvar_t *gl_nosubimage;
cvar_t *gl_allow_software;
cvar_t *gl_vertex_arrays;
cvar_t *gl_particle_min_size;
cvar_t *gl_particle_max_size;
cvar_t *gl_particle_size;
cvar_t *gl_particle_att_a;
cvar_t *gl_particle_att_b;
cvar_t *gl_particle_att_c;
cvar_t *gl_ext_swapinterval;
cvar_t *gl_ext_palettedtexture;
cvar_t *gl_ext_multitexture;
cvar_t *gl_ext_pointparameters;
cvar_t *gl_ext_compiled_vertex_array;
cvar_t *gl_ext_mtexcombine;
cvar_t *gl_log;
cvar_t *gl_bitdepth;
cvar_t *gl_drawbuffer;
cvar_t *gl_driver;
cvar_t *gl_lightmap;
cvar_t *gl_shadows;
cvar_t *gl_stencilshadow;
cvar_t *gl_mode;
cvar_t *gl_customwidth;
cvar_t *gl_customheight;
#ifdef RETEXTURE
2012-07-21 12:09:45 +00:00
cvar_t *gl_retexturing;
#endif
2012-07-21 12:09:45 +00:00
cvar_t *gl_dynamic;
cvar_t *gl_modulate;
cvar_t *gl_nobind;
cvar_t *gl_round_down;
cvar_t *gl_picmip;
cvar_t *gl_skymip;
cvar_t *gl_showtris;
cvar_t *gl_ztrick;
cvar_t *gl_zfix;
2012-07-21 12:09:45 +00:00
cvar_t *gl_finish;
cvar_t *gl_clear;
cvar_t *gl_cull;
cvar_t *gl_polyblend;
cvar_t *gl_flashblend;
cvar_t *gl_playermip;
cvar_t *gl_saturatelighting;
cvar_t *gl_swapinterval;
cvar_t *gl_texturemode;
cvar_t *gl_texturealphamode;
cvar_t *gl_texturesolidmode;
cvar_t *gl_anisotropic;
cvar_t *gl_anisotropic_avail;
cvar_t *gl_lockpvs;
cvar_t *vid_fullscreen;
cvar_t *vid_gamma;
cvar_t *vid_ref;
/*
* Returns true if the box is completely outside the frustom
*/
qboolean
2012-07-21 12:09:45 +00:00
R_CullBox(vec3_t mins, vec3_t maxs)
{
int i;
2012-07-21 12:09:45 +00:00
if (gl_nocull->value)
{
2012-07-21 12:09:45 +00:00
return false;
}
2012-07-21 12:09:45 +00:00
for (i = 0; i < 4; i++)
{
2012-07-21 12:09:45 +00:00
if (BOX_ON_PLANE_SIDE(mins, maxs, &frustum[i]) == 2)
{
2012-07-21 12:09:45 +00:00
return true;
}
}
2012-07-21 12:09:45 +00:00
return false;
}
void
2012-07-21 12:09:45 +00:00
R_RotateForEntity(entity_t *e)
{
2012-07-21 12:09:45 +00:00
qglTranslatef(e->origin[0], e->origin[1], e->origin[2]);
2012-07-21 12:09:45 +00:00
qglRotatef(e->angles[1], 0, 0, 1);
qglRotatef(-e->angles[0], 0, 1, 0);
qglRotatef(-e->angles[2], 1, 0, 0);
}
void
2012-07-21 12:09:45 +00:00
R_DrawSpriteModel(entity_t *e)
{
float alpha = 1.0F;
vec3_t point;
dsprframe_t *frame;
2012-07-21 12:09:45 +00:00
float *up, *right;
dsprite_t *psprite;
2012-07-21 12:09:45 +00:00
/* don't even bother culling, because it's just
a single polygon without a surface cache */
psprite = (dsprite_t *)currentmodel->extradata;
e->frame %= psprite->numframes;
2012-07-21 12:09:45 +00:00
frame = &psprite->frames[e->frame];
2012-07-21 12:09:45 +00:00
/* normal sprite */
up = vup;
right = vright;
2012-07-21 12:09:45 +00:00
if (e->flags & RF_TRANSLUCENT)
{
alpha = e->alpha;
}
2012-07-21 12:09:45 +00:00
if (alpha != 1.0F)
{
2012-07-21 12:09:45 +00:00
qglEnable(GL_BLEND);
}
2012-07-21 12:09:45 +00:00
qglColor4f(1, 1, 1, alpha);
2012-07-21 12:09:45 +00:00
R_Bind(currentmodel->skins[e->frame]->texnum);
2012-07-21 12:09:45 +00:00
R_TexEnv(GL_MODULATE);
2012-07-21 12:09:45 +00:00
if (alpha == 1.0)
{
2012-07-21 12:09:45 +00:00
qglEnable(GL_ALPHA_TEST);
}
else
{
2012-07-21 12:09:45 +00:00
qglDisable(GL_ALPHA_TEST);
}
2012-07-21 12:09:45 +00:00
qglBegin(GL_QUADS);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(0, 1);
VectorMA(e->origin, -frame->origin_y, up, point);
VectorMA(point, -frame->origin_x, right, point);
qglVertex3fv(point);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(0, 0);
VectorMA(e->origin, frame->height - frame->origin_y, up, point);
VectorMA(point, -frame->origin_x, right, point);
qglVertex3fv(point);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(1, 0);
VectorMA(e->origin, frame->height - frame->origin_y, up, point);
VectorMA(point, frame->width - frame->origin_x, right, point);
qglVertex3fv(point);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(1, 1);
VectorMA(e->origin, -frame->origin_y, up, point);
VectorMA(point, frame->width - frame->origin_x, right, point);
qglVertex3fv(point);
qglEnd();
2012-07-21 12:09:45 +00:00
qglDisable(GL_ALPHA_TEST);
R_TexEnv(GL_REPLACE);
2012-07-21 12:09:45 +00:00
if (alpha != 1.0F)
{
2012-07-21 12:09:45 +00:00
qglDisable(GL_BLEND);
}
2012-07-21 12:09:45 +00:00
qglColor4f(1, 1, 1, 1);
}
void
2012-07-21 12:09:45 +00:00
R_DrawNullModel(void)
{
vec3_t shadelight;
int i;
2012-07-21 12:09:45 +00:00
if (currententity->flags & RF_FULLBRIGHT)
{
2012-07-21 12:09:45 +00:00
shadelight[0] = shadelight[1] = shadelight[2] = 1.0F;
}
else
{
2012-07-21 12:09:45 +00:00
R_LightPoint(currententity->origin, shadelight);
}
qglPushMatrix();
2012-07-21 12:09:45 +00:00
R_RotateForEntity(currententity);
2012-07-21 12:09:45 +00:00
qglDisable(GL_TEXTURE_2D);
qglColor3fv(shadelight);
2012-07-21 12:09:45 +00:00
qglBegin(GL_TRIANGLE_FAN);
qglVertex3f(0, 0, -16);
2012-07-21 12:09:45 +00:00
for (i = 0; i <= 4; i++)
{
2012-07-21 12:09:45 +00:00
qglVertex3f(16 * cos(i * M_PI / 2), 16 * sin(i * M_PI / 2), 0);
}
qglEnd();
2012-07-21 12:09:45 +00:00
qglBegin(GL_TRIANGLE_FAN);
qglVertex3f(0, 0, 16);
2012-07-21 12:09:45 +00:00
for (i = 4; i >= 0; i--)
{
2012-07-21 12:09:45 +00:00
qglVertex3f(16 * cos(i * M_PI / 2), 16 * sin(i * M_PI / 2), 0);
}
qglEnd();
2012-07-21 12:09:45 +00:00
qglColor3f(1, 1, 1);
qglPopMatrix();
2012-07-21 12:09:45 +00:00
qglEnable(GL_TEXTURE_2D);
}
void
2012-07-21 12:09:45 +00:00
R_DrawEntitiesOnList(void)
{
int i;
2012-07-21 12:09:45 +00:00
if (!gl_drawentities->value)
{
return;
}
/* draw non-transparent first */
2012-07-21 12:09:45 +00:00
for (i = 0; i < r_newrefdef.num_entities; i++)
{
2012-07-21 12:09:45 +00:00
currententity = &r_newrefdef.entities[i];
2012-07-21 12:09:45 +00:00
if (currententity->flags & RF_TRANSLUCENT)
{
continue; /* solid */
}
2012-07-21 12:09:45 +00:00
if (currententity->flags & RF_BEAM)
{
2012-07-21 12:09:45 +00:00
R_DrawBeam(currententity);
}
else
{
currentmodel = currententity->model;
2012-07-21 12:09:45 +00:00
if (!currentmodel)
{
R_DrawNullModel();
continue;
}
2012-07-21 12:09:45 +00:00
switch (currentmodel->type)
{
case mod_alias:
2012-07-21 12:09:45 +00:00
R_DrawAliasModel(currententity);
break;
case mod_brush:
2012-07-21 12:09:45 +00:00
R_DrawBrushModel(currententity);
break;
case mod_sprite:
2012-07-21 12:09:45 +00:00
R_DrawSpriteModel(currententity);
break;
default:
2012-07-21 12:09:45 +00:00
ri.Sys_Error(ERR_DROP, "Bad modeltype");
break;
}
}
}
/* draw transparent entities
2012-07-21 12:09:45 +00:00
we could sort these if it ever
becomes a problem... */
qglDepthMask(0);
2012-07-21 12:09:45 +00:00
for (i = 0; i < r_newrefdef.num_entities; i++)
{
2012-07-21 12:09:45 +00:00
currententity = &r_newrefdef.entities[i];
2012-07-21 12:09:45 +00:00
if (!(currententity->flags & RF_TRANSLUCENT))
{
continue; /* solid */
}
2012-07-21 12:09:45 +00:00
if (currententity->flags & RF_BEAM)
{
2012-07-21 12:09:45 +00:00
R_DrawBeam(currententity);
}
else
{
currentmodel = currententity->model;
2012-07-21 12:09:45 +00:00
if (!currentmodel)
{
R_DrawNullModel();
continue;
}
2012-07-21 12:09:45 +00:00
switch (currentmodel->type)
{
case mod_alias:
2012-07-21 12:09:45 +00:00
R_DrawAliasModel(currententity);
break;
case mod_brush:
2012-07-21 12:09:45 +00:00
R_DrawBrushModel(currententity);
break;
case mod_sprite:
2012-07-21 12:09:45 +00:00
R_DrawSpriteModel(currententity);
break;
default:
2012-07-21 12:09:45 +00:00
ri.Sys_Error(ERR_DROP, "Bad modeltype");
break;
}
}
}
2012-07-21 12:09:45 +00:00
qglDepthMask(1); /* back to writing */
}
void
2012-07-21 12:09:45 +00:00
R_DrawParticles2(int num_particles, const particle_t particles[],
const unsigned colortable[768])
{
const particle_t *p;
int i;
vec3_t up, right;
float scale;
2012-07-21 12:09:45 +00:00
byte color[4];
2012-07-21 12:09:45 +00:00
R_Bind(r_particletexture->texnum);
qglDepthMask(GL_FALSE); /* no z buffering */
qglEnable(GL_BLEND);
R_TexEnv(GL_MODULATE);
qglBegin(GL_TRIANGLES);
2012-07-21 12:09:45 +00:00
VectorScale(vup, 1.5, up);
VectorScale(vright, 1.5, right);
2012-07-21 12:09:45 +00:00
for (p = particles, i = 0; i < num_particles; i++, p++)
{
/* hack a scale up to keep particles from disapearing */
2012-07-21 12:09:45 +00:00
scale = (p->origin[0] - r_origin[0]) * vpn[0] +
(p->origin[1] - r_origin[1]) * vpn[1] +
(p->origin[2] - r_origin[2]) * vpn[2];
2012-07-21 12:09:45 +00:00
if (scale < 20)
{
scale = 1;
}
else
{
scale = 1 + scale * 0.004;
}
2012-07-21 12:09:45 +00:00
*(int *)color = colortable[p->color];
color[3] = p->alpha * 255;
2012-07-21 12:09:45 +00:00
qglColor4ubv(color);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(0.0625, 0.0625);
qglVertex3fv(p->origin);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(1.0625, 0.0625);
qglVertex3f(p->origin[0] + up[0] * scale,
p->origin[1] + up[1] * scale,
p->origin[2] + up[2] * scale);
2012-07-21 12:09:45 +00:00
qglTexCoord2f(0.0625, 1.0625);
qglVertex3f(p->origin[0] + right[0] * scale,
p->origin[1] + right[1] * scale,
p->origin[2] + right[2] * scale);
}
qglEnd();
2012-07-21 12:09:45 +00:00
qglDisable(GL_BLEND);
qglColor4f(1, 1, 1, 1);
qglDepthMask(1); /* back to normal Z buffering */
R_TexEnv(GL_REPLACE);
}
void
2012-07-21 12:09:45 +00:00
R_DrawParticles(void)
{
2012-07-21 12:09:45 +00:00
if (gl_ext_pointparameters->value && qglPointParameterfEXT)
{
int i;
2012-07-21 12:09:45 +00:00
unsigned char color[4];
const particle_t *p;
2012-07-21 12:09:45 +00:00
qglDepthMask(GL_FALSE);
qglEnable(GL_BLEND);
qglDisable(GL_TEXTURE_2D);
2012-07-21 12:09:45 +00:00
qglPointSize(LittleFloat(gl_particle_size->value));
2012-07-21 12:09:45 +00:00
qglBegin(GL_POINTS);
2012-07-21 12:09:45 +00:00
for (i = 0, p = r_newrefdef.particles;
i < r_newrefdef.num_particles;
i++, p++)
{
2012-07-21 12:09:45 +00:00
*(int *)color = d_8to24table[p->color & 0xFF];
color[3] = p->alpha * 255;
qglColor4ubv(color);
qglVertex3fv(p->origin);
}
qglEnd();
2012-07-21 12:09:45 +00:00
qglDisable(GL_BLEND);
qglColor4f(1.0F, 1.0F, 1.0F, 1.0F);
qglDepthMask(GL_TRUE);
qglEnable(GL_TEXTURE_2D);
}
else
{
2012-07-21 12:09:45 +00:00
R_DrawParticles2(r_newrefdef.num_particles,
r_newrefdef.particles, d_8to24table);
}
}
void
2012-07-21 12:09:45 +00:00
R_PolyBlend(void)
{
2012-07-21 12:09:45 +00:00
if (!gl_polyblend->value)
{
return;
}
2012-07-21 12:09:45 +00:00
if (!v_blend[3])
{
return;
}
2012-07-21 12:09:45 +00:00
qglDisable(GL_ALPHA_TEST);
qglEnable(GL_BLEND);
qglDisable(GL_DEPTH_TEST);
qglDisable(GL_TEXTURE_2D);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglRotatef(-90, 1, 0, 0); /* put Z going up */
qglRotatef(90, 0, 0, 1); /* put Z going up */
2012-07-21 12:09:45 +00:00
qglColor4fv(v_blend);
2012-07-21 12:09:45 +00:00
qglBegin(GL_QUADS);
2012-07-21 12:09:45 +00:00
qglVertex3f(10, 100, 100);
qglVertex3f(10, -100, 100);
qglVertex3f(10, -100, -100);
qglVertex3f(10, 100, -100);
qglEnd();
2012-07-21 12:09:45 +00:00
qglDisable(GL_BLEND);
qglEnable(GL_TEXTURE_2D);
qglEnable(GL_ALPHA_TEST);
2012-07-21 12:09:45 +00:00
qglColor4f(1, 1, 1, 1);
}
int
2012-07-21 12:09:45 +00:00
R_SignbitsForPlane(cplane_t *out)
{
int bits, j;
/* for fast box on planeside test */
bits = 0;
2012-07-21 12:09:45 +00:00
for (j = 0; j < 3; j++)
{
2012-07-21 12:09:45 +00:00
if (out->normal[j] < 0)
{
bits |= 1 << j;
}
}
2012-07-21 12:09:45 +00:00
return bits;
}
void
2012-07-21 12:09:45 +00:00
R_SetFrustum(void)
{
int i;
/* rotate VPN right by FOV_X/2 degrees */
2012-07-21 12:09:45 +00:00
RotatePointAroundVector(frustum[0].normal, vup, vpn,
-(90 - r_newrefdef.fov_x / 2));
/* rotate VPN left by FOV_X/2 degrees */
2012-07-21 12:09:45 +00:00
RotatePointAroundVector(frustum[1].normal,
vup, vpn, 90 - r_newrefdef.fov_x / 2);
/* rotate VPN up by FOV_X/2 degrees */
2012-07-21 12:09:45 +00:00
RotatePointAroundVector(frustum[2].normal,
vright, vpn, 90 - r_newrefdef.fov_y / 2);
/* rotate VPN down by FOV_X/2 degrees */
2012-07-21 12:09:45 +00:00
RotatePointAroundVector(frustum[3].normal, vright, vpn,
-(90 - r_newrefdef.fov_y / 2));
2012-07-21 12:09:45 +00:00
for (i = 0; i < 4; i++)
{
2012-07-21 12:09:45 +00:00
frustum[i].type = PLANE_ANYZ;
frustum[i].dist = DotProduct(r_origin, frustum[i].normal);
frustum[i].signbits = R_SignbitsForPlane(&frustum[i]);
}
}
void
2012-07-21 12:09:45 +00:00
R_SetupFrame(void)
{
int i;
mleaf_t *leaf;
r_framecount++;
/* build the transformation matrix for the given view angles */
2012-07-21 12:09:45 +00:00
VectorCopy(r_newrefdef.vieworg, r_origin);
2012-07-21 12:09:45 +00:00
AngleVectors(r_newrefdef.viewangles, vpn, vright, vup);
/* current viewcluster */
2012-07-21 12:09:45 +00:00
if (!(r_newrefdef.rdflags & RDF_NOWORLDMODEL))
{
r_oldviewcluster = r_viewcluster;
r_oldviewcluster2 = r_viewcluster2;
2012-07-21 12:09:45 +00:00
leaf = Mod_PointInLeaf(r_origin, r_worldmodel);
r_viewcluster = r_viewcluster2 = leaf->cluster;
/* check above and below so crossing solid water doesn't draw wrong */
2012-07-21 12:09:45 +00:00
if (!leaf->contents)
{
/* look down a bit */
vec3_t temp;
2012-07-21 12:09:45 +00:00
VectorCopy(r_origin, temp);
temp[2] -= 16;
leaf = Mod_PointInLeaf(temp, r_worldmodel);
2012-07-21 12:09:45 +00:00
if (!(leaf->contents & CONTENTS_SOLID) &&
(leaf->cluster != r_viewcluster2))
{
r_viewcluster2 = leaf->cluster;
}
}
else
{
/* look up a bit */
vec3_t temp;
2012-07-21 12:09:45 +00:00
VectorCopy(r_origin, temp);
temp[2] += 16;
leaf = Mod_PointInLeaf(temp, r_worldmodel);
2012-07-21 12:09:45 +00:00
if (!(leaf->contents & CONTENTS_SOLID) &&
(leaf->cluster != r_viewcluster2))
{
r_viewcluster2 = leaf->cluster;
}
}
}
2012-07-21 12:09:45 +00:00
for (i = 0; i < 4; i++)
{
2012-07-21 12:09:45 +00:00
v_blend[i] = r_newrefdef.blend[i];
}
c_brush_polys = 0;
c_alias_polys = 0;
/* clear out the portion of the screen that the NOWORLDMODEL defines */
2012-07-21 12:09:45 +00:00
if (r_newrefdef.rdflags & RDF_NOWORLDMODEL)
{
2012-07-21 12:09:45 +00:00
qglEnable(GL_SCISSOR_TEST);
qglClearColor(0.3, 0.3, 0.3, 1);
qglScissor(r_newrefdef.x,
vid.height - r_newrefdef.height - r_newrefdef.y,
r_newrefdef.width, r_newrefdef.height);
qglClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
qglClearColor(1, 0, 0.5, 0.5);
qglDisable(GL_SCISSOR_TEST);
}
}
void
2012-07-21 12:09:45 +00:00
R_MYgluPerspective(GLdouble fovy, GLdouble aspect,
GLdouble zNear, GLdouble zFar)
{
GLdouble xmin, xmax, ymin, ymax;
2012-07-21 12:09:45 +00:00
ymax = zNear * tan(fovy * M_PI / 360.0);
ymin = -ymax;
xmin = ymin * aspect;
xmax = ymax * aspect;
2012-07-21 12:09:45 +00:00
xmin += -(2 * gl_state.camera_separation) / zNear;
xmax += -(2 * gl_state.camera_separation) / zNear;
2012-07-21 12:09:45 +00:00
qglFrustum(xmin, xmax, ymin, ymax, zNear, zFar);
}
void
2012-07-21 12:09:45 +00:00
R_SetupGL(void)
{
float screenaspect;
int x, x2, y2, y, w, h;
/* set up viewport */
2012-07-21 12:09:45 +00:00
x = floor(r_newrefdef.x * vid.width / vid.width);
x2 = ceil((r_newrefdef.x + r_newrefdef.width) * vid.width / vid.width);
y = floor(vid.height - r_newrefdef.y * vid.height / vid.height);
y2 = ceil(vid.height -
(r_newrefdef.y + r_newrefdef.height) * vid.height / vid.height);
w = x2 - x;
h = y - y2;
2012-07-21 12:09:45 +00:00
qglViewport(x, y2, w, h);
/* set up projection matrix */
2012-07-21 12:09:45 +00:00
screenaspect = (float)r_newrefdef.width / r_newrefdef.height;
qglMatrixMode(GL_PROJECTION);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
if (gl_farsee->value == 0)
{
R_MYgluPerspective(r_newrefdef.fov_y, screenaspect, 4, 4096);
}
else
{
R_MYgluPerspective(r_newrefdef.fov_y, screenaspect, 4, 8192);
2012-03-08 11:24:45 +00:00
}
2012-07-21 12:09:45 +00:00
qglCullFace(GL_FRONT);
2012-07-21 12:09:45 +00:00
qglMatrixMode(GL_MODELVIEW);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglRotatef(-90, 1, 0, 0); /* put Z going up */
qglRotatef(90, 0, 0, 1); /* put Z going up */
qglRotatef(-r_newrefdef.viewangles[2], 1, 0, 0);
qglRotatef(-r_newrefdef.viewangles[0], 0, 1, 0);
qglRotatef(-r_newrefdef.viewangles[1], 0, 0, 1);
qglTranslatef(-r_newrefdef.vieworg[0], -r_newrefdef.vieworg[1],
-r_newrefdef.vieworg[2]);
2012-07-21 12:09:45 +00:00
qglGetFloatv(GL_MODELVIEW_MATRIX, r_world_matrix);
/* set drawing parms */
2012-07-21 12:09:45 +00:00
if (gl_cull->value)
{
2012-07-21 12:09:45 +00:00
qglEnable(GL_CULL_FACE);
}
else
{
2012-07-21 12:09:45 +00:00
qglDisable(GL_CULL_FACE);
}
2012-07-21 12:09:45 +00:00
qglDisable(GL_BLEND);
qglDisable(GL_ALPHA_TEST);
qglEnable(GL_DEPTH_TEST);
}
void
2012-07-21 12:09:45 +00:00
R_Clear(void)
{
2012-07-21 12:09:45 +00:00
if (gl_ztrick->value)
{
static int trickframe;
2012-07-21 12:09:45 +00:00
if (gl_clear->value)
{
2012-07-21 12:09:45 +00:00
qglClear(GL_COLOR_BUFFER_BIT);
}
trickframe++;
2012-07-21 12:09:45 +00:00
if (trickframe & 1)
{
gldepthmin = 0;
gldepthmax = 0.49999;
2012-07-21 12:09:45 +00:00
qglDepthFunc(GL_LEQUAL);
}
else
{
gldepthmin = 1;
gldepthmax = 0.5;
2012-07-21 12:09:45 +00:00
qglDepthFunc(GL_GEQUAL);
}
}
else
{
2012-07-21 12:09:45 +00:00
if (gl_clear->value)
{
2012-07-21 12:09:45 +00:00
qglClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
}
else
{
2012-07-21 12:09:45 +00:00
qglClear(GL_DEPTH_BUFFER_BIT);
}
gldepthmin = 0;
gldepthmax = 1;
2012-07-21 12:09:45 +00:00
qglDepthFunc(GL_LEQUAL);
}
2012-07-21 12:09:45 +00:00
qglDepthRange(gldepthmin, gldepthmax);
if (gl_zfix->value)
{
if (gldepthmax > gldepthmin)
{
qglPolygonOffset(0.05, 1);
}
else
{
qglPolygonOffset(-0.05, -1);
}
}
/* stencilbuffer shadows */
2012-07-21 12:09:45 +00:00
if (gl_shadows->value && have_stencil && gl_stencilshadow->value)
{
2012-07-21 12:09:45 +00:00
qglClearStencil(1);
qglClear(GL_STENCIL_BUFFER_BIT);
}
}
void
2012-07-21 12:09:45 +00:00
R_Flash(void)
{
R_PolyBlend();
}
/*
* r_newrefdef must be set before the first call
*/
void
2012-07-21 12:09:45 +00:00
R_RenderView(refdef_t *fd)
{
2012-07-21 12:09:45 +00:00
if (gl_norefresh->value)
{
return;
}
r_newrefdef = *fd;
2012-07-21 12:09:45 +00:00
if (!r_worldmodel && !(r_newrefdef.rdflags & RDF_NOWORLDMODEL))
{
2012-07-21 12:09:45 +00:00
ri.Sys_Error(ERR_DROP, "R_RenderView: NULL worldmodel");
}
2012-07-21 12:09:45 +00:00
if (gl_speeds->value)
{
c_brush_polys = 0;
c_alias_polys = 0;
}
R_PushDlights();
2012-07-21 12:09:45 +00:00
if (gl_finish->value)
{
qglFinish();
}
R_SetupFrame();
R_SetFrustum();
R_SetupGL();
R_MarkLeaves(); /* done here so we know if we're in water */
R_DrawWorld();
R_DrawEntitiesOnList();
R_RenderDlights();
R_DrawParticles();
R_DrawAlphaSurfaces();
R_Flash();
2012-07-21 12:09:45 +00:00
if (gl_speeds->value)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "%4i wpoly %4i epoly %i tex %i lmaps\n",
c_brush_polys, c_alias_polys, c_visible_textures,
c_visible_lightmaps);
}
}
void
2012-07-21 12:09:45 +00:00
R_SetGL2D(void)
{
/* set 2D virtual screen size */
2012-07-21 12:09:45 +00:00
qglViewport(0, 0, vid.width, vid.height);
qglMatrixMode(GL_PROJECTION);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglOrtho(0, vid.width, vid.height, 0, -99999, 99999);
qglMatrixMode(GL_MODELVIEW);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglDisable(GL_DEPTH_TEST);
qglDisable(GL_CULL_FACE);
qglDisable(GL_BLEND);
qglEnable(GL_ALPHA_TEST);
qglColor4f(1, 1, 1, 1);
}
void
2012-07-21 12:09:45 +00:00
R_SetLightLevel(void)
{
vec3_t shadelight;
2012-07-21 12:09:45 +00:00
if (r_newrefdef.rdflags & RDF_NOWORLDMODEL)
{
return;
}
/* save off light value for server to look at */
2012-07-21 12:09:45 +00:00
R_LightPoint(r_newrefdef.vieworg, shadelight);
2012-07-21 12:09:45 +00:00
/* pick the greatest component, which should be the
* same as the mono value returned by software */
if (shadelight[0] > shadelight[1])
{
2012-07-21 12:09:45 +00:00
if (shadelight[0] > shadelight[2])
{
2012-07-21 12:09:45 +00:00
gl_lightlevel->value = 150 * shadelight[0];
}
else
{
2012-07-21 12:09:45 +00:00
gl_lightlevel->value = 150 * shadelight[2];
}
}
else
{
2012-07-21 12:09:45 +00:00
if (shadelight[1] > shadelight[2])
{
2012-07-21 12:09:45 +00:00
gl_lightlevel->value = 150 * shadelight[1];
}
else
{
2012-07-21 12:09:45 +00:00
gl_lightlevel->value = 150 * shadelight[2];
}
}
}
void
2012-07-21 12:09:45 +00:00
R_RenderFrame(refdef_t *fd)
{
2012-07-21 12:09:45 +00:00
R_RenderView(fd);
R_SetLightLevel();
R_SetGL2D();
}
void
2012-07-21 12:09:45 +00:00
R_Register(void)
{
2012-07-21 12:09:45 +00:00
gl_lefthand = ri.Cvar_Get("hand", "0", CVAR_USERINFO | CVAR_ARCHIVE);
gl_farsee = ri.Cvar_Get("gl_farsee", "0", CVAR_LATCH | CVAR_ARCHIVE);
gl_norefresh = ri.Cvar_Get("gl_norefresh", "0", 0);
gl_fullbright = ri.Cvar_Get("gl_fullbright", "0", 0);
gl_drawentities = ri.Cvar_Get("gl_drawentities", "1", 0);
gl_drawworld = ri.Cvar_Get("gl_drawworld", "1", 0);
gl_novis = ri.Cvar_Get("gl_novis", "0", 0);
gl_nocull = ri.Cvar_Get("gl_nocull", "0", 0);
gl_lerpmodels = ri.Cvar_Get("gl_lerpmodels", "1", 0);
gl_speeds = ri.Cvar_Get("gl_speeds", "0", 0);
gl_lightlevel = ri.Cvar_Get("gl_lightlevel", "0", 0);
gl_overbrightbits = ri.Cvar_Get("gl_overbrightbits", "2", CVAR_ARCHIVE);
gl_nosubimage = ri.Cvar_Get("gl_nosubimage", "0", 0);
gl_allow_software = ri.Cvar_Get("gl_allow_software", "0", 0);
gl_particle_min_size = ri.Cvar_Get("gl_particle_min_size", "2", CVAR_ARCHIVE);
gl_particle_max_size = ri.Cvar_Get("gl_particle_max_size", "40", CVAR_ARCHIVE);
gl_particle_size = ri.Cvar_Get("gl_particle_size", "40", CVAR_ARCHIVE);
gl_particle_att_a = ri.Cvar_Get("gl_particle_att_a", "0.01", CVAR_ARCHIVE);
gl_particle_att_b = ri.Cvar_Get("gl_particle_att_b", "0.0", CVAR_ARCHIVE);
gl_particle_att_c = ri.Cvar_Get("gl_particle_att_c", "0.01", CVAR_ARCHIVE);
gl_modulate = ri.Cvar_Get("gl_modulate", "1", CVAR_ARCHIVE);
gl_log = ri.Cvar_Get("gl_log", "0", 0);
gl_bitdepth = ri.Cvar_Get("gl_bitdepth", "0", 0);
gl_mode = ri.Cvar_Get("gl_mode", "5", CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_lightmap = ri.Cvar_Get("gl_lightmap", "0", 0);
gl_shadows = ri.Cvar_Get("gl_shadows", "0", CVAR_ARCHIVE);
gl_stencilshadow = ri.Cvar_Get("gl_stencilshadow", "0", CVAR_ARCHIVE);
gl_dynamic = ri.Cvar_Get("gl_dynamic", "1", 0);
gl_nobind = ri.Cvar_Get("gl_nobind", "0", 0);
gl_round_down = ri.Cvar_Get("gl_round_down", "1", 0);
gl_picmip = ri.Cvar_Get("gl_picmip", "0", 0);
gl_skymip = ri.Cvar_Get("gl_skymip", "0", 0);
gl_showtris = ri.Cvar_Get("gl_showtris", "0", 0);
gl_ztrick = ri.Cvar_Get("gl_ztrick", "0", 0);
gl_zfix = ri.Cvar_Get("gl_zfix", "0", 0);
2012-07-21 12:09:45 +00:00
gl_finish = ri.Cvar_Get("gl_finish", "0", CVAR_ARCHIVE);
gl_clear = ri.Cvar_Get("gl_clear", "0", 0);
gl_cull = ri.Cvar_Get("gl_cull", "1", 0);
gl_polyblend = ri.Cvar_Get("gl_polyblend", "1", 0);
gl_flashblend = ri.Cvar_Get("gl_flashblend", "0", 0);
gl_playermip = ri.Cvar_Get("gl_playermip", "0", 0);
gl_driver = ri.Cvar_Get("gl_driver", LIBGL, CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_texturemode = ri.Cvar_Get("gl_texturemode", "GL_LINEAR_MIPMAP_NEAREST", CVAR_ARCHIVE);
gl_texturealphamode = ri.Cvar_Get("gl_texturealphamode", "default", CVAR_ARCHIVE);
gl_texturesolidmode = ri.Cvar_Get("gl_texturesolidmode", "default", CVAR_ARCHIVE);
gl_anisotropic = ri.Cvar_Get("gl_anisotropic", "0", CVAR_ARCHIVE);
gl_anisotropic_avail = ri.Cvar_Get("gl_anisotropic_avail", "0", 0);
gl_lockpvs = ri.Cvar_Get("gl_lockpvs", "0", 0);
2012-07-21 12:09:45 +00:00
gl_vertex_arrays = ri.Cvar_Get("gl_vertex_arrays", "0", CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_ext_swapinterval = ri.Cvar_Get("gl_ext_swapinterval", "1", CVAR_ARCHIVE);
gl_ext_palettedtexture = ri.Cvar_Get("gl_ext_palettedtexture", "0", CVAR_ARCHIVE);
gl_ext_multitexture = ri.Cvar_Get("gl_ext_multitexture", "1", CVAR_ARCHIVE);
gl_ext_pointparameters = ri.Cvar_Get("gl_ext_pointparameters", "1", CVAR_ARCHIVE);
gl_ext_compiled_vertex_array = ri.Cvar_Get("gl_ext_compiled_vertex_array", "1", CVAR_ARCHIVE);
gl_ext_mtexcombine = ri.Cvar_Get("gl_ext_mtexcombine", "1", CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_drawbuffer = ri.Cvar_Get("gl_drawbuffer", "GL_BACK", 0);
gl_swapinterval = ri.Cvar_Get("gl_swapinterval", "1", CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_saturatelighting = ri.Cvar_Get("gl_saturatelighting", "0", 0);
2012-07-21 12:09:45 +00:00
vid_fullscreen = ri.Cvar_Get("vid_fullscreen", "0", CVAR_ARCHIVE);
vid_gamma = ri.Cvar_Get("vid_gamma", "1.0", CVAR_ARCHIVE);
vid_ref = ri.Cvar_Get("vid_ref", "soft", CVAR_ARCHIVE);
2012-07-21 12:09:45 +00:00
gl_customwidth = ri.Cvar_Get("gl_customwidth", "1024", CVAR_ARCHIVE);
gl_customheight = ri.Cvar_Get("gl_customheight", "768", CVAR_ARCHIVE);
2010-05-22 06:58:09 +00:00
#ifdef RETEXTURE
2012-07-21 12:09:45 +00:00
gl_retexturing = ri.Cvar_Get("gl_retexturing", "1", CVAR_ARCHIVE);
#endif
2012-07-21 12:09:45 +00:00
ri.Cmd_AddCommand("imagelist", R_ImageList_f);
ri.Cmd_AddCommand("screenshot", R_ScreenShot);
ri.Cmd_AddCommand("modellist", Mod_Modellist_f);
ri.Cmd_AddCommand("gl_strings", R_Strings);
}
qboolean
2012-07-21 12:09:45 +00:00
R_SetMode(void)
{
rserr_t err;
qboolean fullscreen;
fullscreen = vid_fullscreen->value;
vid_fullscreen->modified = false;
gl_mode->modified = false;
/* a bit hackish approach to enable custom resolutions:
2012-07-21 12:09:45 +00:00
Glimp_SetMode needs these values set for mode -1 */
vid.width = gl_customwidth->value;
vid.height = gl_customheight->value;
2012-07-21 12:09:45 +00:00
if ((err = GLimp_SetMode(&vid.width, &vid.height, gl_mode->value,
fullscreen)) == rserr_ok)
{
2012-07-21 12:09:45 +00:00
if (gl_mode->value == -1)
{
2012-08-02 12:47:18 +00:00
gl_state.prev_mode = 4; /* safe default for custom mode */
}
else
{
gl_state.prev_mode = gl_mode->value;
}
}
else
{
2012-07-21 12:09:45 +00:00
if (err == rserr_invalid_fullscreen)
{
2012-07-21 12:09:45 +00:00
ri.Cvar_SetValue("vid_fullscreen", 0);
vid_fullscreen->modified = false;
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "ref_gl::R_SetMode() - fullscreen unavailable in this mode\n");
2012-07-21 12:09:45 +00:00
if ((err = GLimp_SetMode(&vid.width, &vid.height, gl_mode->value, false)) == rserr_ok)
{
2012-07-21 12:09:45 +00:00
return true;
}
}
2012-07-21 12:09:45 +00:00
else if (err == rserr_invalid_mode)
{
2012-07-21 12:09:45 +00:00
ri.Cvar_SetValue("gl_mode", gl_state.prev_mode);
gl_mode->modified = false;
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "ref_gl::R_SetMode() - invalid mode\n");
}
/* try setting it back to something safe */
2012-07-21 12:09:45 +00:00
if ((err =GLimp_SetMode(&vid.width, &vid.height, gl_state.prev_mode, false)) != rserr_ok)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "ref_gl::R_SetMode() - could not revert to safe mode\n");
return false;
}
}
2012-07-21 12:09:45 +00:00
return true;
}
int
2012-07-21 12:09:45 +00:00
R_Init(void *hinstance, void *hWnd)
{
2012-07-21 12:09:45 +00:00
char renderer_buffer[1000];
char vendor_buffer[1000];
int err;
int j;
2012-07-21 12:09:45 +00:00
extern float r_turbsin[256];
2012-07-21 12:09:45 +00:00
for (j = 0; j < 256; j++)
{
2012-07-21 12:09:45 +00:00
r_turbsin[j] *= 0.5;
}
2012-04-25 09:50:06 +00:00
/* Options */
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "Refresher build options:\n");
2012-04-25 09:50:06 +00:00
#ifdef RETEXTURE
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, " + Retexturing support\n");
2012-04-25 09:50:06 +00:00
#else
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, " - Retexturing support\n");
2012-04-25 09:50:06 +00:00
#endif
2012-04-28 20:04:24 +00:00
#ifdef X11GAMMA
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, " + Gamma via X11\n");
2012-04-25 09:50:06 +00:00
#else
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, " - Gamma via X11\n");
2012-04-25 09:50:06 +00:00
#endif
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "Refresh: " REF_VERSION "\n");
Draw_GetPalette();
R_Register();
/* initialize our QGL dynamic bindings */
2012-07-21 12:09:45 +00:00
if (!QGL_Init(gl_driver->string))
{
QGL_Shutdown();
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "ref_gl::R_Init() - could not load \"%s\"\n",
gl_driver->string);
return -1;
}
/* initialize OS-specific parts of OpenGL */
2012-07-21 12:09:45 +00:00
if (!GLimp_Init())
{
QGL_Shutdown();
2012-07-21 12:09:45 +00:00
return -1;
}
/* set our "safe" modes */
gl_state.prev_mode = 3;
/* create the window and set up the context */
2012-07-21 12:09:45 +00:00
if (!R_SetMode())
{
QGL_Shutdown();
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "ref_gl::R_Init() - could not R_SetMode()\n");
return -1;
}
ri.Vid_MenuInit();
/* get our various GL strings */
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "\nOpenGL setting:\n", gl_config.vendor_string);
gl_config.vendor_string = (char *)qglGetString(GL_VENDOR);
ri.Con_Printf(PRINT_ALL, "GL_VENDOR: %s\n", gl_config.vendor_string);
gl_config.renderer_string = (char *)qglGetString(GL_RENDERER);
ri.Con_Printf(PRINT_ALL, "GL_RENDERER: %s\n", gl_config.renderer_string);
gl_config.version_string = (char *)qglGetString(GL_VERSION);
ri.Con_Printf(PRINT_ALL, "GL_VERSION: %s\n", gl_config.version_string);
gl_config.extensions_string = (char *)qglGetString(GL_EXTENSIONS);
ri.Con_Printf(PRINT_ALL, "GL_EXTENSIONS: %s\n", gl_config.extensions_string);
strncpy(renderer_buffer, gl_config.renderer_string, sizeof(renderer_buffer));
renderer_buffer[sizeof(renderer_buffer) - 1] = 0;
strlwr(renderer_buffer);
strncpy(vendor_buffer, gl_config.vendor_string, sizeof(vendor_buffer));
vendor_buffer[sizeof(vendor_buffer) - 1] = 0;
strlwr(vendor_buffer);
ri.Cvar_Set("scr_drawall", "0");
gl_config.allow_cds = true;
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "\n\nProbing for OpenGL extensions:\n");
/* grab extensions */
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_EXT_compiled_vertex_array") ||
strstr(gl_config.extensions_string, "GL_SGI_compiled_vertex_array"))
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_EXT_compiled_vertex_array\n");
qglLockArraysEXT = (void *)GetProcAddressGL("glLockArraysEXT");
qglUnlockArraysEXT = (void *)GetProcAddressGL("glUnlockArraysEXT");
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_EXT_compiled_vertex_array not found\n");
}
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_EXT_point_parameters"))
{
2012-07-21 12:09:45 +00:00
if (gl_ext_pointparameters->value)
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_EXT_point_parameters\n");
2012-07-21 12:09:45 +00:00
qglPointParameterfEXT = (void (APIENTRY *)(GLenum, GLfloat))
GetProcAddressGL("glPointParameterfEXT");
2012-07-21 12:09:45 +00:00
qglPointParameterfvEXT = (void (APIENTRY *)(GLenum, const GLfloat *))
GetProcAddressGL("glPointParameterfvEXT");
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_EXT_point_parameters\n");
}
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_EXT_point_parameters not found\n");
}
2012-07-21 12:09:45 +00:00
if (!qglColorTableEXT &&
strstr(gl_config.extensions_string, "GL_EXT_paletted_texture") &&
strstr(gl_config.extensions_string, "GL_EXT_shared_texture_palette"))
{
2012-07-21 12:09:45 +00:00
if (gl_ext_palettedtexture->value)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_EXT_shared_texture_palette\n");
qglColorTableEXT =
2012-07-21 12:09:45 +00:00
(void (APIENTRY *)(GLenum, GLenum, GLsizei, GLenum, GLenum,
const GLvoid *))GetProcAddressGL(
2012-07-21 12:09:45 +00:00
"glColorTableEXT");
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_EXT_shared_texture_palette\n");
}
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_EXT_shared_texture_palette not found\n");
}
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_ARB_multitexture"))
{
2012-07-21 12:09:45 +00:00
if (gl_ext_multitexture->value)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_ARB_multitexture\n");
qglMTexCoord2fSGIS = (void *)GetProcAddressGL("glMultiTexCoord2fARB");
qglActiveTextureARB = (void *)GetProcAddressGL("glActiveTextureARB");
qglClientActiveTextureARB = (void *)GetProcAddressGL("glClientActiveTextureARB");
QGL_TEXTURE0 = GL_TEXTURE0_ARB;
QGL_TEXTURE1 = GL_TEXTURE1_ARB;
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_ARB_multitexture\n");
}
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_ARB_multitexture not found\n");
}
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_SGIS_multitexture"))
{
2012-07-21 12:09:45 +00:00
if (qglActiveTextureARB)
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_SGIS_multitexture deprecated in favor of ARB_multitexture\n");
}
2012-07-21 12:09:45 +00:00
else if (gl_ext_multitexture->value)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_SGIS_multitexture\n");
qglMTexCoord2fSGIS = (void *)GetProcAddressGL("glMTexCoord2fSGIS");
qglSelectTextureSGIS = (void *)GetProcAddressGL("glSelectTextureSGIS");
QGL_TEXTURE0 = GL_TEXTURE0_SGIS;
QGL_TEXTURE1 = GL_TEXTURE1_SGIS;
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_SGIS_multitexture\n");
}
}
else
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_SGIS_multitexture not found\n");
}
2011-10-17 10:43:48 +00:00
gl_config.anisotropic = false;
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_EXT_texture_filter_anisotropic"))
2011-10-17 10:43:48 +00:00
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_EXT_texture_filter_anisotropic\n");
2011-10-17 10:43:48 +00:00
gl_config.anisotropic = true;
qglGetFloatv(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &gl_config.max_anisotropy);
ri.Cvar_SetValue("gl_anisotropic_avail", gl_config.max_anisotropy);
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_EXT_texture_filter_anisotropic not found\n");
2011-10-17 10:43:48 +00:00
gl_config.anisotropic = false;
gl_config.max_anisotropy = 0.0;
ri.Cvar_SetValue("gl_anisotropic_avail", 0.0);
2012-04-29 13:57:33 +00:00
}
2011-10-17 10:43:48 +00:00
2010-10-25 13:21:52 +00:00
gl_config.mtexcombine = false;
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_ARB_texture_env_combine"))
2010-10-25 13:21:52 +00:00
{
2012-07-21 12:09:45 +00:00
if (gl_ext_mtexcombine->value)
2010-10-25 13:21:52 +00:00
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_ARB_texture_env_combine\n");
2010-10-25 13:21:52 +00:00
gl_config.mtexcombine = true;
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_ARB_texture_env_combine\n");
2010-10-25 13:21:52 +00:00
}
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_ARB_texture_env_combine not found\n");
2010-10-25 13:21:52 +00:00
}
2012-07-21 12:09:45 +00:00
if (!gl_config.mtexcombine)
2010-10-25 13:21:52 +00:00
{
2012-07-21 12:09:45 +00:00
if (strstr(gl_config.extensions_string, "GL_EXT_texture_env_combine"))
2010-10-25 13:21:52 +00:00
{
2012-07-21 12:09:45 +00:00
if (gl_ext_mtexcombine->value)
2010-10-25 13:21:52 +00:00
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...using GL_EXT_texture_env_combine\n");
2010-10-25 13:21:52 +00:00
gl_config.mtexcombine = true;
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...ignoring GL_EXT_texture_env_combine\n");
2010-10-25 13:21:52 +00:00
}
}
else
{
2012-11-05 19:12:43 +00:00
ri.Con_Printf(PRINT_ALL, "...GL_EXT_texture_env_combine not found\n");
2010-10-25 13:21:52 +00:00
}
}
R_SetDefaultState();
R_InitImages();
Mod_Init();
R_InitParticleTexture();
Draw_InitLocal();
err = qglGetError();
2012-07-21 12:09:45 +00:00
if (err != GL_NO_ERROR)
{
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "glGetError() = 0x%x\n", err);
}
2012-07-21 12:09:45 +00:00
return true;
}
void
2012-07-21 12:09:45 +00:00
R_Shutdown(void)
{
2012-07-21 12:09:45 +00:00
ri.Cmd_RemoveCommand("modellist");
ri.Cmd_RemoveCommand("screenshot");
ri.Cmd_RemoveCommand("imagelist");
ri.Cmd_RemoveCommand("gl_strings");
Mod_FreeAll();
R_ShutdownImages();
2012-07-21 12:09:45 +00:00
/* shutdown OS specific OpenGL stuff like contexts, etc. */
GLimp_Shutdown();
/* shutdown our QGL subsystem */
QGL_Shutdown();
}
2012-07-21 12:09:45 +00:00
extern void UpdateHardwareGamma();
void
2012-07-21 12:09:45 +00:00
R_BeginFrame(float camera_separation)
{
gl_state.camera_separation = camera_separation;
/* change modes if necessary */
2012-07-21 12:09:45 +00:00
if (gl_mode->modified || vid_fullscreen->modified)
{
2012-07-21 12:09:45 +00:00
cvar_t *ref;
2012-07-21 12:09:45 +00:00
ref = ri.Cvar_Get("vid_ref", "gl", 0);
ref->modified = true;
}
2012-07-21 12:09:45 +00:00
if (gl_log->modified)
{
2012-07-21 12:09:45 +00:00
GLimp_EnableLogging(gl_log->value);
gl_log->modified = false;
}
2012-07-21 12:09:45 +00:00
if (gl_log->value)
{
GLimp_LogNewFrame();
}
2012-07-21 12:09:45 +00:00
if (vid_gamma->modified)
{
vid_gamma->modified = false;
2012-07-21 12:09:45 +00:00
if (gl_state.hwgamma)
{
UpdateHardwareGamma();
}
}
/* go into 2D mode */
2012-07-21 12:09:45 +00:00
qglViewport(0, 0, vid.width, vid.height);
qglMatrixMode(GL_PROJECTION);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglOrtho(0, vid.width, vid.height, 0, -99999, 99999);
qglMatrixMode(GL_MODELVIEW);
qglLoadIdentity();
2012-07-21 12:09:45 +00:00
qglDisable(GL_DEPTH_TEST);
qglDisable(GL_CULL_FACE);
qglDisable(GL_BLEND);
qglEnable(GL_ALPHA_TEST);
qglColor4f(1, 1, 1, 1);
/* draw buffer stuff */
2012-07-21 12:09:45 +00:00
if (gl_drawbuffer->modified)
{
gl_drawbuffer->modified = false;
2012-07-21 12:09:45 +00:00
if ((gl_state.camera_separation == 0) || !gl_state.stereo_enabled)
{
2012-07-21 12:09:45 +00:00
if (Q_stricmp(gl_drawbuffer->string, "GL_FRONT") == 0)
{
2012-07-21 12:09:45 +00:00
qglDrawBuffer(GL_FRONT);
}
else
{
2012-07-21 12:09:45 +00:00
qglDrawBuffer(GL_BACK);
}
}
}
/* texturemode stuff */
2012-07-21 12:09:45 +00:00
if (gl_texturemode->modified)
{
2012-07-21 12:09:45 +00:00
R_TextureMode(gl_texturemode->string);
gl_texturemode->modified = false;
}
2012-07-21 12:09:45 +00:00
if (gl_texturealphamode->modified)
{
2012-07-21 12:09:45 +00:00
R_TextureAlphaMode(gl_texturealphamode->string);
gl_texturealphamode->modified = false;
}
2012-07-21 12:09:45 +00:00
if (gl_texturesolidmode->modified)
{
2012-07-21 12:09:45 +00:00
R_TextureSolidMode(gl_texturesolidmode->string);
gl_texturesolidmode->modified = false;
}
/* clear screen if desired */
R_Clear();
}
void
2012-07-21 12:09:45 +00:00
R_SetPalette(const unsigned char *palette)
{
int i;
2012-07-21 12:09:45 +00:00
byte *rp = (byte *)r_rawpalette;
2012-07-21 12:09:45 +00:00
if (palette)
{
2012-07-21 12:09:45 +00:00
for (i = 0; i < 256; i++)
{
2012-07-21 12:09:45 +00:00
rp[i * 4 + 0] = palette[i * 3 + 0];
rp[i * 4 + 1] = palette[i * 3 + 1];
rp[i * 4 + 2] = palette[i * 3 + 2];
rp[i * 4 + 3] = 0xff;
}
}
else
{
2012-07-21 12:09:45 +00:00
for (i = 0; i < 256; i++)
{
2012-07-21 12:09:45 +00:00
rp[i * 4 + 0] = LittleLong(d_8to24table[i]) & 0xff;
rp[i * 4 + 1] = (LittleLong(d_8to24table[i]) >> 8) & 0xff;
rp[i * 4 + 2] = (LittleLong(d_8to24table[i]) >> 16) & 0xff;
rp[i * 4 + 3] = 0xff;
}
}
2012-07-21 12:09:45 +00:00
R_SetTexturePalette(r_rawpalette);
2012-07-21 12:09:45 +00:00
qglClearColor(0, 0, 0, 0);
qglClear(GL_COLOR_BUFFER_BIT);
qglClearColor(1, 0, 0.5, 0.5);
}
/* R_DrawBeam */
void
2012-07-21 12:09:45 +00:00
R_DrawBeam(entity_t *e)
{
int i;
float r, g, b;
vec3_t perpvec;
vec3_t direction, normalized_direction;
2012-07-21 12:09:45 +00:00
vec3_t start_points[NUM_BEAM_SEGS], end_points[NUM_BEAM_SEGS];
vec3_t oldorigin, origin;
2012-07-21 12:09:45 +00:00
oldorigin[0] = e->oldorigin[0];
oldorigin[1] = e->oldorigin[1];
oldorigin[2] = e->oldorigin[2];
2012-07-21 12:09:45 +00:00
origin[0] = e->origin[0];
origin[1] = e->origin[1];
origin[2] = e->origin[2];
2012-07-21 12:09:45 +00:00
normalized_direction[0] = direction[0] = oldorigin[0] - origin[0];
normalized_direction[1] = direction[1] = oldorigin[1] - origin[1];
normalized_direction[2] = direction[2] = oldorigin[2] - origin[2];
2012-07-21 12:09:45 +00:00
if (VectorNormalize(normalized_direction) == 0)
{
return;
}
2012-07-21 12:09:45 +00:00
PerpendicularVector(perpvec, normalized_direction);
VectorScale(perpvec, e->frame / 2, perpvec);
2012-07-21 12:09:45 +00:00
for (i = 0; i < 6; i++)
{
2012-07-21 12:09:45 +00:00
RotatePointAroundVector(start_points[i], normalized_direction, perpvec,
(360.0 / NUM_BEAM_SEGS) * i);
VectorAdd(start_points[i], origin, start_points[i]);
VectorAdd(start_points[i], direction, end_points[i]);
}
2012-07-21 12:09:45 +00:00
qglDisable(GL_TEXTURE_2D);
qglEnable(GL_BLEND);
qglDepthMask(GL_FALSE);
2012-07-21 12:09:45 +00:00
r = (LittleLong(d_8to24table[e->skinnum & 0xFF])) & 0xFF;
g = (LittleLong(d_8to24table[e->skinnum & 0xFF]) >> 8) & 0xFF;
b = (LittleLong(d_8to24table[e->skinnum & 0xFF]) >> 16) & 0xFF;
r *= 1 / 255.0F;
g *= 1 / 255.0F;
b *= 1 / 255.0F;
2012-07-21 12:09:45 +00:00
qglColor4f(r, g, b, e->alpha);
2012-07-21 12:09:45 +00:00
qglBegin(GL_TRIANGLE_STRIP);
2012-07-21 12:09:45 +00:00
for (i = 0; i < NUM_BEAM_SEGS; i++)
{
2012-07-21 12:09:45 +00:00
qglVertex3fv(start_points[i]);
qglVertex3fv(end_points[i]);
qglVertex3fv(start_points[(i + 1) % NUM_BEAM_SEGS]);
qglVertex3fv(end_points[(i + 1) % NUM_BEAM_SEGS]);
}
qglEnd();
2012-07-21 12:09:45 +00:00
qglEnable(GL_TEXTURE_2D);
qglDisable(GL_BLEND);
qglDepthMask(GL_TRUE);
}
refexport_t
2012-07-21 12:09:45 +00:00
R_GetRefAPI(refimport_t rimp)
{
refexport_t re;
ri = rimp;
re.api_version = API_VERSION;
re.BeginRegistration = R_BeginRegistration;
re.RegisterModel = R_RegisterModel;
re.RegisterSkin = R_RegisterSkin;
re.RegisterPic = Draw_FindPic;
re.SetSky = R_SetSky;
re.EndRegistration = R_EndRegistration;
re.RenderFrame = R_RenderFrame;
re.DrawGetPicSize = Draw_GetPicSize;
re.DrawPic = Draw_Pic;
re.DrawStretchPic = Draw_StretchPic;
re.DrawChar = Draw_Char;
re.DrawTileClear = Draw_TileClear;
re.DrawFill = Draw_Fill;
re.DrawFadeScreen = Draw_FadeScreen;
re.DrawStretchRaw = Draw_StretchRaw;
re.Init = R_Init;
re.Shutdown = R_Shutdown;
re.CinematicSetPalette = R_SetPalette;
re.BeginFrame = R_BeginFrame;
re.EndFrame = GLimp_EndFrame;
2010-10-19 12:43:11 +00:00
re.AppActivate = NULL;
Swap_Init();
2012-07-21 12:09:45 +00:00
return re;
}
/*
2012-11-06 19:35:02 +00:00
* this is only here so the functions
* in shared source files can link
*/
void
2012-07-21 12:09:45 +00:00
Sys_Error(char *error, ...)
{
va_list argptr;
2012-07-21 12:09:45 +00:00
char text[1024];
2012-07-21 12:09:45 +00:00
va_start(argptr, error);
vsprintf(text, error, argptr);
va_end(argptr);
2012-07-21 12:09:45 +00:00
ri.Sys_Error(ERR_FATAL, "%s", text);
}
void
2012-07-21 12:09:45 +00:00
Com_Printf(char *fmt, ...)
{
va_list argptr;
2012-07-21 12:09:45 +00:00
char text[1024];
2012-07-21 12:09:45 +00:00
va_start(argptr, fmt);
vsprintf(text, fmt, argptr);
va_end(argptr);
2012-07-21 12:09:45 +00:00
ri.Con_Printf(PRINT_ALL, "%s", text);
}
2012-07-21 12:09:45 +00:00