2018-03-05 21:20:54 +00:00
|
|
|
#include "finders.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <stdlib.h>
|
2018-03-17 16:53:16 +00:00
|
|
|
#include <math.h>
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
STRUCT(quad_threadinfo_t)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t start, end;
|
2018-07-25 08:54:33 +02:00
|
|
|
StructureConfig sconf;
|
2018-03-05 21:20:54 +00:00
|
|
|
int threadID;
|
|
|
|
int quality;
|
|
|
|
const char *fnam;
|
2018-07-28 16:23:47 +02:00
|
|
|
};
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Globals
|
|
|
|
//==============================================================================
|
|
|
|
|
|
|
|
|
|
|
|
Biome biomes[256];
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t lowerBaseBitsQ1[] = // for quad-structure with quality 1
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
0x3f18,0x520a,0x751a,0x9a0a
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t lowerBaseBitsQ2[] = // for quad-structure with quality 2
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
0x0770,0x0775,0x07ad,0x07b2,0x0c3a,0x0c58,0x0cba,0x0cd8,0x0e38,
|
|
|
|
0x0e5a,0x0ed8,0x0eda,0x111c,0x1c96,0x2048,0x20e8,0x2248,0x224a,
|
|
|
|
0x22c8,0x258d,0x272d,0x2732,0x2739,0x2758,0x275d,0x27c8,0x27c9,
|
|
|
|
0x2aa9,0x2c3a,0x2cba,0x2eb8,0x308c,0x3206,0x371a,0x3890,0x3d0a,
|
|
|
|
0x3f18,0x4068,0x40ca,0x40e8,0x418a,0x4248,0x426a,0x42ea,0x4732,
|
|
|
|
0x4738,0x4739,0x4765,0x4768,0x476a,0x47b0,0x47b5,0x47d4,0x47d9,
|
|
|
|
0x47e8,0x4c58,0x4e38,0x4eb8,0x4eda,0x5118,0x520a,0x5618,0x5918,
|
|
|
|
0x591d,0x5a08,0x5e18,0x5f1c,0x60ca,0x6739,0x6748,0x6749,0x6758,
|
|
|
|
0x6776,0x67b4,0x67b9,0x67c9,0x67d8,0x67dd,0x67ec,0x6c3a,0x6c58,
|
|
|
|
0x6cba,0x6d9a,0x6e5a,0x6ed8,0x6eda,0x7108,0x717a,0x751a,0x7618,
|
|
|
|
0x791c,0x8068,0x8186,0x8248,0x824a,0x82c8,0x82ea,0x8730,0x8739,
|
|
|
|
0x8748,0x8768,0x87b9,0x87c9,0x87ce,0x87d9,0x898d,0x8c3a,0x8cda,
|
|
|
|
0x8e38,0x8eb8,0x951e,0x9718,0x9a0a,0xa04a,0xa068,0xa0ca,0xa0e8,
|
|
|
|
0xa18a,0xa26a,0xa2e8,0xa2ea,0xa43d,0xa4e1,0xa589,0xa76d,0xa7ac,
|
|
|
|
0xa7b1,0xa7ed,0xa85d,0xa86d,0xaa2d,0xb1f8,0xb217,0xb9f8,0xba09,
|
|
|
|
0xba17,0xbb0f,0xc54c,0xc6f9,0xc954,0xc9ce,0xd70b,0xd719,0xdc55,
|
|
|
|
0xdf0b,0xe1c4,0xe556,0xe589,0xea5d
|
2018-03-05 21:20:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Saving & Loading Seeds
|
|
|
|
//==============================================================================
|
|
|
|
|
|
|
|
|
|
|
|
int64_t *loadSavedSeeds(const char *fnam, int64_t *scnt)
|
|
|
|
{
|
|
|
|
FILE *fp = fopen(fnam, "r");
|
|
|
|
|
|
|
|
int64_t seed;
|
|
|
|
int64_t *baseSeeds;
|
|
|
|
|
|
|
|
if (fp == NULL)
|
|
|
|
{
|
|
|
|
perror("ERR loadSavedSeeds: ");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*scnt = 0;
|
|
|
|
|
|
|
|
while (!feof(fp))
|
|
|
|
{
|
|
|
|
if (fscanf(fp, "%"PRId64, &seed) == 1) (*scnt)++;
|
|
|
|
else while (!feof(fp) && fgetc(fp) != '\n');
|
|
|
|
}
|
|
|
|
|
|
|
|
baseSeeds = (int64_t*) calloc(*scnt, sizeof(*baseSeeds));
|
|
|
|
|
|
|
|
rewind(fp);
|
|
|
|
|
|
|
|
for (int64_t i = 0; i < *scnt && !feof(fp);)
|
|
|
|
{
|
|
|
|
if (fscanf(fp, "%"PRId64, &baseSeeds[i]) == 1) i++;
|
|
|
|
else while (!feof(fp) && fgetc(fp) != '\n');
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
return baseSeeds;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
// Multi-Structure Checks
|
|
|
|
//==============================================================================
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int isQuadFeatureBase(const StructureConfig sconf, const int64_t seed, const int qual)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
// seed offsets for the regions (0,0) to (1,1)
|
2018-07-25 08:54:33 +02:00
|
|
|
const int64_t reg00base = sconf.seed;
|
|
|
|
const int64_t reg01base = 341873128712 + sconf.seed;
|
|
|
|
const int64_t reg10base = 132897987541 + sconf.seed;
|
|
|
|
const int64_t reg11base = 341873128712 + 132897987541 + sconf.seed;
|
|
|
|
|
|
|
|
const int range = sconf.chunkRange;
|
|
|
|
const int upper = range - qual - 1;
|
|
|
|
const int lower = qual;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t s;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg00base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper) return 0;
|
|
|
|
|
|
|
|
s = (reg01base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper) return 0;
|
|
|
|
|
|
|
|
s = (reg10base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower) return 0;
|
|
|
|
|
|
|
|
s = (reg11base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower) return 0;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int isTriFeatureBase(const StructureConfig sconf, const int64_t seed, const int qual)
|
2018-03-18 13:10:13 +00:00
|
|
|
{
|
|
|
|
// seed offsets for the regions (0,0) to (1,1)
|
2018-07-25 08:54:33 +02:00
|
|
|
const int64_t reg00base = sconf.seed;
|
|
|
|
const int64_t reg01base = 341873128712 + sconf.seed;
|
|
|
|
const int64_t reg10base = 132897987541 + sconf.seed;
|
|
|
|
const int64_t reg11base = 341873128712 + 132897987541 + sconf.seed;
|
|
|
|
|
|
|
|
const int range = sconf.chunkRange;
|
|
|
|
const int upper = range - qual - 1;
|
|
|
|
const int lower = qual;
|
2018-03-18 13:10:13 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t s;
|
2018-03-18 13:10:13 +00:00
|
|
|
int missing = 0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg00base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper ||
|
|
|
|
(int)(((s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff) >> 17) % range < upper)
|
2018-03-18 13:10:13 +00:00
|
|
|
{
|
|
|
|
missing++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg01base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower ||
|
|
|
|
(int)(((s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff) >> 17) % range < upper)
|
2018-03-18 13:10:13 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (missing) return 0;
|
2018-03-18 13:10:13 +00:00
|
|
|
missing++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg10base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range < upper ||
|
|
|
|
(int)(((s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff) >> 17) % range > lower)
|
2018-03-18 13:10:13 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (missing) return 0;
|
2018-03-18 13:10:13 +00:00
|
|
|
missing++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg11base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
if ((int)(s >> 17) % range > lower ||
|
|
|
|
(int)(((s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff) >> 17) % range > lower)
|
2018-03-18 13:10:13 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (missing) return 0;
|
2018-03-18 13:10:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int isLargeQuadBase(const StructureConfig sconf, const int64_t seed, const int qual)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
// seed offsets for the regions (0,0) to (1,1)
|
2018-07-25 08:54:33 +02:00
|
|
|
const int64_t reg00base = sconf.seed;
|
|
|
|
const int64_t reg01base = 341873128712 + sconf.seed;
|
|
|
|
const int64_t reg10base = 132897987541 + sconf.seed;
|
|
|
|
const int64_t reg11base = 341873128712 + 132897987541 + sconf.seed;
|
|
|
|
|
|
|
|
const int range = sconf.chunkRange;
|
|
|
|
const int rmin1 = range - 1;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int64_t s;
|
|
|
|
int p;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg00base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg01base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg10base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg11base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int isLargeTriBase(const StructureConfig sconf, const int64_t seed, const int qual)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
// seed offsets for the regions (0,0) to (1,1)
|
2018-07-25 08:54:33 +02:00
|
|
|
const int64_t reg00base = sconf.seed;
|
|
|
|
const int64_t reg01base = 341873128712 + sconf.seed;
|
|
|
|
const int64_t reg10base = 132897987541 + sconf.seed;
|
|
|
|
const int64_t reg11base = 341873128712 + 132897987541 + sconf.seed;
|
|
|
|
|
|
|
|
const int range = sconf.chunkRange;
|
|
|
|
const int rmin1 = range - 1;
|
|
|
|
|
|
|
|
int64_t s;
|
|
|
|
int p;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
|
|
|
int incomplete = 0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg00base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) goto incomp11;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) goto incomp11;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) goto incomp11;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) goto incomp11;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (0)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
incomp11:
|
|
|
|
incomplete = 1;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg01base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp01;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp01;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) goto incomp01;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) goto incomp01;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (0)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
incomp01:
|
2018-07-28 16:23:47 +02:00
|
|
|
if (incomplete) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
incomplete = 2;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg10base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < rmin1-qual) goto incomp10;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p < 2*rmin1-qual) goto incomp10;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp10;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp10;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (0)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
incomp10:
|
2018-07-28 16:23:47 +02:00
|
|
|
if (incomplete) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
incomplete = 3;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
s = (reg11base + seed) ^ 0x5deece66dLL; // & 0xffffffffffff;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp00;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp00;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p = (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp00;
|
|
|
|
s = (s * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-25 08:54:33 +02:00
|
|
|
p += (int)(s >> 17) % range;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p > qual) goto incomp00;
|
2018-03-30 13:36:52 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (0)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
incomp00:
|
2018-07-28 16:23:47 +02:00
|
|
|
if (incomplete) return 0;
|
2018-03-30 13:36:52 +02:00
|
|
|
incomplete = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
return incomplete ? incomplete : -1;
|
|
|
|
}
|
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
/* Calls the correct quad-base finder for the structure config, if available.
|
2018-07-25 08:54:33 +02:00
|
|
|
* (Exits program otherwise.)
|
|
|
|
*/
|
|
|
|
int isQuadBase(const StructureConfig sconf, const int64_t seed, const int64_t qual)
|
|
|
|
{
|
|
|
|
switch(sconf.properties)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return isQuadFeatureBase(sconf, seed, qual);
|
|
|
|
case LARGE_STRUCT:
|
|
|
|
return isLargeQuadBase(sconf, seed, qual);
|
|
|
|
case USE_POW2_RNG:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Quad-finder using power of 2 RNG is not implemented yet.\n");
|
|
|
|
exit(-1);
|
|
|
|
case LARGE_STRUCT | USE_POW2_RNG:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Quad-finder for large structures using power of 2 RNG"
|
|
|
|
" is not implemented yet.\n");
|
|
|
|
exit(-1);
|
|
|
|
default:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Unknown properties field for structure: 0x%04X\n",
|
|
|
|
sconf.properties);
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
/* Calls the correct triple-base finder for the structure config, if available.
|
2018-07-28 10:34:19 +02:00
|
|
|
* (Exits program otherwise.)
|
|
|
|
*/
|
|
|
|
int isTriBase(const StructureConfig sconf, const int64_t seed, const int64_t qual)
|
|
|
|
{
|
|
|
|
switch(sconf.properties)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
return isTriFeatureBase(sconf, seed, qual);
|
|
|
|
case LARGE_STRUCT:
|
|
|
|
return isLargeTriBase(sconf, seed, qual);
|
|
|
|
case USE_POW2_RNG:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Quad-finder using power of 2 RNG is not implemented yet.\n");
|
|
|
|
exit(-1);
|
|
|
|
case LARGE_STRUCT | USE_POW2_RNG:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Quad-finder for large structures using power of 2 RNG"
|
|
|
|
" is not implemented yet.\n");
|
|
|
|
exit(-1);
|
|
|
|
default:
|
|
|
|
fprintf(stderr,
|
|
|
|
"Unknown properties field for structure: 0x%04X\n",
|
|
|
|
sconf.properties);
|
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
/* Searches for the optimal AFK position given four structures at positions 'p',
|
|
|
|
* each of volume (ax,ay,az).
|
|
|
|
*
|
|
|
|
* Returned is the number of spawning spaces within reach.
|
|
|
|
*/
|
2018-03-30 13:36:52 +02:00
|
|
|
int countBlocksInSpawnRange(Pos p[4], const int ax, const int ay, const int az)
|
|
|
|
{
|
|
|
|
int minX = 3e7, minZ = 3e7, maxX = -3e7, maxZ = -3e7;
|
|
|
|
int best;
|
|
|
|
|
|
|
|
|
|
|
|
// Find corners
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int i = 0; i < 4; i++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (p[i].x < minX) minX = p[i].x;
|
|
|
|
if (p[i].z < minZ) minZ = p[i].z;
|
|
|
|
if (p[i].x > maxX) maxX = p[i].x;
|
|
|
|
if (p[i].z > maxZ) maxZ = p[i].z;
|
2018-03-30 13:36:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// assume that the search area is bound by the inner corners
|
|
|
|
maxX += ax;
|
|
|
|
maxZ += az;
|
|
|
|
best = 0;
|
|
|
|
|
|
|
|
double thsq = 128.0*128.0 - az*az/4.0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int x = minX; x < maxX; x++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int z = minZ; z < maxZ; z++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
int inrange = 0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int i = 0; i < 4; i++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
double dx = p[i].x - (x+0.5);
|
|
|
|
double dz = p[i].z - (z+0.5);
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int px = 0; px < ax; px++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int pz = 0; pz < az; pz++)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
double ddx = px + dx;
|
|
|
|
double ddz = pz + dz;
|
|
|
|
inrange += (ddx*ddx + ddz*ddz <= thsq);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (inrange > best)
|
2018-03-30 13:36:52 +02:00
|
|
|
{
|
|
|
|
best = inrange;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return best;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-04 16:18:21 +01:00
|
|
|
static void *search4QuadBasesThread(void *data)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
quad_threadinfo_t info = *(quad_threadinfo_t*)data;
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
const int64_t start = info.start;
|
|
|
|
const int64_t end = info.end;
|
2018-07-25 08:54:33 +02:00
|
|
|
const int64_t structureSeed = info.sconf.seed;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t seed;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t *lowerBits;
|
2018-03-05 21:20:54 +00:00
|
|
|
int lowerBitsCnt;
|
|
|
|
int lowerBitsIdx = 0;
|
2018-07-04 16:18:21 +01:00
|
|
|
int i;
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
lowerBits = (int64_t *) malloc(0x10000 * sizeof(int64_t));
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (info.quality == 1)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
lowerBitsCnt = sizeof(lowerBaseBitsQ1) / sizeof(lowerBaseBitsQ1[0]);
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < lowerBitsCnt; i++)
|
2018-07-04 16:18:21 +01:00
|
|
|
{
|
|
|
|
lowerBits[i] = (lowerBaseBitsQ1[i] - structureSeed) & 0xffff;
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
2018-07-28 16:23:47 +02:00
|
|
|
else if (info.quality == 2)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
lowerBitsCnt = sizeof(lowerBaseBitsQ2) / sizeof(lowerBaseBitsQ2[0]);
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < lowerBitsCnt; i++)
|
2018-07-04 16:18:21 +01:00
|
|
|
{
|
|
|
|
lowerBits[i] = (lowerBaseBitsQ2[i] - structureSeed) & 0xffff;
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
printf("WARN search4QuadBasesThread: "
|
2018-03-17 16:53:16 +00:00
|
|
|
"Lower bits for quality %d have not been defined => "
|
|
|
|
"will try all combinations.\n", info.quality);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:18:21 +01:00
|
|
|
lowerBitsCnt = 0x10000;
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < lowerBitsCnt; i++) lowerBits[i] = i;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
char fnam[256];
|
|
|
|
sprintf(fnam, "%s.part%d", info.fnam, info.threadID);
|
|
|
|
|
|
|
|
FILE *fp = fopen(fnam, "a+");
|
2018-07-28 16:23:47 +02:00
|
|
|
if (fp == NULL)
|
|
|
|
{
|
2018-07-02 14:34:01 -07:00
|
|
|
fprintf(stderr, "Could not open \"%s\" for writing.\n", fnam);
|
2018-07-04 16:18:21 +01:00
|
|
|
free(lowerBits);
|
2018-07-02 14:34:01 -07:00
|
|
|
exit(-1);
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
seed = start;
|
|
|
|
|
|
|
|
// Check the last entry in the file and use it as a starting point if it
|
|
|
|
// exists. (I.e. loading the saved progress.)
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!fseek(fp, -31, SEEK_END))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
char buf[32];
|
2018-07-28 16:23:47 +02:00
|
|
|
if (fread(buf, 30, 1, fp) > 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
char *last_newline = strrchr(buf, '\n');
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (sscanf(last_newline, "%"PRId64, &seed) == 1)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
while (lowerBits[lowerBitsIdx] <= (seed & 0xffff))
|
2018-03-05 21:20:54 +00:00
|
|
|
lowerBitsIdx++;
|
|
|
|
|
|
|
|
seed = (seed & 0x0000ffffffff0000) + lowerBits[lowerBitsIdx];
|
|
|
|
|
2018-07-04 17:28:27 +01:00
|
|
|
printf("Thread %d starting from: %"PRId64"\n", info.threadID, seed);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
seed = start;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
while (seed < end)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (isQuadBase(info.sconf, seed, info.quality))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 17:28:27 +01:00
|
|
|
fprintf(fp, "%"PRId64"\n", seed);
|
2018-03-05 21:20:54 +00:00
|
|
|
fflush(fp);
|
2018-07-04 17:28:27 +01:00
|
|
|
//printf("Thread %d: %"PRId64"\n", info.threadID, seed);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
lowerBitsIdx++;
|
2018-07-28 16:23:47 +02:00
|
|
|
if (lowerBitsIdx >= lowerBitsCnt)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
lowerBitsIdx = 0;
|
|
|
|
seed += 0x10000;
|
|
|
|
}
|
|
|
|
seed = (seed & 0x0000ffffffff0000) + lowerBits[lowerBitsIdx];
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fp);
|
2018-07-04 16:18:21 +01:00
|
|
|
free(lowerBits);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-04 16:18:21 +01:00
|
|
|
void search4QuadBases(const char *fnam, const int threads,
|
2018-07-25 08:54:33 +02:00
|
|
|
const StructureConfig structureConfig, const int quality)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
pthread_t threadID[threads];
|
|
|
|
quad_threadinfo_t info[threads];
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t t;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (t = 0; t < threads; t++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
info[t].threadID = t;
|
2018-07-25 10:20:06 +02:00
|
|
|
info[t].start = (t * SEED_BASE_MAX / threads) & 0x0000ffffffff0000;
|
|
|
|
info[t].end = ((info[t].start + (SEED_BASE_MAX-1) / threads) & 0x0000ffffffff0000) + 1;
|
2018-03-05 21:20:54 +00:00
|
|
|
info[t].fnam = fnam;
|
|
|
|
info[t].quality = quality;
|
2018-07-25 08:54:33 +02:00
|
|
|
info[t].sconf = structureConfig;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (t = 0; t < threads; t++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
pthread_create(&threadID[t], NULL, search4QuadBasesThread, (void*)&info[t]);
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (t = 0; t < threads; t++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
pthread_join(threadID[t], NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
// merge thread parts
|
|
|
|
|
|
|
|
char fnamThread[256];
|
|
|
|
char buffer[4097];
|
|
|
|
FILE *fp = fopen(fnam, "w");
|
2018-07-02 14:34:01 -07:00
|
|
|
if (fp == NULL) {
|
|
|
|
fprintf(stderr, "Could not open \"%s\" for writing.\n", fnam);
|
|
|
|
exit(-1);
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
FILE *fpart;
|
|
|
|
int n;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (t = 0; t < threads; t++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
sprintf(fnamThread, "%s.part%d", info[t].fnam, info[t].threadID);
|
|
|
|
|
|
|
|
fpart = fopen(fnamThread, "r");
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (fpart == NULL)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
perror("ERR search4QuadBases: ");
|
2018-03-05 21:20:54 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
while ((n = fread(buffer, sizeof(char), 4096, fpart)))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!fwrite(buffer, sizeof(char), n, fp))
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:18:21 +01:00
|
|
|
perror("ERR search4QuadBases: ");
|
2018-03-05 21:20:54 +00:00
|
|
|
fclose(fp);
|
|
|
|
fclose(fpart);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fpart);
|
|
|
|
|
|
|
|
remove(fnamThread);
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Finding Structure Positions
|
|
|
|
//==============================================================================
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
Pos getStructurePos(const StructureConfig config, int64_t seed,
|
2018-07-28 10:34:19 +02:00
|
|
|
const int regionX, const int regionZ)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
Pos pos;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
// set seed
|
2018-07-08 20:58:43 -07:00
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + config.seed;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed ^ 0x5deece66dLL);// & ((1LL << 48) - 1);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:01:00 -07:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (config.properties & USE_POW2_RNG)
|
|
|
|
{
|
|
|
|
// Java RNG treats powers of 2 as a special case.
|
|
|
|
pos.x = (config.chunkRange * (seed >> 17)) >> 31;
|
|
|
|
|
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
pos.z = (config.chunkRange * (seed >> 17)) >> 31;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pos.x = (int)(seed >> 17) % config.chunkRange;
|
|
|
|
|
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
pos.z = (int)(seed >> 17) % config.chunkRange;
|
|
|
|
}
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x = ((regionX*config.regionSize + pos.x) << 4) + 8;
|
|
|
|
pos.z = ((regionZ*config.regionSize + pos.z) << 4) + 8;
|
2018-03-05 21:20:54 +00:00
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
Pos getStructureChunkInRegion(const StructureConfig config, int64_t seed,
|
2018-07-08 20:01:00 -07:00
|
|
|
const int regionX, const int regionZ)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-03-17 16:53:16 +00:00
|
|
|
/*
|
|
|
|
// Vanilla like implementation.
|
2018-07-04 16:18:21 +01:00
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + structureSeed;
|
2018-03-05 21:20:54 +00:00
|
|
|
setSeed(&(seed));
|
|
|
|
|
|
|
|
Pos pos;
|
|
|
|
pos.x = nextInt(&seed, 24);
|
|
|
|
pos.z = nextInt(&seed, 24);
|
2018-03-17 16:53:16 +00:00
|
|
|
*/
|
|
|
|
Pos pos;
|
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + config.seed;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed ^ 0x5deece66dLL);// & ((1LL << 48) - 1);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:01:00 -07:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (config.properties & USE_POW2_RNG)
|
|
|
|
{
|
|
|
|
// Java RNG treats powers of 2 as a special case.
|
|
|
|
pos.x = (config.chunkRange * (seed >> 17)) >> 31;
|
|
|
|
|
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
pos.z = (config.chunkRange * (seed >> 17)) >> 31;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pos.x = (int)(seed >> 17) % config.chunkRange;
|
|
|
|
|
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
|
|
|
pos.z = (int)(seed >> 17) % config.chunkRange;
|
|
|
|
}
|
2018-03-17 16:53:16 +00:00
|
|
|
|
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
Pos getLargeStructurePos(StructureConfig config, int64_t seed,
|
2018-07-28 10:34:19 +02:00
|
|
|
const int regionX, const int regionZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
Pos pos;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//TODO: if (config.properties & USE_POW2_RNG)...
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
// set seed
|
2018-07-08 20:58:43 -07:00
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + config.seed;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed ^ 0x5deece66dLL) & ((1LL << 48) - 1);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x = (seed >> 17) % config.chunkRange;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x += (seed >> 17) % config.chunkRange;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.z = (seed >> 17) % config.chunkRange;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.z += (seed >> 17) % config.chunkRange;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x = regionX*config.regionSize + (pos.x >> 1);
|
|
|
|
pos.z = regionZ*config.regionSize + (pos.z >> 1);
|
2018-03-30 13:36:52 +02:00
|
|
|
pos.x = pos.x*16 + 8;
|
|
|
|
pos.z = pos.z*16 + 8;
|
2018-03-17 16:53:16 +00:00
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-08 20:58:43 -07:00
|
|
|
Pos getLargeStructureChunkInRegion(StructureConfig config, int64_t seed,
|
2018-07-28 10:34:19 +02:00
|
|
|
const int regionX, const int regionZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
Pos pos;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//TODO: if (config.properties & USE_POW2_RNG)...
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
// set seed
|
2018-07-08 20:58:43 -07:00
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + config.seed;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed ^ 0x5deece66dLL) & ((1LL << 48) - 1);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x = (seed >> 17) % config.chunkRange;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.x += (seed >> 17) % config.chunkRange;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.z = (seed >> 17) % config.chunkRange;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-08 20:58:43 -07:00
|
|
|
pos.z += (seed >> 17) % config.chunkRange;
|
|
|
|
|
|
|
|
pos.x >>= 1;
|
|
|
|
pos.z >>= 1;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Checking Biomes & Biome Helper Functions
|
|
|
|
//==============================================================================
|
|
|
|
|
|
|
|
|
|
|
|
int getBiomeAtPos(const LayerStack g, const Pos pos)
|
|
|
|
{
|
|
|
|
int *map = allocCache(&g.layers[g.layerNum-1], 1, 1);
|
|
|
|
genArea(&g.layers[g.layerNum-1], map, pos.x, pos.z, 1, 1);
|
|
|
|
int biomeID = map[0];
|
|
|
|
free(map);
|
|
|
|
return biomeID;
|
|
|
|
}
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
Pos findBiomePosition(
|
2018-07-28 16:23:47 +02:00
|
|
|
const int mcversion,
|
2018-03-17 16:53:16 +00:00
|
|
|
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
|
|
|
|
)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
|
|
|
int x1 = (centerX-range) >> 2;
|
|
|
|
int z1 = (centerZ-range) >> 2;
|
|
|
|
int x2 = (centerX+range) >> 2;
|
|
|
|
int z2 = (centerZ+range) >> 2;
|
|
|
|
int width = x2 - x1 + 1;
|
|
|
|
int height = z2 - z1 + 1;
|
2018-03-17 16:53:16 +00:00
|
|
|
int *map;
|
2018-07-28 16:23:47 +02:00
|
|
|
int i, j, found;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
Layer *layer = &g.layers[L_RIVER_MIX_4];
|
|
|
|
Pos out;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (layer->scale != 4)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
printf("WARN findBiomePosition: The generator has unexpected scale %d at layer %d.\n",
|
|
|
|
layer->scale, L_RIVER_MIX_4);
|
|
|
|
}
|
|
|
|
|
|
|
|
map = cache ? cache : allocCache(layer, width, height);
|
|
|
|
|
|
|
|
genArea(layer, map, x1, z1, width, height);
|
|
|
|
|
2018-07-05 21:05:34 +01:00
|
|
|
out.x = centerX;
|
|
|
|
out.z = centerZ;
|
2018-03-17 16:53:16 +00:00
|
|
|
found = 0;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (mcversion >= MC_1_13)
|
|
|
|
{
|
|
|
|
for (i = 0, j = 2; i < width*height; i++)
|
|
|
|
{
|
|
|
|
if (!isValid[map[i] & 0xff]) continue;
|
|
|
|
if ((found == 0 || nextInt(seed, j++) == 0))
|
|
|
|
{
|
|
|
|
out.x = (x1 + i%width) << 2;
|
|
|
|
out.z = (z1 + i/width) << 2;
|
|
|
|
found = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
found = j - 2;
|
|
|
|
}
|
|
|
|
else
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < width*height; i++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (isValid[map[i] & 0xff] &&
|
|
|
|
(found == 0 || nextInt(seed, found + 1) == 0))
|
|
|
|
{
|
|
|
|
out.x = (x1 + i%width) << 2;
|
|
|
|
out.z = (z1 + i/width) << 2;
|
|
|
|
++found;
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
|
|
|
|
if (cache == NULL)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
free(map);
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (passes != NULL)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
*passes = found;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
int areBiomesViable(
|
|
|
|
const LayerStack g,
|
|
|
|
int * cache,
|
|
|
|
const int posX,
|
|
|
|
const int posZ,
|
|
|
|
const int radius,
|
|
|
|
const int * isValid
|
|
|
|
)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
int x1 = (posX - radius) >> 2;
|
|
|
|
int z1 = (posZ - radius) >> 2;
|
|
|
|
int x2 = (posX + radius) >> 2;
|
|
|
|
int z2 = (posZ + radius) >> 2;
|
|
|
|
int width = x2 - x1 + 1;
|
|
|
|
int height = z2 - z1 + 1;
|
|
|
|
int i;
|
|
|
|
int *map;
|
|
|
|
|
|
|
|
Layer *layer = &g.layers[L_RIVER_MIX_4];
|
2018-07-04 13:14:28 -07:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (layer->scale != 4)
|
2018-07-04 13:14:28 -07:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
printf("WARN areBiomesViable: The generator has unexpected scale %d at layer %d.\n",
|
|
|
|
layer->scale, L_RIVER_MIX_4);
|
|
|
|
}
|
|
|
|
|
|
|
|
map = cache ? cache : allocCache(layer, width, height);
|
|
|
|
genArea(layer, map, x1, z1, width, height);
|
|
|
|
|
|
|
|
for (i = 0; i < width*height; i++)
|
|
|
|
{
|
|
|
|
if (!isValid[ map[i] & 0xff ])
|
2018-07-04 13:14:28 -07:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (cache == NULL) free(map);
|
|
|
|
return 0;
|
2018-07-04 13:14:28 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (cache == NULL) free(map);
|
|
|
|
return 1;
|
2018-07-04 13:14:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
int getBiomeRadius(
|
|
|
|
const int * map,
|
|
|
|
const int mapSide,
|
|
|
|
const int * biomes,
|
|
|
|
const int bnum,
|
|
|
|
const int ignoreMutations)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
int r, i, b;
|
|
|
|
int blist[0x100];
|
|
|
|
int mask = ignoreMutations ? 0x7f : 0xff;
|
|
|
|
int radiusMax = mapSide / 2;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if ((mapSide & 1) == 0)
|
|
|
|
{
|
|
|
|
printf("WARN getBiomeRadius: Side length of the square map should be an odd integer.\n");
|
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
memset(blist, 0, sizeof(blist));
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (r = 1; r < radiusMax; r++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = radiusMax-r; i <= radiusMax+r; i++)
|
|
|
|
{
|
|
|
|
blist[ map[(radiusMax-r) * mapSide+ i] & mask ] = 1;
|
|
|
|
blist[ map[(radiusMax+r-1) * mapSide + i] & mask ] = 1;
|
|
|
|
blist[ map[mapSide*i + (radiusMax-r)] & mask ] = 1;
|
|
|
|
blist[ map[mapSide*i + (radiusMax+r-1)] & mask ] = 1;
|
|
|
|
}
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (b = 0; b < bnum && blist[biomes[b] & mask]; b++);
|
|
|
|
if (b >= bnum)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
2018-07-04 13:14:28 -07:00
|
|
|
}
|
2018-07-28 16:23:47 +02:00
|
|
|
|
|
|
|
return r != radiusMax ? r : -1;
|
2018-07-04 13:14:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Finding Strongholds and Spawn
|
|
|
|
//==============================================================================
|
2018-07-04 13:14:28 -07:00
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
int* getValidStrongholdBiomes()
|
|
|
|
{
|
|
|
|
static int validStrongholdBiomes[256];
|
|
|
|
|
|
|
|
if (!validStrongholdBiomes[plains])
|
|
|
|
{
|
|
|
|
int id;
|
|
|
|
for (id = 0; id < 256; id++)
|
|
|
|
{
|
|
|
|
if (biomeExists(id) && biomes[id].height > 0.0)
|
|
|
|
validStrongholdBiomes[id] = 1;
|
2018-07-04 13:14:28 -07:00
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
2018-07-04 13:14:28 -07:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
return validStrongholdBiomes;
|
2018-03-17 16:53:16 +00:00
|
|
|
}
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
|
|
|
|
int findStrongholds(const int mcversion, LayerStack *g, int *cache,
|
|
|
|
Pos *locations, int64_t worldSeed, int maxSH, const int maxRadius)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
const int *validStrongholdBiomes = getValidStrongholdBiomes();
|
|
|
|
int i, x, z;
|
|
|
|
double distance;
|
2018-07-28 10:34:19 +02:00
|
|
|
|
|
|
|
int currentRing = 0;
|
|
|
|
int currentCount = 0;
|
|
|
|
int perRing = 3;
|
|
|
|
|
|
|
|
setSeed(&worldSeed);
|
|
|
|
double angle = nextDouble(&worldSeed) * PI * 2.0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (mcversion >= MC_1_9)
|
|
|
|
{
|
|
|
|
if (maxSH <= 0) maxSH = 128;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < maxSH; i++)
|
|
|
|
{
|
|
|
|
distance = (4.0 * 32.0) + (6.0 * currentRing * 32.0) +
|
|
|
|
(nextDouble(&worldSeed) - 0.5) * 32 * 2.5;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (maxRadius && distance*16 > maxRadius)
|
|
|
|
return i;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
x = (int)round(cos(angle) * distance);
|
|
|
|
z = (int)round(sin(angle) * distance);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
locations[i] = findBiomePosition(mcversion, *g, cache,
|
|
|
|
(x << 4) + 8, (z << 4) + 8, 112, validStrongholdBiomes,
|
|
|
|
&worldSeed, NULL);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
angle += 2 * PI / perRing;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
currentCount++;
|
|
|
|
if (currentCount == perRing)
|
|
|
|
{
|
|
|
|
// Current ring is complete, move to next ring.
|
|
|
|
currentRing++;
|
|
|
|
currentCount = 0;
|
|
|
|
perRing = perRing + 2*perRing/(currentRing+1);
|
|
|
|
if (perRing > 128-i)
|
|
|
|
perRing = 128-i;
|
|
|
|
angle = angle + nextDouble(&worldSeed) * PI * 2.0;
|
|
|
|
}
|
|
|
|
}
|
2018-03-17 16:53:16 +00:00
|
|
|
}
|
2018-07-28 16:23:47 +02:00
|
|
|
else
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (maxSH <= 0) maxSH = 3;
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < maxSH; i++)
|
|
|
|
{
|
|
|
|
distance = (1.25 + nextDouble(&worldSeed)) * 32.0;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (maxRadius && distance*16 > maxRadius)
|
|
|
|
return i;
|
2018-03-10 21:45:57 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
x = (int)round(cos(angle) * distance);
|
|
|
|
z = (int)round(sin(angle) * distance);
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
locations[i] = findBiomePosition(mcversion, *g, cache,
|
|
|
|
(x << 4) + 8, (z << 4) + 8, 112, validStrongholdBiomes,
|
|
|
|
&worldSeed, NULL);
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
angle += 2 * PI / 3.0;
|
|
|
|
}
|
|
|
|
}
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
return maxSH;
|
|
|
|
}
|
2018-07-28 10:34:19 +02:00
|
|
|
|
|
|
|
|
2018-07-29 15:12:52 +02:00
|
|
|
static double getGrassProbability(int64_t seed, int biome, int x, int z)
|
|
|
|
{
|
|
|
|
// TODO: Use ChunkGeneratorOverworld.generateHeightmap for better estimate.
|
|
|
|
// TODO: Try to determine the actual probabilities and build a statistic.
|
|
|
|
switch (biome)
|
|
|
|
{
|
|
|
|
case plains: return 1.0;
|
|
|
|
case extremeHills: return 0.8; // height dependent
|
|
|
|
case forest: return 1.0;
|
|
|
|
case taiga: return 1.0;
|
|
|
|
case swampland: return 0.6; // height dependent
|
|
|
|
case river: return 0.2;
|
|
|
|
case beach: return 0.1;
|
|
|
|
case forestHills: return 1.0;
|
|
|
|
case taigaHills: return 1.0;
|
|
|
|
case extremeHillsEdge: return 1.0; // height dependent
|
|
|
|
case jungle: return 1.0;
|
|
|
|
case jungleHills: return 1.0;
|
|
|
|
case jungleEdge: return 1.0;
|
|
|
|
case birchForest: return 1.0;
|
|
|
|
case birchForestHills: return 1.0;
|
|
|
|
case roofedForest: return 0.9;
|
|
|
|
case coldTaiga: return 0.1; // below trees
|
|
|
|
case coldTaigaHills: return 0.1; // below trees
|
|
|
|
case megaTaiga: return 0.6;
|
|
|
|
case megaTaigaHills: return 0.6;
|
|
|
|
case extremeHillsPlus: return 0.2; // height dependent
|
|
|
|
case savanna: return 1.0;
|
|
|
|
case savannaPlateau: return 1.0;
|
|
|
|
case mesaPlateau_F: return 0.1; // height dependent
|
|
|
|
case mesaPlateau: return 0.1; // height dependent
|
|
|
|
// NOTE: in rare circumstances you can get also get grass islands that are
|
|
|
|
// completely ocean variants...
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-29 15:12:52 +02:00
|
|
|
static int canCoordinateBeSpawn(const int64_t seed, LayerStack *g, int *cache, Pos pos)
|
2018-07-28 16:23:47 +02:00
|
|
|
{
|
2018-07-29 15:12:52 +02:00
|
|
|
int biome = getBiomeAtPos(*g, pos);
|
|
|
|
return getGrassProbability(seed, biome, pos.x, pos.z) >= 0.5;
|
2018-07-28 16:23:47 +02:00
|
|
|
}
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
static int* getValidSpawnBiomes()
|
|
|
|
{
|
|
|
|
static int isSpawnBiome[256];
|
|
|
|
unsigned int i;
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!isSpawnBiome[biomesToSpawnIn[0]])
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sizeof(biomesToSpawnIn) / sizeof(int); i++)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
isSpawnBiome[ biomesToSpawnIn[i] ] = 1;
|
2018-07-28 10:34:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
return isSpawnBiome;
|
2018-07-28 10:34:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
Pos getSpawn(const int mcversion, LayerStack *g, int *cache, int64_t worldSeed)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
const int *isSpawnBiome = getValidSpawnBiomes();
|
2018-07-28 10:34:19 +02:00
|
|
|
Pos spawn;
|
|
|
|
int found;
|
2018-07-28 16:23:47 +02:00
|
|
|
int i;
|
2018-07-28 10:34:19 +02:00
|
|
|
|
|
|
|
setSeed(&worldSeed);
|
2018-07-28 16:23:47 +02:00
|
|
|
spawn = findBiomePosition(mcversion, *g, cache, 0, 0, 256, isSpawnBiome,
|
|
|
|
&worldSeed, &found);
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!found)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
//printf("Unable to find spawn biome.\n");
|
|
|
|
spawn.x = spawn.z = 8;
|
2018-07-28 10:34:19 +02:00
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (mcversion >= MC_1_13)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
// TODO: The 1.13 section may need further checking!
|
|
|
|
int n2 = 0;
|
|
|
|
int n3 = 0;
|
|
|
|
int n4 = 0;
|
|
|
|
int n5 = -1;
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < 1024; i++)
|
|
|
|
{
|
|
|
|
if (n2 > -16 && n2 <= 16 && n3 > -16 && n3 <= 16)
|
|
|
|
{
|
|
|
|
int cx = ((spawn.x >> 4) + n2) << 4;
|
|
|
|
int cz = ((spawn.z >> 4) + n3) << 4;
|
2018-07-28 10:34:19 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (int i2 = cx; i2 <= cx+15; i2++)
|
|
|
|
{
|
|
|
|
for (int i3 = cz; i3 <= cz+15; i3++)
|
2018-07-28 10:34:19 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
Pos pos = {i2, i3};
|
2018-07-29 15:12:52 +02:00
|
|
|
if (canCoordinateBeSpawn(worldSeed, g, cache, pos))
|
2018-07-28 16:23:47 +02:00
|
|
|
{
|
|
|
|
return pos;
|
|
|
|
}
|
2018-07-28 10:34:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (n2 == n3 || (n2 < 0 && n2 == - n3) || (n2 > 0 && n2 == 1 - n3))
|
|
|
|
{
|
|
|
|
int n7 = n4;
|
|
|
|
n4 = - n5;
|
|
|
|
n5 = n7;
|
|
|
|
}
|
|
|
|
n2 += n4;
|
|
|
|
n3 += n5;
|
2018-07-28 10:34:19 +02:00
|
|
|
}
|
|
|
|
}
|
2018-07-28 16:23:47 +02:00
|
|
|
else
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-29 15:12:52 +02:00
|
|
|
for (i = 0; i < 1000 && !canCoordinateBeSpawn(worldSeed, g, cache, spawn); i++)
|
2018-03-10 21:45:57 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
spawn.x += nextInt(&worldSeed, 64) - nextInt(&worldSeed, 64);
|
|
|
|
spawn.z += nextInt(&worldSeed, 64) - nextInt(&worldSeed, 64);
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
return spawn;
|
2018-03-10 21:45:57 +00:00
|
|
|
}
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Validating Structure Positions
|
|
|
|
//==============================================================================
|
|
|
|
|
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
int isViableFeaturePos(const int structureType, const LayerStack g, int *cache,
|
|
|
|
const int blockX, const int blockZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-25 08:54:33 +02:00
|
|
|
int *map = cache ? cache : allocCache(&g.layers[g.layerNum-1], 1, 1);
|
|
|
|
genArea(&g.layers[g.layerNum-1], map, blockX, blockZ, 1, 1);
|
|
|
|
int biomeID = map[0];
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!cache) free(map);
|
2018-03-17 16:53:16 +00:00
|
|
|
|
2018-07-25 08:54:33 +02:00
|
|
|
switch(structureType)
|
|
|
|
{
|
|
|
|
case Desert_Pyramid:
|
|
|
|
return biomeID == desert || biomeID == desertHills;
|
|
|
|
case Igloo:
|
|
|
|
return biomeID == icePlains || biomeID == coldTaiga;
|
|
|
|
case Jungle_Pyramid:
|
|
|
|
return biomeID == jungle || biomeID == jungleHills;
|
|
|
|
case Swamp_Hut:
|
|
|
|
return biomeID == swampland;
|
|
|
|
case Ocean_Ruin:
|
|
|
|
case Shipwreck:
|
|
|
|
return isOceanic(biomeID);
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "Structure type is not valid for the scattered feature biome check.\n");
|
|
|
|
exit(1);
|
|
|
|
}
|
2018-03-17 16:53:16 +00:00
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int isViableVillagePos(const LayerStack g, int *cache,
|
2018-07-25 08:54:33 +02:00
|
|
|
const int blockX, const int blockZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
static int isVillageBiome[0x100];
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!isVillageBiome[villageBiomeList[0]])
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
unsigned int i;
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sizeof(villageBiomeList) / sizeof(int); i++)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
isVillageBiome[ villageBiomeList[i] ] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return areBiomesViable(g, cache, blockX, blockZ, 0, isVillageBiome);
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int isViableOceanMonumentPos(const LayerStack g, int *cache,
|
2018-07-25 08:54:33 +02:00
|
|
|
const int blockX, const int blockZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
static int isWaterBiome[0x100];
|
|
|
|
static int isDeepOcean[0x100];
|
|
|
|
|
2018-07-28 18:17:30 +02:00
|
|
|
if (!isWaterBiome[oceanMonumentBiomeList1[1]])
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
unsigned int i;
|
2018-07-28 18:17:30 +02:00
|
|
|
for (i = 0; i < sizeof(oceanMonumentBiomeList1) / sizeof(int); i++)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-28 18:17:30 +02:00
|
|
|
isWaterBiome[ oceanMonumentBiomeList1[i] ] = 1;
|
2018-03-17 16:53:16 +00:00
|
|
|
}
|
|
|
|
|
2018-07-28 18:17:30 +02:00
|
|
|
for (i = 0; i < sizeof(oceanMonumentBiomeList2) / sizeof(int); i++)
|
|
|
|
{
|
|
|
|
isDeepOcean[ oceanMonumentBiomeList2[i] ] = 1;
|
|
|
|
}
|
2018-03-17 16:53:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return areBiomesViable(g, cache, blockX, blockZ, 16, isDeepOcean) &&
|
|
|
|
areBiomesViable(g, cache, blockX, blockZ, 29, isWaterBiome);
|
|
|
|
}
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int isViableMansionPos(const LayerStack g, int *cache,
|
2018-07-25 08:54:33 +02:00
|
|
|
const int blockX, const int blockZ)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
static int isMansionBiome[0x100];
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!isMansionBiome[mansionBiomeList[0]])
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
unsigned int i;
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sizeof(mansionBiomeList) / sizeof(int); i++)
|
2018-03-17 16:53:16 +00:00
|
|
|
{
|
|
|
|
isMansionBiome[ mansionBiomeList[i] ] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return areBiomesViable(g, cache, blockX, blockZ, 32, isMansionBiome);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
|
|
|
|
|
|
|
|
//==============================================================================
|
|
|
|
// Finding Properties of Structures
|
|
|
|
//==============================================================================
|
|
|
|
|
|
|
|
|
|
|
|
int isZombieVillage(const int mcversion, const int64_t worldSeed,
|
|
|
|
const int regionX, const int regionZ)
|
2018-07-28 10:47:46 +02:00
|
|
|
{
|
|
|
|
Pos pos;
|
|
|
|
int64_t seed = worldSeed;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (mcversion < MC_1_10)
|
|
|
|
{
|
|
|
|
printf("Warning: Zombie villages were only introduced in MC 1.10.\n");
|
|
|
|
}
|
|
|
|
|
2018-07-28 10:47:46 +02:00
|
|
|
// get the chunk position of the village
|
|
|
|
seed = regionX*341873128712 + regionZ*132897987541 + seed + VILLAGE_CONFIG.seed;
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed ^ 0x5deece66dLL);// & ((1LL << 48) - 1);
|
2018-07-28 10:47:46 +02:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-28 10:47:46 +02:00
|
|
|
pos.x = (seed >> 17) % VILLAGE_CONFIG.chunkRange;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
seed = (seed * 0x5deece66dLL + 0xbLL) & 0xffffffffffff;
|
2018-07-28 10:47:46 +02:00
|
|
|
pos.z = (seed >> 17) % VILLAGE_CONFIG.chunkRange;
|
|
|
|
|
|
|
|
pos.x += regionX * VILLAGE_CONFIG.regionSize;
|
|
|
|
pos.z += regionZ * VILLAGE_CONFIG.regionSize;
|
|
|
|
|
|
|
|
// jump to the random number check that determines whether this is village
|
|
|
|
// is zombie infested
|
|
|
|
int64_t rnd = chunkGenerateRnd(worldSeed, pos.x , pos.z);
|
2018-07-28 16:23:47 +02:00
|
|
|
// TODO: check for versions <= 1.11
|
|
|
|
skipNextN(&rnd, mcversion >= MC_1_13 ? 10 : 11);
|
2018-07-28 10:47:46 +02:00
|
|
|
|
|
|
|
return nextInt(&rnd, 50) == 0;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
|
|
|
|
int isBabyZombieVillage(const int mcversion, const int64_t worldSeed,
|
|
|
|
const int regionX, const int regionZ)
|
2018-07-28 10:47:46 +02:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!isZombieVillage(mcversion, worldSeed, regionX, regionZ))
|
2018-07-28 10:47:46 +02:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
// Whether the zombie is a child or not is dependent on the world random
|
|
|
|
// object which is not reset for villages. The last reset is instead
|
|
|
|
// performed during the positioning of Mansions.
|
|
|
|
int64_t rnd = worldSeed;
|
|
|
|
rnd = regionX*341873128712 + regionZ*132897987541 + rnd + MANSION_CONFIG.seed;
|
|
|
|
setSeed(&rnd);
|
|
|
|
skipNextN(&rnd, 5);
|
|
|
|
|
|
|
|
int isChild = nextFloat(&rnd) < 0.05;
|
|
|
|
//int mountNearbyChicken = nextFloat(&rnd) < 0.05;
|
|
|
|
//int spawnNewChicken = nextFloat(&rnd) < 0.05;
|
|
|
|
|
|
|
|
return isChild;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
//==============================================================================
|
|
|
|
// Seed Filters
|
|
|
|
//==============================================================================
|
2018-03-17 16:53:16 +00:00
|
|
|
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t filterAllTempCats(
|
2018-07-28 16:23:47 +02:00
|
|
|
LayerStack * g,
|
|
|
|
int * cache,
|
|
|
|
const int64_t * seedsIn,
|
|
|
|
int64_t * seedsOut,
|
|
|
|
const int64_t seedCnt,
|
|
|
|
const int centX,
|
|
|
|
const int centZ)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
/* We require all temperature categories, including the special variations
|
|
|
|
* in order to get all main biomes. This gives 8 required values:
|
|
|
|
* Oceanic, Warm, Lush, Cold, Freezing,
|
|
|
|
* Special Warm, Special Lush, Special Cold
|
|
|
|
* These categories generate at Layer 13: Edge, Special.
|
|
|
|
*
|
|
|
|
* Note: The scale at this layer is 1:1024 and each element can "leak" its
|
|
|
|
* biome values up to 1024 blocks outwards into the negative coordinates
|
|
|
|
* (due to the Zoom layers).
|
|
|
|
*
|
|
|
|
* The plan is to check if the 3x3 area contains all 8 temperature types.
|
|
|
|
* For this, we can check even earlier at Layer 10: Add Island, that each of
|
|
|
|
* the Warm, Cold and Freezing categories are present.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Edit:
|
|
|
|
* All the biomes that are generated by a simple Cold climate can actually
|
|
|
|
* be generated later on. So I have commented out the Cold requirements.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const int pX = centX-1, pZ = centZ-1;
|
|
|
|
const int sX = 3, sZ = 3;
|
2018-03-17 16:53:16 +00:00
|
|
|
int *map;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
Layer *lFilterSnow = &g->layers[L_ADD_SNOW_1024];
|
|
|
|
Layer *lFilterSpecial = &g->layers[L_SPECIAL_1024];
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
map = cache ? cache : allocCache(lFilterSpecial, sX, sZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
// Construct a dummy Edge,Special layer.
|
|
|
|
Layer layerSpecial;
|
2018-03-17 16:53:16 +00:00
|
|
|
setupLayer(1024, &layerSpecial, NULL, 3, NULL);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t sidx, hits, seed;
|
2018-03-05 21:20:54 +00:00
|
|
|
int types[9];
|
|
|
|
int specialCnt;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
hits = 0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (sidx = 0; sidx < seedCnt; sidx++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
seed = seedsIn[sidx];
|
|
|
|
|
|
|
|
/*** Pre-Generation Checks ***/
|
|
|
|
|
|
|
|
// We require at least 3 special temperature categories which can be
|
|
|
|
// tested for without going through the previous layers. (We'll get
|
|
|
|
// false positives due to Oceans, but this works fine to rule out some
|
|
|
|
// seeds early on.)
|
|
|
|
setWorldSeed(&layerSpecial, seed);
|
|
|
|
specialCnt = 0;
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX; i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-28 16:23:47 +02:00
|
|
|
for (j = 0; j < sZ; j++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-07-04 16:48:05 +01:00
|
|
|
setChunkSeed(&layerSpecial, (int64_t)(i+pX), (int64_t)(j+pZ));
|
2018-07-28 16:23:47 +02:00
|
|
|
if (mcNextInt(&layerSpecial, 13) == 0)
|
2018-03-05 21:20:54 +00:00
|
|
|
specialCnt++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (specialCnt < 3)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*** Cold/Warm Check ***/
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
// Continue by checking if enough cold and warm categories are present.
|
2018-03-17 16:53:16 +00:00
|
|
|
setWorldSeed(lFilterSnow, seed);
|
|
|
|
genArea(lFilterSnow, map, pX,pZ, sX,sZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
memset(types, 0, sizeof(types));
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX*sZ; i++)
|
2018-03-17 16:53:16 +00:00
|
|
|
types[map[i]]++;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
// 1xOcean needs to be present
|
|
|
|
// 4xWarm need to turn into Warm, Lush, Special Warm and Special Lush
|
|
|
|
// 1xFreezing that needs to stay Freezing
|
|
|
|
// 3x(Cold + Freezing) for Cold, Special Cold and Freezing
|
2018-07-28 16:23:47 +02:00
|
|
|
if ( types[Ocean] < 1 || types[Warm] < 4 || types[Freezing] < 1 ||
|
2018-03-05 21:20:54 +00:00
|
|
|
types[Cold]+types[Freezing] < 2)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*** Complete Temperature Category Check ***/
|
|
|
|
|
|
|
|
// Check that all temperature variants are present.
|
2018-03-17 16:53:16 +00:00
|
|
|
setWorldSeed(lFilterSpecial, seed);
|
|
|
|
genArea(lFilterSpecial, map, pX,pZ, sX,sZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
memset(types, 0, sizeof(types));
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX*sZ; i++)
|
2018-03-17 16:53:16 +00:00
|
|
|
types[ map[i] > 4 ? (map[i]&0xf) + 4 : map[i] ]++;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if ( types[Ocean] < 1 || types[Warm] < 1 || types[Lush] < 1 ||
|
2018-03-05 21:20:54 +00:00
|
|
|
/*types[Cold] < 1 ||*/ types[Freezing] < 1 ||
|
|
|
|
types[Warm+4] < 1 || types[Lush+4] < 1 || types[Cold+4] < 1)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX*sZ; i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
printf("%c%d ", " s"[cache[i] > 4], cache[i]&0xf);
|
2018-07-28 16:23:47 +02:00
|
|
|
if (i % sX == sX-1) printf("\n");
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
printf("\n");*/
|
|
|
|
|
|
|
|
// Save the candidate.
|
|
|
|
seedsOut[hits] = seed;
|
|
|
|
hits++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (cache == NULL) free(map);
|
2018-03-05 21:20:54 +00:00
|
|
|
return hits;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const int majorBiomes[] = {
|
|
|
|
ocean, plains, desert, extremeHills, forest, taiga, swampland,
|
|
|
|
icePlains, mushroomIsland, jungle, deepOcean, birchForest, roofedForest,
|
|
|
|
coldTaiga, megaTaiga, savanna, mesaPlateau_F, mesaPlateau
|
|
|
|
};
|
|
|
|
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t filterAllMajorBiomes(
|
2018-07-28 16:23:47 +02:00
|
|
|
LayerStack * g,
|
|
|
|
int * cache,
|
|
|
|
const int64_t * seedsIn,
|
|
|
|
int64_t * seedsOut,
|
|
|
|
const int64_t seedCnt,
|
|
|
|
const int pX,
|
|
|
|
const int pZ,
|
|
|
|
const unsigned int sX,
|
|
|
|
const unsigned int sZ)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-03-17 16:53:16 +00:00
|
|
|
Layer *lFilterMushroom = &g->layers[L_ADD_MUSHROOM_ISLAND_256];
|
|
|
|
Layer *lFilterBiomes = &g->layers[L_BIOME_256];
|
2018-03-05 21:20:54 +00:00
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
int *map;
|
2018-07-04 16:48:05 +01:00
|
|
|
int64_t sidx, seed, hits;
|
|
|
|
unsigned int i, id, hasAll;
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
int types[BIOME_NUM];
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
map = cache ? cache : allocCache(lFilterBiomes, sX, sZ);
|
|
|
|
|
2018-03-05 21:20:54 +00:00
|
|
|
hits = 0;
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
for (sidx = 0; sidx < seedCnt; sidx++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
/* We can use the Mushroom layer both to check for mushroomIsland biomes
|
|
|
|
* and to make sure all temperature categories are present in the area.
|
|
|
|
*/
|
|
|
|
seed = seedsIn[sidx];
|
2018-03-17 16:53:16 +00:00
|
|
|
setWorldSeed(lFilterMushroom, seed);
|
|
|
|
genArea(lFilterMushroom, map, pX,pZ, sX,sZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
memset(types, 0, sizeof(types));
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX*sZ; i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-03-17 16:53:16 +00:00
|
|
|
id = map[i];
|
2018-07-28 16:23:47 +02:00
|
|
|
if (id >= BIOME_NUM) id = (id & 0xf) + 4;
|
2018-03-05 21:20:54 +00:00
|
|
|
types[id]++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if ( types[Ocean] < 1 || types[Warm] < 1 || types[Lush] < 1 ||
|
2018-03-05 21:20:54 +00:00
|
|
|
/* types[Cold] < 1 || */ types[Freezing] < 1 ||
|
|
|
|
types[Warm+4] < 1 || types[Lush+4] < 1 || types[Cold+4] < 1 ||
|
|
|
|
types[mushroomIsland] < 1)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*** Find all major biomes ***/
|
|
|
|
|
2018-03-17 16:53:16 +00:00
|
|
|
setWorldSeed(lFilterBiomes, seed);
|
|
|
|
genArea(lFilterBiomes, map, pX,pZ, sX,sZ);
|
2018-03-05 21:20:54 +00:00
|
|
|
|
|
|
|
memset(types, 0, sizeof(types));
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sX*sZ; i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
2018-03-17 16:53:16 +00:00
|
|
|
types[map[i]]++;
|
2018-03-05 21:20:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
hasAll = 1;
|
2018-07-28 16:23:47 +02:00
|
|
|
for (i = 0; i < sizeof(majorBiomes) / sizeof(*majorBiomes); i++)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
// plains, taiga and deepOcean can be generated in later layers.
|
|
|
|
// Also small islands of Forests can be generated in deepOcean
|
|
|
|
// biomes, but we are going to ignore those.
|
2018-07-28 16:23:47 +02:00
|
|
|
if (majorBiomes[i] == plains ||
|
2018-03-05 21:20:54 +00:00
|
|
|
majorBiomes[i] == taiga ||
|
|
|
|
majorBiomes[i] == deepOcean)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (types[majorBiomes[i]] < 1)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
hasAll = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-07-28 16:23:47 +02:00
|
|
|
if (!hasAll)
|
2018-03-05 21:20:54 +00:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
seedsOut[hits] = seed;
|
|
|
|
hits++;
|
|
|
|
}
|
|
|
|
|
2018-07-28 16:23:47 +02:00
|
|
|
if (cache == NULL) free(map);
|
2018-03-05 21:20:54 +00:00
|
|
|
return hits;
|
|
|
|
}
|