mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-11 13:21:10 +01:00
104 lines
2.6 KiB
C++
104 lines
2.6 KiB
C++
#include "wfc.h"
|
|
#include <limits>
|
|
|
|
namespace {
|
|
// Normalize a vector so the sum of its elements is equal to 1.0f
|
|
Vector<double> &normalize(Vector<double> &v) {
|
|
double sum_weights = 0.0;
|
|
int size = v.size();
|
|
const double* vpr = v.ptr();
|
|
for (int i = 0; i < size; ++i) {
|
|
sum_weights += vpr[i];
|
|
}
|
|
|
|
double* vpw = v.ptrw();
|
|
double inv_sum_weights = 1.0 / sum_weights;
|
|
for (int i = 0; i < size; ++i) {
|
|
vpw[i] *= inv_sum_weights;
|
|
}
|
|
|
|
return v;
|
|
}
|
|
} //namespace
|
|
|
|
Array2D<uint32_t> WFC::wave_to_output() const {
|
|
Array2D<uint32_t> output_patterns(wave.height, wave.width);
|
|
|
|
for (uint32_t i = 0; i < wave.size; i++) {
|
|
for (uint32_t k = 0; k < nb_patterns; k++) {
|
|
if (wave.get(i, k)) {
|
|
output_patterns.data.write[i] = k;
|
|
}
|
|
}
|
|
}
|
|
return output_patterns;
|
|
}
|
|
|
|
WFC::WFC(bool periodic_output, int seed,
|
|
Vector<double> patterns_frequencies,
|
|
Propagator::PropagatorState propagator, uint32_t wave_height,
|
|
uint32_t wave_width) :
|
|
gen(seed), patterns_frequencies(normalize(patterns_frequencies)), wave(wave_height, wave_width, patterns_frequencies), nb_patterns(propagator.size()), propagator(wave.height, wave.width, periodic_output, propagator) {}
|
|
|
|
Array2D<uint32_t> WFC::run() {
|
|
while (true) {
|
|
// Define the value of an undefined cell.
|
|
ObserveStatus result = observe();
|
|
|
|
// Check if the algorithm has terminated.
|
|
if (result == failure) {
|
|
return Array2D<uint32_t>(0, 0);
|
|
} else if (result == success) {
|
|
return wave_to_output();
|
|
}
|
|
|
|
// Propagate the information.
|
|
propagator.propagate(wave);
|
|
}
|
|
}
|
|
|
|
WFC::ObserveStatus WFC::observe() {
|
|
// Get the cell with lowest entropy.
|
|
int argmin = wave.get_min_entropy(gen);
|
|
|
|
// If there is a contradiction, the algorithm has failed.
|
|
if (argmin == -2) {
|
|
return failure;
|
|
}
|
|
|
|
// If the lowest entropy is 0, then the algorithm has succeeded and
|
|
// finished.
|
|
if (argmin == -1) {
|
|
wave_to_output();
|
|
return success;
|
|
}
|
|
|
|
// Choose an element according to the pattern distribution
|
|
double s = 0;
|
|
for (uint32_t k = 0; k < nb_patterns; k++) {
|
|
s += wave.get(argmin, k) ? patterns_frequencies[k] : 0;
|
|
}
|
|
|
|
std::uniform_real_distribution<> dis(0, s);
|
|
double random_value = dis(gen);
|
|
size_t chosen_value = nb_patterns - 1;
|
|
|
|
for (uint32_t k = 0; k < nb_patterns; k++) {
|
|
random_value -= wave.get(argmin, k) ? patterns_frequencies[k] : 0;
|
|
if (random_value <= 0) {
|
|
chosen_value = k;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// And define the cell with the pattern.
|
|
for (uint32_t k = 0; k < nb_patterns; k++) {
|
|
if (wave.get(argmin, k) != (k == chosen_value)) {
|
|
propagator.add_to_propagator(argmin / wave.width, argmin % wave.width, k);
|
|
wave.set(argmin, k, false);
|
|
}
|
|
}
|
|
|
|
return to_continue;
|
|
}
|