-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgamato_07.py
192 lines (145 loc) · 9.26 KB
/
gamato_07.py
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
"""
### Day 7: Amplification Circuit ###
--- Part One ---
Based on the navigational maps, you're going to need to send more power to your ship's thrusters to reach Santa in time.
To do this, you'll need to configure a series of amplifiers already installed on the ship.
There are five amplifiers connected in series; each one receives an input signal and produces an output signal.
They are connected such that the first amplifier's output leads to the second amplifier's input, the second amplifier's
output leads to the third amplifier's input, and so on. The first amplifier's input value is 0, and the last
amplifier's output leads to your ship's thrusters.
O-------O O-------O O-------O O-------O O-------O
0 ->| Amp A |->| Amp B |->| Amp C |->| Amp D |->| Amp E |-> (to thrusters)
O-------O O-------O O-------O O-------O O-------O
The Elves have sent you some Amplifier Controller Software (your puzzle input), a program that should run on your
existing Intcode computer. Each amplifier will need to run a copy of the program.
When a copy of the program starts running on an amplifier, it will first use an input instruction to ask the amplifier
for its current phase setting (an integer from 0 to 4). Each phase setting is used exactly once,
but the Elves can't remember which amplifier needs which phase setting.
The program will then call another input instruction to get the amplifier's input signal, compute the correct
output signal, and supply it back to the amplifier with an output instruction.
(If the amplifier has not yet received an input signal, it waits until one arrives.)
Your job is to find the largest output signal that can be sent to the thrusters by trying every possible combination
of phase settings on the amplifiers. Make sure that memory is not shared or reused between copies of the program.
For example, suppose you want to try the phase setting sequence 3,1,2,4,0, which would mean setting amplifier A
to phase setting 3, amplifier B to setting 1, C to 2, D to 4, and E to 0. Then, you could determine the output signal
that gets sent from amplifier E to the thrusters with the following steps:
Start the copy of the amplifier controller software that will run on amplifier A. At its first input instruction,
provide it the amplifier's phase setting, 3. At its second input instruction, provide it the input signal, 0.
After some calculations, it will use an output instruction to indicate the amplifier's output signal.
Start the software for amplifier B. Provide it the phase setting (1) and then whatever output signal was produced
from amplifier A. It will then produce a new output signal destined for amplifier C.
Start the software for amplifier C, provide the phase setting (2) and the value from amplifier B,
then collect its output signal.
Run amplifier D's software, provide the phase setting (4) and input value, and collect its output signal.
Run amplifier E's software, provide the phase setting (0) and input value, and collect its output signal.
The final output signal from amplifier E would be sent to the thrusters. However, this phase setting sequence may not
have been the best one; another sequence might have sent a higher signal to the thrusters.
Here are some example programs:
Max thruster signal 43210 (from phase setting sequence 4,3,2,1,0):
3,15,3,16,1002,16,10,16,1,16,15,15,4,15,99,0,0
Max thruster signal 54321 (from phase setting sequence 0,1,2,3,4):
3,23,3,24,1002,24,10,24,1002,23,-1,23,
101,5,23,23,1,24,23,23,4,23,99,0,0
Max thruster signal 65210 (from phase setting sequence 1,0,4,3,2):
3,31,3,32,1002,32,10,32,1001,31,-2,31,1007,31,0,33,
1002,33,7,33,1,33,31,31,1,32,31,31,4,31,99,0,0,0
Try every combination of phase settings on the amplifiers. What is the highest signal that can be sent to the thrusters?
--- Part Two ---
It's no good - in this configuration, the amplifiers can't generate a large enough output signal to produce the
thrust you'll need. The Elves quickly talk you through rewiring the amplifiers into a feedback loop:
O-------O O-------O O-------O O-------O O-------O
0 -+->| Amp A |->| Amp B |->| Amp C |->| Amp D |->| Amp E |-.
| O-------O O-------O O-------O O-------O O-------O |
| |
'--------------------------------------------------------+
|
v
(to thrusters)
Most of the amplifiers are connected as they were before; amplifier A's output is connected to amplifier B's input,
and so on. However, the output from amplifier E is now connected into amplifier A's input. This creates the feedback
loop: the signal will be sent through the amplifiers many times.
In feedback loop mode, the amplifiers need totally different phase settings: integers from 5 to 9, again each used
exactly once. These settings will cause the Amplifier Controller Software to repeatedly take input and produce output
many times before halting. Provide each amplifier its phase setting at its first input instruction; all further
input/output instructions are for signals.
Don't restart the Amplifier Controller Software on any amplifier during this process. Each one should continue
receiving and sending signals until it halts.
All signals sent or received in this process will be between pairs of amplifiers except the very first signal and the
very last signal. To start the process, a 0 signal is sent to amplifier A's input exactly once.
Eventually, the software on the amplifiers will halt after they have processed the final loop. When this happens,
the last output signal from amplifier E is sent to the thrusters. Your job is to find the largest output signal that
can be sent to the thrusters using the new phase settings and feedback loop arrangement.
Here are some example programs:
Max thruster signal 139629729 (from phase setting sequence 9,8,7,6,5):
3,26,1001,26,-4,26,3,27,1002,27,2,27,1,27,26,
27,4,27,1001,28,-1,28,1005,28,6,99,0,0,5
Max thruster signal 18216 (from phase setting sequence 9,7,8,5,6):
3,52,1001,52,-5,52,3,53,1,52,56,54,1007,54,5,55,1005,55,26,1001,54,
-5,54,1105,1,12,1,53,54,53,1008,54,0,55,1001,55,1,55,2,53,55,53,4,
53,1001,56,-1,56,1005,56,6,99,0,0,0,0,10
Try every combination of the new phase settings on the amplifier feedback loop. What is the highest signal that
can be sent to the thrusters?
"""
def run_intcode(program, inputs, code=None):
if code is None:
code = int(input("INPUT: "))
op_counts = [0, 3, 3, 1, 1, 2, 2, 3, 3]
index = 0
while program[code] != 99:
modes = [int(x) for x in f"{program[code]:0>5}"[:3]][::-1]
instruction = int(f"{program[code]:0>5}"[3:])
op_count = op_counts[instruction]
operands = [program[code + x + 1] if modes[x] else program[program[code + x + 1]] for x in range(op_count)]
if instruction == 1:
program[program[code + 3]] = operands[0] + operands[1]
elif instruction == 2:
program[program[code + 3]] = operands[0] * operands[1]
elif instruction == 3:
program[program[code + 1]] = inputs[index]
index += 1
elif instruction == 4:
return operands[0], code + 2
elif instruction == 5:
code = (operands[1] - 3) if operands[0] != 0 else code
elif instruction == 6:
code = (operands[1] - 3) if operands[0] == 0 else code
elif instruction == 7:
program[program[code + 3]] = int(operands[0] < operands[1])
elif instruction == 8:
program[program[code + 3]] = int(operands[0] == operands[1])
code += op_count + 1
return [False, False]
def amplify_control_system(data, settings, feedback):
prev = 0
program = [int(x) for x in data]
amplifiers = [[program.copy(), 0] for _ in range(5)]
for i in range(5):
output, pc = run_intcode(amplifiers[i][0], [settings[i], prev], code=amplifiers[i][1])
prev = output
amplifiers[i][1] = pc
if not feedback:
return prev
i = 0
while True:
output, pc = run_intcode(amplifiers[i][0], [prev], code=amplifiers[i][1])
if output is False:
break
prev = output
amplifiers[i][1] = pc
i = (i+1) % 5
return prev
def permute_data(arr, tmp, index=0, res=None):
if res is None:
res = []
if index >= len(tmp):
return res.append(tmp.copy())
for i in range(len(arr)):
tmp[index] = arr[i]
permute_data(arr[:i] + arr[i + 1:], tmp, index=index + 1, res=res)
return res
if __name__ == "__main__":
data_set = open("puzzle_inputs/input_day_07.txt", 'r').readline().split(',')
one = permute_data(list(range(5)), list(range(5)))
two = permute_data(list(range(5, 10)), list(range(5, 10)))
print("Part One:", max(amplify_control_system(data_set, permutation, False) for permutation in one))
print("Part Two:", max(amplify_control_system(data_set, permutation, True) for permutation in two))