SegaRaycast/src/main.c

306 lines
8.2 KiB
C
Raw Normal View History

2024-08-29 17:51:49 -06:00
#include <genesis.h>
2024-08-31 09:00:42 -06:00
#include "sinTable.h"
#include "cosTable.h"
#define TABLE_LENGTH 720
2024-08-30 15:32:58 -06:00
#define SPEED 5
2024-08-31 09:00:42 -06:00
#define FRAC_BITS 8
2024-08-31 20:10:10 -06:00
#define TILE_WIDTH 8
#define TILE_HEIGHT 8
#define TILE_SIZE (TILE_WIDTH * TILE_HEIGHT / 2)
#define SCREEN_WIDTH 320
#define SCREEN_HEIGHT 224
#define SCREEN_TILES_X 40
#define SCREEN_TILES_Y 28
#define TILE_EMPTY 0
#define TILE_FILLED 1
#define MAP_WIDTH 10
#define MAP_HEIGHT 10
2024-09-01 07:59:33 -06:00
#define WALL_HEIGHT_SCALE FIX16(1)
2024-08-29 18:49:38 -06:00
2024-08-29 18:17:52 -06:00
Line l;
2024-08-30 09:09:31 -06:00
s16 angle = 0;
2024-08-30 15:54:02 -06:00
s16 x, y;
2024-08-31 20:10:10 -06:00
u8 tileBuffer[TILE_SIZE];
u16 tileIndex = 0;
2024-08-29 18:17:52 -06:00
2024-08-31 09:00:42 -06:00
2024-08-29 19:39:58 -06:00
const u8 map[10][10] = {{1,1,1,1,1,1,1,1,1,1},
{1,0,0,0,0,0,0,0,0,1},
{1,1,1,1,0,0,1,1,1,1},
{1,0,0,1,0,0,1,0,0,1},
{1,0,0,0,0,0,1,0,0,1},
{1,0,0,1,0,0,1,1,0,1},
{1,1,1,1,0,0,1,0,0,1},
{1,0,0,0,0,0,0,0,0,1},
{1,0,0,0,0,0,1,0,0,1},
{1,1,1,1,1,1,1,1,1,1}};
2024-08-30 12:12:20 -06:00
int dist(int ax, int ay, int bx, int by){
return (ax-bx)*(ax-bx) + (ay-by)*(ay-by);
}
2024-09-01 03:56:59 -06:00
void castRay(s16 angle, s16 column){
2024-08-30 15:32:58 -06:00
if(angle >= 360) angle = angle - 360;
if(angle < 0) angle = 360 + angle;
2024-08-30 12:12:20 -06:00
u16 r,mx,my,mp,dof;
2024-08-31 22:11:01 -06:00
fix16 rx,ry,ra,xo,yo;
s16 ind = (int)((float)angle/360.0f*1024.0f);
fix16 dy = sinFix16(ind);
fix16 dx = cosFix16(ind);
fix16 tan = FIX16(0);
if(dx != 0) tan = fix16Div(dy, dx);
fix16 aTan = FIX16(0);
if(tan != 0) aTan = fix16Div(FIX16(-1), tan);
s16 tanInt = fix16ToInt(fix16Mul(aTan, FIX16(100)));
2024-08-30 09:09:31 -06:00
dof = 0;
2024-08-31 22:11:01 -06:00
fix16 distH = FIX16(500);
fix16 hx = FIX16(l.pt1.x);
fix16 hy = FIX16(l.pt1.y);
2024-08-30 14:04:02 -06:00
if(dy == 0){
2024-08-31 22:11:01 -06:00
rx = FIX16(l.pt1.x);
ry = FIX16(l.pt1.y);
2024-08-30 12:12:20 -06:00
dof = 8;
}else {
if(angle > 180) {
2024-08-31 22:11:01 -06:00
ry = FIX16(l.pt1.y / 10 * 10)-FIX16(1);
rx = fix16Mul(FIX16(l.pt1.y) - ry,aTan) + FIX16(l.pt1.x);
yo = FIX16(-10);
xo = fix16Mul(0-yo,aTan);
2024-08-30 12:12:20 -06:00
}
if(angle < 180) {
2024-08-31 22:11:01 -06:00
ry = FIX16((l.pt1.y / 10 * 10)+10);
rx = fix16Mul(FIX16(l.pt1.y) - ry,aTan) + FIX16(l.pt1.x);
yo = FIX16(10);
xo = fix16Mul(0-yo,aTan);
2024-08-30 12:12:20 -06:00
}
2024-08-30 09:09:31 -06:00
}
2024-08-30 12:12:20 -06:00
while(dof<8){
2024-08-31 22:11:01 -06:00
mx = fix16ToInt(rx) / 10;
my = fix16ToInt(ry) / 10;
2024-08-30 14:04:02 -06:00
if(rx < 0 || ry < 0){
dof = 8;
continue;
}
if(mx < 10 && my < 10 && map[my][mx] == 1){
2024-08-30 12:12:20 -06:00
dof = 8;
hx = rx;
hy = ry;
2024-08-31 22:11:01 -06:00
distH = dist(l.pt1.x, l.pt1.y, fix16ToInt(rx), fix16ToInt(ry));
2024-08-30 12:12:20 -06:00
}else{
rx = rx + xo;
ry = ry + yo;
dof += 1;
}
2024-08-30 09:09:31 -06:00
}
2024-08-30 12:12:20 -06:00
dof = 0;
2024-08-31 22:11:01 -06:00
fix16 distV = FIX16(500);
fix16 vx = FIX16(l.pt1.x);
fix16 vy = FIX16(l.pt1.y);
fix16 nTan = 0-tan;
if(fix16ToInt(nTan) > 51){
nTan = FIX16(51);
}
2024-08-31 22:11:01 -06:00
if(fix16ToInt(nTan) < -51){
nTan = FIX16(-51);
}
2024-08-31 22:11:01 -06:00
if(angle > 90 && r < 270) {
rx = FIX16(l.pt1.x / 10 * 10)-FIX16(1);
ry = fix16Mul(FIX16(l.pt1.x) - rx,nTan) + FIX16(l.pt1.y);
xo = FIX16(-10);
yo = fix16Mul(0-xo,nTan);
2024-08-30 12:12:20 -06:00
}
2024-08-31 22:11:01 -06:00
if(angle < 90 || angle > 270) {
rx = FIX16((l.pt1.x / 10 * 10)+10);
ry = fix16Mul(FIX16(l.pt1.x) - rx,nTan) + FIX16(l.pt1.y);
xo = FIX16(10);
yo = fix16Mul(0-xo,nTan);
2024-08-30 12:12:20 -06:00
}
2024-08-30 14:04:02 -06:00
if(dx == 0){
2024-08-31 22:11:01 -06:00
rx = FIX16(l.pt1.x);
ry = FIX16(l.pt1.y);
xo = FIX16(0);
yo = FIX16(0);
2024-08-30 09:09:31 -06:00
dof = 8;
}
while(dof<8){
2024-08-31 22:11:01 -06:00
mx = fix16ToInt(rx) / 10;
my = fix16ToInt(ry) / 10;
2024-08-30 21:09:42 -06:00
if(rx < 0 || ry < 0){
dof = 8;
continue;
}
2024-08-30 16:58:36 -06:00
if(mx < 10 && my < 10 && map[my][mx] == 1){
2024-08-30 12:12:20 -06:00
dof = 8;
vx = rx;
vy = ry;
2024-08-31 22:11:01 -06:00
distV = dist(l.pt1.x, l.pt1.y, fix16ToInt(rx), fix16ToInt(ry));
2024-08-30 12:12:20 -06:00
}else{
rx = rx + xo;
ry = ry + yo;
dof += 1;
}
}
2024-09-01 03:56:59 -06:00
fix16 shortestDist = distH;
if(distV < distH){
2024-09-01 03:56:59 -06:00
shortestDist = distV;
2024-08-31 22:11:01 -06:00
l.pt2.x = fix16ToInt(vx);
l.pt2.y = fix16ToInt(vy);
}else {
2024-08-31 22:11:01 -06:00
l.pt2.x = fix16ToInt(hx);
l.pt2.y = fix16ToInt(hy);
}
2024-08-30 22:53:22 -06:00
2024-09-01 03:56:59 -06:00
2024-09-01 07:56:27 -06:00
//Wall height calcs
fix16 wallHeightFix16 = fix16Div(FIX16(SCREEN_HEIGHT), (shortestDist + FIX16(1)));
wallHeightFix16 = fix16Mul(wallHeightFix16, WALL_HEIGHT_SCALE);
int wallHeight = fix16ToInt(wallHeightFix16);
int wallTop = (SCREEN_HEIGHT / 2) - (wallHeight / 2);
int wallBottom = (SCREEN_HEIGHT / 2) + (wallHeight / 2);
if (wallTop < 0) wallTop = 0;
if (wallBottom >= SCREEN_HEIGHT) wallBottom = SCREEN_HEIGHT - 1;
u16 tileTop = wallTop / TILE_HEIGHT;
u16 tileBottom = wallBottom / TILE_HEIGHT;
u16 tileHeight = tileBottom - tileTop + 1;
for (int yTile = 0; yTile <= SCREEN_TILES_Y; yTile++) {
if(yTile >= tileTop && yTile <= tileBottom){
VDP_setTileMapXY(BG_B, TILE_ATTR_FULL(PAL1, 0, FALSE, FALSE, TILE_FILLED), column, yTile);
}else{
VDP_setTileMapXY(BG_B, TILE_ATTR_FULL(PAL1, 0, FALSE, FALSE, TILE_EMPTY), column, yTile);
2024-09-01 03:56:59 -06:00
}
2024-09-01 07:56:27 -06:00
}
2024-08-29 21:04:02 -06:00
}
2024-08-29 19:39:58 -06:00
2024-08-31 20:10:10 -06:00
void initTileBuffer(u8 color) {
u8 colorValue = (color << 4) | color; // Color for both nibbles
for (int i = 0; i < TILE_SIZE; i++) {
tileBuffer[i] = colorValue;
}
2024-08-29 19:39:58 -06:00
}
2024-08-29 18:49:38 -06:00
2024-08-31 20:10:10 -06:00
void uploadTileDataToVRAM(u16 vramTileIndex) {
VDP_loadTileData(tileBuffer, vramTileIndex, 1, DMA); // Load 1 tile to VRAM
}
2024-08-29 18:49:38 -06:00
2024-08-31 20:10:10 -06:00
void mapscan() {
2024-08-31 22:11:01 -06:00
u8 mapcolor = 15; // map color
2024-08-31 20:10:10 -06:00
mapcolor |= mapcolor << 4;
2024-08-29 18:17:52 -06:00
2024-08-31 20:10:10 -06:00
for (u8 x = 0; x < 10; x++) {
for (u8 y = 0; y < 10; y++) {
if (map[y][x] == 1) {
// Draw a filled tile for the map block
for (u8 px = 0; px < TILE_WIDTH; px++) {
for (u8 py = 0; py < TILE_HEIGHT; py++) {
setPixelInBuffer(px, py, mapcolor);
}
}
// Upload the filled tile to VRAM
uploadTileDataToVRAM(tileIndex);
2024-08-31 22:11:01 -06:00
VDP_setTileMapXY(BG_A, tileIndex, x, y);
2024-08-31 20:10:10 -06:00
}
}
}
}
2024-08-29 18:17:52 -06:00
2024-08-31 20:10:10 -06:00
void clearScreenWithTile(u16 tileIndex) {
// Fill the entire screen with the background tile
VDP_fillTileMapRect(BG_A, TILE_ATTR_FULL(PAL1, 0, FALSE, FALSE, tileIndex), 0, 0, SCREEN_TILES_X, SCREEN_TILES_Y);
2024-09-01 07:56:27 -06:00
//VDP_fillTileMapRect(BG_B, TILE_ATTR_FULL(PAL1, 0, FALSE, FALSE, tileIndex), 0, 0, SCREEN_TILES_X, SCREEN_TILES_Y);
2024-08-31 20:10:10 -06:00
}
2024-08-29 18:49:38 -06:00
2024-08-31 20:10:10 -06:00
void drawMap() {
for (u16 y = 0; y < MAP_HEIGHT; y++) {
for (u16 x = 0; x < MAP_WIDTH; x++) {
u8 tileType = map[y][x] ? TILE_FILLED : TILE_EMPTY;
VDP_setTileMapXY(BG_A, TILE_ATTR_FULL(PAL1, 0, FALSE, FALSE, tileType), x, y);
}
}
2024-08-29 18:17:52 -06:00
}
2024-08-31 20:10:10 -06:00
void render() {
2024-09-01 07:58:22 -06:00
//clearScreenWithTile(TILE_EMPTY);
2024-09-01 07:56:27 -06:00
s16 castStart = angle - 20;
if(angle >= 360) angle = angle - 360;
2024-09-01 03:56:59 -06:00
if(angle < 0) angle = 360 + angle;
2024-09-01 07:56:27 -06:00
if(castStart >= 360) castStart = castStart - 360;
if(castStart < 0) castStart = 360 + castStart;
for(int i = 0; i < 40; i++){
s16 cast = castStart+i;
if(cast >= 360) cast = cast - 360;
if(cast < 0) cast = 360 + cast;
castRay(cast, i);
//VDP_waitVSync();
}
//drawMap();
2024-09-01 07:58:22 -06:00
VDP_setTextPlane(BG_A);
2024-09-01 07:56:27 -06:00
VDP_setTextPalette(2);
VDP_showFPS(1);
char debugCastText[20];
sprintf(debugCastText, "Cast: %03d", castStart);
VDP_drawText(debugCastText, 20, 2);
angle = angle + 1;
SYS_doVBlankProcess();
2024-08-29 18:49:38 -06:00
}
2024-08-29 18:17:52 -06:00
2024-08-31 20:10:10 -06:00
void update() {
u16 joy = JOY_readJoypad(JOY_1);
if (joy & BUTTON_LEFT) {
angle -= 1;
}
if (joy & BUTTON_RIGHT) {
angle += 1;
}
if (joy & BUTTON_UP) {
s16 ind = (int)((float)angle / 360.0f * 1024.0f);
fix16 dy = sinFix16(ind);
fix16 dx = cosFix16(ind);
y += fix16ToInt(fix16Mul(dy, FIX16(SPEED)));
x += fix16ToInt(fix16Mul(dx, FIX16(SPEED)));
}
if (joy & BUTTON_DOWN) {
s16 ind = (int)((float)angle / 360.0f * 1024.0f);
fix16 dy = sinFix16(ind);
fix16 dx = cosFix16(ind);
y -= fix16ToInt(fix16Mul(dy, FIX16(SPEED)));
x -= fix16ToInt(fix16Mul(dx, FIX16(SPEED)));
}
2024-08-29 18:17:52 -06:00
}
2024-08-31 20:10:10 -06:00
void initVDP() {
SYS_disableInts();
VDP_setPlaneSize(64, 32, TRUE);
SYS_enableInts();
}
2024-08-29 17:51:49 -06:00
2024-08-31 20:10:10 -06:00
int main() {
SYS_disableInts();
VDP_setPlaneSize(64, 32, TRUE);
SYS_enableInts();
2024-08-31 22:11:01 -06:00
PAL_setColor(14, RGB24_TO_VDPCOLOR(0x1c1c14));
PAL_setColor(15, RGB24_TO_VDPCOLOR(0x17ffe8));
2024-08-31 20:10:10 -06:00
VDP_setBackgroundColor(14);
2024-08-29 17:51:49 -06:00
2024-08-31 20:10:10 -06:00
// Initialize and upload tile data for TILE_EMPTY and TILE_FILLED
2024-08-31 22:11:01 -06:00
initTileBuffer(15); // Color for TILE_FILLED
2024-08-31 20:10:10 -06:00
VDP_loadTileData(tileBuffer, TILE_FILLED, 1, DMA);
initTileBuffer(0); // Color for TILE_EMPTY
VDP_loadTileData(tileBuffer, TILE_EMPTY, 1, DMA);
2024-08-29 17:51:49 -06:00
2024-09-01 08:09:50 -06:00
l.pt1.x = 15;
l.pt1.y = 15;
l.pt2.x = 0;
l.pt2.y = 0;
2024-08-31 20:10:10 -06:00
while (TRUE) {
render();
}
2024-08-29 17:51:49 -06:00
2024-08-31 20:10:10 -06:00
return 0;
2024-08-29 17:51:49 -06:00
}