Initial commit

This commit is contained in:
2024-11-14 19:16:08 -05:00
commit 4f0091f23f
18 changed files with 2277 additions and 0 deletions

BIN
d1.bin Executable file

Binary file not shown.

7
d1/d1-p2-test.input Normal file
View File

@@ -0,0 +1,7 @@
two1nine
eightwothree
abcone2threexyz
xtwone3four
4nineeightseven2
zoneight234
7pqrstsixteen

1000
d1/d1.input Normal file

File diff suppressed because it is too large Load Diff

393
d1/main.odin Normal file
View File

@@ -0,0 +1,393 @@
package d1;
import "core:os"
import "core:fmt"
import "core:testing"
main_attempt_1 :: proc () {
if len(os.args) < 2 {
fmt.printf("Usage: ./d1 FILENAME\n")
os.exit(-1)
}
fh: os.Handle
eno: os.Errno
fmt.printf("Reached line 13!\n")
if fh, eno = os.open(os.args[1]); eno != os.ERROR_NONE {
fmt.printf("Cannot open %s: %s", os.args[1], os.get_last_error_string())
os.exit(-1)
}
defer os.close(fh)
buffer: []u8 = make([]u8, 1024)
defer delete(buffer)
idx := -1
n_read := 0
digits: []u8 = make([]u8, 2)
defer delete(digits)
digits[0] = 'A'
digits[1] = 'A'
total := 0
line_ctr := 0
outer_loop: for {
n_read, eno = os.read(fh, buffer)
if eno != os.ERROR_NONE {
break outer_loop;
}
if n_read <= 0 {
if digits[0] != 'A' {
fmt.printf("DEBUG: Line %d: First digit: %c\n", line_ctr, digits[0])
total += 10 * int(digits[0] - '0')
if digits[1] == 'A' {
fmt.printf("DEBUG: Line %d: Repeating first digit\n", line_ctr)
total += int(digits[0] - '0')
} else {
fmt.printf("DEBUG: Line %d: Second digit: %c\n", line_ctr, digits[1])
total += int(digits[1] - '0')
}
}
break outer_loop;
}
for idx = 0; idx < n_read; idx += 1 {
switch buffer[idx] {
case '\n':
assert(digits[0] != 'A')
fmt.printf("DEBUG: Line %d: First digit: %c\n", line_ctr, digits[0])
total += 10 * int(digits[0] - '0')
if digits[1] == 'A' {
fmt.printf("DEBUG: Line %d: Repeating first digit\n", line_ctr)
total += int(digits[0] - '0')
} else {
fmt.printf("DEBUG: Line %d: Second digit: %c\n", line_ctr, digits[0])
total += int(digits[1] - '0')
}
digits[0] = 'A'
digits[1] = 'A'
line_ctr += 1
case '0'..='9':
if digits[0] == 'A' {
digits[0] = buffer[idx]
} else {
digits[1] = buffer[idx]
}
}
}
}
fmt.printf("total: %d\n", total)
}
line_no := 0
fmt_debug :: proc(msg: string, args: ..any) {
fmt.printf("DEBUG: L%d: ", line_no)
fmt.printf(msg, ..args)
}
digit_int :: proc(digit: u8) -> int {
return int(digit - '0')
}
add_digits :: proc(digits: []u8) -> (x: int) {
x = 0
assert(len(digits) == 2)
assert(digits[0] != 'A')
fmt_debug("First digit: %c\n", digits[0])
x += 10 * digit_int(digits[0])
if digits[1] == 'A' {
fmt_debug("No second digit, reusing first digit: %c\n", digits[0])
x += digit_int(digits[0])
} else {
fmt_debug("Second digit: %c\n", digits[1])
x += digit_int(digits[1])
}
fmt_debug("Total value: %d\n", x)
return
}
main_attempt_2 :: proc () {
if len(os.args) < 2 {
fmt.printf("Usage: ./d1 FILENAME\n")
os.exit(-1)
}
fh: os.Handle
eno: os.Errno
fmt.printf("Reached line 13!\n")
if fh, eno = os.open(os.args[1]); eno != os.ERROR_NONE {
fmt.printf("Cannot open %s: %s", os.args[1], os.get_last_error_string())
os.exit(-1)
}
defer os.close(fh)
buffer: []u8 = make([]u8, 1024)
defer delete(buffer)
idx := -1
n_read := 0
digits: []u8 = make([]u8, 2)
defer delete(digits)
digits[0] = 'A'
digits[1] = 'A'
total := 0
outer_loop: for {
n_read, eno = os.read(fh, buffer)
if eno != os.ERROR_NONE {
break outer_loop;
}
if n_read <= 0 {
if digits[0] != 'A' {
total += add_digits(digits)
fmt_debug("Total: %d\n", total)
}
break outer_loop;
}
for idx = 0; idx < n_read; idx += 1 {
switch buffer[idx] {
case '\n':
total += add_digits(digits)
fmt_debug("Total: %d\n", total)
digits[0] = 'A'
digits[1] = 'A'
line_no += 1
case '0'..='9':
if digits[0] == 'A' {
digits[0] = buffer[idx]
} else {
digits[1] = buffer[idx]
}
}
}
}
fmt.printf("Total: %d\n", total)
}
WORD_LITERALS := []string{"zero", "one", "two", "three", "four", "five", "six", "seven", "eight", "nine"};
string_at :: proc(buffer: []u8, prefix: string, start: int = 0) -> bool {
for idx := 0; idx < len(prefix); idx += 1 {
buffer_idx := start + idx;
if buffer_idx >= len(buffer) {
return false;
}
if prefix[idx] != buffer[buffer_idx] {
return false;
}
}
return true;
}
starts_with :: proc(buffer: []u8, prefix: string) -> bool {
return string_at(buffer, prefix, 0);
}
looking_at_numword :: proc(buffer: []u8, start: int = 0) -> (bool, int, int) {
assert(start < len(buffer))
for idx := 0; idx < len(WORD_LITERALS); idx += 1 {
if string_at(buffer, WORD_LITERALS[idx], start) {
return true, idx, start + len(WORD_LITERALS[idx]);
}
}
return false, 0, 0;
};
looking_at_num :: proc(buffer: []u8, start: int = 0) -> (bool, int, int) {
assert(0 <= start);
switch buffer[start] {
case '0'..='9':
return true, digit_int(buffer[start]), start + 1;
case 'z','o','t','f','s','e','n':
return looking_at_numword(buffer, start)
}
return false, 0, 0;
}
@(test)
test_looking_at_num :: proc(t: ^testing.T) {
input: []u8;
num: int;
ok: bool;
next_num: int;
input = transmute([]u8) string("fivethreeonezblqnsfk1");
next_num = 5;
ok, num, _ = looking_at_num(input, 0);
testing.expect(t, ok, "Failure: not looking at a number or digit");
testing.expect(t, num == next_num, fmt.tprint("%d != %d\n", next_num, num));
next_num = 3;
ok, num, _ = looking_at_num(input, 4);
testing.expect(t, ok, "Failure: not looking at a number or digit");
testing.expect(t, num == next_num, fmt.tprintf("%d != %d\n", next_num, num));
input = transmute([]u8) string("two74119onebtqgnine");
next_num = 2;
ok, num, _ = looking_at_num(input, 0);
testing.expect(t, ok, "Failure: not looking at a number or digit");
testing.expect(t, num == next_num, fmt.tprintf("%d != %d\n", next_num, num));
next_num = 1;
ok, num, _ = looking_at_num(input, 6);
testing.expect(t, ok, "Failure: not looking at a number or digit");
testing.expect(t, num == next_num, fmt.tprintf("%d != %d\n", next_num, num));
// input = transmute([]u8) string("jrjh5vsrxbhsfour3");
// input = transmute([]u8) string("tn5eightfncnzcdtthree8");
// input = transmute([]u8) string("kpmrk5flx");
// input = transmute([]u8) string("fkxxqxdfsixgthreepvzjxrkcfk6twofour");
// input = transmute([]u8) string("dqbx6six5twoone");
// input = transmute([]u8) string("glmsckj2bvmts1spctnjrtqhmbxzq");
// input = transmute([]u8) string("7sixthreerzmpbffcx");
// input = transmute([]u8) string("zhss9gfxfgmrmzthreefivevpkljfourtwoeight");
}
slice_next_line :: proc(buffer: []u8, start: uint) -> (int, []u8) {
assert(start < len(buffer))
for idx := 0; idx < len(buffer); idx += 1 {
if buffer[idx] == '\n' {
return idx, buffer[start:idx]
}
}
return -1, buffer[start:]
}
BUFFER_SIZE :: 1024
process_line :: proc(buffer: []u8, start: int) -> (int, int) {
assert(0 <= start && start < len(buffer))
a, b := -1, -1
index := start
for index < len(buffer) {
ok, x, _ := looking_at_num(buffer, index)
if !ok {
index += 1;
} else if a == -1 {
// Save first digit
a = 10 * int(x);
index += len(WORD_LITERALS[x])
} else {
// Save second digit
b += int(x)
index += len(WORD_LITERALS[x])
}
}
assert(a > -1);
if b == -1 {
return index, 10 * a + a;
}
return index, 10 * a + b;
}
os_blksize :: proc() -> i32 {
info := os.OS_Stat{};
_ = os._unix_stat("/", &info);
return info.block_size;
}
scan_to_next :: proc(fh: os.Handle, chr: u8, pos: uint = 0) -> (success: bool, char_pos: uint) {
buf := make([]u8, os_blksize());
file_pos := pos;
char_pos = 0
success = false
for {
nr, errno := os.read_at(fh, buf, i64(file_pos));
if errno == os.EINVAL {
break;
} else if errno != os.ERROR_NONE {
fmt.printf("Fatal error: %s\n", os.get_last_error_string());
os.exit(int(errno));
}
for idx : uint = 0; idx < len(buf); idx += 1 {
if buf[idx] == '\n' {
success = true;
char_pos = file_pos + idx;
break;
}
}
if success || nr < len(buf) {
break;
}
file_pos += len(buf);
}
os.seek(fh, i64(pos), os.SEEK_SET)
return;
}
die :: proc (eno: os.Errno) {
fmt.printf("Fatal error: %s\n", os.get_last_error_string());
os.exit(int(eno));
}
main_attempt_3 :: proc() {
if len(os.args) < 2 {
fmt.printf("Usage: ./d1 FILENAME\n")
os.exit(-1)
}
fh: os.Handle
eno: os.Errno
if fh, eno = os.open(os.args[1]); eno != os.ERROR_NONE {
die(eno);
}
defer os.close(fh);
buffer: []u8 = make([]u8, BUFFER_SIZE);
defer delete(buffer);
nr: int;
num: int;
ok: bool;
total := 0
digits: []int = {-1, -1};
for {
nr, eno = os.read(fh, buffer);
if eno != os.ERROR_NONE {
die(eno);
}
assert(0 < nr && nr <= BUFFER_SIZE);
for idx := 0; idx < nr; idx += 1 {
if buffer[idx] == '\n' {
if digits[0] > -1 {
total += 10 * digits[0];
if digits[1] < 0 {
fmt_debug("Number for line: %d%d\n", digits[0], digits[0]);
total += digits[0];
} else {
fmt_debug("Number for line: %d%d\n", digits[0], digits[1]);
total += digits[1];
}
digits[0] = -1;
digits[1] = -1;
} else {
fmt_debug("Hit newline without setting digits\n");
}
line_no += 1
continue;
}
new_idx: int;
ok, num, new_idx = looking_at_num(buffer, idx);
if ok && digits[0] == -1 {
fmt_debug("Found first number: %d\n", num);
digits[0] = num;
idx = new_idx - 1;
} else if ok {
fmt_debug("Found additional number: %d\n", num);
digits[1] = num;
idx = new_idx - 1;
}
}
if nr != BUFFER_SIZE {
break;
}
}
// off by one, as usual
if digits[0] != -1 {
total += 10 * digits[0];
if digits[1] < 0 {
fmt_debug("Number for line: %d%d\n", digits[0], digits[0]);
total += digits[0];
} else {
fmt_debug("Number for line: %d%d\n", digits[0], digits[1]);
total += digits[1];
}
}
fmt.printf("Total: %d\n", total);
}
main :: proc() {
main_attempt_3()
}

