// //--------------------------------------------------------------------------- // // Copyright(C) 2000-2018 Christoph Oelckers // All rights reserved. // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 2 of the License, or // (at your option) any later version. // // 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 Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with this program. If not, see http://www.gnu.org/licenses/ // //-------------------------------------------------------------------------- // /* ** gl_drawinfo.cpp ** Basic scene draw info management class ** */ #include "hw_portal.h" #include "build.h" #include "hw_renderstate.h" #include "hw_drawinfo.h" #include "models/modeldata.h" #include "hw_clock.h" #include "hw_cvars.h" #include "hw_viewpointbuffer.h" #include "flatvertices.h" #include "hw_lightbuffer.h" #include "hw_vrmodes.h" #include "hw_clipper.h" #include "v_draw.h" #include "gamecvars.h" #include "gamestruct.h" #include "automap.h" #include "hw_voxels.h" #include "coreactor.h" #include "tiletexture.h" #include "buildtiles.h" EXTERN_CVAR(Float, r_visibility) CVAR(Bool, gl_no_skyclear, false, CVAR_ARCHIVE | CVAR_GLOBALCONFIG) CVAR(Bool, gl_texture, true, 0) CVAR(Float, gl_mask_threshold, 0.5f, CVAR_ARCHIVE | CVAR_GLOBALCONFIG) CVAR(Float, gl_mask_sprite_threshold, 0.5f, CVAR_ARCHIVE | CVAR_GLOBALCONFIG) BitArray gotsector; //========================================================================== // // // //========================================================================== class FDrawInfoList { public: TDeletingArray mList; HWDrawInfo * GetNew(); void Release(HWDrawInfo *); }; FDrawInfoList di_list; //========================================================================== // // Try to reuse the lists as often as possible as they contain resources that // are expensive to create and delete. // // Note: If multithreading gets used, this class needs synchronization. // //========================================================================== HWDrawInfo *FDrawInfoList::GetNew() { if (mList.Size() > 0) { HWDrawInfo *di; mList.Pop(di); return di; } return new HWDrawInfo(); } void FDrawInfoList::Release(HWDrawInfo * di) { di->ClearBuffers(); mList.Push(di); } //========================================================================== // // Sets up a new drawinfo struct // //========================================================================== HWDrawInfo *HWDrawInfo::StartDrawInfo(HWDrawInfo *parent, FRenderViewpoint &parentvp, HWViewpointUniforms *uniforms) { HWDrawInfo *di = di_list.GetNew(); di->StartScene(parentvp, uniforms); return di; } //========================================================================== // // // //========================================================================== static Clipper staticClipper; // Since all scenes are processed sequentially we only need one clipper. static HWDrawInfo * gl_drawinfo; // This is a linked list of all active DrawInfos and needed to free the memory arena after the last one goes out of scope. void HWDrawInfo::StartScene(FRenderViewpoint& parentvp, HWViewpointUniforms* uniforms) { mClipper = &staticClipper; Viewpoint = parentvp; //lightmode = Level->lightMode; if (uniforms) { VPUniforms = *uniforms; // The clip planes will never be inherited from the parent drawinfo. VPUniforms.mClipLine.X = -1000001.f; VPUniforms.mClipHeight = 0; } else { VPUniforms.mProjectionMatrix.loadIdentity(); VPUniforms.mViewMatrix.loadIdentity(); VPUniforms.mNormalViewMatrix.loadIdentity(); //VPUniforms.mViewHeight = viewheight; VPUniforms.mGlobVis = (2 / 65536.f) * (g_visibility + g_relvisibility) / r_ambientlight; VPUniforms.mPalLightLevels = numshades | (static_cast(gl_fogmode) << 8); if (isBuildSoftwareLighting()) VPUniforms.mPalLightLevels |= (5 << 16); VPUniforms.mClipLine.X = -10000000.0f; VPUniforms.mShadowmapFilter = gl_shadowmap_filter; } ClearBuffers(); for (int i = 0; i < GLDL_TYPES; i++) drawlists[i].Reset(); vpIndex = 0; // Fullbright information needs to be propagated from the main view. if (outer != nullptr) FullbrightFlags = outer->FullbrightFlags; else FullbrightFlags = 0; outer = gl_drawinfo; gl_drawinfo = this; } //========================================================================== // // // //========================================================================== HWDrawInfo *HWDrawInfo::EndDrawInfo() { assert(this == gl_drawinfo); for (int i = 0; i < GLDL_TYPES; i++) drawlists[i].Reset(); gl_drawinfo = outer; di_list.Release(this); if (gl_drawinfo == nullptr) ResetRenderDataAllocator(); return gl_drawinfo; } //========================================================================== // // // //========================================================================== void HWDrawInfo::ClearBuffers() { spriteindex = 0; mClipPortal = nullptr; mCurrentPortal = nullptr; } //----------------------------------------------------------------------------- // // R_FrustumAngle // //----------------------------------------------------------------------------- angle_t HWDrawInfo::FrustumAngle() { float WidescreenRatio = (float)screen->GetWidth() / screen->GetHeight(); float tilt = fabs(Viewpoint.HWAngles.Pitch.Degrees()); // If the pitch is larger than this you can look all around at a FOV of 90° if (tilt > 46.0f) return 0xffffffff; // ok, this is a gross hack that barely works... // but at least it doesn't overestimate too much... double floatangle = 2.0 + (45.0 + ((tilt / 1.9)))*Viewpoint.FieldOfView.Degrees() * 48.0 / AspectMultiplier(WidescreenRatio) / 90.0; angle_t a1 = DAngle::fromDeg(floatangle).BAMs(); if (a1 >= ANGLE_90) return 0xffffffff; // it's either below 90 or bust. return a1; } //----------------------------------------------------------------------------- // // Setup the modelview matrix // //----------------------------------------------------------------------------- void HWDrawInfo::SetViewMatrix(const FRotator &angles, float vx, float vy, float vz, bool mirror, bool planemirror) { float mult = mirror ? -1.f : 1.f; float planemult = planemirror ? -1 : 1;// Level->info->pixelstretch : Level->info->pixelstretch; VPUniforms.mViewMatrix.loadIdentity(); VPUniforms.mViewMatrix.rotate(angles.Roll.Degrees(), 0.0f, 0.0f, 1.0f); VPUniforms.mViewMatrix.rotate(angles.Pitch.Degrees(), 1.0f, 0.0f, 0.0f); VPUniforms.mViewMatrix.rotate(angles.Yaw.Degrees(), 0.0f, mult, 0.0f); VPUniforms.mViewMatrix.translate(vx * mult, -vz * planemult, -vy); VPUniforms.mViewMatrix.scale(-mult, planemult, 1); } //----------------------------------------------------------------------------- // // SetupView // Setup the view rotation matrix for the given viewpoint // //----------------------------------------------------------------------------- void HWDrawInfo::SetupView(FRenderState &state, float vx, float vy, float vz, bool mirror, bool planemirror) { auto &vp = Viewpoint; //vp.SetViewAngle(r_viewwindow); // todo: need to pass in. SetViewMatrix(vp.HWAngles, vx, vy, vz, mirror, planemirror); SetCameraPos(vp.Pos); VPUniforms.CalcDependencies(); VPUniforms.mLightBlendMode = 0; vpIndex = screen->mViewpoints->SetViewpoint(state, &VPUniforms); } //----------------------------------------------------------------------------- // // // //----------------------------------------------------------------------------- HWPortal * HWDrawInfo::FindPortal(const void * src) { int i = Portals.Size() - 1; while (i >= 0 && Portals[i] && Portals[i]->GetSource() != src) i--; return i >= 0 ? Portals[i] : nullptr; } //----------------------------------------------------------------------------- // // // //----------------------------------------------------------------------------- void HWDrawInfo::DispatchSprites() { for (unsigned i = 0; i < tsprites.Size(); i++) { auto tspr = tsprites.get(i); auto actor = tspr->ownerActor; auto texid = tspr->spritetexture(); if (actor == nullptr || tspr->scale.X == 0 || tspr->scale.Y == 0 || !texid.isValid()) continue; actor->spr.cstat2 |= CSTAT2_SPRITE_MAPPED; if (!(tspr->cstat2 & CSTAT2_SPRITE_NOANIMATE)) { tileUpdatePicnum(texid, (actor->GetIndex() & 16383)); } if (tspr->cstat2 & CSTAT2_SPRITE_FULLBRIGHT) tspr->shade = -127; tspr->picnum = legacyTileNum(texid); int tilenum = tspr->picnum; if (!(actor->sprext.renderflags & SPREXT_NOTMD) && !(tspr->cstat2 & CSTAT2_SPRITE_NOMODEL)) { auto pt = modelManager.GetModel(tspr->spritetexture(), tspr->pal); if (hw_models && pt && pt->modelid >= 0 && pt->framenum >= 0) { //HWSprite hwsprite; //if (hwsprite.ProcessModel(pt, tspr)) continue; } if (r_voxels) { auto vox = GetExtInfo(texid).tiletovox; if (vox >= 0 && voxmodels[vox]) { HWSprite hwsprite; if (hwsprite.ProcessVoxel(this, voxmodels[vox], tspr, tspr->sectp, voxrotate[vox])) continue; } } } if (actor->sprext.renderflags & SPREXT_AWAY1) { tspr->pos.XY() += tspr->Angles.Yaw.ToVector() * 0.125; } else if (actor->sprext.renderflags & SPREXT_AWAY2) { tspr->pos.XY() -= tspr->Angles.Yaw.ToVector() * 0.125; } switch (tspr->cstat & CSTAT_SPRITE_ALIGNMENT_MASK) { case CSTAT_SPRITE_ALIGNMENT_FACING: { HWSprite sprite; sprite.Process(this, tspr, tspr->sectp, false); break; } case CSTAT_SPRITE_ALIGNMENT_WALL: { HWWall wall; wall.ProcessWallSprite(this, tspr, tspr->sectp); break; } case CSTAT_SPRITE_ALIGNMENT_FLOOR: { HWFlat flat; flat.ProcessFlatSprite(this, tspr, tspr->sectp); break; } default: break; } } } //----------------------------------------------------------------------------- // // CreateScene // // creates the draw lists for the current scene // //----------------------------------------------------------------------------- void HWDrawInfo::CreateScene(bool portal) { const auto& vp = Viewpoint; angle_t a1 = FrustumAngle(); // reset the portal manager portalState.StartFrame(); ProcessAll.Clock(); // clip the scene and fill the drawlists screen->mVertexData->Map(); screen->mLights->Map(); tsprites.clear(); ingeo = false; geoofs = { 0,0 }; if(!portal) mClipper->SetVisibleRange(vp.RotAngle, a1); if (a1 != 0xffffffff) mDrawer.Init(this, mClipper, vp.Pos, vp.RotAngle - a1, vp.RotAngle + a1); else mDrawer.Init(this, mClipper, vp.Pos, 0, 0); if (vp.SectNums) mDrawer.RenderScene(vp.SectNums, vp.SectCount, portal); else mDrawer.RenderScene(&vp.SectCount, 1, portal); SetupSprite.Clock(); // vp is in render space, so we must convert back. gi->processSprites(tsprites, DVector3(vp.Pos.X, -vp.Pos.Y, -vp.Pos.Z), DAngle::fromBam(vp.RotAngle), vp.TicFrac); DispatchSprites(); SetupSprite.Unclock(); GeoEffect eff; int effsect = vp.SectNums ? vp.SectNums[0] : vp.SectCount; auto drawsectp = §or[effsect]; auto orgdrawsectp = drawsectp; // RR geometry hack. Ugh... // This just adds to the existing render list, so we must offset the effect areas to the same xy-space as the main one as we cannot change the view matrix. if (gi->GetGeoEffect(&eff, drawsectp)) { ingeo = true; geoofs = { (float)eff.geox[0], (float)eff.geoy[0] }; // process the first layer. for (int i = 0; i < eff.geocnt; i++) { auto sect = eff.geosectorwarp[i]; for (unsigned w = 0; w < sect->walls.Size(); w++) { auto wal = §->walls[w]; wal->pos.X += eff.geox[i]; wal->pos.Y += eff.geoy[i]; } sect->dirty = EDirty::AllDirty; if (eff.geosector[i] == drawsectp) drawsectp = eff.geosectorwarp[i]; } if (a1 != 0xffffffff) mDrawer.Init(this, mClipper, vp.Pos, vp.RotAngle - a1, vp.RotAngle + a1); else mDrawer.Init(this, mClipper, vp.Pos, 0, 0); int drawsect = sectindex(drawsectp); mDrawer.RenderScene(&drawsect, 1, false); for (int i = 0; i < eff.geocnt; i++) { auto sect = eff.geosectorwarp[i]; for (unsigned w = 0; w < sect->walls.Size(); w++) { auto wal = §->walls[w]; wal->pos.X -= eff.geox[i]; wal->pos.Y -= eff.geoy[i]; } } // Now the second layer. Same shit, different arrays. geoofs = { (float)eff.geox2[0], (float)eff.geoy2[0] }; for (int i = 0; i < eff.geocnt; i++) { auto sect = eff.geosectorwarp2[i]; for (unsigned w = 0; w < sect->walls.Size(); w++) { auto wal = §->walls[w]; wal->pos.X += eff.geox2[i]; wal->pos.Y += eff.geoy2[i]; } sect->dirty = EDirty::AllDirty; if (eff.geosector[i] == orgdrawsectp) drawsectp = eff.geosectorwarp2[i]; } if (a1 != 0xffffffff) mDrawer.Init(this, mClipper, vp.Pos, vp.RotAngle - a1, vp.RotAngle + a1); else mDrawer.Init(this, mClipper, vp.Pos, 0, 0); drawsect = sectindex(drawsectp); mDrawer.RenderScene(&drawsect, 1, false); for (int i = 0; i < eff.geocnt; i++) { auto sect = eff.geosectorwarp2[i]; for (unsigned w = 0; w < sect->walls.Size(); w++) { auto wal = §->walls[w]; wal->pos.X -= eff.geox2[i]; wal->pos.Y -= eff.geoy2[i]; } } ingeo = false; } screen->mLights->Unmap(); screen->mVertexData->Unmap(); ProcessAll.Unclock(); } //----------------------------------------------------------------------------- // // RenderScene // // Draws the current draw lists for the non GLSL renderer // //----------------------------------------------------------------------------- void HWDrawInfo::RenderScene(FRenderState &state) { RenderAll.Clock(); state.SetDepthMask(true); state.EnableFog(true); state.SetRenderStyle(STYLE_Source); // Part 1: solid geometry. This is set up so that there are no transparent parts state.SetDepthFunc(DF_Less); state.AlphaFunc(Alpha_GEqual, 0.f); state.ClearDepthBias(); state.EnableTexture(gl_texture); state.EnableBrightmap(true); drawlists[GLDL_PLAINWALLS].DrawWalls(this, state, false); drawlists[GLDL_PLAINFLATS].DrawFlats(this, state, false); // Part 2: masked geometry. This is set up so that only pixels with alpha>gl_mask_threshold will show state.AlphaFunc(Alpha_GEqual, gl_mask_threshold); // This list is masked, non-translucent walls. drawlists[GLDL_MASKEDWALLS].DrawWalls(this, state, false); // These lists must be drawn in two passes for color and depth to avoid depth fighting with overlapping entries drawlists[GLDL_MASKEDFLATS].SortFlats(this); drawlists[GLDL_MASKEDWALLSV].SortWallsHorz(this); drawlists[GLDL_MASKEDWALLSH].SortWallsVert(this); drawlists[GLDL_MASKEDWALLSD].SortWallsDiag(this); // these lists are only wall and floor sprites - often attached to walls and floors - so they need to be offset from the plane they may be attached to. drawlists[GLDL_MASKEDWALLSS].DrawWalls(this, state, false); // Each list must draw both its passes before the next one to ensure proper depth buffer contents. auto& list = drawlists[GLDL_MASKEDWALLSD].drawitems; unsigned i = 0; RenderWall.Clock(); while (i < list.Size()) { unsigned j; auto check = drawlists[GLDL_MASKEDWALLSD].walls[list[i].index]->walldist; state.SetDepthMask(false); for (j = i; j < list.Size() && drawlists[GLDL_MASKEDWALLSD].walls[list[j].index]->walldist == check; j++) { drawlists[GLDL_MASKEDWALLSD].walls[list[j].index]->DrawWall(this, state, false); } state.SetDepthMask(true); for (unsigned k = i; k < j; k++) { drawlists[GLDL_MASKEDWALLSD].walls[list[k].index]->DrawWall(this, state, false); } i = j; } RenderWall.Unclock(); state.SetDepthMask(false); drawlists[GLDL_MASKEDWALLSD].DrawWalls(this, state, false); state.SetDepthMask(true); state.SetColorMask(false); drawlists[GLDL_MASKEDWALLSD].DrawWalls(this, state, false); state.SetColorMask(true); state.SetDepthMask(false); drawlists[GLDL_MASKEDWALLSV].DrawWalls(this, state, false); state.SetDepthMask(true); state.SetColorMask(false); drawlists[GLDL_MASKEDWALLSV].DrawWalls(this, state, false); state.SetColorMask(true); state.SetDepthMask(false); drawlists[GLDL_MASKEDWALLSH].DrawWalls(this, state, false); state.SetDepthMask(true); state.SetColorMask(false); drawlists[GLDL_MASKEDWALLSH].DrawWalls(this, state, false); state.SetColorMask(true); state.SetDepthBias(-1, -128); drawlists[GLDL_MASKEDSLOPEFLATS].DrawFlats(this, state, false); state.SetDepthMask(false); drawlists[GLDL_MASKEDFLATS].DrawFlats(this, state, false); state.SetDepthMask(true); state.SetColorMask(false); drawlists[GLDL_MASKEDFLATS].DrawFlats(this, state, false); state.SetColorMask(true); state.ClearDepthBias(); drawlists[GLDL_MODELS].Draw(this, state, false); state.SetRenderStyle(STYLE_Translucent); state.SetDepthFunc(DF_LEqual); RenderAll.Unclock(); } //----------------------------------------------------------------------------- // // RenderTranslucent // //----------------------------------------------------------------------------- void HWDrawInfo::RenderTranslucent(FRenderState &state) { RenderAll.Clock(); state.SetDepthBias(-1, -160); // final pass: translucent stuff state.AlphaFunc(Alpha_GEqual, gl_mask_sprite_threshold); state.SetRenderStyle(STYLE_Translucent); state.EnableBrightmap(true); drawlists[GLDL_TRANSLUCENTBORDER].Draw(this, state, true); state.SetDepthMask(false); drawlists[GLDL_TRANSLUCENT].DrawSorted(this, state); state.EnableBrightmap(false); state.ClearDepthBias(); state.AlphaFunc(Alpha_GEqual, 0.5f); state.SetDepthMask(true); RenderAll.Unclock(); } //----------------------------------------------------------------------------- // // RenderTranslucent // //----------------------------------------------------------------------------- void HWDrawInfo::RenderPortal(HWPortal *p, FRenderState &state, bool usestencil) { auto gp = static_cast(p); gp->SetupStencil(this, state, usestencil); auto new_di = StartDrawInfo(this, Viewpoint, &VPUniforms); new_di->visibility = visibility; new_di->rellight = rellight; new_di->mCurrentPortal = gp; state.SetLightIndex(-1); gp->DrawContents(new_di, state); new_di->EndDrawInfo(); state.SetVertexBuffer(screen->mVertexData); screen->mViewpoints->Bind(state, vpIndex); gp->RemoveStencil(this, state, usestencil); } //----------------------------------------------------------------------------- // // Draws player sprites and color blend // //----------------------------------------------------------------------------- void HWDrawInfo::EndDrawScene(FRenderState &state) { state.EnableFog(false); #if 0 // [BB] HUD models need to be rendered here. const bool renderHUDModel = IsHUDModelForPlayerAvailable(players[consoleplayer].camera->player); if (renderHUDModel) { // [BB] The HUD model should be drawn over everything else already drawn. state.Clear(CT_Depth); DrawPlayerSprites(true, state); } #endif state.EnableStencil(false); state.SetViewport(screen->mScreenViewport.left, screen->mScreenViewport.top, screen->mScreenViewport.width, screen->mScreenViewport.height); // Restore standard rendering state state.SetRenderStyle(STYLE_Translucent); state.ResetColor(); state.EnableTexture(true); state.SetScissor(0, 0, -1, -1); } //----------------------------------------------------------------------------- // // sets 3D viewport and initial state // //----------------------------------------------------------------------------- void HWDrawInfo::Set3DViewport(FRenderState &state) { // Always clear all buffers with scissor test disabled. // This is faster on newer hardware because it allows the GPU to skip // reading from slower memory where the full buffers are stored. state.SetScissor(0, 0, -1, -1); state.Clear(CT_Color | CT_Depth | CT_Stencil); const auto &bounds = screen->mSceneViewport; state.SetViewport(bounds.left, bounds.top, bounds.width, bounds.height); state.SetScissor(bounds.left, bounds.top, bounds.width, bounds.height); state.EnableMultisampling(true); state.EnableDepthTest(true); state.EnableStencil(true); state.SetStencil(0, SOP_Keep, SF_AllOn); } //----------------------------------------------------------------------------- // // gl_drawscene - this function renders the scene from the current // viewpoint, including mirrors and skyboxes and other portals // It is assumed that the HWPortal::EndFrame returns with the // stencil, z-buffer and the projection matrix intact! // //----------------------------------------------------------------------------- void HWDrawInfo::DrawScene(int drawmode, bool portal) { static int recursion = 0; static int ssao_portals_available = 0; bool applySSAO = false; if (drawmode == DM_MAINVIEW) { ssao_portals_available = gl_ssao_portals; applySSAO = true; } else if (drawmode == DM_OFFSCREEN) { ssao_portals_available = 0; } else if (drawmode == DM_PORTAL && ssao_portals_available > 0) { applySSAO = (mCurrentPortal->AllowSSAO()/* || Level->flags3&LEVEL3_SKYBOXAO*/); ssao_portals_available--; } CreateScene(portal); auto& RenderState = *screen->RenderState(); RenderState.SetDepthMask(true); if (!gl_no_skyclear) portalState.RenderFirstSkyPortal(recursion, this, RenderState); RenderScene(RenderState); if (applySSAO && RenderState.GetPassType() == GBUFFER_PASS) { screen->AmbientOccludeScene(VPUniforms.mProjectionMatrix.get()[5]); screen->mViewpoints->Bind(RenderState, vpIndex); } // Handle all portals after rendering the opaque objects but before // doing all translucent stuff recursion++; portalState.EndFrame(this, RenderState); recursion--; RenderTranslucent(RenderState); } //----------------------------------------------------------------------------- // // R_RenderView - renders one view - either the screen or a camera texture // //----------------------------------------------------------------------------- void HWDrawInfo::ProcessScene(bool toscreen) { portalState.BeginScene(); DrawScene(toscreen ? DM_MAINVIEW : DM_OFFSCREEN, false); if (toscreen && isBlood()) { gotsector = mDrawer.GotSector(); // Blood needs this to implement some lighting effect hacks. Needs to be refactored to use better info. } }