redo evolver, add warrior serialization / deserialization to library
This commit is contained in:
parent
0d44b60612
commit
9778e5c129
5 changed files with 348 additions and 246 deletions
|
@ -2,294 +2,359 @@
|
||||||
#include <filesystem>
|
#include <filesystem>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
#include <cstdlib>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <random>
|
#include <sstream>
|
||||||
#include <vector>
|
#include <mpi.h>
|
||||||
#include <ctime>
|
|
||||||
|
|
||||||
const lib94::warrior *warrior_to_beat;
|
constexpr size_t max_good_warriors = 100;
|
||||||
|
constexpr int rounds_per_matchup = 100;
|
||||||
|
constexpr int steps_to_tie = 1000000;
|
||||||
|
|
||||||
constexpr int score_rounds = 50;
|
constexpr lib94::instruction background_instruction = {
|
||||||
constexpr int rounds_to_tie = 1000000;
|
.op = lib94::DAT, .mod = lib94::F,
|
||||||
constexpr int warriors_per_generation = 100;
|
.amode = lib94::DIRECT, .bmode = lib94::DIRECT,
|
||||||
|
.anumber = 0, .bnumber = 0
|
||||||
struct winfo {
|
|
||||||
lib94::warrior w;
|
|
||||||
int score;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
std::vector<winfo> current_generation;
|
lib94::warrior *warrior_to_beat;
|
||||||
int generation_number;
|
int comm_rank, comm_size;
|
||||||
|
|
||||||
std::default_random_engine prng;
|
std::filesystem::path output_dir;
|
||||||
|
|
||||||
//sets random opcode, modifier, and modes
|
std::vector<lib94::warrior *> seed_warriors;
|
||||||
void random_instruction(lib94::instruction &instr) {
|
|
||||||
std::uniform_int_distribution<int> opcode_dist(lib94::DAT, lib94::NOP);
|
|
||||||
std::uniform_int_distribution<int> modifier_dist(lib94::A, lib94::I);
|
|
||||||
std::uniform_int_distribution<int> mode_dist(lib94::IMMEDIATE, lib94::B_INCREMENT);
|
|
||||||
instr.op = (lib94::opcode)opcode_dist(prng);
|
|
||||||
instr.mod = (lib94::modifier)modifier_dist(prng);
|
|
||||||
instr.amode = (lib94::mode)mode_dist(prng);
|
|
||||||
instr.bmode = (lib94::mode)mode_dist(prng);
|
|
||||||
}
|
|
||||||
|
|
||||||
//sets random numbers
|
void head_main();
|
||||||
void random_numbers(lib94::instruction &instr) {
|
void child_main();
|
||||||
std::uniform_int_distribution<lib94::number_t> number_dist(0, LIB94_CORE_SIZE - 1);
|
|
||||||
instr.anumber = number_dist(prng);
|
|
||||||
instr.bnumber = number_dist(prng);
|
|
||||||
}
|
|
||||||
|
|
||||||
//sets random numbers within a range; left_limit and right_limit are assumed to be >= -LIB94_CORE_SIZE
|
|
||||||
void random_numbers(lib94::instruction &instr, std::make_signed_t<lib94::number_t> left_limit, std::make_signed_t<lib94::number_t> right_limit) {
|
|
||||||
std::uniform_int_distribution<std::make_signed_t<lib94::number_t>> number_dist(left_limit, right_limit);
|
|
||||||
instr.anumber = (number_dist(prng) + LIB94_CORE_SIZE) % LIB94_CORE_SIZE;
|
|
||||||
instr.bnumber = (number_dist(prng) + LIB94_CORE_SIZE) % LIB94_CORE_SIZE;
|
|
||||||
}
|
|
||||||
|
|
||||||
void random_warrior(lib94::warrior &w) {
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> length_dist(1, 10);
|
|
||||||
int length = length_dist(prng);
|
|
||||||
w.instructions.resize(length);
|
|
||||||
|
|
||||||
for (int i = 0; i < length; ++i) {
|
|
||||||
random_instruction(w.instructions[i]);
|
|
||||||
//allow numbers to point up to one instruction before or after warrior
|
|
||||||
random_numbers(w.instructions[i], -i - 1, length - i);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> org_dist(0, length - 1);
|
|
||||||
w.org = org_dist(prng);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void mutate(lib94::warrior &w) {
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> case_dist(0, 9);
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> instr_dist(0, w.instructions.size() - 1);
|
|
||||||
int instr = instr_dist(prng);
|
|
||||||
|
|
||||||
lib94::instruction i;
|
|
||||||
|
|
||||||
switch (case_dist(prng)) {
|
|
||||||
|
|
||||||
case 0:
|
|
||||||
return;
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
case 2:
|
|
||||||
random_instruction(w.instructions[instr]);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 3:
|
|
||||||
case 4:
|
|
||||||
random_numbers(w.instructions[instr], -instr - 1, w.instructions.size() - instr);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 5:
|
|
||||||
random_numbers(w.instructions[instr]);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 6:
|
|
||||||
random_instruction(i);
|
|
||||||
random_numbers(i, -instr - 1, w.instructions.size() + 1 - instr);
|
|
||||||
w.instructions.insert(w.instructions.cbegin() + instr, i);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 7:
|
|
||||||
random_instruction(i);
|
|
||||||
random_numbers(i, -instr - 2, w.instructions.size() - instr);
|
|
||||||
w.instructions.insert(w.instructions.cbegin() + instr + 1, i);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 8:
|
|
||||||
case 9:
|
|
||||||
if (w.instructions.size() > 1)
|
|
||||||
w.instructions.erase(w.instructions.cbegin() + instr);
|
|
||||||
break;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
void crossover(const lib94::warrior &a, const lib94::warrior &b, lib94::warrior &c) {
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> b_offset_dist(0, a.instructions.size());
|
|
||||||
int b_offset = b_offset_dist(prng);
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> b_start_dist(b_offset, std::min(a.instructions.size(), b_offset + b.instructions.size()));
|
|
||||||
int b_start = b_start_dist(prng);
|
|
||||||
|
|
||||||
std::uniform_int_distribution<int> b_end_dist(b_start, std::min(a.instructions.size(), b_offset + b.instructions.size()));
|
|
||||||
int b_end = b_end_dist(prng);
|
|
||||||
|
|
||||||
c.instructions.resize(a.instructions.size());
|
|
||||||
std::copy(a.instructions.cbegin(), a.instructions.cbegin() + b_start, c.instructions.begin());
|
|
||||||
std::copy(b.instructions.cbegin() + b_start - b_offset, b.instructions.cbegin() + b_end - b_offset, c.instructions.begin() + b_start);
|
|
||||||
std::copy(a.instructions.cbegin() + b_end, a.instructions.cend(), c.instructions.begin() + b_end);
|
|
||||||
|
|
||||||
c.org = a.org;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
const lib94::instruction background = {
|
|
||||||
.op = lib94::DAT,
|
|
||||||
.mod = lib94::F,
|
|
||||||
.amode = lib94::DIRECT,
|
|
||||||
.bmode = lib94::DIRECT,
|
|
||||||
.anumber = 0,
|
|
||||||
.bnumber = 0
|
|
||||||
};
|
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
|
|
||||||
prng.seed(time(0));
|
MPI_Init(&argc, &argv);
|
||||||
|
|
||||||
if (argc != 3) {
|
MPI_Comm_rank(MPI_COMM_WORLD, &comm_rank);
|
||||||
std::cerr << "please call this with the path to a warrior to beat followed by the path to a directory to store the outputs." << std::endl;
|
MPI_Comm_size(MPI_COMM_WORLD, &comm_size);
|
||||||
|
|
||||||
|
if (comm_size < 2) {
|
||||||
|
std::cerr << "you need at least two processes." << std::endl;
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (argc < 4) {
|
||||||
|
if (comm_rank == 0)
|
||||||
|
std::cerr << "usage: " << argv[0] << " <warrior to beat> <output directory> <seed 1> <seed 2> ..." << std::endl;
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
warrior_to_beat = lib94::compile_warrior_from_file(argv[1]);
|
warrior_to_beat = lib94::compile_warrior_from_file(argv[1]);
|
||||||
}
|
}
|
||||||
catch (const lib94::compiler_exception &ex) {
|
catch (const lib94::compiler_exception &ex) {
|
||||||
std::cerr << "error compiling " << argv[1] << ": " << ex.message << " on line " << ex.source_line_number << '.' << std::endl;
|
if (comm_rank == 0)
|
||||||
return 1;
|
std::cerr
|
||||||
|
<< "syntax error on line " << ex.source_line_number
|
||||||
|
<< " of " << argv[1] << ":\n " << ex.message << std::endl;
|
||||||
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!warrior_to_beat) {
|
if (warrior_to_beat == 0) {
|
||||||
std::cerr << "could not open " << argv[1] << '.' << std::endl;
|
if (comm_rank == 0)
|
||||||
return 1;
|
std::cerr << "could not read file " << argv[1] << std::endl;
|
||||||
|
return 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::filesystem::path output_dir(argv[2]);
|
output_dir = argv[2];
|
||||||
|
|
||||||
if (std::filesystem::exists(output_dir)) {
|
for (int i = 3; i < argc; ++i) {
|
||||||
std::cerr << output_dir << " already exists." << std::endl;
|
|
||||||
return 1;
|
try {
|
||||||
|
seed_warriors.push_back(lib94::compile_warrior_from_file(argv[i]));
|
||||||
|
}
|
||||||
|
catch (const lib94::compiler_exception &ex) {
|
||||||
|
if (comm_rank == 0)
|
||||||
|
std::cerr
|
||||||
|
<< "syntax error on line " << ex.source_line_number
|
||||||
|
<< " of " << argv[i] << ":\n " << ex.message << std::endl;
|
||||||
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!std::filesystem::create_directories(output_dir)) {
|
if (seed_warriors.back() == 0) {
|
||||||
std::cerr << "could not create " << output_dir << '.' << std::endl;
|
if (comm_rank == 0)
|
||||||
return 1;
|
std::cerr << "could not read file " << argv[i] << std::endl;
|
||||||
|
return 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
generation_number = 1;
|
|
||||||
current_generation.resize(warriors_per_generation);
|
|
||||||
for (int i = 0; i < warriors_per_generation; ++i) {
|
|
||||||
random_warrior(current_generation[i].w);
|
|
||||||
current_generation[i].w.name = "g1w" + std::to_string(i + 1);
|
|
||||||
current_generation[i].score = -1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int last_best_score = 0;
|
if (comm_rank == 0)
|
||||||
|
head_main();
|
||||||
|
|
||||||
|
else
|
||||||
|
child_main();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
struct scored_warrior {
|
||||||
|
int score;
|
||||||
|
lib94::warrior *w;
|
||||||
|
bool operator <(const scored_warrior &other) const {
|
||||||
|
return score > other.score || (score == other.score && w->instructions.size() < other.w->instructions.size());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void head_main() {
|
||||||
|
|
||||||
|
std::cout << "\x1b""7\x1b[?47h\x1b[2J" << std::flush;
|
||||||
|
|
||||||
|
std::vector<scored_warrior> warriors;
|
||||||
|
for (lib94::warrior *const &w : seed_warriors)
|
||||||
|
warriors.push_back({.score = -1, .w = w});
|
||||||
|
|
||||||
|
int *warrior_counts = new int[comm_size];
|
||||||
|
|
||||||
|
int round = 1;
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
|
|
||||||
std::shuffle(current_generation.begin(), current_generation.end(), prng);
|
std::cout << "\x1b[1;1Hhead node: sending warriors\x1b[0K" << std::flush;
|
||||||
|
|
||||||
std::cout << "generation " << generation_number << ". " << std::flush;
|
int warrior_count = warriors.size();
|
||||||
|
MPI_Bcast(&warrior_count, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
|
||||||
int best_score = 0;
|
for (scored_warrior &sw : warriors) {
|
||||||
const lib94::warrior *best_w = ¤t_generation[0].w;
|
std::vector<uint8_t> serialization;
|
||||||
|
lib94::serialize_warrior(sw.w, serialization);
|
||||||
|
int serialization_length = serialization.size();
|
||||||
|
MPI_Bcast(&serialization_length, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
MPI_Bcast(serialization.data(), serialization_length, MPI_UINT8_T, 0, MPI_COMM_WORLD);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (round == 1) {
|
||||||
|
for (scored_warrior &sw : warriors)
|
||||||
|
delete sw.w;
|
||||||
|
warriors.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "\x1b[1;1Hhead node: waiting for child nodes\x1b[0K" << std::flush;
|
||||||
|
|
||||||
|
MPI_Gather(warrior_counts, 1, MPI_INT, warrior_counts, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
|
||||||
|
for (int i = 1; i < comm_size; ++i) {
|
||||||
|
std::cout << "\x1b[1;1Hhead node: receiving warriors from child node " << i << "\x1b[0K" << std::flush;
|
||||||
|
for (int j = 0; j < warrior_counts[i]; ++j) {
|
||||||
|
int score;
|
||||||
|
MPI_Recv(&score, 1, MPI_INT, i, 0, MPI_COMM_WORLD, 0);
|
||||||
|
int serialization_length;
|
||||||
|
MPI_Recv(&serialization_length, 1, MPI_INT, i, 0, MPI_COMM_WORLD, 0);
|
||||||
|
uint8_t *buffer = new uint8_t[serialization_length];
|
||||||
|
MPI_Recv(buffer, serialization_length, MPI_UINT8_T, i, 0, MPI_COMM_WORLD, 0);
|
||||||
|
warriors.push_back({.score = score, .w = lib94::deserialize_warrior(buffer)});
|
||||||
|
delete[] buffer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::sort(warriors.begin(), warriors.end());
|
||||||
|
|
||||||
|
std::filesystem::path out_path = output_dir / (std::to_string(round) + ".red");
|
||||||
|
|
||||||
int total_score = 0;
|
int total_score = 0;
|
||||||
|
for (const scored_warrior &sw : warriors)
|
||||||
|
total_score += sw.score;
|
||||||
|
|
||||||
for (winfo &wi : current_generation) {
|
std::cout << "\x1b[" << (comm_size + 2) << ";1Hmin/avg/max score from round " << round << ":\n "
|
||||||
|
<< (float)(warriors[warriors.size() - 1].score * 100) / (rounds_per_matchup * 2) << "% / "
|
||||||
|
<< (float)(total_score * 100) / (warriors.size() * rounds_per_matchup * 2) << "% / "
|
||||||
|
<< (float)(warriors[0].score * 100) / (rounds_per_matchup * 2) << "%\x1b[0K\n\nbest was:\n";
|
||||||
|
|
||||||
if (wi.score == -1) {
|
std::ofstream out_file(out_path);
|
||||||
|
if (!out_file) {
|
||||||
const lib94::warrior *ws[2] = { warrior_to_beat, &wi.w };
|
std::cout << "\x1b[?47l\x1b""8" << std::flush;
|
||||||
int score = score_rounds;
|
std::cerr << "could not open file " << out_path << std::endl;
|
||||||
|
int zero = 0;
|
||||||
for (int i = 0; i < score_rounds; ++i) {
|
MPI_Bcast(&zero, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
const lib94::warrior *winner = lib94::do_round(background, ws, 2, 0, true, rounds_to_tie);
|
MPI_Finalize();
|
||||||
if (winner == warrior_to_beat)
|
exit(5);
|
||||||
--score;
|
|
||||||
else if (winner != 0)
|
|
||||||
++score;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
wi.score = score;
|
out_file << ";author evolver\n;name evolved\n";
|
||||||
|
out_file << ";score was " << (float)(warriors[0].score * 100) / (rounds_per_matchup * 2) << "%\n";
|
||||||
|
out_file << "org " << warriors[0].w->org << '\n';
|
||||||
|
std::cout << " org " << warriors[0].w->org << "\x1b[0K\n";
|
||||||
|
for (const lib94::instruction &i : warriors[0].w->instructions) {
|
||||||
|
out_file << lib94::instruction_to_string(i) << '\n';
|
||||||
|
std::cout << " " << lib94::instruction_to_string(i) << "\x1b[0K\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (wi.score > best_score) {
|
std::cout << "\x1b[0J" << std::flush;
|
||||||
best_score = wi.score;
|
|
||||||
best_w = &wi.w;
|
out_file.close();
|
||||||
|
|
||||||
|
if (warriors[0].score == rounds_per_matchup * 2) {
|
||||||
|
std::cout << "\x1b[?47l\x1b""8completed in " << round << " rounds\n";
|
||||||
|
int zero = 0;
|
||||||
|
MPI_Bcast(&zero, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
MPI_Finalize();
|
||||||
|
exit(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
total_score += wi.score;
|
size_t good_warriors = std::min(max_good_warriors, warriors.size());
|
||||||
|
|
||||||
}
|
for (size_t i = good_warriors; i < warriors.size(); ++i)
|
||||||
|
delete warriors[i].w;
|
||||||
|
|
||||||
std::cout << "average / maximum score " << (double)total_score / current_generation.size() << " / " << best_score << ". " << std::endl;
|
warriors.resize(good_warriors);
|
||||||
|
|
||||||
if (best_score > last_best_score) {
|
++round;
|
||||||
|
|
||||||
last_best_score = best_score;
|
|
||||||
|
|
||||||
std::filesystem::path file_path = output_dir / (std::to_string(generation_number) + ".red");
|
|
||||||
std::ofstream file(file_path);
|
|
||||||
|
|
||||||
if (!file) {
|
|
||||||
std::cerr << "could not open " << file_path << '.' << std::endl;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
file << ";author evolver\n;name " << best_w->name << "\n;score = " << best_score << " / " << score_rounds * 2 << "\norg " << best_w->org << '\n';
|
|
||||||
for (const lib94::instruction &i : best_w->instructions)
|
|
||||||
file << lib94::instruction_to_string(i) << '\n';
|
|
||||||
|
|
||||||
file.close();
|
|
||||||
|
|
||||||
if (best_score == score_rounds * 2) {
|
|
||||||
std::cout << "perfect score reached." << std::endl;
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
++generation_number;
|
void randomize_in_range(lib94::instruction &instr, int min, int max) {
|
||||||
|
instr.op = (lib94::opcode)(rand() % 16);
|
||||||
|
instr.mod = (lib94::modifier)(rand() % 7);
|
||||||
|
instr.amode = (lib94::mode)(rand() % 8);
|
||||||
|
instr.bmode = (lib94::mode)(rand() % 8);
|
||||||
|
instr.anumber = (rand() % (max - min + 1) + min + LIB94_CORE_SIZE) % LIB94_CORE_SIZE;
|
||||||
|
instr.bnumber = (rand() % (max - min + 1) + min + LIB94_CORE_SIZE) % LIB94_CORE_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<winfo> new_warriors;
|
void randomize(lib94::instruction &instr) {
|
||||||
new_warriors.reserve(current_generation.size() + 2);
|
instr.op = (lib94::opcode)(rand() % 16);
|
||||||
|
instr.mod = (lib94::modifier)(rand() % 7);
|
||||||
|
instr.amode = (lib94::mode)(rand() % 8);
|
||||||
|
instr.bmode = (lib94::mode)(rand() % 8);
|
||||||
|
instr.anumber = rand() % LIB94_CORE_SIZE;
|
||||||
|
instr.bnumber = rand() % LIB94_CORE_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
new_warriors.push_back({.w = *best_w, .score = best_score});
|
void mutate_step(lib94::warrior *w) {
|
||||||
|
|
||||||
std::uniform_int_distribution<int> score_at_least(best_score > score_rounds ? score_rounds : 0, best_score > score_rounds ? score_rounds * 2 + 1 : score_rounds);
|
if (w->instructions.size() == 0) {
|
||||||
|
bool in_range = rand() % 2 == 0;
|
||||||
|
lib94::instruction instruction;
|
||||||
|
if (in_range)
|
||||||
|
randomize_in_range(instruction, 0, 0);
|
||||||
|
else
|
||||||
|
randomize(instruction);
|
||||||
|
w->instructions.push_back(instruction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
for (const winfo &wi : current_generation)
|
int rand_inclusive = rand() % (w->instructions.size() + 1);
|
||||||
if (wi.score >= score_at_least(prng))
|
int rand_exclusive = rand() % w->instructions.size();
|
||||||
new_warriors.push_back(wi);
|
bool in_range = rand() % 2 == 0;
|
||||||
|
|
||||||
std::uniform_int_distribution<int> warrior_dist(0, new_warriors.size() - 1);
|
switch (rand() % 4) {
|
||||||
|
|
||||||
do {
|
//org
|
||||||
|
case 0:
|
||||||
|
w->org = rand_exclusive;
|
||||||
|
return;
|
||||||
|
|
||||||
lib94::warrior w1 = new_warriors[warrior_dist(prng)].w;
|
//insert
|
||||||
lib94::warrior w2 = new_warriors[warrior_dist(prng)].w;
|
case 1:
|
||||||
|
w->instructions.insert(w->instructions.cbegin() + rand_inclusive, (lib94::instruction){});
|
||||||
|
if (w->org > rand_inclusive)
|
||||||
|
++w->org;
|
||||||
|
rand_exclusive = rand_inclusive;
|
||||||
|
//FALLTHRU
|
||||||
|
|
||||||
mutate(w1);
|
//modify
|
||||||
mutate(w2);
|
case 2:
|
||||||
|
if (in_range)
|
||||||
|
randomize_in_range(w->instructions[rand_exclusive], -rand_exclusive, w->instructions.size() - rand_exclusive - 1);
|
||||||
|
else
|
||||||
|
randomize(w->instructions[rand_exclusive]);
|
||||||
|
return;
|
||||||
|
|
||||||
lib94::warrior w;
|
//remove
|
||||||
crossover(w1, w2, w);
|
case 3:
|
||||||
w.name = "g" + std::to_string(generation_number) + "w" + std::to_string(new_warriors.size() + 1);
|
w->instructions.erase(w->instructions.cbegin() + rand_exclusive);
|
||||||
|
if (w->org > rand_exclusive)
|
||||||
new_warriors.push_back({.w = w, .score = -1});
|
--w->org;
|
||||||
|
return;
|
||||||
} while (new_warriors.size() < warriors_per_generation);
|
|
||||||
|
}
|
||||||
current_generation = std::move(new_warriors);
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void mutate(lib94::warrior *w) {
|
||||||
|
do
|
||||||
|
mutate_step(w);
|
||||||
|
while (rand() % 2 == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int score(const lib94::warrior *w) {
|
||||||
|
int the_score = 0;
|
||||||
|
const lib94::warrior *const warriors[2] = {warrior_to_beat, w};
|
||||||
|
for (int i = 0; i < rounds_per_matchup; ++i) {
|
||||||
|
const lib94::warrior *winner = lib94::do_round(background_instruction, warriors, 2, 0, true, steps_to_tie);
|
||||||
|
if (winner == 0)
|
||||||
|
++the_score;
|
||||||
|
else if (winner == w)
|
||||||
|
the_score += 2;
|
||||||
|
}
|
||||||
|
return the_score;
|
||||||
|
}
|
||||||
|
|
||||||
|
void child_main() {
|
||||||
|
|
||||||
|
srand(time(0));
|
||||||
|
|
||||||
|
std::vector<scored_warrior> warriors;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
|
||||||
|
int warrior_count;
|
||||||
|
MPI_Bcast(&warrior_count, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
|
||||||
|
if (warrior_count == 0) {
|
||||||
|
MPI_Finalize();
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
warriors.resize(warrior_count);
|
||||||
|
|
||||||
|
std::cout << "\x1b[" << (comm_rank + 1) << ";1Hchild node " << comm_rank << ": receiving warriors\x1b[0K" << std::flush;
|
||||||
|
|
||||||
|
for (int i = 0; i < warrior_count; ++i) {
|
||||||
|
int serialization_length;
|
||||||
|
MPI_Bcast(&serialization_length, 1, MPI_INT, 0, MPI_COMM_WORLD);
|
||||||
|
uint8_t *buffer = new uint8_t[serialization_length];
|
||||||
|
MPI_Bcast(buffer, serialization_length, MPI_UINT8_T, 0, MPI_COMM_WORLD);
|
||||||
|
warriors[i].w = lib94::deserialize_warrior(buffer);
|
||||||
|
delete[] buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "\x1b[" << (comm_rank + 1) << ";1Hchild node " << comm_rank << ": mutating warriors\x1b[0K" << std::flush;
|
||||||
|
|
||||||
|
for (scored_warrior &sw : warriors)
|
||||||
|
mutate(sw.w);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < warriors.size(); ++i) {
|
||||||
|
std::cout << "\x1b[" << (comm_rank + 1) << ";1Hchild node " << comm_rank << ": scoring warriors " << (float)(i * 100) / warriors.size() << "%\x1b[0K" << std::flush;
|
||||||
|
warriors[i].score = score(warriors[i].w);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::cout << "\x1b[" << (comm_rank + 1) << ";1Hchild node " << comm_rank << ": waiting for other child nodes\x1b[0K" << std::flush;
|
||||||
|
|
||||||
|
MPI_Gather(&warrior_count, 1, MPI_INT, 0, 0, 0, 0, MPI_COMM_WORLD);
|
||||||
|
|
||||||
|
for (scored_warrior &sw : warriors) {
|
||||||
|
std::vector<uint8_t> serialization;
|
||||||
|
lib94::serialize_warrior(sw.w, serialization);
|
||||||
|
int serialization_length = serialization.size();
|
||||||
|
MPI_Send(&sw.score, 1, MPI_INT, 0, 0, MPI_COMM_WORLD);
|
||||||
|
MPI_Send(&serialization_length, 1, MPI_INT, 0, 0, MPI_COMM_WORLD);
|
||||||
|
MPI_Send(serialization.data(), serialization_length, MPI_UINT8_T, 0, 0, MPI_COMM_WORLD);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (scored_warrior &sw : warriors)
|
||||||
|
delete sw.w;
|
||||||
|
|
||||||
|
warriors.clear();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -63,6 +63,15 @@ namespace lib94 {
|
||||||
std::vector<instruction> instructions;
|
std::vector<instruction> instructions;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//serialize a warrior in such a way that it can be read back by deserialize_warrior.
|
||||||
|
//does not consider name and author.
|
||||||
|
void serialize_warrior(const warrior *the_warrior, std::vector<uint8_t> &into);
|
||||||
|
|
||||||
|
//deserialize a warrior that has been serialized by serialize_warrior.
|
||||||
|
//no error checking is performed.
|
||||||
|
//does not consider name and author.
|
||||||
|
warrior *deserialize_warrior(const uint8_t *from);
|
||||||
|
|
||||||
//this seeds the prng used to place warriors into the core at the start of a round
|
//this seeds the prng used to place warriors into the core at the start of a round
|
||||||
//if this is never called, 0 is used as a seed and the placements are deterministic.
|
//if this is never called, 0 is used as a seed and the placements are deterministic.
|
||||||
void seed_prng(uint_fast64_t seed);
|
void seed_prng(uint_fast64_t seed);
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
#include <lib94/lib94.hpp>
|
#include <lib94/lib94.hpp>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
#include <cstring>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
|
@ -47,6 +48,26 @@ namespace lib94 {
|
||||||
mode_chars[instr.bmode] + std::to_string(instr.bnumber);
|
mode_chars[instr.bmode] + std::to_string(instr.bnumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void serialize_warrior(const warrior *the_warrior, std::vector<uint8_t> &into) {
|
||||||
|
size_t ic = the_warrior->instructions.size();
|
||||||
|
into.resize(sizeof(number_t) + sizeof(size_t) + sizeof(instruction) * ic);
|
||||||
|
memcpy(into.data(), &the_warrior->org, sizeof(number_t));
|
||||||
|
memcpy(into.data() + sizeof(number_t), &ic, sizeof(size_t));
|
||||||
|
memcpy(into.data() + sizeof(number_t) + sizeof(size_t),
|
||||||
|
the_warrior->instructions.data(), sizeof(instruction) * ic);
|
||||||
|
}
|
||||||
|
|
||||||
|
warrior *deserialize_warrior(const uint8_t *from) {
|
||||||
|
warrior *w = new warrior();
|
||||||
|
size_t ic;
|
||||||
|
memcpy(&w->org, from, sizeof(number_t));
|
||||||
|
memcpy(&ic, from + sizeof(number_t), sizeof(size_t));
|
||||||
|
w->instructions.resize(ic);
|
||||||
|
memcpy(w->instructions.data(),
|
||||||
|
from + sizeof(number_t) + sizeof(size_t), sizeof(instruction) * ic);
|
||||||
|
return w;
|
||||||
|
}
|
||||||
|
|
||||||
[[noreturn]] static void throw_compiler_exception(unsigned source_line_number, std::string message) {
|
[[noreturn]] static void throw_compiler_exception(unsigned source_line_number, std::string message) {
|
||||||
compiler_exception ex;
|
compiler_exception ex;
|
||||||
ex.source_line_number = source_line_number;
|
ex.source_line_number = source_line_number;
|
||||||
|
|
6
makefile
6
makefile
|
@ -18,7 +18,7 @@ bin/bench: obj/bench/main.o obj/bench/bench_window.o obj/bench/core_widget.o obj
|
||||||
|
|
||||||
bin/evolver: obj/evolver/evolver.o obj/lib94.o
|
bin/evolver: obj/evolver/evolver.o obj/lib94.o
|
||||||
@mkdir -p $(dir $@)
|
@mkdir -p $(dir $@)
|
||||||
g++ ${CPP_ARGS} $^ -o $@
|
g++ ${CPP_ARGS} $^ ${MPI_LD_ARGS} -o $@
|
||||||
|
|
||||||
bin/tabulator-mpi: obj/tabulator-mpi/source.o obj/lib94.o
|
bin/tabulator-mpi: obj/tabulator-mpi/source.o obj/lib94.o
|
||||||
@mkdir -p $(dir $@)
|
@mkdir -p $(dir $@)
|
||||||
|
@ -31,6 +31,10 @@ obj/bench/%.o: bench/%.cpp
|
||||||
@mkdir -p $(dir $@)
|
@mkdir -p $(dir $@)
|
||||||
g++ -c ${CPP_ARGS} ${GTKMM_CPP_ARGS} $^ -o $@
|
g++ -c ${CPP_ARGS} ${GTKMM_CPP_ARGS} $^ -o $@
|
||||||
|
|
||||||
|
obj/evolver/%.o: evolver/%.cpp
|
||||||
|
@mkdir -p $(dir $@)
|
||||||
|
g++ -c ${CPP_ARGS} ${MPI_CPP_ARGS} $^ -o $@
|
||||||
|
|
||||||
obj/tabulator-mpi/%.o: tabulator-mpi/%.cpp
|
obj/tabulator-mpi/%.o: tabulator-mpi/%.cpp
|
||||||
@mkdir -p $(dir $@)
|
@mkdir -p $(dir $@)
|
||||||
g++ -c ${CPP_ARGS} ${MPI_CPP_ARGS} $^ -o $@
|
g++ -c ${CPP_ARGS} ${MPI_CPP_ARGS} $^ -o $@
|
||||||
|
|
11
readme.txt
11
readme.txt
|
@ -41,8 +41,11 @@ Note that tabulator expects at least two processes (one head process and at leas
|
||||||
|
|
||||||
=== evolver ===
|
=== evolver ===
|
||||||
|
|
||||||
The "evolver" program attempts to create warriors who are good against a particular other warrior via a process similar to genetic
|
The experimental "evolver" program attempts to create warriors who are good against a particular other
|
||||||
evolution. It is highly experimental and does not always work. I may or may not attempt to improve it in the future.
|
warrior via a process similar to genetic evolution. I may or may not attempt to improve it in the future.
|
||||||
|
|
||||||
To evolve a warrior against the included Epson warrior, with the outputs from each round stored in a directory named output, run
|
The evolver takes an arbitrary number of input warriors as "seeds". To evolve a warrior against Dwarf, using
|
||||||
bin/evolver warriors/epson.red output
|
all of the included warriors as seeds, and storing the outputs from each round in a directory named output, run
|
||||||
|
mpirun bin/evolver warriors/dwarf.red output warriors/*.red
|
||||||
|
|
||||||
|
As with tabulator, evolver requires at least two processes, and you will have to oversubscribe if you only have one core.
|
||||||
|
|
Loading…
Add table
Reference in a new issue