-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathDay13.cs
146 lines (121 loc) · 4.73 KB
/
Day13.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
145
146
using System;
using System.Numerics;
using System.Runtime.InteropServices;
using System.Runtime.Intrinsics;
using AdventOfCode.CSharp.Common;
namespace AdventOfCode.CSharp.Y2023.Solvers;
public class Day13 : ISolver
{
public static void Solve(ReadOnlySpan<byte> input, Solution solution)
{
int part1 = 0;
int part2 = 0;
while (true)
{
int gridEnd = input.IndexOf("\n\n"u8);
ReadOnlySpan<byte> grid = gridEnd == -1 ? input : input.Slice(0, gridEnd + 1);
FindMirrorsInGrid(grid, ref part1, ref part2);
// last grid will end with single newline
if (gridEnd == -1)
break;
input = input.Slice(gridEnd + 2);
}
solution.SubmitPart1(part1);
solution.SubmitPart2(part2);
}
private static void FindMirrorsInGrid(ReadOnlySpan<byte> grid, ref int part1, ref int part2)
{
Span<uint> rows = stackalloc uint[24];
Span<uint> cols = stackalloc uint[24];
ExtractRowsAndCols(grid, rows, cols, out int width, out int height);
bool hasSolvedPart1 = false;
bool hasSolvedPart2 = false;
// Find mirror on the rows
FindMirrors(rows.Slice(0, height), 100, ref part1, ref part2, ref hasSolvedPart1, ref hasSolvedPart2);
if (hasSolvedPart1 && hasSolvedPart2)
return;
// Find mirrors on the cols
FindMirrors(cols.Slice(0, width), 1, ref part1, ref part2, ref hasSolvedPart1, ref hasSolvedPart2);
}
private static void ExtractRowsAndCols(ReadOnlySpan<byte> grid, Span<uint> rows, Span<uint> cols, out int width, out int height)
{
width = grid.IndexOf((byte)'\n');
height = grid.Length / (width + 1);
ref byte gridRef = ref MemoryMarshal.GetReference(grid);
int rowOffset = 0;
for (int y = 0; y < height; y++)
{
// load row using SIMD
uint row;
if (grid.Length - rowOffset > Vector256<byte>.Count)
{
Vector256<byte> rowVec = Vector256.LoadUnsafe(ref gridRef, (nuint)rowOffset);
uint bits = Vector256.Equals(rowVec, Vector256.Create((byte)'#')).ExtractMostSignificantBits();
row = bits & ((1U << width) - 1);
}
else
{
// If the row is too small for the buffer, then load the row into the end of the vector
Vector256<byte> rowVec = Vector256.LoadUnsafe(ref gridRef, (nuint)(rowOffset + width - Vector256<byte>.Count));
uint bits = Vector256.Equals(rowVec, Vector256.Create((byte)'#')).ExtractMostSignificantBits();
row = bits >> (Vector256<byte>.Count - width);
}
rows[y] = row;
uint rowBit = 1U << y;
// iterate over row bits with https://lemire.me/blog/2018/02/21/iterating-over-set-bits-quickly/
while (row != 0)
{
uint t = row & (~row + 1);
cols[BitOperations.TrailingZeroCount(t)] |= rowBit;
row ^= t;
}
rowOffset += width + 1;
}
}
private static void FindMirrors(Span<uint> rowsOrCols, int multiplier, ref int part1, ref int part2, ref bool hasSolvedPart1, ref bool hasSolvedPart2)
{
uint prev = rowsOrCols[0];
for (int i = 1; i < rowsOrCols.Length; i++)
{
uint cur = rowsOrCols[i];
uint xorValue = cur ^ prev;
if (xorValue == 0 || BitOperations.IsPow2(xorValue))
{
bool hasOneDiff = xorValue != 0;
if (VerifyCandidateMirror(rowsOrCols, i, ref hasOneDiff))
{
if (!hasOneDiff)
{
part1 += i * multiplier;
hasSolvedPart1 = true;
if (hasSolvedPart2)
return;
}
else
{
part2 += i * multiplier;
hasSolvedPart2 = true;
if (hasSolvedPart1)
return;
}
}
}
prev = cur;
}
}
private static bool VerifyCandidateMirror(Span<uint> rowsOrCols, int i, ref bool hasOneDiff)
{
int maxJ = Math.Min(i, rowsOrCols.Length - i);
for (int j = 1; j < maxJ; j++)
{
uint pairXor = rowsOrCols[i - 1 - j] ^ rowsOrCols[i + j];
if (pairXor != 0)
{
if (hasOneDiff || !BitOperations.IsPow2(pairXor))
return false;
hasOneDiff = true;
}
}
return true;
}
}