mirror of
https://github.com/ZDoom/gzdoom-gles.git
synced 2025-01-20 15:40:45 +00:00
f45371e231
Using I_MSTime is not precise enough, because some camera textures can be done quicker. It was pointless anyway trying to make this multithreading-safe, the entire caching idea here makes no sense if two clippers can simultaneously work on the same level data without changing the memory organization and rendering it ineffective.
155 lines
3.1 KiB
C++
155 lines
3.1 KiB
C++
#ifndef __GL_CLIPPER
|
|
#define __GL_CLIPPER
|
|
|
|
#include "doomtype.h"
|
|
#include "xs_Float.h"
|
|
#include "r_utility.h"
|
|
#include "memarena.h"
|
|
|
|
angle_t R_PointToPseudoAngle(double x, double y);
|
|
|
|
// Used to speed up angle calculations during clipping
|
|
inline angle_t vertex_t::GetClipAngle()
|
|
{
|
|
return R_PointToPseudoAngle(p.X, p.Y);
|
|
}
|
|
|
|
class ClipNode
|
|
{
|
|
friend class Clipper;
|
|
|
|
ClipNode *prev, *next;
|
|
angle_t start, end;
|
|
|
|
bool operator== (const ClipNode &other)
|
|
{
|
|
return other.start == start && other.end == end;
|
|
}
|
|
};
|
|
|
|
|
|
class Clipper
|
|
{
|
|
static unsigned starttime;
|
|
FMemArena nodearena;
|
|
ClipNode * freelist = nullptr;
|
|
|
|
ClipNode * clipnodes = nullptr;
|
|
ClipNode * cliphead = nullptr;
|
|
ClipNode * silhouette = nullptr; // will be preserved even when RemoveClipRange is called
|
|
bool blocked = false;
|
|
|
|
static angle_t AngleToPseudo(angle_t ang);
|
|
bool IsRangeVisible(angle_t startangle, angle_t endangle);
|
|
void RemoveRange(ClipNode * cn);
|
|
void AddClipRange(angle_t startangle, angle_t endangle);
|
|
void RemoveClipRange(angle_t startangle, angle_t endangle);
|
|
void DoRemoveClipRange(angle_t start, angle_t end);
|
|
|
|
public:
|
|
|
|
Clipper();
|
|
|
|
void Clear();
|
|
|
|
void Free(ClipNode *node)
|
|
{
|
|
node->next = freelist;
|
|
freelist = node;
|
|
}
|
|
|
|
ClipNode * GetNew()
|
|
{
|
|
if (freelist)
|
|
{
|
|
ClipNode * p = freelist;
|
|
freelist = p->next;
|
|
return p;
|
|
}
|
|
else return (ClipNode*)nodearena.Alloc(sizeof(ClipNode));
|
|
}
|
|
|
|
ClipNode * NewRange(angle_t start, angle_t end)
|
|
{
|
|
ClipNode * c = GetNew();
|
|
|
|
c->start = start;
|
|
c->end = end;
|
|
c->next = c->prev = NULL;
|
|
return c;
|
|
}
|
|
|
|
void SetSilhouette();
|
|
|
|
bool SafeCheckRange(angle_t startAngle, angle_t endAngle)
|
|
{
|
|
if(startAngle > endAngle)
|
|
{
|
|
return (IsRangeVisible(startAngle, ANGLE_MAX) || IsRangeVisible(0, endAngle));
|
|
}
|
|
|
|
return IsRangeVisible(startAngle, endAngle);
|
|
}
|
|
|
|
void SafeAddClipRange(angle_t startangle, angle_t endangle)
|
|
{
|
|
if(startangle > endangle)
|
|
{
|
|
// The range has to added in two parts.
|
|
AddClipRange(startangle, ANGLE_MAX);
|
|
AddClipRange(0, endangle);
|
|
}
|
|
else
|
|
{
|
|
// Add the range as usual.
|
|
AddClipRange(startangle, endangle);
|
|
}
|
|
}
|
|
|
|
void SafeAddClipRangeRealAngles(angle_t startangle, angle_t endangle)
|
|
{
|
|
SafeAddClipRange(AngleToPseudo(startangle), AngleToPseudo(endangle));
|
|
}
|
|
|
|
|
|
void SafeRemoveClipRange(angle_t startangle, angle_t endangle)
|
|
{
|
|
if(startangle > endangle)
|
|
{
|
|
// The range has to added in two parts.
|
|
RemoveClipRange(startangle, ANGLE_MAX);
|
|
RemoveClipRange(0, endangle);
|
|
}
|
|
else
|
|
{
|
|
// Add the range as usual.
|
|
RemoveClipRange(startangle, endangle);
|
|
}
|
|
}
|
|
|
|
void SafeRemoveClipRangeRealAngles(angle_t startangle, angle_t endangle)
|
|
{
|
|
SafeRemoveClipRange(AngleToPseudo(startangle), AngleToPseudo(endangle));
|
|
}
|
|
|
|
void SetBlocked(bool on)
|
|
{
|
|
blocked = on;
|
|
}
|
|
|
|
bool IsBlocked() const
|
|
{
|
|
return blocked;
|
|
}
|
|
|
|
bool CheckBox(const float *bspcoord);
|
|
|
|
// Used to speed up angle calculations during clipping
|
|
inline angle_t GetClipAngle(vertex_t *v)
|
|
{
|
|
return unsigned(v->angletime) == starttime ? v->viewangle : (v->angletime = starttime, v->viewangle = R_PointToPseudoAngle(v->p.X, v->p.Y));
|
|
}
|
|
|
|
};
|
|
|
|
#endif
|