-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathmain.cpp
More file actions
128 lines (120 loc) · 3.69 KB
/
main.cpp
File metadata and controls
128 lines (120 loc) · 3.69 KB
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
#include <stdio.h>
#include "AoC64.h"
#if __has_include("input.h")
#include "input.h" // user’s private input (gitignored)
#else
#include "input.example.h" // checked-in sample input
#endif
// based on https://github.com/FransFaase/AdventOfCode2021/blob/main/src/day18_4sol.cpp
constexpr int8_t MAX_TREE_SIZE = 32;
struct Tree {
int16_t v[MAX_TREE_SIZE];
Tree() { clear(); }
void clear() { for (int8_t i = 1; i < MAX_TREE_SIZE; i++) v[i] = -1; }
};
void parse(const char * &s, int16_t *v, int8_t d) {
if (*s == '[') {
s++;
parse(s, v - d, d/2);
s++;
parse(s, v + d, d/2);
s++;
} else
*v = *s++ - '0';
}
void add(Tree &l, Tree &r, Tree &sum) {
sum.clear();
// temporarily use a 2x larger tree
int16_t isum[2*MAX_TREE_SIZE];
isum[MAX_TREE_SIZE] = -1;
for (int8_t i = 1; i < MAX_TREE_SIZE; ++i) {
isum[i] = l.v[i];
isum[MAX_TREE_SIZE + i] = r.v[i];
}
// make a combined tree
for (int8_t i = 0; i < 2*MAX_TREE_SIZE; ++i) {
if ((i & 3) == 1 && isum[i] != -1) {
for (int8_t j = i - 1; j > 0; --j)
if (isum[j] != -1) {
isum[j] += isum[i];
break;
}
isum[i++] = -1;
isum[i++] = 0;
for (int8_t j = i + 1; j < 2*MAX_TREE_SIZE; j++)
if (isum[j] != -1) {
isum[j] += isum[i];
break;
}
isum[i] = -1;
}
}
// shrink it
for (int8_t i = 1; i < MAX_TREE_SIZE; i++) {
sum.v[i] = isum[2*i];
}
for (int8_t i = 1; i < MAX_TREE_SIZE;) {
if (sum.v[i] < 10)
i++;
else {
int16_t n1 = sum.v[i] / 2;
int16_t n2 = sum.v[i] - n1;
if ((i & 1) == 1) {
int8_t p_i;
for (p_i = i - 1; p_i > 0; --p_i)
if (sum.v[p_i] != -1) {
sum.v[p_i] += n1;
break;
}
int8_t n_i;
for (n_i = i + 1; n_i < MAX_TREE_SIZE; ++n_i)
if (sum.v[n_i] != -1) {
sum.v[n_i] += n2;
break;
}
sum.v[i] = 0;
i = p_i > 0 ? p_i : n_i;
} else {
sum.v[i] = -1;
int8_t o;
if ((i & 3) == 0) o = 2;
else if ((i & 7) == 0) o = 4;
else if ((i & 15) == 0) o = 8;
else o = 1;
sum.v[i] = -1;
sum.v[i - o] = n1;
sum.v[i + o] = n2;
i -= o;
}
}
}
}
inline int16_t magnitude(int16_t *v, int8_t d) {
return *v == -1 ? 3 * magnitude(v - d, d/2) + 2 * magnitude(v + d, d/2) : *v;
}
int main(void) {
init(18);
Tree numbers[n_input], result;
parse(input[0], numbers[0].v + MAX_TREE_SIZE/2, 8);
Tree sum = numbers[0];
for (int8_t i = 1; i < n_input; ++i) {
tick(i & (uint8_t)7);
parse(input[i], numbers[i].v + MAX_TREE_SIZE/2, 8);
add(sum, numbers[i], result);
sum = result;
}
printf("part 1: %d\n", magnitude(sum.v + MAX_TREE_SIZE/2, 8));
int16_t largest = 0;
int16_t t = 0;
for (int8_t i = 0; i < n_input; ++i) {
for (int8_t j = 0; j < n_input; ++j) {
tick((++t >> 3) & (uint8_t)7);
add(numbers[i], numbers[j], result);
int16_t m = magnitude(result.v + MAX_TREE_SIZE/2, 8);
if (m > largest)
largest = m;
}
}
printf("part 2: %d\n", largest);
finish();
}