summaryrefslogtreecommitdiff
path: root/src/procgen.cpp
blob: 5cda92e7e54341ff7805a743d9c34d4819672fe5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
This file is part of Minitrem.

Minitrem is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 2 of the License, or
(at your option) any later version.

Minitrem is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with Minitrem.  If not, see <http://www.gnu.org/licenses/>.
*/

#include "common.hpp"
#include <cmath>

namespace procgen {

void prng_t::seed(uint32_t seed)
{
	state = seed;
	next();
	next();
}

uint32_t prng_t::next(void)
{
	// glibc's LCG parameters (with m = 2^32 - 1).
	state *= 1103515245;
	state += 12345;
	return state;
}

float prng_t::next_float(void)
{
	return (float)next() / 4294967295.0f;
}

float prng_t::next_float(float n, float p)
{
	return remap(0.0f, 1.0f, n, p, next_float());
}

void prng_t::unit_vec(float out[2])
{
	float t;

	t = next_float() * 2.0f * M_PI;
	out[0] = cos(t);
	out[1] = sin(t);
}

v2f_t prng_t::unit_vec2(void)
{
	float t;
	t = next_float() * 2.0f * M_PI;
	return {cosf(t), sinf(t)};
}

void perlin_noise_t::generate(prng_t *prng, size_t size_)
{
	if (table)
		delete[] table;

	size = size_;
	table = new float[size * size][2];

	for (size_t y = 0; y < size; y++)
	for (size_t x = 0; x < size; x++)
		prng->unit_vec(table[y * size + x]);
}

float perlin_noise_t::table_dot(size_t nx, size_t ny, float dx, float dy)
{
	return table[ny * size + nx][0] * dx + table[ny * size + nx][1] * dy;
}

static float smooth_step(float x)
{
	return 3 * x * x - 2 * x * x * x;
}

float perlin_noise_t::get(v2f_t x, float scale)
{
	size_t nx, ny, nx1, ny1;
	float s, t, a, b, c, d;

	x /= scale;

	nx = (ssize_t)floor(x[0]) & (size - 1);
	ny = (ssize_t)floor(x[1]) & (size - 1);

	nx1 = (nx == size - 1 ? 0 : nx + 1);
	ny1 = (ny == size - 1 ? 0 : ny + 1);

	s = smooth_step(x[0] - floor(x[0]));
	t = smooth_step(x[1] - floor(x[1]));

	a = table_dot(nx,  ny,  -s,        -t);
	b = table_dot(nx1, ny,  -s + 1.0f, -t);
	c = table_dot(nx,  ny1, -s,        -t + 1.0f);
	d = table_dot(nx1, ny1, -s + 1.0f, -t + 1.0f);

	return bilerp(a, b, c, d, s, t);
}

perlin_noise_t::~perlin_noise_t()
{
	delete[] table;
}

} // namespace procgen