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>
|
|
|
|
|
|
|
|
|
|
|
|
#define THREADS 6
|
2018-07-04 17:28:27 +01:00
|
|
|
#define SEEDMAX (1LL << 48)
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:18:21 +01:00
|
|
|
#define FEATURE_SEED 14357617
|
|
|
|
#define VILLAGE_SEED 10387312
|
|
|
|
#define MONUMENT_SEED 10387313
|
|
|
|
#define MANSION_SEED 10387319
|
|
|
|
|
|
|
|
/* 1.13 separated feature seeds by type */
|
|
|
|
#define DESERT_PYRAMID_SEED 14357617
|
|
|
|
#define IGLOO_SEED 14357618
|
|
|
|
#define JUNGLE_PYRAMID_SEED 14357619
|
|
|
|
#define SWAMP_HUT_SEED 14357620
|
|
|
|
#define OCEAN_RUIN_SEED 14357621
|
|
|
|
|
|
|
|
enum {Desert_Pyramid=1, Igloo, Jungle_Pyramid, Swamp_Hut, Ocean_Ruin};
|
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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-06-30 16:34:31 -07:00
|
|
|
/*************************** 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
|
|
|
*
|
2018-06-30 16:34:31 -07: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
|
|
|
*
|
2018-06-30 16:34:31 -07: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
|
2018-06-30 16:34:31 -07:00
|
|
|
* 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
|
|
|
|
2018-03-18 13:10:13 +00:00
|
|
|
// helper functions
|
2018-07-04 16:48:05 +01:00
|
|
|
int isQuadFeatureBase(const int64_t structureSeed, const int64_t seed,
|
|
|
|
const int64_t lower, const int64_t upper);
|
|
|
|
int isTriFeatureBase(const int64_t structureSeed, const int64_t seed,
|
|
|
|
const int64_t lower, const int64_t upper);
|
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'.
|
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
void search4QuadBases(const char *fnam, int threads, const int64_t structureSeed,
|
2018-07-04 16:18:21 +01:00
|
|
|
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.)
|
2018-07-02 14:30:12 -07:00
|
|
|
* 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);
|
|
|
|
|
|
|
|
|
2018-07-04 16:18:21 +01:00
|
|
|
/* getStructureChunkInRegion
|
|
|
|
* -------------------------
|
2018-03-17 16:53:16 +00:00
|
|
|
* Finds the chunk position within the specified region (32x32 chunks) where
|
2018-07-04 16:18:21 +01:00
|
|
|
* the structure generation attempt will occur.
|
|
|
|
* This function applies for scattered-feature structureSeeds and villages.
|
2018-03-17 16:53:16 +00:00
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
Pos getStructureChunkInRegion(const int64_t structureSeed, int64_t seed,
|
2018-07-04 16:18:21 +01:00
|
|
|
const int regionX, const int 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
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
Pos getStructurePos(const int64_t structureSeed, int64_t seed, const int64_t regionX,
|
|
|
|
const int64_t regionZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
|
|
|
|
/* getOceanMonumentPos
|
|
|
|
* -------------------
|
|
|
|
* Fast implementation for finding the block position at which the ocean
|
|
|
|
* monument generation attempt will occur in the specified region.
|
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
Pos getOceanMonumentPos(int64_t seed, const int64_t regionX, const int64_t regionZ);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
|
|
|
/* getMansionPos
|
|
|
|
* -------------
|
|
|
|
* Fast implementation for finding the block position at which the woodland
|
|
|
|
* mansions generation attempt will occur in the specified 80x80 chunk area.
|
|
|
|
*
|
|
|
|
* area80X, area80Z: area coordinates in units 1280 blocks (= 80 chunks)
|
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
Pos getMansionPos(int64_t seed, const int64_t area80X, const int64_t area80Z);
|
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
|
|
|
|
|
|
|
/* 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
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-06-30 16:34:31 -07:00
|
|
|
/* isViableWitchHutPos
|
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
|
|
|
|
*
|
|
|
|
* The return value is non-zero if the position is valid, and in the case of
|
2018-06-30 16:34:31 -07:00
|
|
|
* isViableWitchHutPos() the return value is an enum of the temple type.
|
2018-03-17 16:53:16 +00:00
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
int isViableWitchHutPos(const LayerStack g, int *cache, const int64_t blockX, const int64_t blockZ);
|
|
|
|
int isViableVillagePos(const LayerStack g, int *cache, const int64_t blockX, const int64_t blockZ);
|
|
|
|
int isViableOceanMonumentPos(const LayerStack g, int *cache, const int64_t blockX, const int64_t blockZ);
|
|
|
|
int isViableMansionPos(const LayerStack g, int *cache, const int64_t blockX, const int64_t 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
|
|
|
|
|
|
|
/********************** C copy of the Java Random methods **********************
|
|
|
|
*/
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline void setSeed(int64_t *seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 17:28:27 +01:00
|
|
|
*seed = (*seed ^ 0x5DEECE66DLL) & ((1LL << 48) - 1);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int next(int64_t *seed, const int bits)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 17:28:27 +01:00
|
|
|
*seed = (*seed * 0x5DEECE66DLL + 0xBLL) & ((1LL << 48) - 1);
|
2018-03-05 21:20:54 +00:00
|
|
|
return (int) (*seed >> (48 - bits));
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int nextInt(int64_t *seed, const int n)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
int bits, val;
|
|
|
|
do {
|
|
|
|
bits = next(seed, 31);
|
|
|
|
val = bits % n;
|
|
|
|
}
|
|
|
|
while(bits - val + (n - 1) < 0);
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int64_t nextint64_t(int64_t *seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
return ((int64_t) next(seed, 32) << 32) + next(seed, 32);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline float nextFloat(int64_t *seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
return next(seed, 24) / (float) (1 << 24);
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline double nextDouble(int64_t *seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 17:28:27 +01:00
|
|
|
return (((int64_t) next(seed, 26) << 27) + next(seed, 27)) / (double) (1LL << 53);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Custom, faster alternative for the first and second call to nextInt(24)
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int firstInt24(int64_t seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
seed ^= 0x5deece66d;
|
|
|
|
seed = (seed * 0x5deece66d) & 0xffffffffffff;
|
|
|
|
seed >>= 17;
|
|
|
|
return seed % 24;
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int secondInt24(int64_t seed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
seed ^= 0x5deece66d;
|
|
|
|
seed = (seed * 0x5deece66d + 0xB) & 0xffffffffffff;
|
|
|
|
seed = (seed * 0x5deece66d) & 0xffffffffffff;
|
|
|
|
seed >>= 17;
|
|
|
|
return seed % 24;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* invSeed48()
|
|
|
|
* -----------
|
|
|
|
* Returns the previous 48-bit seed which will generate 'nseed'.
|
|
|
|
* The upper 16 bits are ignored, both here and in the generator.
|
|
|
|
*/
|
2018-07-04 16:48:05 +01:00
|
|
|
static inline int64_t invSeed48(int64_t nseed)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t x = 0x5deece66d;
|
2018-07-04 17:28:27 +01:00
|
|
|
const int64_t xinv = 0xdfe05bcb1365LL;
|
|
|
|
const int64_t y = 0xbLL;
|
|
|
|
const int64_t m48 = 0xffffffffffffLL;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t a = nseed >> 32;
|
2018-07-04 17:28:27 +01:00
|
|
|
int64_t b = nseed & 0xffffffffLL;
|
|
|
|
if(b & 0x80000000LL) a++;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t q = ((b << 16) - y - (a << 16)*x) & m48;
|
|
|
|
for(int64_t k = 0; k <= 5; k++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t d = (x - (q + (k << 48))) % x;
|
2018-03-05 21:20:54 +00:00
|
|
|
d = (d + x) % x; // force the modulo and keep it positive
|
|
|
|
if(d < 65536)
|
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t c = ((q + d) * xinv) & m48;
|
2018-03-05 21:20:54 +00:00
|
|
|
if(c < 65536)
|
|
|
|
{
|
|
|
|
return ((((a << 16) + c) - y) * xinv) & m48;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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_ */
|