cubiomes/finders.h

444 lines
15 KiB
C
Raw Normal View History

2018-03-05 21:20:54 +00:00
#ifndef FINDERS_H_
#define FINDERS_H_
#include "generator.h"
#include <stdio.h>
#include <string.h>
#include <pthread.h>
#include <stdlib.h>
2018-07-05 21:05:34 +01:00
#define DEFAULT_THREADS 6
2018-07-04 17:28:27 +01:00
#define SEEDMAX (1LL << 48)
#define PI 3.141592653589793
2018-03-05 21:20:54 +00:00
#define LARGE_STRUCT 1
#define USE_POW2_RNG 2
STRUCT(StructureConfig) {
int64_t seed;
int regionSize, chunkRange;
int properties;
};
/* For desert temples, igloos, jungle temples and witch huts prior to 1.13. */
extern const StructureConfig FEATURE_CONFIG;
2018-07-04 16:18:21 +01:00
/* 1.13 separated feature seeds by type */
extern const StructureConfig DESERT_PYRAMID_CONFIG;
extern const StructureConfig IGLOO_CONFIG;
extern const StructureConfig JUNGLE_PYRAMID_CONFIG;
extern const StructureConfig SWAMP_HUT_CONFIG;
extern const StructureConfig VILLAGE_CONFIG;
extern const StructureConfig OCEAN_RUIN_CONFIG;
extern const StructureConfig SHIPWRECK_CONFIG;
extern const StructureConfig MONUMENT_CONFIG;
extern const StructureConfig MANSION_CONFIG;
2018-07-04 16:18:21 +01:00
enum {
Desert_Pyramid, Igloo, Jungle_Pyramid, Swamp_Hut,
Village, Ocean_Ruin, Shipwreck, Monument, Mansion
} StructureType;
2018-03-05 21:20:54 +00:00
2018-03-17 16:53:16 +00:00
static const int templeBiomeList[] = {desert, desertHills, jungle, jungleHills, swampland, icePlains, coldTaiga};
static const int biomesToSpawnIn[] = {forest, plains, taiga, taigaHills, forestHills, jungle, jungleHills};
2018-03-05 21:20:54 +00:00
static const int oceanMonumentBiomeList[] = {ocean, deepOcean, river, frozenOcean, frozenRiver};
2018-03-17 16:53:16 +00:00
static const int villageBiomeList[] = {plains, desert, savanna, taiga};
static const int mansionBiomeList[] = {roofedForest, roofedForest+128};
static const int achievementBiomes[] =
{
ocean, plains, desert, extremeHills, forest, taiga, swampland, river, /*hell, sky,*/ // 0-9
/*frozenOcean,*/ frozenRiver, icePlains, iceMountains, mushroomIsland, mushroomIslandShore, beach, desertHills, forestHills, taigaHills, // 10-19
/*extremeHillsEdge,*/ jungle, jungleHills, jungleEdge, deepOcean, stoneBeach, coldBeach, birchForest, birchForestHills, roofedForest, // 20-29
coldTaiga, coldTaigaHills, megaTaiga, megaTaigaHills, extremeHillsPlus, savanna, savannaPlateau, mesa, mesaPlateau_F, mesaPlateau // 30-39
};
2018-03-05 21:20:54 +00:00
STRUCT(Pos)
{
int x, z;
};
extern Biome biomes[256];
/******************************** SEED FINDING *********************************
*
* If we want to find rare seeds that meet multiple custom criteria then we
* should test each condition, starting with the one that is the cheapest
* to test for, while ruling out the most seeds.
*
* Biome checks are quite expensive and should be applied late in the
* condition chain (to avoid as many unnecessary checks as possible).
* Fortunately we can often rule out vast amounts of seeds before hand.
*/
/*************************** Quad-Structure Checks *****************************
2018-03-05 21:20:54 +00:00
*
* Several tricks can be applied to determine candidate seeds for quad
* temples (inc. witch huts).
*
* Minecraft uses a 48-bit pseudo random number generator (PRNG) to determine
* the position of it's structures. The remaining top 16 bits do not influence
2018-07-04 16:18:21 +01:00
* the structure positioning. Additionally the position of most structures in a
2018-03-05 21:20:54 +00:00
* world can be translated by applying the following transformation to the
* seed:
*
2018-07-04 16:18:21 +01:00
* seed2 = seed1 - dregX * 341873128712 - dregZ * 132897987541;
2018-03-05 21:20:54 +00:00
*
* Here seed1 and seed2 have the same structure positioning, but moved by a
2018-07-04 16:18:21 +01:00
* region offset of (dregX,dregZ). [a region is 32x32 chunks].
2018-03-05 21:20:54 +00:00
*
* For a quad-structure, we mainly care about relative positioning, so we can
* get away with just checking the regions near the origin: (0,0),(0,1),(1,0)
* and (1,1) and then move the structures to the desired position.
2018-03-05 21:20:54 +00:00
*
* Lastly we can recognise a that the transformation of relative region-
* coordinates imposes some restrictions in the PRNG, such that
* perfect-position quad-structure-seeds can only have certain values for the
* lower 16-bits in their seeds.
*
*
** The Set of all Quad-Witch-Huts
*
* These conditions only leave 32 free bits which can comfortably be brute-
* forced to get the entire set of quad-structure candidates. Each of the seeds
* found this way describes an entire set of possible quad-witch-huts (with
* degrees of freedom for region-transposition, and the top 16-bit bits).
2018-03-05 21:20:54 +00:00
*/
2018-07-04 16:18:21 +01:00
/* isQuadBase
* ----------
* Calls the correct quad-base finder for the structure config, if available.
* (Exits program otherwise.)
*/
int isQuadBase(const StructureConfig sconf, const int64_t seed, const int64_t qual);
2018-03-18 13:10:13 +00:00
2018-07-04 16:18:21 +01:00
/* moveStructure
* -------------
2018-03-17 16:53:16 +00:00
* Transposes a base seed such that structures are moved by the specified region
* vector, (regX, regZ).
*/
2018-07-04 16:48:05 +01:00
int64_t moveStructure(const int64_t baseSeed, const int regX, const int regZ);
2018-03-05 21:20:54 +00:00
2018-03-17 16:53:16 +00:00
/* loadSavedSeeds
* --------------
* Loads a list of seeds from a file. The seeds should be written as decimal
* UFT-8 numbers separated by newlines.
*
* fnam: file path
* scnt: number of valid seeds found in the file = length of the returned buffer
*
* Return a pointer to dynamically allocated seed list.
*/
2018-07-04 16:48:05 +01:00
int64_t *loadSavedSeeds(const char *fnam, int64_t *scnt);
2018-03-05 21:20:54 +00:00
2018-07-04 16:18:21 +01:00
/* search4QuadBases
* ----------------
* Starts a multi-threaded search for structure base seeds of the specified
2018-03-17 16:53:16 +00:00
* quality (chunk tolerance). The result is saved in a file of path 'fnam'.
*/
void search4QuadBases(const char *fnam, int threads,
const StructureConfig structureConfig, int quality);
2018-03-05 21:20:54 +00:00
2018-03-17 16:53:16 +00:00
2018-03-05 21:20:54 +00:00
/**************************** General Biome Checks *****************************
*/
/* getBiomeAtPos
* -------------
2018-03-17 16:53:16 +00:00
* Returns the biome for the specified block position.
* (Alternatives should be considered in performance critical code.)
* This function is not threadsafe.
2018-03-05 21:20:54 +00:00
*/
2018-03-17 16:53:16 +00:00
int getBiomeAtPos(const LayerStack g, const Pos pos);
/* getOceanRuinPos
* ---------------
* Fast implementation for finding the block position at which an ocean ruin
* generation attempt will occur in the specified region.
2018-03-17 16:53:16 +00:00
*/
Pos getOceanRuinPos(int64_t seed, const int64_t regionX, const int64_t regionZ);
2018-03-05 21:20:54 +00:00
2018-07-04 16:18:21 +01:00
/* getStructurePos
* ---------------
* Fast implementation for finding the block position at which the structure
2018-03-05 21:20:54 +00:00
* generation attempt will occur in the specified region.
2018-07-04 16:18:21 +01:00
* This function applies for scattered-feature structureSeeds and villages.
2018-03-05 21:20:54 +00:00
*/
Pos getStructurePos(StructureConfig config, int64_t seed,
const int64_t regionX, const int64_t regionZ);
2018-03-05 21:20:54 +00:00
/* getStructureChunkInRegion
* -------------------------
* Finds the chunk position within the specified region (a square region of
* chunks depending on structure type) where the structure generation attempt
* will occur.
*
* This function applies for scattered-feature structureSeeds and villages.
*/
Pos getStructureChunkInRegion(StructureConfig config, int64_t seed,
const int regionX, const int regionZ);
2018-03-17 16:53:16 +00:00
/* getLargeStructurePos
* --------------------
2018-03-17 16:53:16 +00:00
* Fast implementation for finding the block position at which the ocean
* monument or woodland mansion generation attempt will occur in the
* specified region.
2018-03-17 16:53:16 +00:00
*/
Pos getLargeStructurePos(StructureConfig config, int64_t seed,
const int64_t regionX, const int64_t regionZ);
2018-03-17 16:53:16 +00:00
/* getLargeStructureChunkInRegion
* ------------------------------
* Fast implementation for finding the chunk position at which the ocean
* monument or woodland mansion generation attempt will occur in the
* specified region.
2018-03-17 16:53:16 +00:00
*/
Pos getLargeStructureChunkInRegion(StructureConfig config, int64_t seed,
const int64_t regionX, const int64_t regionZ);
2018-03-17 16:53:16 +00:00
/************************ Biome Checks for Structures **************************
*
* Scattered features only do a simple check of the biome at the block position
* of the structure origin (i.e. the north-west corner). Before 1.13 the type of
* structure was determined by the biome, while in 1.13 the scattered feature
* positions are calculated separately for each type. However, the biome
* requirements remain the same:
*
* Desert Pyramid: desert or desertHills
* Igloo : icePlains or coldTaiga
* Jungle Pyramid: jungle or jungleHills
* Swamp Hut : swampland
*
* Similarly, Ocean Ruins and Shipwrecks require any oceanic biome at their
* block position.
*
* Villages, Monuments and Mansions on the other hand require a certain area to
* be of a valid biome and the check is performed at a 1:4 scale instead of 1:1.
* (Actually the area for villages has a radius zero, which means it is a simple
* biome check at a 1:4 scale.)
*/
2018-03-17 16:53:16 +00:00
/* findBiomePosition
* -----------------
* Finds a suitable pseudo-random location in the specified area.
* Used to determine the positions of spawn and stongholds.
* Warning: accurate, but slow!
*
* g : generator layer stack
* cache : biome buffer, set to NULL for temporary allocation
* centreX, centreZ : origin for the search
* range : 'radius' of the search
* isValid : boolean array of valid biome ids (size = 256)
* seed : seed used for the RNG
* (initialise RNG using setSeed(&seed))
* passes : number of valid biomes passed, set to NULL to ignore this
*/
Pos findBiomePosition(
const LayerStack g,
int *cache,
const int centerX,
const int centerZ,
const int range,
const int *isValid,
2018-07-04 16:48:05 +01:00
int64_t *seed,
2018-03-17 16:53:16 +00:00
int *passes
);
/* findStrongholds_pre19
* ---------------------
* Finds the 3 stronghold positions for the specified world seed up to MC 1.9.
* Warning: Slow!
*
* g : generator layer stack [world seed will be updated]
* cache : biome buffer, set to NULL for temporary allocation
* locations : output block positions for the 3 strongholds
* worldSeed : world seed used for the generator
*/
2018-07-04 16:48:05 +01:00
void findStrongholds_pre19(LayerStack *g, int *cache, Pos *locations, int64_t worldSeed);
2018-03-17 16:53:16 +00:00
/* findStrongholds
* ---------------
2018-07-04 13:47:46 -07:00
* Finds up to 128 strongholds which generate since MC 1.9. Returns the number
* of strongholds found within the specified radius.
* Warning: Slow!
*
* g : generator layer stack [world seed will be updated]
* cache : biome buffer, set to NULL for temporary allocation
* locations : output block positions for the 128 strongholds
* worldSeed : world seed used for the generator
2018-07-05 21:05:34 +01:00
* maxRadius : Stop searching if the radius exceeds this value in meters. Set to
* 0 to return all strongholds.
*/
int findStrongholds(LayerStack *g, int *cache, Pos *locations, int64_t worldSeed, int maxRadius);
2018-03-17 16:53:16 +00:00
/* getSpawn
* --------
* Finds the spawn point in the world.
* Warning: Slow, and may be inaccurate because the world spawn depends on
* grass blocks!
*
* g : generator layer stack [world seed will be updated]
* cache : biome buffer, set to NULL for temporary allocation
* worldSeed : world seed used for the generator
*/
2018-07-04 16:48:05 +01:00
Pos getSpawn(LayerStack *g, int *cache, int64_t worldSeed);
2018-03-17 16:53:16 +00:00
/* areBiomesViable
* ---------------
* Determines if the given area contains only biomes specified by 'biomeList'.
* Used to determine the positions of villages, ocean monument and mansions.
* Warning: accurate, but slow!
*
* g : generator layer stack
* cache : biome buffer, set to NULL for temporary allocation
* posX, posZ : centre for the check
* radius : 'radius' of the check area
* isValid : boolean array of valid biome ids (size = 256)
*/
int areBiomesViable(
const LayerStack g,
int *cache,
const int posX,
const int posZ,
const int radius,
const int *isValid
);
/* isViableFeaturePos
2018-03-17 16:53:16 +00:00
* isViableVillagePos
* isViableOceanMonumentPos
* isViableMansionPos
* ------------------------
* Perform the biome check at the specified block coordinates to determine
* whether the corresponding structure would spawn. You can get the block
* positions using the appropriate getXXXPos() function.
*
* g : generator layer stack [set seed beforehand with applySeed()]
* cache : biome buffer, set to NULL for temporary allocation
* blockX, blockZ : block coordinates
*
* In the case of isViableFeaturePos() the 'type' argument specifies the type of
* scattered feature the check is performed for.
*
* The return value is non-zero if the position is valid.
2018-03-17 16:53:16 +00:00
*/
int isViableFeaturePos(const int type, const LayerStack g, int *cache, const int blockX, const int blockZ);
int isViableVillagePos(const LayerStack g, int *cache, const int blockX, const int blockZ);
int isViableOceanMonumentPos(const LayerStack g, int *cache, const int blockX, const int blockZ);
int isViableMansionPos(const LayerStack g, int *cache, const int blockX, const int blockZ);
2018-03-17 16:53:16 +00:00
/* getBiomeRadius
* --------------
* Finds the smallest radius (by square around the origin) at which all the
* specified biomes are present. The input map is assumed to be a square of
* side length 'sideLen'.
*
* map : square biome map to be tested
* sideLen : side length of the square map (should be 2*radius+1)
* biomes : list of biomes to check for
* bnum : length of 'biomes'
* ignoreMutations : flag to count mutated biomes as their original form
*
* Return the radius on the square map that covers all biomes in the list.
* If the map does not contain all the specified biomes, -1 is returned.
*/
int getBiomeRadius(
const int *map,
const int mapSide,
const int *biomes,
const int bnum,
const int ignoreMutations);
/******************************** Seed Filters *********************************
*/
2018-03-05 21:20:54 +00:00
/* filterAllTempCats
* -----------------
* Looks through the seeds in 'seedsIn' and copies those for which all
* temperature categories are present in the 3x3 area centred on the specified
* coordinates into 'seedsOut'. The map scale at this layer is 1:1024.
*
2018-03-17 16:53:16 +00:00
* g : generator layer stack, (NOTE: seed will be modified)
* cache : biome buffer, set to NULL for temporary allocation
* seedsIn : list of seeds to check
* seedsOut : output buffer for the candidate seeds
* seedCnt : number of seeds in 'seedsIn'
* centX, centZ: search origin centre (in 1024 block units)
2018-03-05 21:20:54 +00:00
*
* Returns the number of found candidates.
*/
2018-07-04 16:48:05 +01:00
int64_t filterAllTempCats(
2018-03-17 16:53:16 +00:00
LayerStack *g,
int *cache,
2018-07-04 16:48:05 +01:00
const int64_t *seedsIn,
int64_t *seedsOut,
const int64_t seedCnt,
2018-03-17 16:53:16 +00:00
const int centX,
const int centZ);
2018-03-05 21:20:54 +00:00
/* filterAllMajorBiomes
* --------------------
* Looks through the list of seeds in 'seedsIn' and copies those that have all
* major overworld biomes in the specified area into 'seedsOut'. These checks
* are done at a scale of 1:256.
*
2018-03-17 16:53:16 +00:00
* g : generator layer stack, (NOTE: seed will be modified)
* cache : biome buffer, set to NULL for temporary allocation
* seedsIn : list of seeds to check
* seedsOut : output buffer for the candidate seeds
* seedCnt : number of seeds in 'seedsIn'
* pX, pZ : search starting coordinates (in 256 block units)
* sX, sZ : size of the searching area (in 256 block units)
*
2018-03-05 21:20:54 +00:00
* Returns the number of seeds found.
*/
2018-07-04 16:48:05 +01:00
int64_t filterAllMajorBiomes(
2018-03-17 16:53:16 +00:00
LayerStack *g,
int *cache,
2018-07-04 16:48:05 +01:00
const int64_t *seedsIn,
int64_t *seedsOut,
const int64_t seedCnt,
2018-03-17 16:53:16 +00:00
const int pX,
const int pZ,
2018-07-04 17:27:45 +01:00
const unsigned int sX,
const unsigned int sZ
2018-03-17 16:53:16 +00:00
);
2018-03-05 21:20:54 +00:00
static inline const char *int2binstr(int x, int bit)
{
static char str[33];
str[0] = '\0';
for(bit = (1 << bit); bit > 0; bit >>= 1)
strcat(str, ((x & bit) == bit) ? "1" : "0");
return str;
}
#endif /* FINDERS_H_ */