csgo-2018-source/public/vjobs/edgegeom_shared.h
2021-07-24 21:11:47 -07:00

131 lines
3.3 KiB
C++

//========== Copyright © Valve Corporation, All rights reserved. ========
#if !defined( JOB_EDGE_GEOM_SHARED_HDR ) && defined( _PS3 )
#define JOB_EDGE_GEOM_SHARED_HDR
#include "ps3/spu_job_shared.h"
#include "vjobs/pcring.h"
#define EDGEGEOMRING_DEBUG_TRACE 128 // set to 0 to disable debug trace
// the max allocation, in bytes, made from job_edgegeom
#define EDGEGEOMRING_MAX_ALLOCATION ( 32 * 1024 )
#define EDGEGEOMRING_JOBQUEUE_TAG_COUNT 4
struct ALIGN16 EdgeGeomDebugTrace_t
{
uint8 m_nAllocResult;
uint8 m_nQueueTag;
uint16 m_nJobId;
uint32 m_eaEdgeGeomJts;
uint32 m_nPut;
uint32 m_nEnd;
uint32 m_nTagSignal[EDGEGEOMRING_JOBQUEUE_TAG_COUNT];
}ALIGN16_POST;
struct ALIGN16 CEdgeGeomRing_Mutable
{
SysFifo m_ibvbRing; // edge geom index buffer/vertex buffer ring
// WARNING. Although logically there may be any (2^x) number of queue tags for edgegeom jobs,
// The SPURS must only see 0 and 1 (even and odd) tags. Even tags must synchronize with even,
// odd tags must synchronize with odd.
uint32 m_ibvbRingSignal[EDGEGEOMRING_JOBQUEUE_TAG_COUNT];
uint32 m_nMaxJobId[EDGEGEOMRING_JOBQUEUE_TAG_COUNT];
uint m_nAtomicCollisionSpins;
uint m_nRsxWaitSpins;
uint m_nRingIncarnation;
uint m_nUsedSpus;
#if EDGEGEOMRING_DEBUG_TRACE
uint m_nNextDebugTrace;
#endif
}
ALIGN16_POST;
struct ALIGN128 CEdgeGeomRing: public CEdgeGeomRing_Mutable
{
// immutable part
uint m_eaLocalBaseAddress;
uint m_nIoOffsetDelta;
uint m_nIbvbRingLabel;
uint32 * m_eaIbvbRingLabel;
uint m_eaThis;
uint m_nDebuggerBreakMask;
#if EDGEGEOMRING_DEBUG_TRACE
uint m_nUseCounter;
EdgeGeomDebugTrace_t *m_eaDebugTrace;
bool m_enableDebugTrace;
#endif
public:
uintp Allocate( CellGcmContextData *pGcmCtx, uint nBytes, uint nTag );
void Test();
#ifndef SPU
void Init( void* eaBuffer, uint nBufferSize, uint nIoOffsetDelta, void * eaLocalBaseAddress, uint nLabel );
void Shutdown();
#endif
}
ALIGN128_POST;
enum EdgeGeomJobConstEnum_t
{
EDGEGEOMJOB_SCRATCH_SIZE = 64 * 1024
};
struct CEdgeGeomFeeder
{
uint m_nJobQueueTag; // the tag we're spawning jobs with, currently
uint m_nSpawnedJobsWithTag; // number of jobs we spawned with the tag
// max bytes that the jobs spawned with current tag can allocate;
// must not exceed 1/8 of the full ring buffer size because there may be 4 full tag switches between setting label
uint m_nSpawnedJobsWithTagReserveAllocate;
uint m_nTotalEdgeGeomJobCounter;
uint m_nIbvbRingSize;
bool Tick( uint nReserve );
#ifndef SPU
void Init( uint nIbvbRingSize );
#endif
};
inline bool CEdgeGeomFeeder::Tick( uint nReserve )
{
++m_nTotalEdgeGeomJobCounter;
++m_nSpawnedJobsWithTag;
m_nSpawnedJobsWithTagReserveAllocate += nReserve; // tentatively add the reserve to the same tag..
//if( ++m_nSpawnedJobsWithTag > 6 )
if( m_nSpawnedJobsWithTagReserveAllocate > m_nIbvbRingSize / 8 )
{
m_nJobQueueTag = ( m_nJobQueueTag + 1 ) & ( EDGEGEOMRING_JOBQUEUE_TAG_COUNT - 1 );
m_nSpawnedJobsWithTag = 0;
m_nSpawnedJobsWithTagReserveAllocate = nReserve;
return true ;// !( m_nJobQueueTag & 1 ); // only insert labels in one tag, to make sure they insert in serial fashion
}
else
{
return false;
}
}
namespace job_edgegeom
{
enum FlagEnum_t
{
FLAG_SWITCH_JOBQUEUE_TAG = 0x80,
FLAG_SKIP_VERTEX_CACHE_INVALIDATE = 0x40000000
};
}
#endif