2018-03-05 21:20:54 +00:00
|
|
|
#ifndef LAYER_H_
|
|
|
|
#define LAYER_H_
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
#include "javarnd.h"
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
#define __STDC_FORMAT_MACROS 1
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
#include <stdlib.h>
|
2018-07-04 16:48:05 +01:00
|
|
|
#include <stdint.h>
|
2018-07-04 17:28:27 +01:00
|
|
|
#include <inttypes.h>
|
|
|
|
|
2019-05-06 23:14:25 +02:00
|
|
|
#ifndef NULL
|
|
|
|
#define NULL ((void*)0)
|
|
|
|
#endif
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
#define SIMD_NOTIFY 0
|
|
|
|
|
2018-04-25 00:36:45 +01:00
|
|
|
#if defined USE_SIMD && __AVX2__
|
2018-04-21 22:29:42 +02:00
|
|
|
#include <emmintrin.h>
|
|
|
|
#include <smmintrin.h>
|
|
|
|
#include <immintrin.h>
|
2020-03-15 16:09:06 +01:00
|
|
|
#if SIMD_NOTIFY
|
2018-04-21 22:29:42 +02:00
|
|
|
#warning "Using AVX2 extensions."
|
2020-03-15 16:09:06 +01:00
|
|
|
#endif
|
2018-04-25 00:36:45 +01:00
|
|
|
#elif defined USE_SIMD && defined __SSE4_2__
|
2018-04-21 22:29:42 +02:00
|
|
|
#include <emmintrin.h>
|
|
|
|
#include <smmintrin.h>
|
2020-03-15 16:09:06 +01:00
|
|
|
#if SIMD_NOTIFY
|
2018-04-21 22:29:42 +02:00
|
|
|
#warning "Using SSE4.2 extensions."
|
2020-03-15 16:09:06 +01:00
|
|
|
#endif
|
2018-04-21 22:29:42 +02:00
|
|
|
#else
|
2018-04-25 00:36:45 +01:00
|
|
|
//#warning "Using no SIMD extensions."
|
2018-04-21 22:29:42 +02:00
|
|
|
#endif
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
#define STRUCT(S) typedef struct S S; struct S
|
|
|
|
|
|
|
|
#define OPT_O2 __attribute__((optimize("O2")))
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
enum BiomeID
|
|
|
|
{
|
2018-03-05 21:20:54 +00:00
|
|
|
none = -1,
|
2019-05-06 23:14:25 +02:00
|
|
|
// 0
|
|
|
|
ocean = 0,
|
|
|
|
plains,
|
|
|
|
desert,
|
|
|
|
mountains, extremeHills = mountains,
|
|
|
|
forest,
|
|
|
|
taiga,
|
|
|
|
swamp, swampland = swamp,
|
|
|
|
river,
|
|
|
|
nether, hell = nether,
|
|
|
|
the_end, sky = the_end,
|
|
|
|
// 10
|
|
|
|
frozen_ocean, frozenOcean = frozen_ocean,
|
|
|
|
frozen_river, frozenRiver = frozen_river,
|
|
|
|
snowy_tundra, icePlains = snowy_tundra,
|
|
|
|
snowy_mountains, iceMountains = snowy_mountains,
|
|
|
|
mushroom_fields, mushroomIsland = mushroom_fields,
|
|
|
|
mushroom_field_shore, mushroomIslandShore = mushroom_field_shore,
|
|
|
|
beach,
|
|
|
|
desert_hills, desertHills = desert_hills,
|
|
|
|
wooded_hills, forestHills = wooded_hills,
|
|
|
|
taiga_hills, taigaHills = taiga_hills,
|
|
|
|
// 20
|
|
|
|
mountain_edge, extremeHillsEdge = mountain_edge,
|
|
|
|
jungle,
|
|
|
|
jungle_hills, jungleHills = jungle_hills,
|
|
|
|
jungle_edge, jungleEdge = jungle_edge,
|
|
|
|
deep_ocean, deepOcean = deep_ocean,
|
|
|
|
stone_shore, stoneBeach = stone_shore,
|
|
|
|
snowy_beach, coldBeach = snowy_beach,
|
|
|
|
birch_forest, birchForest = birch_forest,
|
|
|
|
birch_forest_hills, birchForestHills = birch_forest_hills,
|
|
|
|
dark_forest, roofedForest = dark_forest,
|
|
|
|
// 30
|
|
|
|
snowy_taiga, coldTaiga = snowy_taiga,
|
|
|
|
snowy_taiga_hills, coldTaigaHills = snowy_taiga_hills,
|
|
|
|
giant_tree_taiga, megaTaiga = giant_tree_taiga,
|
|
|
|
giant_tree_taiga_hills, megaTaigaHills = giant_tree_taiga_hills,
|
|
|
|
wooded_mountains, extremeHillsPlus = wooded_mountains,
|
|
|
|
savanna,
|
|
|
|
savanna_plateau, savannaPlateau = savanna_plateau,
|
|
|
|
badlands, mesa = badlands,
|
|
|
|
wooded_badlands_plateau, mesaPlateau_F = wooded_badlands_plateau,
|
|
|
|
badlands_plateau, mesaPlateau = badlands_plateau,
|
|
|
|
// 40 -- 1.13
|
|
|
|
small_end_islands,
|
|
|
|
end_midlands,
|
|
|
|
end_highlands,
|
|
|
|
end_barrens,
|
|
|
|
warm_ocean, warmOcean = warm_ocean,
|
|
|
|
lukewarm_ocean, lukewarmOcean = lukewarm_ocean,
|
|
|
|
cold_ocean, coldOcean = cold_ocean,
|
|
|
|
deep_warm_ocean, warmDeepOcean = deep_warm_ocean,
|
|
|
|
deep_lukewarm_ocean, lukewarmDeepOcean = deep_lukewarm_ocean,
|
|
|
|
deep_cold_ocean, coldDeepOcean = deep_cold_ocean,
|
|
|
|
// 50
|
|
|
|
deep_frozen_ocean, frozenDeepOcean = deep_frozen_ocean,
|
|
|
|
BIOME_NUM,
|
|
|
|
|
|
|
|
the_void = 127,
|
|
|
|
|
|
|
|
// mutated variants
|
|
|
|
sunflower_plains = plains+128,
|
|
|
|
desert_lakes = desert+128,
|
|
|
|
gravelly_mountains = mountains+128,
|
|
|
|
flower_forest = forest+128,
|
|
|
|
taiga_mountains = taiga+128,
|
|
|
|
swamp_hills = swamp+128,
|
|
|
|
ice_spikes = snowy_tundra+128,
|
|
|
|
modified_jungle = jungle+128,
|
|
|
|
modified_jungle_edge = jungle_edge+128,
|
|
|
|
tall_birch_forest = birch_forest+128,
|
|
|
|
tall_birch_hills = birch_forest_hills+128,
|
|
|
|
dark_forest_hills = dark_forest+128,
|
|
|
|
snowy_taiga_mountains = snowy_taiga+128,
|
|
|
|
giant_spruce_taiga = giant_tree_taiga+128,
|
|
|
|
giant_spruce_taiga_hills = giant_tree_taiga_hills+128,
|
|
|
|
modified_gravelly_mountains = wooded_mountains+128,
|
|
|
|
shattered_savanna = savanna+128,
|
|
|
|
shattered_savanna_plateau = savanna_plateau+128,
|
|
|
|
eroded_badlands = badlands+128,
|
|
|
|
modified_wooded_badlands_plateau = wooded_badlands_plateau+128,
|
|
|
|
modified_badlands_plateau = badlands_plateau+128,
|
|
|
|
// 1.14
|
|
|
|
bamboo_jungle = 168,
|
|
|
|
bamboo_jungle_hills = 169
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
enum BiomeType
|
|
|
|
{
|
2019-05-06 23:14:25 +02:00
|
|
|
Void = -1,
|
|
|
|
Ocean, Plains, Desert, Hills, Forest, Taiga, Swamp, River, Hell, Sky, Snow, MushroomIsland, Beach, Jungle, StoneBeach, Savanna, Mesa,
|
|
|
|
BTYPE_NUM
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
enum BiomeTempCategory
|
|
|
|
{
|
2018-08-02 23:55:43 +02:00
|
|
|
Oceanic, Warm, Lush, Cold, Freezing, Special
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
|
|
|
|
STRUCT(Biome)
|
|
|
|
{
|
2018-03-05 21:20:54 +00:00
|
|
|
int id;
|
|
|
|
int type;
|
|
|
|
double height;
|
|
|
|
double temp;
|
|
|
|
int tempCat;
|
|
|
|
};
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
STRUCT(OceanRnd)
|
|
|
|
{
|
|
|
|
int d[512];
|
|
|
|
double a, b, c;
|
|
|
|
};
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
STRUCT(Layer)
|
|
|
|
{
|
|
|
|
int64_t baseSeed; // Generator seed (depends only on layer hierarchy)
|
|
|
|
int64_t worldSeed; // based on the seed of the world
|
|
|
|
|
|
|
|
int64_t chunkSeed; // randomiser seed
|
|
|
|
|
|
|
|
OceanRnd *oceanRnd; // world seed dependent data for ocean temperatures
|
|
|
|
|
|
|
|
int scale; // map scale of this layer (map entry = scale x scale blocks)
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
void (*getMap)(Layer *layer, int *out, int x, int z, int w, int h);
|
|
|
|
|
2018-07-08 15:13:09 +01:00
|
|
|
Layer *p, *p2; // parent layers
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
#endif
|
2018-07-08 15:13:09 +01:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Essentials
|
|
|
|
//==============================================================================
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
extern Biome biomes[256];
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
/* initBiomes() has to be called before any of the generators can be used */
|
|
|
|
void initBiomes();
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
/* Applies the given world seed to the layer and all dependent layers. */
|
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
|
|
|
//==============================================================================
|
|
|
|
// Static Helpers
|
|
|
|
//==============================================================================
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
static inline int getBiomeType(int id)
|
|
|
|
{
|
|
|
|
return biomes[id & 0xff].type;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int biomeExists(int id)
|
|
|
|
{
|
|
|
|
return !(biomes[id & 0xff].id & (~0xff));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int getTempCategory(int id)
|
|
|
|
{
|
|
|
|
return biomes[id & 0xff].tempCat;
|
|
|
|
}
|
|
|
|
|
2018-07-09 01:58:46 +01:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
static inline int equalOrPlateau(int id1, int id2)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (id1 == id2) return 1;
|
2019-05-07 19:38:29 +02:00
|
|
|
if (id1 == wooded_badlands_plateau || id1 == badlands_plateau)
|
|
|
|
return id2 == wooded_badlands_plateau || id2 == badlands_plateau;
|
2018-07-28 16:57:29 +02:00
|
|
|
if (!biomeExists(id1) || !biomeExists(id2)) return 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
// adjust for asymmetric equality (workaround to simulate a bug in the MC java code)
|
2018-07-28 16:57:29 +02:00
|
|
|
if (id1 >= 128 || id2 >= 128) {
|
2018-03-05 21:20:54 +00:00
|
|
|
// skip biomes that did not overload the isEqualTo() method
|
2018-07-28 16:57:29 +02:00
|
|
|
if (id2 == 130 || id2 == 133 || id2 == 134 || id2 == 149 || id2 == 151 || id2 == 155 ||
|
2019-05-04 14:19:35 +02:00
|
|
|
id2 == 156 || id2 == 157 || id2 == 158 || id2 == 163 || id2 == 164) return 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
return getBiomeType(id1) == getBiomeType(id2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int canBeNeighbors(int id1, int id2)
|
|
|
|
{
|
2018-07-28 16:57:29 +02:00
|
|
|
if (equalOrPlateau(id1, id2)) return 1;
|
|
|
|
if (!biomeExists(id1) || !biomeExists(id2)) return 0;
|
|
|
|
int tempCat1 = getTempCategory(id1); if (tempCat1 == Lush) return 1;
|
|
|
|
int tempCat2 = getTempCategory(id2); if (tempCat2 == Lush) return 1;
|
2018-03-05 21:20:54 +00:00
|
|
|
return tempCat1 == tempCat2;
|
|
|
|
}
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
static inline int isShallowOcean(int id)
|
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
const uint64_t shallow_bits =
|
|
|
|
(1ULL << ocean) |
|
|
|
|
(1ULL << frozen_ocean) |
|
|
|
|
(1ULL << warm_ocean) |
|
|
|
|
(1ULL << lukewarm_ocean) |
|
|
|
|
(1ULL << cold_ocean);
|
|
|
|
return id < 64 && ((1ULL << id) & shallow_bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int isDeepOcean(int id)
|
|
|
|
{
|
|
|
|
const uint64_t deep_bits =
|
|
|
|
(1ULL << deep_ocean) |
|
|
|
|
(1ULL << deep_warm_ocean) |
|
|
|
|
(1ULL << deep_lukewarm_ocean) |
|
|
|
|
(1ULL << deep_cold_ocean) |
|
|
|
|
(1ULL << deep_frozen_ocean);
|
|
|
|
return id < 64 && ((1ULL << id) & deep_bits);
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
static inline int isOceanic(int id)
|
|
|
|
{
|
2019-05-07 19:38:29 +02:00
|
|
|
const uint64_t ocean_bits =
|
|
|
|
(1ULL << ocean) |
|
|
|
|
(1ULL << frozen_ocean) |
|
|
|
|
(1ULL << warm_ocean) |
|
|
|
|
(1ULL << lukewarm_ocean) |
|
|
|
|
(1ULL << cold_ocean) |
|
|
|
|
(1ULL << deep_ocean) |
|
|
|
|
(1ULL << deep_warm_ocean) |
|
|
|
|
(1ULL << deep_lukewarm_ocean) |
|
|
|
|
(1ULL << deep_cold_ocean) |
|
|
|
|
(1ULL << deep_frozen_ocean);
|
|
|
|
return id < 64 && ((1ULL << id) & ocean_bits);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
static inline int isBiomeSnowy(int id)
|
|
|
|
{
|
|
|
|
return biomeExists(id) && biomes[id&0xff].temp < 0.1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int mcNextInt(Layer *layer, int mod)
|
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
int ret = (int)((layer->chunkSeed >> 24) % (int64_t)mod);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ret += mod;
|
|
|
|
}
|
|
|
|
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->chunkSeed *= layer->chunkSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->chunkSeed += layer->worldSeed;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-08-02 23:55:43 +02:00
|
|
|
|
|
|
|
|
|
|
|
static inline int64_t processWorldSeed(register int64_t ws, const int64_t bs)
|
|
|
|
{
|
|
|
|
ws *= ws * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ws += bs;
|
|
|
|
ws *= ws * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ws += bs;
|
|
|
|
ws *= ws * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ws += bs;
|
|
|
|
ws *= ws * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
return ws;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int64_t getChunkSeed(register int64_t ss, const int64_t x, const int64_t z)
|
|
|
|
{
|
|
|
|
ss += x;
|
|
|
|
ss *= ss * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ss += z;
|
|
|
|
ss *= ss * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ss += x;
|
|
|
|
ss *= ss * 6364136223846793005LL + 1442695040888963407LL;
|
|
|
|
ss += z;
|
|
|
|
return ss;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline void setChunkSeed(Layer *layer, int64_t chunkX, int64_t chunkZ)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
layer->chunkSeed = layer->worldSeed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->chunkSeed *= layer->chunkSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->chunkSeed += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->chunkSeed *= layer->chunkSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->chunkSeed += chunkZ;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->chunkSeed *= layer->chunkSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->chunkSeed += chunkX;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->chunkSeed *= layer->chunkSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->chunkSeed += chunkZ;
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline void setBaseSeed(Layer *layer, int64_t seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
layer->baseSeed = seed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->baseSeed *= layer->baseSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->baseSeed += seed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->baseSeed *= layer->baseSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->baseSeed += seed;
|
2018-07-04 17:28:27 +01:00
|
|
|
layer->baseSeed *= layer->baseSeed * 6364136223846793005LL + 1442695040888963407LL;
|
2018-03-05 21:20:54 +00:00
|
|
|
layer->baseSeed += seed;
|
|
|
|
|
|
|
|
layer->p = NULL;
|
|
|
|
layer->worldSeed = 0;
|
|
|
|
layer->chunkSeed = 0;
|
|
|
|
}
|
|
|
|
|
2018-04-25 00:36:45 +01:00
|
|
|
#if defined USE_SIMD && __AVX2__
|
2018-07-28 16:57:29 +02:00
|
|
|
|
|
|
|
static inline __m256i set8ChunkSeeds(int ws, __m256i xs, __m256i zs)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m256i out = _mm256_set1_epi32(ws);
|
|
|
|
__m256i mul = _mm256_set1_epi32(1284865837);
|
|
|
|
__m256i add = _mm256_set1_epi32(4150755663);
|
|
|
|
out = _mm256_add_epi32(xs, _mm256_mullo_epi32(out, _mm256_add_epi32(add, _mm256_mullo_epi32(out, mul))));
|
|
|
|
out = _mm256_add_epi32(zs, _mm256_mullo_epi32(out, _mm256_add_epi32(add, _mm256_mullo_epi32(out, mul))));
|
|
|
|
out = _mm256_add_epi32(xs, _mm256_mullo_epi32(out, _mm256_add_epi32(add, _mm256_mullo_epi32(out, mul))));
|
|
|
|
return _mm256_add_epi32(zs, _mm256_mullo_epi32(out, _mm256_add_epi32(add, _mm256_mullo_epi32(out, mul))));
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m256i mc8NextInt(__m256i* cs, int ws, int mask)
|
|
|
|
{
|
2020-03-15 16:09:06 +01:00
|
|
|
__m256i andm = _mm256_set1_epi32(mask);
|
|
|
|
__m256i ret = _mm256_and_si256(andm, _mm256_srli_epi32(*cs, 24));
|
2018-04-21 22:29:42 +02:00
|
|
|
*cs = _mm256_add_epi32(_mm256_set1_epi32(ws), _mm256_mullo_epi32(*cs, _mm256_add_epi32(_mm256_set1_epi32(4150755663), _mm256_mullo_epi32(*cs, _mm256_set1_epi32(1284865837)))));
|
2020-03-15 16:09:06 +01:00
|
|
|
return _mm256_add_epi32(ret, _mm256_and_si256(andm, _mm256_cmpgt_epi32(_mm256_set1_epi32(0), ret)));
|
2018-04-21 22:29:42 +02:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m256i select8Random2(__m256i* cs, int ws, __m256i a1, __m256i a2)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m256i cmp = _mm256_cmpeq_epi32(_mm256_set1_epi32(0), mc8NextInt(cs, ws, 0x1));
|
|
|
|
return _mm256_or_si256(_mm256_and_si256(cmp, a1), _mm256_andnot_si256(cmp, a2));
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m256i select8Random4(__m256i* cs, int ws, __m256i a1, __m256i a2, __m256i a3, __m256i a4)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m256i val = mc8NextInt(cs, ws, 0x3);
|
|
|
|
__m256i v2 = _mm256_set1_epi32(2);
|
|
|
|
__m256i cmp1 = _mm256_cmpeq_epi32(val, _mm256_set1_epi32(0));
|
|
|
|
__m256i cmp2 = _mm256_cmpeq_epi32(v2, val);
|
|
|
|
__m256i cmp3 = _mm256_cmpgt_epi32(v2, val);
|
|
|
|
return _mm256_or_si256(
|
|
|
|
_mm256_and_si256(cmp3, _mm256_or_si256(_mm256_and_si256(cmp1, a1), _mm256_andnot_si256(cmp1, a2))),
|
|
|
|
_mm256_andnot_si256(cmp3, _mm256_or_si256(_mm256_and_si256(cmp2, a3), _mm256_andnot_si256(cmp2, a4)))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m256i select8ModeOrRandom(__m256i* cs, int ws, __m256i a1, __m256i a2, __m256i a3, __m256i a4)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m256i cmp1 = _mm256_cmpeq_epi32(a1, a2);
|
|
|
|
__m256i cmp2 = _mm256_cmpeq_epi32(a1, a3);
|
|
|
|
__m256i cmp3 = _mm256_cmpeq_epi32(a1, a4);
|
|
|
|
__m256i cmp4 = _mm256_cmpeq_epi32(a2, a3);
|
|
|
|
__m256i cmp5 = _mm256_cmpeq_epi32(a2, a4);
|
|
|
|
__m256i cmp6 = _mm256_cmpeq_epi32(a3, a4);
|
|
|
|
__m256i isa1 = _mm256_or_si256(
|
|
|
|
_mm256_andnot_si256(cmp6, cmp1),
|
|
|
|
_mm256_or_si256 (
|
|
|
|
_mm256_andnot_si256(cmp5, cmp2),
|
|
|
|
_mm256_andnot_si256(cmp4, cmp3)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
__m256i isa2 = _mm256_or_si256(
|
|
|
|
_mm256_andnot_si256(cmp3, cmp4),
|
|
|
|
_mm256_andnot_si256(cmp2, cmp5)
|
|
|
|
);
|
|
|
|
__m256i isa3 = _mm256_andnot_si256(cmp1, cmp6);
|
2018-07-28 16:57:29 +02:00
|
|
|
|
2018-04-21 22:29:42 +02:00
|
|
|
return _mm256_or_si256(
|
|
|
|
_mm256_andnot_si256(
|
|
|
|
_mm256_or_si256(
|
|
|
|
isa1,
|
|
|
|
_mm256_or_si256(isa2, isa3)
|
|
|
|
),
|
|
|
|
select8Random4(cs, ws, a1, a2, a3, a4)
|
|
|
|
),
|
|
|
|
_mm256_or_si256(
|
|
|
|
_mm256_and_si256(isa1, a1),
|
|
|
|
_mm256_or_si256(
|
|
|
|
_mm256_and_si256(isa2, a2),
|
|
|
|
_mm256_and_si256(isa3, a3)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
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
|
|
|
|
|
|
|
static inline __m128i set4ChunkSeeds(int ws, __m128i xs, __m128i zs)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m128i out = _mm_set1_epi32(ws);
|
|
|
|
__m128i mul = _mm_set1_epi32(1284865837);
|
|
|
|
__m128i add = _mm_set1_epi32(4150755663);
|
|
|
|
out = _mm_add_epi32(xs, _mm_mullo_epi32(out, _mm_add_epi32(add, _mm_mullo_epi32(out, mul))));
|
|
|
|
out = _mm_add_epi32(zs, _mm_mullo_epi32(out, _mm_add_epi32(add, _mm_mullo_epi32(out, mul))));
|
|
|
|
out = _mm_add_epi32(xs, _mm_mullo_epi32(out, _mm_add_epi32(add, _mm_mullo_epi32(out, mul))));
|
|
|
|
return _mm_add_epi32(zs, _mm_mullo_epi32(out, _mm_add_epi32(add, _mm_mullo_epi32(out, mul))));
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m128i mc4NextInt(__m128i* cs, int ws, int mask)
|
|
|
|
{
|
2020-03-15 16:09:06 +01:00
|
|
|
__m128i andm = _mm_set1_epi32(mask);
|
|
|
|
__m128i ret = _mm_and_si128(andm, _mm_srli_epi32(*cs, 24));
|
2018-04-21 22:29:42 +02:00
|
|
|
*cs = _mm_add_epi32( _mm_set1_epi32(ws), _mm_mullo_epi32(*cs, _mm_add_epi32(_mm_set1_epi32(4150755663), _mm_mullo_epi32(*cs, _mm_set1_epi32(1284865837)))));
|
2020-03-15 16:09:06 +01:00
|
|
|
return _mm_add_epi32(ret, _mm_and_si128(andm, _mm_cmplt_epi32(ret, _mm_set1_epi32(0))));
|
2018-04-21 22:29:42 +02:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m128i select4Random2(__m128i* cs, int ws, __m128i a1, __m128i a2)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m128i cmp = _mm_cmpeq_epi32(_mm_set1_epi32(0), mc4NextInt(cs, ws, 0x1));
|
|
|
|
return _mm_or_si128(_mm_and_si128(cmp, a1), _mm_andnot_si128(cmp, a2));
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m128i select4Random4(__m128i* cs, int ws, __m128i a1, __m128i a2, __m128i a3, __m128i a4)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
__m128i val = mc4NextInt(cs, ws, 0x3);
|
|
|
|
__m128i v2 = _mm_set1_epi32(2);
|
|
|
|
__m128i cmp1 = _mm_cmpeq_epi32(val, _mm_set1_epi32(0));
|
|
|
|
__m128i cmp2 = _mm_cmpeq_epi32(val, v2);
|
|
|
|
__m128i cmp3 = _mm_cmplt_epi32(val, v2);
|
|
|
|
return _mm_or_si128(
|
|
|
|
_mm_and_si128(cmp3, _mm_or_si128(_mm_and_si128(cmp1, a1), _mm_andnot_si128(cmp1, a2))),
|
|
|
|
_mm_andnot_si128(cmp3, _mm_or_si128(_mm_and_si128(cmp2, a3), _mm_andnot_si128(cmp2, a4)))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
static inline __m128i select4ModeOrRandom(__m128i* cs, int ws, __m128i a1, __m128i a2, __m128i a3, __m128i a4)
|
|
|
|
{
|
2018-04-21 22:29:42 +02:00
|
|
|
//((a == b)&(c != d) | (a == c)&(b != d) | (a == d)&(b != c))&a | ((b == c)&(a != d) | (b == d)&(a != c))&b | ((c == d)&(a != b))&c
|
|
|
|
__m128i cmp1 = _mm_cmpeq_epi32(a1, a2);
|
|
|
|
__m128i cmp2 = _mm_cmpeq_epi32(a1, a3);
|
|
|
|
__m128i cmp3 = _mm_cmpeq_epi32(a1, a4);
|
|
|
|
__m128i cmp4 = _mm_cmpeq_epi32(a2, a3);
|
|
|
|
__m128i cmp5 = _mm_cmpeq_epi32(a2, a4);
|
|
|
|
__m128i cmp6 = _mm_cmpeq_epi32(a3, a4);
|
|
|
|
__m128i isa1 = _mm_or_si128(
|
|
|
|
_mm_andnot_si128(cmp6, cmp1),
|
|
|
|
_mm_or_si128 (
|
|
|
|
_mm_andnot_si128(cmp5, cmp2),
|
|
|
|
_mm_andnot_si128(cmp4, cmp3)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
__m128i isa2 = _mm_or_si128(
|
|
|
|
_mm_andnot_si128(cmp3, cmp4),
|
|
|
|
_mm_andnot_si128(cmp2, cmp5)
|
|
|
|
);
|
|
|
|
__m128i isa3 = _mm_andnot_si128(cmp1, cmp6);
|
|
|
|
return _mm_or_si128(
|
|
|
|
_mm_andnot_si128(
|
|
|
|
_mm_or_si128(
|
|
|
|
isa1,
|
|
|
|
_mm_or_si128(isa2, isa3)
|
|
|
|
),
|
|
|
|
select4Random4(cs, ws, a1, a2, a3, a4)
|
|
|
|
),
|
|
|
|
_mm_or_si128(
|
|
|
|
_mm_and_si128(isa1, a1),
|
|
|
|
_mm_or_si128(
|
|
|
|
_mm_and_si128(isa2, a2),
|
|
|
|
_mm_and_si128(isa3, a3)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
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
|
|
|
static inline int selectRandom2(Layer *l, int a1, int a2)
|
|
|
|
{
|
|
|
|
int i = mcNextInt(l, 2);
|
|
|
|
return i == 0 ? a1 : a2;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int selectRandom4(Layer *l, int a1, int a2, int a3, int a4)
|
|
|
|
{
|
|
|
|
int i = mcNextInt(l, 4);
|
|
|
|
return i == 0 ? a1 : i == 1 ? a2 : i == 2 ? a3 : a4;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int selectModeOrRandom(Layer *l, int a1, int a2, int a3, int a4)
|
|
|
|
{
|
|
|
|
int rndarg = selectRandom4(l, a1, a2, a3, a4);
|
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
if (a2 == a3 && a3 == a4) return a2;
|
|
|
|
if (a1 == a2 && a1 == a3) return a1;
|
|
|
|
if (a1 == a2 && a1 == a4) return a1;
|
|
|
|
if (a1 == a3 && a1 == a4) return a1;
|
|
|
|
if (a1 == a2 && a3 != a4) return a1;
|
|
|
|
if (a1 == a3 && a2 != a4) return a1;
|
|
|
|
if (a1 == a4 && a2 != a3) return a1;
|
|
|
|
if (a2 == a3 && a1 != a4) return a2;
|
|
|
|
if (a2 == a4 && a1 != a3) return a2;
|
|
|
|
if (a3 == a4 && a1 != a2) return a3;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
return rndarg;
|
|
|
|
}
|
2018-07-28 16:57:29 +02:00
|
|
|
|
2018-04-21 22:29:42 +02:00
|
|
|
#endif
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:57:29 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Layers
|
|
|
|
//==============================================================================
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
// A null layer does nothing, and can be used to apply a layer to existing data.
|
|
|
|
void mapNull(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-07-09 02:21:34 +01:00
|
|
|
// A skip layer simply calls its first parent without modification.
|
|
|
|
// This can be used as an easy way to skip a layer in a generator.
|
|
|
|
void mapSkip(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
void mapIsland(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapZoom(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapAddIsland(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapRemoveTooMuchOcean(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapAddSnow(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapCoolWarm(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapHeatIce(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapSpecial(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapAddMushroomIsland(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapDeepOcean(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapBiome(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-11-29 04:32:12 +00:00
|
|
|
void mapBiomeBE(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2019-05-06 23:14:25 +02:00
|
|
|
void mapAddBamboo(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-03-05 21:20:54 +00:00
|
|
|
void mapRiverInit(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapBiomeEdge(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapHills(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapRiver(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapSmooth(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapRareBiome(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapShore(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
void mapRiverMix(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
|
|
|
|
2018-07-09 01:58:46 +01:00
|
|
|
// 1.13 layers
|
|
|
|
void mapHills113(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-03-10 21:45:57 +00:00
|
|
|
void mapOceanTemp(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight);
|
|
|
|
void mapOceanMix(Layer *l, int * __restrict out, int areaX, int areaZ, int areaWidth, int areaHeight);
|
|
|
|
|
|
|
|
void mapVoronoiZoom(Layer *l, int * __restrict out, int x, int z, int w, int h);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2020-03-15 16:09:06 +01:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
#endif /* LAYER_H_ */
|