aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMarvin Borner2020-12-11 17:36:41 +0100
committerMarvin Borner2020-12-11 17:36:41 +0100
commit51a1115dc9f44286c6c19d48ce5261147792e8c4 (patch)
tree21ed6f7cb0e578ee09f40009889570b1ae84a09f
parentde7fae88afbcab153a8dcc749d0ece099057b77a (diff)
B
-rw-r--r--2020/11/Makefile2
-rw-r--r--2020/11/solve.c137
2 files changed, 127 insertions, 12 deletions
diff --git a/2020/11/Makefile b/2020/11/Makefile
index 7f68e56..58d64e6 100644
--- a/2020/11/Makefile
+++ b/2020/11/Makefile
@@ -1,7 +1,7 @@
.PHONY: solve.c
solve.o: solve.c
- @gcc $+ -o $@ -lm
+ @gcc -Ofast $+ -o $@ -lm
clean:
@rm -f *.o
diff --git a/2020/11/solve.c b/2020/11/solve.c
index 6fd8691..2b613cf 100644
--- a/2020/11/solve.c
+++ b/2020/11/solve.c
@@ -9,15 +9,21 @@
#define Y_MAX 97
// Quite magical, isn't it?
+#define TOP_LEFT(x, y, p) (x != 0 && y != 0 ? *(p - X_MAX - 1) : 0)
+#define TOP_CENTER(x, y, p) (y != 0 ? *(p - X_MAX) : 0)
+#define TOP_RIGHT(x, y, p) (x != X_MAX && y != 0 ? *(p - X_MAX + 1) : 0)
+#define LEFT(x, y, p) (x != 0 ? *(p - 1) : 0)
+#define RIGHT(x, y, p) (x != X_MAX ? *(p + 1) : 0)
+#define BOTTOM_LEFT(x, y, p) (x != 0 && y != Y_MAX ? *(p + X_MAX - 1) : 0)
+#define BOTTOM_CENTER(x, y, p) (y != Y_MAX ? *(p + X_MAX) : 0)
+#define BOTTOM_RIGHT(x, y, p) (x != X_MAX && y != Y_MAX ? *(p + X_MAX + 1) : 0)
+
#define ADJACENT(x, y, p) \
- (x != 0 && y != 0 ? *(p - X_MAX - 1) == '#' : 0) + (y != 0 ? *(p - X_MAX) == '#' : 0) + \
- (x != X_MAX && y != 0 ? *(p - X_MAX + 1) == '#' : 0) + \
- (x != 0 ? *(p - 1) == '#' : 0) + (x != X_MAX ? *(p + 1) == '#' : 0) + \
- (x != 0 && y != Y_MAX ? *(p + X_MAX - 1) == '#' : 0) + \
- (y != Y_MAX ? *(p + X_MAX) == '#' : 0) + \
- (x != X_MAX && y != Y_MAX ? *(p + X_MAX + 1) == '#' : 0);
-
-int count(char *buf1, char *buf2)
+ ((TOP_LEFT(x, y, p) == '#') + (TOP_CENTER(x, y, p) == '#') + (TOP_RIGHT(x, y, p) == '#') + \
+ (LEFT(x, y, p) == '#') + (RIGHT(x, y, p) == '#') + (BOTTOM_LEFT(x, y, p) == '#') + \
+ (BOTTOM_CENTER(x, y, p) == '#') + (BOTTOM_RIGHT(x, y, p) == '#'))
+
+int count_one(char *buf1, char *buf2)
{
int x = 0, y = 0;
for (char *p = buf1; *p; p++) {
@@ -48,7 +54,111 @@ int count(char *buf1, char *buf2)
}
if (memcmp(buf1, buf2, SIZE)) {
- return count(buf2, buf1); // Swap buffers
+ return count_one(buf2, buf1); // Swap buffers
+ } else {
+ int cnt = 0;
+ for (char *p = buf1; *p; p++) {
+ if (*p == '#')
+ cnt++;
+ }
+ return cnt;
+ }
+}
+
+#define TL (1 << 0)
+#define TC (1 << 1)
+#define TR (1 << 2)
+#define R (1 << 3)
+#define BR (1 << 4)
+#define BC (1 << 5)
+#define BL (1 << 6)
+#define L (1 << 7)
+#define SET(d) \
+ { \
+ if (type == '#' || type == 'L') \
+ found |= d; \
+ if (type == '#') \
+ cnt++; \
+ }
+int count_two(char *buf1, char *buf2)
+{
+ int x = 0, y = 0;
+ for (char *p = buf1; *p; p++) {
+ if (*p == '.') {
+ x++;
+ buf2[p - buf1] = '.';
+ continue;
+ }
+
+ if (*p == '\n') {
+ x = 0;
+ y++;
+ buf2[p - buf1] = '\n';
+ continue;
+ }
+
+ int i = 0, found = 0, cnt = 0;
+ char type;
+ while (found != 0xff) {
+ // Handle overflows (could be combined with the checks below)
+ if (x - i <= 0 || y - i <= 0)
+ found |= TL;
+ if (y - i <= 0)
+ found |= TC;
+ if (x + i >= X_MAX - 2 || y - i <= 0)
+ found |= TR;
+ if (x + i >= X_MAX - 2)
+ found |= R;
+ if (x + i >= X_MAX - 2 || y + i >= Y_MAX)
+ found |= BR;
+ if (y + i >= Y_MAX)
+ found |= BC;
+ if (x - i <= 0 || y + i >= Y_MAX)
+ found |= BL;
+ if (x - i <= 0)
+ found |= L;
+
+ // Find next seat
+ if ((found & TL) == 0 &&
+ (type = TOP_LEFT(x - i, y - i, &buf1[(y - i) * X_MAX + (x - i)])))
+ SET(TL);
+ if ((found & TC) == 0 &&
+ (type = TOP_CENTER(x, y - i, &buf1[(y - i) * X_MAX + x])))
+ SET(TC);
+ if ((found & TR) == 0 &&
+ (type = TOP_RIGHT(x + i, y - i, &buf1[(y - i) * X_MAX + (x + i)])))
+ SET(TR);
+ if ((found & R) == 0 &&
+ (type = RIGHT(x + i, y, &buf1[y * X_MAX + (x + i)])))
+ SET(R);
+ if ((found & BR) == 0 &&
+ (type = BOTTOM_RIGHT(x + i, y + i, &buf1[(y + i) * X_MAX + (x + i)])))
+ SET(BR);
+ if ((found & BC) == 0 &&
+ (type = BOTTOM_CENTER(x, y + i, &buf1[(y + i) * X_MAX + x])))
+ SET(BC);
+ if ((found & BL) == 0 &&
+ (type = BOTTOM_LEFT(x - i, y + i, &buf1[(y + i) * X_MAX + (x - i)])))
+ SET(BL);
+ if ((found & L) == 0 && (type = LEFT(x - i, y, &buf1[y * X_MAX + (x - i)])))
+ SET(L);
+
+ i++;
+ }
+
+ if (*p == 'L' && !cnt) {
+ buf2[p - buf1] = '#';
+ } else if (*p == '#' && cnt >= 5) {
+ buf2[p - buf1] = 'L';
+ } else {
+ buf2[p - buf1] = *p;
+ }
+
+ x++;
+ }
+
+ if (memcmp(buf1, buf2, SIZE)) {
+ return count_two(buf2, buf1); // Swap buffers
} else {
int cnt = 0;
for (char *p = buf1; *p; p++) {
@@ -68,14 +178,19 @@ int part_one(char *data)
char buf2[SIZE] = { 0 };
memcpy(buf2, data, SIZE);
- return count(buf1, buf2);
+ return count_one(buf1, buf2);
}
int part_two(char *data)
{
int res = 0;
- return res;
+ char buf1[SIZE] = { 0 };
+ memcpy(buf1, data, SIZE);
+ char buf2[SIZE] = { 0 };
+ memcpy(buf2, data, SIZE);
+
+ return count_two(buf1, buf2);
}
int main(int argc, char *argv[])