-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDay06.hs
74 lines (64 loc) · 2.04 KB
/
Day06.hs
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
66
67
68
69
70
71
72
73
74
-- |
-- Module : AOC2021.Day06
-- License : BSD3
--
-- Stability : experimental
-- Portability : non-portable
--
-- Day 6. See "AOC.Solver" for the types used in this module!
module AOC2021.Day06 (
day06a,
day06b,
) where
import AOC.Common ()
import AOC.Solver ((:~>) (..))
import Control.Monad ((<=<))
import Control.Monad.ST (runST)
import Data.Finite (Finite, packFinite)
import Data.Foldable (traverse_)
import Data.List.Split (splitOn)
import Data.Semigroup (stimes)
import qualified Data.Vector.Mutable.Sized as SMV
import qualified Data.Vector.Sized as SV
import GHC.TypeNats (KnownNat)
import qualified Linear as L
import Text.Read (readMaybe)
newtype SquareMat n = SquareMat (SV.Vector n (SV.Vector n Int))
deriving stock (Show)
applyMat :: KnownNat n => SquareMat n -> SV.Vector n Int -> SV.Vector n Int
applyMat (SquareMat xs) x = xs L.!* x
step :: SquareMat 9
step =
SquareMat $
SV.fromTuple
( SV.fromTuple (0, 1, 0, 0, 0, 0, 0, 0, 0)
, SV.fromTuple (0, 0, 1, 0, 0, 0, 0, 0, 0)
, SV.fromTuple (0, 0, 0, 1, 0, 0, 0, 0, 0)
, SV.fromTuple (0, 0, 0, 0, 1, 0, 0, 0, 0)
, SV.fromTuple (0, 0, 0, 0, 0, 1, 0, 0, 0)
, SV.fromTuple (0, 0, 0, 0, 0, 0, 1, 0, 0)
, SV.fromTuple (1, 0, 0, 0, 0, 0, 0, 1, 0)
, SV.fromTuple (0, 0, 0, 0, 0, 0, 0, 0, 1)
, SV.fromTuple (1, 0, 0, 0, 0, 0, 0, 0, 0)
)
instance KnownNat n => Semigroup (SquareMat n) where
SquareMat x <> SquareMat y = SquareMat (x L.!*! y)
stepN :: Int -> SquareMat 9
stepN n = stimes n step
allocate :: forall n. KnownNat n => [Finite n] -> SV.Vector n Int
allocate xs = runST do
v <- SMV.replicate 0
traverse_ (SMV.modify v (+ 1)) xs
SV.freeze v
day06 :: Int -> [Finite 9] :~> Int
day06 n =
MkSol
{ sParse = traverse (packFinite <=< readMaybe) . splitOn ","
, sShow = show
, -- this isn't going to bench very accurately because stepN n is auto-cached
sSolve = Just . sum . (stepN n `applyMat`) . allocate
}
day06a :: [Finite 9] :~> Int
day06a = day06 80
day06b :: [Finite 9] :~> Int
day06b = day06 256