2018-03-05 21:20:54 +00:00
|
|
|
#include "layers.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
|
|
|
|
static void oceanRndInit(OceanRnd *rnd, int64_t seed);
|
|
|
|
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void initAddBiome(int id, int tempCat, int biometype, float temp, float height)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (id & (~0xff)) return;
|
2018-03-05 21:20:54 +00:00
|
|
|
biomes[id].id = id;
|
|
|
|
biomes[id].type = biometype;
|
|
|
|
biomes[id].temp = temp;
|
|
|
|
biomes[id].height = height;
|
|
|
|
biomes[id].tempCat = tempCat;
|
|
|
|
}
|
|
|
|
|
|
|
|
void createMutation(int id)
|
|
|
|
{
|
|
|
|
biomes[id+128] = biomes[id];
|
|
|
|
biomes[id+128].id = id+128;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* initBiomes() has to be called before any of the generators can be used */
|
|
|
|
void initBiomes()
|
|
|
|
{
|
|
|
|
int i;
|
2018-07-28 16:57:29 +02:00
|
|
|
for (i = 0; i < 256; i++) biomes[i].id = none;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
const double hDefault = 0.1, hShallowWaters = -0.5, hOceans = -1.0, hDeepOceans = -1.8, hLowPlains = 0.125;
|
|
|
|
const double hMidPlains = 0.2, hLowHills = 0.45, hHighPlateaus = 1.5, hMidHills = 1.0, hShores = 0.0;
|
|
|
|
const double hRockyWaters = 0.1, hLowIslands = 0.2, hPartiallySubmerged = -0.2;
|
|
|
|
|
|
|
|
initAddBiome(ocean, Oceanic, Ocean, 0.5, hOceans);
|
|
|
|
initAddBiome(plains, Lush, Plains, 0.8, hDefault);
|
|
|
|
initAddBiome(desert, Warm, Desert, 2.0, hLowPlains);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(mountains, Lush, Hills, 0.2, hMidHills);
|
2018-03-05 21:20:54 +00:00
|
|
|
initAddBiome(forest, Lush, Forest, 0.7, hDefault);
|
|
|
|
initAddBiome(taiga, Lush, Taiga, 0.25, hMidPlains);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(swamp, Lush, Swamp, 0.8, hPartiallySubmerged);
|
2018-03-05 21:20:54 +00:00
|
|
|
initAddBiome(river, Lush, River, 0.5, hShallowWaters);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(nether, Warm, Hell, 2.0, hDefault);
|
|
|
|
initAddBiome(the_end, Lush, Sky, 0.5, hDefault);
|
|
|
|
initAddBiome(frozen_ocean, Oceanic, Ocean, 0.0, hOceans);
|
|
|
|
initAddBiome(frozen_river, Cold, River, 0.0, hShallowWaters);
|
|
|
|
initAddBiome(snowy_tundra, Cold, Snow, 0.0, hLowPlains);
|
|
|
|
initAddBiome(snowy_mountains, Cold, Snow, 0.0, hLowHills);
|
|
|
|
initAddBiome(mushroom_fields, Lush, MushroomIsland, 0.9, hLowIslands);
|
|
|
|
initAddBiome(mushroom_field_shore, Lush, MushroomIsland, 0.9, hShores);
|
2018-03-05 21:20:54 +00:00
|
|
|
initAddBiome(beach, Lush, Beach, 0.8, hShores);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(desert_hills, Warm, Desert, 2.0, hLowHills);
|
|
|
|
initAddBiome(wooded_hills, Lush, Forest, 0.7, hLowHills);
|
|
|
|
initAddBiome(taiga_hills, Lush, Taiga, 0.25, hLowHills);
|
|
|
|
initAddBiome(mountain_edge, Lush, Hills, 0.2, hMidHills);
|
2018-03-05 21:20:54 +00:00
|
|
|
initAddBiome(jungle, Lush, Jungle, 0.95, hDefault);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(jungle_hills, Lush, Jungle, 0.95, hLowHills);
|
|
|
|
initAddBiome(jungle_edge, Lush, Jungle, 0.95, hDefault);
|
|
|
|
initAddBiome(deep_ocean, Oceanic, Ocean, 0.5, hDeepOceans);
|
|
|
|
initAddBiome(stone_shore, Lush, StoneBeach, 0.2, hRockyWaters);
|
|
|
|
initAddBiome(snowy_beach, Cold, Beach, 0.05, hShores);
|
|
|
|
initAddBiome(birch_forest, Lush, Forest, 0.6, hDefault);
|
|
|
|
initAddBiome(birch_forest_hills, Lush, Forest, 0.6, hLowHills);
|
|
|
|
initAddBiome(dark_forest, Lush, Forest, 0.7, hDefault);
|
|
|
|
initAddBiome(snowy_taiga, Cold, Taiga, -0.5, hMidPlains);
|
|
|
|
initAddBiome(snowy_taiga_hills, Cold, Taiga, -0.5, hLowHills);
|
|
|
|
initAddBiome(giant_tree_taiga, Lush, Taiga, 0.3, hMidPlains);
|
|
|
|
initAddBiome(giant_tree_taiga_hills, Lush, Taiga, 0.3, hLowHills);
|
|
|
|
initAddBiome(wooded_mountains, Lush, Hills, 0.2, hMidHills);
|
2018-03-05 21:20:54 +00:00
|
|
|
initAddBiome(savanna, Warm, Savanna, 1.2, hLowPlains);
|
2019-05-06 23:14:25 +02:00
|
|
|
initAddBiome(savanna_plateau, Warm, Savanna, 1.0, hHighPlateaus);
|
|
|
|
initAddBiome(badlands, Warm, Mesa, 2.0, hDefault);
|
|
|
|
initAddBiome(wooded_badlands_plateau, Warm, Mesa, 2.0, hHighPlateaus);
|
|
|
|
initAddBiome(badlands_plateau, Warm, Mesa, 2.0, hHighPlateaus);
|
|
|
|
|
|
|
|
initAddBiome(small_end_islands, Lush, Sky, 0.5, hDefault);
|
|
|
|
initAddBiome(end_midlands, Lush, Sky, 0.5, hDefault);
|
|
|
|
initAddBiome(end_highlands, Lush, Sky, 0.5, hDefault);
|
|
|
|
initAddBiome(end_barrens, Lush, Sky, 0.5, hDefault);
|
|
|
|
initAddBiome(warm_ocean, Oceanic, Ocean, 0.5, hOceans);
|
|
|
|
initAddBiome(lukewarm_ocean, Oceanic, Ocean, 0.5, hOceans);
|
|
|
|
initAddBiome(cold_ocean, Oceanic, Ocean, 0.5, hOceans);
|
|
|
|
initAddBiome(deep_warm_ocean, Oceanic, Ocean, 0.5, hDeepOceans);
|
|
|
|
initAddBiome(deep_lukewarm_ocean, Oceanic, Ocean, 0.5, hDeepOceans);
|
|
|
|
initAddBiome(deep_cold_ocean, Oceanic, Ocean, 0.5, hDeepOceans);
|
|
|
|
initAddBiome(deep_frozen_ocean, Oceanic, Ocean, 0.5, hDeepOceans);
|
|
|
|
|
|
|
|
initAddBiome(the_void, Void, Void, 0.5, 0);
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
createMutation(plains);
|
|
|
|
createMutation(desert);
|
2019-05-06 23:14:25 +02:00
|
|
|
createMutation(mountains);
|
2018-03-05 21:20:54 +00:00
|
|
|
createMutation(forest);
|
|
|
|
createMutation(taiga);
|
2019-05-06 23:14:25 +02:00
|
|
|
createMutation(swamp);
|
|
|
|
createMutation(snowy_tundra);
|
2018-03-05 21:20:54 +00:00
|
|
|
createMutation(jungle);
|
2019-05-06 23:14:25 +02:00
|
|
|
createMutation(jungle_edge);
|
|
|
|
createMutation(birch_forest);
|
|
|
|
createMutation(birch_forest_hills);
|
|
|
|
createMutation(dark_forest);
|
|
|
|
createMutation(snowy_taiga);
|
|
|
|
createMutation(giant_tree_taiga);
|
|
|
|
createMutation(giant_tree_taiga_hills);
|
|
|
|
createMutation(wooded_mountains);
|
2018-03-05 21:20:54 +00:00
|
|
|
createMutation(savanna);
|
2019-05-06 23:14:25 +02:00
|
|
|
createMutation(savanna_plateau);
|
|
|
|
createMutation(badlands);
|
|
|
|
createMutation(wooded_badlands_plateau);
|
|
|
|
createMutation(badlands_plateau);
|
|
|
|
|
|
|
|
initAddBiome(bamboo_jungle, Lush, Jungle, 0.95, hDefault);
|
|
|
|
initAddBiome(bamboo_jungle_hills, Lush, Jungle, 0.95, hLowHills);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
void setWorldSeed(Layer *layer, int64_t seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (layer->p2 != NULL && layer->getMap != mapHills)
|
2018-03-10 21:45:57 +00:00
|
|
|
setWorldSeed(layer->p2, seed);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (layer->p != NULL)
|
2018-03-10 21:45:57 +00:00
|
|
|
setWorldSeed(layer->p, seed);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (layer->oceanRnd != NULL)
|
2018-07-08 15:13:09 +01:00
|
|
|
oceanRndInit(layer->oceanRnd, seed);
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->worldSeed = seed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->worldSeed *= layer->worldSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->worldSeed += layer->baseSeed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->worldSeed *= layer->worldSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->worldSeed += layer->baseSeed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->worldSeed *= layer->worldSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->worldSeed += layer->baseSeed;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapNull(Layer *l, int * __restrict out, int x, int z, int w, int h)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-07-09 02:21:34 +01:00
|
|
|
void mapSkip(Layer *l, int * __restrict out, int x, int z, int w, int h)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p == NULL)
|
2018-07-09 02:21:34 +01:00
|
|
|
{
|
|
|
|
printf("mapSkip() requires a non-null parent layer.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
l->p->getMap(l->p, out, x, z, w, h);
|
|
|
|
}
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
void mapIsland(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
register int x, z;
|
2018-03-21 21:30:55 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t ws = l->worldSeed;
|
2018-07-04 17:28:27 +01:00
|
|
|
const int64_t ss = ws * (ws * 6364136223846793005LL + 1442695040888963407LL);
|
2018-03-21 21:30:55 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t chunkX = (int64_t)(x + areaX);
|
|
|
|
const int64_t chunkZ = (int64_t)(z + areaZ);
|
|
|
|
register int64_t cs = ss;
|
2018-03-21 21:30:55 +00:00
|
|
|
cs += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 21:30:55 +00:00
|
|
|
cs += chunkZ;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 21:30:55 +00:00
|
|
|
cs += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 21:30:55 +00:00
|
|
|
cs += chunkZ;
|
|
|
|
|
|
|
|
out[x + z*areaWidth] = (cs >> 24) % 10 == 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (areaX > -areaWidth && areaX <= 0 && areaZ > -areaHeight && areaZ <= 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[-areaX + -areaZ * areaWidth] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
// FIXME: currently SIMD only works properly for certain sizes
|
2018-04-25 00:36:45 +01:00
|
|
|
#if defined USE_SIMD && defined __AVX2__
|
2018-07-28 16:57:29 +02:00
|
|
|
|
|
|
|
void mapZoom(Layer *l, int* __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
int pWidth = (areaWidth>>1)+2, pHeight = (areaHeight>>1)+1;
|
2018-04-25 00:36:45 +01:00
|
|
|
|
|
|
|
l->p->getMap(l->p, out, areaX>>1, areaZ>>1, pWidth, pHeight+1);
|
|
|
|
|
2018-04-21 22:29:42 +02:00
|
|
|
__m256i (*selectRand)(__m256i* cs, int ws, __m256i a1, __m256i a2, __m256i a3, __m256i a4) = (l->p->getMap == mapIsland) ? select8Random4 : select8ModeOrRandom;
|
2018-04-25 00:36:45 +01:00
|
|
|
int newWidth = (areaWidth+10)&0xFFFFFFFE;//modified to ignore ends
|
2018-04-21 22:29:42 +02:00
|
|
|
int x, z;
|
|
|
|
__m256i cs, a, b, a1, b1, toBuf1, toBuf2, aSuf;
|
|
|
|
__m256i mask1 = _mm256_setr_epi32(0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0), mask2 = _mm256_setr_epi32(0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF);
|
|
|
|
__m256i shuffle = _mm256_setr_epi32(0, 4, 1, 5, 2, 6, 3, 7);
|
|
|
|
int pX = areaX&0xFFFFFFFE;
|
|
|
|
__m256i xs = _mm256_set_epi32(pX+14, pX+12, pX+10, pX+8, pX+6, pX+4, pX+2, pX), zs;
|
|
|
|
__m256i v2 = _mm256_set1_epi32(2), v16 = _mm256_set1_epi32(16);
|
2020-03-15 16:09:06 +01:00
|
|
|
int* buf = (int*) malloc((newWidth+1)*((areaHeight+2)|1)*sizeof(*buf));
|
2018-04-21 22:29:42 +02:00
|
|
|
int* idx = buf;
|
|
|
|
int* outIdx = out;
|
|
|
|
//z first!
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < pWidth-1; x += 8)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
a = _mm256_loadu_si256((__m256i*)(outIdx));//0, 0
|
|
|
|
b = _mm256_loadu_si256((__m256i*)(outIdx+1));//1, 0
|
|
|
|
zs = _mm256_set1_epi32(areaZ&0xFFFFFFFE);
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < pHeight; z++)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
cs = set8ChunkSeeds(l->worldSeed, xs, zs);
|
|
|
|
outIdx += pWidth;
|
|
|
|
a1 = _mm256_loadu_si256((__m256i*)(outIdx));//0, 1
|
|
|
|
b1 = _mm256_loadu_si256((__m256i*)(outIdx+1));//1, 1
|
|
|
|
toBuf1 = _mm256_permutevar8x32_epi32(select8Random2(&cs, l->worldSeed, a, a1), shuffle);
|
|
|
|
toBuf2 = _mm256_permutevar8x32_epi32(select8Random2(&cs, l->worldSeed, a, b), shuffle);
|
|
|
|
aSuf = _mm256_permutevar8x32_epi32(a, shuffle);
|
|
|
|
_mm256_maskstore_epi32(idx, mask1, aSuf);
|
|
|
|
_mm256_maskstore_epi32(idx+1, mask1, toBuf2);
|
|
|
|
_mm256_maskstore_epi32(idx+7, mask2, aSuf);
|
|
|
|
_mm256_maskstore_epi32(idx+8, mask2, toBuf2);
|
|
|
|
idx += newWidth;
|
|
|
|
toBuf2 = _mm256_permutevar8x32_epi32(selectRand(&cs, l->worldSeed, a, b, a1, b1), shuffle);
|
|
|
|
_mm256_maskstore_epi32(idx, mask1, toBuf1);
|
|
|
|
_mm256_maskstore_epi32(idx+1, mask1, toBuf2);
|
|
|
|
_mm256_maskstore_epi32(idx+7, mask2, toBuf1);
|
|
|
|
_mm256_maskstore_epi32(idx+8, mask2, toBuf2);
|
|
|
|
idx += newWidth;
|
|
|
|
a = a1;
|
|
|
|
b = b1;
|
|
|
|
zs = _mm256_add_epi32(zs, v2);
|
|
|
|
}
|
|
|
|
outIdx += 8-pHeight*pWidth;
|
|
|
|
idx += 16-pHeight*2*newWidth;
|
|
|
|
xs = _mm256_add_epi32(xs, v16);
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-04-21 22:29:42 +02:00
|
|
|
{
|
|
|
|
memcpy(&out[z*areaWidth], &buf[(z + (areaZ & 1))*newWidth + (areaX & 1)], areaWidth*sizeof(int));
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
|
2018-04-25 00:36:45 +01:00
|
|
|
#elif defined USE_SIMD && defined __SSE4_2__
|
2018-07-28 16:57:29 +02:00
|
|
|
|
|
|
|
void mapZoom(Layer *l, int* __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
int pWidth = (areaWidth>>1)+2, pHeight = (areaHeight>>1)+1;
|
2018-04-25 00:36:45 +01:00
|
|
|
|
|
|
|
l->p->getMap(l->p, out, areaX>>1, areaZ>>1, pWidth, pHeight+1);
|
|
|
|
|
2018-04-21 22:29:42 +02:00
|
|
|
__m128i (*selectRand)(__m128i* cs, int ws, __m128i a1, __m128i a2, __m128i a3, __m128i a4) = (l->p->getMap == mapIsland) ? select4Random4 : select4ModeOrRandom;
|
|
|
|
int newWidth = areaWidth+6&0xFFFFFFFE;//modified to ignore ends
|
|
|
|
int x, z;
|
|
|
|
__m128i cs, a, b, a1, b1, toBuf1, toBuf2, aSuf;
|
|
|
|
__m128i mask1 = _mm_setr_epi32(0xFFFFFFFF, 0x0, 0xFFFFFFFF, 0x0), mask2 = _mm_setr_epi32(0x0, 0xFFFFFFFF, 0x0, 0xFFFFFFFF);
|
|
|
|
int pX = areaX&0xFFFFFFFE;
|
|
|
|
__m128i xs = _mm_set_epi32(pX+6, pX+4, pX+2, pX), zs;
|
|
|
|
__m128i v2 = _mm_set1_epi32(2), v8 = _mm_set1_epi32(8);
|
2020-03-15 16:09:06 +01:00
|
|
|
int* buf = (int*) malloc((newWidth+1)*(areaHeight+2|1)*sizeof(*buf));
|
2018-04-21 22:29:42 +02:00
|
|
|
int* idx = buf;
|
|
|
|
int* outIdx = out;
|
|
|
|
//z first!
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < pWidth-1; x += 4)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
a = _mm_loadu_si128((__m128i_u*)(outIdx));//0, 0
|
|
|
|
b = _mm_loadu_si128((__m128i_u*)(outIdx+1));//1, 0
|
|
|
|
zs = _mm_set1_epi32(areaZ&0xFFFFFFFE);
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < pHeight; z++)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
cs = set4ChunkSeeds(l->worldSeed, xs, zs);
|
|
|
|
outIdx += pWidth;
|
|
|
|
a1 = _mm_loadu_si128((__m128i_u*)(outIdx));//0, 1
|
|
|
|
b1 = _mm_loadu_si128((__m128i_u*)(outIdx+1));//1, 1
|
|
|
|
toBuf1 = _mm_shuffle_epi32(select4Random2(&cs, l->worldSeed, a, a1), 0xD8);//11011000->3120->1324
|
|
|
|
toBuf2 = _mm_shuffle_epi32(select4Random2(&cs, l->worldSeed, a, b), 0xD8);
|
|
|
|
aSuf = _mm_shuffle_epi32(a, 0xD8);
|
|
|
|
_mm_maskmoveu_si128(aSuf, mask1, (char*)(idx));
|
|
|
|
_mm_maskmoveu_si128(toBuf2, mask1, (char*)(idx+1));
|
|
|
|
_mm_maskmoveu_si128(aSuf, mask2, (char*)(idx+3));
|
|
|
|
_mm_maskmoveu_si128(toBuf2, mask2, (char*)(idx+4));
|
|
|
|
idx += newWidth;
|
|
|
|
toBuf2 = _mm_shuffle_epi32(selectRand(&cs, l->worldSeed, a, b, a1, b1), 0xD8);
|
|
|
|
_mm_maskmoveu_si128(toBuf1, mask1, (char*)(idx));
|
|
|
|
_mm_maskmoveu_si128(toBuf2, mask1, (char*)(idx+1));
|
|
|
|
_mm_maskmoveu_si128(toBuf1, mask2, (char*)(idx+3));
|
|
|
|
_mm_maskmoveu_si128(toBuf2, mask2, (char*)(idx+4));
|
|
|
|
idx += newWidth;
|
|
|
|
a = a1;
|
|
|
|
b = b1;
|
|
|
|
zs = _mm_add_epi32(zs, v2);
|
|
|
|
}
|
|
|
|
outIdx += 4-pHeight*pWidth;
|
|
|
|
idx += 8-pHeight*2*newWidth;
|
|
|
|
xs = _mm_add_epi32(xs, v8);
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-04-21 22:29:42 +02:00
|
|
|
{
|
|
|
|
memcpy(&out[z*areaWidth], &buf[(z + (areaZ & 1))*newWidth + (areaX & 1)], areaWidth*sizeof(int));
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
|
2018-04-21 22:29:42 +02:00
|
|
|
#else
|
2018-07-28 16:57:29 +02:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void mapZoom(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX >> 1;
|
|
|
|
int pZ = areaZ >> 1;
|
2020-03-15 16:09:06 +01:00
|
|
|
int pWidth = ((areaX + areaWidth ) >> 1) - pX + 1;
|
|
|
|
int pHeight = ((areaZ + areaHeight) >> 1) - pZ + 1;
|
2018-03-05 21:20:54 +00:00
|
|
|
int x, z;
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
//printf("[%d %d] [%d %d]\n", pX, pZ, pWidth, pHeight);
|
2018-03-05 21:20:54 +00:00
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
int newWidth = (pWidth) << 1;
|
|
|
|
int newHeight = (pHeight) << 1;
|
2018-03-05 21:20:54 +00:00
|
|
|
int idx, a, b;
|
|
|
|
int *buf = (int *)malloc((newWidth+1)*(newHeight+1)*sizeof(*buf));
|
|
|
|
|
2018-03-21 18:09:47 +00:00
|
|
|
const int ws = (int)l->worldSeed;
|
|
|
|
const int ss = ws * (ws * 1284865837 + 4150755663);
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
for (z = 0; z < pHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
idx = (z << 1) * newWidth;
|
|
|
|
a = out[(z+0)*pWidth];
|
|
|
|
b = out[(z+1)*pWidth];
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
for (x = 0; x < pWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int a1 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int b1 = out[x+1 + (z+1)*pWidth];
|
2018-03-21 18:09:47 +00:00
|
|
|
|
|
|
|
const int chunkX = (x + pX) << 1;
|
|
|
|
const int chunkZ = (z + pZ) << 1;
|
|
|
|
|
|
|
|
register int cs = ss;
|
|
|
|
cs += chunkX;
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += chunkZ;
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += chunkX;
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += chunkZ;
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
buf[idx] = a;
|
2018-03-21 18:09:47 +00:00
|
|
|
buf[idx + newWidth] = (cs >> 24) & 1 ? b : a;
|
2018-03-05 21:20:54 +00:00
|
|
|
idx++;
|
2018-03-21 18:09:47 +00:00
|
|
|
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += ws;
|
|
|
|
buf[idx] = (cs >> 24) & 1 ? a1 : a;
|
|
|
|
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p->getMap == mapIsland)
|
2018-03-21 18:09:47 +00:00
|
|
|
{
|
|
|
|
//selectRandom4
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += ws;
|
|
|
|
const int i = (cs >> 24) & 3;
|
|
|
|
buf[idx + newWidth] = i==0 ? a : i==1 ? a1 : i==2 ? b : b1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//selectModeOrRandom
|
2018-03-21 21:30:55 +00:00
|
|
|
if (a1 == b && b == b1) buf[idx + newWidth] = a1;
|
|
|
|
else if (a == a1 && a == b ) buf[idx + newWidth] = a;
|
|
|
|
else if (a == a1 && a == b1) buf[idx + newWidth] = a;
|
|
|
|
else if (a == b && a == b1) buf[idx + newWidth] = a;
|
|
|
|
else if (a == a1 && b != b1) buf[idx + newWidth] = a;
|
|
|
|
else if (a == b && a1 != b1) buf[idx + newWidth] = a;
|
|
|
|
else if (a == b1 && a1 != b ) buf[idx + newWidth] = a;
|
|
|
|
else if (a1 == b && a != b1) buf[idx + newWidth] = a1;
|
|
|
|
else if (a1 == b1 && a != b ) buf[idx + newWidth] = a1;
|
|
|
|
else if (b == b1 && a != a1) buf[idx + newWidth] = b;
|
2018-03-21 18:09:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
cs *= cs * 1284865837 + 4150755663;
|
|
|
|
cs += ws;
|
|
|
|
const int i = (cs >> 24) & 3;
|
|
|
|
buf[idx + newWidth] = i==0 ? a : i==1 ? a1 : i==2 ? b : b1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
idx++;
|
|
|
|
a = a1;
|
|
|
|
b = b1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
memcpy(&out[z*areaWidth], &buf[(z + (areaZ & 1))*newWidth + (areaX & 1)], areaWidth*sizeof(int));
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
2018-04-21 22:29:42 +02:00
|
|
|
#endif
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
void mapAddIsland(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t ws = l->worldSeed;
|
2018-07-04 17:28:27 +01:00
|
|
|
const int64_t ss = ws * (ws * 6364136223846793005LL + 1442695040888963407LL);
|
2018-03-21 18:09:47 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v00 = out[x+0 + (z+0)*pWidth];
|
|
|
|
int v20 = out[x+2 + (z+0)*pWidth];
|
|
|
|
int v02 = out[x+0 + (z+2)*pWidth];
|
|
|
|
int v22 = out[x+2 + (z+2)*pWidth];
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == 0 && (v00 != 0 || v20 != 0 || v02 != 0 || v22 != 0))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-11 15:36:48 +01:00
|
|
|
const int64_t chunkX = (int64_t)(x + areaX);
|
|
|
|
const int64_t chunkZ = (int64_t)(z + areaZ);
|
|
|
|
register int64_t cs = ss;
|
|
|
|
cs += chunkX;
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += chunkZ;
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += chunkX;
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += chunkZ;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
int v = 1;
|
2018-07-11 15:36:48 +01:00
|
|
|
int inc = 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v00 != 0)
|
2018-07-11 15:36:48 +01:00
|
|
|
{
|
|
|
|
++inc; v = v00;
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += ws;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v20 != 0)
|
2018-07-11 15:36:48 +01:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (++inc == 1 || (cs & (1LL << 24)) == 0) v = v20;
|
2018-07-11 15:36:48 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += ws;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v02 != 0)
|
2018-07-11 15:36:48 +01:00
|
|
|
{
|
|
|
|
switch(++inc)
|
|
|
|
{
|
|
|
|
case 1: v = v02; break;
|
2018-07-28 16:57:29 +02:00
|
|
|
case 2: if ((cs & (1LL << 24)) == 0) v = v02; break;
|
|
|
|
default: if (((cs >> 24) % 3) == 0) v = v02;
|
2018-07-11 15:36:48 +01:00
|
|
|
}
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += ws;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v22 != 0)
|
2018-07-11 15:36:48 +01:00
|
|
|
{
|
|
|
|
switch(++inc)
|
|
|
|
{
|
|
|
|
case 1: v = v22; break;
|
2018-07-28 16:57:29 +02:00
|
|
|
case 2: if ((cs & (1LL << 24)) == 0) v = v22; break;
|
|
|
|
case 3: if (((cs >> 24) % 3) == 0) v = v22; break;
|
|
|
|
default: if ((cs & (3LL << 24)) == 0) v = v22;
|
2018-07-11 15:36:48 +01:00
|
|
|
}
|
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
cs += ws;
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if ((cs >> 24) % 3 == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = v;
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (v == 4)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = 4;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = 0;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (v11 > 0 && (v00 == 0 || v20 == 0 || v02 == 0 || v22 == 0))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
//setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-07-28 16:57:29 +02:00
|
|
|
//if (mcNextInt(l, 5) == 0)...
|
2018-03-21 18:09:47 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t chunkX = (int64_t)(x + areaX);
|
|
|
|
const int64_t chunkZ = (int64_t)(z + areaZ);
|
2018-03-21 18:09:47 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
register int64_t cs = ss;
|
2018-03-21 18:09:47 +00:00
|
|
|
cs += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 18:09:47 +00:00
|
|
|
cs += chunkZ;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 18:09:47 +00:00
|
|
|
cs += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
cs *= cs * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-21 18:09:47 +00:00
|
|
|
cs += chunkZ;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if ((cs >> 24) % 5 == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = (v11 == 4) ? 4 : 0;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapRemoveTooMuchOcean(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (out[x+1 + (z+0)*pWidth] != 0) continue;
|
|
|
|
if (out[x+2 + (z+1)*pWidth] != 0) continue;
|
|
|
|
if (out[x+0 + (z+1)*pWidth] != 0) continue;
|
|
|
|
if (out[x+1 + (z+2)*pWidth] != 0) continue;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (mcNextInt(l, 2) == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapAddSnow(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isShallowOcean(v11))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-03-10 21:45:57 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
int r = mcNextInt(l, 6);
|
2018-03-10 21:45:57 +00:00
|
|
|
int v;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (r == 0) v = 4;
|
|
|
|
else if (r <= 1) v = 3;
|
2018-03-05 21:20:54 +00:00
|
|
|
else v = 1;
|
|
|
|
|
|
|
|
out[x + z*areaWidth] = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void mapCoolWarm(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == 1)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int v21 = out[x+2 + (z+1)*pWidth];
|
|
|
|
int v01 = out[x+0 + (z+1)*pWidth];
|
|
|
|
int v12 = out[x+1 + (z+2)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v10 == 3 || v10 == 4 || v21 == 3 || v21 == 4 || v01 == 3 || v01 == 4 || v12 == 3 || v12 == 4)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
v11 = 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapHeatIce(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == 4)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int v21 = out[x+2 + (z+1)*pWidth];
|
|
|
|
int v01 = out[x+0 + (z+1)*pWidth];
|
|
|
|
int v12 = out[x+1 + (z+2)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v10 == 1 || v10 == 2 || v21 == 1 || v21 == 2 || v01 == 1 || v01 == 2 || v12 == 1 || v12 == 2)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
v11 = 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapSpecial(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
|
|
|
|
int x, z;
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v = out[x + z*areaWidth];
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v == 0) continue;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (mcNextInt(l, 13) == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
v |= (1 + mcNextInt(l, 15)) << 8 & 0xf00;
|
|
|
|
// 1 to 1 mapping so 'out' can be overwritten immediately
|
|
|
|
out[x + z*areaWidth] = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapAddMushroomIsland(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
|
|
|
// surrounded by ocean?
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == 0 && !out[x+0 + (z+0)*pWidth] && !out[x+2 + (z+0)*pWidth] && !out[x+0 + (z+2)*pWidth] && !out[x+2 + (z+2)*pWidth])
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-07-28 16:57:29 +02:00
|
|
|
if (mcNextInt(l, 100) == 0) {
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = mushroom_fields;
|
2018-03-05 21:20:54 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapDeepOcean(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[(x+1) + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isShallowOcean(v11))
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
|
|
|
// count adjacent oceans
|
|
|
|
int oceans = 0;
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isShallowOcean(out[(x+1) + (z+0)*pWidth])) oceans++;
|
|
|
|
if (isShallowOcean(out[(x+2) + (z+1)*pWidth])) oceans++;
|
|
|
|
if (isShallowOcean(out[(x+0) + (z+1)*pWidth])) oceans++;
|
|
|
|
if (isShallowOcean(out[(x+1) + (z+2)*pWidth])) oceans++;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (oceans > 3)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
switch (v11)
|
|
|
|
{
|
|
|
|
case warm_ocean:
|
|
|
|
v11 = deep_warm_ocean;
|
|
|
|
break;
|
|
|
|
case lukewarm_ocean:
|
|
|
|
v11 = deep_lukewarm_ocean;
|
|
|
|
break;
|
|
|
|
case ocean:
|
|
|
|
v11 = deep_ocean;
|
|
|
|
break;
|
|
|
|
case cold_ocean:
|
|
|
|
v11 = deep_cold_ocean;
|
|
|
|
break;
|
|
|
|
case frozen_ocean:
|
|
|
|
v11 = deep_frozen_ocean;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
v11 = deep_ocean;
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const int warmBiomes[] = {desert, desert, desert, savanna, savanna, plains};
|
2019-05-07 19:38:29 +02:00
|
|
|
const int lushBiomes[] = {forest, dark_forest, mountains, plains, birch_forest, swamp};
|
2019-05-06 23:14:25 +02:00
|
|
|
const int coldBiomes[] = {forest, mountains, taiga, plains};
|
2019-05-07 19:38:29 +02:00
|
|
|
const int snowBiomes[] = {snowy_tundra, snowy_tundra, snowy_tundra, snowy_taiga};
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
void mapBiome(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
|
|
|
|
int x, z;
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int idx = x + z*areaWidth;
|
|
|
|
int id = out[idx];
|
|
|
|
int hasHighBit = (id & 0xf00) >> 8;
|
|
|
|
id &= -0xf01;
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (getBiomeType(id) == Ocean || id == mushroom_fields)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[idx] = id;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
switch(id){
|
|
|
|
case Warm:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (hasHighBit) out[idx] = (mcNextInt(l, 3) == 0) ? badlands_plateau : wooded_badlands_plateau;
|
2018-03-05 21:20:54 +00:00
|
|
|
else out[idx] = warmBiomes[mcNextInt(l, 6)];
|
|
|
|
break;
|
|
|
|
case Lush:
|
2018-07-28 16:57:29 +02:00
|
|
|
if (hasHighBit) out[idx] = jungle;
|
2018-03-05 21:20:54 +00:00
|
|
|
else out[idx] = lushBiomes[mcNextInt(l, 6)];
|
|
|
|
break;
|
|
|
|
case Cold:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (hasHighBit) out[idx] = giant_tree_taiga;
|
2018-03-05 21:20:54 +00:00
|
|
|
else out[idx] = coldBiomes[mcNextInt(l, 4)];
|
|
|
|
break;
|
|
|
|
case Freezing:
|
|
|
|
out[idx] = snowBiomes[mcNextInt(l, 4)];
|
|
|
|
break;
|
|
|
|
default:
|
2019-05-06 23:14:25 +02:00
|
|
|
out[idx] = mushroom_fields;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-07 19:38:29 +02:00
|
|
|
const int lushBiomesBE[] = {forest, dark_forest, mountains, plains, plains, plains, birch_forest, swamp};
|
2018-11-29 04:32:12 +00:00
|
|
|
|
|
|
|
void mapBiomeBE(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
|
|
|
|
int x, z;
|
|
|
|
for (z = 0; z < areaHeight; z++)
|
|
|
|
{
|
|
|
|
for (x = 0; x < areaWidth; x++)
|
|
|
|
{
|
|
|
|
int idx = x + z*areaWidth;
|
|
|
|
int id = out[idx];
|
|
|
|
int hasHighBit = (id & 0xf00) >> 8;
|
|
|
|
id &= -0xf01;
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (getBiomeType(id) == Ocean || id == mushroom_fields)
|
2018-11-29 04:32:12 +00:00
|
|
|
{
|
|
|
|
out[idx] = id;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
|
|
|
|
|
|
|
switch(id){
|
|
|
|
case Warm:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (hasHighBit) out[idx] = (mcNextInt(l, 3) == 0) ? badlands_plateau : wooded_badlands_plateau;
|
2018-11-29 04:32:12 +00:00
|
|
|
else out[idx] = warmBiomes[mcNextInt(l, 6)];
|
|
|
|
break;
|
|
|
|
case Lush:
|
|
|
|
if (hasHighBit) out[idx] = jungle;
|
|
|
|
else out[idx] = lushBiomesBE[mcNextInt(l, 6)];
|
|
|
|
break;
|
|
|
|
case Cold:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (hasHighBit) out[idx] = giant_tree_taiga;
|
2018-11-29 04:32:12 +00:00
|
|
|
else out[idx] = coldBiomes[mcNextInt(l, 4)];
|
|
|
|
break;
|
|
|
|
case Freezing:
|
|
|
|
out[idx] = snowBiomes[mcNextInt(l, 4)];
|
|
|
|
break;
|
|
|
|
default:
|
2019-05-06 23:14:25 +02:00
|
|
|
out[idx] = mushroom_fields;
|
2018-11-29 04:32:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void mapRiverInit(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
|
|
|
|
int x, z;
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (out[x + z*areaWidth] > 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = mcNextInt(l, 299999)+2;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
void mapAddBamboo(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
|
|
|
|
int x, z;
|
|
|
|
for (z = 0; z < areaHeight; z++)
|
|
|
|
{
|
|
|
|
for (x = 0; x < areaWidth; x++)
|
|
|
|
{
|
|
|
|
int idx = x + z*areaWidth;
|
|
|
|
if (out[idx] != jungle) continue;
|
|
|
|
|
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
|
|
|
if (mcNextInt(l, 10) == 0)
|
|
|
|
{
|
|
|
|
out[idx] = bamboo_jungle;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
static inline int replaceEdge(int *out, int idx, int v10, int v21, int v01, int v12, int id, int baseID, int edgeID)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (id != baseID) return 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equalOrPlateau(v10, baseID) && equalOrPlateau(v21, baseID) && equalOrPlateau(v01, baseID) && equalOrPlateau(v12, baseID))
|
2018-03-05 21:20:54 +00:00
|
|
|
out[idx] = id;
|
|
|
|
else
|
|
|
|
out[idx] = edgeID;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mapBiomeEdge(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int v21 = out[x+2 + (z+1)*pWidth];
|
|
|
|
int v01 = out[x+0 + (z+1)*pWidth];
|
|
|
|
int v12 = out[x+1 + (z+2)*pWidth];
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (/*!replaceEdgeIfNecessary(out, x + z*areaWidth, v10, v21, v01, v12, v11, mountains, mountain_edge) &&*/
|
2019-05-07 19:38:29 +02:00
|
|
|
!replaceEdge(out, x + z*areaWidth, v10, v21, v01, v12, v11, wooded_badlands_plateau, badlands) &&
|
|
|
|
!replaceEdge(out, x + z*areaWidth, v10, v21, v01, v12, v11, badlands_plateau, badlands) &&
|
|
|
|
!replaceEdge(out, x + z*areaWidth, v10, v21, v01, v12, v11, giant_tree_taiga, taiga))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v11 == desert)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
if (v10 != snowy_tundra && v21 != snowy_tundra && v01 != snowy_tundra && v12 != snowy_tundra)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
out[x + z*areaWidth] = wooded_mountains;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
2019-05-06 23:14:25 +02:00
|
|
|
else if (v11 == swamp)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v10 != desert && v21 != desert && v01 != desert && v12 != desert &&
|
2019-05-07 19:38:29 +02:00
|
|
|
v10 != snowy_taiga && v21 != snowy_taiga && v01 != snowy_taiga && v12 != snowy_taiga &&
|
2019-05-06 23:14:25 +02:00
|
|
|
v10 != snowy_tundra && v21 != snowy_tundra && v01 != snowy_tundra && v12 != snowy_tundra)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
if (v10 != jungle && v12 != jungle && v21 != jungle && v01 != jungle &&
|
|
|
|
v10 != bamboo_jungle && v12 != bamboo_jungle &&
|
|
|
|
v21 != bamboo_jungle && v01 != bamboo_jungle)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = jungleEdge;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = plains;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapHills(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
2018-07-04 16:18:21 +01:00
|
|
|
int *buf = NULL;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p2 == NULL)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
printf("mapHills() requires two parents! Use setupMultiLayer()\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:53:35 +01:00
|
|
|
buf = (int *) malloc(pWidth*pHeight*sizeof(int));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
memcpy(buf, out, pWidth*pHeight*sizeof(int));
|
|
|
|
|
|
|
|
l->p2->getMap(l->p2, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
int a11 = buf[x+1 + (z+1)*pWidth]; // biome branch
|
|
|
|
int b11 = out[x+1 + (z+1)*pWidth]; // river branch
|
|
|
|
int idx = x + z*areaWidth;
|
|
|
|
|
|
|
|
int var12 = (b11 - 2) % 29 == 0;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (a11 != 0 && b11 >= 2 && (b11 - 2) % 29 == 1 && a11 < 128)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[idx] = (biomeExists(a11 + 128)) ? a11 + 128 : a11;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (mcNextInt(l, 3) != 0 && !var12)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int hillID = a11;
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
switch(a11)
|
|
|
|
{
|
2018-03-05 21:20:54 +00:00
|
|
|
case desert:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = desert_hills; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
case forest:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = wooded_hills; break;
|
2019-05-07 19:38:29 +02:00
|
|
|
case birch_forest:
|
|
|
|
hillID = birch_forest_hills; break;
|
|
|
|
case dark_forest:
|
2018-03-05 21:20:54 +00:00
|
|
|
hillID = plains; break;
|
|
|
|
case taiga:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = taiga_hills; break;
|
2019-05-07 19:38:29 +02:00
|
|
|
case giant_tree_taiga:
|
|
|
|
hillID = giant_tree_taiga_hills; break;
|
|
|
|
case snowy_taiga:
|
|
|
|
hillID = snowy_taiga_hills; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
case plains:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = (mcNextInt(l, 3) == 0) ? wooded_hills : forest; break;
|
|
|
|
case snowy_tundra:
|
|
|
|
hillID = snowy_mountains; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
case jungle:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = jungle_hills; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
case ocean:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = deep_ocean; break;
|
2019-05-06 23:14:25 +02:00
|
|
|
case mountains:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = wooded_mountains; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
case savanna:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = savanna_plateau; break;
|
2018-03-05 21:20:54 +00:00
|
|
|
default:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (equalOrPlateau(a11, wooded_badlands_plateau))
|
|
|
|
hillID = badlands;
|
|
|
|
else if (a11 == deep_ocean && mcNextInt(l, 3) == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
hillID = (mcNextInt(l, 2) == 0) ? plains : forest;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (var12 && hillID != a11)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (biomeExists(hillID + 128))
|
2018-03-05 21:20:54 +00:00
|
|
|
hillID += 128;
|
|
|
|
else
|
|
|
|
hillID = a11;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (hillID == a11)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int a10 = buf[x+1 + (z+0)*pWidth];
|
|
|
|
int a21 = buf[x+2 + (z+1)*pWidth];
|
|
|
|
int a01 = buf[x+0 + (z+1)*pWidth];
|
|
|
|
int a12 = buf[x+1 + (z+2)*pWidth];
|
|
|
|
int equals = 0;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equalOrPlateau(a10, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a21, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a01, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a12, a11)) equals++;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equals >= 3)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[idx] = hillID;
|
|
|
|
else
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-04 10:53:35 +01:00
|
|
|
|
|
|
|
free(buf);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-09 01:58:46 +01:00
|
|
|
void mapHills113(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
int *buf = NULL;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p2 == NULL)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
|
|
|
printf("mapHills() requires two parents! Use setupMultiLayer()\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
buf = (int *) malloc(pWidth*pHeight*sizeof(int));
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
memcpy(buf, out, pWidth*pHeight*sizeof(int));
|
|
|
|
|
|
|
|
l->p2->getMap(l->p2, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
|
|
|
int a11 = buf[x+1 + (z+1)*pWidth]; // biome branch
|
|
|
|
int b11 = out[x+1 + (z+1)*pWidth]; // river branch
|
|
|
|
int idx = x + z*areaWidth;
|
|
|
|
|
|
|
|
int bn = (b11 - 2) % 29;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!(isOceanic(a11) || b11 < 2 || bn != 1 || a11 >= 128))
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
|
|
|
out[idx] = (biomeExists(a11 + 128)) ? a11 + 128 : a11;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (mcNextInt(l, 3) == 0 || bn == 0)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
|
|
|
int hillID = a11;
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
switch(a11)
|
|
|
|
{
|
2018-07-09 01:58:46 +01:00
|
|
|
case desert:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = desert_hills; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
case forest:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = wooded_hills; break;
|
2019-05-07 19:38:29 +02:00
|
|
|
case birch_forest:
|
|
|
|
hillID = birch_forest_hills; break;
|
|
|
|
case dark_forest:
|
2018-07-09 01:58:46 +01:00
|
|
|
hillID = plains; break;
|
|
|
|
case taiga:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = taiga_hills; break;
|
2019-05-07 19:38:29 +02:00
|
|
|
case giant_tree_taiga:
|
|
|
|
hillID = giant_tree_taiga_hills; break;
|
|
|
|
case snowy_taiga:
|
|
|
|
hillID = snowy_taiga_hills; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
case plains:
|
2019-05-06 23:14:25 +02:00
|
|
|
hillID = (mcNextInt(l, 3) == 0) ? wooded_hills : forest; break;
|
|
|
|
case snowy_tundra:
|
|
|
|
hillID = snowy_mountains; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
case jungle:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = jungle_hills; break;
|
2019-05-06 23:14:25 +02:00
|
|
|
case bamboo_jungle:
|
|
|
|
hillID = bamboo_jungle_hills; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
case ocean:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = deep_ocean; break;
|
2019-05-06 23:14:25 +02:00
|
|
|
case mountains:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = wooded_mountains; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
case savanna:
|
2019-05-07 19:38:29 +02:00
|
|
|
hillID = savanna_plateau; break;
|
2018-07-09 01:58:46 +01:00
|
|
|
default:
|
2019-05-07 19:38:29 +02:00
|
|
|
if (equalOrPlateau(a11, wooded_badlands_plateau))
|
|
|
|
hillID = badlands;
|
2019-05-07 23:22:50 +02:00
|
|
|
else if (isDeepOcean(a11) && mcNextInt(l, 3) == 0)
|
2018-07-09 01:58:46 +01:00
|
|
|
hillID = (mcNextInt(l, 2) == 0) ? plains : forest;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (bn == 0 && hillID != a11)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (biomeExists(hillID + 128))
|
2018-07-09 01:58:46 +01:00
|
|
|
hillID += 128;
|
|
|
|
else
|
|
|
|
hillID = a11;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (hillID != a11)
|
2018-07-09 01:58:46 +01:00
|
|
|
{
|
|
|
|
int a10 = buf[x+1 + (z+0)*pWidth];
|
|
|
|
int a21 = buf[x+2 + (z+1)*pWidth];
|
|
|
|
int a01 = buf[x+0 + (z+1)*pWidth];
|
|
|
|
int a12 = buf[x+1 + (z+2)*pWidth];
|
|
|
|
int equals = 0;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equalOrPlateau(a10, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a21, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a01, a11)) equals++;
|
|
|
|
if (equalOrPlateau(a12, a11)) equals++;
|
2018-07-09 01:58:46 +01:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equals >= 3)
|
2018-07-09 01:58:46 +01:00
|
|
|
out[idx] = hillID;
|
|
|
|
else
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[idx] = a11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
static inline int reduceID(int id)
|
|
|
|
{
|
|
|
|
return id >= 2 ? 2 + (id & 1) : id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void mapRiver(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v01 = reduceID(out[x+0 + (z+1)*pWidth]);
|
|
|
|
int v21 = reduceID(out[x+2 + (z+1)*pWidth]);
|
|
|
|
int v10 = reduceID(out[x+1 + (z+0)*pWidth]);
|
|
|
|
int v12 = reduceID(out[x+1 + (z+2)*pWidth]);
|
|
|
|
int v11 = reduceID(out[x+1 + (z+1)*pWidth]);
|
|
|
|
|
|
|
|
if (v11 == v01 && v11 == v10 && v11 == v21 && v11 == v12)
|
|
|
|
{
|
|
|
|
out[x + z * areaWidth] = -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z * areaWidth] = river;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapSmooth(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int v21 = out[x+2 + (z+1)*pWidth];
|
|
|
|
int v01 = out[x+0 + (z+1)*pWidth];
|
|
|
|
int v12 = out[x+1 + (z+2)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v01 == v21 && v10 == v12)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (mcNextInt(l, 2) == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
v11 = v01;
|
|
|
|
else
|
|
|
|
v11 = v10;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v01 == v21) v11 = v01;
|
|
|
|
if (v10 == v12) v11 = v10;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
out[x + z * areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapRareBiome(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(l, (int64_t)(x + areaX), (int64_t)(z + areaZ));
|
2018-03-05 21:20:54 +00:00
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (mcNextInt(l, 57) == 0 && v11 == plains)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-09 01:58:46 +01:00
|
|
|
// Sunflower Plains
|
2018-03-10 21:45:57 +00:00
|
|
|
out[x + z*areaWidth] = plains + 128;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline static int replaceOcean(int *out, int idx, int v10, int v21, int v01, int v12, int id, int replaceID)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isOceanic(id)) return 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!isOceanic(v10) && !isOceanic(v21) && !isOceanic(v01) && !isOceanic(v12))
|
2018-03-05 21:20:54 +00:00
|
|
|
out[idx] = id;
|
|
|
|
else
|
|
|
|
out[idx] = replaceID;
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline static int isBiomeJFTO(int id)
|
|
|
|
{
|
2018-03-06 12:20:41 +00:00
|
|
|
return biomeExists(id) && (getBiomeType(id) == Jungle || id == forest || id == taiga || isOceanic(id));
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void mapShore(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int pX = areaX - 1;
|
|
|
|
int pZ = areaZ - 1;
|
|
|
|
int pWidth = areaWidth + 2;
|
|
|
|
int pHeight = areaHeight + 2;
|
|
|
|
int x, z;
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth];
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth];
|
|
|
|
int v21 = out[x+2 + (z+1)*pWidth];
|
|
|
|
int v01 = out[x+0 + (z+1)*pWidth];
|
|
|
|
int v12 = out[x+1 + (z+2)*pWidth];
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
int biome = biomeExists(v11) ? v11 : 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (v11 == mushroom_fields)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (v10 != ocean && v21 != ocean && v01 != ocean && v12 != ocean)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
else
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = mushroom_field_shore;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (/*biome < 128 &&*/ getBiomeType(biome) == Jungle)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isBiomeJFTO(v10) && isBiomeJFTO(v21) && isBiomeJFTO(v01) && isBiomeJFTO(v12))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
if (!isOceanic(v10) && !isOceanic(v21) && !isOceanic(v01) && !isOceanic(v12))
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = beach;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = jungleEdge;
|
|
|
|
}
|
|
|
|
}
|
2019-05-07 19:38:29 +02:00
|
|
|
else if (v11 != mountains && v11 != wooded_mountains && v11 != mountain_edge)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isBiomeSnowy(biome))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
replaceOcean(out, x + z*areaWidth, v10, v21, v01, v12, v11, snowy_beach);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
2019-05-07 19:38:29 +02:00
|
|
|
else if (v11 != badlands && v11 != wooded_badlands_plateau)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
if (v11 != ocean && v11 != deep_ocean && v11 != river && v11 != swamp)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!isOceanic(v10) && !isOceanic(v21) && !isOceanic(v01) && !isOceanic(v12))
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = beach;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!isOceanic(v10) && !isOceanic(v21) && !isOceanic(v01) && !isOceanic(v12))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (getBiomeType(v10) == Mesa && getBiomeType(v21) == Mesa && getBiomeType(v01) == Mesa && getBiomeType(v12) == Mesa)
|
2018-03-05 21:20:54 +00:00
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
else
|
|
|
|
out[x + z*areaWidth] = desert;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[x + z*areaWidth] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
replaceOcean(out, x + z*areaWidth, v10, v21, v01, v12, v11, stone_shore);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void mapRiverMix(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int idx;
|
2018-07-04 10:53:35 +01:00
|
|
|
int len;
|
|
|
|
int *buf;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p2 == NULL)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
printf("mapRiverMix() requires two parents! Use setupMultiLayer()\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:53:35 +01:00
|
|
|
len = areaWidth*areaHeight;
|
|
|
|
buf = (int *) malloc(len*sizeof(int));
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
l->p->getMap(l->p, out, areaX, areaZ, areaWidth, areaHeight); // biome chain
|
2018-07-04 10:53:35 +01:00
|
|
|
memcpy(buf, out, len*sizeof(int));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
l->p2->getMap(l->p2, out, areaX, areaZ, areaWidth, areaHeight); // rivers
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (idx = 0; idx < len; idx++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isOceanic(buf[idx]))
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
|
|
|
out[idx] = buf[idx];
|
|
|
|
}
|
|
|
|
else
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (out[idx] == river)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
if (buf[idx] == snowy_tundra)
|
|
|
|
out[idx] = frozen_river;
|
|
|
|
else if (buf[idx] == mushroom_fields || buf[idx] == mushroom_field_shore)
|
|
|
|
out[idx] = mushroom_field_shore;
|
2018-03-10 21:45:57 +00:00
|
|
|
else
|
|
|
|
out[idx] = out[idx] & 255;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out[idx] = buf[idx];
|
|
|
|
}
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
/* Initialises data for the ocean temperature types using the world seed.
|
2018-07-08 15:13:09 +01:00
|
|
|
* This function is called when the world seed is applied in setWorldSeed().
|
|
|
|
*/
|
|
|
|
static void oceanRndInit(OceanRnd *rnd, int64_t seed)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
int i = 0;
|
2018-07-08 15:13:09 +01:00
|
|
|
memset(rnd, 0, sizeof(*rnd));
|
|
|
|
setSeed(&seed);
|
|
|
|
rnd->a = nextDouble(&seed) * 256.0;
|
|
|
|
rnd->b = nextDouble(&seed) * 256.0;
|
|
|
|
rnd->c = nextDouble(&seed) * 256.0;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (i = 0; i < 256; i++)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
rnd->d[i] = i;
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
for (i = 0; i < 256; i++)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
int n3 = nextInt(&seed, 256 - i) + i;
|
|
|
|
int n4 = rnd->d[i];
|
|
|
|
rnd->d[i] = rnd->d[n3];
|
2018-07-08 15:13:09 +01:00
|
|
|
rnd->d[n3] = n4;
|
2018-07-08 17:26:49 +01:00
|
|
|
rnd->d[i + 256] = rnd->d[i];
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-08 17:26:49 +01:00
|
|
|
static double lerp(const double part, const double from, const double to)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
return from + part * (to - from);
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-08 17:26:49 +01:00
|
|
|
/* Table of vectors to cube edge centres (12 + 4 extra), used for ocean PRNG */
|
|
|
|
const double cEdgeX[] = {1.0,-1.0, 1.0,-1.0, 1.0,-1.0, 1.0,-1.0, 0.0, 0.0, 0.0, 0.0, 1.0, 0.0,-1.0, 0.0};
|
|
|
|
const double cEdgeY[] = {1.0, 1.0,-1.0,-1.0, 0.0, 0.0, 0.0, 0.0, 1.0,-1.0, 1.0,-1.0, 1.0,-1.0, 1.0,-1.0};
|
|
|
|
const double cEdgeZ[] = {0.0, 0.0, 0.0, 0.0, 1.0, 1.0,-1.0,-1.0, 1.0, 1.0,-1.0,-1.0, 0.0, 1.0, 0.0,-1.0};
|
|
|
|
|
|
|
|
static double indexedLerp(int idx, const double d1, const double d2, const double d3)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
idx &= 0xf;
|
|
|
|
return cEdgeX[idx] * d1 + cEdgeY[idx] * d2 + cEdgeZ[idx] * d3;
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
|
|
|
|
2018-07-08 17:26:49 +01:00
|
|
|
|
|
|
|
static double getOceanTemp(const OceanRnd *rnd, double d1, double d2, double d3)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
d1 += rnd->a;
|
|
|
|
d2 += rnd->b;
|
|
|
|
d3 += rnd->c;
|
|
|
|
int i1 = (int)d1 - (int)(d1 < 0);
|
|
|
|
int i2 = (int)d2 - (int)(d2 < 0);
|
|
|
|
int i3 = (int)d3 - (int)(d3 < 0);
|
|
|
|
d1 -= i1;
|
|
|
|
d2 -= i2;
|
|
|
|
d3 -= i3;
|
|
|
|
double t1 = d1*d1*d1 * (d1 * (d1*6.0-15.0) + 10.0);
|
|
|
|
double t2 = d2*d2*d2 * (d2 * (d2*6.0-15.0) + 10.0);
|
|
|
|
double t3 = d3*d3*d3 * (d3 * (d3*6.0-15.0) + 10.0);
|
|
|
|
|
|
|
|
i1 &= 0xff;
|
|
|
|
i2 &= 0xff;
|
|
|
|
i3 &= 0xff;
|
|
|
|
|
|
|
|
int a1 = rnd->d[i1] + i2;
|
|
|
|
int a2 = rnd->d[a1] + i3;
|
|
|
|
int a3 = rnd->d[a1+1] + i3;
|
|
|
|
int b1 = rnd->d[i1+1] + i2;
|
|
|
|
int b2 = rnd->d[b1] + i3;
|
|
|
|
int b3 = rnd->d[b1+1] + i3;
|
|
|
|
|
|
|
|
double l1 = indexedLerp(rnd->d[a2], d1, d2, d3);
|
|
|
|
double l2 = indexedLerp(rnd->d[b2], d1-1, d2, d3);
|
|
|
|
double l3 = indexedLerp(rnd->d[a3], d1, d2-1, d3);
|
|
|
|
double l4 = indexedLerp(rnd->d[b3], d1-1, d2-1, d3);
|
|
|
|
double l5 = indexedLerp(rnd->d[a2+1], d1, d2, d3-1);
|
|
|
|
double l6 = indexedLerp(rnd->d[b2+1], d1-1, d2, d3-1);
|
|
|
|
double l7 = indexedLerp(rnd->d[a3+1], d1, d2-1, d3-1);
|
|
|
|
double l8 = indexedLerp(rnd->d[b3+1], d1-1, d2-1, d3-1);
|
|
|
|
|
|
|
|
l1 = lerp(t1, l1, l2);
|
|
|
|
l3 = lerp(t1, l3, l4);
|
|
|
|
l5 = lerp(t1, l5, l6);
|
|
|
|
l7 = lerp(t1, l7, l8);
|
|
|
|
|
|
|
|
l1 = lerp(t2, l1, l3);
|
|
|
|
l5 = lerp(t2, l5, l7);
|
|
|
|
|
|
|
|
return lerp(t3, l1, l5);
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void mapOceanTemp(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
int x, z;
|
|
|
|
OceanRnd *rnd = l->oceanRnd;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 15:13:09 +01:00
|
|
|
double tmp = getOceanTemp(rnd, (x + areaX) / 8.0, (z + areaZ) / 8.0, 0);
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (tmp > 0.4)
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = warm_ocean;
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (tmp > 0.2)
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = lukewarm_ocean;
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (tmp < -0.4)
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = frozen_ocean;
|
2018-07-28 16:57:29 +02:00
|
|
|
else if (tmp < -0.2)
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = cold_ocean;
|
2018-03-10 21:45:57 +00:00
|
|
|
else
|
2018-07-08 17:26:49 +01:00
|
|
|
out[x + z*areaWidth] = ocean;
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-08 17:35:42 +01:00
|
|
|
/* Warning: this function is horribly slow compared to other layers! */
|
2018-07-08 15:13:09 +01:00
|
|
|
void mapOceanMix(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 15:13:09 +01:00
|
|
|
int landX = areaX-8, landZ = areaZ-8;
|
|
|
|
int landWidth = areaWidth+17, landHeight = areaHeight+17;
|
|
|
|
int *map1, *map2;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (l->p2 == NULL)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
|
|
|
printf("mapOceanMix() requires two parents! Use setupMultiLayer()\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, landX, landZ, landWidth, landHeight);
|
|
|
|
map1 = (int *) malloc(landWidth*landHeight*sizeof(int));
|
|
|
|
memcpy(map1, out, landWidth*landHeight*sizeof(int));
|
|
|
|
|
|
|
|
l->p2->getMap(l->p2, out, areaX, areaZ, areaWidth, areaHeight);
|
|
|
|
map2 = (int *) malloc(areaWidth*areaHeight*sizeof(int));
|
|
|
|
memcpy(map2, out, areaWidth*areaHeight*sizeof(int));
|
|
|
|
|
|
|
|
|
|
|
|
int x, z;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < areaWidth; x++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
int landID = map1[(x+8) + (z+8)*landWidth];
|
|
|
|
int oceanID = map2[x + z*areaWidth];
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!isOceanic(landID))
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 17:26:49 +01:00
|
|
|
out[x + z*areaWidth] = landID;
|
2018-07-08 15:13:09 +01:00
|
|
|
continue;
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (int i = -8; i <= 8; i += 4)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
for (int j = -8; j <= 8; j += 4)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-08 17:35:42 +01:00
|
|
|
int nearbyID = map1[(x+i+8) + (z+j+8)*landWidth];
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (isOceanic(nearbyID)) continue;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (oceanID == warm_ocean)
|
2018-07-08 15:13:09 +01:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = lukewarm_ocean;
|
2018-07-08 15:13:09 +01:00
|
|
|
goto loop_x;
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
if (oceanID == frozen_ocean)
|
2018-07-08 17:35:42 +01:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
out[x + z*areaWidth] = cold_ocean;
|
2018-07-08 17:35:42 +01:00
|
|
|
goto loop_x;
|
|
|
|
}
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2019-05-07 19:38:29 +02:00
|
|
|
if (landID == deep_ocean)
|
2018-07-08 17:35:42 +01:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
switch (oceanID)
|
2018-07-08 17:35:42 +01:00
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
case lukewarm_ocean:
|
|
|
|
oceanID = deep_lukewarm_ocean;
|
|
|
|
break;
|
|
|
|
case ocean:
|
|
|
|
oceanID = deep_ocean;
|
|
|
|
break;
|
|
|
|
case cold_ocean:
|
|
|
|
oceanID = deep_cold_ocean;
|
|
|
|
break;
|
|
|
|
case frozen_ocean:
|
|
|
|
oceanID = deep_frozen_ocean;
|
|
|
|
break;
|
2018-07-08 15:13:09 +01:00
|
|
|
}
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-08 17:26:49 +01:00
|
|
|
out[x + z*areaWidth] = oceanID;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
loop_x:;
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
free(map1);
|
|
|
|
free(map2);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void mapVoronoiZoom(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight)
|
|
|
|
{
|
|
|
|
areaX -= 2;
|
|
|
|
areaZ -= 2;
|
|
|
|
int pX = areaX >> 2;
|
|
|
|
int pZ = areaZ >> 2;
|
|
|
|
int pWidth = (areaWidth >> 2) + 2;
|
|
|
|
int pHeight = (areaHeight >> 2) + 2;
|
|
|
|
int newWidth = (pWidth-1) << 2;
|
|
|
|
int newHeight = (pHeight-1) << 2;
|
|
|
|
int x, z, i, j;
|
|
|
|
int *buf = (int *)malloc((newWidth+1)*(newHeight+1)*sizeof(*buf));
|
|
|
|
|
|
|
|
l->p->getMap(l->p, out, pX, pZ, pWidth, pHeight);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < pHeight - 1; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int v00 = out[(z+0)*pWidth];
|
|
|
|
int v01 = out[(z+1)*pWidth];
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (x = 0; x < pWidth - 1; x++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
setChunkSeed(l, (x+pX) << 2, (z+pZ) << 2);
|
|
|
|
double da1 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6;
|
|
|
|
double da2 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6;
|
|
|
|
|
|
|
|
setChunkSeed(l, (x+pX+1) << 2, (z+pZ) << 2);
|
|
|
|
double db1 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6 + 4.0;
|
|
|
|
double db2 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6;
|
|
|
|
|
|
|
|
setChunkSeed(l, (x+pX) << 2, (z+pZ+1) << 2);
|
|
|
|
double dc1 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6;
|
|
|
|
double dc2 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6 + 4.0;
|
|
|
|
|
|
|
|
setChunkSeed(l, (x+pX+1) << 2, (z+pZ+1) << 2);
|
|
|
|
double dd1 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6 + 4.0;
|
|
|
|
double dd2 = (mcNextInt(l, 1024) / 1024.0 - 0.5) * 3.6 + 4.0;
|
|
|
|
|
|
|
|
int v10 = out[x+1 + (z+0)*pWidth] & 255;
|
|
|
|
int v11 = out[x+1 + (z+1)*pWidth] & 255;
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (j = 0; j < 4; j++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int idx = ((z << 2) + j) * newWidth + (x << 2);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (i = 0; i < 4; i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
double da = (j-da2)*(j-da2) + (i-da1)*(i-da1);
|
|
|
|
double db = (j-db2)*(j-db2) + (i-db1)*(i-db1);
|
|
|
|
double dc = (j-dc2)*(j-dc2) + (i-dc1)*(i-dc1);
|
|
|
|
double dd = (j-dd2)*(j-dd2) + (i-dd1)*(i-dd1);
|
|
|
|
|
|
|
|
if (da < db && da < dc && da < dd)
|
|
|
|
{
|
|
|
|
buf[idx++] = v00;
|
|
|
|
}
|
|
|
|
else if (db < da && db < dc && db < dd)
|
|
|
|
{
|
|
|
|
buf[idx++] = v10;
|
|
|
|
}
|
|
|
|
else if (dc < da && dc < db && dc < dd)
|
|
|
|
{
|
|
|
|
buf[idx++] = v01;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
buf[idx++] = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
v00 = v10;
|
|
|
|
v01 = v11;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
for (z = 0; z < areaHeight; z++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
memcpy(&out[z * areaWidth], &buf[(z + (areaZ & 3))*newWidth + (areaX & 3)], areaWidth*sizeof(int));
|
|
|
|
}
|
|
|
|
|
|
|
|
free(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
|