BIN
d2.bin Executable file

Binary file not shown.

100
d2/d2.input Normal file
View File

@@ -0,0 +1,100 @@
Game 1: 1 red, 3 blue, 11 green; 1 blue, 5 red; 3 blue, 5 green, 13 red; 6 red, 1 blue, 4 green; 16 red, 12 green
Game 2: 3 red, 13 blue, 5 green; 14 green, 14 blue; 9 blue, 10 green, 3 red; 2 green, 5 blue; 11 green, 3 blue, 3 red; 16 blue, 2 red, 9 green
Game 3: 17 blue, 5 red; 3 red, 11 green, 17 blue; 1 red, 6 blue, 9 green; 3 blue, 11 green, 1 red; 3 green, 10 red, 11 blue; 12 red, 3 green, 15 blue
Game 4: 14 green, 14 red, 1 blue; 15 red, 13 green, 1 blue; 6 green, 15 red; 7 green
Game 5: 3 green, 1 blue, 3 red; 6 red, 2 green, 2 blue; 12 red, 3 green, 1 blue; 2 green, 9 red; 1 blue; 2 blue, 10 red
Game 6: 5 blue, 5 green; 4 blue, 1 red, 10 green; 16 green, 1 red, 6 blue; 1 red, 1 blue, 13 green; 1 red, 5 blue, 7 green; 14 green, 17 blue
Game 7: 1 green, 8 blue, 4 red; 1 green, 4 blue, 4 red; 6 blue, 4 red, 4 green; 1 red, 8 green
Game 8: 2 red, 5 blue, 1 green; 1 blue, 4 red, 8 green; 6 blue, 12 green, 6 red; 3 blue, 5 red; 8 red, 2 blue, 13 green; 5 green, 4 red, 3 blue
Game 9: 11 red; 1 green, 2 red, 2 blue; 1 blue, 2 green, 9 red; 4 red, 2 green, 2 blue; 1 blue, 2 green; 1 blue, 9 red, 2 green
Game 10: 9 red, 4 green; 1 blue, 3 red, 7 green; 3 green, 1 red, 1 blue; 7 green, 4 red, 1 blue; 1 blue, 5 green, 10 red; 1 red, 5 green
Game 11: 2 blue, 4 red, 3 green; 1 blue, 7 red; 4 green, 7 red, 1 blue; 3 blue, 6 green, 4 red; 3 red, 1 green, 3 blue
Game 12: 1 green, 6 red, 5 blue; 3 green, 2 red, 4 blue; 3 green, 1 red, 3 blue
Game 13: 6 green, 1 red, 9 blue; 11 red, 4 blue, 12 green; 6 green, 9 red, 19 blue; 2 green, 6 blue; 10 green, 1 red, 16 blue; 4 green, 14 blue
Game 14: 7 blue, 2 red; 1 green, 2 red, 19 blue; 12 blue, 6 green, 11 red
Game 15: 4 red, 4 green, 7 blue; 15 blue, 1 green, 8 red; 2 red, 10 green, 11 blue; 5 red, 4 blue, 6 green; 9 red, 8 blue, 3 green; 9 blue, 9 red
Game 16: 7 red, 2 blue, 19 green; 6 blue, 9 green; 8 green, 6 red, 19 blue; 11 green, 7 red, 1 blue; 9 blue, 3 red, 17 green
Game 17: 3 blue, 4 green, 5 red; 2 red, 4 green, 11 blue; 6 blue, 13 green; 3 blue, 12 green, 7 red
Game 18: 9 red, 6 blue, 7 green; 3 green, 3 blue, 5 red; 18 red, 6 blue, 4 green; 3 green, 10 red, 8 blue
Game 19: 3 red, 6 green; 1 red, 5 green, 4 blue; 3 red, 14 blue
Game 20: 2 green, 2 blue, 4 red; 14 red, 6 blue, 5 green; 1 blue, 5 red, 3 green; 10 red, 6 green, 6 blue
Game 21: 10 blue, 12 green, 3 red; 1 green, 14 red; 5 blue, 7 green; 12 blue, 1 red, 13 green; 7 red, 4 green
Game 22: 2 red, 1 blue; 1 red, 2 blue; 1 red, 1 green, 3 blue; 3 blue; 1 red; 1 green, 2 red
Game 23: 4 blue, 4 green, 1 red; 3 blue, 1 red, 6 green; 1 red, 1 blue
Game 24: 5 blue, 15 green, 13 red; 20 green, 13 blue, 6 red; 5 blue, 11 red, 16 green; 6 red, 5 blue, 13 green; 12 blue, 13 green, 3 red
Game 25: 10 blue, 17 red; 12 red, 16 blue, 3 green; 4 green, 12 blue, 10 red; 8 blue, 3 green, 10 red; 5 green, 2 red, 12 blue
Game 26: 11 red, 9 blue; 3 blue, 3 red, 3 green; 10 blue, 3 green, 4 red; 1 green, 4 blue, 9 red; 5 green, 1 red, 7 blue; 1 red, 3 blue, 3 green
Game 27: 1 green, 12 red, 4 blue; 5 red, 2 green, 1 blue; 3 green, 6 blue, 10 red; 1 green, 4 red, 3 blue
Game 28: 6 blue; 2 green; 2 green, 8 blue, 1 red; 2 green, 2 blue; 6 blue, 8 green; 9 green, 5 blue
Game 29: 1 green, 9 blue, 9 red; 13 green, 4 red, 9 blue; 3 green, 8 blue, 15 red; 15 green, 18 blue, 3 red; 16 green, 10 red; 16 green, 12 blue, 16 red
Game 30: 14 blue, 4 green, 1 red; 7 red, 14 blue; 2 blue, 4 red, 1 green
Game 31: 2 red, 14 green, 3 blue; 3 blue, 3 green, 4 red; 8 blue, 4 red, 1 green; 8 green, 3 blue; 10 blue, 1 red, 11 green; 13 green, 2 red, 3 blue
Game 32: 8 blue, 16 red; 2 green, 8 blue, 16 red; 16 blue, 4 green, 17 red; 2 red, 5 green, 4 blue
Game 33: 2 red, 2 green, 1 blue; 5 red, 1 blue; 8 green, 14 red
Game 34: 4 red, 4 green; 9 green; 1 blue, 16 green; 1 blue, 5 red, 9 green; 2 red, 15 green, 1 blue
Game 35: 1 green, 5 red; 1 green, 15 red, 13 blue; 2 red, 13 blue, 17 green; 9 blue, 3 red, 11 green; 7 green, 8 blue, 14 red
Game 36: 19 green; 3 green, 1 blue, 1 red; 1 green, 8 blue; 13 green, 5 red, 5 blue
Game 37: 12 red, 7 green, 3 blue; 12 blue, 10 red, 9 green; 17 green, 8 red, 13 blue; 9 blue, 9 green, 8 red; 4 red, 13 green, 13 blue; 15 green, 12 red, 14 blue
Game 38: 5 blue, 1 green, 20 red; 1 green, 13 red, 18 blue; 17 blue, 9 red, 10 green; 4 blue, 4 red, 12 green; 12 blue, 12 red, 6 green; 12 green, 13 red, 2 blue
Game 39: 7 blue, 6 red, 2 green; 6 blue, 1 red; 7 blue, 1 red
Game 40: 1 blue, 3 red; 15 blue, 1 green; 1 green, 16 red, 2 blue
Game 41: 2 blue, 4 green; 8 green, 3 red; 2 blue, 9 red, 4 green; 4 red, 3 blue, 10 green; 5 green, 3 blue, 2 red
Game 42: 7 green, 2 blue, 1 red; 8 green, 4 red; 5 blue, 1 red, 3 green
Game 43: 3 red, 1 blue; 1 blue, 2 green, 2 red; 1 red, 2 blue; 3 blue
Game 44: 3 green, 14 blue, 1 red; 16 blue, 5 red, 11 green; 12 green, 1 blue; 13 blue, 1 red; 5 blue, 2 red, 6 green; 3 blue, 5 red, 11 green
Game 45: 7 blue, 1 red; 1 red, 3 blue; 3 green, 14 blue, 1 red; 4 blue, 3 green, 1 red; 15 blue, 1 red, 3 green
Game 46: 15 red, 4 blue; 15 red, 11 blue, 3 green; 14 red, 2 green, 2 blue; 14 red, 8 blue, 3 green; 4 red, 1 blue
Game 47: 4 green, 2 blue, 3 red; 8 red, 2 green, 18 blue; 1 green, 17 blue, 1 red
Game 48: 2 green, 4 red, 2 blue; 15 blue, 16 red, 5 green; 14 blue, 2 green, 10 red; 3 green, 13 red, 6 blue; 8 green, 4 red, 12 blue; 15 red, 3 green, 9 blue
Game 49: 1 green, 6 red, 7 blue; 1 blue, 9 green, 9 red; 4 green, 8 red; 9 blue, 1 red, 14 green; 2 blue, 9 red
Game 50: 3 red, 10 blue, 14 green; 2 red, 9 blue, 7 green; 4 blue, 12 green; 1 red, 4 green, 5 blue
Game 51: 2 green, 6 blue; 1 green, 10 blue, 1 red; 3 blue, 2 green
Game 52: 1 green, 4 red, 1 blue; 3 red, 5 green, 4 blue; 1 blue, 3 red, 5 green; 1 red, 1 green, 1 blue; 12 green, 2 red, 4 blue; 10 blue, 7 green, 1 red
Game 53: 12 red, 1 blue; 8 red, 11 blue, 11 green; 8 red, 6 blue, 13 green; 11 blue, 11 red, 16 green; 6 red, 9 green, 4 blue
Game 54: 2 red, 8 blue, 15 green; 4 green, 3 blue, 6 red; 12 green, 13 blue, 4 red
Game 55: 1 green, 16 blue, 4 red; 3 red, 1 blue, 1 green; 12 red, 16 blue; 3 red
Game 56: 4 green; 1 red, 4 green; 2 red, 3 blue, 7 green; 2 red, 3 blue, 15 green
Game 57: 17 green; 1 green, 9 blue; 1 red, 1 green, 9 blue
Game 58: 3 green, 8 red, 7 blue; 4 green, 9 blue, 2 red; 1 red, 2 green, 11 blue; 8 blue, 4 green
Game 59: 6 green, 1 red; 4 blue, 6 green; 4 green, 5 blue
Game 60: 3 green, 5 blue, 1 red; 7 green, 5 blue, 16 red; 14 red, 1 green, 1 blue; 7 green, 2 blue; 13 red, 5 green, 5 blue
Game 61: 1 green, 2 blue, 2 red; 2 green; 6 red, 1 blue, 1 green
Game 62: 5 red, 8 blue, 1 green; 1 red, 1 blue; 2 green, 8 blue
Game 63: 2 red, 2 blue, 2 green; 9 blue, 7 green; 1 green, 4 blue; 18 green, 3 blue
Game 64: 13 green, 1 blue, 6 red; 13 green, 15 red, 8 blue; 5 green, 14 red, 4 blue; 2 green, 8 blue, 12 red; 1 blue, 5 red, 13 green; 7 blue, 8 green, 2 red
Game 65: 7 blue, 12 red, 6 green; 11 red, 8 green, 8 blue; 9 red, 7 green, 7 blue; 14 red, 2 blue, 17 green
Game 66: 2 green, 5 red; 7 red, 14 blue; 19 blue, 2 green; 7 blue, 4 green, 6 red
Game 67: 4 green, 17 red, 7 blue; 4 blue, 6 green; 7 green, 7 red, 12 blue; 2 red, 14 blue
Game 68: 1 red, 11 green, 4 blue; 17 blue, 1 red, 10 green; 3 red, 7 green, 1 blue; 7 green, 3 red, 6 blue; 2 red, 3 green; 2 green, 2 red, 4 blue
Game 69: 5 blue, 4 red; 3 red, 11 green, 1 blue; 6 green, 2 blue; 10 green, 4 red, 5 blue; 2 red, 11 green
Game 70: 16 red, 7 blue, 1 green; 14 red, 1 blue, 4 green; 4 red, 4 green; 7 blue, 5 red, 2 green
Game 71: 14 red, 2 blue, 13 green; 7 green, 5 red, 2 blue; 3 blue, 9 green, 11 red; 10 red, 4 blue, 1 green
Game 72: 1 green, 2 red, 6 blue; 4 green, 4 red, 9 blue; 6 green, 8 blue, 1 red; 5 red, 4 green, 9 blue; 15 blue, 2 green, 7 red; 10 blue, 2 green, 10 red
Game 73: 7 green, 6 red, 7 blue; 6 blue, 5 red, 8 green; 5 blue, 5 red
Game 74: 11 red, 1 blue; 2 green, 4 blue, 1 red; 1 green, 2 blue, 11 red; 9 red, 5 blue; 15 red, 10 blue; 9 red, 3 blue
Game 75: 1 blue, 6 red, 9 green; 5 red, 1 blue, 8 green; 2 green, 2 red, 1 blue; 7 red, 1 green; 3 green, 6 red, 2 blue; 1 green, 1 red
Game 76: 16 red, 3 blue, 9 green; 4 blue, 4 green; 5 blue, 1 green, 10 red; 6 blue, 13 red; 1 blue, 2 green, 8 red
Game 77: 4 red, 4 blue; 5 blue, 5 red; 6 red, 3 green
Game 78: 11 green, 1 red; 1 blue, 18 green, 1 red; 6 green, 5 red, 2 blue; 6 red, 1 blue, 15 green; 5 green, 5 red
Game 79: 2 red, 3 green, 13 blue; 7 blue, 5 green; 4 blue, 2 red, 6 green; 6 green, 15 blue
Game 80: 9 green, 2 blue, 1 red; 8 green, 1 red; 1 blue, 7 green; 2 green, 1 blue; 3 green; 5 green, 1 red, 2 blue
Game 81: 2 blue, 8 green, 1 red; 3 green, 1 blue; 6 blue, 1 green; 3 blue, 3 green, 1 red; 2 green, 8 blue; 1 red, 8 blue, 2 green
Game 82: 5 blue, 4 red, 1 green; 9 red, 12 green, 8 blue; 9 red, 6 green, 15 blue; 8 blue, 10 red, 6 green
Game 83: 2 green, 7 blue, 4 red; 2 blue, 11 red, 9 green; 7 red, 7 green, 6 blue; 12 blue, 4 red, 11 green; 11 green, 7 blue; 7 green, 5 red, 2 blue
Game 84: 9 red, 1 blue, 7 green; 5 red, 5 green; 4 green, 4 blue; 4 green, 5 red
Game 85: 5 green, 13 red, 11 blue; 5 blue, 19 green, 15 red; 17 red, 3 green, 8 blue; 13 green, 10 red; 3 green, 17 red, 11 blue
Game 86: 1 green, 11 blue; 11 blue, 1 green, 8 red; 6 blue, 4 red; 4 blue, 17 red; 1 green, 15 red
Game 87: 3 green, 8 red, 6 blue; 6 red, 13 green, 1 blue; 4 blue, 8 red, 8 green
Game 88: 5 green, 5 blue; 3 green, 10 blue, 2 red; 6 blue, 7 red, 1 green; 5 green, 3 red, 11 blue; 8 red, 4 green, 6 blue
Game 89: 5 green, 10 blue, 12 red; 1 green, 13 red, 8 blue; 4 red, 11 green, 12 blue
Game 90: 4 green, 3 red, 11 blue; 1 green, 12 red, 12 blue; 9 blue, 5 red, 1 green; 2 green, 12 blue, 12 red
Game 91: 5 red, 8 blue, 1 green; 5 green, 3 blue; 9 blue, 7 green, 5 red; 1 green, 3 blue, 6 red; 9 blue, 11 green, 4 red; 2 green, 4 red, 10 blue
Game 92: 11 blue, 1 red, 6 green; 10 blue, 2 red; 4 red, 6 green, 19 blue
Game 93: 1 green, 3 blue, 3 red; 3 red; 5 blue, 3 red; 1 green, 4 red
Game 94: 9 red, 4 blue, 4 green; 1 blue, 6 red, 15 green; 10 red, 5 blue, 1 green; 2 blue, 4 green, 8 red
Game 95: 13 blue, 4 green, 3 red; 15 green, 3 red, 2 blue; 16 green, 8 blue, 2 red
Game 96: 15 blue, 7 green, 3 red; 5 red, 7 green, 17 blue; 6 red, 12 blue; 5 green, 10 blue, 4 red
Game 97: 5 red, 2 green; 8 red; 1 blue, 7 green, 2 red; 7 red, 15 green
Game 98: 6 green, 1 blue, 1 red; 3 green, 3 red; 1 blue, 13 green, 4 red
Game 99: 16 red, 5 blue, 9 green; 2 green, 7 blue, 2 red; 10 blue, 3 green; 9 red, 8 blue, 13 green; 16 green, 13 red, 10 blue
Game 100: 16 blue, 12 red, 3 green; 2 green, 7 blue; 5 blue, 4 green; 10 blue, 6 red, 6 green; 5 red, 12 blue, 2 green; 9 red, 12 blue, 11 green

