251 lines
9.4 KiB
C++
251 lines
9.4 KiB
C++
/*
|
|
* Created by Brett on 11/02/23.
|
|
* Licensed under GNU General Public License V3.0
|
|
* See LICENSE file for license detail
|
|
*/
|
|
|
|
#ifndef FINALPROJECT_WORLD_H
|
|
#define FINALPROJECT_WORLD_H
|
|
|
|
#include <world/chunk/storage.h>
|
|
#include <render/gl.h>
|
|
#include <phmap.h>
|
|
#include "blt/profiling/profiler.h"
|
|
#include "util/threadpool.h"
|
|
#include <render/frustum.h>
|
|
|
|
namespace fp {
|
|
|
|
namespace _static {
|
|
/**
|
|
* Converts from world coord to chunk-internal coords
|
|
* @param coord world space coordinate
|
|
* @return chunk internal coord
|
|
*/
|
|
static inline int world_to_internal(int coord) {
|
|
auto val = coord % CHUNK_SIZE;
|
|
return val < 0 ? CHUNK_SIZE + val : val;
|
|
}
|
|
|
|
static inline block_pos world_to_internal(const block_pos& coord) {
|
|
return {world_to_internal(coord.x), world_to_internal(coord.y),
|
|
world_to_internal(coord.z)};
|
|
}
|
|
|
|
/**
|
|
* Converts from world coord to chunk pos coords
|
|
*
|
|
* consider: (int) (-31 / 32) which equals 0
|
|
* but a negative chunk would be stored at -1, not 0 (since that is taken by the positive coord chunk)
|
|
* an arithmetic right shift would produce the desired -1 (see Java, which performs a signed right bit shift)
|
|
* however in C++ shifting on a signed type is undefined behaviour. So we must emulate an arithmetic right shift.
|
|
*
|
|
* @param coord x,y, or z coordinate to convert
|
|
* @return a right arithmetic bit shift resulting in a signed division of the coordinate by CHUNK_SIZE
|
|
*/
|
|
static inline int world_to_chunk(int coord) {
|
|
auto ucoord = (unsigned int) coord;
|
|
|
|
ucoord >>= CHUNK_SHIFT;
|
|
|
|
if (coord < 0) {
|
|
// the mask only has to be generated once since it is never modified at runtime beyond assignment
|
|
static unsigned int mask = 0;
|
|
if (mask == 0) {
|
|
for (int i = 0; i < CHUNK_SHIFT; i++)
|
|
mask |= (1 << ((sizeof(int) * 8 - 1) - i));
|
|
}
|
|
ucoord |= mask;
|
|
}
|
|
|
|
return (int) (ucoord);
|
|
}
|
|
|
|
static inline chunk_pos world_to_chunk(const block_pos& pos) {
|
|
return {world_to_chunk(pos.x), world_to_chunk(pos.y), world_to_chunk(pos.z)};
|
|
}
|
|
|
|
}
|
|
|
|
struct chunk {
|
|
private:
|
|
block_storage* storage;
|
|
mesh_storage* mesh = nullptr;
|
|
VAO* chunk_vao;
|
|
chunk_pos pos;
|
|
|
|
chunk_mesh_status dirtiness = OKAY;
|
|
chunk_update_status status = NONE;
|
|
unsigned long render_size = 0;
|
|
public:
|
|
explicit chunk(chunk_pos pos): pos(pos) {
|
|
storage = new block_storage();
|
|
chunk_vao = new VAO();
|
|
auto vbo = new VBO(ARRAY_BUFFER, nullptr, 0, DYNAMIC);
|
|
//auto data_size = 3 * sizeof(float) + 3 * sizeof(float);
|
|
//chunk_vao->bindVBO(vbo, 0, 3, GL_FLOAT, (int) data_size, 0);
|
|
//chunk_vao->bindVBO(vbo, 1, 3, GL_FLOAT, (int) data_size, 3 * sizeof(float), true);
|
|
chunk_vao->bindVBO(vbo, 0, 1, GL_FLOAT, sizeof(float), 0);
|
|
chunk_vao->bindElementVBO(new VBO(ELEMENT_BUFFER, nullptr, 0, DYNAMIC));
|
|
}
|
|
|
|
void render(shader& shader);
|
|
|
|
void updateChunkMesh();
|
|
|
|
/**
|
|
* Mark the chunk as completely dirty and in need of a full chunk refresh
|
|
*/
|
|
inline void markDirty() {
|
|
dirtiness = DIRTY;
|
|
}
|
|
|
|
/**
|
|
* Full chunk mesh is now completely generated and waiting on uploading to the GPU
|
|
*/
|
|
inline void markRefresh() {
|
|
dirtiness = REFRESH;
|
|
}
|
|
|
|
[[nodiscard]] inline block_storage* getBlockStorage() {
|
|
return storage;
|
|
}
|
|
|
|
[[nodiscard]] inline mesh_storage*& getMeshStorage() {
|
|
return mesh;
|
|
}
|
|
|
|
[[nodiscard]] inline VAO*& getVAO() {
|
|
return chunk_vao;
|
|
}
|
|
|
|
[[nodiscard]] inline chunk_pos getPos() const {
|
|
return pos;
|
|
}
|
|
|
|
[[nodiscard]] inline chunk_mesh_status getDirtiness() const {
|
|
return dirtiness;
|
|
}
|
|
|
|
[[nodiscard]] inline chunk_update_status& getStatus() {
|
|
return status;
|
|
}
|
|
|
|
inline void setStatus(const chunk_update_status& new_status) {
|
|
status = new_status;
|
|
}
|
|
|
|
~chunk() {
|
|
delete storage;
|
|
delete chunk_vao;
|
|
delete mesh;
|
|
}
|
|
};
|
|
|
|
struct chunk_neighbours {
|
|
fp::chunk* neighbours[6];
|
|
|
|
inline chunk*& operator[](int i) {
|
|
return neighbours[i];
|
|
}
|
|
};
|
|
|
|
class world {
|
|
private:
|
|
// not using the parallel functions of the map, we will do manual concurrency control
|
|
// "The parallel hash maps are preferred when you have a few hash maps that will store a very large number of values.
|
|
// The non-parallel hash maps are preferred if you have a large number of hash maps, each storing a relatively small number of values."
|
|
phmap::parallel_flat_hash_map<chunk_pos, chunk*, _static::chunk_pos_hash, _static::chunk_pos_equality> chunk_storage;
|
|
std::mutex chunkInsertMutex;
|
|
std::thread* chunkGenerationThread;
|
|
const unsigned int THREAD_COUNT = 16;
|
|
volatile bool running = true;
|
|
protected:
|
|
void generateChunkMesh(chunk* chunk);
|
|
|
|
chunk* generateChunk(const chunk_pos& pos);
|
|
|
|
inline void getNeighbours(const chunk_pos& pos, chunk_neighbours& neighbours) {
|
|
neighbours[X_POS] = getChunk(chunk_pos{pos.x + 1, pos.y, pos.z});
|
|
neighbours[X_NEG] = getChunk(chunk_pos{pos.x - 1, pos.y, pos.z});
|
|
neighbours[Y_POS] = getChunk(chunk_pos{pos.x, pos.y + 1, pos.z});
|
|
neighbours[Y_NEG] = getChunk(chunk_pos{pos.x, pos.y - 1, pos.z});
|
|
neighbours[Z_POS] = getChunk(chunk_pos{pos.x, pos.y, pos.z + 1});
|
|
neighbours[Z_NEG] = getChunk(chunk_pos{pos.x, pos.y, pos.z - 1});
|
|
}
|
|
|
|
inline bool spawnChunk(chunk* chunk) {
|
|
// make sure not to insert null chunks
|
|
if (chunk == nullptr)
|
|
return false;
|
|
// we must lock for the entire insertion including neighbour updates.
|
|
// otherwise chunk updates might not be properly handled
|
|
std::scoped_lock<std::mutex> lock(chunkInsertMutex);
|
|
// or overwrite existing chunks (*memory leak*)
|
|
if (chunk_storage.find(chunk->getPos()) != chunk_storage.end())
|
|
return false;
|
|
chunk_storage.insert({chunk->getPos(), chunk});
|
|
|
|
chunk_neighbours chunkNeighbours{};
|
|
getNeighbours(chunk->getPos(), chunkNeighbours);
|
|
|
|
for (auto* p : chunkNeighbours.neighbours) {
|
|
if (p)
|
|
p->setStatus(NEIGHBOUR_CREATE);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
inline static chunk_pos offsetCameraByChunk(int i, int j, int k) {
|
|
const auto& pos = fp::camera::getPosition();
|
|
int x = (int) pos.x();
|
|
int y = (int) pos.y();
|
|
int z = (int) pos.z();
|
|
auto camera_chunk_pos = fp::_static::world_to_chunk({x, y, z});
|
|
return chunk_pos{camera_chunk_pos.x + i, // chunk x
|
|
camera_chunk_pos.y + j, // chunk y
|
|
camera_chunk_pos.z + k}; // chunk z
|
|
}
|
|
|
|
inline chunk* getChunk(const chunk_pos& pos) {
|
|
const auto map_pos = chunk_storage.find(pos);
|
|
if (map_pos == chunk_storage.end())
|
|
return nullptr;
|
|
return map_pos->second;
|
|
}
|
|
|
|
inline chunk* getChunk(const block_pos& pos) {
|
|
return chunk_storage[_static::world_to_chunk(pos)];
|
|
}
|
|
|
|
public:
|
|
world();
|
|
|
|
void update();
|
|
|
|
void render(fp::shader& shader);
|
|
|
|
inline bool setBlock(const block_pos& pos, block_type blockID) {
|
|
auto c = getChunk(pos);
|
|
if (!c)
|
|
return false;
|
|
// mark the chunk for a mesh update
|
|
c->markDirty();
|
|
c->getBlockStorage()->set(_static::world_to_internal(pos), blockID);
|
|
return true;
|
|
}
|
|
|
|
inline block_type getBlock(const block_pos& pos) {
|
|
auto c = getChunk(pos);
|
|
if (!c)
|
|
return fp::registry::AIR;
|
|
return c->getBlockStorage()->get(_static::world_to_internal(pos));
|
|
}
|
|
|
|
~world();
|
|
};
|
|
|
|
}
|
|
|
|
#endif //FINALPROJECT_WORLD_H
|