{-# Language ImportQualifiedPost, LambdaCase, MonadComprehensions #-}
module Main
(
main
, part1
, part2
, feed
, Effect(..)
, interpreter
, Instruction(..)
, Expression(..)
, Register(..)
, instruction
, register
, expression
) where
import Advent.Input ( getInputLines )
import Advent.ReadS (P(..), runP)
import Control.Applicative ((<|>), empty)
import Data.Char (isAlpha, isDigit)
import Data.Map qualified as Map
import Data.Map.Strict (Map)
import Data.Maybe (fromMaybe)
import Data.Vector qualified as V
import Text.ParserCombinators.ReadP
main :: IO ()
IO ()
main =
do pgm <- (String -> Instruction) -> [String] -> [Instruction]
forall a b. (a -> b) -> [a] -> [b]
map (P Instruction -> String -> Instruction
forall a. P a -> String -> a
runP P Instruction
instruction) ([String] -> [Instruction]) -> IO [String] -> IO [Instruction]
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> Int -> Int -> IO [String]
getInputLines Int
2017 Int
18
let start = [Instruction] -> Integer -> Effect
interpreter [Instruction]
pgm
print (part1 start)
print (part2 start)
part1 ::
(Integer -> Effect) ->
Maybe Integer
part1 :: (Integer -> Effect) -> Maybe Integer
part1 Integer -> Effect
start = Maybe Integer -> Effect -> Maybe Integer
go Maybe Integer
forall a. Maybe a
Nothing (Integer -> Effect
start Integer
0)
where
go :: Maybe Integer -> Effect -> Maybe Integer
go :: Maybe Integer -> Effect -> Maybe Integer
go Maybe Integer
_ (Send Integer
x Effect
p) = Maybe Integer -> Effect -> Maybe Integer
go (Integer -> Maybe Integer
forall a. a -> Maybe a
Just Integer
x) Effect
p
go Maybe Integer
s (Receive Integer
0 Integer -> Effect
p) = Maybe Integer -> Effect -> Maybe Integer
go Maybe Integer
s (Integer -> Effect
p Integer
0)
go Maybe Integer
s (Receive Integer
_ Integer -> Effect
_) = Maybe Integer
s
go Maybe Integer
_ Effect
Halt = Maybe Integer
forall a. Maybe a
Nothing
part2 ::
(Integer -> Effect) ->
Int
part2 :: (Integer -> Effect) -> Int
part2 Integer -> Effect
start = Effect -> Effect -> Int -> Int
go (Integer -> Effect
start Integer
0) (Integer -> Effect
start Integer
1) Int
0
where
go :: Effect -> Effect -> Int -> Int
go :: Effect -> Effect -> Int -> Int
go (Send Integer
o Effect
p0) Effect
p1 Int
ctr = Effect -> Effect -> Int -> Int
go Effect
p0 (Integer -> Effect -> Effect
feed Integer
o Effect
p1) Int
ctr
go Effect
p0 (Send Integer
o Effect
p1) Int
ctr = Effect -> Effect -> Int -> Int
go (Integer -> Effect -> Effect
feed Integer
o Effect
p0) Effect
p1 (Int
ctrInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1)
go Effect
_ Effect
_ Int
ctr = Int
ctr
feed :: Integer -> Effect -> Effect
feed :: Integer -> Effect -> Effect
feed Integer
i (Send Integer
o Effect
p) = Integer -> Effect -> Effect
Send Integer
o (Integer -> Effect -> Effect
feed Integer
i Effect
p)
feed Integer
i (Receive Integer
_ Integer -> Effect
k) = Integer -> Effect
k Integer
i
feed Integer
_ Effect
Halt = Effect
Halt
data Effect
= Halt
| Send Integer Effect
| Receive Integer (Integer -> Effect)
interpreter ::
[Instruction] ->
Integer ->
Effect
interpreter :: [Instruction] -> Integer -> Effect
interpreter [Instruction]
cmds Integer
pid = Int -> Map Register Integer -> Effect
go Int
0 Map Register Integer
initialRegs
where
v :: Vector Instruction
v = [Instruction] -> Vector Instruction
forall a. [a] -> Vector a
V.fromList [Instruction]
cmds
initialRegs :: Map Register Integer
initialRegs = Register -> Integer -> Map Register Integer
forall k a. k -> a -> Map k a
Map.singleton (Char -> Register
Register Char
'p') Integer
pid
go ::
Int ->
Map Register Integer ->
Effect
go :: Int -> Map Register Integer -> Effect
go Int
pc Map Register Integer
regs =
case Vector Instruction
v Vector Instruction -> Int -> Maybe Instruction
forall a. Vector a -> Int -> Maybe a
V.!? Int
pc of
Maybe Instruction
Nothing -> Effect
Halt
Just (Snd Expression
e ) -> Integer -> Effect -> Effect
Send (Expression -> Integer
val Expression
e) (Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) Map Register Integer
regs)
Just (Rcv Register
r ) -> Integer -> (Integer -> Effect) -> Effect
Receive (Register -> Integer
reg Register
r) (Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (Map Register Integer -> Effect)
-> (Integer -> Map Register Integer) -> Integer -> Effect
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Register -> Integer -> Map Register Integer
set Register
r)
Just (Set Register
r Expression
e) -> Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (Register -> Integer -> Map Register Integer
set Register
r (Expression -> Integer
val Expression
e))
Just (Add Register
r Expression
e) -> Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (Register -> (Integer -> Integer) -> Map Register Integer
upd Register
r (Expression -> Integer
val Expression
e Integer -> Integer -> Integer
forall a. Num a => a -> a -> a
+))
Just (Mul Register
r Expression
e) -> Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (Register -> (Integer -> Integer) -> Map Register Integer
upd Register
r (Expression -> Integer
val Expression
e Integer -> Integer -> Integer
forall a. Num a => a -> a -> a
*))
Just (Mod Register
r Expression
e) -> Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
1) (Register -> (Integer -> Integer) -> Map Register Integer
upd Register
r (Integer -> Integer -> Integer
forall a. Integral a => a -> a -> a
`rem` Expression -> Integer
val Expression
e))
Just (Jgz Expression
x Expression
y) -> Int -> Map Register Integer -> Effect
go (Int
pcInt -> Int -> Int
forall a. Num a => a -> a -> a
+Int
o) Map Register Integer
regs
where o :: Int
o | Expression -> Integer
val Expression
x Integer -> Integer -> Bool
forall a. Ord a => a -> a -> Bool
> Integer
0 = Integer -> Int
forall a b. (Integral a, Num b) => a -> b
fromIntegral (Expression -> Integer
val Expression
y)
| Bool
otherwise = Int
1
where
val :: Expression -> Integer
val (RegisterExpression Register
r) = Register -> Integer
reg Register
r
val (IntegerExpression Integer
i) = Integer
i
reg :: Register -> Integer
reg Register
r = Integer -> Register -> Map Register Integer -> Integer
forall k a. Ord k => a -> k -> Map k a -> a
Map.findWithDefault Integer
0 Register
r Map Register Integer
regs
set :: Register -> Integer -> Map Register Integer
set Register
r Integer
x = Register -> Integer -> Map Register Integer -> Map Register Integer
forall k a. Ord k => k -> a -> Map k a -> Map k a
Map.insert Register
r Integer
x Map Register Integer
regs
upd :: Register -> (Integer -> Integer) -> Map Register Integer
upd Register
r Integer -> Integer
f = (Maybe Integer -> Maybe Integer)
-> Register -> Map Register Integer -> Map Register Integer
forall k a.
Ord k =>
(Maybe a -> Maybe a) -> k -> Map k a -> Map k a
Map.alter (Integer -> Maybe Integer
forall a. a -> Maybe a
Just (Integer -> Maybe Integer)
-> (Maybe Integer -> Integer) -> Maybe Integer -> Maybe Integer
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Integer -> Integer
f (Integer -> Integer)
-> (Maybe Integer -> Integer) -> Maybe Integer -> Integer
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Integer -> Maybe Integer -> Integer
forall a. a -> Maybe a -> a
fromMaybe Integer
0) Register
r Map Register Integer
regs
newtype Register = Register Char
deriving (ReadPrec [Register]
ReadPrec Register
Int -> ReadS Register
ReadS [Register]
(Int -> ReadS Register)
-> ReadS [Register]
-> ReadPrec Register
-> ReadPrec [Register]
-> Read Register
forall a.
(Int -> ReadS a)
-> ReadS [a] -> ReadPrec a -> ReadPrec [a] -> Read a
$creadsPrec :: Int -> ReadS Register
readsPrec :: Int -> ReadS Register
$creadList :: ReadS [Register]
readList :: ReadS [Register]
$creadPrec :: ReadPrec Register
readPrec :: ReadPrec Register
$creadListPrec :: ReadPrec [Register]
readListPrec :: ReadPrec [Register]
Read, Int -> Register -> ShowS
[Register] -> ShowS
Register -> String
(Int -> Register -> ShowS)
-> (Register -> String) -> ([Register] -> ShowS) -> Show Register
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Register -> ShowS
showsPrec :: Int -> Register -> ShowS
$cshow :: Register -> String
show :: Register -> String
$cshowList :: [Register] -> ShowS
showList :: [Register] -> ShowS
Show, Register -> Register -> Bool
(Register -> Register -> Bool)
-> (Register -> Register -> Bool) -> Eq Register
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
$c== :: Register -> Register -> Bool
== :: Register -> Register -> Bool
$c/= :: Register -> Register -> Bool
/= :: Register -> Register -> Bool
Eq, Eq Register
Eq Register =>
(Register -> Register -> Ordering)
-> (Register -> Register -> Bool)
-> (Register -> Register -> Bool)
-> (Register -> Register -> Bool)
-> (Register -> Register -> Bool)
-> (Register -> Register -> Register)
-> (Register -> Register -> Register)
-> Ord Register
Register -> Register -> Bool
Register -> Register -> Ordering
Register -> Register -> Register
forall a.
Eq a =>
(a -> a -> Ordering)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> Bool)
-> (a -> a -> a)
-> (a -> a -> a)
-> Ord a
$ccompare :: Register -> Register -> Ordering
compare :: Register -> Register -> Ordering
$c< :: Register -> Register -> Bool
< :: Register -> Register -> Bool
$c<= :: Register -> Register -> Bool
<= :: Register -> Register -> Bool
$c> :: Register -> Register -> Bool
> :: Register -> Register -> Bool
$c>= :: Register -> Register -> Bool
>= :: Register -> Register -> Bool
$cmax :: Register -> Register -> Register
max :: Register -> Register -> Register
$cmin :: Register -> Register -> Register
min :: Register -> Register -> Register
Ord)
data Expression
= RegisterExpression Register
| IntegerExpression Integer
deriving (ReadPrec [Expression]
ReadPrec Expression
Int -> ReadS Expression
ReadS [Expression]
(Int -> ReadS Expression)
-> ReadS [Expression]
-> ReadPrec Expression
-> ReadPrec [Expression]
-> Read Expression
forall a.
(Int -> ReadS a)
-> ReadS [a] -> ReadPrec a -> ReadPrec [a] -> Read a
$creadsPrec :: Int -> ReadS Expression
readsPrec :: Int -> ReadS Expression
$creadList :: ReadS [Expression]
readList :: ReadS [Expression]
$creadPrec :: ReadPrec Expression
readPrec :: ReadPrec Expression
$creadListPrec :: ReadPrec [Expression]
readListPrec :: ReadPrec [Expression]
Read, Int -> Expression -> ShowS
[Expression] -> ShowS
Expression -> String
(Int -> Expression -> ShowS)
-> (Expression -> String)
-> ([Expression] -> ShowS)
-> Show Expression
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Expression -> ShowS
showsPrec :: Int -> Expression -> ShowS
$cshow :: Expression -> String
show :: Expression -> String
$cshowList :: [Expression] -> ShowS
showList :: [Expression] -> ShowS
Show)
data Instruction
= Snd Expression
| Rcv Register
| Set Register Expression
| Add Register Expression
| Mul Register Expression
| Mod Register Expression
| Jgz Expression Expression
deriving (ReadPrec [Instruction]
ReadPrec Instruction
Int -> ReadS Instruction
ReadS [Instruction]
(Int -> ReadS Instruction)
-> ReadS [Instruction]
-> ReadPrec Instruction
-> ReadPrec [Instruction]
-> Read Instruction
forall a.
(Int -> ReadS a)
-> ReadS [a] -> ReadPrec a -> ReadPrec [a] -> Read a
$creadsPrec :: Int -> ReadS Instruction
readsPrec :: Int -> ReadS Instruction
$creadList :: ReadS [Instruction]
readList :: ReadS [Instruction]
$creadPrec :: ReadPrec Instruction
readPrec :: ReadPrec Instruction
$creadListPrec :: ReadPrec [Instruction]
readListPrec :: ReadPrec [Instruction]
Read, Int -> Instruction -> ShowS
[Instruction] -> ShowS
Instruction -> String
(Int -> Instruction -> ShowS)
-> (Instruction -> String)
-> ([Instruction] -> ShowS)
-> Show Instruction
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
$cshowsPrec :: Int -> Instruction -> ShowS
showsPrec :: Int -> Instruction -> ShowS
$cshow :: Instruction -> String
show :: Instruction -> String
$cshowList :: [Instruction] -> ShowS
showList :: [Instruction] -> ShowS
Show)
instruction :: P Instruction
instruction :: P Instruction
instruction = ReadS String -> P String
forall a. ReadS a -> P a
P ReadS String
lex P String -> (String -> P Instruction) -> P Instruction
forall a b. P a -> (a -> P b) -> P b
forall (m :: * -> *) a b. Monad m => m a -> (a -> m b) -> m b
>>= \case
String
"snd" -> Expression -> Instruction
Snd (Expression -> Instruction) -> P Expression -> P Instruction
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Expression
expression
String
"rcv" -> Register -> Instruction
Rcv (Register -> Instruction) -> P Register -> P Instruction
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register
String
"set" -> Register -> Expression -> Instruction
Set (Register -> Expression -> Instruction)
-> P Register -> P (Expression -> Instruction)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register P (Expression -> Instruction) -> P Expression -> P Instruction
forall a b. P (a -> b) -> P a -> P b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> P Expression
expression
String
"add" -> Register -> Expression -> Instruction
Add (Register -> Expression -> Instruction)
-> P Register -> P (Expression -> Instruction)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register P (Expression -> Instruction) -> P Expression -> P Instruction
forall a b. P (a -> b) -> P a -> P b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> P Expression
expression
String
"mul" -> Register -> Expression -> Instruction
Mul (Register -> Expression -> Instruction)
-> P Register -> P (Expression -> Instruction)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register P (Expression -> Instruction) -> P Expression -> P Instruction
forall a b. P (a -> b) -> P a -> P b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> P Expression
expression
String
"mod" -> Register -> Expression -> Instruction
Mod (Register -> Expression -> Instruction)
-> P Register -> P (Expression -> Instruction)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register P (Expression -> Instruction) -> P Expression -> P Instruction
forall a b. P (a -> b) -> P a -> P b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> P Expression
expression
String
"jgz" -> Expression -> Expression -> Instruction
Jgz (Expression -> Expression -> Instruction)
-> P Expression -> P (Expression -> Instruction)
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Expression
expression P (Expression -> Instruction) -> P Expression -> P Instruction
forall a b. P (a -> b) -> P a -> P b
forall (f :: * -> *) a b. Applicative f => f (a -> b) -> f a -> f b
<*> P Expression
expression
String
_ -> P Instruction
forall a. P a
forall (f :: * -> *) a. Alternative f => f a
empty
expression :: P Expression
expression :: P Expression
expression =
Register -> Expression
RegisterExpression (Register -> Expression) -> P Register -> P Expression
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> P Register
register P Expression -> P Expression -> P Expression
forall a. P a -> P a -> P a
forall (f :: * -> *) a. Alternative f => f a -> f a -> f a
<|>
Integer -> Expression
IntegerExpression (Integer -> Expression) -> P Integer -> P Expression
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> ReadS Integer -> P Integer
forall a. ReadS a -> P a
P ReadS Integer
forall a. Read a => ReadS a
reads
register :: P Register
register :: P Register
register = [Char -> Register
Register Char
c | [Char
c] <- ReadS String -> P String
forall a. ReadS a -> P a
P ReadS String
lex, Char -> Bool
isAlpha Char
c]