OpenMiner/source/server/world/TerrainGenerator.cpp

330 lines
12 KiB
C++

/*
* =====================================================================================
*
* OpenMiner
*
* Copyright (C) 2018-2020 Unarelith, Quentin Bazin <openminer@unarelith.net>
* Copyright (C) 2019-2020 the OpenMiner contributors (see CONTRIBUTORS.md)
*
* This file is part of OpenMiner.
*
* OpenMiner is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* OpenMiner is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with OpenMiner; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*
* =====================================================================================
*/
#include "EngineConfig.hpp"
#include "Registry.hpp"
#include "ServerChunk.hpp"
#include "TerrainGenerator.hpp"
#include "World.hpp"
#include <glm/gtc/noise.hpp>
#include "FastNoise.hpp"
TerrainGenerator::TerrainGenerator(Heightmap &heightmap, const Dimension &dimension, s32 seed)
: m_biomeSampler(dimension, seed), m_heightmap(heightmap)
{
m_caveNoise.SetFrequency(1.0 / 128.0);
m_caveNoise.SetFractalOctaves(2);
setSeed(seed);
}
void TerrainGenerator::generate(ServerChunk &chunk) const {
fastNoiseGeneration(chunk);
}
void TerrainGenerator::fastNoiseGeneration(ServerChunk &chunk) const {
HeightmapChunk &heightmap = m_heightmap.getOrCreateChunk(chunk.x(), chunk.y());
Random_t rand;
rand.seed(chunk.x() + chunk.y() * CHUNK_WIDTH + chunk.z() * CHUNK_WIDTH * CHUNK_HEIGHT + 1337);
Chunk *topChunk = chunk.getSurroundingChunk(Chunk::Top);
for(int y = 0 ; y < CHUNK_DEPTH ; y++) {
for(int x = 0 ; x < CHUNK_WIDTH ; x++) {
u16 biomeIndex = m_biomeSampler.getBiomeIndexAt(x + chunk.x() * CHUNK_WIDTH, y + chunk.y() * CHUNK_DEPTH);
const Biome &biome = Registry::getInstance().getBiome(biomeIndex);
// Land height
double h = heightmap.landHeightAt(x, y);
// Land blocks
for(int z = 0 ; z < CHUNK_HEIGHT ; z++) {
// Are we above "ground" level?
if(z + chunk.z() * CHUNK_HEIGHT > h) {
// If we are not yet up to sea level, fill with water blocks
if (z + chunk.z() * CHUNK_HEIGHT < SEALEVEL) {
chunk.setBlockRaw(x, y, z, biome.getLiquidBlockID());
}
// Otherwise we are in the air
else if (chunk.getBlock(x, y, z) == 0 && z == CHUNK_HEIGHT - 1) {
// Add sunlight at the top of the chunk if possible
chunk.lightmap().addSunlight(x, y, z, 15);
}
}
else {
bool isGeneratingTopBlock = z + chunk.z() * CHUNK_HEIGHT == h && z + chunk.z() * CHUNK_HEIGHT > SEALEVEL - 1;
if (isGeneratingTopBlock)
chunk.setBlockRaw(x, y, z, biome.getTopBlockID());
else if (z + chunk.z() * CHUNK_HEIGHT <= SEALEVEL - 1 && h < SEALEVEL && z + chunk.z() * CHUNK_HEIGHT > h - 3)
chunk.setBlockRaw(x, y, z, biome.getBeachBlockID());
else if (z + chunk.z() * CHUNK_HEIGHT > h - 3)
chunk.setBlockRaw(x, y, z, biome.getGroundBlockID());
else
chunk.setBlockRaw(x, y, z, biome.getDeepBlockID());
// Caves
generateCaves(chunk, x, y, z);
// Populate ores.
generateOres(chunk, x, y, z, biome, rand);
// Generate trees, flora and portals
if (isGeneratingTopBlock && chunk.getBlock(x, y, z)) {
// Try to place a tree
bool placedTree = tryPlaceTree(chunk, x, y, z + 1, biome, rand);
// Otherwise try to place flora.
bool placedFlora = false;
if (!placedTree)
placedFlora = tryPlaceFlora(chunk, x, y, z + 1, biome, rand);
// Or a portal
if (!placedTree && !placedFlora)
tryPlacePortal(chunk, x, y, z + 1, biome, rand);
}
}
if (topChunk && topChunk->isInitialized()) {
int sunlightLevel = topChunk->lightmap().getSunlight(x, y, 0);
if (sunlightLevel) {
chunk.lightmap().addSunlight(x, y, CHUNK_HEIGHT - 1, sunlightLevel);
}
}
}
}
}
}
inline bool TerrainGenerator::tryPlaceTree(ServerChunk &chunk, int x, int y, int z, const Biome &biome, Random_t &rand) const {
if (chunk.getBlock(x, y, z - 1) == biome.getTopBlockID()) {
for (const PlacementEntry::Tree &treePlacement : biome.getTrees()) {
if (!rand.get<bool>(treePlacement.probability))
continue;
const Tree &tree = Registry::getInstance().getTree(treePlacement.treeID);
// Trunk
int h = rand.get(tree.trunkMinHeight(), tree.trunkMaxHeight());
for (int i = 0; i < h; i++) {
chunk.setBlockRaw(x, y, z + i, tree.getLogBlockID());
}
// Leaves
if (tree.hasLeaves()) {
for (int iz = -3; iz <= 3; iz++) {
for (int iy = -3; iy <= 3; iy++) {
for (int ix = -3; ix <= 3; ix++) {
u16 block = chunk.getBlock(x + ix, y + iy, z + h + iz);
if (ix * ix + iy * iy + iz * iz < 8 + rand.get(0, 1) && !block) {
chunk.setBlockRaw(x + ix, y + iy, z + h + iz, tree.getLeavesBlockID());
// FIXME: This is a temporary fix for the second part of #41
chunk.lightmap().setSunlight(x + ix, y + iy, z + h + iz, 0);
}
}
}
}
}
return true;
}
}
return false;
}
inline bool TerrainGenerator::tryPlaceFlora(ServerChunk &chunk, int x, int y, int z, const Biome &biome, Random_t &rand) const {
for (const PlacementEntry::Flora &flora : biome.getFlora()) {
if (chunk.getBlock(x, y, z - 1) != flora.spawnsOnBlockID)
continue;
if (!rand.get<bool>(flora.probability))
continue;
chunk.setBlockRaw(x, y, z, flora.blockID);
return true;
}
return false;
}
inline bool TerrainGenerator::tryPlacePortal(ServerChunk &chunk, int x, int y, int z, const Biome &biome, Random_t &rand) const {
// FIXME: This is a temporary portal generation
// This code should be replaced by a proper "feature" implementation
// which will also allow making stuff like villages easier
if (chunk.getBlock(x, y, z - 1) == biome.getTopBlockID() && rand.get<bool>(0.0002)) {
for (int ix = 0 ; ix < 4 ; ++ix) {
for (int iz = 0 ; iz < 5 ; ++iz) {
if (ix == 0 || iz == 0 || ix == 3 || iz == 4)
chunk.setBlockRaw(x + ix, y, z + iz, biome.getPortalFrameBlockID());
else
chunk.setBlockRaw(x + ix, y, z + iz, biome.getPortalBlockID());
}
}
return true;
}
return false;
}
inline void TerrainGenerator::generateOres(ServerChunk &chunk, int x, int y, int z, const Biome &biome, Random_t &rand) const {
// TODO: Like trees, ores should be able to seamlessly cross chunk boundaries.
// This could be achieved either by setting up a generation pipeline with stages,
// processing neighboring chunks' ores every time, or generating them with noise.
for (const PlacementEntry::Ore &ore : biome.getOres()) {
if (!rand.get<bool>(ore.probability))
continue;
if (ore.genType == PlacementEntry::Ore::Gen::RandomWalk)
randomWalkOrePlace(chunk, x, y, z, rand, ore.blockID, biome.getDeepBlockID(), ore.size);
else if (ore.genType == PlacementEntry::Ore::Gen::FloodFill)
oreFloodFill(chunk, x, y, z, biome.getDeepBlockID(), ore.blockID, ore.size, rand);
break;
}
}
inline void TerrainGenerator::randomWalkOrePlace(ServerChunk &chunk, int x, int y, int z, Random_t &rand, u16 oreBlock, u16 deepBlock, int size) const {
if (size <= 0 || chunk.getBlock(x, y, z) != deepBlock)
return;
chunk.setBlockRaw(x, y, z, oreBlock);
int direction = rand.get(1, 6);
switch(direction) {
case 1:
randomWalkOrePlace(chunk, x + 1, y, z, rand, oreBlock, deepBlock, size - 1);
break;
case 2:
randomWalkOrePlace(chunk, x - 1, y, z, rand, oreBlock, deepBlock, size - 1);
break;
case 3:
randomWalkOrePlace(chunk, x, y + 1, z, rand, oreBlock, deepBlock, size - 1);
break;
case 4:
randomWalkOrePlace(chunk, x, y - 1, z, rand, oreBlock, deepBlock, size - 1);
break;
case 5:
randomWalkOrePlace(chunk, x, y, z + 1, rand, oreBlock, deepBlock, size - 1);
break;
case 6:
randomWalkOrePlace(chunk, x, y, z - 1, rand, oreBlock, deepBlock, size - 1);
break;
}
}
void TerrainGenerator::oreFloodFill(ServerChunk &chunk, double x, double y, double z, u16 toReplace, u16 replaceWith, int depth, Random_t &rand) const {
if (depth < 0) return;
if (chunk.getBlock(x, y, z) == replaceWith) return;
if (chunk.getBlock(x, y, z) == toReplace)
chunk.setBlockRaw(x, y, z, replaceWith);
oreFloodFill(chunk, x + 1, y, z, toReplace, replaceWith, depth - 1, rand);
oreFloodFill(chunk, x - 1, y, z, toReplace, replaceWith, depth - 1, rand);
oreFloodFill(chunk, x, y + 1, z, toReplace, replaceWith, depth - 1, rand);
oreFloodFill(chunk, x, y - 1, z, toReplace, replaceWith, depth - 1, rand);
oreFloodFill(chunk, x, y, z + 1, toReplace, replaceWith, depth - 1, rand);
oreFloodFill(chunk, x, y, z - 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x + 1, y + 1, z + 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x + 1, y + 1, z - 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x + 1, y - 1, z + 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x + 1, y - 1, z - 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x - 1, y + 1, z + 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x - 1, y + 1, z - 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x - 1, y - 1, z + 1, toReplace, replaceWith, depth - 1, rand);
if (rand.get<bool>(1.f / 15.f))
oreFloodFill(chunk, x - 1, y - 1, z - 1, toReplace, replaceWith, depth - 1, rand);
}
inline void TerrainGenerator::generateCavesOld(ServerChunk &chunk, int x, int y, int z, int h, HeightmapChunk &heightmap) const {
float n2 = noise2d(-(x + chunk.x() * CHUNK_WIDTH) / 256.0, (y + chunk.y() * CHUNK_DEPTH) / 256.0, 8, 0.3) * 4;
float r2 = noise3d_abs(-(x + chunk.x() * CHUNK_WIDTH) / 512.0f, (z + chunk.z() * CHUNK_HEIGHT) / 512.0f, (y + chunk.y() * CHUNK_DEPTH) / 512.0f, 4, 0.1);
float r3 = noise3d_abs(-(x + chunk.x() * CHUNK_WIDTH) / 512.0f, (z + chunk.z() * CHUNK_HEIGHT) / 128.0f, (y + chunk.y() * CHUNK_DEPTH) / 512.0f, 4, 1);
float r4 = n2 * 5 + r2 * r3 * 20;
if (r4 > 6 && r4 < 8 && h > SEALEVEL) {
chunk.setBlockRaw(x, y, z - 1, 0);
chunk.setBlockRaw(x, y, z, 0);
chunk.setBlockRaw(x, y, z + 1, 0);
s32 landHeight = heightmap.landHeightAt(x, y);
if (landHeight == z + 1 || landHeight == z || landHeight == z - 1)
heightmap.setLandHeight(x, y, z - 2);
}
}
inline void TerrainGenerator::generateCaves(ServerChunk &chunk, int x, int y, int z) const {
int rx = x + chunk.x() * CHUNK_WIDTH;
int ry = y + chunk.y() * CHUNK_DEPTH;
int rz = z + chunk.z() * CHUNK_HEIGHT;
// Density map (not textured image)
double n1 = m_caveNoise.GetSimplexFractal(rx, ry, rz);
double n2 = m_caveNoise.GetSimplexFractal(rx, ry + 88.0, rz);
double finalNoise = n1 * n1 + n2 * n2;
if (finalNoise < 0.02) {
// FIXME: Update heightmap
chunk.setBlockRaw(x, y, z, 0);
}
}
inline float TerrainGenerator::noise2d(double x, double y, int octaves, float persistence) {
float sum = 0;
float strength = 1.0;
float scale = 1.0;
for(int i = 0 ; i < octaves ; i++) {
sum += strength * glm::simplex(glm::vec2{x, y} * scale);
scale *= 2.0;
strength *= persistence;
}
return sum;
}
inline float TerrainGenerator::noise3d_abs(double x, double y, double z, int octaves, float persistence) {
float sum = 0;
float strength = 1.0;
float scale = 1.0;
for(int i = 0 ; i < octaves ; i++) {
sum += strength * fabsf(glm::simplex(glm::vec3{x, y, z} * scale));
scale *= 2.0;
strength *= persistence;
}
return sum;
}