#include "Dimension.h" #include "Constants.h" #include "Datei.h" #include "Game.h" #include "NoBlock.h" using namespace Framework; Dimension::Dimension(int id) : Thread(), dimensionId(id), gravity(9.8f), chunks(new Trie()), entities(new RCArray()) { start(); } Dimension::~Dimension() { entities->release(); chunks->release(); } void Dimension::api(Framework::InMemoryBuffer* zRequest, NetworkMessage* zResponse, Entity* zSource) { DoLaterHandler laterHandler; char type; zRequest->lese(&type, 1); switch (type) { case 0: // chunk message { Punkt center; zRequest->lese((char*)¢er.x, 4); zRequest->lese((char*)¢er.y, 4); cs.lock(); Chunk* cC = zChunk(center); if (!cC) { // TODO: have a max amount of waiting requests per player waitingRequests.add({ dynamic_cast(zRequest->getThis()), center, zSource->getId() }); Game::INSTANCE->requestArea({ center.x - CHUNK_SIZE / 2, center.y - CHUNK_SIZE / 2, center.x + CHUNK_SIZE / 2 - 1, center.y + CHUNK_SIZE / 2 - 1, dimensionId }); } else { cC->api(zRequest, zSource, laterHandler); } cs.unlock(); break; } } } void Dimension::tickEntities() { for (auto entity : *entities) { if (!entity->isRemoved() && zChunk(Punkt((int)entity->getPosition().x, (int)entity->getPosition().y))) entity->prepareTick(this); } int index = 0; for (auto entity : *entities) { if (!entity->isRemoved() && zChunk(Punkt((int)entity->getPosition().x, (int)entity->getPosition().y))) entity->tick(this); index++; } } void Dimension::thread() { // light calculation int index = 0; ZeitMesser messer; messer.messungStart(); double time = 0; Framework::Array> internalLightUpdateQueue; while (true) { Vec3 position; if (internalLightUpdateQueue.getEintragAnzahl()) { position = internalLightUpdateQueue.get(0); internalLightUpdateQueue.remove(0); } else { if (priorizedLightUpdateQueue.getEintragAnzahl()) { prioLightCs.lock(); position = priorizedLightUpdateQueue.get(0); priorizedLightUpdateQueue.remove(0); prioLightCs.unlock(); } else { if (!lightUpdateQueue.getEintragAnzahl()) { messer.messungEnde(); time += messer.getSekunden(); Sleep(500); messer.messungStart(); continue; } lightCs.lock(); position = lightUpdateQueue.get(0); lightUpdateQueue.remove(0); lightCs.unlock(); } } // TODO: do not remove chunks while light calculation Chunk* chunk = zChunk(Game::INSTANCE->getChunkCenter(position.x, position.y)); if (position.z >= 0 && position.z < WORLD_HEIGHT) { if (chunk) { int x = position.x % CHUNK_SIZE; int y = position.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; unsigned char* light = chunk->getLightData(Vec3(x, y, position.z)); unsigned char dayLight[6] = { 255, 255, 255, 0, 0, 0 }; unsigned char noLight[6] = { 0, 0, 0, 0, 0, 0 }; unsigned char newLight[6] = { 0, 0, 0, 0, 0, 0 }; // add neighbor light emission for (int i = 0; i < 6; i++) { unsigned char* neighborLeight; Vec3 neighborPos = position + getDirection(getDirectionFromIndex(i)); if (neighborPos.z < 0) { neighborLeight = noLight; } else if (neighborPos.z >= WORLD_HEIGHT) { neighborLeight = dayLight; } else { Chunk* neighborChunk = zChunk(Game::INSTANCE->getChunkCenter(neighborPos.x, neighborPos.y)); int x = neighborPos.x % CHUNK_SIZE; int y = neighborPos.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; if (neighborChunk) neighborLeight = neighborChunk->getLightData(Vec3(x, y, neighborPos.z)); else neighborLeight = noLight; } for (int j = 0; j < 3; j++) newLight[j] = (unsigned char)MAX(newLight[j], i == getDirectionIndex(TOP) ? neighborLeight[j] : (unsigned char)((float)neighborLeight[j] * 0.8f)); for (int j = 3; j < 6; j++) newLight[j] = (unsigned char)MAX(newLight[j], (unsigned char)((float)neighborLeight[j] * 0.8f)); } const Block* current = zBlockOrDefault(position); if (!current) current = &NoBlock::INSTANCE; // add own light emission for (int j = 3; j < 6; j++) newLight[j] = (unsigned char)MAX(newLight[j], current->getLightEmisionColor()[j - 3]); current->filterPassingLight(newLight); current->filterPassingLight(newLight + 3); for (int i = 0; i < 6; i++) { if (newLight[i] != light[i]) { chunk->setLightData(Vec3(x, y, position.z), newLight); for (int j = 0; j < 6; j++) internalLightUpdateQueue.add(position + getDirection(getDirectionFromIndex(j)), 0); break; } } } } index++; if (index > 100000) { messer.messungEnde(); time += messer.getSekunden(); std::cout << "100000 light updates needed " << time << " seconds\n"; time = 0; index = 0; Sleep(250); messer.messungStart(); } } } void Dimension::getAddrOf(Punkt cPos, char* addr) const { *(int*)addr = cPos.x; *((int*)addr + 1) = cPos.y; } void Dimension::getAddrOfWorld(Punkt wPos, char* addr) const { if (wPos.x < 0) wPos.x -= CHUNK_SIZE; if (wPos.y < 0) // needed because otherwise would (-8, -8) have the same adress as (8, 8) wPos.y -= CHUNK_SIZE; wPos /= CHUNK_SIZE; getAddrOf(wPos, addr); } Chunk* Dimension::zChunk(Punkt wPos) const { char addr[8]; getAddrOfWorld(wPos, addr); return chunks->z(addr, 8); } Framework::Either Dimension::zBlock(Vec3 location) { Chunk* c = zChunk(Game::INSTANCE->getChunkCenter(location.x, location.y)); if (c) { int x = location.x % CHUNK_SIZE; int y = location.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; return c->zBlockAt(Vec3(x, y, location.z)); } return 0; } Block* Dimension::zRealBlockInstance(Framework::Vec3 location) { Chunk* c = zChunk(Game::INSTANCE->getChunkCenter(location.x, location.y)); if (c) { int x = location.x % CHUNK_SIZE; int y = location.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; c->instantiateBlock(Vec3(x, y, location.z)); auto result = c->zBlockAt(Vec3(x, y, location.z)); return result.isA() ? result.getA() : 0; } return 0; } const Block* Dimension::zBlockOrDefault(Framework::Vec3 location) { Chunk* c = zChunk(Game::INSTANCE->getChunkCenter(location.x, location.y)); if (c) { int x = location.x % CHUNK_SIZE; int y = location.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; return c->zBlockConst(Vec3(x, y, location.z)); } return 0; } void Dimension::placeBlock(Framework::Vec3 location, Framework::Either block) { Chunk* c = zChunk(Game::getChunkCenter(location.x, location.y)); if (c) { int x = location.x % CHUNK_SIZE; int y = location.y % CHUNK_SIZE; if (x < 0) x += CHUNK_SIZE; if (y < 0) y += CHUNK_SIZE; if (block.isA()) c->putBlockAt(Vec3(x, y, location.z), block); else { c->putBlockAt(Vec3(x, y, location.z), 0); c->putBlockTypeAt(Vec3(x, y, location.z), block); } } else if (block.isA()) block.getA()->release(); } void Dimension::addEntity(Entity* entity) { entities->add(entity); } void Dimension::setChunk(Chunk* chunk, Punkt center) { chunkCs.lock(); char addr[8]; getAddrOfWorld(center, addr); Chunk* old = chunks->z(addr, 8); if (old) { for (int i = 0; i < chunkList.getEintragAnzahl(); i++) { if (chunkList.get(i) == old) { chunkList.remove(i); break; } } } chunks->set(addr, 8, chunk); if (chunk) { chunkList.add(chunk); chunk->setAdded(); } getAddrOfWorld(center + Punkt(CHUNK_SIZE, 0), addr); Chunk* zChunk = chunks->z(addr, 8); if (zChunk) { zChunk->setNeighbor(WEST, chunk); if (chunk) { chunk->setNeighbor(EAST, zChunk); } } getAddrOfWorld(center + Punkt(-CHUNK_SIZE, 0), addr); zChunk = chunks->z(addr, 8); if (zChunk) { zChunk->setNeighbor(EAST, chunk); if (chunk) chunk->setNeighbor(WEST, zChunk); } getAddrOfWorld(center + Punkt(0, CHUNK_SIZE), addr); zChunk = chunks->z(addr, 8); if (zChunk) { zChunk->setNeighbor(NORTH, chunk); if (chunk) chunk->setNeighbor(SOUTH, zChunk); } getAddrOfWorld(center + Punkt(0, -CHUNK_SIZE), addr); zChunk = chunks->z(addr, 8); if (zChunk) { zChunk->setNeighbor(SOUTH, chunk); if (chunk) chunk->setNeighbor(NORTH, zChunk); } DoLaterHandler laterHandler; if (chunk) { cs.lock(); int index = 0; for (Iterator iterator = waitingRequests.begin(); iterator; ) { Entity* zE = Game::INSTANCE->zEntity(iterator.val().sourceId); if (zE) { if (iterator.val().chunkCenter == chunk->getCenter()) { chunk->api(iterator.val().request, zE, laterHandler); iterator.val().request->release(); iterator.remove(); continue; } } else { iterator.val().request->release(); iterator.remove(); continue; } iterator++; index++; } cs.unlock(); } chunkCs.unlock(); updateLightAtChunkBorders(center); } void Dimension::save(Text worldDir) const { for (auto chunk = chunks->getIterator(); chunk; chunk++) { if (!chunk._) continue; Datei* file = new Datei(); Text filePath = worldDir + "/dim/" + dimensionId + "/"; filePath.appendHex(chunk->getCenter().x); filePath += "_"; filePath.appendHex(chunk->getCenter().y); filePath += ".chunk"; file->setDatei(filePath); if (file->open(Datei::Style::schreiben)) chunk->save(file); file->close(); file->release(); } Text filePath = worldDir + "/dim/" + dimensionId + "/entities"; Datei* file = new Datei(); file->setDatei(filePath); if (file->open(Datei::Style::schreiben)) { for (Entity* entity : *entities) { if (entity->zType()->getId() != PlayerEntityType::ID) { if (!entity->isRemoved()) { int type = entity->zType()->getId(); file->schreibe((char*)&type, 4); StaticRegistry::INSTANCE.zElement(type)->saveEntity(entity, file); } } else { Datei pFile; pFile.setDatei(worldDir + "/player/" + ((Player*)entity)->getName()); if (pFile.open(Datei::Style::schreiben)) PlayerEntityType::INSTANCE->saveEntity(entity, &pFile); } } file->close(); } } int Dimension::getDimensionId() const { return dimensionId; } bool Dimension::hasChunck(int x, int y) const { return zChunk(Punkt(x, y)); } float Dimension::getGravity() const { return gravity; } void Dimension::removeOldChunks() { chunkCs.lock(); Array removed; int index = 0; for (Chunk* chunk : chunkList) { if (!chunk->hasObservers()) removed.add(index, 0); index++; } for (int i : removed) { Chunk* chunk = chunkList.get(i); // TODO: save chunk in a seperate thread? Text filePath = Game::INSTANCE->getWorldDirectory() + "/dim/" + getDimensionId() + "/"; filePath.appendHex(chunk->getCenter().x); filePath += "_"; filePath.appendHex(chunk->getCenter().y); filePath += ".chunk"; Datei d; d.setDatei(filePath); d.erstellen(); d.open(Datei::Style::schreiben); chunk->save(&d); d.close(); chunk->prepareRemove(); setChunk(0, chunk->getCenter()); } chunkCs.unlock(); } Entity* Dimension::zEntity(int id) { for (auto entity : *entities) { if (!entity->isRemoved() && entity->getId() == id) return entity; } return 0; } Entity* Dimension::zNearestEntity(Framework::Vec3 pos, std::function filter) { Entity* result = 0; float sqDist = 0; for (auto entity : *entities) { if (!entity->isRemoved() && filter(entity)) { float d = pos.abstandSq(entity->getPosition()); if (!result || d < sqDist) { result = entity; sqDist = d; } } } return result; } void Dimension::removeEntity(int id) { int index = 0; for (auto entity : *entities) { if (entity->getId() == id) { entities->remove(index); return; } index++; } } void Dimension::removeSubscriptions(Entity* zEntity) { for (Chunk* chunk : chunkList) chunk->removeObserver(zEntity); } void Dimension::updateLightning(Vec3 location) { lightCs.lock(); lightUpdateQueue.add(location, 0); lightCs.unlock(); } void Dimension::updateLightningWithoutWait(Framework::Vec3 location) { prioLightCs.lock(); priorizedLightUpdateQueue.add(location, 0); prioLightCs.unlock(); } void Dimension::updateLightAtChunkBorders(Punkt chunkCenter) { if (lightUpdateQueue.getEintragAnzahl() > 300000) { std::cout << "warning: light calculation queue is over 300000 blocks long"; } for (int i = WORLD_HEIGHT - 1; i >= 0; i--) { for (int j = 0; j < CHUNK_SIZE; j++) { updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8 - 1, chunkCenter.y - CHUNK_SIZE / 8 + j, i)); updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8, chunkCenter.y - CHUNK_SIZE / 8 + j, i)); updateLightning(Vec3(chunkCenter.x + CHUNK_SIZE / 8 - 1, chunkCenter.y - CHUNK_SIZE / 8 + j, i)); updateLightning(Vec3(chunkCenter.x + CHUNK_SIZE / 8, chunkCenter.y - CHUNK_SIZE / 8 + j, i)); updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8 + j, chunkCenter.y - CHUNK_SIZE / 8 - 1, i)); updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8 + j, chunkCenter.y - CHUNK_SIZE / 8, i)); updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8 + j, chunkCenter.y + CHUNK_SIZE / 8 - 1, i)); updateLightning(Vec3(chunkCenter.x - CHUNK_SIZE / 8 + j, chunkCenter.y + CHUNK_SIZE / 8, i)); } } }