-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathDay20.cs
144 lines (117 loc) · 4.98 KB
/
Day20.cs
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
using System;
using System.Collections.Generic;
using System.Numerics;
using AdventOfCode.CSharp.Common;
namespace AdventOfCode.CSharp.Y2023.Solvers;
public class Day20 : ISolver
{
/**
* This solution is quite cheaty to be honest as it directly exploits how the inputs are structured and will not
* solve any kind of general case. I suppose a less cheaty implementation would try and guess which conjunction
* nodes appear to be cyclic in nature and then perform an LCM (or CRT?) on those cycles to determine what the
* value of rx is, but even then that's still assuming a lot of things so I'm just going to stick with this.
*/
public enum ModuleType : byte { FlipFlop, Conjunction, Broadcast, Final }
record Module(ModuleType Type, List<int> Destinations, List<int> Senders);
public static void Solve(ReadOnlySpan<byte> input, Solution solution)
{
Dictionary<int, Module> modules = new(64);
var rxModule = new Module(ModuleType.Final, [], new(4));
modules['r' << 8 | 'x'] = rxModule;
while (!input.IsEmpty)
{
ModuleType type = input[0] switch { (byte)'%' => ModuleType.FlipFlop, (byte)'&' => ModuleType.Conjunction, _ => ModuleType.Broadcast };
int name;
if (type == ModuleType.Broadcast)
{
name = -1;
input = input.Slice("broadcaster -> ".Length);
}
else
{
name = input[1] << 8 | input[2];
input = input.Slice("%jp -> ".Length);
}
var destinations = new List<int>(10);
int i = 0;
while (true)
{
byte c1 = input[i++];
byte c2 = input[i++];
destinations.Add(c1 << 8 | c2);
if (input[i++] == '\n')
break;
i++;
}
modules[name] = new Module(type, destinations, new(10));
input = input.Slice(i);
}
foreach ((int name, Module module) in modules)
foreach (int destination in module.Destinations)
modules[destination].Senders.Add(name);
long lowPulses = 1000; // 1000 because of 1000 button presses
long highPulses = 0;
long part2 = 1;
int nodeBeforeLast = rxModule.Senders[0];
foreach (int inputNode in modules[nodeBeforeLast].Senders)
{
lowPulses += 1000; // For each pulse from the broadcast node
int inputCenter = modules[inputNode].Senders[0];
Module centerModule = modules[inputCenter];
Module start = modules[centerModule.Senders[0]]; // could be in the middle, we don't know
int number = 1;
Module cur = start;
int bit = 2;
while (true)
{
List<int> destinations = cur.Destinations;
int nextId = destinations[0];
if (nextId == inputCenter)
{
if (destinations.Count == 1)
{
number |= bit >> 1; // we reached the end, so set the previous bit
break;
}
nextId = destinations[1];
}
cur = modules[nextId];
if (cur.Destinations.Count == 2)
number |= bit;
bit <<= 1;
}
cur = start;
while (true)
{
List<int> senders = cur.Senders;
int prevId = senders[0];
if (prevId == inputCenter)
prevId = senders[1];
if (prevId == -1)
break;
cur = modules[prevId];
number = (number << 1) | (cur.Destinations.Count - 1);
}
part2 *= number;
int bitsInNumber = 32 - BitOperations.LeadingZeroCount((uint)number);
int highPerPulseToCenter = bitsInNumber - BitOperations.PopCount((uint)number) + 3; // each pulse to center, sends a high pulse to each zero, and 3 extra modules
int flips = 1000;
while (flips > 0)
{
int isOneBit = number & 1;
int numDestinations = isOneBit + 1;
int numLowPulses = flips / 2;
// half of the flips are low pulse, the other high pulse
lowPulses += numDestinations * numLowPulses;
highPulses += numDestinations * ((flips + 1) / 2);
// if this is a 1 bit, then each pulse to the center will cause additional pulses
lowPulses += isOneBit * flips; // 1 low pulse between the modules that connect the center to rx
highPulses += isOneBit * flips * highPerPulseToCenter;
number >>= 1;
flips >>= 1;
}
}
solution.SubmitPart1(lowPulses * highPulses);
solution.SubmitPart2(part2);
}
}