-
Notifications
You must be signed in to change notification settings - Fork 10
/
Copy pathDay19.fs
65 lines (54 loc) · 2.18 KB
/
Day19.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
59
60
61
62
63
64
65
module Year2019Day19
open AdventOfCode.FSharp.Common
open AdventOfCode.FSharp.Y2019.Common.Intcode
let getIsBeamCallback intcode =
let comp = Computer.create intcode |> run
let isBeam i j=
match comp with
| Input f ->
match f (int64 i) with
| Input f ->
match f (int64 j) with
| Output (o, _) -> List.head o = 1L
| _ -> failwith "Expected output"
| _ -> failwith "Expected a second input"
| _ -> failwith "Expected an input"
isBeam
let solvePart1 intcode =
let isBeam = getIsBeamCallback intcode
Seq.init (50*50) (fun i -> isBeam (i / 50) (i % 50))
|> Seq.filter id
|> Seq.length
type BeamArea =
{ TopRight : int * int
BottomLeft : int * int }
static member topLeft { TopRight = (_, y); BottomLeft = (x, _) } = (x, y)
static member height { TopRight = (_, y1); BottomLeft = (_, y2) } = y2 - y1
static member width { TopRight = (x1, _); BottomLeft = (x2, _) } = x1 - x2
static member increaseWidth amount isBeam beamArea =
let x, y = beamArea.TopRight
let rec findNext y =
if isBeam (x + amount) y then y
else findNext (y + 1)
{ beamArea with TopRight = x + amount, findNext y }
static member increaseHeight amount isBeam beamArea =
let x, y = beamArea.BottomLeft
let rec findNext x =
if isBeam x (y + amount) then x
else findNext (x + 1)
{ beamArea with BottomLeft = findNext x, y + amount }
let solvePart2 intcode =
let isBeam = getIsBeamCallback intcode
let rec seek beamArea =
let width = BeamArea.width beamArea
let height = BeamArea.height beamArea
if width = 99 && height = 99 then
let (x, y) = BeamArea.topLeft beamArea
x * 10000 + y
else
if width <= height then
BeamArea.increaseWidth (99 - width) isBeam beamArea |> seek
else
BeamArea.increaseHeight (99 - height) isBeam beamArea |> seek
seek { TopRight = (0, 0); BottomLeft = (0, 0) }
let solver = { parse = parseIntCode; part1 = solvePart1; part2 = solvePart2 }