100
d2/d2.structured-input Normal file
View File

@@ -0,0 +1,100 @@
001: 1r/3b/11g/1b/5r/3b/5b/13r/6r/1b/4g/16r/12g/
002: 3r/13b/5g/14b/14b/9b/10b/3r/2b/5b/11b/3b/3r/16b/2r/9g/
003: 17b/5r/3r/11b/17b/1r/6b/9g/3b/11b/1r/3b/10r/11b/12r/3b/15b/
004: 14b/14r/1b/15r/13b/1b/6b/15r/7g/
005: 3b/1b/3r/6r/2b/2b/12r/3b/1b/2b/9r/1b/2b/10r/
006: 5b/5g/4b/1r/10g/16b/1r/6b/1r/1b/13g/1r/5b/7g/14b/17b/
007: 1b/8b/4r/1b/4b/4r/6b/4r/4g/1r/8g/
008: 2r/5b/1g/1b/4r/8g/6b/12b/6r/3b/5r/8r/2b/13g/5b/4r/3b/
009: 11r/1b/2r/2b/1b/2b/9r/4r/2b/2b/1b/2g/1b/9r/2g/
010: 9r/4g/1b/3r/7g/3b/1r/1b/7b/4r/1b/1b/5b/10r/1r/5g/
011: 2b/4r/3g/1b/7r/4b/7r/1b/3b/6b/4r/3r/1b/3b/
012: 1b/6r/5b/3b/2r/4b/3b/1r/3b/
013: 6b/1r/9b/11r/4b/12g/6b/9r/19b/2b/6b/10b/1r/16b/4b/14b/
014: 7b/2r/1b/2r/19b/12b/6b/11r/
015: 4r/4b/7b/15b/1b/8r/2r/10b/11b/5r/4b/6g/9r/8b/3g/9b/9r/
016: 7r/2b/19g/6b/9g/8b/6r/19b/11b/7r/1b/9b/3r/17g/
017: 3b/4b/5r/2r/4b/11b/6b/13g/3b/12b/7r/
018: 9r/6b/7g/3b/3b/5r/18r/6b/4g/3b/10r/8b/
019: 3r/6g/1r/5b/4b/3r/14b/
020: 2b/2b/4r/14r/6b/5g/1b/5r/3g/10r/6b/6b/
021: 10b/12b/3r/1b/14r/5b/7g/12b/1r/13g/7r/4g/
022: 2r/1b/1r/2b/1r/1b/3b/3b/1r/1b/2r/
023: 4b/4b/1r/3b/1r/6g/1r/1b/
024: 5b/15b/13r/20b/13b/6r/5b/11r/16g/6r/5b/13g/12b/13b/3r/
025: 10b/17r/12r/16b/3g/4b/12b/10r/8b/3b/10r/5b/2r/12b/
026: 11r/9b/3b/3r/3g/10b/3b/4r/1b/4b/9r/5b/1r/7b/1r/3b/3g/
027: 1b/12r/4b/5r/2b/1b/3b/6b/10r/1b/4r/3b/
028: 6b/2g/2b/8b/1r/2b/2b/6b/8g/9b/5b/
029: 1b/9b/9r/13b/4r/9b/3b/8b/15r/15b/18b/3r/16b/10r/16b/12b/16r/
030: 14b/4b/1r/7r/14b/2b/4r/1g/
031: 2r/14b/3b/3b/3b/4r/8b/4r/1g/8b/3b/10b/1r/11g/13b/2r/3b/
032: 8b/16r/2b/8b/16r/16b/4b/17r/2r/5b/4b/
033: 2r/2b/1b/5r/1b/8b/14r/
034: 4r/4g/9g/1b/16g/1b/5r/9g/2r/15b/1b/
035: 1b/5r/1b/15r/13b/2r/13b/17g/9b/3r/11g/7b/8b/14r/
036: 19g/3b/1b/1r/1b/8b/13b/5r/5b/
037: 12r/7b/3b/12b/10r/9g/17b/8r/13b/9b/9b/8r/4r/13b/13b/15b/12r/14b/
038: 5b/1b/20r/1b/13r/18b/17b/9r/10g/4b/4r/12g/12b/12r/6g/12b/13r/2b/
039: 7b/6r/2g/6b/1r/7b/1r/
040: 1b/3r/15b/1g/1b/16r/2b/
041: 2b/4g/8b/3r/2b/9r/4g/4r/3b/10g/5b/3b/2r/
042: 7b/2b/1r/8b/4r/5b/1r/3g/
043: 3r/1b/1b/2b/2r/1r/2b/3b/
044: 3b/14b/1r/16b/5r/11g/12b/1b/13b/1r/5b/2r/6g/3b/5r/11g/
045: 7b/1r/1r/3b/3b/14b/1r/4b/3b/1r/15b/1r/3g/
046: 15r/4b/15r/11b/3g/14r/2b/2b/14r/8b/3g/4r/1b/
047: 4b/2b/3r/8r/2b/18b/1b/17b/1r/
048: 2b/4r/2b/15b/16r/5g/14b/2b/10r/3b/13r/6b/8b/4r/12b/15r/3b/9b/
049: 1b/6r/7b/1b/9b/9r/4b/8r/9b/1r/14g/2b/9r/
050: 3r/10b/14g/2r/9b/7g/4b/12g/1r/4b/5b/
051: 2b/6b/1b/10b/1r/3b/2g/
052: 1b/4r/1b/3r/5b/4b/1b/3r/5g/1r/1b/1b/12b/2r/4b/10b/7b/1r/
053: 12r/1b/8r/11b/11g/8r/6b/13g/11b/11r/16g/6r/9b/4b/
054: 2r/8b/15g/4b/3b/6r/12b/13b/4r/
055: 1b/16b/4r/3r/1b/1g/12r/16b/3r/
056: 4g/1r/4g/2r/3b/7g/2r/3b/15g/
057: 17g/1b/9b/1r/1b/9b/
058: 3b/8r/7b/4b/9b/2r/1r/2b/11b/8b/4g/
059: 6b/1r/4b/6g/4b/5b/
060: 3b/5b/1r/7b/5b/16r/14r/1b/1b/7b/2b/13r/5b/5b/
061: 1b/2b/2r/2g/6r/1b/1g/
062: 5r/8b/1g/1r/1b/2b/8b/
063: 2r/2b/2g/9b/7g/1b/4b/18b/3b/
064: 13b/1b/6r/13b/15r/8b/5b/14r/4b/2b/8b/12r/1b/5r/13g/7b/8b/2r/
065: 7b/12r/6g/11r/8b/8b/9r/7b/7b/14r/2b/17g/
066: 2b/5r/7r/14b/19b/2g/7b/4b/6r/
067: 4b/17r/7b/4b/6g/7b/7r/12b/2r/14b/
068: 1r/11b/4b/17b/1r/10g/3r/7b/1b/7b/3r/6b/2r/3g/2b/2r/4b/
069: 5b/4r/3r/11b/1b/6b/2b/10b/4r/5b/2r/11g/
070: 16r/7b/1g/14r/1b/4g/4r/4g/7b/5r/2g/
071: 14r/2b/13g/7b/5r/2b/3b/9b/11r/10r/4b/1g/
072: 1b/2r/6b/4b/4r/9b/6b/8b/1r/5r/4b/9b/15b/2b/7r/10b/2b/10r/
073: 7b/6r/7b/6b/5r/8g/5b/5r/
074: 11r/1b/2b/4b/1r/1b/2b/11r/9r/5b/15r/10b/9r/3b/
075: 1b/6r/9g/5r/1b/8g/2b/2r/1b/7r/1g/3b/6r/2b/1b/1r/
076: 16r/3b/9g/4b/4g/5b/1b/10r/6b/13r/1b/2b/8r/
077: 4r/4b/5b/5r/6r/3g/
078: 11b/1r/1b/18b/1r/6b/5r/2b/6r/1b/15g/5b/5r/
079: 2r/3b/13b/7b/5g/4b/2r/6g/6b/15b/
080: 9b/2b/1r/8b/1r/1b/7g/2b/1b/3g/5b/1r/2b/
081: 2b/8b/1r/3b/1b/6b/1g/3b/3b/1r/2b/8b/1r/8b/2g/
082: 5b/4r/1g/9r/12b/8b/9r/6b/15b/8b/10r/6g/
083: 2b/7b/4r/2b/11r/9g/7r/7b/6b/12b/4r/11g/11b/7b/7b/5r/2b/
084: 9r/1b/7g/5r/5g/4b/4b/4b/5r/
085: 5b/13r/11b/5b/19b/15r/17r/3b/8b/13b/10r/3b/17r/11b/
086: 1b/11b/11b/1b/8r/6b/4r/4b/17r/1b/15r/
087: 3b/8r/6b/6r/13b/1b/4b/8r/8g/
088: 5b/5b/3b/10b/2r/6b/7r/1g/5b/3r/11b/8r/4b/6b/
089: 5b/10b/12r/1b/13r/8b/4r/11b/12b/
090: 4b/3r/11b/1b/12r/12b/9b/5r/1g/2b/12b/12r/
091: 5r/8b/1g/5b/3b/9b/7b/5r/1b/3b/6r/9b/11b/4r/2b/4r/10b/
092: 11b/1r/6g/10b/2r/4r/6b/19b/
093: 1b/3b/3r/3r/5b/3r/1b/4r/
094: 9r/4b/4g/1b/6r/15g/10r/5b/1g/2b/4b/8r/
095: 13b/4b/3r/15b/3r/2b/16b/8b/2r/
096: 15b/7b/3r/5r/7b/17b/6r/12b/5b/10b/4r/
097: 5r/2g/8r/1b/7b/2r/7r/15g/
098: 6b/1b/1r/3b/3r/1b/13b/4r/
099: 16r/5b/9g/2b/7b/2r/10b/3g/9r/8b/13g/16b/13r/10b/
100: 16b/12r/3g/2b/7b/5b/4g/10b/6r/6g/5r/12b/2g/9r/12b/11g/

