mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-21 08:31:00 +00:00
More stencil sky stuff
This commit is contained in:
parent
049ceecca5
commit
373038231c
5 changed files with 75 additions and 20 deletions
|
@ -53,6 +53,8 @@ void RenderPolyBsp::Render()
|
||||||
PvsSectors.clear();
|
PvsSectors.clear();
|
||||||
ScreenSprites.clear();
|
ScreenSprites.clear();
|
||||||
PolyStencilBuffer::Instance()->Clear(viewwidth, viewheight, 0);
|
PolyStencilBuffer::Instance()->Clear(viewwidth, viewheight, 0);
|
||||||
|
PolySubsectorGBuffer::Instance()->Resize(dc_pitch, viewheight);
|
||||||
|
NextSubsectorDepth = 0;
|
||||||
|
|
||||||
// Perspective correct:
|
// Perspective correct:
|
||||||
float ratio = WidescreenRatio;
|
float ratio = WidescreenRatio;
|
||||||
|
@ -75,7 +77,7 @@ void RenderPolyBsp::Render()
|
||||||
else
|
else
|
||||||
RenderNode(nodes + numnodes - 1); // The head node is the last node output.
|
RenderNode(nodes + numnodes - 1); // The head node is the last node output.
|
||||||
|
|
||||||
// Render front to back using the stencil buffer
|
// Render front to back
|
||||||
if (r_debug_cull)
|
if (r_debug_cull)
|
||||||
{
|
{
|
||||||
for (auto it = PvsSectors.rbegin(); it != PvsSectors.rend(); ++it)
|
for (auto it = PvsSectors.rbegin(); it != PvsSectors.rend(); ++it)
|
||||||
|
@ -128,17 +130,21 @@ void RenderPolyBsp::RenderSubsector(subsector_t *sub)
|
||||||
sector_t *frontsector = sub->sector;
|
sector_t *frontsector = sub->sector;
|
||||||
frontsector->MoreFlags |= SECF_DRAWN;
|
frontsector->MoreFlags |= SECF_DRAWN;
|
||||||
|
|
||||||
|
uint32_t subsectorDepth = NextSubsectorDepth++;
|
||||||
|
|
||||||
for (uint32_t i = 0; i < sub->numlines; i++)
|
for (uint32_t i = 0; i < sub->numlines; i++)
|
||||||
{
|
{
|
||||||
seg_t *line = &sub->firstline[i];
|
seg_t *line = &sub->firstline[i];
|
||||||
if (line->sidedef == nullptr || !(line->sidedef->Flags & WALLF_POLYOBJ))
|
if (line->sidedef == nullptr || !(line->sidedef->Flags & WALLF_POLYOBJ))
|
||||||
AddLine(line, frontsector);
|
AddLine(line, frontsector, subsectorDepth);
|
||||||
}
|
}
|
||||||
|
|
||||||
FTextureID floorpicnum = frontsector->GetTexture(sector_t::floor);
|
FTextureID floorpicnum = frontsector->GetTexture(sector_t::floor);
|
||||||
FTexture *floortex = TexMan(floorpicnum);
|
FTexture *floortex = TexMan(floorpicnum);
|
||||||
if (floortex->UseType != FTexture::TEX_Null)
|
if (floortex->UseType != FTexture::TEX_Null)
|
||||||
{
|
{
|
||||||
|
bool isSky = floorpicnum == skyflatnum;
|
||||||
|
|
||||||
TriVertex *vertices = PolyVertexBuffer::GetVertices(sub->numlines);
|
TriVertex *vertices = PolyVertexBuffer::GetVertices(sub->numlines);
|
||||||
if (!vertices)
|
if (!vertices)
|
||||||
return;
|
return;
|
||||||
|
@ -157,17 +163,22 @@ void RenderPolyBsp::RenderSubsector(subsector_t *sub)
|
||||||
else if (fixedcolormap)
|
else if (fixedcolormap)
|
||||||
uniforms.light = 256;
|
uniforms.light = 256;
|
||||||
uniforms.flags = 0;
|
uniforms.flags = 0;
|
||||||
|
uniforms.subsectorDepth = isSky ? SkySubsectorDepth : subsectorDepth;
|
||||||
bool isSky = floorpicnum == skyflatnum;
|
|
||||||
if (!isSky)
|
if (!isSky)
|
||||||
|
{
|
||||||
PolyTriangleDrawer::draw(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, floortex, 0);
|
PolyTriangleDrawer::draw(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, floortex, 0);
|
||||||
PolyTriangleDrawer::stencil(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, isSky ? 255 : 254);
|
PolyTriangleDrawer::stencil(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
PolyTriangleDrawer::fill(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
FTextureID ceilpicnum = frontsector->GetTexture(sector_t::ceiling);
|
FTextureID ceilpicnum = frontsector->GetTexture(sector_t::ceiling);
|
||||||
FTexture *ceiltex = TexMan(ceilpicnum);
|
FTexture *ceiltex = TexMan(ceilpicnum);
|
||||||
if (ceiltex->UseType != FTexture::TEX_Null)
|
if (ceiltex->UseType != FTexture::TEX_Null)
|
||||||
{
|
{
|
||||||
|
bool isSky = ceilpicnum == skyflatnum;
|
||||||
|
|
||||||
TriVertex *vertices = PolyVertexBuffer::GetVertices(sub->numlines);
|
TriVertex *vertices = PolyVertexBuffer::GetVertices(sub->numlines);
|
||||||
if (!vertices)
|
if (!vertices)
|
||||||
return;
|
return;
|
||||||
|
@ -186,11 +197,14 @@ void RenderPolyBsp::RenderSubsector(subsector_t *sub)
|
||||||
else if (fixedcolormap)
|
else if (fixedcolormap)
|
||||||
uniforms.light = 256;
|
uniforms.light = 256;
|
||||||
uniforms.flags = 0;
|
uniforms.flags = 0;
|
||||||
|
uniforms.subsectorDepth = isSky ? SkySubsectorDepth : subsectorDepth;
|
||||||
bool isSky = ceilpicnum == skyflatnum;
|
|
||||||
if (!isSky)
|
if (!isSky)
|
||||||
|
{
|
||||||
PolyTriangleDrawer::draw(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, ceiltex, 0);
|
PolyTriangleDrawer::draw(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, ceiltex, 0);
|
||||||
PolyTriangleDrawer::stencil(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, isSky ? 255 : 254);
|
PolyTriangleDrawer::stencil(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
PolyTriangleDrawer::fill(uniforms, vertices, sub->numlines, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
SpriteRange sprites = GetSpritesForSector(sub->sector);
|
SpriteRange sprites = GetSpritesForSector(sub->sector);
|
||||||
|
@ -198,9 +212,9 @@ void RenderPolyBsp::RenderSubsector(subsector_t *sub)
|
||||||
{
|
{
|
||||||
AActor *thing = SortedSprites[sprites.Start + i].Thing;
|
AActor *thing = SortedSprites[sprites.Start + i].Thing;
|
||||||
if ((thing->renderflags & RF_SPRITETYPEMASK) == RF_WALLSPRITE)
|
if ((thing->renderflags & RF_SPRITETYPEMASK) == RF_WALLSPRITE)
|
||||||
AddWallSprite(thing, sub);
|
AddWallSprite(thing, sub, subsectorDepth);
|
||||||
else
|
else
|
||||||
AddSprite(thing, sub);
|
AddSprite(thing, sub, subsectorDepth);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -224,7 +238,7 @@ SpriteRange RenderPolyBsp::GetSpritesForSector(sector_t *sector)
|
||||||
return range;
|
return range;
|
||||||
}
|
}
|
||||||
|
|
||||||
void RenderPolyBsp::AddLine(seg_t *line, sector_t *frontsector)
|
void RenderPolyBsp::AddLine(seg_t *line, sector_t *frontsector, uint32_t subsectorDepth)
|
||||||
{
|
{
|
||||||
// Reject lines not facing viewer
|
// Reject lines not facing viewer
|
||||||
DVector2 pt1 = line->v1->fPos() - ViewPos;
|
DVector2 pt1 = line->v1->fPos() - ViewPos;
|
||||||
|
@ -241,6 +255,7 @@ void RenderPolyBsp::AddLine(seg_t *line, sector_t *frontsector)
|
||||||
wall.Line = line;
|
wall.Line = line;
|
||||||
wall.Colormap = frontsector->ColorMap;
|
wall.Colormap = frontsector->ColorMap;
|
||||||
wall.Masked = false;
|
wall.Masked = false;
|
||||||
|
wall.SubsectorDepth = subsectorDepth;
|
||||||
|
|
||||||
if (line->backsector == nullptr)
|
if (line->backsector == nullptr)
|
||||||
{
|
{
|
||||||
|
@ -341,7 +356,7 @@ bool RenderPolyBsp::IsThingCulled(AActor *thing)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void RenderPolyBsp::AddSprite(AActor *thing, subsector_t *sub)
|
void RenderPolyBsp::AddSprite(AActor *thing, subsector_t *sub, uint32_t subsectorDepth)
|
||||||
{
|
{
|
||||||
if (IsThingCulled(thing))
|
if (IsThingCulled(thing))
|
||||||
return;
|
return;
|
||||||
|
@ -428,10 +443,11 @@ void RenderPolyBsp::AddSprite(AActor *thing, subsector_t *sub)
|
||||||
uniforms.objectToClip = worldToClip;
|
uniforms.objectToClip = worldToClip;
|
||||||
uniforms.light = (uint32_t)((thing->Sector->lightlevel + actualextralight) / 255.0f * 256.0f);
|
uniforms.light = (uint32_t)((thing->Sector->lightlevel + actualextralight) / 255.0f * 256.0f);
|
||||||
uniforms.flags = 0;
|
uniforms.flags = 0;
|
||||||
PolyTriangleDrawer::draw(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, tex, 254);
|
uniforms.subsectorDepth = subsectorDepth;
|
||||||
|
PolyTriangleDrawer::draw(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, tex, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void RenderPolyBsp::AddWallSprite(AActor *thing, subsector_t *sub)
|
void RenderPolyBsp::AddWallSprite(AActor *thing, subsector_t *sub, uint32_t subsectorDepth)
|
||||||
{
|
{
|
||||||
if (IsThingCulled(thing))
|
if (IsThingCulled(thing))
|
||||||
return;
|
return;
|
||||||
|
@ -1076,10 +1092,15 @@ void RenderPolyWall::Render(const TriMatrix &worldToClip)
|
||||||
uniforms.objectToClip = worldToClip;
|
uniforms.objectToClip = worldToClip;
|
||||||
uniforms.light = (uint32_t)(GetLightLevel() / 255.0f * 256.0f);
|
uniforms.light = (uint32_t)(GetLightLevel() / 255.0f * 256.0f);
|
||||||
uniforms.flags = 0;
|
uniforms.flags = 0;
|
||||||
|
uniforms.subsectorDepth = IsSky ? RenderPolyBsp::SkySubsectorDepth : SubsectorDepth;
|
||||||
|
|
||||||
if (!IsSky)
|
if (!IsSky)
|
||||||
|
{
|
||||||
PolyTriangleDrawer::draw(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, tex, 0);
|
PolyTriangleDrawer::draw(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, tex, 0);
|
||||||
PolyTriangleDrawer::stencil(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, IsSky ? 255 : 254);
|
PolyTriangleDrawer::stencil(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
PolyTriangleDrawer::fill(uniforms, vertices, 4, TriangleDrawMode::Fan, true, 0, viewwidth, 0, viewheight, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
FTexture *RenderPolyWall::GetTexture()
|
FTexture *RenderPolyWall::GetTexture()
|
||||||
|
@ -1407,7 +1428,7 @@ void PolySkyDome::CreateDome()
|
||||||
|
|
||||||
void PolySkyDome::RenderRow(const TriUniforms &uniforms, FTexture *skytex, int row)
|
void PolySkyDome::RenderRow(const TriUniforms &uniforms, FTexture *skytex, int row)
|
||||||
{
|
{
|
||||||
PolyTriangleDrawer::draw(uniforms, &mVertices[mPrimStart[row]], mPrimStart[row + 1] - mPrimStart[row], TriangleDrawMode::Strip, false, 0, viewwidth, 0, viewheight, skytex, 255);
|
PolyTriangleDrawer::draw(uniforms, &mVertices[mPrimStart[row]], mPrimStart[row + 1] - mPrimStart[row], TriangleDrawMode::Strip, false, 0, viewwidth, 0, viewheight, skytex, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void PolySkyDome::RenderCapColorRow(const TriUniforms &uniforms, FTexture *skytex, int row, bool bottomCap)
|
void PolySkyDome::RenderCapColorRow(const TriUniforms &uniforms, FTexture *skytex, int row, bool bottomCap)
|
||||||
|
@ -1415,7 +1436,7 @@ void PolySkyDome::RenderCapColorRow(const TriUniforms &uniforms, FTexture *skyte
|
||||||
uint32_t solid = skytex->GetSkyCapColor(bottomCap);
|
uint32_t solid = skytex->GetSkyCapColor(bottomCap);
|
||||||
if (!r_swtruecolor)
|
if (!r_swtruecolor)
|
||||||
solid = RGB32k.RGB[(RPART(solid) >> 3)][(GPART(solid) >> 3)][(BPART(solid) >> 3)];
|
solid = RGB32k.RGB[(RPART(solid) >> 3)][(GPART(solid) >> 3)][(BPART(solid) >> 3)];
|
||||||
PolyTriangleDrawer::fill(uniforms, &mVertices[mPrimStart[row]], mPrimStart[row + 1] - mPrimStart[row], TriangleDrawMode::Fan, bottomCap, 0, viewwidth, 0, viewheight, solid, 255);
|
PolyTriangleDrawer::fill(uniforms, &mVertices[mPrimStart[row]], mPrimStart[row + 1] - mPrimStart[row], TriangleDrawMode::Fan, bottomCap, 0, viewwidth, 0, viewheight, solid, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void PolySkyDome::Render(const TriMatrix &worldToClip)
|
void PolySkyDome::Render(const TriMatrix &worldToClip)
|
||||||
|
|
11
src/r_poly.h
11
src/r_poly.h
|
@ -97,14 +97,16 @@ public:
|
||||||
void Render();
|
void Render();
|
||||||
void RenderScreenSprites();
|
void RenderScreenSprites();
|
||||||
|
|
||||||
|
static const uint32_t SkySubsectorDepth = 0xffffffff;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void RenderNode(void *node);
|
void RenderNode(void *node);
|
||||||
void RenderSubsector(subsector_t *sub);
|
void RenderSubsector(subsector_t *sub);
|
||||||
void AddLine(seg_t *line, sector_t *frontsector);
|
void AddLine(seg_t *line, sector_t *frontsector, uint32_t subsectorDepth);
|
||||||
TriVertex PlaneVertex(vertex_t *v1, sector_t *sector, const secplane_t &plane);
|
TriVertex PlaneVertex(vertex_t *v1, sector_t *sector, const secplane_t &plane);
|
||||||
|
|
||||||
void AddSprite(AActor *thing, subsector_t *sub);
|
void AddSprite(AActor *thing, subsector_t *sub, uint32_t subsectorDepth);
|
||||||
void AddWallSprite(AActor *thing, subsector_t *sub);
|
void AddWallSprite(AActor *thing, subsector_t *sub, uint32_t subsectorDepth);
|
||||||
bool IsThingCulled(AActor *thing);
|
bool IsThingCulled(AActor *thing);
|
||||||
visstyle_t GetSpriteVisStyle(AActor *thing, double z);
|
visstyle_t GetSpriteVisStyle(AActor *thing, double z);
|
||||||
FTexture *GetSpriteTexture(AActor *thing, /*out*/ bool &flipX);
|
FTexture *GetSpriteTexture(AActor *thing, /*out*/ bool &flipX);
|
||||||
|
@ -125,6 +127,8 @@ private:
|
||||||
bool IsSegmentCulled(int x1, int x2) const;
|
bool IsSegmentCulled(int x1, int x2) const;
|
||||||
|
|
||||||
std::vector<subsector_t *> PvsSectors;
|
std::vector<subsector_t *> PvsSectors;
|
||||||
|
uint32_t NextSubsectorDepth = 0;
|
||||||
|
|
||||||
TriMatrix worldToClip;
|
TriMatrix worldToClip;
|
||||||
|
|
||||||
std::vector<SpriteRange> SectorSpriteRanges;
|
std::vector<SpriteRange> SectorSpriteRanges;
|
||||||
|
@ -176,6 +180,7 @@ public:
|
||||||
FSWColormap *Colormap = nullptr;
|
FSWColormap *Colormap = nullptr;
|
||||||
bool Masked = false;
|
bool Masked = false;
|
||||||
bool IsSky = false;
|
bool IsSky = false;
|
||||||
|
uint32_t SubsectorDepth = 0;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
FTexture *GetTexture();
|
FTexture *GetTexture();
|
||||||
|
|
|
@ -94,6 +94,7 @@ void PolyTriangleDrawer::draw_arrays(const TriUniforms &uniforms, const TriVerte
|
||||||
args.stencilPitch = PolyStencilBuffer::Instance()->BlockWidth();
|
args.stencilPitch = PolyStencilBuffer::Instance()->BlockWidth();
|
||||||
args.stencilValues = PolyStencilBuffer::Instance()->Values();
|
args.stencilValues = PolyStencilBuffer::Instance()->Values();
|
||||||
args.stencilMasks = PolyStencilBuffer::Instance()->Masks();
|
args.stencilMasks = PolyStencilBuffer::Instance()->Masks();
|
||||||
|
args.subsectorGBuffer = PolySubsectorGBuffer::Instance()->Values();
|
||||||
|
|
||||||
TriVertex vert[3];
|
TriVertex vert[3];
|
||||||
if (mode == TriangleDrawMode::Normal)
|
if (mode == TriangleDrawMode::Normal)
|
||||||
|
|
|
@ -125,6 +125,32 @@ private:
|
||||||
uint32_t &ValueMask; // 4 * 4 + 2 * 2 + 1 bits indicating is Values are the same
|
uint32_t &ValueMask; // 4 * 4 + 2 * 2 + 1 bits indicating is Values are the same
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class PolySubsectorGBuffer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static PolySubsectorGBuffer *Instance()
|
||||||
|
{
|
||||||
|
static PolySubsectorGBuffer buffer;
|
||||||
|
return &buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Resize(int newwidth, int newheight)
|
||||||
|
{
|
||||||
|
width = newwidth;
|
||||||
|
height = newheight;
|
||||||
|
values.resize(width * height);
|
||||||
|
}
|
||||||
|
|
||||||
|
int Width() const { return width; }
|
||||||
|
int Height() const { return height; }
|
||||||
|
uint32_t *Values() { return values.data(); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
std::vector<uint32_t> values;
|
||||||
|
};
|
||||||
|
|
||||||
class PolyStencilBuffer
|
class PolyStencilBuffer
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -186,6 +212,7 @@ struct ScreenPolyTriangleDrawerArgs
|
||||||
int stencilPitch;
|
int stencilPitch;
|
||||||
uint8_t stencilTestValue;
|
uint8_t stencilTestValue;
|
||||||
uint8_t stencilWriteValue;
|
uint8_t stencilWriteValue;
|
||||||
|
uint32_t *subsectorGBuffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ScreenPolyTriangleDrawer
|
class ScreenPolyTriangleDrawer
|
||||||
|
|
|
@ -63,6 +63,7 @@ struct TriMatrix
|
||||||
struct TriUniforms
|
struct TriUniforms
|
||||||
{
|
{
|
||||||
uint32_t light;
|
uint32_t light;
|
||||||
|
uint32_t subsectorDepth;
|
||||||
|
|
||||||
uint16_t light_alpha;
|
uint16_t light_alpha;
|
||||||
uint16_t light_red;
|
uint16_t light_red;
|
||||||
|
|
Loading…
Reference in a new issue