banan-os/userspace/aoc2024/day15/main.cpp

327 lines
6.5 KiB
C++

#include <BAN/Vector.h>
#include <inttypes.h>
#include <stdio.h>
using i8 = int8_t;
using i16 = int16_t;
using i32 = int32_t;
using i64 = int64_t;
using isize = ssize_t;
using u8 = uint8_t;
using u16 = uint16_t;
using u32 = uint32_t;
using u64 = uint64_t;
using usize = size_t;
struct Grid2D
{
usize width { 0 };
usize height { 0 };
BAN::Vector<char> data;
inline char get(usize x, usize y) const
{
ASSERT(x < width && y < height);
return data[y * width + x];
}
inline char& get(usize x, usize y)
{
ASSERT(x < width && y < height);
return data[y * width + x];
}
};
struct Position
{
i32 x, y;
};
enum class Dir
{
Up, Down,
Left, Right,
};
struct ParseInputResult
{
Grid2D grid;
Position robot;
BAN::Vector<Dir> moves;
};
static ParseInputResult parse_input(FILE* fp, bool wide)
{
Grid2D grid;
char buffer[1024] {};
while (fgets(buffer, sizeof(buffer), fp))
{
const usize len = strlen(buffer);
if (len == 0 || buffer[0] == '\n')
break;
if (grid.data.empty())
grid.width = len - 1;
grid.height++;
ASSERT(buffer[grid.width] == '\n');
if (grid.data.capacity() < grid.height * grid.width)
MUST(grid.data.reserve(2 * grid.height * grid.width));
MUST(grid.data.resize(grid.height * grid.width));
memcpy(&grid.data[(grid.height - 1) * grid.width], buffer, grid.width);
}
if (wide)
{
BAN::Vector<char> wide_data;
MUST(wide_data.resize(grid.data.size() * 2));
for (usize i = 0; i < grid.data.size(); i++)
{
char l = 0, r = 0;
switch (grid.data[i])
{
case '#': l = '#'; r = '#'; break;
case 'O': l = '['; r = ']'; break;
case '.': l = '.'; r = '.'; break;
case '@': l = '@'; r = '.'; break;
}
wide_data[i * 2 + 0] = l;
wide_data[i * 2 + 1] = r;
}
grid.data = BAN::move(wide_data);
grid.width *= 2;
}
constexpr auto char_to_dir =
[](char ch) -> Dir
{
switch (ch)
{
case '^': return Dir::Up;
case 'v': return Dir::Down;
case '<': return Dir::Left;
case '>': return Dir::Right;
}
ASSERT_NOT_REACHED();
};
BAN::Vector<Dir> moves;
for (;;)
{
usize nread = fread(buffer, 1, sizeof(buffer), fp);
if (nread == 0)
break;
MUST(moves.reserve(moves.size() + nread));
for (usize i = 0; i < nread; i++)
if (buffer[i] != '\n')
MUST(moves.push_back(char_to_dir(buffer[i])));
}
Position robot;
for (u32 y = 0; y < grid.height; y++)
{
for (u32 x = 0; x < grid.width; x++)
{
if (grid.data[y * grid.width + x] != '@')
continue;
grid.data[y * grid.width + x] = '.';
robot = { (i32)x, (i32)y };
}
}
return ParseInputResult {
.grid = BAN::move(grid),
.robot = robot,
.moves = BAN::move(moves),
};
}
i64 part1(FILE* fp)
{
auto [grid, robot, moves] = parse_input(fp, false);
for (usize i = 0; i < moves.size(); i++)
{
#if 0
printf("\e[H");
for (usize y = 0; y < grid.height; y++) {
for (usize x = 0; x < grid.width; x++)
printf("%c ", grid.get(x, y));
printf("\n");
}
printf("\e[%u;%uH\e[31m@\e[m", robot.y + 1, robot.x * 2 + 1);
printf("\e[%zuH%zu/%zu\n", grid.height + 1, i, moves.size());
getchar();
#endif
i32 vx = 0, vy = 0;
switch (moves[i])
{
case Dir::Up: vx = 0; vy = -1; break;
case Dir::Down: vx = 0; vy = 1; break;
case Dir::Left: vx = -1; vy = 0; break;
case Dir::Right: vx = 1; vy = 0; break;
}
Position empty { -1, -1 };
for (usize j = 1;; j++)
{
const auto pos = Position {
(i32)(robot.x + j * vx),
(i32)(robot.y + j * vy),
};
const char ch = grid.get(pos.x, pos.y);
if (ch == '#')
break;
if (ch == 'O')
continue;
ASSERT(ch == '.');
empty = pos;
break;
}
if (empty.x == -1 || empty.y == -1)
continue;
while (empty.x != robot.x || empty.y != robot.y)
{
const auto pos = Position {
(i32)(empty.x - vx),
(i32)(empty.y - vy),
};
grid.get(empty.x, empty.y) = grid.get(pos.x, pos.y);
empty = pos;
}
grid.get(robot.x, robot.y) = '.';
robot.x += vx;
robot.y += vy;
}
i64 result = 0;
for (usize y = 0; y < grid.height; y++)
for (usize x = 0; x < grid.width; x++)
if (grid.get(x, y) == 'O')
result += 100 * y + x;
return result;
}
static bool can_move(const Grid2D& grid, Position pos, Position dir)
{
const char ch = grid.get(pos.x, pos.y);
if (ch == '.')
return true;
if (ch == '#')
return false;
if (dir.x)
return can_move(grid, { pos.x + dir.x, pos.y }, dir);
ASSERT(ch == '[' || ch == ']');
const i32 dir_x = (ch == '[') ? 1 : -1;
return can_move(grid, { pos.x, pos.y + dir.y }, dir)
&& can_move(grid, { pos.x + dir_x, pos.y + dir.y }, dir);
}
static void do_move(Grid2D& grid, Position pos, Position dir)
{
const char ch = grid.get(pos.x, pos.y);
ASSERT(ch != '#');
if (ch == '.')
;
else if (dir.x)
do_move(grid, { pos.x + dir.x, pos.y }, dir);
else
{
ASSERT(ch == '[' || ch == ']');
const i32 dir_x = (ch == '[') ? 1 : -1;
do_move(grid, { pos.x, pos.y + dir.y }, dir);
do_move(grid, { pos.x + dir_x, pos.y + dir.y }, dir);
}
grid.get(pos.x, pos.y) = grid.get(pos.x - dir.x, pos.y - dir.y);
grid.get(pos.x - dir.x, pos.y - dir.y) = '.';
//grid.get(pos.x + dir.x, pos.y + dir.y) = grid.get(pos.x, pos.y);
//grid.get(pos.x, pos.y) = '.';
}
i64 part2(FILE* fp)
{
auto [grid, robot, moves] = parse_input(fp, true);
for (usize i = 0; i < moves.size(); i++)
{
#if 0
printf("\e[H");
for (usize y = 0; y < grid.height; y++) {
for (usize x = 0; x < grid.width; x++)
printf("%c", grid.get(x, y));
printf("\n");
}
printf("\e[%u;%uH\e[31m@\e[m", robot.y + 1, robot.x + 1);
printf("\e[%zuH%zu/%zu\n", grid.height + 1, i, moves.size());
getchar();
#endif
Position dir {};
switch (moves[i])
{
case Dir::Up: dir = { 0, -1 }; break;
case Dir::Down: dir = { 0, 1 }; break;
case Dir::Left: dir = { -1, 0 }; break;
case Dir::Right: dir = { 1, 0 }; break;
}
const auto next = Position {
robot.x + dir.x,
robot.y + dir.y,
};
if (!can_move(grid, next, dir))
continue;
do_move(grid, next, dir);
robot = next;
}
i64 result = 0;
for (usize y = 0; y < grid.height; y++)
for (usize x = 0; x < grid.width; x++)
if (grid.get(x, y) == '[')
result += 100 * y + x;
return result;
}
int main(int argc, char** argv)
{
const char* file_path = "/usr/share/aoc2024/day15_input.txt";
if (argc >= 2)
file_path = argv[1];
FILE* fp = fopen(file_path, "r");
if (fp == nullptr)
{
perror("fopen");
return 1;
}
printf("part1: %" PRId64 "\n", part1(fp));
fseek(fp, 0, SEEK_SET);
printf("part2: %" PRId64 "\n", part2(fp));
fclose(fp);
}