-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathDay23.fs
58 lines (48 loc) · 1.87 KB
/
Day23.fs
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
module Year2015Day23
open AdventOfCode.FSharp.Common
type Instruction =
| Half of string
| Triple of string
| Increment of string
| Jump of int
| JumpIfEven of string * int
| JumpIfOne of string * int
let asInstr line =
let toks = splitBy " " id line
match toks.[0] with
| "hlf" -> Half toks.[1]
| "tpl" -> Triple toks.[1]
| "inc" -> Increment toks.[1]
| "jmp" -> Jump (int toks.[1])
| "jie" -> JumpIfEven (toks.[1].TrimEnd(','), int toks.[2])
| "jio" -> JumpIfOne (toks.[1].TrimEnd(','), int toks.[2])
| _ -> failwithf "Invalid Instruction: %s" line
let parse = parseEachLine asInstr >> Seq.toArray
type State = { PC : int; Regs : Map<string, int>; Code : Instruction [] }
let setReg reg value state = { state with Regs = Map.add reg value state.Regs }
let getReg reg state = Map.tryFind reg state.Regs |> Option.defaultValue 0
let mapReg f reg state = state |> setReg reg (getReg reg state |> f)
let jmp offset state = { state with PC = state.PC + offset }
let step state =
match state.Code.[state.PC] with
| Half r -> state |> mapReg (fun r -> r / 2) r |> jmp 1
| Triple r -> state |> mapReg (fun r -> r * 3) r |> jmp 1
| Increment r -> state |> mapReg ((+) 1) r |> jmp 1
| Jump o -> state |> jmp o
| JumpIfEven (r, o) -> state |> jmp (if getReg r state % 2 = 0 then o else 1)
| JumpIfOne (r, o) -> state |> jmp (if getReg r state = 1 then o else 1)
let runProgram state =
let rec run' state =
if state.PC < 0 || state.PC >= state.Code.Length then state
else step state |> run'
run' state
let solvePart1 input =
{ PC = 0; Regs = Map.empty; Code = input }
|> runProgram
|> getReg "b"
let solvePart2 input =
{ PC = 0; Regs = Map.empty; Code = input }
|> setReg "a" 1
|> runProgram
|> getReg "b"
let solver = { parse = parse; part1 = solvePart1; part2 = solvePart2 }