beco
beco is a file format that is used for storing per-coordinate data.
Structure
beco files are little endian on Switch and big endian on Wii U.
Header
Offset | Type | Description |
---|---|---|
0x0 | u32 | Magic (0x00112233) |
0x4 | u32 | Number of rows |
0x8 | u32 | Divisor |
0xc | u32 | Padding |
Offsets
Offset | Type | Description |
---|---|---|
0x0 | u32[num_rows] | Offsets to row data, divided by 2 and relative to the start of the row section |
Rows
Segment
Offset | Type | Description |
---|---|---|
0x0 | u16 | Custom data |
0x2 | u16 | Length (X axis) |
This structure is repeated until the entire row has been covered by segments. The sum of all segment lengths is equal to the length of the map on the X axis.
Operations
Getting data for a given coordinate
The algorithm that returns the custom data that is associated with a coordinate is very simple. It consists of determining the correct row index in the table, then iterating over segments in that row and returning the segment data when the X coordinate is within its bounds.
For the reference, here is the official, mostly unreversed function[1] that implements the aforementioned algorithm:
A cleaner version of this algorithm is given below:
unsigned int eco::getCurrentAreaNum(float posX, float posZ, Ecosystem* unused, EcoMapInfo* info)
{
posX = std::clamp(posX, -5000.0, 4999.0);
posZ = std::clamp(posZ, -4000.0, 4000.0);
float epsilon1 = (posX + 5000.0 < 0.0) ? -0.5 : 0.5;
float epsilon2 = (posZ + 4000.0 < 0.0) ? -0.5 : 0.5;
int x = posX + 5000.0 + epsilon1;
int z = (posZ + 4000.0 + epsilon2) / info->header->divisor;
uint row = std::clamp(z, 0, info->header->numRows - 2);
if (info->header->divisor == 10)
x = x / 10 + ((0x66666667LL * x) >> 0x3F);
uint* offsets = &info->rowOffsets[row];
if ( offsets[0] >= offsets[1] )
return 0xFFFFFFFF;
Segment* segment = reinterpret_cast<Segment*>((char *)info->rows + 2 * offsets[0]);
Segment* segmentEnd = reinterpret_cast<Segment*>((char *)info->rows + 2 * offsets[1]);
uint totalLength = 0;
while (true)
{
totalLength += segment->length;
if (x < totalLength)
return segment->value;
++segment;
if (segment >= segmentEnd)
return 0xFFFFFFFF;
}
}
Usage in Breath of the Wild
beco files are used in Breath of the Wild to map coordinates to arbitrary u16 data. Three beco files are used for different purposes: FieldMapArea.beco, MapTower.beco and LoadBalancer.beco.
- ↑ 0x7100E41C40 in Switch 1.5.0