Skip to content

Commit

Permalink
Puzzle 18 - part 2
Browse files Browse the repository at this point in the history
  • Loading branch information
mody committed Dec 18, 2017
1 parent 68e8fff commit 59a5a6c
Show file tree
Hide file tree
Showing 3 changed files with 190 additions and 1 deletion.
10 changes: 9 additions & 1 deletion CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -281,5 +281,13 @@ sklik_compile_executable(
SOURCES
puzzle_18_1.cc
COMPILE_OPTIONS
-Wall -std=c++14 -g -O0
-Wall -std=c++14 -g -O2
)

sklik_compile_executable(
NAME puzzle_18_2
SOURCES
puzzle_18_2.cc
COMPILE_OPTIONS
-Wall -std=c++14 -g -O2
)
41 changes: 41 additions & 0 deletions input_18_2.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
set i 31
set a 1
mul p 17
jgz p p
mul a 2
add i -1
jgz i -2
add a -1
set i 127
set p 618
mul p 8505
mod p a
mul p 129749
add p 12345
mod p a
set b p
mod b 10000
snd b
add i -1
jgz i -9
jgz a 3
rcv b
jgz b -1
set f 0
set i 126
rcv a
rcv b
set p a
mul p -1
add p b
jgz p 4
snd a
set a b
jgz 1 3
snd b
set f 1
add i -1
jgz i -11
snd a
jgz f -16
jgz a -19
140 changes: 140 additions & 0 deletions puzzle_18_2.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,140 @@
#include <iomanip>
#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <deque>
#include <map>

// snd X plays a sound with a frequency equal to the value of X.
// set X Y sets register X to the value of Y.
// add X Y increases register X by the value of Y.
// mul X Y sets register X to the result of multiplying the value contained in
// register X by the value of Y.
// mod X Y sets register X to the remainder of dividing the value contained in
// register X by the value of Y (that is, it sets X to the result of X modulo Y).
// rcv X recovers the frequency of the last sound played, but only when the
// value of X is not zero. (If it is zero, the command does nothing.)
// jgz X Y jumps with an offset of the value of Y, but only if the value of X
// is greater than zero. (An offset of 2 skips the next instruction, an offset
// of -1 jumps to the previous instruction, and so on.)

using Program = std::vector<std::string>;
using Queue = std::deque<int>;

struct Registers {
using Storage = std::map<std::string, long>;
using iterator = Storage::iterator;

Registers(int pid) : registers{{"p", pid}}
{}

iterator regOrVal(const std::string& input)
{
// std::cout << "input: " << input << " is ";
if (input[0] == '-' || (input[0] >= '0' && input[0] <= '9')) {
auto it = registers.insert({input, atoi(&input[0])}).first;
// std::cout << "val: " << it->second << "\n";
return it;
} else {
auto it = registers.insert({input, 0}).first;
// std::cout << "reg: " << it->second << "\n";
return it;
}
}

private:
Storage registers;
};

struct Context {
Context(const Program& _program, int _pid, Queue& _rq, Queue& _wq)
: program(_program)
, registers{_pid}
, rq(_rq)
, wq(_wq)
, pid(_pid)
{}

const Program& program;
Registers registers;
Queue& rq;
Queue& wq;
int pid;
int ip = 0;
int freq = 0;

bool step() {
if (ip >= program.size()) {
return false;
}

std::stringstream ss{program[ip]};
std::string c, p1, p2;
ss >> c >> p1 >> p2;
// std::cout << "\nip: " << ip << " : " << c << " : " << p1 << " : " << p2 << "\n";
if (c == "snd") {
wq.push_back(registers.regOrVal(p1)->second);
std::cout << "pid: " << pid << " - send " << wq.back() << "\n";
} else if (c == "set") {
auto r1 = registers.regOrVal(p1);
r1->second = registers.regOrVal(p2)->second;
} else if (c == "add") {
auto r1 = registers.regOrVal(p1);
r1->second += registers.regOrVal(p2)->second;
} else if (c == "mul") {
auto r1 = registers.regOrVal(p1);
r1->second *= registers.regOrVal(p2)->second;
} else if (c == "mod") {
auto r1 = registers.regOrVal(p1);
r1->second %= registers.regOrVal(p2)->second;
} else if (c == "rcv") {
if (rq.empty()) {
// block
std::cout << "pid: " << pid << " - recv blocked\n";
return true;
}
auto r1 = registers.regOrVal(p1);
r1->second = rq.front();
rq.pop_front();
std::cout << "pid: " << pid << " - recv " << r1->second << "\n";
} else if (c == "jgz") {
auto r1 = registers.regOrVal(p1);
if (r1->second > 0) {
ip += registers.regOrVal(p2)->second;
return true;
}
} else {
std::cerr << "ERROR: " << c << "\n";
}
++ip;
return true;
}
};




int main()
{
Program program;

std::string line;
while (std::getline(std::cin, line)) {
program.emplace_back(std::move(line));
}

Queue q0, q1;
Context c0(program, 0, q0, q1);
Context c1(program, 1, q1, q0);

for (;;) {
bool pid0 = c0.step();
bool pid1 = c1.step();
if (!pid0 && !pid1) {
break;
}
}

return 0;
}

0 comments on commit 59a5a6c

Please sign in to comment.