33
d2/d2.test.input Normal file
View File

@@ -0,0 +1,33 @@
Game 1:
3 blue, 4 red;
1 red, 2 green, 6 blue;
2 green
Game 2:
1 blue, 2 green;
3 green, 4 blue, 1 red;
1 green, 1 blue
Game 3:
8 green, 6 blue, 20 red;
5 blue, 4 red, 13 green;
5 green, 1 red
Game 4:
1 green, 3 red, 6 blue;
3 green, 6 red;
3 green, 15 blue, 14 red
Game 5:
6 red, 1 blue, 3 green;
2 blue, 1 red, 2 green
982
0 * 10 = 0
0 + 9 = 9
9 * 10 = 90
90 + 8 = 98
98 * 10 = 980
980 + 2 = 982

View File

@@ -0,0 +1,5 @@
1: 4/3/1; 1/2/6; 0/2/0;
2: 0/2/1; 1/3/4; 0/1/1;
3: 20/8/6; 4/13/5; 1/5/0;
4: 3/1/6; 6/3/0; 14/3/15;
5: 6/3/1; 1/2/2;

102
d2/main.odin Normal file
View File

@@ -0,0 +1,102 @@
package d2;
import "core:os"
import "core:fmt"
die :: proc (eno: os.Errno) {
fmt.printf("Fatal error: %s\n", os.get_last_error_string());
os.exit(int(eno));
}
usage :: proc() {
fmt.printf("Usage: ./%s FILENAME\n", os.args[0]);
}
BUFFER_SIZE :: 1024;
State :: enum {
NONE, READ_ID, READ_RESULT, READ_DELIM, READ_COLON, READ_NEWLINE
}
Result :: struct{
red: int, green: int, blue: int,
};
TARGET_GAME :: Result{12, 13, 14};
games: [dynamic]Result = make([dynamic]Result);
stack: [dynamic]State = make([dynamic]State);
collect_digits :: proc(buffer: []u8, index: ^int) -> int {
assert(index^ < len(buffer));
result := 0
for '0' <= buffer[index^] && buffer[index^] <= '9' && index^ < len(buffer) {
result *= 10;
result += int(buffer[index^] - '0');
index^ += 1;
}
return result;
}
consume_token :: proc(token: string, buffer: []u8, index: ^int) -> bool {
bi := index^;
ti := 0;
for ti < len(token) {
if bi >= len(buffer) || buffer[bi] != token[ti] {
return false;
} else {
ti += 1; bi += 1;
}
}
return true;
}
attempt_1 :: proc() {
if len(os.args) < 2 {
usage();
os.exit(-1);
}
fh, eno := os.open(os.args[1], os.O_RDONLY);
if eno != os.ERROR_NONE { die(eno); }
defer os.close(fh);
id := -1;
red, green, blue, num_read: int;
state := State.READ_ID;
total := 0;
buffer, ok := os.read_entire_file_from_handle(fh);
if !ok {
// die(os.get_last_error());
// TODO
os.exit(-1);
}
ix := 0;
for ix < len(buffer) {
if eno != os.ERROR_NONE { die(eno); }
switch pop(&stack) {
case .READ_ID:
id = collect_digits(buffer, &ix);
append(&stack, State.READ_RESULT);
append(&stack, State.READ_DELIM);
append(&stack, State.READ_COLON);
case .READ_COLON:
if !consume_token(":", buffer, &ix) {
// TODO
os.exit(-1);
}
case .READ_DELIM:
for ix < len(buffer) && buffer[ix] == ' ' {
ix += 1;
}
case .READ_NEWLINE:
if !consume_token("\n", buffer, &ix) {
// TODO
os.exit(-1);
}
case .READ_RESULT:
append(&stack, State.READ_ID);
append(&stack, State.READ_NEWLINE);
case .NONE:
os.exit(-1);
}
}
}
main :: proc() {
attempt_1();
}

