Licar/map.h

916 lines
28 KiB
C
Raw Normal View History

2023-07-21 21:17:49 +02:00
#ifndef _LCR_MAP
#define _LCR_MAP
/**
The map (track) module for Licar.
2023-07-23 16:51:09 +02:00
Map coordinates/size:
- map size is 64x64x64 blocks
- [0,0,0] is is bottom-left-front-most
- x goes right, y goes up, z goes forward
- coordinate number is a single number obtained as x + 64 * y + 64 * 64 * z
2024-11-27 23:44:00 +01:00
The TEXT format serves for editing maps in human readable format, it more or
less corresponds to the binary storage format (below) with some exceptions.
It has the following structure:
2024-12-27 09:06:21 +01:00
- Number specifying environment (0, 1, 2, ...)
- A series of block strings. Blocks may be separated by characters that
aren't ':' (comments may be added this way). Block format
2024-11-27 23:44:00 +01:00
is following:
2024-11-27 20:30:54 +01:00
2024-12-27 09:06:21 +01:00
:BXYZMT
2024-11-27 20:30:54 +01:00
where:
2024-12-02 22:47:22 +01:00
- B is block type
2024-11-27 20:30:54 +01:00
- X, Y and Z are block coordinates, each one a single character. The
following are characters signifying numbers 0 to 63:
0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ$@
2024-11-27 23:44:00 +01:00
- M is block material ('0' to '3').
- T is an optinal transform string (for more detail see the binary format)
consisting from 0 to 3 characters, which may be:
2024-11-27 20:30:54 +01:00
- '|': flip horizontally
- '-': flip vertically
- 'L': rotate 90 degrees
- 'I': rotate 180 degrees
- 'J': rotate 270 degrees
2024-11-27 23:44:00 +01:00
Here the block order matters, latter blocks rewrite previously placed ones.
The internal BINARY map format is made from the text string. It can't contain
special blocks and blocks are ordered by their coordinate number (for fast
lookup). The format consists of blocks, each of format:
2024-12-02 22:47:22 +01:00
- 1 byte type: says the type of block, is the same value as B in the text
format.
2024-11-27 23:44:00 +01:00
- 3 bytes: A, B, C, such that:
- A, B and lowest 2 bits of C form the block coordinate number (A being
the lowest part etc.)
- bits C2 and C3 say the block material
- highest 4 bits of C (C4, C5, C6, C7) say the block's transform:
- first if C4 is set, the block is flipped in the X direction
- then the block is rotated around vertical axis by 0, 90, 180 or 270
degrees if C5C6 is 00, 01, 10 or 11.
- last if C7 is set, the block is flipped vertically
2023-07-21 21:17:49 +02:00
*/
2025-01-14 13:59:44 +01:00
#include "general.h"
2025-01-14 13:40:22 +01:00
#define LCR_MAP_NAME_MAX_LEN 15 /**< Maximum map name length (without
terminating zero. */
2024-12-27 09:06:21 +01:00
#define LCR_BLOCK_START_CHAR ':'
2023-07-23 16:51:09 +02:00
#define LCR_BLOCK_TRANSFORM_FLIP_H 0x10
#define LCR_BLOCK_TRANSFORM_ROT_90 0x20
#define LCR_BLOCK_TRANSFORM_ROT_180 0x40
#define LCR_BLOCK_TRANSFORM_ROT_270 0x60
#define LCR_BLOCK_TRANSFORM_FLIP_V 0x80
2024-07-24 20:28:57 +02:00
#define LCR_BLOCK_XYZ_TO_COORD(x,y,z) // ???
2023-07-23 16:51:09 +02:00
2024-07-24 23:16:13 +02:00
#define LCR_MAP_BLOCK(t,x,y,z,m,r) t,(uint8_t) (x | (y << 6)), \
(uint8_t) ((y >> 2) | (z << 4)), \
(uint8_t) ((z >> 4) | (m << 2) | (r))
2023-07-23 16:51:09 +02:00
2024-07-24 20:28:57 +02:00
#define LCR_BLOCK_SIZE 4 ///< size of map block, in bytes
2023-07-23 16:51:09 +02:00
#define LCR_BLOCK_MATERIAL_CONCRETE 0x00
2024-08-01 21:41:21 +02:00
#define LCR_BLOCK_MATERIAL_GRASS 0x01
#define LCR_BLOCK_MATERIAL_DIRT 0x02
#define LCR_BLOCK_MATERIAL_ICE 0x03
2023-07-21 21:17:49 +02:00
2023-08-03 21:12:23 +02:00
// normal blocks:
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_FULL '=' ///< completely filled block
#define LCR_BLOCK_BOTTOM '-' ///< filled bottom half
#define LCR_BLOCK_LEFT ';' ///< filled left half
#define LCR_BLOCK_BOTTOM_LEFT ',' ///< filled bottom left quarter
#define LCR_BLOCK_BOTTOM_LEFT_FRONT '.' ///< filled bottom left front eigth
#define LCR_BLOCK_RAMP '^' ///< plain ramp
#define LCR_BLOCK_RAMP_34 '/' ///< plain ramp, 3/4 size
#define LCR_BLOCK_RAMP_12 '<' ///< plain ramp, 1/2 size
#define LCR_BLOCK_RAMP_14 '_' ///< plain ramp, 1/4 size
2024-12-11 22:53:52 +01:00
#define LCR_BLOCK_RAMP_CORNER 'v' ///< corner of ramp
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_RAMP_CURVED_PLAT ']' ///< curved ramp with top platgform
#define LCR_BLOCK_RAMP_CURVED ')' ///< curv. ramp without top platf.
#define LCR_BLOCK_RAMP_CURVED_WALL '}' ///< curved ramp plus small wall
#define LCR_BLOCK_RAMP_STEEP '|' ///< extremely steep ramp
#define LCR_BLOCK_CORNER 'A' ///< diagonal corner
#define LCR_BLOCK_CORNER_12 '\\' ///< diagonal corner (1/2 wide)
#define LCR_BLOCK_HILL '(' ///< curved "hill"
2024-12-18 13:01:27 +01:00
#define LCR_BLOCK_BUMP '~' ///< small bump on the road
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_FULL_ACCEL '>'
2024-12-18 13:01:27 +01:00
#define LCR_BLOCK_BOTTOM_ACCEL 'z'
#define LCR_BLOCK_RAMP_ACCEL 'y'
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_FULL_FAN 'o'
2024-12-18 13:01:27 +01:00
#define LCR_BLOCK_RAMP_FAN 'V'
2024-12-02 23:05:41 +01:00
2024-12-12 23:17:06 +01:00
#define LCR_BLOCK_CORNER_CONVEX 'n'
#define LCR_BLOCK_CORNER_CONCAVE 'l'
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_CHECKPOINT_0 '+' ///< checkpoint, not taken
#define LCR_BLOCK_CHECKPOINT_1 '\'' ///< checkpoint, taken
#define LCR_BLOCK_FINISH '!' ///< finish
2024-11-27 20:30:54 +01:00
// special blocks:
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_NONE 'x' ///< no block, e.g to make holes
2024-11-27 20:30:54 +01:00
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_CUBOID_FILL 'f' /**< makes a cuboid from the
2024-11-27 20:30:54 +01:00
previously specified block, the
size is given by block coords */
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_CUBOID_HOLLOW 'h' /**< same as cuboid special block,
2024-11-27 20:30:54 +01:00
but makes a hollow one */
2024-12-02 22:47:22 +01:00
#define LCR_BLOCK_START '*' ///< specifies start block position
2024-09-23 23:31:30 +02:00
2024-10-03 00:24:28 +02:00
/*
TODO:
- ramp corner???
- curved corner?
- curved out corner?
- curved "hill"
- bumpy road
2024-12-02 22:47:22 +01:00
- bigger structures like a loop, sloped road etc?
2024-10-03 00:24:28 +02:00
*/
2024-11-21 00:16:00 +01:00
#define LCR_MAP_BLOCK_CACHE_SIZE (8 * 2) /// do not change
/**
Cache for accelerating LCR_mapGetBlockAtFast, consists of 8 2-item records,
the first record item stores block coord number, the second one stores the
value returned by LCR_mapGetBlockAtFast (-1 is 0xffffffff). The record index
depends on the block coords: lowest bit is x % 2, middle bit y % 2, highest
one z % 2.
*/
uint32_t _LCR_mapBlockCache[LCR_MAP_BLOCK_CACHE_SIZE];
2023-08-03 21:12:23 +02:00
struct
2023-07-21 21:17:49 +02:00
{
2023-08-03 21:12:23 +02:00
uint16_t blockCount;
2024-09-23 23:31:30 +02:00
uint8_t blocks[LCR_SETTING_MAP_MAX_BLOCKS * LCR_BLOCK_SIZE];
2024-09-29 20:52:52 +02:00
uint8_t startPos[4]; ///< Initial position and rotation.
2023-08-03 21:12:23 +02:00
2023-08-08 16:17:51 +02:00
uint8_t environment;
2024-11-24 21:21:29 +01:00
uint8_t checkpointCount;
2025-01-09 15:34:14 +01:00
uint32_t hash; ///< Hash of the processed binary map.
2025-01-14 13:40:22 +01:00
char name[LCR_MAP_NAME_MAX_LEN + 1];
2023-08-03 21:12:23 +02:00
} LCR_currentMap;
2023-07-21 21:17:49 +02:00
2024-08-02 00:05:03 +02:00
void LCR_makeMapBlock(uint8_t type, uint8_t x, uint8_t y, uint8_t z,
uint8_t material, uint8_t transform, uint8_t block[LCR_BLOCK_SIZE])
{
block[0] = type;
block[1] = x | (y << 6);
block[2] = (y >> 2) | (z << 4);
block[3] = (z >> 4) | (material << 2) | transform;
}
2024-07-24 20:28:57 +02:00
void LCR_mapBlockGetCoords(const uint8_t block[LCR_BLOCK_SIZE],
uint8_t *x, uint8_t *y, uint8_t *z)
2023-08-03 21:12:23 +02:00
{
*x = block[1] & 0x3f;
*y = (block[1] >> 6) | ((block[2] & 0x0f) << 2);
*z = (block[2] >> 4) | ((block[3] & 0x03) << 4);
}
2024-09-26 14:56:39 +02:00
uint8_t LCR_mapBlockOppositeTransform(uint8_t transform)
{
if (!(transform & LCR_BLOCK_TRANSFORM_FLIP_H))
{
if ((transform & 0x60) == LCR_BLOCK_TRANSFORM_ROT_90)
return ((transform & (~0x60)) | LCR_BLOCK_TRANSFORM_ROT_270);
if ((transform & 0x60) == LCR_BLOCK_TRANSFORM_ROT_270)
return ((transform & (~0x60)) | LCR_BLOCK_TRANSFORM_ROT_90);
}
return transform;
}
2024-12-18 13:01:27 +01:00
uint8_t LCR_mapBlockIsAccelerator(uint8_t block)
{
return block == LCR_BLOCK_FULL_ACCEL || block == LCR_BLOCK_RAMP_ACCEL ||
block == LCR_BLOCK_BOTTOM_ACCEL;
}
uint8_t LCR_mapBlockIsFan(uint8_t block)
{
return block == LCR_BLOCK_FULL_FAN || block == LCR_BLOCK_RAMP_FAN;
}
2024-07-24 23:16:13 +02:00
uint8_t LCR_mapBlockGetTransform(const uint8_t block[LCR_BLOCK_SIZE])
{
return block[3] & 0xf0;
}
2024-08-01 21:41:21 +02:00
uint8_t LCR_mapBlockGetMaterial(const uint8_t block[LCR_BLOCK_SIZE])
{
2024-12-18 13:01:27 +01:00
return (LCR_mapBlockIsAccelerator(block[0]) ||
LCR_mapBlockIsFan(block[0])) ? LCR_BLOCK_MATERIAL_CONCRETE :
((block[3] >> 2) & 0x03);
2024-08-01 21:41:21 +02:00
}
2024-07-24 20:28:57 +02:00
uint32_t LCR_mapBlockGetCoordNumber(const uint8_t block[LCR_BLOCK_SIZE])
2023-08-03 21:12:23 +02:00
{
return block[1] | (((uint32_t) block[2]) << 8) |
((((uint32_t) block[3]) & 0x3) << 16);
}
2024-09-23 20:21:08 +02:00
uint32_t LCR_mapBlockCoordsToCoordNumber(uint8_t x, uint8_t y, uint8_t z)
{
uint8_t b[LCR_BLOCK_SIZE];
LCR_makeMapBlock(0,x,y,z,0,0,b);
return LCR_mapBlockGetCoordNumber(b);
}
2024-10-03 00:24:28 +02:00
/**
Gets dimensions of a non-curved ramp.
*/
void LCR_rampGetDimensions(uint8_t rampType, uint8_t *height4ths,
2024-10-01 22:08:03 +02:00
uint8_t *length6ths)
{
2024-12-18 20:45:35 +01:00
*height4ths = (rampType == LCR_BLOCK_RAMP_14) +
2024-12-18 13:01:27 +01:00
(rampType == LCR_BLOCK_RAMP || rampType == LCR_BLOCK_RAMP_ACCEL ||
rampType == LCR_BLOCK_RAMP_FAN || rampType == LCR_BLOCK_RAMP_STEEP) * 4 +
2024-10-01 22:08:03 +02:00
(rampType == LCR_BLOCK_RAMP_12 || rampType == LCR_BLOCK_RAMP_34) * 2 +
(rampType == LCR_BLOCK_RAMP_34);
*length6ths = rampType != LCR_BLOCK_RAMP_STEEP ? 6 : 1;
}
2023-08-08 16:17:51 +02:00
uint8_t *LCR_getMapBlockAtCoordNumber(uint32_t coord)
{
// binary search the block:
uint16_t a = 0, b = LCR_currentMap.blockCount - 1;
while (b >= a)
{
uint16_t mid = (a + b) / 2;
2024-07-24 20:28:57 +02:00
uint8_t *block = LCR_currentMap.blocks + mid * LCR_BLOCK_SIZE;
2024-12-18 20:45:35 +01:00
uint32_t coord2 = LCR_mapBlockGetCoordNumber(block);
2023-08-08 16:17:51 +02:00
if (coord2 == coord)
return block;
else if (coord2 > coord)
b = mid - 1;
else
a = mid + 1;
}
return 0;
}
2023-08-03 21:12:23 +02:00
/**
Adds given block to current map, including possibly deleting a block by
adding LCR_BLOCK_NONE. The function handles sorting the block to the right
2024-11-27 20:30:54 +01:00
position. Returns pointer to the block on success, else 0.
2023-08-03 21:12:23 +02:00
*/
2024-12-16 20:17:58 +01:00
void _LCR_mapAddBlock(const uint8_t block[LCR_BLOCK_SIZE])
2023-08-03 21:12:23 +02:00
{
2024-09-29 20:52:52 +02:00
LCR_LOG2("adding map block");
2024-09-23 23:31:30 +02:00
if (LCR_currentMap.blockCount >= LCR_SETTING_MAP_MAX_BLOCKS)
2024-11-27 20:30:54 +01:00
{
LCR_LOG0("couldn't add block");
2024-12-16 20:17:58 +01:00
return;
2024-11-27 20:30:54 +01:00
}
2023-08-03 21:12:23 +02:00
uint32_t coord = LCR_mapBlockGetCoordNumber(block);
uint16_t insertAt = 0;
while (insertAt < LCR_currentMap.blockCount &&
2024-07-24 20:28:57 +02:00
coord > LCR_mapBlockGetCoordNumber(LCR_currentMap.blocks +
insertAt * LCR_BLOCK_SIZE))
2023-08-03 21:12:23 +02:00
insertAt++;
if (block[0] == LCR_BLOCK_NONE)
{
if (insertAt < LCR_currentMap.blockCount &&
2024-07-24 20:28:57 +02:00
coord == LCR_mapBlockGetCoordNumber(LCR_currentMap.blocks +
insertAt * LCR_BLOCK_SIZE))
2023-08-03 21:12:23 +02:00
{
// shift all left (remove the block):
2024-12-16 20:17:58 +01:00
for (int i = insertAt * LCR_BLOCK_SIZE;
i < (LCR_currentMap.blockCount - 1) * LCR_BLOCK_SIZE; ++i)
LCR_currentMap.blocks[i] = LCR_currentMap.blocks[i + LCR_BLOCK_SIZE];
2023-08-03 21:12:23 +02:00
LCR_currentMap.blockCount--;
}
2024-12-16 20:17:58 +01:00
return;
2023-08-03 21:12:23 +02:00
}
if (insertAt == LCR_currentMap.blockCount ||
2024-07-24 20:28:57 +02:00
coord != LCR_mapBlockGetCoordNumber(LCR_currentMap.blocks +
insertAt * LCR_BLOCK_SIZE))
2023-08-03 21:12:23 +02:00
{
// shift from here to the right, make room for the new block
LCR_currentMap.blockCount++;
2024-07-24 20:28:57 +02:00
for (int16_t i = ((int16_t) LCR_currentMap.blockCount) - 1;
i > insertAt; i--)
for (uint8_t j = 0; j < LCR_BLOCK_SIZE; ++j)
LCR_currentMap.blocks[i * LCR_BLOCK_SIZE + j] =
LCR_currentMap.blocks[(i - 1) * LCR_BLOCK_SIZE + j];
2023-08-03 21:12:23 +02:00
}
2024-07-24 20:28:57 +02:00
insertAt *= LCR_BLOCK_SIZE;
2023-08-03 21:12:23 +02:00
2024-07-24 20:28:57 +02:00
for (uint8_t j = 0; j < LCR_BLOCK_SIZE; ++j)
2023-08-03 21:12:23 +02:00
LCR_currentMap.blocks[insertAt + j] = block[j];
}
2024-11-24 21:21:29 +01:00
/**
Resets the map to start a run (including e.g. unmarking checkpoints etc.).
*/
void LCR_mapReset(void)
{
LCR_LOG0("resetting map");
for (int i = 0; i < LCR_currentMap.blockCount; ++i)
if (LCR_currentMap.blocks[i * LCR_BLOCK_SIZE] == LCR_BLOCK_CHECKPOINT_1)
LCR_currentMap.blocks[i * LCR_BLOCK_SIZE] = LCR_BLOCK_CHECKPOINT_0;
}
2024-11-27 20:30:54 +01:00
int _LCR_mapCharToCoord(char c)
{
if (c >= '0' && c <= '9')
return c - '0';
if (c >= 'a' && c <= 'z')
return c - 'a' + 10;
if (c >= 'A' && c <= 'Z')
return c - 'A' + 36;
if (c == '@')
return 62;
if (c == '&')
return 63;
return -1;
}
2025-01-09 15:34:14 +01:00
void _LCR_mapComputeHash(void)
{
LCR_LOG1("computing map hash");
const uint8_t *data = LCR_currentMap.startPos;
LCR_currentMap.hash = 11 + LCR_currentMap.environment;
for (int i = 0; i < LCR_currentMap.blockCount * LCR_BLOCK_SIZE + 4; ++i)
{
LCR_currentMap.hash = LCR_currentMap.hash * 101 + *data;
data = i != 3 ? data + 1 : LCR_currentMap.blocks;
}
LCR_currentMap.hash *= 251;
LCR_currentMap.hash = ((LCR_currentMap.hash << 19) |
(LCR_currentMap.hash >> 13)) * 113;
}
2025-01-14 13:40:22 +01:00
uint8_t LCR_mapLoadFromStr(char (*getNextCharFunc)(void), const char *name)
2024-11-27 20:30:54 +01:00
{
LCR_LOG0("loading map string");
2025-01-14 13:40:22 +01:00
for (int i = 0; i < LCR_MAP_NAME_MAX_LEN; ++i)
{
LCR_currentMap.name[i] = *name;
LCR_currentMap.name[i + 1] = 0;
if (*name == 0)
break;
name++;
}
2024-12-27 09:06:21 +01:00
char c;
2024-12-16 20:17:58 +01:00
uint8_t prevBlock[LCR_BLOCK_SIZE];
prevBlock[0] = LCR_BLOCK_NONE;
2024-11-27 20:30:54 +01:00
for (int i = 0; i < 4; ++i)
LCR_currentMap.startPos[i] = 0;
LCR_currentMap.checkpointCount = 0;
LCR_currentMap.blockCount = 0;
LCR_currentMap.environment = 0;
2025-01-09 15:34:14 +01:00
LCR_currentMap.hash = 0;
2024-11-27 20:30:54 +01:00
2024-12-27 09:06:21 +01:00
c = getNextCharFunc();
2024-11-27 20:30:54 +01:00
2024-12-27 09:06:21 +01:00
if (c < '0' || c > '3')
2024-11-27 20:30:54 +01:00
{
2024-12-27 09:06:21 +01:00
LCR_LOG0("bad environment char");
return 0;
2024-11-27 20:30:54 +01:00
}
2024-12-27 09:06:21 +01:00
LCR_currentMap.environment = c - '0';
2024-11-27 20:30:54 +01:00
2024-12-27 09:06:21 +01:00
while (c)
2024-11-27 20:30:54 +01:00
{
2024-12-27 09:06:21 +01:00
if (c == LCR_BLOCK_START_CHAR)
2024-11-27 20:30:54 +01:00
{
2024-12-27 09:06:21 +01:00
uint8_t block = getNextCharFunc();
2024-11-27 20:30:54 +01:00
uint8_t trans = 0;
uint8_t mat = 0;
int coords[3];
for (int i = 0; i < 3; ++i)
{
2024-12-27 09:06:21 +01:00
c = getNextCharFunc();
coords[i] = _LCR_mapCharToCoord(c);
2024-11-27 20:30:54 +01:00
if (coords[i] < 0)
{
LCR_LOG0("bad coord");
return 0;
}
}
2024-12-27 09:06:21 +01:00
c = getNextCharFunc();
2024-11-27 20:30:54 +01:00
2024-12-27 09:06:21 +01:00
if (c < '0' || c > '3')
2024-11-27 20:30:54 +01:00
{
LCR_LOG0("bad material");
return 0;
}
2024-12-27 09:06:21 +01:00
mat = c - '0';
2024-11-27 20:30:54 +01:00
while (1)
{
2024-12-27 09:06:21 +01:00
c = getNextCharFunc();
if (c == '|')
2024-11-27 20:30:54 +01:00
trans |= LCR_BLOCK_TRANSFORM_FLIP_H;
2024-12-27 09:06:21 +01:00
else if (c == '-')
2024-11-27 20:30:54 +01:00
trans |= LCR_BLOCK_TRANSFORM_FLIP_V;
2024-12-27 09:06:21 +01:00
else if (c == 'L')
2024-11-27 20:30:54 +01:00
trans |= LCR_BLOCK_TRANSFORM_ROT_90;
2024-12-27 09:06:21 +01:00
else if (c == 'I')
2024-11-27 20:30:54 +01:00
trans |= LCR_BLOCK_TRANSFORM_ROT_180;
2024-12-27 09:06:21 +01:00
else if (c == 'J')
2024-11-27 20:30:54 +01:00
trans |= LCR_BLOCK_TRANSFORM_ROT_270;
else
break;
}
2024-12-27 09:06:21 +01:00
while (c && c != LCR_BLOCK_START_CHAR)
c = getNextCharFunc();
2024-11-27 20:30:54 +01:00
switch (block)
{
case LCR_BLOCK_CUBOID_FILL:
case LCR_BLOCK_CUBOID_HOLLOW:
{
uint8_t x, y, z, mat, transform;
uint8_t tmpBlock[LCR_BLOCK_SIZE];
mat = LCR_mapBlockGetMaterial(prevBlock);
transform = LCR_mapBlockGetTransform(prevBlock);
LCR_mapBlockGetCoords(prevBlock,&x,&y,&z);
2024-11-29 00:51:43 +01:00
2024-11-27 20:30:54 +01:00
for (uint8_t k = 0; k < coords[2]; ++k)
for (uint8_t j = 0; j < coords[1]; ++j)
for (uint8_t i = 0; i < coords[0]; ++i)
2024-11-29 00:51:43 +01:00
if ((block == LCR_BLOCK_CUBOID_FILL ||
2024-11-27 20:30:54 +01:00
k == 0 || k == coords[2] - 1 ||
j == 0 || j == coords[1] - 1 ||
2024-11-29 00:51:43 +01:00
i == 0 || i == coords[0] - 1) &&
(x + i < LCR_MAP_SIZE_BLOCKS &&
y + j < LCR_MAP_SIZE_BLOCKS &&
z + k < LCR_MAP_SIZE_BLOCKS))
2024-11-27 20:30:54 +01:00
{
LCR_makeMapBlock(prevBlock[0],x + i,y + j,z + k,mat,transform,
tmpBlock);
2024-12-16 20:17:58 +01:00
_LCR_mapAddBlock(tmpBlock);
2024-11-27 20:30:54 +01:00
}
break;
}
case LCR_BLOCK_START:
LCR_currentMap.startPos[0] = coords[0];
LCR_currentMap.startPos[1] = coords[1];
LCR_currentMap.startPos[2] = coords[2];
2024-12-19 21:10:18 +01:00
LCR_currentMap.startPos[3] = trans;
2024-11-27 20:30:54 +01:00
break;
case LCR_BLOCK_CHECKPOINT_0:
LCR_currentMap.checkpointCount++;
// fall through
default: // normal block
{
LCR_makeMapBlock(block,coords[0],coords[1],coords[2],mat,trans,
2024-12-16 20:17:58 +01:00
prevBlock);
2024-11-27 20:30:54 +01:00
2024-12-16 20:17:58 +01:00
_LCR_mapAddBlock(prevBlock);
2024-11-27 20:30:54 +01:00
break;
}
2024-12-02 22:47:22 +01:00
// TODO: check for invalid blocks?
2024-11-27 20:30:54 +01:00
}
}
2024-12-27 09:06:21 +01:00
else
c = getNextCharFunc();
2024-11-27 20:30:54 +01:00
}
LCR_LOG2("clearing map block cache")
for (int i = 0; i < LCR_MAP_BLOCK_CACHE_SIZE; ++i)
_LCR_mapBlockCache[i] = 0xffffffff;
2025-01-09 15:34:14 +01:00
_LCR_mapComputeHash();
2024-11-27 20:30:54 +01:00
LCR_LOG2("map loaded")
LCR_mapReset();
return 1;
}
2023-08-03 21:12:23 +02:00
/**
2024-09-23 20:21:08 +02:00
Same as LCR_mapGetBlockAt, but allows to specify start and end block of the
of the search to make it faster.
2023-08-03 21:12:23 +02:00
*/
2024-09-23 20:21:08 +02:00
int LCR_mapGetBlockAtFast(uint8_t x, uint8_t y, uint8_t z,
int start, int end)
2023-08-03 21:12:23 +02:00
{
2024-09-23 20:21:08 +02:00
// binary search (the blocks are sorted)
uint32_t n = LCR_mapBlockCoordsToCoordNumber(x,y,z);
2024-11-21 00:16:00 +01:00
uint8_t cacheIndex = 2 * ((x % 2) | ((y % 2) << 1) | ((z % 2) << 2));
if (_LCR_mapBlockCache[cacheIndex] == n)
return
(_LCR_mapBlockCache[cacheIndex + 1] != 0xffffffff) ?
((int) _LCR_mapBlockCache[cacheIndex + 1]) : -1;
_LCR_mapBlockCache[cacheIndex] = n;
2024-09-23 20:21:08 +02:00
while (start <= end)
{
int m = (start + end) / 2;
uint32_t n2 = LCR_mapBlockGetCoordNumber(
LCR_currentMap.blocks + m * LCR_BLOCK_SIZE);
if (n2 < n)
start = m + 1;
else if (n2 > n)
end = m - 1;
else
2024-11-21 00:16:00 +01:00
{
_LCR_mapBlockCache[cacheIndex + 1] = m;
2024-09-23 20:21:08 +02:00
return m;
2024-11-21 00:16:00 +01:00
}
2024-09-23 20:21:08 +02:00
}
2024-11-21 00:16:00 +01:00
_LCR_mapBlockCache[cacheIndex + 1] = 0xffffffff;
2024-09-23 20:21:08 +02:00
return -1;
}
/**
Gets an index to a map block of the currently loaded map at given
coordinates. If there is no block at given coordinates, -1 is returned.
*/
int LCR_mapGetBlockAt(uint8_t x, uint8_t y, uint8_t z)
{
if (LCR_currentMap.blockCount == 0)
return -1;
return LCR_mapGetBlockAtFast(x,y,z,0,LCR_currentMap.blockCount - 1);
2023-08-03 21:12:23 +02:00
}
2024-07-23 19:57:29 +02:00
uint8_t _LCR_encodeMapBlockCoords(uint8_t x, uint8_t y, uint8_t z)
{
return (5 * 7) * z + 7 * y + x;
}
void _LCR_decodeMapBlockCoords(uint8_t byte, uint8_t *x, uint8_t *y, uint8_t *z)
{
*x = (byte % 7);
*y = ((byte / 7) % 5);
*z = (byte / 35);
}
2024-08-01 21:41:21 +02:00
#define LCR_BLOCK_SHAPE_COORD_MAX 12
2024-07-23 19:57:29 +02:00
/**
Decodes XYZ coordinates encoded in a byte returned by LCR_mapGetBlockShape.
Each coordinate will be in range 0 to 12 (including both). This unusual range
is intentional as it for example has an exact mid value.
*/
void LCR_decodeMapBlockCoords(uint8_t byte, uint8_t *x, uint8_t *y, uint8_t *z)
{
_LCR_decodeMapBlockCoords(byte,x,y,z);
*x *= 2;
*y *= 3;
*z *= 2;
}
2024-07-22 01:16:16 +02:00
void _LCR_addBlockShapeByte(uint8_t *bytes, uint8_t *byteCount,
int x, int y, int z)
{
if (*byteCount >= LCR_MAP_BLOCK_SHAPE_MAX_BYTES)
return;
2024-07-23 19:57:29 +02:00
bytes[*byteCount] = _LCR_encodeMapBlockCoords(x,y,z);
2024-07-22 01:16:16 +02:00
*byteCount += 1;
}
2024-09-29 20:52:52 +02:00
/**
Macro that transforms coordinates according to block transformation.
*/
2024-09-26 14:56:39 +02:00
#define LCR_TRANSFORM_COORDS(trans,cx,cy,cz,maxXZ,maxY)\
if (trans & LCR_BLOCK_TRANSFORM_FLIP_H) cx = maxXZ - cx;\
if (trans & 0x20) { /* for both 90 and 270 */ \
cx ^= cz; cz ^= cx; cx ^= cz; /* swap */ \
cx = maxXZ - cx; } \
if (trans & 0x40) { /* for both 180 and 270 */ \
cx = maxXZ - cx; \
cz = maxXZ - cz; } \
if (trans & LCR_BLOCK_TRANSFORM_FLIP_V) \
cy = maxY - cy;
2023-08-03 21:12:23 +02:00
/**
Gets a shape of given map block type as a 3D model composed of triangles. The
2024-07-22 01:16:16 +02:00
model is returned as an array of byte triplets (triangles), with each byte
representing one coordinate. These coordinates can be decoded with
LCR_decodeMapBlockCoords function.
2023-08-03 21:12:23 +02:00
*/
void LCR_mapGetBlockShape(uint8_t blockType, uint8_t transform,
2024-07-22 01:16:16 +02:00
uint8_t bytes[LCR_MAP_BLOCK_SHAPE_MAX_BYTES], uint8_t *byteCount)
{
/*
The coordinate format is following: byte B specifies coordinates X (0 to 6)
2024-07-23 19:57:29 +02:00
= B % 7, Y (vertical, 0 to 4) = (B / 7) % 5, Z (0 to 6) = B % 35. Helper
side view grid:
4 . . . . . . . ^
3 . . . . . . . | y
2 . . . . . . .
1 . . . . . . .
0 . . . . . . .
0 1 2 3 4 5 6 -> x/z
2024-07-22 01:16:16 +02:00
*/
*byteCount = 0;
2024-07-23 19:57:29 +02:00
#define ADD(a,b,c) _LCR_addBlockShapeByte(bytes,byteCount,a,b,c);
2024-07-22 01:16:16 +02:00
switch (blockType)
{
2024-10-07 22:07:58 +02:00
case LCR_BLOCK_CHECKPOINT_0:
case LCR_BLOCK_CHECKPOINT_1:
case LCR_BLOCK_FINISH:
ADD(3,0,3) ADD(0,2,6) ADD(6,2,6)
ADD(3,0,3) ADD(0,2,0) ADD(0,2,6)
ADD(3,0,3) ADD(6,2,0) ADD(0,2,0)
ADD(3,0,3) ADD(6,2,6) ADD(6,2,0)
ADD(3,4,3) ADD(0,2,6) ADD(0,2,0)
ADD(3,4,3) ADD(0,2,0) ADD(6,2,0)
ADD(3,4,3) ADD(6,2,0) ADD(6,2,6)
ADD(3,4,3) ADD(6,2,6) ADD(0,2,6)
break;
2024-10-07 15:52:39 +02:00
2024-07-22 01:16:16 +02:00
case LCR_BLOCK_FULL:
2024-07-24 20:28:57 +02:00
case LCR_BLOCK_BOTTOM:
case LCR_BLOCK_LEFT:
case LCR_BLOCK_BOTTOM_LEFT:
case LCR_BLOCK_BOTTOM_LEFT_FRONT:
2024-08-01 21:41:21 +02:00
case LCR_BLOCK_FULL_ACCEL:
2024-08-06 01:34:38 +02:00
case LCR_BLOCK_FULL_FAN:
2024-12-18 13:01:27 +01:00
case LCR_BLOCK_BOTTOM_ACCEL:
2024-07-24 20:28:57 +02:00
{
uint8_t xRight = 6, yTop = 4,
zBack = 6 >> (blockType == LCR_BLOCK_BOTTOM_LEFT_FRONT);
2024-12-18 13:01:27 +01:00
if (blockType == LCR_BLOCK_BOTTOM ||
blockType == LCR_BLOCK_BOTTOM_ACCEL ||
blockType == LCR_BLOCK_BOTTOM_LEFT ||
2024-07-24 20:28:57 +02:00
blockType == LCR_BLOCK_BOTTOM_LEFT_FRONT)
yTop /= 2;
if (blockType == LCR_BLOCK_LEFT ||
blockType == LCR_BLOCK_BOTTOM_LEFT ||
blockType == LCR_BLOCK_BOTTOM_LEFT_FRONT)
xRight /= 2;
ADD(0,0,0) ADD(xRight,0,0) ADD(xRight,yTop,0) // front
ADD(0,0,0) ADD(xRight,yTop,0) ADD(0,yTop,0)
ADD(xRight,0,0) ADD(xRight,0,zBack) ADD(xRight,yTop,zBack) // right
ADD(xRight,0,0) ADD(xRight,yTop,zBack) ADD(xRight,yTop,0)
ADD(0,0,0) ADD(0,yTop,0) ADD(0,yTop,zBack) // left
ADD(0,0,0) ADD(0,yTop,zBack) ADD(0,0,zBack)
ADD(0,0,zBack) ADD(0,yTop,zBack) ADD(xRight,yTop,zBack) // back
ADD(0,0,zBack) ADD(xRight,yTop,zBack) ADD(xRight,0,zBack)
ADD(0,yTop,0) ADD(xRight,yTop,0) ADD(xRight,yTop,zBack) // top
ADD(0,yTop,0) ADD(xRight,yTop,zBack) ADD(0,yTop,zBack)
ADD(0,0,0) ADD(xRight,0,zBack) ADD(xRight,0,0) // bottom
ADD(0,0,0) ADD(0,0,zBack) ADD(xRight,0,zBack)
break;
}
2024-10-04 00:59:15 +02:00
case LCR_BLOCK_RAMP_CURVED_PLAT:
ADD(0,0,6) ADD(0,4,5) ADD(0,4,6) // left
ADD(6,0,6) ADD(6,4,6) ADD(6,4,5) // right
ADD(0,4,5) ADD(6,4,5) ADD(0,4,6) // top
ADD(0,4,6) ADD(6,4,5) ADD(6,4,6)
2024-10-06 21:53:11 +02:00
// fall through
2024-10-03 00:24:28 +02:00
case LCR_BLOCK_RAMP_CURVED:
2024-10-04 00:59:15 +02:00
{
uint8_t plusZ = blockType == LCR_BLOCK_RAMP_CURVED;
ADD(0,0,0) ADD(6,0,0) ADD(0,1,3 + plusZ) // ramp
ADD(0,1,3 + plusZ) ADD(6,0,0) ADD(6,1,3 + plusZ)
ADD(0,1,3 + plusZ) ADD(6,1,3 + plusZ) ADD(0,2,4 + plusZ) // ramp
ADD(0,2,4 + plusZ) ADD(6,1,3 + plusZ) ADD(6,2,4 + plusZ)
ADD(0,2,4 + plusZ) ADD(6,2,4 + plusZ) ADD(0,4,5 + plusZ) // ramp
ADD(0,4,5 + plusZ) ADD(6,2,4 + plusZ) ADD(6,4,5 + plusZ)
ADD(0,0,0) ADD(0,1,3 + plusZ) ADD(0,0,6) // left
ADD(0,0,6) ADD(0,1,3 + plusZ) ADD(0,2,4 + plusZ)
ADD(0,0,6) ADD(0,2,4 + plusZ) ADD(0,4,5 + plusZ)
ADD(6,0,0) ADD(6,0,6) ADD(6,1,3 + plusZ) // right
ADD(6,0,6) ADD(6,2,4 + plusZ) ADD(6,1,3 + plusZ)
ADD(6,0,6) ADD(6,4,5 + plusZ) ADD(6,2,4 + plusZ)
ADD(0,0,6) ADD(0,4,6) ADD(6,0,6) // back
ADD(6,0,6) ADD(0,4,6) ADD(6,4,6)
ADD(0,0,0) ADD(6,0,6) ADD(6,0,0) // bottom
ADD(0,0,0) ADD(0,0,6) ADD(6,0,6)
2024-10-03 00:24:28 +02:00
break;
2024-10-04 00:59:15 +02:00
}
2024-10-03 00:24:28 +02:00
2024-07-24 20:28:57 +02:00
case LCR_BLOCK_RAMP_CURVED_WALL:
2024-07-23 19:57:29 +02:00
ADD(0,0,0) ADD(5,0,0) ADD(0,1,3) // ramp
ADD(0,1,3) ADD(5,0,0) ADD(5,1,3)
ADD(0,1,3) ADD(5,1,3) ADD(0,2,4) // ramp
ADD(0,2,4) ADD(5,1,3) ADD(5,2,4)
ADD(0,2,4) ADD(5,2,4) ADD(0,4,5) // ramp
ADD(0,4,5) ADD(5,2,4) ADD(5,4,5)
ADD(0,4,5) ADD(5,4,5) ADD(0,4,6) // top
ADD(0,4,6) ADD(5,4,5) ADD(6,4,6)
ADD(5,4,5) ADD(6,4,0) ADD(6,4,6) // top
ADD(5,4,5) ADD(5,4,0) ADD(6,4,0)
ADD(5,4,0) ADD(5,4,5) ADD(5,2,4) // inner side
ADD(5,4,0) ADD(5,2,4) ADD(5,1,3)
ADD(5,4,0) ADD(5,1,3) ADD(5,0,0)
ADD(5,4,0) ADD(5,0,0) ADD(6,4,0) // front
ADD(6,4,0) ADD(5,0,0) ADD(6,0,0)
ADD(6,4,0) ADD(6,0,0) ADD(6,4,6) // right
ADD(6,4,6) ADD(6,0,0) ADD(6,0,6)
ADD(0,0,6) ADD(0,4,5) ADD(0,4,6) // left
ADD(0,0,6) ADD(0,2,4) ADD(0,4,5)
ADD(0,0,6) ADD(0,1,3) ADD(0,2,4)
ADD(0,0,6) ADD(0,0,0) ADD(0,1,3)
ADD(0,0,6) ADD(0,4,6) ADD(6,0,6) // back
ADD(6,0,6) ADD(0,4,6) ADD(6,4,6)
ADD(0,0,0) ADD(6,0,6) ADD(6,0,0) // bottom
ADD(0,0,0) ADD(0,0,6) ADD(6,0,6)
2024-07-22 01:16:16 +02:00
break;
2024-07-24 20:28:57 +02:00
2024-09-23 23:31:30 +02:00
case LCR_BLOCK_RAMP:
case LCR_BLOCK_RAMP_12:
case LCR_BLOCK_RAMP_14:
case LCR_BLOCK_RAMP_34:
2024-10-01 22:08:03 +02:00
case LCR_BLOCK_RAMP_STEEP:
2024-12-18 13:01:27 +01:00
case LCR_BLOCK_RAMP_ACCEL:
case LCR_BLOCK_RAMP_FAN:
2024-09-23 23:31:30 +02:00
{
2024-10-01 22:08:03 +02:00
uint8_t front, top;
LCR_rampGetDimensions(blockType,&top,&front);
front = 6 - front;
ADD(0,0,front) ADD(0,top,6) ADD(0,0,6) // side
ADD(6,0,front) ADD(6,0,6) ADD(6,top,6) // side
ADD(0,0,front) ADD(6,0,front) ADD(0,top,6) // top
ADD(6,0,front) ADD(6,top,6) ADD(0,top,6) // top
ADD(0,0,6) ADD(6,top,6) ADD(6,0,6) // back
ADD(0,0,6) ADD(0,top,6) ADD(6,top,6) // back
ADD(0,0,front) ADD(0,0,6) ADD(6,0,6) // bottom
ADD(0,0,front) ADD(6,0,6) ADD(6,0,front) // bottom
2024-09-23 23:31:30 +02:00
break;
}
2024-10-01 22:38:49 +02:00
case LCR_BLOCK_CORNER:
case LCR_BLOCK_CORNER_12:
{
uint8_t right = blockType == LCR_BLOCK_CORNER ? 6 : 3;
ADD(0,0,0) ADD(right,0,6) ADD(right,4,6) // front/right
ADD(0,0,0) ADD(right,4,6) ADD(0,4,0) // front/right
ADD(0,0,0) ADD(0,4,6) ADD(0,0,6) // left
ADD(0,0,0) ADD(0,4,0) ADD(0,4,6) // left
ADD(right,0,6) ADD(0,0,6) ADD(0,4,6) // back
ADD(0,4,6) ADD(right,4,6) ADD(right,0,6) // back
ADD(0,4,0) ADD(right,4,6) ADD(0,4,6) // top
ADD(0,0,6) ADD(right,0,6) ADD(0,0,0) // bottom
break;
}
2024-12-15 21:43:43 +01:00
case LCR_BLOCK_CORNER_CONVEX:
case LCR_BLOCK_CORNER_CONCAVE:
{
uint8_t
mx = blockType == LCR_BLOCK_CORNER_CONVEX ? 4 : 2,
mz = blockType == LCR_BLOCK_CORNER_CONVEX ? 2 : 4;
ADD(0,0,0) ADD(0,4,6) ADD(0,0,6) // left
ADD(0,0,0) ADD(0,4,0) ADD(0,4,6) // left
ADD(6,0,6) ADD(0,0,6) ADD(0,4,6) // back
ADD(0,4,6) ADD(6,4,6) ADD(6,0,6) // back
ADD(0,0,0) ADD(mx,4,mz) ADD(0,4,0) // right
ADD(mx,0,mz) ADD(mx,4,mz) ADD(0,0,0)
ADD(6,4,6) ADD(mx,4,mz) ADD(6,0,6)
ADD(6,0,6) ADD(mx,4,mz) ADD(mx,0,mz)
ADD(0,4,0) ADD(mx,4,mz) ADD(0,4,6) // top
ADD(0,4,6) ADD(mx,4,mz) ADD(6,4,6)
ADD(0,0,0) ADD(0,0,6) ADD(mx,0,mz) // bottom
ADD(0,0,6) ADD(6,0,6) ADD(mx,0,mz)
break;
}
2024-12-12 23:17:06 +01:00
2024-12-02 23:05:41 +01:00
case LCR_BLOCK_BUMP:
ADD(3,0,0) ADD(6,0,3) ADD(3,1,3) // top
ADD(6,0,3) ADD(3,0,6) ADD(3,1,3)
ADD(3,0,6) ADD(0,0,3) ADD(3,1,3)
ADD(0,0,3) ADD(3,0,0) ADD(3,1,3)
ADD(3,0,0) ADD(3,0,6) ADD(6,0,3) // bottom
ADD(3,0,6) ADD(3,0,0) ADD(0,0,3) // bottom
break;
2024-12-11 22:53:52 +01:00
case LCR_BLOCK_RAMP_CORNER:
ADD(6,0,6) ADD(0,0,0) ADD(6,4,0) // diagonal
ADD(6,0,6) ADD(6,4,0) ADD(6,0,0) // right
ADD(0,0,0) ADD(6,0,0) ADD(6,4,0) // front
ADD(0,0,0) ADD(6,0,6) ADD(6,0,0) // bottom
break;
2024-11-29 00:51:43 +01:00
case LCR_BLOCK_HILL:
ADD(0,0,0) ADD(6,0,0) ADD(0,2,1) // front
ADD(6,0,0) ADD(6,2,1) ADD(0,2,1)
ADD(0,2,1) ADD(6,2,1) ADD(0,3,2) // front 2
ADD(6,2,1) ADD(6,3,2) ADD(0,3,2)
ADD(0,3,2) ADD(6,3,2) ADD(0,4,4) // front 3
ADD(6,3,2) ADD(6,4,4) ADD(0,4,4)
ADD(0,4,4) ADD(6,4,4) ADD(0,4,6) // top
ADD(6,4,4) ADD(6,4,6) ADD(0,4,6)
ADD(0,0,0) ADD(0,0,6) ADD(6,0,0) // bottom
ADD(6,0,0) ADD(0,0,6) ADD(6,0,6)
ADD(0,0,6) ADD(0,4,6) ADD(6,4,6) // back
ADD(0,0,6) ADD(6,4,6) ADD(6,0,6)
ADD(0,0,0) ADD(0,2,1) ADD(0,0,6) // left
ADD(0,2,1) ADD(0,3,2) ADD(0,0,6)
ADD(0,3,2) ADD(0,4,4) ADD(0,0,6)
ADD(0,4,4) ADD(0,4,6) ADD(0,0,6)
ADD(6,0,0) ADD(6,0,6) ADD(6,2,1) // right
ADD(6,2,1) ADD(6,0,6) ADD(6,3,2)
ADD(6,3,2) ADD(6,0,6) ADD(6,4,4)
ADD(6,4,4) ADD(6,0,6) ADD(6,4,6)
break;
2024-07-24 20:28:57 +02:00
default: break;
2024-07-22 01:16:16 +02:00
}
2024-07-23 19:57:29 +02:00
if (transform)
{
for (int i = 0; i < *byteCount; ++i)
{
uint8_t x, y, z, tmp;
_LCR_decodeMapBlockCoords(bytes[i],&x,&y,&z);
2024-09-26 14:56:39 +02:00
2024-09-29 20:52:52 +02:00
LCR_TRANSFORM_COORDS(transform,x,y,z,6,4)
2024-07-23 19:57:29 +02:00
bytes[i] = _LCR_encodeMapBlockCoords(x,y,z);
}
if (((transform & LCR_BLOCK_TRANSFORM_FLIP_H) == 0) !=
((transform & LCR_BLOCK_TRANSFORM_FLIP_V) == 0)) // flip triangles
for (int i = 0; i < *byteCount; i += 3)
{
uint8_t tmp = bytes[i];
bytes[i] = bytes[i + 1];
bytes[i + 1] = tmp;
}
}
#undef ADD
2023-08-03 21:12:23 +02:00
}
2023-07-21 21:17:49 +02:00
#endif // guard