Added 2017/18 part 1

This commit is contained in:
FrederikBaerentsen 2024-11-27 16:45:03 +01:00
parent bfb1ae0d73
commit 1ecac41ed4
2 changed files with 92 additions and 38 deletions

View File

@ -74,6 +74,67 @@ last sound played is `4`.
recently played sound) the *first* time a `rcv` instruction is executed recently played sound) the *first* time a `rcv` instruction is executed
with a non-zero value? with a non-zero value?
To begin, [get your puzzle input](18/input). Your puzzle answer was `8600`.
The first half of this puzzle is complete! It provides one gold star: \*
## \-\-- Part Two \-\-- {#part2}
As you congratulate yourself for a job well done, you notice that the
documentation has been on the back of the tablet this entire time. While
you actually got most of the instructions correct, there are a few key
differences. This assembly code isn\'t about sound at all - it\'s meant
to be run *twice at the same time*.
Each running copy of the program has its own set of registers and
follows the code independently - in fact, the programs don\'t even
necessarily run at the same speed. To coordinate, they use the *send*
(`snd`) and *receive* (`rcv`) instructions:
- `snd X` *sends* the value of `X` to the other program. These values
wait in a queue until that program is ready to receive them. Each
program has its own message queue, so a program can never receive a
message it sent.
- `rcv X` *receives* the next value and stores it in register `X`. If
no values are in the queue, the program *waits for a value to be
sent to it*. Programs do not continue to the next instruction until
they have received a value. Values are received in the order they
are sent.
Each program also has its own *program ID* (one `0` and the other `1`);
the register `p` should begin with this value.
For example:
snd 1
snd 2
snd p
rcv a
rcv b
rcv c
rcv d
Both programs begin by sending three values to the other. Program `0`
sends `1, 2, 0`; program `1` sends `1, 2, 1`. Then, each program
receives a value (both `1`) and stores it in `a`, receives another value
(both `2`) and stores it in `b`, and then each receives the program ID
of the other program (program `0` receives `1`; program `1` receives
`0`) and stores it in `c`. Each program now sees a different value in
its own copy of register `c`.
Finally, both programs try to `rcv` a *fourth* time, but no data is
waiting for either of them, and they reach a *deadlock*. When this
happens, both programs terminate.
It should be noted that it would be equally valid for the programs to
run at different speeds; for example, program `0` might have sent all
three values and then stopped at the first `rcv` before program `1`
executed even its first instruction.
Once both of your programs have terminated (regardless of what caused
them to do so), *how many times did program `1` send a value*?
Answer: Answer:
Although it hasn\'t changed, you can still [get your puzzle
input](18/input).

View File

@ -4,7 +4,7 @@ from pprint import pprint
sys.path.insert(0, '../../') sys.path.insert(0, '../../')
from fred import list2int from fred import list2int
input_f = 'test' input_f = 'input'
part = 1 part = 1
######################################### #########################################
@ -25,9 +25,20 @@ def parse_input(input_str):
return None return None
def sets_return(x):
if x in Sets:
value = Sets[x]
if isinstance(value, str) and not x.lstrip('-').isdigit():
return sets_return(value)
return value
elif x.isdigit() or x.lstrip('-').isdigit():
return int(x)
else:
return None
instructions = [] instructions = []
sets = {} Sets = {}
last_sound = 0 last_sound = 0
@ -40,63 +51,45 @@ if part == 1:
while True: while True:
i = instructions[x] i = instructions[x]
print(i)
if isinstance(i[1], str):
if i[1] not in Sets:
Sets[i[1]] = 0
if i[0] == 'set': if i[0] == 'set':
if isinstance(i[2], str): Sets[i[1]] = int(sets_return(i[2]))
sets[i[1]] = int(i[2])
else:
sets[i[1]] = int(sets[i[2]])
x += 1 x += 1
elif i[0] == 'add': elif i[0] == 'add':
sets[i[1]] = int(sets[i[1]]) + int(i[2]) Sets[i[1]] += sets_return(i[2])
x += 1 x += 1
elif i[0] == 'mul': elif i[0] == 'mul':
if i[2] in sets: Sets[i[1]] *= sets_return(i[2])
sets[i[1]] = int(sets[i[1]]) * int(sets[i[2]])
else:
sets[i[1]] = int(sets[i[1]]) * int(i[2])
x += 1 x += 1
elif i[0] == 'mod': elif i[0] == 'mod':
sets[i[1]] = int(sets[i[1]]) % int(i[2]) Sets[i[1]] %= sets_return(i[2])
x += 1 x += 1
elif i[0] == 'snd': elif i[0] == 'snd':
last_sound = sets[i[1]] last_sound = sets_return(i[1])
x += 1 x += 1
elif i[0] == 'rcv': elif i[0] == 'rcv':
if sets[i[1]] != 0: if sets_return(i[1]) != 0:
sets[i[1]] = int(sets[i[1]]) Sets[i[1]] = sets_return(i[1])
x += 1 print(last_sound)
else: exit()
print('skipped')
x += 1 x += 1
elif i[0] == 'jgz': #this is the one that fails elif i[0] == 'jgz':
if isinstance(i[2], str): if sets_return(i[1]) > 0:
if sets[i[1]] > 0: x += sets_return(i[2])
x += int(sets[i[2]])
else:
x += int(i[2])
else: else:
x += 1 x += 1
print(x,sets,last_sound)
input()
#print(programs)
#print(instructions)
#print(len(instructions))
#for idx,i in enumerate(instructions):