BIN
lexer_gen.bin Executable file

Binary file not shown.

20
lexer_gen/main.odin Normal file
View File

@@ -0,0 +1,20 @@
package lexer_gen
import "core:os"
import "core:fmt"
main :: proc() {
if len(os.args) < 2 {
// TODO
return
}
buf, ok := os.read_entire_file_from_filename(os.args[1])
if !ok {
// TODO
return
}
tokenizer := tokenize(buf)
parser := parse_token_stream(tokenizer)
print_parse_tree(parser)
fmt.print('\n')
}

500
lexer_gen/re.odin Normal file
View File

@@ -0,0 +1,500 @@
package lexer_gen
import "core:os"
import "core:fmt"
import "core:strings"
import "core:testing"
import "core:log"
logger := log.create_console_logger()
INDENTATION :: " "
print_indentation :: proc (level: int) {
for ii := 0; ii < level; ii += 1 {
fmt.print(INDENTATION)
}
}
Tokens :: enum {
ASTERISK,
BACK_SLASH,
CARROT,
CLOSE_BRACKET,
CLOSE_PAREN,
DOLLAR,
DOT,
FORWARD_SLASH,
LITERAL,
OPEN_BRACKET,
OPEN_PAREN,
PIPE,
TACK,
UNDERLINE,
_EOS,
}
token_type_string :: proc (tt: Tokens) -> string {
switch tt {
case .ASTERISK:
return "ASTERISK"
case .BACK_SLASH:
return "BACK_SLASH"
case .CARROT:
return "CARROT"
case .CLOSE_BRACKET:
return "CLOSE_BRACKET"
case .CLOSE_PAREN:
return "CLOSE_PAREN"
case .DOLLAR:
return "DOLLAR"
case .DOT:
return "DOT"
case .FORWARD_SLASH:
return "FORWARD_SLASH"
case .LITERAL:
return "LITERAL"
case .OPEN_BRACKET:
return "OPEN_BRACKET"
case .OPEN_PAREN:
return "OPEN_PAREN"
case .PIPE:
return "PIPE"
case .TACK:
return "TACK"
case .UNDERLINE:
return "UNDERLINE"
case ._EOS:
return "_EOS"
}
panic("Unreachable");
}
print_token_type :: proc(tt: Tokens) {
fmt.print(token_type_string(tt));
}
Token :: struct {
type_: Tokens,
start: int,
end: int,
}
print_token :: proc (token: ^Token) {
using token
fmt.printf("(%s start=%d end=%d)", token_type_string(type_), start, end)
}
Tokenizer :: struct {
buffer: []u8,
tokens: [dynamic](^Token),
idx: int,
}
Parser :: struct {
tokenizer: ^Tokenizer,
ast: ^AstNode,
current: ^AstNode,
stack: [dynamic]^AstNode
}
new_parser :: proc(tokenizer: ^Tokenizer) -> ^Parser {
p := new(Parser);
p.tokenizer = tokenizer;
p.ast = new_ast_node(Group);
p.current = p.ast;
p.ast.start = 0;
return p;
}
AstNode :: struct {
start: int,
end: int,
next: ^AstNode,
variant: union{
^Group, ^Literal, ^Range, ^Repeater
}
}
new_ast_node :: proc($T: typeid) -> ^T {
e := new(T)
e.variant = e
return e
}
Literal :: struct {
using node: AstNode,
}
print_literal :: proc(parser: ^Parser, node: ^Literal, level := 0) {
using parser
value := tokenizer.buffer[node.start:node.end]
print_indentation(level)
fmt.printf("(Literal start=%d end=%d value='%s'", node.start, node.end, value)
}
Range :: struct {
using node: AstNode,
lbound: rune,
ubound: rune,
};
print_range :: proc (parser: ^Parser, range: ^Range, level := 0) {
print_indentation(level);
fmt.printf("(range %s %c)", range.lbound, range.ubound);
}
Group :: struct {
using node: AstNode,
child: ^AstNode
};
print_group :: proc(parser: ^Parser, group: ^Group, level := 0) {
// TODO
if parser == nil || group == nil {
// TODO handle error
}
fmt.print("(group ")
current: ^AstNode = group.child;
for current != nil {
print_node(parser, current, level + 1)
current = current.next
}
fmt.print(")")
}
RepeaterType :: enum {
One,
OneOrMore,
ZeroOrMore,
ExactRange,
_None
}
repeater_type_string :: proc(rt: RepeaterType) -> string {
switch rt {
case .One:
return "One"
case .OneOrMore:
return "OneOrMore"
case .ZeroOrMore:
return "ZeroOrMore"
case .ExactRange:
return "ExactRange"
case ._None:
return "_None"
}
panic("Unreachable")
}
Repeater :: struct {
using node: AstNode,
target: ^AstNode,
type_: RepeaterType
}
print_repeater :: proc(parser: ^Parser, node: ^Repeater, level := 0) {
using node
print_indentation(level);
fmt.printf("(repeat %s \n", repeater_type_string(type_));
print_node(parser, target, level + 1);
fmt.print(")");
}
print_node :: proc(parser: ^Parser, node: $T/^AstNode, level := 0) {
switch node_ in node.variant {
case ^Group: print_group(parser, node_, level);
case ^Literal: print_literal(parser, node_, level);
case ^Repeater: print_repeater(parser, node_, level);
case ^Range: print_range(parser, node_, level);
}
}
new_tokenizer :: proc() -> (state: ^Tokenizer) {
state = new(Tokenizer)
using state
tokens = make([dynamic](^Token))
idx = 0
return
}
scan_one :: proc(tokenizer: ^Tokenizer, tt: Tokens) {
using tokenizer;
fmt.printf("%s: Call with type %s\n", #procedure, tt);
new_tok := new(Token);
new_tok.type_ = tt;
new_tok.start = idx;
new_tok.end = idx + 1;
append(&tokens, new_tok);
}
peek_front_safe :: proc(array: [dynamic]$E) -> (E, bool) {
if array == nil {
return nil, false;
}
return array[0], true;
}
peek_safe :: proc(array: [dynamic]$E) -> (E, bool) {
if array == nil || len(array) == 0 {
return nil, false;
}
return array[len(array) - 1], true;
}
peek_type :: proc (tokens: [dynamic]^Token, start := 0) -> Tokens {
tok, ok := peek_front_safe(tokens);
if !ok {
return ._EOS;
}
return tok.type_;
}
scan_one_maybe :: proc(tokenizer: ^Tokenizer) -> bool {
switch tokenizer.buffer[tokenizer.idx] {
case '[':
scan_one(tokenizer, Tokens.OPEN_BRACKET);
case ']':
scan_one(tokenizer, Tokens.CLOSE_BRACKET);
case '(':
scan_one(tokenizer, Tokens.OPEN_PAREN);
case ')':
scan_one(tokenizer, Tokens.CLOSE_PAREN);
case '|':
scan_one(tokenizer, Tokens.PIPE);
case '^':
scan_one(tokenizer, Tokens.CARROT);
case '$':
scan_one(tokenizer, Tokens.DOLLAR);
case '*':
scan_one(tokenizer, Tokens.ASTERISK);
case '-':
scan_one(tokenizer, Tokens.TACK);
case '_':
scan_one(tokenizer, Tokens.UNDERLINE);
case '.':
scan_one(tokenizer, Tokens.DOT);
case '\\':
scan_one(tokenizer, Tokens.BACK_SLASH);
case '/':
scan_one(tokenizer, Tokens.FORWARD_SLASH);
case:
return false;
}
return true;
}
tokenize :: proc (buffer: []u8) -> ^Tokenizer {
tokenizer := new_tokenizer();
tokenizer.buffer = buffer;
new_tok: ^Token = nil;
literal_tok: ^Token = nil;
for ; tokenizer.idx < len(tokenizer.buffer); tokenizer.idx += 1 {
scanned := scan_one_maybe(tokenizer);
if scanned && literal_tok != nil {
// End a literal
literal_tok.end = tokenizer.idx;
literal_tok = nil;
} else if !scanned && literal_tok == nil {
// Start a literal
literal_tok = new(Token);
literal_tok.type_ = .LITERAL;
literal_tok.start = tokenizer.idx;
literal_tok.end = tokenizer.idx;
fmt.printf("Appending!\n");
append(&tokenizer.tokens, literal_tok);
}
}
assert(len(tokenizer.tokens) > 0);
return tokenizer;
}
print_token_stream :: proc(stream: []^Token) {
for tok, index in stream {
fmt.printf("%d: ", index);
print_token(tok);
fmt.print("\n");
}
}
@(test)
test_tokenize_basic :: proc(sig: ^testing.T) {
buf, ok := os.read_entire_file_from_filename("./test/test_tokenizer__basic.input");
tokenizer := tokenize(buf);
assert(
len(tokenizer.tokens) == 2,
fmt.tprintf("len(tokens) = %d != 2\n", len(tokenizer.tokens)));
print_token_stream(tokenizer.tokens[:]);
assert(
tokenizer.tokens[0].type_ == Tokens.LITERAL,
fmt.tprintf("tt = %s\n", tokenizer.tokens[0].type_));
assert(
tokenizer.tokens[1].type_ == Tokens.ASTERISK,
fmt.tprintf("tt = %s\n", tokenizer.tokens[0].type_));
}
parse_escape :: proc(parser: ^Parser) -> ^AstNode {
using parser
node := new_ast_node(Literal)
tok, ok := pop_front_safe(&(tokenizer.tokens))
if !ok {
free(node)
return nil
}
node.start = tok.start
assert(tok.type_ == .BACK_SLASH)
tok, ok = pop_front_safe(&tokenizer.tokens)
if !ok {
// TODO
}
if tok.type_ == .LITERAL {
// TODO Cannot escape a literal
}
node.end = tok.end
return node
}
parse_zero_or_more_repeater :: proc(parser: ^Parser) -> ^AstNode {
using parser
// Create an AST node representing the repeater
ok: bool
repeater_token: ^Token
repeater_token, ok = pop_safe(&tokenizer.tokens)
if !ok {
// TODO Handle error
return nil
}
assert(repeater_token.type_ == .ASTERISK)
// Attach the ast node that the repeater operates on
node := new_ast_node(Repeater)
node.type_ = .ZeroOrMore
node.start = repeater_token.start
node.end = repeater_token.end
return node
}
parse_literal :: proc(parser: ^Parser) -> ^AstNode {
tok, ok := pop_front_safe(&parser.tokenizer.tokens)
if !ok { return nil }
node := new(Literal)
node.start = tok.start
node.end = tok.end
node.next = nil
value := parser.tokenizer.buffer[node.start:node.end]
log.debugf("Finished parsing type=Literal with value='%s'\n", value)
return node
}
parse_token_stream :: proc(tokenizer: ^Tokenizer) -> ^Parser {
parser := new_parser(tokenizer)
parser.ast = parse_token_stream__inner(parser)
fmt.printf("%x\n", &parser.ast)
return parser
}
parse_token_stream__inner :: proc(parser: ^Parser) -> ^AstNode {
using parser
next: ^AstNode
for len(tokenizer.tokens) > 0 {
type_ := peek_type(tokenizer.tokens)
log.debugf("Parsing token of type=%s\n", type_)
#partial switch type_ {
case .LITERAL:
next = parse_literal(parser)
case .BACK_SLASH:
next = parse_escape(parser)
case .ASTERISK:
next = parse_zero_or_more_repeater(parser)
case:
msg := fmt.tprintf("Don't know how to handle token of type=%s\n", type_)
assert(false, msg)
}
if next == nil {
return current
} else if current != nil {
current.next = next
}
current = next
}
return current
}
print_parse_tree :: proc(parser: ^Parser) {
for current := parser.ast; current != nil; current = current.next {
print_node(parser, current, 0);
}
}
@(test)
test_parse_token_stream :: proc(sig: ^testing.T) {
buf, ok := os.read_entire_file_from_filename("./test/test_parser__basic.input");
assert(ok)
tokenizer := tokenize(buf)
parser := parse_token_stream(tokenizer)
print_parse_tree(parser)
}
iter_lines :: proc(buffer: []u8, index: ^int) -> ([]u8, bool) {
if index^ < 0 || index^ >= len(buffer) {
return buffer, true
}
lbound := index^
for index^ < len(buffer) {
if buffer[index^] == '\n' {
index^ += 1 // Skip the newline
return buffer[lbound:index^ - 1], false
}
index^ += 1
}
return buffer[lbound:index^], false
}
@(test)
test_iter_lines :: proc(sig: ^testing.T) {
input, ok_ := os.read_entire_file_from_filename("./test/test_iter_lines__basic.input")
assert(ok_)
linum := 1
index := 0
line, stop_iter := iter_lines(input, &index)
for !stop_iter {
linum += 1
fmt.printf("Line %0d: %s\n", linum, line)
line, stop_iter = iter_lines(input, &index)
}
}
parse_token_expectation :: proc(filename: string) {
buffer, ok := os.read_entire_file_from_filename(filename);
if !ok {
// TODO handle error
log.errorf("%s\n", os.get_last_error_string());
return;
}
start := 0
line, ok_ := iter_lines(buffer, &start)
for ; ok; line, ok = iter_lines(buffer, &start) {
}
}
re_compile :: proc(pattern: string) -> ^AstNode {
pattern_ := strings.clone(pattern);
tokenizer := tokenize(transmute([]u8)pattern_)
parser := parse_token_stream(tokenizer);
return parser.ast;
}
re_match :: proc() {
assert(false);
return;
}
re_search :: proc() {
assert(false);
return;
}

View File

@@ -0,0 +1,7 @@
Line 1: Hello world!
Line 2: How
Line 3: ,.sd1
Line 4: Are
Line 5: you? ??? !
Line 6:
Line 7: I'm doing well, thanks

View File

@@ -0,0 +1,7 @@
Hello world!
How
,.sd1
Are
you? ??? !
I'm doing well, thanks

View File

@@ -0,0 +1 @@
\(a

View File

@@ -0,0 +1 @@
A*

View File

@@ -0,0 +1 @@
A*