mirror of
https://github.com/thug1src/thug.git
synced 2025-01-22 05:43:47 +00:00
2347 lines
71 KiB
C++
2347 lines
71 KiB
C++
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <libgraph.h>
|
||
|
#include <libdma.h>
|
||
|
#include <libsn.h>
|
||
|
#include <devvu1.h>
|
||
|
#include <math.h>
|
||
|
#include <sifrpc.h>
|
||
|
#include <sifdev.h>
|
||
|
#include <libpad.h>
|
||
|
#include <core\math.h>
|
||
|
#include <core\defines.h>
|
||
|
#include <core\debug.h>
|
||
|
#include <gfx\nxviewman.h>
|
||
|
#include <sys\profiler.h>
|
||
|
#include <sys\timer.h>
|
||
|
#include "mikemath.h"
|
||
|
#include "dma.h"
|
||
|
#include "vif.h"
|
||
|
#include "vu1.h"
|
||
|
#include "gif.h"
|
||
|
#include "gs.h"
|
||
|
#include "vu1code.h"
|
||
|
#include "dmacalls.h"
|
||
|
#include "line.h"
|
||
|
#include "chars.h"
|
||
|
#include "texture.h"
|
||
|
#include "material.h"
|
||
|
#include "group.h"
|
||
|
#include "scene.h"
|
||
|
#include "mesh.h"
|
||
|
#include "nx_init.h"
|
||
|
#include "render.h"
|
||
|
#include "interrupts.h"
|
||
|
#include "switches.h"
|
||
|
#include "sprite.h"
|
||
|
#include "instance.h"
|
||
|
#include "geomnode.h"
|
||
|
#include "occlude.h"
|
||
|
#include "resource.h"
|
||
|
#include "vu1context.h"
|
||
|
#include "light.h"
|
||
|
#include "fx.h"
|
||
|
#include "asmdma.h"
|
||
|
#include "vu1\newvu1code.h"
|
||
|
|
||
|
#include "gfx\nx.h"
|
||
|
#include "gfx\ngps\p_nxscene.h"
|
||
|
#include "gfx\ngps\p_nxtexture.h"
|
||
|
|
||
|
|
||
|
//#undef __USE_PROFILER__
|
||
|
|
||
|
|
||
|
namespace Inp
|
||
|
{
|
||
|
extern uint32 gDebugButtons[];
|
||
|
extern uint32 gDebugMakes[];
|
||
|
|
||
|
}
|
||
|
|
||
|
namespace NxPs2
|
||
|
{
|
||
|
|
||
|
extern uint32 *p_patch_PRMODE;
|
||
|
extern uint32 gPassMask1;
|
||
|
|
||
|
|
||
|
extern int geom_stats_total;
|
||
|
extern int geom_stats_inactive ;
|
||
|
extern int geom_stats_sky;
|
||
|
extern int geom_stats_transformed;
|
||
|
extern int geom_stats_skeletal;
|
||
|
extern int geom_stats_camera_sphere;
|
||
|
extern int geom_stats_clipcull;
|
||
|
extern int geom_stats_culled;
|
||
|
extern int geom_stats_leaf_culled;
|
||
|
extern int geom_stats_boxcheck;
|
||
|
extern int geom_stats_occludecheck;
|
||
|
extern int geom_stats_occluded;
|
||
|
extern int geom_stats_occludecheck;
|
||
|
extern int geom_stats_occluded;
|
||
|
extern int geom_stats_occludecheck;
|
||
|
extern int geom_stats_occluded;
|
||
|
extern int geom_stats_colored;
|
||
|
extern int geom_stats_leaf;
|
||
|
extern int geom_stats_wibbleUV;
|
||
|
extern int geom_stats_wibbleVC;
|
||
|
extern int geom_stats_sendcontext;
|
||
|
extern int geom_stats_sorted;
|
||
|
extern int geom_stats_shadow;
|
||
|
|
||
|
// Mick: Letterboxing on the ps2 is a post-processing
|
||
|
// step one frame ahead in the rendering pipeline (as
|
||
|
// compared to things like setting the FOV), so
|
||
|
// it must be delayed by 1 frame to synchronize it
|
||
|
// with such events.
|
||
|
bool ActualDoLetterbox = false;
|
||
|
bool DoLetterbox = false;
|
||
|
|
||
|
int DMAOverflowOK = false;
|
||
|
|
||
|
bool DoFlipCopy = true;
|
||
|
sGroup PrologueGroup;
|
||
|
float SkaterY;
|
||
|
|
||
|
int SingleRender = 0;
|
||
|
|
||
|
// skinning globals
|
||
|
Mat SkaterTransform;
|
||
|
Mat BoneTransforms[64];
|
||
|
int NumBones=0;
|
||
|
|
||
|
|
||
|
#if STENCIL_SHADOW
|
||
|
void CastStencilShadow(CInstance *pInstance, sGroup *p_group, Mth::Vector &ShadowVec, Mth::Vector &TweakVec);
|
||
|
#else
|
||
|
void CastShadow(CInstance *pInstance, sGroup *p_group);
|
||
|
#endif
|
||
|
|
||
|
void EnableFlipCopy(bool flip)
|
||
|
{
|
||
|
DoFlipCopy = flip;
|
||
|
}
|
||
|
|
||
|
bool FlipCopyEnabled()
|
||
|
{
|
||
|
return DoFlipCopy;
|
||
|
}
|
||
|
|
||
|
//////////////////////////////////////////////////////////////////
|
||
|
// void RenderPrologue(void)
|
||
|
//
|
||
|
// called at the start of a frame
|
||
|
// sets up the initial DMA list
|
||
|
// by setting up a prologue "group"
|
||
|
// consisting of an empty upload portion, and
|
||
|
// a render portion that flips the visible buffers,
|
||
|
// and sets up various vif settings
|
||
|
// finally we initilize vu1::Loc = vu1::Buffer=0;
|
||
|
//
|
||
|
// Note, Field will be 0 or 1, depending on which buffer we are drawing into
|
||
|
|
||
|
void RenderPrologue(void)
|
||
|
{
|
||
|
#ifdef __NOPT_ASSERT__
|
||
|
// a nasty patch by Mick to allow us to see the extra passes in flat shaded mode
|
||
|
if (gPassMask1)
|
||
|
{
|
||
|
if (Inp::gDebugMakes[0] & (1<<5)) // check for Circle pressed
|
||
|
{
|
||
|
*p_patch_PRMODE ^= 1;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
*p_patch_PRMODE = 1;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
|
||
|
|
||
|
// record the time
|
||
|
render::sTime = (int)Tmr::GetTime();
|
||
|
|
||
|
// set dma pointer to the start of the dma buffer
|
||
|
dma::pLoc = dma::pList[render::Field];
|
||
|
dma::pBase = dma::pLoc;
|
||
|
|
||
|
PrologueGroup.profile_color = 0x008080; // yellow = prologue group
|
||
|
|
||
|
// prologue render begins here
|
||
|
PrologueGroup.pRender[render::Field] = dma::pLoc;
|
||
|
PrologueGroup.Used[render::Field] = true; // prologue always used
|
||
|
|
||
|
// call the DMA routine to flip frame buffers and clear the draw buffer
|
||
|
// currently the flip is accomplished by rendering a full screen sprite from the
|
||
|
// draw buffer to the display buffer. This converts it from 32-bit to 16-bit (dithered)
|
||
|
// and means there is no actual buffer switching going on
|
||
|
// the draw buffer is always at the same address.
|
||
|
// When the game is running at 60fps, this copying takes place during the vblank.
|
||
|
|
||
|
//static bool last_flip_copy = true;
|
||
|
if (DoFlipCopy /*&& last_flip_copy*/) // DoFlipCopy is set/cleared by the loading screen code (p_NxLoadScreen.cpp)
|
||
|
{
|
||
|
if (ActualDoLetterbox)
|
||
|
dma::Gosub(FLIP_N_CLEAR_LETTERBOX,2);
|
||
|
else
|
||
|
dma::Gosub(FLIP_N_CLEAR,2);
|
||
|
ActualDoLetterbox=DoLetterbox;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
dma::Gosub(CLEAR_ZBUFFER,2);
|
||
|
}
|
||
|
//last_flip_copy = DoFlipCopy;
|
||
|
|
||
|
// set the GS to the default rendering state
|
||
|
dma::Gosub(SET_RENDERSTATE,2);
|
||
|
dma::Gosub(SET_FOGCOL,2);
|
||
|
|
||
|
// upload VU1 microcode (the code in microcode.dsm)
|
||
|
// This is done every frame, at the start of the frame
|
||
|
// it actually only needs doing once, but it does not really take
|
||
|
// any time, and doing it every frame gives us the flexibility to uploaded new
|
||
|
// microcode at a later stage in rendering.
|
||
|
// for now, all the microcode fits in the 16K code area of VU1
|
||
|
dma::Tag(dma::ref, ((uint8 *)&MPGEnd-(uint8 *)&MPGStart+15)/16, (uint)&MPGStart);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
|
||
|
// VIF1 and VU1 setup
|
||
|
dma::BeginTag(dma::end, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::STMASK(0);
|
||
|
vif::STMOD(0);
|
||
|
vif::STCYCL(1,1);
|
||
|
vif::BASE(0);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
dma::EndTag();
|
||
|
|
||
|
// prepare all groups for rendering
|
||
|
for (sGroup *p_group=sGroup::pHead; p_group; p_group=p_group->pNext)
|
||
|
{
|
||
|
// initialise the dma list
|
||
|
dma::BeginList(p_group);
|
||
|
|
||
|
// assume textures aren't used until something actually gets rendered
|
||
|
p_group->Used[render::Field] = false;
|
||
|
}
|
||
|
dma::sp_group = NULL;
|
||
|
|
||
|
|
||
|
// while the dma context is null, set up an empty upload for the particles group and fog group
|
||
|
sGroup::pParticles->pUpload[render::Field] = dma::pLoc;
|
||
|
sGroup::pFog->pUpload[render::Field] = dma::pLoc;
|
||
|
dma::Tag(dma::end, 0, 0);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
|
||
|
// signal that the particle group is used
|
||
|
sGroup::pParticles->Used[render::Field] = true;
|
||
|
|
||
|
// reset current marker number
|
||
|
render::sMarkerIndex = 0;
|
||
|
|
||
|
// count particles
|
||
|
render::sTotalNewParticles = 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void RenderWorld(Mth::Matrix* camera_orient, Mth::Vector* camera_pos, float view_angle, float screen_aspect)
|
||
|
{
|
||
|
Mth::Matrix transform(*camera_orient);
|
||
|
transform[3] = *camera_pos;
|
||
|
|
||
|
// need the transform to be well-formed as a 4x4 matrix
|
||
|
transform[0][3] = 0.0f;
|
||
|
transform[1][3] = 0.0f;
|
||
|
transform[2][3] = 0.0f;
|
||
|
transform[3][3] = 1.0f;
|
||
|
|
||
|
Mth::Rect viewport1(0.0f, 0.0f, 1.0f, 1.0f);
|
||
|
RenderViewport(0, transform, viewport1, view_angle, screen_aspect, -1.0f, -100000.0f);
|
||
|
}
|
||
|
|
||
|
void RenderInitImmediateMode()
|
||
|
{
|
||
|
dma::SetList(sGroup::pEpilogue);
|
||
|
|
||
|
// epilogue render (a terrible hack to put this here!)
|
||
|
//sGroup::pEpilogue->pRender[render::Field] = dma::pLoc;
|
||
|
sGroup::pEpilogue->Used[render::Field] = true; // epilogue always rendered
|
||
|
|
||
|
// VIF1 and VU1 setup
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::STROW((int)render::RowRegI[0], (int)render::RowRegI[1], (int)render::RowRegI[2], (int)render::RowRegI[3]);
|
||
|
vif::STMASK(0);
|
||
|
vif::STMOD(0);
|
||
|
vif::STCYCL(1,1);
|
||
|
vif::BASE(0);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vu1::Loc = vu1::Buffer=0;
|
||
|
//vu1::BeginPrim(ABS, VU1_ADDR(L_VF10));
|
||
|
//vu1::StoreVec(*(Vec *)&render::ViewportScale); // VF10
|
||
|
//vu1::StoreVec(*(Vec *)&render::ViewportOffset); // VF11
|
||
|
//vu1::StoreMat(*(Mat *)&render::AdjustedWorldToFrustum); // VF12-15
|
||
|
//vu1::StoreMat(*(Mat *)&render::AdjustedWorldToViewport); // VF16-19
|
||
|
//vu1::EndPrim(0);
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::XYOFFSET_1, PackXYOFFSET(XOFFSET, YOFFSET));
|
||
|
gs::Reg1(gs::SCISSOR_1, PackSCISSOR(0,HRES - 1,0,VRES - 1));
|
||
|
gs::Reg1(gs::TEST_1, PackTEST(0,0,0,0,0,0,1,ZGEQUAL));
|
||
|
gs::Reg1(gs::CLAMP_1,PackCLAMP(CLAMP,CLAMP,0,0,0,0));
|
||
|
gs::Reg1(gs::PRMODECONT, PackPRMODECONT(1));
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
}
|
||
|
|
||
|
void RenderSwitchImmediateMode()
|
||
|
{
|
||
|
dma::SetList(sGroup::pEpilogue);
|
||
|
}
|
||
|
|
||
|
|
||
|
// camera_transform must be a well-formed 4x4 matrix
|
||
|
void RenderViewport(uint viewport_num, const Mth::Matrix& camera_transform, const Mth::Rect& viewport_rec, float view_angle,
|
||
|
float viewport_aspect, float near, float far, bool just_setup)
|
||
|
{
|
||
|
// set up a load of persistent data for current viewport
|
||
|
render::SetupVars(viewport_num, camera_transform, viewport_rec, view_angle, viewport_aspect, near, far);
|
||
|
|
||
|
// Perform the DMA seperately, as the actual VARS might be re-set by update_render_vars called from plat_transform_to_screen
|
||
|
render::SetupVarsDMA(viewport_num, near, far);
|
||
|
|
||
|
// old and new style renders
|
||
|
# ifdef __USE_PROFILER__
|
||
|
Sys::CPUProfiler->PushContext( 255, 0, 0 ); // Red (Under Yellow) = OldStyle Rendering (Skins and stuff)
|
||
|
# endif // __USE_PROFILER__
|
||
|
render::OldStyle();
|
||
|
# ifdef __USE_PROFILER__
|
||
|
Sys::CPUProfiler->PopContext( );
|
||
|
# endif // __USE_PROFILER__
|
||
|
|
||
|
|
||
|
# ifdef __USE_PROFILER__
|
||
|
Sys::CPUProfiler->PushContext( 0, 255, 0 ); // Green (Under Yellow) = NewStyle rendering (Environment)
|
||
|
# endif // __USE_PROFILER__
|
||
|
|
||
|
render::NewStyle(just_setup);
|
||
|
# ifdef __USE_PROFILER__
|
||
|
Sys::CPUProfiler->PopContext( );
|
||
|
# endif // __USE_PROFILER__
|
||
|
}
|
||
|
|
||
|
// Uncomment the next line to use the color-per-mesh feature
|
||
|
#define USE_COLOR_PER_MESH
|
||
|
|
||
|
void render::OldStyle()
|
||
|
{
|
||
|
// Don't render anything if it's not going to be displayed
|
||
|
if (!FlipCopyEnabled())
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
float x,y,z,R;
|
||
|
sMesh *pMesh;
|
||
|
sGroup *p_group;
|
||
|
int j;
|
||
|
uint32 vu1_flags;
|
||
|
uint32 colour;
|
||
|
float colourComponent;
|
||
|
|
||
|
Mth::Matrix LocalToWorld;
|
||
|
Mth::Matrix LocalToCamera;
|
||
|
Mth::Matrix LocalToFrustum;
|
||
|
Mth::Matrix LocalToViewport;
|
||
|
Mth::Matrix RootToWorld;
|
||
|
Mth::Matrix RootToCamera;
|
||
|
Mth::Matrix RootToFrustum;
|
||
|
Mth::Matrix RootToViewport;
|
||
|
Mth::Matrix LightMatrix;
|
||
|
Mth::Matrix ColourMatrix;
|
||
|
|
||
|
Mth::Vector ambientColour;
|
||
|
Mth::Vector diffuseColour0;
|
||
|
Mth::Vector diffuseColour1;
|
||
|
Mth::Vector diffuseColour2;
|
||
|
Mth::Vector colouredAmbientColour;
|
||
|
Mth::Vector colouredDiffuseColour0;
|
||
|
Mth::Vector colouredDiffuseColour1;
|
||
|
Mth::Vector colouredDiffuseColour2;
|
||
|
Mth::Vector v;
|
||
|
|
||
|
// Mick: Initialize these local matricies to known values....
|
||
|
LocalToWorld.Zero();
|
||
|
LocalToCamera.Zero();
|
||
|
LocalToFrustum.Zero();
|
||
|
LocalToViewport.Zero();
|
||
|
RootToWorld.Zero();
|
||
|
RootToCamera.Zero();
|
||
|
RootToFrustum.Zero();
|
||
|
RootToViewport.Zero();
|
||
|
LightMatrix.Zero();
|
||
|
// And the vectors...
|
||
|
ambientColour.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
diffuseColour0.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
diffuseColour1.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
diffuseColour2.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
colouredAmbientColour.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
colouredDiffuseColour0.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
colouredDiffuseColour1.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
colouredDiffuseColour2.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
v.Set(0.0f,0.0f,0.0f,1.0f);
|
||
|
|
||
|
|
||
|
|
||
|
#if STENCIL_SHADOW
|
||
|
// prepare stencil shadow dma list for generating the stencil buffer
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
|
||
|
// send the GS setup
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::XYOFFSET_2, PackXYOFFSET(XOFFSET, YOFFSET));
|
||
|
gs::Reg1(gs::ZBUF_2, PackZBUF(ZBUFFER_START,PSMZ24,1));
|
||
|
gs::Reg1(gs::SCISSOR_2, PackSCISSOR(0,HRES - 1,0,VRES - 1));
|
||
|
gs::Reg1(gs::FRAME_2, PackFRAME(0x1BA,HRES/64,PSMCT16S,0x00000000));
|
||
|
//gs::Reg1(gs::FRAME_2, PackFRAME(FRAME_START,HRES/64,PSMCT32,0x00000000));
|
||
|
gs::Reg1(gs::ALPHA_2, PackALPHA(0,2,2,1,128));
|
||
|
gs::Reg1(gs::FBA_2, PackFBA(0));
|
||
|
gs::Reg1(gs::PABE, PackPABE(0));
|
||
|
gs::Reg1(gs::COLCLAMP, PackCOLCLAMP(0));
|
||
|
gs::Reg1(gs::PRMODECONT, PackPRMODECONT(1));
|
||
|
gs::Reg1(gs::DTHE, PackDTHE(0));
|
||
|
gs::Reg1(gs::TEST_2, PackTEST(0,0,0,0,0,0,1,ZALWAYS));
|
||
|
gs::Reg1(gs::PRIM, PackPRIM(SPRITE,0,0,0,0,0,0,1,0)); // clear the stencil buffer
|
||
|
gs::Reg1(gs::RGBAQ, PackRGBAQ(0,0,0,0,0));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(XOFFSET,YOFFSET,0));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(0x10000-XOFFSET,0x10000-YOFFSET,0));
|
||
|
gs::Reg1(gs::TEST_2, PackTEST(0,0,0,0,0,0,1,ZGREATER));
|
||
|
gs::Reg1(gs::FRAME_2, PackFRAME(0x1BA,HRES/64,PSMCT16S,0xFF000000));
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
#else
|
||
|
// prepare shadow dma list for generating the shadow texture
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
|
||
|
// send the GS setup
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::XYOFFSET_2, PackXYOFFSET(0x7800,0x7800));
|
||
|
gs::Reg1(gs::ZBUF_2, PackZBUF(0,PSMZ16,1));
|
||
|
gs::Reg1(gs::TEST_2, PackTEST(0,0,0,0,0,0,1,ZALWAYS));
|
||
|
gs::Reg1(gs::SCISSOR_2, PackSCISSOR(0,255,0,255));
|
||
|
gs::Reg1(gs::FRAME_2, PackFRAME(0x1F0,4,PSMCT16,0x00000000));
|
||
|
//gs::Reg1(gs::FRAME_2, PackFRAME(0x000,HRES/64,PSMCT32,0x00000000));
|
||
|
gs::Reg1(gs::FBA_2, PackFBA(0));
|
||
|
gs::Reg1(gs::PRIM, PackPRIM(SPRITE,0,0,0,0,0,0,1,0)); // clear the texture
|
||
|
gs::Reg1(gs::RGBAQ, PackRGBAQ(0,0,0,0,0));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(0x7800,0x7800,0));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(0x8800,0x8800,0));
|
||
|
gs::Reg1(gs::FBA_2, PackFBA(1)); // set A on all pixels we touch
|
||
|
gs::Reg1(gs::SCISSOR_2, PackSCISSOR(1,254,1,254)); // don't touch edge of texture to avoid streaks
|
||
|
gs::Reg1(gs::PRMODECONT, PackPRMODECONT(0));
|
||
|
gs::Reg1(gs::PRMODE, PackPRMODE(0,0,0,0,0,0,1,0));
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
#endif
|
||
|
|
||
|
|
||
|
// assume no dma context
|
||
|
dma::SetList(NULL);
|
||
|
|
||
|
// render old-style groups
|
||
|
for (p_group=sGroup::pHead; p_group!=sGroup::pEpilogue; p_group=p_group->pNext)
|
||
|
{
|
||
|
// skip the old render mechanism if the group belongs to a pip-style scene
|
||
|
if (p_group==sGroup::pShadow || !p_group->pScene || p_group->pScene->Flags & SCENEFLAG_USESPIP)
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
dma::SetList(p_group);
|
||
|
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
|
||
|
// VIF1 and VU1 setup
|
||
|
vif::FLUSH();
|
||
|
vif::STMASK(0);
|
||
|
vif::STMOD(0);
|
||
|
vif::STCYCL(1,1);
|
||
|
vif::STCOL(0x3F800000,0,0,0);
|
||
|
vif::BASE(0);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vu1::Loc = vu1::Buffer=0;
|
||
|
|
||
|
// constant part of vu1 context data
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF09));
|
||
|
// vu1::StoreVec(*(Vec *)&render::AltFrustum); // VF09
|
||
|
// vu1::StoreVec(*(Vec *)&render::InverseViewportScale); // VF10
|
||
|
// vu1::StoreVec(*(Vec *)&render::InverseViewportOffset); // VF11
|
||
|
vu1::CopyQuads((uint32*)&render::AltFrustum,
|
||
|
(uint32*)&render::InverseViewportScale,
|
||
|
(uint32*)&render::InverseViewportOffset
|
||
|
);
|
||
|
vu1::EndPrim(0);
|
||
|
|
||
|
// send the GS viewport context
|
||
|
gs::BeginPrim(ABS, 0, 0);
|
||
|
gs::Reg1(gs::XYOFFSET_1, render::reg_XYOFFSET);
|
||
|
gs::Reg1(gs::SCISSOR_1, render::reg_SCISSOR);
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
|
||
|
Mth::Matrix refl_temp;
|
||
|
Mth::Matrix ReflVecs;
|
||
|
refl_temp[0] = Mth::Vector(1.0f, 0.0f, 0.0f, 0.0f);
|
||
|
refl_temp[1] = Mth::Vector(0.0f, 1.0f, 0.0f, 0.0f);
|
||
|
refl_temp[2] = Mth::Vector(0.0f, 0.0f, 1.4f, 0.0f);
|
||
|
refl_temp[3] = Mth::Vector(0.0f, 0.0f, 0.0f, 0.0f);
|
||
|
|
||
|
// this part handles the instanced non-skinned models
|
||
|
if (p_group->pScene && (p_group->pScene->Flags & SCENEFLAG_INSTANCEABLE) &&
|
||
|
p_group->pMeshes && !(p_group->pMeshes->Flags & MESHFLAG_SKINNED))
|
||
|
{
|
||
|
for (CInstance *pInstance=p_group->pScene->pInstances; pInstance; pInstance=pInstance->GetNext())
|
||
|
{
|
||
|
// make sure the instance is active
|
||
|
if (!pInstance->IsActive())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
LocalToWorld = *(Mth::Matrix *)pInstance->GetTransform();
|
||
|
LocalToCamera = LocalToWorld * render::WorldToCamera;
|
||
|
LocalToFrustum = LocalToWorld * render::WorldToFrustum;
|
||
|
LocalToViewport = LocalToFrustum * render::FrustumToViewport;
|
||
|
|
||
|
// set up reflection map transform
|
||
|
sceVu0MulMatrix((sceVu0FVECTOR*)&ReflVecs,
|
||
|
(sceVu0FVECTOR*)&refl_temp,
|
||
|
(sceVu0FVECTOR*)&LocalToCamera);
|
||
|
|
||
|
// send VIF/VU context
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::STROW(0,0,0,0);
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF12));
|
||
|
vu1::StoreMat(*(Mat *)&LocalToViewport); // VF12-15
|
||
|
vu1::StoreMat(*(Mat *)&ReflVecs); // VF16-19
|
||
|
vu1::EndPrim(0);
|
||
|
|
||
|
//gs::BeginPrim(ABS,0,0);
|
||
|
//gs::Reg1(gs::FOGCOL, PackFOGCOL(0x60,0x40,0xC0));
|
||
|
//gs::EndPrim(0);
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
// traverse array of meshes in this group
|
||
|
for (j=0,pMesh=p_group->pMeshes; j<p_group->NumMeshes; j++,pMesh++)
|
||
|
{
|
||
|
// skip if invisible
|
||
|
if (!pMesh->IsActive())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// do nothing if mesh doesn't have a dma subroutine
|
||
|
if (!pMesh->pSubroutine)
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
v = pMesh->Sphere;
|
||
|
v[3] = 1.0f;
|
||
|
v *= LocalToCamera;
|
||
|
x = v[0];
|
||
|
y = v[1];
|
||
|
z = v[2];
|
||
|
R = pMesh->Sphere[3];
|
||
|
|
||
|
// if all outside view volume then cull
|
||
|
if (R<z-render::Near || R<render::Far-z ||
|
||
|
R<render::sy*z+render::cy*y || R<render::sy*z-render::cy*y ||
|
||
|
R<render::sx*z-render::cx*x || R<render::sx*z+render::cx*x)
|
||
|
continue;
|
||
|
|
||
|
// else if all inside outer volume then render normally
|
||
|
else if (R<render::Near-z && R<z-render::Far &&
|
||
|
R<-render::Sy*z-render::Cy*y && R<-render::Sy*z+render::Cy*y &&
|
||
|
R<-render::Sx*z+render::Cx*x && R<-render::Sx*z-render::Cx*x)
|
||
|
{
|
||
|
dma::Gosub3D(pMesh->pSubroutine, PROJ);
|
||
|
}
|
||
|
|
||
|
// otherwise cull at the vertex level
|
||
|
else
|
||
|
{
|
||
|
dma::Gosub3D(pMesh->pSubroutine, CLIP);
|
||
|
}
|
||
|
|
||
|
p_group->Used[render::Field] = true; // Note, should have culling before here
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
// this part handles the instanced skinned models
|
||
|
if (p_group->pScene && (p_group->pScene->Flags & SCENEFLAG_INSTANCEABLE) &&
|
||
|
p_group->pMeshes && (p_group->pMeshes->Flags & MESHFLAG_SKINNED))
|
||
|
{
|
||
|
for (CInstance *pInstance=p_group->pScene->pInstances; pInstance; pInstance=pInstance->GetNext())
|
||
|
{
|
||
|
// make sure the instance is active
|
||
|
if (!pInstance->IsActive())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// shadow
|
||
|
if (pInstance->CastsShadow())
|
||
|
{
|
||
|
#if STENCIL_SHADOW
|
||
|
Mth::Vector shadow_vec(20.0f, -100.0f, 0.0f, 0.0f);
|
||
|
Mth::Vector tweak_vec(0.4f, -2.0f, 0.0f, 0.0f);
|
||
|
CastStencilShadow(pInstance, p_group, shadow_vec, tweak_vec);
|
||
|
#else
|
||
|
CastShadow(pInstance, p_group);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
// mat for bounding sphere test
|
||
|
RootToWorld = *(Mth::Matrix *)pInstance->GetTransform();
|
||
|
RootToCamera = RootToWorld * render::WorldToCamera;
|
||
|
|
||
|
// get bounding sphere
|
||
|
v = pInstance->GetBoundingSphere();
|
||
|
R = v[3];
|
||
|
v[3] = 1.0f;
|
||
|
v *= RootToCamera;
|
||
|
x = v[0];
|
||
|
y = v[1];
|
||
|
z = v[2];
|
||
|
|
||
|
// if all outside view volume then cull
|
||
|
if (R<z-render::Near || R<render::Far-z ||
|
||
|
R<render::sy*z+render::cy*y || R<render::sy*z-render::cy*y ||
|
||
|
R<render::sx*z-render::cx*x || R<render::sx*z+render::cx*x)
|
||
|
continue;
|
||
|
|
||
|
// else if all inside outer volume then render normally
|
||
|
else if (R<render::Near-z && R<z-render::Far &&
|
||
|
R<-render::Sy*z-render::Cy*y && R<-render::Sy*z+render::Cy*y &&
|
||
|
R<-render::Sx*z+render::Cx*x && R<-render::Sx*z-render::Cx*x)
|
||
|
{
|
||
|
vu1_flags = PROJ;
|
||
|
}
|
||
|
|
||
|
// otherwise cull at the vertex level
|
||
|
else
|
||
|
{
|
||
|
vu1_flags = CULL;
|
||
|
}
|
||
|
|
||
|
// wireframe mode
|
||
|
if (pInstance->IsWireframe())
|
||
|
{
|
||
|
vu1_flags |= WIRE;
|
||
|
}
|
||
|
|
||
|
p_group->Used[render::Field] = true; // Note, should have culling before here
|
||
|
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
|
||
|
|
||
|
// adjust matrix for 0:8 fixed point weights format (used to be 1:15)
|
||
|
RootToCamera[0] *= 128.0f;
|
||
|
RootToCamera[1] *= 128.0f;
|
||
|
RootToCamera[2] *= 128.0f;
|
||
|
|
||
|
// send vu1 context
|
||
|
RootToFrustum = RootToCamera * render::CameraToFrustum;
|
||
|
RootToViewport = RootToFrustum * render::FrustumToIntViewport;
|
||
|
|
||
|
CLightGroup *pLightGroup = pInstance->GetLightGroup();
|
||
|
|
||
|
if (pLightGroup)
|
||
|
{
|
||
|
const Mth::Vector &light_dir0 = pLightGroup->GetDirection(0);
|
||
|
const Mth::Vector &light_dir1 = pLightGroup->GetDirection(1);
|
||
|
const Mth::Vector &light_dir2 = pLightGroup->GetDirection(2);
|
||
|
|
||
|
LightMatrix[0][0] = light_dir0[0];
|
||
|
LightMatrix[1][0] = light_dir0[1];
|
||
|
LightMatrix[2][0] = light_dir0[2];
|
||
|
LightMatrix[0][1] = light_dir1[0];
|
||
|
LightMatrix[1][1] = light_dir1[1];
|
||
|
LightMatrix[2][1] = light_dir1[2];
|
||
|
LightMatrix[0][2] = light_dir2[0];
|
||
|
LightMatrix[1][2] = light_dir2[1];
|
||
|
LightMatrix[2][2] = light_dir2[2];
|
||
|
} else {
|
||
|
const Mth::Vector &light_dir0 = CLightGroup::sGetDefaultDirection(0);
|
||
|
const Mth::Vector &light_dir1 = CLightGroup::sGetDefaultDirection(1);
|
||
|
const Mth::Vector &light_dir2 = CLightGroup::sGetDefaultDirection(2);
|
||
|
|
||
|
LightMatrix[0][0] = light_dir0[0];
|
||
|
LightMatrix[1][0] = light_dir0[1];
|
||
|
LightMatrix[2][0] = light_dir0[2];
|
||
|
LightMatrix[0][1] = light_dir1[0];
|
||
|
LightMatrix[1][1] = light_dir1[1];
|
||
|
LightMatrix[2][1] = light_dir1[2];
|
||
|
LightMatrix[0][2] = light_dir2[0];
|
||
|
LightMatrix[1][2] = light_dir2[1];
|
||
|
LightMatrix[2][2] = light_dir2[2];
|
||
|
}
|
||
|
|
||
|
Mth::Matrix temp = *(Mth::Matrix *)pInstance->GetTransform();
|
||
|
temp[0].Normalize(); // do something sensible with a scaled matrix
|
||
|
temp[1].Normalize();
|
||
|
temp[2].Normalize();
|
||
|
LightMatrix = temp * LightMatrix;
|
||
|
|
||
|
if (pLightGroup)
|
||
|
{
|
||
|
ambientColour = pLightGroup->GetAmbientColor();
|
||
|
diffuseColour0 = pLightGroup->GetDiffuseColor(0);
|
||
|
diffuseColour1 = pLightGroup->GetDiffuseColor(1);
|
||
|
diffuseColour2 = pLightGroup->GetDiffuseColor(2);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ambientColour = CLightGroup::sGetDefaultAmbientColor();
|
||
|
diffuseColour0 = CLightGroup::sGetDefaultDiffuseColor(0);
|
||
|
diffuseColour1 = CLightGroup::sGetDefaultDiffuseColor(1);
|
||
|
diffuseColour2 = CLightGroup::sGetDefaultDiffuseColor(2);
|
||
|
}
|
||
|
|
||
|
// apply instance colour
|
||
|
|
||
|
#ifdef USE_COLOR_PER_MESH
|
||
|
if (pInstance->HasColorPerMaterial())
|
||
|
{
|
||
|
// Get first color for now
|
||
|
colour = pInstance->GetMaterialColorByIndex(0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
colour = pInstance->GetColor();
|
||
|
}
|
||
|
#else
|
||
|
colour = pInstance->GetColor();
|
||
|
#endif
|
||
|
colourComponent = (float)(colour & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[0] = ambientColour[0] * colourComponent;
|
||
|
colouredDiffuseColour0[0] = diffuseColour0[0] * colourComponent;
|
||
|
colouredDiffuseColour1[0] = diffuseColour1[0] * colourComponent;
|
||
|
colouredDiffuseColour2[0] = diffuseColour2[0] * colourComponent;
|
||
|
colourComponent = (float)(colour>>8 & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[1] = ambientColour[1] * colourComponent;
|
||
|
colouredDiffuseColour0[1] = diffuseColour0[1] * colourComponent;
|
||
|
colouredDiffuseColour1[1] = diffuseColour1[1] * colourComponent;
|
||
|
colouredDiffuseColour2[1] = diffuseColour2[1] * colourComponent;
|
||
|
colourComponent = (float)(colour>>16 & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[2] = ambientColour[2] * colourComponent;
|
||
|
colouredDiffuseColour0[2] = diffuseColour0[2] * colourComponent;
|
||
|
colouredDiffuseColour1[2] = diffuseColour1[2] * colourComponent;
|
||
|
colouredDiffuseColour2[2] = diffuseColour2[2] * colourComponent;
|
||
|
|
||
|
LightMatrix[3] = colouredAmbientColour * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[0] = colouredDiffuseColour0 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[1] = colouredDiffuseColour1 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[2] = colouredDiffuseColour2 * 1.0f/(128.0f*8388608.0f);
|
||
|
|
||
|
// vu context data
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF10));
|
||
|
|
||
|
// vu1::StoreVec(*(Vec *)&render::InverseIntViewportScale); // VF10
|
||
|
// vu1::StoreVec(*(Vec *)&render::InverseIntViewportOffset); // VF11
|
||
|
vu1::CopyQuads((uint32*)&render::InverseIntViewportScale,
|
||
|
(uint32*)&render::InverseIntViewportOffset);
|
||
|
|
||
|
vu1::StoreMat(*(Mat *)&RootToFrustum); // VF12-15
|
||
|
vu1::StoreMat(*(Mat *)&RootToViewport); // VF16-19
|
||
|
vu1::StoreMat(*(Mat *)&LightMatrix); // VF20-23
|
||
|
vu1::StoreMat(*(Mat *)&ColourMatrix); // VF24-27
|
||
|
vu1::EndPrim(0);
|
||
|
|
||
|
// gs context data, specifically the fog coefficient
|
||
|
float w,f;
|
||
|
w = -RootToCamera[3][2];
|
||
|
if ((w > FogNear) && EnableFog) // Garrett: We have to check for EnableFog here because the VU1 code isn't
|
||
|
{
|
||
|
f = 1.0 + EffectiveFogAlpha * (FogNear/w - 1.0f);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
f = 1.0f;
|
||
|
}
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::FOG, PackFOG((int)(f*255.99f)));
|
||
|
//gs::Reg1(gs::FOGCOL, PackFOGCOL(0x60,0x40,0xC0));
|
||
|
gs::EndPrim(1);
|
||
|
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
|
||
|
// get free run of the VUMem
|
||
|
vif::FLUSHA();
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
// send transforms
|
||
|
vu1::Loc = 0;
|
||
|
dma::Tag(dma::ref, pInstance->GetNumBones()<<2, (uint)pInstance->GetBoneTransforms());
|
||
|
vif::NOP();
|
||
|
vif::UNPACK(0,V4_32,pInstance->GetNumBones()<<2,ABS,SIGNED,0);
|
||
|
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
|
||
|
vif::ITOP(pInstance->GetNumBones()<<2);
|
||
|
vif::MSCAL(VU1_ADDR(ReformatXforms));
|
||
|
|
||
|
// not using pre-translate
|
||
|
// don't forget to switch off the offset mode on xyz unpack if using the row reg for something else
|
||
|
vif::STROW(0,0,0,0);
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
// traverse colour-unlocked meshes
|
||
|
for (j=0,pMesh=p_group->pMeshes; j<p_group->NumMeshes; j++,pMesh++)
|
||
|
{
|
||
|
// skip if invisible, no subroutine, or not affected by dynamic colouring
|
||
|
if (!pMesh->IsActive() || !pMesh->pSubroutine || (pMesh->Flags & MESHFLAG_COLOR_LOCKED))
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
#ifdef USE_COLOR_PER_MESH
|
||
|
uint32 mesh_color;
|
||
|
// Garrett: Thought I had the material color array in the right order, but the render order
|
||
|
// is different. Right now I'm just searching out each color, but I'll need to go back
|
||
|
// and see if it is possible to put the color array in render order (i.e. is the render order
|
||
|
// static).
|
||
|
if (pInstance->HasColorPerMaterial() && (colour != (mesh_color = pInstance->GetMaterialColor(pMesh->MaterialName, pMesh->Pass))))
|
||
|
{
|
||
|
colour = mesh_color;
|
||
|
colourComponent = (float)(colour & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[0] = ambientColour[0] * colourComponent;
|
||
|
colouredDiffuseColour0[0] = diffuseColour0[0] * colourComponent;
|
||
|
colouredDiffuseColour1[0] = diffuseColour1[0] * colourComponent;
|
||
|
colouredDiffuseColour2[0] = diffuseColour2[0] * colourComponent;
|
||
|
colourComponent = (float)(colour>>8 & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[1] = ambientColour[1] * colourComponent;
|
||
|
colouredDiffuseColour0[1] = diffuseColour0[1] * colourComponent;
|
||
|
colouredDiffuseColour1[1] = diffuseColour1[1] * colourComponent;
|
||
|
colouredDiffuseColour2[1] = diffuseColour2[1] * colourComponent;
|
||
|
colourComponent = (float)(colour>>16 & 0xFF) * 0.0078125f;
|
||
|
colouredAmbientColour[2] = ambientColour[2] * colourComponent;
|
||
|
colouredDiffuseColour0[2] = diffuseColour0[2] * colourComponent;
|
||
|
colouredDiffuseColour1[2] = diffuseColour1[2] * colourComponent;
|
||
|
colouredDiffuseColour2[2] = diffuseColour2[2] * colourComponent;
|
||
|
|
||
|
LightMatrix[3] = colouredAmbientColour * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[0] = colouredDiffuseColour0 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[1] = colouredDiffuseColour1 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[2] = colouredDiffuseColour2 * 1.0f/(128.0f*8388608.0f);
|
||
|
|
||
|
// change lighting context
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vu1::Loc = 256;
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF20));
|
||
|
vu1::StoreMat(*(Mat *)&LightMatrix); // VF20-23
|
||
|
vu1::StoreMat(*(Mat *)&ColourMatrix); // VF24-27
|
||
|
vu1::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(ParseInit));
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
}
|
||
|
#endif // USE_COLOR_PER_MESH
|
||
|
|
||
|
// render the mesh
|
||
|
vu1::Loc = vu1::Buffer = 256;
|
||
|
dma::BeginTag(dma::call,(uint)pMesh->pSubroutine);
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vif::ITOP(vu1_flags);
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
vu1::Loc += ((uint16 *)pMesh->pSubroutine)[1];
|
||
|
}
|
||
|
|
||
|
|
||
|
// change lighting context
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vu1::Loc = 256;
|
||
|
LightMatrix[3] = ambientColour * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[0] = diffuseColour0 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[1] = diffuseColour1 * 1.0f/(128.0f*8388608.0f);
|
||
|
ColourMatrix[2] = diffuseColour2 * 1.0f/(128.0f*8388608.0f);
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF20));
|
||
|
vu1::StoreMat(*(Mat *)&LightMatrix); // VF20-23
|
||
|
vu1::StoreMat(*(Mat *)&ColourMatrix); // VF24-27
|
||
|
vu1::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(ParseInit));
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
|
||
|
// traverse colour-locked meshes
|
||
|
for (j=0,pMesh=p_group->pMeshes; j<p_group->NumMeshes; j++,pMesh++)
|
||
|
{
|
||
|
// skip if invisible, no subroutine, or affected by dynamic colouring
|
||
|
if (!pMesh->IsActive() || !pMesh->pSubroutine || !(pMesh->Flags & MESHFLAG_COLOR_LOCKED))
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// render the mesh
|
||
|
vu1::Loc = vu1::Buffer = 256;
|
||
|
dma::BeginTag(dma::call,(uint)pMesh->pSubroutine);
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vif::ITOP(vu1_flags);
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
vu1::Loc += ((uint16 *)pMesh->pSubroutine)[1];
|
||
|
}
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
#if STENCIL_SHADOW
|
||
|
// postprocess stencil buffer
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
|
||
|
// send the GS setup
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
|
||
|
gs::Reg1(gs::TEXFLUSH, 0);
|
||
|
|
||
|
gs::Reg1(gs::COLCLAMP, PackCOLCLAMP(1));
|
||
|
gs::Reg1(gs::TEST_1, PackTEST(0,0,0,0,0,0,1,ZALWAYS));
|
||
|
gs::Reg1(gs::TEX0_1, PackTEX0(0x3740,HRES/64,PSMCT16S,10,10,1,DECAL,0,0,0,0,0));
|
||
|
gs::Reg1(gs::TEX1_1, PackTEX1(1,0,0,0,0,0,0));
|
||
|
gs::Reg1(gs::ALPHA_1, PackALPHA(2,1,0,1,0));
|
||
|
gs::Reg1(gs::TEXA, PackTEXA(ShadowDensity, 1, ShadowDensity));
|
||
|
gs::Reg1(gs::ZBUF_1, PackZBUF(ZBUFFER_START,PSMZ24,1));
|
||
|
|
||
|
gs::Reg1(gs::PRIM, PackPRIM(SPRITE,0,1,0,1,0,1,0,0));
|
||
|
gs::Reg1(gs::UV, PackUV(0x0008,0x0008));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(XOFFSET,YOFFSET,0));
|
||
|
gs::Reg1(gs::UV, PackUV((HRES<<4)+8, (VRES<<4)+8));
|
||
|
gs::Reg1(gs::XYZ2, PackXYZ(XOFFSET+(HRES<<4), YOFFSET+(VRES<<4), 0));//0x10000-XOFFSET,0x10000-YOFFSET,0));
|
||
|
|
||
|
gs::Reg1(gs::TEXA, PackTEXA(0x00,0,0x80));
|
||
|
gs::Reg1(gs::ZBUF_1, PackZBUF(ZBUFFER_START,PSMZ24,0));
|
||
|
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
#endif
|
||
|
|
||
|
|
||
|
|
||
|
// assume no dma context
|
||
|
dma::SetList(NULL);
|
||
|
|
||
|
}
|
||
|
|
||
|
void patch_texture_dma()
|
||
|
{
|
||
|
Nx::CScene *p_nxscene = Nx::CEngine::sGetMainScene();
|
||
|
if (p_nxscene)
|
||
|
{
|
||
|
Nx::CPs2TexDict *p_tex_dict = (Nx::CPs2TexDict *)p_nxscene->GetTexDict();
|
||
|
if (p_tex_dict)
|
||
|
{
|
||
|
sScene *p_scene = p_tex_dict->GetEngineTextureDictionary();
|
||
|
|
||
|
|
||
|
sTexture *p_textures = p_scene->pTextures;
|
||
|
int num_textures = p_scene->NumTextures;
|
||
|
|
||
|
|
||
|
// int total_referenced = 0;
|
||
|
// int num_unique = 0;
|
||
|
int i;
|
||
|
for (i=0;i<num_textures;i++)
|
||
|
{
|
||
|
|
||
|
uint32* p_dma = (uint32*)p_textures[i].mp_dma;
|
||
|
|
||
|
// gif::Tag2(0, 0, IMAGE, 0, 0, 0, NumTexQWords[j]); // 0..3 words, NumTexQWords is in the lower 15 bits of the first word [0]
|
||
|
// dma::EndTag(); // Just Aligns upo to QW boundry
|
||
|
// dma::Tag(dma::ref, NumTexQWords[j], (uint)pTextureSource); // 0..1 (after alignment) NumTexQWords goes in the lower 28 bits of word 0
|
||
|
|
||
|
if (p_textures[i].m_render_count)
|
||
|
{
|
||
|
// Reset it for next time around
|
||
|
p_textures[i].m_render_count = 0;
|
||
|
// fix it, so it gets uploaded
|
||
|
// This test does speed up the routine by about 20%
|
||
|
// Generally the visibility does not change
|
||
|
uint32 current_value = (p_dma[0]);
|
||
|
if (current_value == 0)
|
||
|
{
|
||
|
p_dma[0] = p_textures[i].m_quad_words;;
|
||
|
uint32* p_dma2 = (uint32*)((((int)(p_dma+4)) + 3)&0xfffffff0);
|
||
|
p_dma2[0] = (dma::ref << 28) | p_textures[i].m_quad_words;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// patch it, so nothing gets uploaded
|
||
|
uint32 current_value = (p_dma[0]);
|
||
|
if (current_value != 0)
|
||
|
{
|
||
|
p_dma[0] = 0;
|
||
|
uint32* p_dma2 = (uint32*)((((int)(p_dma+4)) + 3)&0xfffffff0);
|
||
|
p_dma2[0] = (dma::ref << 28);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void render::NewStyle(bool just_setup)
|
||
|
{
|
||
|
|
||
|
sGroup *p_group;
|
||
|
|
||
|
// add viewport-specific GS registers to each new-style group
|
||
|
for (p_group=sGroup::pHead; p_group!=sGroup::pEpilogue; p_group=p_group->pNext)
|
||
|
{
|
||
|
if ((p_group!=sGroup::pShadow) && p_group->pScene && (p_group->pScene->Flags & SCENEFLAG_USESPIP))
|
||
|
{
|
||
|
// send the GS viewport context
|
||
|
dma::SetList(p_group);
|
||
|
|
||
|
// mark our place if it's a (the?) sorted group
|
||
|
// remember to assert there's no overflow
|
||
|
if (p_group->flags & GROUPFLAG_SORT)
|
||
|
{
|
||
|
//printf("Marking location %08X, p_group==%08X, flags=%08X\n", (int)dma::pLoc, (int)p_group, p_group->flags);
|
||
|
Dbg_MsgAssert(sMarkerIndex < MAX_SORTED_LIST_MARKERS, ("too many sorted list markers - tell Mike"));
|
||
|
render::sSortedListMarker[render::sMarkerIndex++] = (int)dma::pLoc;
|
||
|
}
|
||
|
|
||
|
dma::BeginTag(dma::cnt, 0xFE000000);
|
||
|
vif::BASE(vu1::Loc);
|
||
|
vif::OFFSET(0);
|
||
|
uint vu1_loc = vu1::Loc;
|
||
|
vu1::Loc = 0; // must do this as a relative prim for a sortable list...
|
||
|
|
||
|
// constant part of vu1 context data
|
||
|
vu1::BeginPrim(REL, VU1_ADDR(L_VF09));
|
||
|
vu1::StoreVec(*(Vec *)&render::AltFrustum); // VF09
|
||
|
if (p_group->flags & GROUPFLAG_SKY)
|
||
|
{
|
||
|
vu1::StoreVec(*(Vec *)&render::SkyInverseViewportScale); // VF10
|
||
|
vu1::StoreVec(*(Vec *)&render::SkyInverseViewportOffset); // VF11
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
vu1::StoreVec(*(Vec *)&render::InverseViewportScale); // VF10
|
||
|
vu1::StoreVec(*(Vec *)&render::InverseViewportOffset); // VF11
|
||
|
}
|
||
|
vu1::EndPrim(0);
|
||
|
|
||
|
gs::BeginPrim(REL, 0, 0);
|
||
|
gs::Reg1(gs::XYOFFSET_1, render::reg_XYOFFSET);
|
||
|
gs::Reg1(gs::SCISSOR_1, render::reg_SCISSOR);
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
((uint16 *)dma::pTag)[1] |= vu1::Loc & 0x3FF; // must write some code for doing this automatically
|
||
|
vu1::Loc += vu1_loc;
|
||
|
|
||
|
if (p_group->flags & GROUPFLAG_SORT)
|
||
|
{
|
||
|
dma::Tag(dma::cnt,0,0);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
}
|
||
|
|
||
|
// assume no vu1 context for each group
|
||
|
p_group->pVu1Context = NULL;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
// reserve space from dma list for new trans context
|
||
|
dma::SetList(NULL);
|
||
|
//CVu1Context *p_ctxt = new(dma::pLoc) CVu1Context;
|
||
|
CVu1Context *p_ctxt = (CVu1Context *)dma::pLoc;
|
||
|
//dma::pLoc += sizeof(CVu1Context);
|
||
|
dma::pLoc += (STD_CTXT_SIZE+7)*16;
|
||
|
p_ctxt->Init();
|
||
|
Mth::Matrix I;
|
||
|
I.Ident();
|
||
|
p_ctxt->StandardSetup(I);
|
||
|
|
||
|
// shadow stuff
|
||
|
sGroup::pShadow->Used[render::Field] = true;
|
||
|
|
||
|
#if !STENCIL_SHADOW
|
||
|
|
||
|
// while the dma context is null, set up an empty upload for the shadow group
|
||
|
sGroup::pShadow->pUpload[render::Field] = dma::pLoc;
|
||
|
dma::Tag(dma::end, 0, 0);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
|
||
|
// ...and also a vu1 context for the shadow group
|
||
|
CVu1Context *p_shdw_ctxt = new(dma::pLoc) CVu1Context(*p_ctxt);
|
||
|
dma::pLoc += sizeof(CVu1Context);
|
||
|
p_shdw_ctxt->InitExtended();
|
||
|
p_shdw_ctxt->SetShadowVecs(SkaterY);
|
||
|
|
||
|
// add a z-push
|
||
|
p_shdw_ctxt->AddZPush(16.0f);
|
||
|
|
||
|
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
|
||
|
// initial cnt tag, for sending row reg and z-sort key
|
||
|
Mth::Vector *p_trans = p_shdw_ctxt->GetTranslation();
|
||
|
dma::Tag(dma::cnt, 1, 0);
|
||
|
vif::NOP();
|
||
|
vif::STROW((int)(*p_trans)[0], (int)(*p_trans)[1], (int)(*p_trans)[2], 0);
|
||
|
|
||
|
// send transform context
|
||
|
dma::Tag(dma::ref, (EXT_CTXT_SIZE+2)|(EXT_CTXT_SIZE+1)<<16, p_shdw_ctxt->GetDma());
|
||
|
vif::BASE(vu1::Loc);
|
||
|
vif::OFFSET(0);
|
||
|
vu1::Loc += EXT_CTXT_SIZE+1;
|
||
|
sGroup::pShadow->pVu1Context = p_shdw_ctxt;
|
||
|
|
||
|
// send the GS context for the shadow list
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF10));
|
||
|
vu1::StoreVec(*(Vec *)&render::InverseViewportScale); // VF10
|
||
|
vu1::StoreVec(*(Vec *)&render::InverseViewportOffset); // VF11
|
||
|
vu1::EndPrim(0);
|
||
|
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::TEXFLUSH, PackTEXFLUSH(0));
|
||
|
gs::Reg1(gs::FBA_2, PackFBA(0));
|
||
|
gs::Reg1(gs::TEX0_2, PackTEX0(0x3E00,4,PSMCT16,8,8,1,DECAL,0,0,0,0,0));
|
||
|
gs::Reg1(gs::TEX1_2, PackTEX1(1,0,NEAREST,NEAREST,0,0,0));
|
||
|
gs::Reg1(gs::CLAMP_2, PackCLAMP(CLAMP,CLAMP,0,0,0,0));
|
||
|
gs::Reg1(gs::FRAME_2, PackFRAME(FRAME_START,HRES/64,PSMCT32,0xFF000000));
|
||
|
gs::Reg1(gs::ZBUF_2, PackZBUF(ZBUFFER_START,PSMZ24,1));
|
||
|
gs::Reg1(gs::XYOFFSET_2, reg_XYOFFSET);
|
||
|
gs::Reg1(gs::SCISSOR_2, reg_SCISSOR);
|
||
|
gs::Reg1(gs::ALPHA_2, PackALPHA(2,1,2,1,ShadowDensity));
|
||
|
gs::Reg1(gs::TEST_2, PackTEST(1,AGEQUAL,0x80,KEEP,0,0,1,ZGEQUAL));
|
||
|
gs::Reg1(gs::PRMODECONT, PackPRMODECONT(0));
|
||
|
gs::Reg1(gs::PRMODE, PackPRMODE(0,1,0,1,0,0,1,0));
|
||
|
gs::EndPrim(0);
|
||
|
dma::EndTag();
|
||
|
#endif
|
||
|
|
||
|
|
||
|
geom_stats_total=0;
|
||
|
geom_stats_inactive =0;
|
||
|
geom_stats_sky=0;
|
||
|
geom_stats_transformed=0;
|
||
|
geom_stats_skeletal=0;
|
||
|
geom_stats_camera_sphere=0;
|
||
|
geom_stats_clipcull=0;
|
||
|
geom_stats_culled=0;
|
||
|
geom_stats_leaf_culled=0;
|
||
|
geom_stats_clipcull=0;
|
||
|
geom_stats_boxcheck=0;
|
||
|
geom_stats_occludecheck=0;
|
||
|
geom_stats_occluded=0;
|
||
|
geom_stats_colored=0;
|
||
|
geom_stats_leaf=0;
|
||
|
geom_stats_wibbleUV=0;
|
||
|
geom_stats_wibbleVC=0;
|
||
|
geom_stats_sendcontext=0;
|
||
|
geom_stats_sorted=0;
|
||
|
geom_stats_shadow=0;
|
||
|
|
||
|
// render the world
|
||
|
if (!just_setup)
|
||
|
{
|
||
|
|
||
|
#ifdef __NOPT_ASSERT__
|
||
|
const int span = 60;
|
||
|
static int avg[span];
|
||
|
static int slot = 0;
|
||
|
Tmr::CPUCycles start = Tmr::GetTimeInCPUCycles();
|
||
|
#endif
|
||
|
|
||
|
#if STENCIL_SHADOW
|
||
|
CGeomNode::sWorld.RenderWorld(*p_ctxt, RENDERFLAG_CLIP);
|
||
|
#else
|
||
|
CGeomNode::sWorld.RenderWorld(*p_ctxt, RENDERFLAG_CLIP | RENDERFLAG_SHADOW);
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#ifdef __NOPT_ASSERT__
|
||
|
Tmr::CPUCycles len = Tmr::GetTimeInCPUCycles() - start;
|
||
|
int ticks = len;
|
||
|
avg[slot++] = ticks;
|
||
|
if (slot == span) slot = 0;
|
||
|
int tot = 0;
|
||
|
for (int a=0;a<span;a++)
|
||
|
{
|
||
|
tot += avg[a];
|
||
|
}
|
||
|
tot/=span;
|
||
|
// printf ("%9d %9d\n",tot,ticks); // printf average, and the last frame
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
printf ("\ngeom_stats_total = %d\n", geom_stats_total);
|
||
|
printf ("geom_stats_inactive = %d\n", geom_stats_inactive );
|
||
|
printf ("geom_stats_sky = %d\n", geom_stats_sky);
|
||
|
printf ("geom_stats_transformed = %d\n", geom_stats_transformed);
|
||
|
printf ("geom_stats_skeletal = %d\n", geom_stats_skeletal);
|
||
|
printf ("geom_stats_camera_sphere = %d\n", geom_stats_camera_sphere);
|
||
|
printf ("geom_stats_clipcull = %d\n", geom_stats_clipcull);
|
||
|
printf ("geom_stats_boxcheck = %d\n", geom_stats_boxcheck);
|
||
|
printf ("geom_stats_occludecheck = %d\n", geom_stats_occludecheck);
|
||
|
printf ("geom_stats_occluded = %d\n", geom_stats_occluded);
|
||
|
printf ("geom_stats_colored = %d\n", geom_stats_colored);
|
||
|
printf ("geom_stats_leaf = %d\n", geom_stats_leaf);
|
||
|
printf ("geom_stats_wibbleUV = %d\n", geom_stats_wibbleUV);
|
||
|
printf ("geom_stats_wibbleVC = %d\n", geom_stats_wibbleVC);
|
||
|
printf ("geom_stats_sendcontext = %d\n", geom_stats_sendcontext);
|
||
|
printf ("geom_stats_sorted = %d\n", geom_stats_sorted);
|
||
|
printf ("geom_stats_shadow = %d\n", geom_stats_shadow);
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#if !STENCIL_SHADOW
|
||
|
// undo the havoc caused by the shadow group
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
gs::BeginPrim(ABS,0,0);
|
||
|
gs::Reg1(gs::PRMODECONT, PackPRMODECONT(1));
|
||
|
gs::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
dma::EndTag();
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#if !STENCIL_SHADOW
|
||
|
void SetTextureProjectionCamera( Mth::Vector *p_pos, Mth::Vector *p_at )
|
||
|
{
|
||
|
// sTextureProjectionDetails *p_details = pTextureProjectionDetailsTable->GetItem((uint32)p_texture );
|
||
|
// if( p_details )
|
||
|
{
|
||
|
render::ShadowCameraPosition[0] = p_pos->GetX();
|
||
|
render::ShadowCameraPosition[1] = p_pos->GetY();
|
||
|
render::ShadowCameraPosition[2] = p_pos->GetZ();
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
// Added by Mick
|
||
|
// quick determination of if something is visible or not
|
||
|
// uses the previously calculated s and c vectors
|
||
|
// and the WorldToCamera transform
|
||
|
// (note, no attempt is made to ensure this is the same camera that the object
|
||
|
// will eventually be rendered with)
|
||
|
bool IsVisible(Mth::Vector ¢er, float radius)
|
||
|
{
|
||
|
|
||
|
|
||
|
float x,y,z;
|
||
|
Mth::Vector v(center);
|
||
|
|
||
|
v[3] = 1.0f; // needed?
|
||
|
v *= render::WorldToCamera;
|
||
|
x = v[0];
|
||
|
y = v[1];
|
||
|
z = v[2];
|
||
|
|
||
|
// if all outside view volume then cull
|
||
|
if (radius<z-render::Near || radius<render::Far-z ||
|
||
|
radius<render::sy*z+render::cy*y || radius<render::sy*z-render::cy*y ||
|
||
|
radius<render::sx*z-render::cx*x || radius<render::sx*z+render::cx*x)
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// return true;
|
||
|
|
||
|
if( TestSphereAgainstOccluders( ¢er, radius ))
|
||
|
{
|
||
|
// Occluded.
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
// The same function, but without occlusion
|
||
|
bool IsInFrame(Mth::Vector ¢er, float radius)
|
||
|
{
|
||
|
|
||
|
float x,y,z;
|
||
|
Mth::Vector v(center);
|
||
|
|
||
|
v[3] = 1.0f; // needed?
|
||
|
v *= render::WorldToCamera;
|
||
|
x = v[0];
|
||
|
y = v[1];
|
||
|
z = v[2];
|
||
|
|
||
|
// if all outside view volume then cull
|
||
|
if (radius<z-render::Near || radius<render::Far-z ||
|
||
|
radius<render::sy*z+render::cy*y || radius<render::sy*z-render::cy*y ||
|
||
|
radius<render::sx*z-render::cx*x || radius<render::sx*z+render::cx*x)
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
void RenderMesh(sMesh *pMesh, uint RenderFlags)
|
||
|
{
|
||
|
dma::BeginTag(dma::call,(uint)pMesh->pSubroutine);
|
||
|
vif::BASE(vu1::Loc);
|
||
|
vif::OFFSET(0);
|
||
|
vif::ITOP(RenderFlags);
|
||
|
dma::EndTag();
|
||
|
vu1::Loc += ((uint16 *)(pMesh->pSubroutine))[1];
|
||
|
}
|
||
|
|
||
|
|
||
|
void RenderEpilogue(void)
|
||
|
{
|
||
|
sGroup *p_group;
|
||
|
|
||
|
// ---------------------------------------------
|
||
|
// tag onto end of particle group
|
||
|
dma::SetList(sGroup::pParticles);
|
||
|
|
||
|
// restore vu1 code (and buffering scheme) at end of particles group
|
||
|
dma::Tag(dma::ref, ((uint8 *)&MPGEnd-(uint8 *)&MPGStart+15)/16, (uint)&MPGStart);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::STMASK(0);
|
||
|
vif::STMOD(0);
|
||
|
vif::STCYCL(1,1);
|
||
|
vif::BASE(0);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vu1::Loc = vu1::Buffer = 0;
|
||
|
dma::EndTag();
|
||
|
|
||
|
dma::SetList(NULL);
|
||
|
// ---------------------------------------------
|
||
|
|
||
|
// add a GS SIGNAL primitive to each group
|
||
|
for (p_group=sGroup::pHead; p_group; p_group=p_group->pNext)
|
||
|
{
|
||
|
// end the dma list
|
||
|
dma::EndList(p_group);
|
||
|
}
|
||
|
|
||
|
dma::SetList(NULL);
|
||
|
|
||
|
// epilogue VRAM upload (fonts and sprites)
|
||
|
|
||
|
// Upload the fonts, so we can draw 2D text on screen
|
||
|
sGroup::pEpilogue->pUpload[render::Field] = dma::pLoc;
|
||
|
uint32 *p = (uint32*)(dma::pLoc);
|
||
|
for (SFont *pFont=pFontList; pFont; pFont=pFont->pNext)
|
||
|
{
|
||
|
// Mick: we build this upload list directly, to cut down on the number of writes
|
||
|
*p++ = (dma::ref << 28) + ((pFont->VifSize+15)>>4); // dma::Tag(dma::ref, (pFont->VifSize+15)>>4, (uint)pFont->pVifData);
|
||
|
*p++ = (uint32) pFont->pVifData;
|
||
|
*p++ = 0; // vif::NOP();
|
||
|
*p++ = 0; // vif::NOP();
|
||
|
}
|
||
|
|
||
|
// Upload the "Single Textures", the 2D screen elements (sprites)
|
||
|
for (SSingleTexture *pTexture=SSingleTexture::sp_stexture_list; pTexture; pTexture=pTexture->mp_next)
|
||
|
{
|
||
|
if (pTexture->IsInVRAM())
|
||
|
{
|
||
|
*p++ = (dma::call << 28) + (0); // dma::Tag(dma::call, 0, (uint)pTexture->mp_VifData);
|
||
|
*p++ = (uint32) pTexture->mp_VifData;
|
||
|
*p++ = 0; // vif::NOP();
|
||
|
*p++ = 0; // vif::NOP();
|
||
|
}
|
||
|
}
|
||
|
dma::pLoc = (uint8*)p;
|
||
|
|
||
|
|
||
|
// (?) writing to TEXFLUSH will stall the gs until all the fonts and sprites
|
||
|
// have been uploaded.
|
||
|
// since we are not double buffering anything now, we have to wait until
|
||
|
// everything is in VRAM before we can start rendering any 2D
|
||
|
// this is a potential GS hiccup, but removing it would be tricky
|
||
|
// and possibly not worth the effort
|
||
|
// Garrett: Now the 2D uses both buffers at once.
|
||
|
|
||
|
dma::BeginTag(dma::end, 0);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
gif::Tag2(gs::A_D,1,PACKED,0,0,1,1);
|
||
|
gs::Reg2(gs::TEXFLUSH, 0);
|
||
|
dma::EndTag();
|
||
|
|
||
|
|
||
|
// We need the scratchpad in SortGroup()
|
||
|
bool got_scratch = CSystemResources::sRequestResource(CSystemResources::vSCRATCHPAD_MEMORY);
|
||
|
Dbg_Assert(got_scratch);
|
||
|
|
||
|
// sort the dma lists of transparent groups
|
||
|
for (p_group=sGroup::pHead; p_group!=sGroup::pEpilogue; p_group=p_group->pNext)
|
||
|
{
|
||
|
if ((p_group->flags & GROUPFLAG_SORT) // Mick, most groups are not SORT, so this test first is the fastest
|
||
|
&& p_group!=sGroup::pShadow
|
||
|
&& p_group->pScene
|
||
|
&& (p_group->pScene->Flags & SCENEFLAG_USESPIP)
|
||
|
)
|
||
|
{
|
||
|
p_group->pRender[render::Field] = dma::SortGroup(p_group->pRender[render::Field]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (got_scratch)
|
||
|
{
|
||
|
CSystemResources::sFreeResource(CSystemResources::vSCRATCHPAD_MEMORY);
|
||
|
}
|
||
|
|
||
|
// swap epilogue and particle texture upload pointers (a cheat!)
|
||
|
uint8 *temp_ptr = sGroup::pParticles->pUpload[render::Field];
|
||
|
sGroup::pParticles->pUpload[render::Field] = sGroup::pEpilogue->pUpload[render::Field];
|
||
|
sGroup::pEpilogue->pUpload[render::Field] = temp_ptr;
|
||
|
|
||
|
int size = (int)dma::pLoc - (int)dma::pBase;
|
||
|
static int max = 0;
|
||
|
static int max2 = 0;
|
||
|
if (size > max) max = size;
|
||
|
if (size > max2) max2 = size;
|
||
|
static int tick = 0;
|
||
|
tick++;
|
||
|
tick &= 0x1f;
|
||
|
if (!tick)
|
||
|
{
|
||
|
// printf ("%6d, %6d, %6d (of %6d)\n",size,max2,max, dma::size);
|
||
|
max2 -= 50000;
|
||
|
}
|
||
|
|
||
|
Dbg_MsgAssert(DMAOverflowOK || size < (dma::size*98/100), ("DMA Buffer Overflow used %d\nShow screenshot to Mick (see .bmp saved, above)",size));
|
||
|
|
||
|
if (DMAOverflowOK)
|
||
|
{
|
||
|
DMAOverflowOK--;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
void SendDisplayList(void)
|
||
|
{
|
||
|
|
||
|
// Mick: Mike added this just before he left, but did not check it in
|
||
|
// sceGsSyncPath(0, 0);
|
||
|
|
||
|
// patch up the DMA list
|
||
|
// Moved to StuffAfterGSFinished
|
||
|
// patch_texture_dma();
|
||
|
|
||
|
// Writeback data from D-Cache to Memory
|
||
|
FlushCache(WRITEBACK_DCACHE);
|
||
|
|
||
|
# ifdef __USE_PROFILER__
|
||
|
// Initial render group does not cause an interrupt
|
||
|
// so don't push context
|
||
|
// Sys::VUProfiler->PushContext( 80,80, 228 );
|
||
|
# endif // __USE_PROFILER__
|
||
|
// kick prologue render
|
||
|
*D1_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D1_TADR = (uint)PrologueGroup.pRender[!render::Field]; // address of 1st tag
|
||
|
*D1_CHCR = 0x145; // start transfer, tte=1, chain mode, from memory
|
||
|
sceGsSyncPath(0, 0);
|
||
|
|
||
|
#if USE_INTERRUPTS
|
||
|
|
||
|
// use interrupts
|
||
|
sGroup::pUploadGroup = sGroup::pHead;
|
||
|
sGroup::pRenderGroup = sGroup::pUploadGroup;
|
||
|
|
||
|
|
||
|
// SingleRender = 0;
|
||
|
|
||
|
#if 0 // debugging code to force all gorups to be executed, even if empty
|
||
|
sGroup * p_group = (sGroup*)sGroup::pUploadGroup; // Skip prologue and sky
|
||
|
while (p_group->pNext) // all except the last one
|
||
|
{
|
||
|
p_group->Used[!render::Field] = 1;
|
||
|
p_group = p_group->pNext;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
|
||
|
// Some debugging code to only render/upload a single selectable group
|
||
|
// so we can see what is taking all the time
|
||
|
if (SingleRender)
|
||
|
{
|
||
|
sGroup * p_group = (sGroup*)sGroup::pUploadGroup; // Skip prologue and sky
|
||
|
int g = 0;
|
||
|
while (p_group->pNext) // all except the last one
|
||
|
{
|
||
|
g++;
|
||
|
if (g > 2 && g != SingleRender)
|
||
|
{
|
||
|
p_group->Used[!render::Field] = 0;
|
||
|
}
|
||
|
p_group = p_group->pNext;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
UploadStalled = 0;
|
||
|
RenderStalled = 1;
|
||
|
FlushCache(WRITEBACK_DCACHE);
|
||
|
|
||
|
# ifdef __USE_PROFILER__
|
||
|
Sys::DMAProfiler->PushContext( sGroup::pUploadGroup->profile_color );
|
||
|
# endif // __USE_PROFILER__
|
||
|
|
||
|
|
||
|
// kick off first upload, should trigger all the rest by interrupt
|
||
|
*D2_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D2_TADR = (uint)sGroup::pUploadGroup->pUpload[!render::Field]; // address of 1st tag
|
||
|
*D2_CHCR = 0x105; // start transfer, tte=0, chain mode, from memory
|
||
|
|
||
|
|
||
|
|
||
|
#else
|
||
|
|
||
|
// iterate over groups, uploading textures then rendering meshes in each
|
||
|
// last group will be epilogue group
|
||
|
for (sGroup *p_group=sGroup::pHead; p_group!=sGroup::pEpilogue; p_group=p_group->pNext)
|
||
|
{
|
||
|
if (p_group->Used[!render::Field])
|
||
|
{
|
||
|
// kick dma to upload textures from this group
|
||
|
*D2_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D2_TADR = (uint)p_group->pUpload[!render::Field]; // address of 1st tag
|
||
|
*D2_CHCR = 0x105; // start transfer, tte=0, chain mode, from memory
|
||
|
sceGsSyncPath(0, 0);
|
||
|
|
||
|
// kick dma to render scene
|
||
|
*D1_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D1_TADR = (uint)p_group->pRender[!render::Field]; // address of 1st tag
|
||
|
*D1_CHCR = 0x145; // start transfer, tte=1, chain mode, from memory
|
||
|
sceGsSyncPath(0, 0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// kick dma to upload textures from this group
|
||
|
*D2_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D2_TADR = (uint)sGroup::pEpilogue->pUpload[!render::Field]; // address of 1st tag
|
||
|
*D2_CHCR = 0x105; // start transfer, tte=0, chain mode, from memory
|
||
|
sceGsSyncPath(0, 0);
|
||
|
|
||
|
// kick dma to render scene
|
||
|
*D1_QWC = 0; // must zero QWC because the first action will be to use current MADR & QWC
|
||
|
*D1_TADR = (uint)sGroup::pEpilogue->pRender[!render::Field]; // address of 1st tag
|
||
|
*D1_CHCR = 0x145; // start transfer, tte=1, chain mode, from memory
|
||
|
sceGsSyncPath(0, 0);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
static bool sUpdate2DDma = false;
|
||
|
static bool sVRAMNeedsUpdating = false;
|
||
|
|
||
|
|
||
|
void Clear2DVRAM()
|
||
|
{
|
||
|
// Reset VRAM pointer
|
||
|
FontVramBase = FontVramStart;
|
||
|
|
||
|
// And set flag
|
||
|
sUpdate2DDma = true;
|
||
|
}
|
||
|
|
||
|
void VRAMNeedsUpdating()
|
||
|
{
|
||
|
sVRAMNeedsUpdating = true;
|
||
|
}
|
||
|
|
||
|
void Reallocate2DVRAM()
|
||
|
{
|
||
|
if (!sVRAMNeedsUpdating)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Reset VRAM pointer
|
||
|
Clear2DVRAM();
|
||
|
|
||
|
for (SFont *pFont=pFontList; pFont; pFont=pFont->pNext)
|
||
|
{
|
||
|
pFont->ReallocateVRAM();
|
||
|
}
|
||
|
|
||
|
for (SSingleTexture *pTexture=SSingleTexture::sp_stexture_list; pTexture; pTexture=pTexture->mp_next)
|
||
|
{
|
||
|
pTexture->ReallocateVRAM();
|
||
|
}
|
||
|
|
||
|
// And set flags
|
||
|
sUpdate2DDma = true;
|
||
|
sVRAMNeedsUpdating = false;
|
||
|
}
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
|
||
|
void Update2DDMA()
|
||
|
{
|
||
|
// Only go through the lists if it was requested
|
||
|
if (!sUpdate2DDma)
|
||
|
return;
|
||
|
|
||
|
// Go through each item
|
||
|
for (SFont *pFont=pFontList; pFont; pFont=pFont->pNext)
|
||
|
{
|
||
|
pFont->UpdateDMA();
|
||
|
}
|
||
|
|
||
|
for (SSingleTexture *pTexture=SSingleTexture::sp_stexture_list; pTexture; pTexture=pTexture->mp_next)
|
||
|
{
|
||
|
pTexture->UpdateDMA();
|
||
|
}
|
||
|
|
||
|
// Clear flag
|
||
|
sUpdate2DDma = false;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#if STENCIL_SHADOW
|
||
|
|
||
|
void CastStencilShadow(CInstance *pInstance, sGroup *p_group, Mth::Vector &ShadowVec, Mth::Vector &TweakVec)
|
||
|
{
|
||
|
sShadowVolumeHeader * p_shadow = pInstance->GetScene()->mp_shadow_volume_header;
|
||
|
if (!p_shadow)
|
||
|
return;
|
||
|
|
||
|
Mth::Matrix RootToFrustum,RootToViewport, WorldToRoot;
|
||
|
|
||
|
RootToFrustum = *(Mth::Matrix *)pInstance->GetTransform() * render::WorldToFrustum;
|
||
|
RootToViewport = RootToFrustum * render::FrustumToIntViewport;
|
||
|
|
||
|
WorldToRoot = *(Mth::Matrix *)pInstance->GetTransform();
|
||
|
WorldToRoot.Transpose();
|
||
|
ShadowVec *= WorldToRoot;
|
||
|
TweakVec *= WorldToRoot;
|
||
|
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF14));
|
||
|
vu1::StoreVec(*(Vec *)&ShadowVec); // VF14, shadow vec
|
||
|
vu1::StoreVec(*(Vec *)&TweakVec); // VF15, tweak vec
|
||
|
vu1::StoreMat(*(Mat *)&RootToViewport); // VF16-19
|
||
|
vu1::EndPrim(1);
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
|
||
|
// send transforms
|
||
|
vu1::Loc = 0;
|
||
|
dma::Tag(dma::ref, pInstance->GetNumBones()<<2, (uint)pInstance->GetBoneTransforms());
|
||
|
vif::STCYCL(1,1);
|
||
|
vif::UNPACK(0,V4_32,pInstance->GetNumBones()<<2,ABS,SIGNED,0);
|
||
|
|
||
|
// send data
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::STCYCL(7,21);
|
||
|
|
||
|
sShadowVertex * p_vertex = p_shadow->p_vertex;
|
||
|
sShadowConnect * p_connect = p_shadow->p_connect;
|
||
|
sShadowNeighbor * p_neighbor = p_shadow->p_neighbor;
|
||
|
|
||
|
int vert, num_faces;
|
||
|
for (uint32 i=0; i<p_shadow->num_faces; i++)
|
||
|
{
|
||
|
if (i%36==0)
|
||
|
{
|
||
|
vu1::Loc = 256;
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
}
|
||
|
|
||
|
if (i%12==0)
|
||
|
{
|
||
|
num_faces = p_shadow->num_faces-i;
|
||
|
if (num_faces>12)
|
||
|
num_faces=12;
|
||
|
|
||
|
|
||
|
vif::UNPACK(0, V4_32, 7*num_faces, ABS, UNSIGNED, 0);
|
||
|
}
|
||
|
|
||
|
if ((i%12==11) || (i==p_shadow->num_faces-1))
|
||
|
{
|
||
|
vif::StoreV4_32(0x3480800A, 0x21224000, 0x00000051, 0x8000);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
vif::StoreV4_32(0x3480800A, 0x21224000, 0x00000051, 0x0000);
|
||
|
}
|
||
|
|
||
|
uint32 * p32 = (uint32 *)dma::pLoc;
|
||
|
|
||
|
vert = p_connect[i].corner[0];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
vert = p_connect[i].corner[1];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
vert = p_connect[i].corner[2];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
vert = p_neighbor[i].edge[0];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
vert = p_neighbor[i].edge[1];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
vert = p_neighbor[i].edge[2];
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].x));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].y));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].z));
|
||
|
*p32++ = *((uint32 *)(&p_vertex[vert].idx));
|
||
|
|
||
|
dma::pLoc = (uint8 *)p32;
|
||
|
|
||
|
vu1::Loc += 21;
|
||
|
|
||
|
if ((i%12 == 11) || (i==p_shadow->num_faces-1))
|
||
|
{
|
||
|
vif::MSCAL(VU1_ADDR(ShadowVolumeSkin));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
dma::SetList(p_group);
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
|
||
|
void CastShadow(CInstance *pInstance, sGroup *p_group)
|
||
|
{
|
||
|
int j;
|
||
|
sMesh *pMesh;
|
||
|
Mth::Matrix ProjMat, temp;
|
||
|
|
||
|
// set up projection matrix
|
||
|
ProjMat = *pInstance->GetTransform();
|
||
|
SkaterY = ProjMat[3][1];
|
||
|
ProjMat[3].Set(-36.0f/**SUB_INCH_PRECISION*/*ProjMat[1][0],
|
||
|
-36.0f/**SUB_INCH_PRECISION*/*ProjMat[1][1],
|
||
|
-36.0f/**SUB_INCH_PRECISION*/*ProjMat[1][2], 1.0f);
|
||
|
temp[0].Set(32.0f/**RECIPROCAL_SUB_INCH_PRECISION*/, 0.0f, 0.0f, 0.0f);
|
||
|
temp[1].Set(0.0f, 0.0f, 0.0f, 0.0f);
|
||
|
temp[2].Set(0.0f, 32.0f/**RECIPROCAL_SUB_INCH_PRECISION*/, 0.0f, 0.0f);
|
||
|
temp[3].Set(8421376.0f, 8421376.0f, 0.0f, 1.0f);
|
||
|
ProjMat *= temp;
|
||
|
|
||
|
dma::SetList(sGroup::pShadow);
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vu1::BeginPrim(ABS, VU1_ADDR(L_VF12));
|
||
|
vu1::StoreMat(*(Mat *)&ProjMat); // VF16-19
|
||
|
vu1::EndPrim(1);
|
||
|
|
||
|
vif::MSCAL(VU1_ADDR(Parser));
|
||
|
|
||
|
// not using pre-translate
|
||
|
// don't forget to switch off the offset mode on xyz unpack if using the row reg for somthing else
|
||
|
vif::STROW(0,0,0,0);
|
||
|
|
||
|
// get free run of the VUMem
|
||
|
vif::FLUSH();
|
||
|
|
||
|
dma::EndTag();
|
||
|
|
||
|
// send transforms
|
||
|
vu1::Loc = 0;
|
||
|
dma::Tag(dma::ref, pInstance->GetNumBones()<<2, (uint)pInstance->GetBoneTransforms());
|
||
|
vif::NOP();
|
||
|
vif::UNPACK(0,V4_32,pInstance->GetNumBones()<<2,ABS,SIGNED,0);
|
||
|
|
||
|
dma::Tag(dma::cnt, 0, 0);
|
||
|
vif::ITOP(pInstance->GetNumBones()<<2);
|
||
|
vif::MSCAL(VU1_ADDR(ReformatXforms));
|
||
|
|
||
|
// traverse array of meshes in the model
|
||
|
for (j=0,pMesh=p_group->pMeshes; j<p_group->NumMeshes; j++,pMesh++)
|
||
|
{
|
||
|
// skip if invisible
|
||
|
if (!pMesh->IsActive())
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// do nothing if mesh doesn't have a dma subroutine
|
||
|
if (!pMesh->pSubroutine)
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// render the mesh
|
||
|
vu1::Loc = vu1::Buffer = 256;
|
||
|
dma::BeginTag(dma::call,(uint)pMesh->pSubroutine);
|
||
|
vif::BASE(256);
|
||
|
vif::OFFSET(0);
|
||
|
vif::MSCAL(VU1_ADDR(Setup));
|
||
|
vif::ITOP(SHDW);
|
||
|
vif::FLUSH();
|
||
|
dma::EndTag();
|
||
|
vu1::Loc += ((uint16 *)pMesh->pSubroutine)[1];
|
||
|
}
|
||
|
|
||
|
dma::Tag(dma::cnt, 0, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::NOP();
|
||
|
|
||
|
dma::SetList(p_group);
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
namespace render
|
||
|
{
|
||
|
Mth::Matrix WorldToCamera;
|
||
|
Mth::Matrix WorldToCameraRotation; // shouldn't need this
|
||
|
Mth::Matrix WorldToFrustum;
|
||
|
Mth::Matrix WorldToViewport;
|
||
|
Mth::Matrix WorldToIntViewport;
|
||
|
Mth::Matrix CameraToWorld;
|
||
|
Mth::Matrix CameraToWorldRotation;
|
||
|
Mth::Matrix CameraToFrustum;
|
||
|
Mth::Matrix CameraToViewport;
|
||
|
Mth::Matrix FrustumToViewport;
|
||
|
Mth::Matrix FrustumToIntViewport;
|
||
|
Mth::Matrix AdjustedWorldToCamera;
|
||
|
Mth::Matrix AdjustedWorldToFrustum;
|
||
|
Mth::Matrix AdjustedWorldToViewport;
|
||
|
Mth::Matrix AdjustedWorldToIntViewport;
|
||
|
Mth::Matrix SkyWorldToCamera;
|
||
|
Mth::Matrix SkyWorldToFrustum;
|
||
|
Mth::Matrix SkyWorldToViewport;
|
||
|
Mth::Matrix SkyFrustumToViewport;
|
||
|
|
||
|
Mth::Vector ViewportScale;
|
||
|
Mth::Vector ViewportOffset;
|
||
|
Mth::Vector IntViewportScale;
|
||
|
Mth::Vector IntViewportOffset;
|
||
|
Mth::Vector InverseViewportScale;
|
||
|
Mth::Vector InverseViewportOffset;
|
||
|
Mth::Vector InverseIntViewportScale;
|
||
|
Mth::Vector InverseIntViewportOffset;
|
||
|
Mth::Vector SkyViewportScale;
|
||
|
Mth::Vector SkyViewportOffset;
|
||
|
Mth::Vector SkyInverseViewportScale;
|
||
|
Mth::Vector SkyInverseViewportOffset;
|
||
|
Mth::Vector RowReg;
|
||
|
Mth::Vector RowRegI;
|
||
|
Mth::Vector RowRegF;
|
||
|
#if !STENCIL_SHADOW
|
||
|
Mth::Vector ShadowCameraPosition;
|
||
|
#endif
|
||
|
Mth::Vector AltFrustum;
|
||
|
Mth::Vector CameraPosition;
|
||
|
|
||
|
uint64 reg_XYOFFSET;
|
||
|
uint64 reg_SCISSOR;
|
||
|
|
||
|
float sx,sy,cx,cy,tx,ty,Sx,Sy,Cx,Cy,Tx,Ty,Near,Far;
|
||
|
//float sMultipassMaxDist = 1000000000.0f;
|
||
|
float sMultipassMaxDist = 4000.0f;
|
||
|
|
||
|
uint Frame, Field;
|
||
|
uint ViewportMask;
|
||
|
|
||
|
uint RenderVarFrame = 0xFFFFFFFF;
|
||
|
uint RenderVarViewport = 0xFFFFFFFF;
|
||
|
|
||
|
bool FrustumFlagsPlus[3], FrustumFlagsMinus[3];
|
||
|
|
||
|
int sSortedListMarker[MAX_SORTED_LIST_MARKERS];
|
||
|
int sMarkerIndex;
|
||
|
int sTime;
|
||
|
int sTotalNewParticles;
|
||
|
|
||
|
float FogNear=3000.0f, FogAlpha=0.80f;
|
||
|
uint32 FogCol=0xFF4070;
|
||
|
bool EnableFog=false;
|
||
|
float EffectiveFogAlpha;
|
||
|
|
||
|
uint32 *p_patch_FOGCOL;
|
||
|
|
||
|
uint8 ShadowDensity = 48;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void render::SetupVars(uint viewport_num, const Mth::Matrix& camera_transform, const Mth::Rect& viewport_rec, float view_angle,
|
||
|
float viewport_aspect, float near, float far)
|
||
|
{
|
||
|
|
||
|
// fog variables
|
||
|
EffectiveFogAlpha = FogAlpha;
|
||
|
if (FogAlpha < 0.003f)
|
||
|
{
|
||
|
EffectiveFogAlpha = 0.003f;
|
||
|
}
|
||
|
if (FogAlpha > 0.997f)
|
||
|
{
|
||
|
EffectiveFogAlpha = 0.997f;
|
||
|
}
|
||
|
if (!EnableFog) // TT5158: ZPush is affected by FogAlpha, so having a very low number causes z-fighting. Garrett
|
||
|
{
|
||
|
EffectiveFogAlpha = 0.75f;
|
||
|
}
|
||
|
*p_patch_FOGCOL = FogCol;
|
||
|
|
||
|
|
||
|
// Check if we already set up the current variables
|
||
|
if (VarsUpToDate(viewport_num))
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// camera transform
|
||
|
CameraToWorld = camera_transform;
|
||
|
|
||
|
Dbg_MsgAssert(CameraToWorld[0][3] == 0.0f, ("CameraToWorld[0][3] is %f, not 0.0", CameraToWorld[0][3]));
|
||
|
Dbg_MsgAssert(CameraToWorld[1][3] == 0.0f, ("CameraToWorld[1][3] is %f, not 0.0", CameraToWorld[1][3]));
|
||
|
Dbg_MsgAssert(CameraToWorld[2][3] == 0.0f, ("CameraToWorld[2][3] is %f, not 0.0", CameraToWorld[2][3]));
|
||
|
Dbg_MsgAssert(CameraToWorld[3][3] == 1.0f, ("CameraToWorld[3][3] is %f, not 1.0", CameraToWorld[3][3]));
|
||
|
|
||
|
// need the transform to be well-formed as a 4x4 matrix
|
||
|
//CameraToWorld[0][3] = 0.0f;
|
||
|
//CameraToWorld[1][3] = 0.0f;
|
||
|
//CameraToWorld[2][3] = 0.0f;
|
||
|
//CameraToWorld[3][3] = 1.0f;
|
||
|
|
||
|
CameraToWorldRotation = CameraToWorld;
|
||
|
CameraToWorldRotation[3].Set(0.0f, 0.0f, 0.0f, 1.0f);
|
||
|
|
||
|
// world view transform
|
||
|
WorldToCameraRotation.Transpose(CameraToWorldRotation);
|
||
|
WorldToCamera = CameraToWorld;
|
||
|
WorldToCamera.InvertUniform();
|
||
|
|
||
|
// massage to get ROW register value
|
||
|
RowReg = SUB_INCH_PRECISION * WorldToCamera[3] * CameraToWorldRotation;
|
||
|
RowRegI[0] = (float)(int)RowReg[0];
|
||
|
RowRegI[1] = (float)(int)RowReg[1];
|
||
|
RowRegI[2] = (float)(int)RowReg[2];
|
||
|
RowRegI[3] = 0.0f;
|
||
|
RowRegF = RowReg - RowRegI;
|
||
|
RowRegF[3] = 0.0f;
|
||
|
|
||
|
// massage to get modified matrix
|
||
|
AdjustedWorldToCamera = WorldToCamera;
|
||
|
AdjustedWorldToCamera[3] = RowRegF * WorldToCameraRotation;
|
||
|
AdjustedWorldToCamera[3] *= RECIPROCAL_SUB_INCH_PRECISION;
|
||
|
AdjustedWorldToCamera[3][3] = 1.0f;
|
||
|
|
||
|
// view frustum
|
||
|
tx = tanf(view_angle*8.72664626e-03f); // tan of half (angle in radians)
|
||
|
ty = -(tx / viewport_aspect);
|
||
|
sx = 1.0f/sqrtf(1.0f+1.0f/(tx*tx));
|
||
|
sy = 1.0f/sqrtf(1.0f+1.0f/(ty*ty));
|
||
|
cx = 1.0f/sqrtf(1.0f+tx*tx);
|
||
|
cy = 1.0f/sqrtf(1.0f+ty*ty);
|
||
|
Near = near;
|
||
|
Far = far;
|
||
|
|
||
|
// outer frustum
|
||
|
Tx = tx * (620.0f/(((float) HRES) * 0.5f * viewport_rec.GetWidth()));
|
||
|
Ty = ty * (524.0f/(((float) VRES) * 0.5f * viewport_rec.GetHeight()));
|
||
|
Sx = 1.0f/sqrtf(1.0f+1.0f/(Tx*Tx));
|
||
|
Sy = 1.0f/sqrtf(1.0f+1.0f/(Ty*Ty));
|
||
|
Cx = 1.0f/sqrtf(1.0f+Tx*Tx);
|
||
|
Cy = 1.0f/sqrtf(1.0f+Ty*Ty);
|
||
|
|
||
|
// frustum transform
|
||
|
CameraToFrustum.Zero();
|
||
|
CameraToFrustum[0][0] = 1.0f/Tx;
|
||
|
CameraToFrustum[1][1] = 1.0f/Ty;
|
||
|
CameraToFrustum[2][2] = (near+far) / (far-near);
|
||
|
CameraToFrustum[3][2] = 2.0f * near * far / (near-far);
|
||
|
CameraToFrustum[2][3] = -1.0f;
|
||
|
AdjustedWorldToFrustum = AdjustedWorldToCamera * CameraToFrustum;
|
||
|
WorldToFrustum = WorldToCamera * CameraToFrustum;
|
||
|
|
||
|
// viewport (scale and offset from outer frustum)
|
||
|
ViewportScale.Set( 620.0f, 524.0f, 0.999f*powf(2,22), EffectiveFogAlpha);
|
||
|
ViewportOffset.Set(2048.0f, 2048.0f, 1.5f*powf(2,23), 0);
|
||
|
IntViewportScale.Set(620.0f, 524.0f, 0.999f*powf(2,-95), powf(2,32));
|
||
|
IntViewportOffset.Set(2048.0f+powf(2,19), 2048.0f+powf(2,19), 1.5f*powf(2,-94), powf(2,-32));
|
||
|
|
||
|
// viewport transform
|
||
|
FrustumToViewport.Zero();
|
||
|
FrustumToViewport[0][0] = ViewportScale[0];
|
||
|
FrustumToViewport[1][1] = ViewportScale[1];
|
||
|
FrustumToViewport[2][2] = ViewportScale[2];
|
||
|
FrustumToViewport[3][3] = ViewportScale[3];
|
||
|
FrustumToViewport[3][0] = ViewportOffset[0];
|
||
|
FrustumToViewport[3][1] = ViewportOffset[1];
|
||
|
FrustumToViewport[3][2] = ViewportOffset[2];
|
||
|
AdjustedWorldToViewport = AdjustedWorldToFrustum * FrustumToViewport;
|
||
|
WorldToViewport = WorldToFrustum * FrustumToViewport;
|
||
|
CameraToViewport = CameraToFrustum * FrustumToViewport;
|
||
|
|
||
|
FrustumToIntViewport.Zero();
|
||
|
FrustumToIntViewport[0][0] = IntViewportScale[0];
|
||
|
FrustumToIntViewport[1][1] = IntViewportScale[1];
|
||
|
FrustumToIntViewport[2][2] = IntViewportScale[2];
|
||
|
FrustumToIntViewport[3][3] = IntViewportScale[3];
|
||
|
FrustumToIntViewport[3][0] = IntViewportOffset[0];
|
||
|
FrustumToIntViewport[3][1] = IntViewportOffset[1];
|
||
|
FrustumToIntViewport[3][2] = IntViewportOffset[2];
|
||
|
AdjustedWorldToIntViewport = AdjustedWorldToFrustum * FrustumToIntViewport;
|
||
|
WorldToIntViewport = WorldToFrustum * FrustumToIntViewport;
|
||
|
|
||
|
|
||
|
// sky transforms
|
||
|
//SkyViewportScale.Set(1900.0f, 1900.0f, powf(2,-102), powf(2,32)); // convert these to constants
|
||
|
//SkyViewportOffset.Set(2048.0f+powf(2,19), 2048.0f+powf(2,19), 1.5f*powf(2,-101), powf(2,-32)); // convert these to constants
|
||
|
SkyViewportScale.Set( 620.0f, 524.0f, 0.01f, EffectiveFogAlpha);
|
||
|
SkyViewportOffset.Set(2048.0f, 2048.0f, 1.0f, 0);
|
||
|
|
||
|
SkyWorldToCamera = AdjustedWorldToCamera;
|
||
|
SkyWorldToCamera[3][0] = 0;
|
||
|
SkyWorldToCamera[3][1] = 0;
|
||
|
SkyWorldToCamera[3][2] = 0;
|
||
|
SkyWorldToFrustum = SkyWorldToCamera * CameraToFrustum;
|
||
|
SkyFrustumToViewport = FrustumToViewport;
|
||
|
SkyFrustumToViewport[2][2] = SkyViewportScale[2];
|
||
|
SkyFrustumToViewport[3][2] = SkyViewportOffset[2];
|
||
|
SkyWorldToViewport = SkyWorldToFrustum * SkyFrustumToViewport;
|
||
|
|
||
|
// used in 3D clipping
|
||
|
AltFrustum[0] = Near;
|
||
|
AltFrustum[1] = Far;
|
||
|
AltFrustum[2] = 620.0f/(((float) HRES) * 0.5f * viewport_rec.GetWidth());
|
||
|
AltFrustum[3] = 524.0f/(((float) VRES) * 0.5f * viewport_rec.GetHeight());
|
||
|
|
||
|
// invert the viewport scale and offset
|
||
|
InverseViewportScale[0] = 1.0f / ViewportScale[0];
|
||
|
InverseViewportScale[1] = 1.0f / ViewportScale[1];
|
||
|
InverseViewportScale[2] = 1.0f / ViewportScale[2];
|
||
|
InverseViewportScale[3] = 1.0f / ViewportScale[3];
|
||
|
|
||
|
InverseViewportOffset[0] = -InverseViewportScale[0] * ViewportOffset[0] * InverseViewportScale[3];
|
||
|
InverseViewportOffset[1] = -InverseViewportScale[1] * ViewportOffset[1] * InverseViewportScale[3];
|
||
|
InverseViewportOffset[2] = -InverseViewportScale[2] * ViewportOffset[2] * InverseViewportScale[3];
|
||
|
InverseViewportOffset[3] = -InverseViewportScale[3] * ViewportOffset[3] * InverseViewportScale[3];
|
||
|
InverseViewportOffset[3] = FogNear;
|
||
|
|
||
|
InverseIntViewportScale[0] = 1.0f / IntViewportScale[0];
|
||
|
InverseIntViewportScale[1] = 1.0f / IntViewportScale[1];
|
||
|
InverseIntViewportScale[2] = 1.0f / IntViewportScale[2];
|
||
|
InverseIntViewportScale[3] = 1.0f / IntViewportScale[3];
|
||
|
|
||
|
InverseIntViewportOffset[0] = -InverseIntViewportScale[0] * IntViewportOffset[0] * InverseIntViewportScale[3];
|
||
|
InverseIntViewportOffset[1] = -InverseIntViewportScale[1] * IntViewportOffset[1] * InverseIntViewportScale[3];
|
||
|
InverseIntViewportOffset[2] = -InverseIntViewportScale[2] * IntViewportOffset[2] * InverseIntViewportScale[3];
|
||
|
InverseIntViewportOffset[3] = 0.0f;
|
||
|
|
||
|
SkyInverseViewportScale[0] = 1.0f / SkyViewportScale[0];
|
||
|
SkyInverseViewportScale[1] = 1.0f / SkyViewportScale[1];
|
||
|
SkyInverseViewportScale[2] = 1.0f / SkyViewportScale[2];
|
||
|
SkyInverseViewportScale[3] = 1.0f / SkyViewportScale[3];
|
||
|
|
||
|
SkyInverseViewportOffset[0] = -SkyInverseViewportScale[0] * SkyViewportOffset[0] * SkyInverseViewportScale[3];
|
||
|
SkyInverseViewportOffset[1] = -SkyInverseViewportScale[1] * SkyViewportOffset[1] * SkyInverseViewportScale[3];
|
||
|
SkyInverseViewportOffset[2] = -SkyInverseViewportScale[2] * SkyViewportOffset[2] * SkyInverseViewportScale[3];
|
||
|
SkyInverseViewportOffset[3] = 0.0f;
|
||
|
|
||
|
|
||
|
|
||
|
// generate the GS register values we'll need for the viewport
|
||
|
reg_XYOFFSET = PackXYOFFSET((int)(16.0f * (2048.0f - ((float) HRES) * (viewport_rec.GetOriginX() + 0.5f * viewport_rec.GetWidth()))),
|
||
|
(int)(16.0f * (2048.0f - ((float) VRES) * (viewport_rec.GetOriginY() + 0.5f * viewport_rec.GetHeight()))));
|
||
|
reg_SCISSOR = PackSCISSOR( (int)(((float) HRES) * viewport_rec.GetOriginX()),
|
||
|
(int)(((float) HRES) * (viewport_rec.GetOriginX() + viewport_rec.GetWidth())) - 1,
|
||
|
(int)(((float) VRES) * viewport_rec.GetOriginY()),
|
||
|
(int)(((float) VRES) * (viewport_rec.GetOriginY() + viewport_rec.GetHeight())) - 1);
|
||
|
|
||
|
|
||
|
// inverse frustum transform
|
||
|
//ViewFrustumToCamera.Zero();
|
||
|
//ViewFrustumToCamera[0][0] = tx;
|
||
|
//ViewFrustumToCamera[1][1] = ty;
|
||
|
//ViewFrustumToCamera[2][3] = (near - far) / (2.0f * near * far);
|
||
|
//ViewFrustumToCamera[3][2] = -1.0f;
|
||
|
//ViewFrustumToCamera[3][3] = -(near + far) / (2.0f * near * far);
|
||
|
|
||
|
// get world coords of the far frustum vertices
|
||
|
Mth::Vector vFTL, vFTR, vFBL, vFBR;
|
||
|
vFTL.Set( tx*far, ty*far, far, 1.0f);
|
||
|
vFTL *= CameraToWorld;
|
||
|
vFTR.Set(-tx*far, ty*far, far, 1.0f);
|
||
|
vFTR *= CameraToWorld;
|
||
|
vFBL.Set( tx*far,-ty*far, far, 1.0f);
|
||
|
vFBL *= CameraToWorld;
|
||
|
vFBR.Set(-tx*far,-ty*far, far, 1.0f);
|
||
|
vFBR *= CameraToWorld;
|
||
|
|
||
|
Mth::Vector pos(WorldToCamera[3]);
|
||
|
for (int i=0; i<3; i++)
|
||
|
{
|
||
|
FrustumFlagsPlus[i] = (vFTL[i]>=pos[i] && vFTR[i]>=pos[i] && vFBL[i]>=pos[i] && vFBR[i]>=pos[i]);
|
||
|
FrustumFlagsMinus[i] = (vFTL[i]<=pos[i] && vFTR[i]<=pos[i] && vFBL[i]<=pos[i] && vFBR[i]<=pos[i]);
|
||
|
}
|
||
|
|
||
|
|
||
|
// Set ViewportMask. Will only draw CGeomNodes that have a non-zero result with (ViewportMask & Visibility)
|
||
|
ViewportMask = (1 << (VISIBILITY_FLAG_BIT + viewport_num));
|
||
|
|
||
|
// Mark current viewport and frame
|
||
|
RenderVarFrame = Frame;
|
||
|
RenderVarViewport = viewport_num;
|
||
|
|
||
|
}
|
||
|
|
||
|
void render::SetupVarsDMA(uint viewport_num, float near, float far)
|
||
|
{
|
||
|
|
||
|
|
||
|
// initialisation for new bounding volume tests
|
||
|
Mth::Matrix Normals;
|
||
|
|
||
|
asm ("vmaxw vf31,vf00,vf00w": :);
|
||
|
|
||
|
// view frustum R, L, B, T
|
||
|
Normals[0].Set(-cx, cx, 0, 0);
|
||
|
Normals[1].Set( 0, 0, cy,-cy);
|
||
|
Normals[2].Set(-sx,-sx,-sy,-sy);
|
||
|
Normals[3].Set( 0, 0, 0, 0);
|
||
|
|
||
|
Normals = WorldToCamera * Normals;
|
||
|
|
||
|
asm __volatile__("
|
||
|
|
||
|
lqc2 vf10,(%0)
|
||
|
lqc2 vf11,(%1)
|
||
|
lqc2 vf12,(%2)
|
||
|
lqc2 vf16,(%3)
|
||
|
vabs vf13,vf10
|
||
|
vabs vf14,vf11
|
||
|
vabs vf15,vf12
|
||
|
|
||
|
": : "r" (&Normals[0]), "r" (&Normals[1]), "r" (&Normals[2]), "r" (&Normals[3]));
|
||
|
|
||
|
// both frusta ?, ? N, F,
|
||
|
Normals[0].Set( 0, 0, 0, 0);
|
||
|
Normals[1].Set( 0, 0, 0, 0);
|
||
|
Normals[2].Set( 0, 0, -1, 1);
|
||
|
Normals[3].Set( 0, 0,near,-far);
|
||
|
|
||
|
Normals = WorldToCamera * Normals;
|
||
|
|
||
|
asm __volatile__("
|
||
|
|
||
|
lqc2 vf17,(%0)
|
||
|
lqc2 vf18,(%1)
|
||
|
lqc2 vf19,(%2)
|
||
|
lqc2 vf23,(%3)
|
||
|
vabs vf20,vf17
|
||
|
vabs vf21,vf18
|
||
|
vabs vf22,vf19
|
||
|
|
||
|
": : "r" (&Normals[0]), "r" (&Normals[1]), "r" (&Normals[2]), "r" (&Normals[3]));
|
||
|
|
||
|
|
||
|
// outer frustum R, L, B, T
|
||
|
Normals[0].Set(-Cx, Cx, 0, 0);
|
||
|
Normals[1].Set( 0, 0, Cy,-Cy);
|
||
|
Normals[2].Set(-Sx,-Sx,-Sy,-Sy);
|
||
|
Normals[3].Set( 0, 0, 0, 0);
|
||
|
|
||
|
Normals = WorldToCamera * Normals;
|
||
|
|
||
|
asm __volatile__("
|
||
|
|
||
|
lqc2 vf24,(%0)
|
||
|
lqc2 vf25,(%1)
|
||
|
lqc2 vf26,(%2)
|
||
|
lqc2 vf30,(%3)
|
||
|
vabs vf27,vf24
|
||
|
vabs vf28,vf25
|
||
|
vabs vf29,vf26
|
||
|
|
||
|
": : "r" (&Normals[0]), "r" (&Normals[1]), "r" (&Normals[2]), "r" (&Normals[3]));
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
#if OLD_FOG
|
||
|
// link in the dma list that does fogging postprocessing
|
||
|
dma::SetList(sGroup::pFog);
|
||
|
dma::Tag(dma::call, 0, (uint)&FogDma);
|
||
|
vif::NOP();
|
||
|
vif::NOP();
|
||
|
sGroup::pFog->Used[render::Field] = true;
|
||
|
|
||
|
dma::Gosub(SET_RENDERSTATE,2);
|
||
|
dma::Gosub(SET_FOGCOL,2);
|
||
|
#endif
|
||
|
|
||
|
|
||
|
// upload new vu1 code and set up double-buffering
|
||
|
dma::SetList(sGroup::pParticles);
|
||
|
dma::Tag(dma::ref, ((uint8 *)&NewMPGEnd-(uint8 *)&NewMPGStart+15)/16, (uint)&NewMPGStart);
|
||
|
vif::BASE(0);
|
||
|
vif::OFFSET(32);
|
||
|
|
||
|
// send vu1 data for particle setup
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::BeginDIRECT();
|
||
|
gif::Tag2(gs::A_D,1,PACKED,0,0,1,3);
|
||
|
gs::Reg2(gs::SCISSOR_1, render::reg_SCISSOR);
|
||
|
gs::Reg2(gs::XYOFFSET_1, render::reg_XYOFFSET);
|
||
|
gs::Reg2(gs::ZBUF_1, PackZBUF(ZBUFFER_START,PSMZ24,1));
|
||
|
vif::EndDIRECT();
|
||
|
vif::UNPACK(0, V4_32, 7, REL, SIGNED, 0);
|
||
|
vu1::StoreMat(*(Mat *)&render::WorldToIntViewport); // view transform
|
||
|
vif::StoreV4_32F(320.0f, 320.0f, 0.0f, 0.0f); // kvec
|
||
|
vif::StoreV4_32(0x39005000, 0x39006000, 0, 0x3F800000); // NTL cull test vec - w-component limits tan of apparent radius
|
||
|
vif::StoreV4_32(0x3900B000, 0x3900A000, 0, 0); // BR cull test vec
|
||
|
vif::MSCAL(8); // init sprites
|
||
|
dma::EndTag();
|
||
|
dma::SetList(NULL);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// Checks to see if the render variables are current
|
||
|
bool render::VarsUpToDate(uint viewport_num)
|
||
|
{
|
||
|
return (RenderVarFrame == Frame) && (RenderVarViewport == viewport_num);
|
||
|
}
|
||
|
|
||
|
// Invalidates the render variables
|
||
|
void render::InvalidateVars()
|
||
|
{
|
||
|
RenderVarFrame = 0xFFFFFFFF;
|
||
|
RenderVarViewport = 0xFFFFFFFF;
|
||
|
}
|
||
|
|
||
|
// a quick hack for Dave...
|
||
|
void DrawRectangle(int x0, int y0, int x1, int y1, uint32 rgba)
|
||
|
{
|
||
|
dma::SetList(sGroup::pEpilogue);
|
||
|
dma::BeginTag(dma::cnt, 0);
|
||
|
vif::FLUSH();
|
||
|
vif::BeginDIRECT();
|
||
|
gif::BeginTag2(gs::A_D, 1, PACKED, SPRITE|ABE, 1);
|
||
|
gs::Reg2(gs::ALPHA_1, PackALPHA(0,1,0,1,0));
|
||
|
gs::Reg2(gs::RGBAQ, (uint64)rgba);
|
||
|
gs::Reg2(gs::XYZ2, PackXYZ(x0,y0,0xFFFFFF));
|
||
|
gs::Reg2(gs::XYZ2, PackXYZ(x1,y1,0xFFFFFF));
|
||
|
gif::EndTag2(1);
|
||
|
vif::EndDIRECT();
|
||
|
dma::EndTag();
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
} // namespace NxPs2
|
||
|
|