2021-03-02 02:13:17 +00:00
|
|
|
// Project64 - A Nintendo 64 emulator
|
|
|
|
// http://www.pj64-emu.com/
|
|
|
|
// Copyright(C) 2001-2021 Project64
|
|
|
|
// Copyright(C) 2003-2009 Sergey 'Gonetz' Lipski
|
|
|
|
// Copyright(C) 2002 Dave2001
|
|
|
|
// GNU/GPLv2 licensed: https://gnu.org/licenses/gpl-2.0.html
|
2017-05-16 07:41:05 +00:00
|
|
|
#include <Project64-video/rdp.h>
|
|
|
|
#include <Project64-video/Gfx_1.3.h>
|
|
|
|
#include <Project64-video/trace.h>
|
|
|
|
#include <Project64-video/ucode.h>
|
|
|
|
#include <math.h>
|
|
|
|
#include "3dmath.h"
|
|
|
|
#include "Util.h"
|
|
|
|
|
|
|
|
void rsp_vertex(int v0, int n)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t addr = segoffset(rdp.cmd1) & 0x00FFFFFF;
|
|
|
|
int i;
|
|
|
|
float x, y, z;
|
|
|
|
|
|
|
|
rdp.v0 = v0; // Current vertex
|
|
|
|
rdp.vn = n; // Number to copy
|
|
|
|
|
|
|
|
// This is special, not handled in update(), but here
|
|
|
|
// * Matrix Pre-multiplication idea by Gonetz (Gonetz@ngs.ru)
|
|
|
|
if (rdp.update & UPDATE_MULT_MAT)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update ^= UPDATE_MULT_MAT;
|
|
|
|
MulMatrices(rdp.model, rdp.proj, rdp.combined);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
// *
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// This is special, not handled in update()
|
|
|
|
if (rdp.update & UPDATE_LIGHTS)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update ^= UPDATE_LIGHTS;
|
|
|
|
|
|
|
|
// Calculate light vectors
|
|
|
|
for (uint32_t l = 0; l < rdp.num_lights; l++)
|
|
|
|
{
|
|
|
|
InverseTransformVector(&rdp.light[l].dir_x, rdp.light_vector[l], rdp.model);
|
|
|
|
NormalizeVector(rdp.light_vector[l]);
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "rsp:vertex v0:%d, n:%d, from: %08lx", v0, n, addr);
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
for (i = 0; i < (n << 4); i += 16)
|
|
|
|
{
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX &v = rdp.vtx(v0 + (i >> 4));
|
2016-01-25 11:14:01 +00:00
|
|
|
x = (float)((short*)gfx.RDRAM)[(((addr + i) >> 1) + 0) ^ 1];
|
|
|
|
y = (float)((short*)gfx.RDRAM)[(((addr + i) >> 1) + 1) ^ 1];
|
|
|
|
z = (float)((short*)gfx.RDRAM)[(((addr + i) >> 1) + 2) ^ 1];
|
2017-05-19 20:03:52 +00:00
|
|
|
v.flags = ((uint16_t*)gfx.RDRAM)[(((addr + i) >> 1) + 3) ^ 1];
|
|
|
|
v.ou = (float)((short*)gfx.RDRAM)[(((addr + i) >> 1) + 4) ^ 1];
|
|
|
|
v.ov = (float)((short*)gfx.RDRAM)[(((addr + i) >> 1) + 5) ^ 1];
|
|
|
|
v.uv_scaled = 0;
|
|
|
|
v.a = ((uint8_t*)gfx.RDRAM)[(addr + i + 15) ^ 3];
|
|
|
|
|
|
|
|
v.x = x*rdp.combined[0][0] + y*rdp.combined[1][0] + z*rdp.combined[2][0] + rdp.combined[3][0];
|
|
|
|
v.y = x*rdp.combined[0][1] + y*rdp.combined[1][1] + z*rdp.combined[2][1] + rdp.combined[3][1];
|
|
|
|
v.z = x*rdp.combined[0][2] + y*rdp.combined[1][2] + z*rdp.combined[2][2] + rdp.combined[3][2];
|
|
|
|
v.w = x*rdp.combined[0][3] + y*rdp.combined[1][3] + z*rdp.combined[2][3] + rdp.combined[3][3];
|
|
|
|
|
|
|
|
if (fabs(v.w) < 0.001) v.w = 0.001f;
|
|
|
|
v.oow = 1.0f / v.w;
|
|
|
|
v.x_w = v.x * v.oow;
|
|
|
|
v.y_w = v.y * v.oow;
|
|
|
|
v.z_w = v.z * v.oow;
|
2016-01-25 11:14:01 +00:00
|
|
|
CalculateFog(v);
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
v.uv_calculated = 0xFFFFFFFF;
|
|
|
|
v.screen_translated = 0;
|
|
|
|
v.shade_mod = 0;
|
2016-01-25 11:14:01 +00:00
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
v.scr_off = 0;
|
|
|
|
if (v.x < -v.w) v.scr_off |= 1;
|
|
|
|
if (v.x > v.w) v.scr_off |= 2;
|
|
|
|
if (v.y < -v.w) v.scr_off |= 4;
|
|
|
|
if (v.y > v.w) v.scr_off |= 8;
|
|
|
|
if (v.w < 0.1f) v.scr_off |= 16;
|
|
|
|
// if (v.z_w > 1.0f) v.scr_off |= 32;
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
if (rdp.geom_mode & 0x00020000)
|
|
|
|
{
|
2017-05-19 20:03:52 +00:00
|
|
|
v.vec[0] = ((char*)gfx.RDRAM)[(addr + i + 12) ^ 3];
|
|
|
|
v.vec[1] = ((char*)gfx.RDRAM)[(addr + i + 13) ^ 3];
|
|
|
|
v.vec[2] = ((char*)gfx.RDRAM)[(addr + i + 14) ^ 3];
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.geom_mode & 0x40000)
|
|
|
|
{
|
|
|
|
if (rdp.geom_mode & 0x80000)
|
|
|
|
calc_linear(v);
|
|
|
|
else
|
|
|
|
calc_sphere(v);
|
|
|
|
}
|
2017-05-19 20:03:52 +00:00
|
|
|
NormalizeVector(v.vec);
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
calc_light(v);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-05-19 20:03:52 +00:00
|
|
|
v.r = ((uint8_t*)gfx.RDRAM)[(addr + i + 12) ^ 3];
|
|
|
|
v.g = ((uint8_t*)gfx.RDRAM)[(addr + i + 13) ^ 3];
|
|
|
|
v.b = ((uint8_t*)gfx.RDRAM)[(addr + i + 14) ^ 3];
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2017-05-19 20:03:52 +00:00
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "v%d - x: %f, y: %f, z: %f, w: %f, u: %f, v: %f, f: %f, z_w: %f, r=%d, g=%d, b=%d, a=%d", i >> 4, v.x, v.y, v.z, v.w, v.ou*rdp.tiles(rdp.cur_tile).s_scale, v.ov*rdp.tiles(rdp.cur_tile).t_scale, v.f, v.z_w, v.r, v.g, v.b, v.a);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-08-16 22:18:19 +00:00
|
|
|
void rsp_tri1(gfxVERTEX **v, uint16_t linew = 0)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (cull_tri(v))
|
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
update();
|
|
|
|
draw_tri(v, linew);
|
|
|
|
rdp.tri_n++;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-08-16 22:18:19 +00:00
|
|
|
void rsp_tri2(gfxVERTEX **v)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
int updated = 0;
|
|
|
|
|
|
|
|
if (cull_tri(v))
|
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
updated = 1;
|
|
|
|
update();
|
|
|
|
|
|
|
|
draw_tri(v);
|
|
|
|
rdp.tri_n++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cull_tri(v + 3))
|
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!updated)
|
|
|
|
update();
|
|
|
|
|
|
|
|
draw_tri(v + 3);
|
|
|
|
rdp.tri_n++;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:vertex - loads vertices
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_vertex()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
int v0 = (rdp.cmd0 >> 16) & 0xF; // Current vertex
|
|
|
|
int n = ((rdp.cmd0 >> 20) & 0xF) + 1; // Number of vertices to copy
|
|
|
|
rsp_vertex(v0, n);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ** Definitions **
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_load(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
memcpy(rdp.model, m, 64); // 4*4*4(float)
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update |= UPDATE_MULT_MAT | UPDATE_LIGHTS;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_mul(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
DECLAREALIGN16VAR(m_src[4][4]);
|
|
|
|
memcpy(m_src, rdp.model, 64);
|
|
|
|
MulMatrices(m, m_src, rdp.model);
|
|
|
|
rdp.update |= UPDATE_MULT_MAT | UPDATE_LIGHTS;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_push()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.model_i == rdp.model_stack_size)
|
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "** Model matrix stack overflow ** too many pushes");
|
2016-01-25 11:14:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(rdp.model_stack[rdp.model_i], rdp.model, 64);
|
|
|
|
rdp.model_i++;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_pop(int num = 1)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.model_i > num - 1)
|
|
|
|
{
|
|
|
|
rdp.model_i -= num;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "** Model matrix stack error ** too many pops");
|
2016-01-25 11:14:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
memcpy(rdp.model, rdp.model_stack[rdp.model_i], 64);
|
|
|
|
rdp.update |= UPDATE_MULT_MAT | UPDATE_LIGHTS;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_load_push(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_push();
|
|
|
|
modelview_load(m);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void modelview_mul_push(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_push();
|
|
|
|
modelview_mul(m);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void projection_load(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
memcpy(rdp.proj, m, 64); // 4*4*4(float)
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update |= UPDATE_MULT_MAT;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void projection_mul(float m[4][4])
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
DECLAREALIGN16VAR(m_src[4][4]);
|
|
|
|
memcpy(m_src, rdp.proj, 64);
|
|
|
|
MulMatrices(m, m_src, rdp.proj);
|
|
|
|
rdp.update |= UPDATE_MULT_MAT;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
void load_matrix(float m[4][4], uint32_t addr)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "matrix - addr: %08lx", addr);
|
2016-01-25 11:14:01 +00:00
|
|
|
int x, y; // matrix index
|
|
|
|
addr >>= 1;
|
|
|
|
uint16_t * src = (uint16_t*)gfx.RDRAM;
|
|
|
|
for (x = 0; x < 16; x += 4) { // Adding 4 instead of one, just to remove mult. later
|
|
|
|
for (y = 0; y < 4; y++) {
|
|
|
|
m[x >> 2][y] = (float)(
|
|
|
|
(((int32_t)src[(addr + x + y) ^ 1]) << 16) |
|
|
|
|
src[(addr + x + y + 16) ^ 1]
|
|
|
|
) / 65536.0f;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:matrix - performs matrix operations
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_matrix()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:matrix ");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// Use segment offset to get the address
|
|
|
|
uint32_t addr = segoffset(rdp.cmd1) & 0x00FFFFFF;
|
|
|
|
uint8_t command = (uint8_t)((rdp.cmd0 >> 16) & 0xFF);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
DECLAREALIGN16VAR(m[4][4]);
|
|
|
|
load_matrix(m, addr);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
switch (command)
|
|
|
|
{
|
|
|
|
case 0: // modelview mul nopush
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "modelview mul");
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_mul(m);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 1: // projection mul nopush
|
|
|
|
case 5: // projection mul push, can't push projection
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "projection mul");
|
2016-01-25 11:14:01 +00:00
|
|
|
projection_mul(m);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 2: // modelview load nopush
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "modelview load");
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_load(m);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 3: // projection load nopush
|
|
|
|
case 7: // projection load push, can't push projection
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "projection load");
|
2016-01-25 11:14:01 +00:00
|
|
|
projection_load(m);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 4: // modelview mul push
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "modelview mul push");
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_mul_push(m);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 6: // modelview load push
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "modelview load push");
|
2016-01-25 11:14:01 +00:00
|
|
|
modelview_load_push(m);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "Unknown matrix command, %02lx", command);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", m[0][0], m[0][1], m[0][2], m[0][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", m[1][0], m[1][1], m[1][2], m[1][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", m[2][0], m[2][1], m[2][2], m[2][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", m[3][0], m[3][1], m[3][2], m[3][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "\nmodel\n{%f,%f,%f,%f}", rdp.model[0][0], rdp.model[0][1], rdp.model[0][2], rdp.model[0][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.model[1][0], rdp.model[1][1], rdp.model[1][2], rdp.model[1][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.model[2][0], rdp.model[2][1], rdp.model[2][2], rdp.model[2][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.model[3][0], rdp.model[3][1], rdp.model[3][2], rdp.model[3][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "\nproj\n{%f,%f,%f,%f}", rdp.proj[0][0], rdp.proj[0][1], rdp.proj[0][2], rdp.proj[0][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.proj[1][0], rdp.proj[1][1], rdp.proj[1][2], rdp.proj[1][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.proj[2][0], rdp.proj[2][1], rdp.proj[2][2], rdp.proj[2][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.proj[3][0], rdp.proj[3][1], rdp.proj[3][2], rdp.proj[3][3]);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:movemem - loads a structure with data
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_movemem()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:movemem ");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t i, a;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// Check the command
|
|
|
|
switch ((rdp.cmd0 >> 16) & 0xFF)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x80:
|
|
|
|
{
|
|
|
|
a = (segoffset(rdp.cmd1) & 0xFFFFFF) >> 1;
|
|
|
|
|
|
|
|
short scale_x = ((short*)gfx.RDRAM)[(a + 0) ^ 1] / 4;
|
|
|
|
short scale_y = ((short*)gfx.RDRAM)[(a + 1) ^ 1] / 4;
|
|
|
|
short scale_z = ((short*)gfx.RDRAM)[(a + 2) ^ 1];
|
|
|
|
short trans_x = ((short*)gfx.RDRAM)[(a + 4) ^ 1] / 4;
|
|
|
|
short trans_y = ((short*)gfx.RDRAM)[(a + 5) ^ 1] / 4;
|
|
|
|
short trans_z = ((short*)gfx.RDRAM)[(a + 6) ^ 1];
|
2017-03-13 06:17:52 +00:00
|
|
|
if (g_settings->correct_viewport())
|
2016-01-25 11:14:01 +00:00
|
|
|
{
|
2017-03-13 06:17:52 +00:00
|
|
|
scale_x = (short)abs(scale_x);
|
|
|
|
scale_y = (short)abs(scale_y);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
rdp.view_scale[0] = scale_x * rdp.scale_x;
|
|
|
|
rdp.view_scale[1] = -scale_y * rdp.scale_y;
|
|
|
|
rdp.view_scale[2] = 32.0f * scale_z;
|
|
|
|
rdp.view_trans[0] = trans_x * rdp.scale_x;
|
|
|
|
rdp.view_trans[1] = trans_y * rdp.scale_y;
|
|
|
|
rdp.view_trans[2] = 32.0f * trans_z;
|
|
|
|
|
|
|
|
// there are other values than x and y, but I don't know what they do
|
|
|
|
|
|
|
|
rdp.update |= UPDATE_VIEWPORT;
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "viewport scale(%d, %d, %d), trans(%d, %d, %d), from:%08lx", scale_x, scale_y, scale_z,
|
2016-01-25 11:14:01 +00:00
|
|
|
trans_x, trans_y, trans_z, rdp.cmd1);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x82:
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
a = segoffset(rdp.cmd1) & 0x00ffffff;
|
|
|
|
char dir_x = ((char*)gfx.RDRAM)[(a + 8) ^ 3];
|
|
|
|
rdp.lookat[1][0] = (float)(dir_x) / 127.0f;
|
|
|
|
char dir_y = ((char*)gfx.RDRAM)[(a + 9) ^ 3];
|
|
|
|
rdp.lookat[1][1] = (float)(dir_y) / 127.0f;
|
|
|
|
char dir_z = ((char*)gfx.RDRAM)[(a + 10) ^ 3];
|
|
|
|
rdp.lookat[1][2] = (float)(dir_z) / 127.0f;
|
|
|
|
if (!dir_x && !dir_y)
|
|
|
|
rdp.use_lookat = FALSE;
|
|
|
|
else
|
|
|
|
rdp.use_lookat = TRUE;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "lookat_y (%f, %f, %f)", rdp.lookat[1][0], rdp.lookat[1][1], rdp.lookat[1][2]);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x84:
|
|
|
|
a = segoffset(rdp.cmd1) & 0x00ffffff;
|
|
|
|
rdp.lookat[0][0] = (float)(((char*)gfx.RDRAM)[(a + 8) ^ 3]) / 127.0f;
|
|
|
|
rdp.lookat[0][1] = (float)(((char*)gfx.RDRAM)[(a + 9) ^ 3]) / 127.0f;
|
|
|
|
rdp.lookat[0][2] = (float)(((char*)gfx.RDRAM)[(a + 10) ^ 3]) / 127.0f;
|
|
|
|
rdp.use_lookat = TRUE;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "lookat_x (%f, %f, %f)", rdp.lookat[1][0], rdp.lookat[1][1], rdp.lookat[1][2]);
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x86:
|
|
|
|
case 0x88:
|
|
|
|
case 0x8a:
|
|
|
|
case 0x8c:
|
|
|
|
case 0x8e:
|
|
|
|
case 0x90:
|
|
|
|
case 0x92:
|
|
|
|
case 0x94:
|
|
|
|
// Get the light #
|
|
|
|
i = (((rdp.cmd0 >> 16) & 0xff) - 0x86) >> 1;
|
|
|
|
a = segoffset(rdp.cmd1) & 0x00ffffff;
|
|
|
|
|
|
|
|
// Get the data
|
|
|
|
rdp.light[i].r = (float)(((uint8_t*)gfx.RDRAM)[(a + 0) ^ 3]) / 255.0f;
|
|
|
|
rdp.light[i].g = (float)(((uint8_t*)gfx.RDRAM)[(a + 1) ^ 3]) / 255.0f;
|
|
|
|
rdp.light[i].b = (float)(((uint8_t*)gfx.RDRAM)[(a + 2) ^ 3]) / 255.0f;
|
|
|
|
rdp.light[i].a = 1.0f;
|
|
|
|
// ** Thanks to Icepir8 for pointing this out **
|
|
|
|
// Lighting must be signed byte instead of byte
|
|
|
|
rdp.light[i].dir_x = (float)(((char*)gfx.RDRAM)[(a + 8) ^ 3]) / 127.0f;
|
|
|
|
rdp.light[i].dir_y = (float)(((char*)gfx.RDRAM)[(a + 9) ^ 3]) / 127.0f;
|
|
|
|
rdp.light[i].dir_z = (float)(((char*)gfx.RDRAM)[(a + 10) ^ 3]) / 127.0f;
|
|
|
|
// **
|
|
|
|
|
|
|
|
//rdp.update |= UPDATE_LIGHTS;
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "light: n: %d, r: %.3f, g: %.3f, b: %.3f, x: %.3f, y: %.3f, z: %.3f",
|
2016-01-25 11:14:01 +00:00
|
|
|
i, rdp.light[i].r, rdp.light[i].g, rdp.light[i].b,
|
|
|
|
rdp.light_vector[i][0], rdp.light_vector[i][1], rdp.light_vector[i][2]);
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x9E: //gSPForceMatrix command. Modification of uc2_movemem:matrix. Gonetz.
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
// do not update the combined matrix!
|
|
|
|
rdp.update &= ~UPDATE_MULT_MAT;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t addr = segoffset(rdp.cmd1) & 0x00FFFFFF;
|
|
|
|
load_matrix(rdp.combined, addr);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
addr = rdp.pc[rdp.pc_i] & BMASK;
|
|
|
|
rdp.pc[rdp.pc_i] = (addr + 24) & BMASK; //skip next 3 command, b/c they all are part of gSPForceMatrix
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.combined[0][0], rdp.combined[0][1], rdp.combined[0][2], rdp.combined[0][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.combined[1][0], rdp.combined[1][1], rdp.combined[1][2], rdp.combined[1][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.combined[2][0], rdp.combined[2][1], rdp.combined[2][2], rdp.combined[2][3]);
|
|
|
|
WriteTrace(TraceRDP, TraceVerbose, "{%f,%f,%f,%f}", rdp.combined[3][0], rdp.combined[3][1], rdp.combined[3][2], rdp.combined[3][3]);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//next 3 command should never appear since they will be skipped in previous command
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x98:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "matrix 0 - IGNORED");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x9A:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "matrix 1 - IGNORED");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x9C:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "matrix 2 - IGNORED");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "uc0:movemem unknown (index: 0x%08lx)", (rdp.cmd0 >> 16) & 0xFF);
|
|
|
|
WriteTrace(TraceRDP, TraceDebug, "unknown (index: 0x%08lx)", (rdp.cmd0 >> 16) & 0xFF);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2016-02-11 09:58:42 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:displaylist - makes a call to another section of code
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_displaylist()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t addr = segoffset(rdp.cmd1) & 0x00FFFFFF;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// This fixes partially Gauntlet: Legends
|
2016-02-11 09:58:42 +00:00
|
|
|
if (addr == rdp.pc[rdp.pc_i] - 8) { WriteTrace(TraceRDP, TraceDebug, "display list not executed!"); return; }
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t push = (rdp.cmd0 >> 16) & 0xFF; // push the old location?
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:displaylist: %08lx, push:%s", addr, push ? "no" : "yes");
|
|
|
|
WriteTrace(TraceRDP, TraceDebug, " (seg %d, offset %08lx)", (rdp.cmd1 >> 24) & 0x0F, rdp.cmd1 & 0x00FFFFFF);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
switch (push)
|
|
|
|
{
|
|
|
|
case 0: // push
|
2016-02-11 09:58:42 +00:00
|
|
|
if (rdp.pc_i >= 9)
|
|
|
|
{
|
|
|
|
WriteTrace(TraceRDP, TraceWarning, "** DL stack overflow **");
|
2016-01-25 11:14:01 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
rdp.pc_i++; // go to the next PC in the stack
|
|
|
|
rdp.pc[rdp.pc_i] = addr; // jump to the address
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 1: // no push
|
|
|
|
rdp.pc[rdp.pc_i] = addr; // just jump to the address
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "Unknown displaylist operation");
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// tri1 - renders a triangle
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_tri1()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:tri1 #%d - %d, %d, %d", rdp.tri_n,
|
2016-01-25 11:14:01 +00:00
|
|
|
((rdp.cmd1 >> 16) & 0xFF) / 10,
|
|
|
|
((rdp.cmd1 >> 8) & 0xFF) / 10,
|
|
|
|
(rdp.cmd1 & 0xFF) / 10);
|
|
|
|
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX *vtx[3] = {
|
2017-05-19 20:03:52 +00:00
|
|
|
&rdp.vtx(((rdp.cmd1 >> 16) & 0xFF) / 10),
|
|
|
|
&rdp.vtx(((rdp.cmd1 >> 8) & 0xFF) / 10),
|
|
|
|
&rdp.vtx((rdp.cmd1 & 0xFF) / 10)
|
2016-01-25 11:14:01 +00:00
|
|
|
};
|
2017-01-29 06:26:41 +00:00
|
|
|
if (g_settings->hacks(CSettings::hack_Makers))
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.force_wrap = FALSE;
|
|
|
|
for (int i = 0; i < 3; i++)
|
|
|
|
{
|
2017-05-19 20:03:52 +00:00
|
|
|
if (vtx[i]->ou < 0.0f || vtx[i]->ov < 0.0f)
|
2016-01-25 11:14:01 +00:00
|
|
|
{
|
|
|
|
rdp.force_wrap = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2017-05-19 20:03:52 +00:00
|
|
|
rsp_tri1(vtx);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:enddl - ends a call made by uc0:displaylist
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_enddl()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:enddl");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.pc_i == 0)
|
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "RDP end");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// Halt execution here
|
2017-05-15 07:53:33 +00:00
|
|
|
rdp.halt = true;
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.pc_i--;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_culldl()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
uint8_t vStart = (uint8_t)((rdp.cmd0 & 0x00FFFFFF) / 40) & 0xF;
|
|
|
|
uint8_t vEnd = (uint8_t)(rdp.cmd1 / 40) & 0x0F;
|
|
|
|
uint32_t cond = 0;
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:culldl start: %d, end: %d", vStart, vEnd);
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
if (vEnd < vStart) return;
|
|
|
|
for (uint16_t i = vStart; i <= vEnd; i++)
|
|
|
|
{
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX &v = rdp.vtx(i);
|
2016-01-25 11:14:01 +00:00
|
|
|
// Check if completely off the screen (quick frustrum clipping for 90 FOV)
|
2017-05-19 20:03:52 +00:00
|
|
|
if (v.x >= -v.w)
|
2016-01-25 11:14:01 +00:00
|
|
|
cond |= 0x01;
|
2017-05-19 20:03:52 +00:00
|
|
|
if (v.x <= v.w)
|
2016-01-25 11:14:01 +00:00
|
|
|
cond |= 0x02;
|
2017-05-19 20:03:52 +00:00
|
|
|
if (v.y >= -v.w)
|
2016-01-25 11:14:01 +00:00
|
|
|
cond |= 0x04;
|
2017-05-19 20:03:52 +00:00
|
|
|
if (v.y <= v.w)
|
2016-01-25 11:14:01 +00:00
|
|
|
cond |= 0x08;
|
2017-05-19 20:03:52 +00:00
|
|
|
if (v.w >= 0.1f)
|
2016-01-25 11:14:01 +00:00
|
|
|
cond |= 0x10;
|
|
|
|
|
|
|
|
if (cond == 0x1F)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, " - "); // specify that the enddl is not a real command
|
2016-01-25 11:14:01 +00:00
|
|
|
uc0_enddl();
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_popmatrix()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:popmatrix");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t param = rdp.cmd1;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
switch (param)
|
|
|
|
{
|
|
|
|
case 0: // modelview
|
|
|
|
modelview_pop();
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 1: // projection, can't
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "Unknown uc0:popmatrix command: 0x%08lx", param);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc6_obj_sprite();
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_modifyvtx(uint8_t where, uint16_t vtx, uint32_t val)
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX &v = rdp.vtx(vtx);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
switch (where)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
uc6_obj_sprite();
|
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x10: // RGBA
|
2017-05-19 20:03:52 +00:00
|
|
|
v.r = (uint8_t)(val >> 24);
|
|
|
|
v.g = (uint8_t)((val >> 16) & 0xFF);
|
|
|
|
v.b = (uint8_t)((val >> 8) & 0xFF);
|
|
|
|
v.a = (uint8_t)(val & 0xFF);
|
|
|
|
v.shade_mod = 0;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "RGBA: %d, %d, %d, %d", v.r, v.g, v.b, v.a);
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x14: // ST
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
float scale = rdp.Persp_en ? 0.03125f : 0.015625f;
|
2017-05-19 20:03:52 +00:00
|
|
|
v.ou = (float)((short)(val >> 16)) * scale;
|
|
|
|
v.ov = (float)((short)(val & 0xFFFF)) * scale;
|
|
|
|
v.uv_calculated = 0xFFFFFFFF;
|
|
|
|
v.uv_scaled = 1;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "u/v: (%04lx, %04lx), (%f, %f)", (short)(val >> 16), (short)(val & 0xFFFF),
|
2017-05-19 20:03:52 +00:00
|
|
|
v.ou, v.ov);
|
2013-04-04 21:22:19 +00:00
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x18: // XY screen
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
float scr_x = (float)((short)(val >> 16)) / 4.0f;
|
|
|
|
float scr_y = (float)((short)(val & 0xFFFF)) / 4.0f;
|
2017-05-19 20:03:52 +00:00
|
|
|
v.screen_translated = 2;
|
|
|
|
v.sx = scr_x * rdp.scale_x + rdp.offset_x;
|
|
|
|
v.sy = scr_y * rdp.scale_y + rdp.offset_y;
|
|
|
|
if (v.w < 0.01f)
|
2016-01-25 11:14:01 +00:00
|
|
|
{
|
2017-05-19 20:03:52 +00:00
|
|
|
v.w = 1.0f;
|
|
|
|
v.oow = 1.0f;
|
|
|
|
v.z_w = 1.0f;
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2017-05-19 20:03:52 +00:00
|
|
|
v.sz = rdp.view_trans[2] + v.z_w * rdp.view_scale[2];
|
2016-01-25 11:14:01 +00:00
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
v.scr_off = 0;
|
|
|
|
if (scr_x < 0) v.scr_off |= 1;
|
|
|
|
if (scr_x > rdp.vi_width) v.scr_off |= 2;
|
|
|
|
if (scr_y < 0) v.scr_off |= 4;
|
|
|
|
if (scr_y > rdp.vi_height) v.scr_off |= 8;
|
|
|
|
if (v.w < 0.1f) v.scr_off |= 16;
|
2016-01-25 11:14:01 +00:00
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "x/y: (%f, %f)", scr_x, scr_y);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x1C: // Z screen
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
float scr_z = (float)((short)(val >> 16));
|
2017-05-19 20:03:52 +00:00
|
|
|
v.z_w = (scr_z - rdp.view_trans[2]) / rdp.view_scale[2];
|
|
|
|
v.z = v.z_w * v.w;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "z: %f", scr_z);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "UNKNOWN");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// uc0:moveword - moves a word to someplace, like the segment pointers
|
|
|
|
//
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_moveword()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:moveword ");
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
// Find which command this is (lowest byte of cmd0)
|
|
|
|
switch (rdp.cmd0 & 0xFF)
|
|
|
|
{
|
|
|
|
case 0x00:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "matrix - IGNORED");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x02:
|
|
|
|
rdp.num_lights = ((rdp.cmd1 - 0x80000000) >> 5) - 1; // inverse of equation
|
|
|
|
if (rdp.num_lights > 8) rdp.num_lights = 0;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update |= UPDATE_LIGHTS;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "numlights: %d", rdp.num_lights);
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x04:
|
|
|
|
if (((rdp.cmd0 >> 8) & 0xFFFF) == 0x04)
|
|
|
|
{
|
|
|
|
rdp.clip_ratio = sqrt((float)rdp.cmd1);
|
|
|
|
rdp.update |= UPDATE_VIEWPORT;
|
|
|
|
}
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "clip %08lx, %08lx", rdp.cmd0, rdp.cmd1);
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x06: // segment
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "segment: %08lx -> seg%d", rdp.cmd1, (rdp.cmd0 >> 10) & 0x0F);
|
2016-01-25 11:14:01 +00:00
|
|
|
if ((rdp.cmd1&BMASK) < BMASK)
|
|
|
|
rdp.segment[(rdp.cmd0 >> 10) & 0x0F] = rdp.cmd1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x08:
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.fog_multiplier = (short)(rdp.cmd1 >> 16);
|
|
|
|
rdp.fog_offset = (short)(rdp.cmd1 & 0x0000FFFF);
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "fog: multiplier: %f, offset: %f", rdp.fog_multiplier, rdp.fog_offset);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x0a: // moveword LIGHTCOL
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
int n = (rdp.cmd0 & 0xE000) >> 13;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "lightcol light:%d, %08lx", n, rdp.cmd1);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.light[n].r = (float)((rdp.cmd1 >> 24) & 0xFF) / 255.0f;
|
|
|
|
rdp.light[n].g = (float)((rdp.cmd1 >> 16) & 0xFF) / 255.0f;
|
|
|
|
rdp.light[n].b = (float)((rdp.cmd1 >> 8) & 0xFF) / 255.0f;
|
|
|
|
rdp.light[n].a = 255;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x0c:
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
uint16_t val = (uint16_t)((rdp.cmd0 >> 8) & 0xFFFF);
|
|
|
|
uint16_t vtx = val / 40;
|
|
|
|
uint8_t where = val % 40;
|
|
|
|
uc0_modifyvtx(where, vtx, rdp.cmd1);
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:modifyvtx: vtx: %d, where: 0x%02lx, val: %08lx - ", vtx, where, rdp.cmd1);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
case 0x0e:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "perspnorm - IGNORED");
|
2016-01-25 11:14:01 +00:00
|
|
|
break;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
default:
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceWarning, "uc0:moveword unknown (index: 0x%08lx)", rdp.cmd0 & 0xFF);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_texture()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
int tile = (rdp.cmd0 >> 8) & 0x07;
|
2017-01-29 06:26:41 +00:00
|
|
|
if (tile == 7 && g_settings->hacks(CSettings::hack_Supercross))
|
|
|
|
{
|
|
|
|
tile = 0; //fix for supercross 2000
|
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.mipmap_level = (rdp.cmd0 >> 11) & 0x07;
|
|
|
|
uint32_t on = (rdp.cmd0 & 0xFF);
|
|
|
|
rdp.cur_tile = tile;
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
if (on)
|
|
|
|
{
|
|
|
|
uint16_t s = (uint16_t)((rdp.cmd1 >> 16) & 0xFFFF);
|
|
|
|
uint16_t t = (uint16_t)(rdp.cmd1 & 0xFFFF);
|
|
|
|
|
2017-05-19 11:19:34 +00:00
|
|
|
TILE *tmp_tile = &rdp.tiles(tile);
|
2016-01-25 11:14:01 +00:00
|
|
|
tmp_tile->on = 1;
|
|
|
|
tmp_tile->org_s_scale = s;
|
|
|
|
tmp_tile->org_t_scale = t;
|
|
|
|
tmp_tile->s_scale = (float)(s + 1) / 65536.0f;
|
|
|
|
tmp_tile->t_scale = (float)(t + 1) / 65536.0f;
|
|
|
|
tmp_tile->s_scale /= 32.0f;
|
|
|
|
tmp_tile->t_scale /= 32.0f;
|
|
|
|
|
|
|
|
rdp.update |= UPDATE_TEXTURE;
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:texture: tile: %d, mipmap_lvl: %d, on: %d, s_scale: %f, t_scale: %f", tile, rdp.mipmap_level, on, tmp_tile->s_scale, tmp_tile->t_scale);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:texture skipped b/c of off");
|
2017-05-19 11:19:34 +00:00
|
|
|
rdp.tiles(tile).on = 0;
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_setothermode_h()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:setothermode_h: ");
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
int shift, len;
|
2017-01-30 17:34:36 +00:00
|
|
|
if (g_settings->ucode() == CSettings::ucode_F3DEX2 || g_settings->ucode() == CSettings::ucode_CBFD)
|
2016-01-25 11:14:01 +00:00
|
|
|
{
|
|
|
|
len = (rdp.cmd0 & 0xFF) + 1;
|
|
|
|
shift = 32 - ((rdp.cmd0 >> 8) & 0xFF) - len;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
shift = (rdp.cmd0 >> 8) & 0xFF;
|
|
|
|
len = rdp.cmd0 & 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t mask = 0;
|
|
|
|
int i = len;
|
|
|
|
for (; i; i--)
|
|
|
|
mask = (mask << 1) | 1;
|
|
|
|
mask <<= shift;
|
|
|
|
|
|
|
|
rdp.cmd1 &= mask;
|
|
|
|
rdp.othermode_h &= ~mask;
|
|
|
|
rdp.othermode_h |= rdp.cmd1;
|
|
|
|
|
|
|
|
if (mask & 0x00000030) // alpha dither mode
|
|
|
|
{
|
|
|
|
rdp.alpha_dither_mode = (rdp.othermode_h >> 4) & 0x3;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "alpha dither mode: %s", str_dither[rdp.alpha_dither_mode]);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x000000C0) // rgb dither mode
|
|
|
|
{
|
|
|
|
uint32_t dither_mode = (rdp.othermode_h >> 6) & 0x3;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "rgb dither mode: %s", str_dither[dither_mode]);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x00003000) // filter mode
|
|
|
|
{
|
|
|
|
rdp.filter_mode = (int)((rdp.othermode_h & 0x00003000) >> 12);
|
|
|
|
rdp.update |= UPDATE_TEXTURE;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "filter mode: %s", str_filter[rdp.filter_mode]);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x0000C000) // tlut mode
|
|
|
|
{
|
|
|
|
rdp.tlut_mode = (uint8_t)((rdp.othermode_h & 0x0000C000) >> 14);
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "tlut mode: %s", str_tlut[rdp.tlut_mode]);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x00300000) // cycle type
|
|
|
|
{
|
|
|
|
rdp.cycle_mode = (uint8_t)((rdp.othermode_h & 0x00300000) >> 20);
|
|
|
|
rdp.update |= UPDATE_ZBUF_ENABLED;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "cycletype: %d", rdp.cycle_mode);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x00010000) // LOD enable
|
|
|
|
{
|
|
|
|
rdp.LOD_en = (rdp.othermode_h & 0x00010000) ? TRUE : FALSE;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "LOD_en: %d", rdp.LOD_en);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x00080000) // Persp enable
|
|
|
|
{
|
|
|
|
if (rdp.persp_supported)
|
|
|
|
rdp.Persp_en = (rdp.othermode_h & 0x00080000) ? TRUE : FALSE;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "Persp_en: %d", rdp.Persp_en);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t unk = mask & 0x0FFC60F0F;
|
|
|
|
if (unk) // unknown portions, LARGE
|
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "UNKNOWN PORTIONS: shift: %d, len: %d, unknowns: %08lx", shift, len, unk);
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_setothermode_l()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:setothermode_l ");
|
2016-01-25 11:14:01 +00:00
|
|
|
|
|
|
|
int shift, len;
|
2017-01-30 17:34:36 +00:00
|
|
|
if (g_settings->ucode() == CSettings::ucode_F3DEX2 || g_settings->ucode() == CSettings::ucode_CBFD)
|
2016-01-25 11:14:01 +00:00
|
|
|
{
|
|
|
|
len = (rdp.cmd0 & 0xFF) + 1;
|
|
|
|
shift = 32 - ((rdp.cmd0 >> 8) & 0xFF) - len;
|
|
|
|
if (shift < 0) shift = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
len = rdp.cmd0 & 0xFF;
|
|
|
|
shift = (rdp.cmd0 >> 8) & 0xFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t mask = 0;
|
|
|
|
int i = len;
|
|
|
|
for (; i; i--)
|
|
|
|
mask = (mask << 1) | 1;
|
|
|
|
mask <<= shift;
|
|
|
|
|
|
|
|
rdp.cmd1 &= mask;
|
|
|
|
rdp.othermode_l &= ~mask;
|
|
|
|
rdp.othermode_l |= rdp.cmd1;
|
|
|
|
|
|
|
|
if (mask & 0x00000003) // alpha compare
|
|
|
|
{
|
|
|
|
rdp.acmp = rdp.othermode_l & 0x00000003;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "alpha compare %s", ACmp[rdp.acmp]);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update |= UPDATE_ALPHA_COMPARE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0x00000004) // z-src selection
|
|
|
|
{
|
|
|
|
rdp.zsrc = (rdp.othermode_l & 0x00000004) >> 2;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "z-src sel: %s", str_zs[rdp.zsrc]);
|
|
|
|
WriteTrace(TraceRDP, TraceDebug, "z-src sel: %08lx", rdp.zsrc);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.update |= UPDATE_ZBUF_ENABLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mask & 0xFFFFFFF8) // rendermode / blender bits
|
|
|
|
{
|
|
|
|
rdp.update |= UPDATE_FOG_ENABLED; //if blender has no fog bits, fog must be set off
|
|
|
|
rdp.render_mode_changed |= rdp.rm ^ rdp.othermode_l;
|
|
|
|
rdp.rm = rdp.othermode_l;
|
2017-03-12 20:19:09 +00:00
|
|
|
if (g_settings->flame_corona() && (rdp.rm == 0x00504341)) //hack for flame's corona
|
2017-01-29 06:26:41 +00:00
|
|
|
{
|
|
|
|
rdp.othermode_l |= 0x00000010;
|
|
|
|
}
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "rendermode: %08lx", rdp.othermode_l); // just output whole othermode_l
|
2016-01-25 11:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// there is not one setothermode_l that's not handled :)
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_setgeometrymode()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.geom_mode |= rdp.cmd1;
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:setgeometrymode %08lx; result: %08lx", rdp.cmd1, rdp.geom_mode);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00000001) // Z-Buffer enable
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!(rdp.flags & ZBUF_ENABLED))
|
|
|
|
{
|
|
|
|
rdp.flags |= ZBUF_ENABLED;
|
|
|
|
rdp.update |= UPDATE_ZBUF_ENABLED;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00001000) // Front culling
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!(rdp.flags & CULL_FRONT))
|
|
|
|
{
|
|
|
|
rdp.flags |= CULL_FRONT;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00002000) // Back culling
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!(rdp.flags & CULL_BACK))
|
|
|
|
{
|
|
|
|
rdp.flags |= CULL_BACK;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
//Added by Gonetz
|
|
|
|
if (rdp.cmd1 & 0x00010000) // Fog enable
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!(rdp.flags & FOG_ENABLED))
|
|
|
|
{
|
|
|
|
rdp.flags |= FOG_ENABLED;
|
|
|
|
rdp.update |= UPDATE_FOG_ENABLED;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_cleargeometrymode()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:cleargeometrymode %08lx", rdp.cmd1);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.geom_mode &= (~rdp.cmd1);
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00000001) // Z-Buffer enable
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.flags & ZBUF_ENABLED)
|
|
|
|
{
|
|
|
|
rdp.flags ^= ZBUF_ENABLED;
|
|
|
|
rdp.update |= UPDATE_ZBUF_ENABLED;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00001000) // Front culling
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.flags & CULL_FRONT)
|
|
|
|
{
|
|
|
|
rdp.flags ^= CULL_FRONT;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.cmd1 & 0x00002000) // Back culling
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.flags & CULL_BACK)
|
|
|
|
{
|
|
|
|
rdp.flags ^= CULL_BACK;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2016-01-25 11:14:01 +00:00
|
|
|
//Added by Gonetz
|
|
|
|
if (rdp.cmd1 & 0x00010000) // Fog enable
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (rdp.flags & FOG_ENABLED)
|
|
|
|
{
|
|
|
|
rdp.flags ^= FOG_ENABLED;
|
|
|
|
rdp.update |= UPDATE_FOG_ENABLED;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_line3d()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
uint32_t v0 = ((rdp.cmd1 >> 16) & 0xff) / 10;
|
|
|
|
uint32_t v1 = ((rdp.cmd1 >> 8) & 0xff) / 10;
|
|
|
|
uint16_t width = (uint16_t)(rdp.cmd1 & 0xFF) + 3;
|
|
|
|
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX *vtx[3] = {
|
2017-05-19 20:03:52 +00:00
|
|
|
&rdp.vtx(v1),
|
|
|
|
&rdp.vtx(v0),
|
|
|
|
&rdp.vtx(v0)
|
2016-01-25 11:14:01 +00:00
|
|
|
};
|
|
|
|
uint32_t cull_mode = (rdp.flags & CULLMASK) >> CULLSHIFT;
|
|
|
|
rdp.flags |= CULLMASK;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
2017-05-19 20:03:52 +00:00
|
|
|
rsp_tri1(vtx, width);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.flags ^= CULLMASK;
|
|
|
|
rdp.flags |= cull_mode << CULLSHIFT;
|
|
|
|
rdp.update |= UPDATE_CULL_MODE;
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:line3d v0:%d, v1:%d, width:%d", v0, v1, width);
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-16 07:41:05 +00:00
|
|
|
void uc0_tri4()
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
// c0: 0000 0123, c1: 456789ab
|
|
|
|
// becomes: 405 617 829 a3b
|
|
|
|
|
2016-02-11 09:58:42 +00:00
|
|
|
WriteTrace(TraceRDP, TraceDebug, "uc0:tri4");
|
|
|
|
WriteTrace(TraceRDP, TraceDebug, " #%d, #%d, #%d, #%d - %d, %d, %d - %d, %d, %d - %d, %d, %d - %d, %d, %d", rdp.tri_n, rdp.tri_n + 1, rdp.tri_n + 2, rdp.tri_n + 3,
|
2016-01-25 11:14:01 +00:00
|
|
|
(rdp.cmd1 >> 28) & 0xF,
|
|
|
|
(rdp.cmd0 >> 12) & 0xF,
|
|
|
|
(rdp.cmd1 >> 24) & 0xF,
|
|
|
|
(rdp.cmd1 >> 20) & 0xF,
|
|
|
|
(rdp.cmd0 >> 8) & 0xF,
|
|
|
|
(rdp.cmd1 >> 16) & 0xF,
|
|
|
|
(rdp.cmd1 >> 12) & 0xF,
|
|
|
|
(rdp.cmd0 >> 4) & 0xF,
|
|
|
|
(rdp.cmd1 >> 8) & 0xF,
|
|
|
|
(rdp.cmd1 >> 4) & 0xF,
|
|
|
|
(rdp.cmd0 >> 0) & 0xF,
|
|
|
|
(rdp.cmd1 >> 0) & 0xF);
|
|
|
|
|
2017-08-16 22:18:19 +00:00
|
|
|
gfxVERTEX *vtx[12] = {
|
2017-05-19 20:03:52 +00:00
|
|
|
&rdp.vtx((rdp.cmd1 >> 28) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd0 >> 12) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 24) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 20) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd0 >> 8) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 16) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 12) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd0 >> 4) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 8) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 4) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd0 >> 0) & 0xF),
|
|
|
|
&rdp.vtx((rdp.cmd1 >> 0) & 0xF),
|
2016-01-25 11:14:01 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
int updated = 0;
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
if (cull_tri(vtx))
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
updated = 1;
|
|
|
|
update();
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
draw_tri(vtx);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
}
|
2013-04-04 21:22:19 +00:00
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
if (cull_tri(vtx + 3))
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!updated)
|
|
|
|
{
|
|
|
|
updated = 1;
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
draw_tri(vtx + 3);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
if (cull_tri(vtx + 6))
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
2013-04-04 21:22:19 +00:00
|
|
|
{
|
2016-01-25 11:14:01 +00:00
|
|
|
if (!updated)
|
|
|
|
{
|
|
|
|
updated = 1;
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
draw_tri(vtx + 6);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
2013-04-04 21:22:19 +00:00
|
|
|
}
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
if (cull_tri(vtx + 9))
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!updated)
|
|
|
|
{
|
|
|
|
updated = 1;
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
|
2017-05-19 20:03:52 +00:00
|
|
|
draw_tri(vtx + 9);
|
2016-01-25 11:14:01 +00:00
|
|
|
rdp.tri_n++;
|
|
|
|
}
|
2017-05-16 07:41:05 +00:00
|
|
|
}
|