cubiomes/find_quadhuts.c

171 lines
5.4 KiB
C
Raw Normal View History

2018-03-05 21:20:54 +00:00
/**
* This is an example program that demonstrates how to find seeds with a
* quad-witch-hut located around the specified region (512x512 area).
*
* It uses some optimisations that cause it miss a small number of seeds, in
* exchange for a major speed upgrade. (~99% accuracy, ~1200% speed)
*/
#include "finders.h"
#include "generator.h"
#include "layers.h"
#include <unistd.h>
int main(int argc, char *argv[])
{
// Always initialize the biome list before starting any seed finder or
// biome generator.
initBiomes();
2018-03-17 16:53:16 +00:00
// Translate the positions to the desired regions.
int regPosX;
int regPosZ;
if(argc > 2)
{
if(sscanf(argv[1], "%d", &regPosX) != 1) regPosX = 0;
if(sscanf(argv[2], "%d", &regPosZ) != 1) regPosZ = 0;
}
else
{
printf("Usage:\n"
"find_quadhuts [regionX] [regionZ]\n"
"Defaulting to origin.\n\n");
}
regPosX -= 1;
regPosZ -= 1;
2018-03-18 13:10:13 +00:00
const char *seedFileName = "./seeds/quadbases_Q1.txt";
2018-03-05 21:20:54 +00:00
if(access(seedFileName, F_OK))
{
printf("Seed base file does not exist: Creating new one.\n"
"This may take a few minutes...\n");
int threads = 6;
int quality = 1;
baseQuadTempleSearch(seedFileName, threads, quality);
}
long i, j, qhcnt;
long base, seed;
long *qhcandidates = loadSavedSeeds(seedFileName, &qhcnt);
2018-03-17 16:53:16 +00:00
LayerStack g = setupGenerator();
2018-03-05 21:20:54 +00:00
2018-03-17 16:53:16 +00:00
Layer *lFilterBiome = &g.layers[L_BIOME_256];
int *biomeCache = allocCache(lFilterBiome, 3, 3);
2018-03-05 21:20:54 +00:00
// Load the positions of the four structures that make up the quad-structure
// so we can test the biome at these positions.
Pos qhpos[4];
// Setup a dummy layer for Layer 19: Biome.
2018-03-17 16:53:16 +00:00
Layer layerBiomeDummy;
setupLayer(256, &layerBiomeDummy, NULL, 200, NULL);
2018-03-05 21:20:54 +00:00
int areaX = (regPosX << 1) + 1;
int areaZ = (regPosZ << 1) + 1;
// Search for a swamp at the structure positions
for(i = 0; i < qhcnt; i++)
{
base = moveTemple(qhcandidates[i], regPosX, regPosZ);
qhpos[0] = getTemplePos(base, 0+regPosX, 0+regPosZ);
qhpos[1] = getTemplePos(base, 0+regPosX, 1+regPosZ);
qhpos[2] = getTemplePos(base, 1+regPosX, 0+regPosZ);
qhpos[3] = getTemplePos(base, 1+regPosX, 1+regPosZ);
/*
for(j = 0; j < 4; j++)
{
printf("(%d,%d) ", qhpos[j].x, qhpos[j].z);
}
printf("\n");
//*/
// This little magic code checks if there is a meaningful chance for
// this seed base to generate swamps in the area.
// The idea is that the conversion from Lush temperature to swampland is
// independent of surroundings, so we can test the conversion
// beforehand. Furthermore biomes tend to leak into the negative
// coordinates because of the Zoom layers, so the majority of hits will
// occur when SouthEast corner (at a 1:256 scale) of the quad-hut has a
// swampland. (This assumption misses about 1 in 500 quad-hut seeds.)
// Finally, here we also exploit that the minecraft random number
// generator is quite bad, such that for the "mcNextRand() mod 6" check
// it has a period pattern of ~3 on the high seed-bits.
for(j = 0; j < 5; j++)
{
seed = base + ((j+0x53) << 48);
2018-03-17 16:53:16 +00:00
setWorldSeed(&layerBiomeDummy, seed);
setChunkSeed(&layerBiomeDummy, areaX+1, areaZ+1);
if(mcNextInt(&layerBiomeDummy, 6) == 5)
2018-03-05 21:20:54 +00:00
break;
}
if(j >= 5) continue;
long hits = 0, swpc;
for(j = 0; j < 0x10000; j++)
{
seed = base + (j << 48);
/** Pre-Generation Checks **/
// We can check that at least one swamp could generate in this area
// before doing the biome generator checks.
2018-03-17 16:53:16 +00:00
setWorldSeed(&layerBiomeDummy, seed);
2018-03-05 21:20:54 +00:00
2018-03-17 16:53:16 +00:00
setChunkSeed(&layerBiomeDummy, areaX+1, areaZ+1);
if(mcNextInt(&layerBiomeDummy, 6) != 5)
2018-03-05 21:20:54 +00:00
continue;
// This seed base does not seem to contain many quad huts, so make
// a more detailed analysis of the surroundings and see if there is
// enough potential for more swamps to justify searching fruther.
if(hits == 0 && (j & 0xfff) == 0xfff)
{
swpc = 0;
2018-03-17 16:53:16 +00:00
setChunkSeed(&layerBiomeDummy, areaX, areaZ+1);
swpc += mcNextInt(&layerBiomeDummy, 6) == 5;
setChunkSeed(&layerBiomeDummy, areaX+1, areaZ);
swpc += mcNextInt(&layerBiomeDummy, 6) == 5;
setChunkSeed(&layerBiomeDummy, areaX, areaZ);
swpc += mcNextInt(&layerBiomeDummy, 6) == 5;
2018-03-05 21:20:54 +00:00
if(swpc < (j > 0x1000 ? 2 : 1)) break;
}
// Dismiss seeds that don't have a swamp near the quad temple.
2018-03-17 16:53:16 +00:00
setWorldSeed(lFilterBiome, seed);
genArea(lFilterBiome, biomeCache, (regPosX<<1)+2, (regPosZ<<1)+2, 1, 1);
2018-03-05 21:20:54 +00:00
if(biomeCache[0] != swampland)
continue;
applySeed(&g, seed);
2018-03-17 16:53:16 +00:00
if(getBiomeAtPos(g, qhpos[0]) != swampland) continue;
if(getBiomeAtPos(g, qhpos[1]) != swampland) continue;
if(getBiomeAtPos(g, qhpos[2]) != swampland) continue;
if(getBiomeAtPos(g, qhpos[3]) != swampland) continue;
2018-03-05 21:20:54 +00:00
printf("%ld\n", seed);
hits++;
}
}
free(biomeCache);
2018-03-17 16:53:16 +00:00
freeGenerator(g);
2018-03-05 21:20:54 +00:00
return 0;
}