aboutsummaryrefslogtreecommitdiff
path: root/lllars/parser.hs
diff options
context:
space:
mode:
authorMarvin Borner2024-12-27 16:39:41 +0100
committerMarvin Borner2024-12-27 16:39:41 +0100
commitd6210473f0a4c837c5299722d6f63fa5125cd32c (patch)
tree7fc4a2551db24298de3c6b35e61a2f67cf5e47b5 /lllars/parser.hs
parent36c2ca51e07cffa180226f41a950ff86f47d1c3e (diff)
parser
Co-Authored-By: AnyUnderstanding <christopher.alan.w@gmail.com>
Diffstat (limited to 'lllars/parser.hs')
-rw-r--r--lllars/parser.hs90
1 files changed, 90 insertions, 0 deletions
diff --git a/lllars/parser.hs b/lllars/parser.hs
new file mode 100644
index 0000000..88f9951
--- /dev/null
+++ b/lllars/parser.hs
@@ -0,0 +1,90 @@
+import Data.Functor ( ($>) )
+import Data.Void
+import Text.Megaparsec hiding ( Label
+ , Pos
+ , label
+ )
+import Text.Megaparsec.Char
+import qualified Text.Megaparsec.Char.Lexer as L
+
+type Parser = Parsec Void String
+type Program = [Instr]
+
+type Address = Int
+type Label = String
+
+data Access = Access Address | SAccess Access
+ deriving Show
+data Operation = ADD | MUL | SUB | DIV | AND | OR | XOR
+ deriving Show
+data Addressation = Address Access | BinaryOperation Access Operation Access
+ deriving Show
+data Call = WriteCall | ReadCall
+ deriving Show
+data BranchPolarity = IfTrue | IfFalse
+ deriving Show
+data Instr = Comment String | Write Address Addressation | LarsCall Call | Label Label | GoTo Label | Branch BranchPolarity Address Label
+ deriving Show
+
+comment :: Parser Instr
+comment =
+ Comment <$> (some (string "lars") *> many (satisfy $ not . (== '\n')))
+
+access :: Parser Access
+access = (SAccess <$> (string "sral" *> access)) <|> (Access <$> L.decimal)
+
+binaryOperation :: Parser Operation
+binaryOperation =
+ (char '+' $> ADD)
+ <|> (char '-' $> SUB)
+ <|> (char '*' $> MUL)
+ <|> (char '/' $> DIV)
+
+addressation :: Parser Addressation
+addressation =
+ try (BinaryOperation <$> access <*> binaryOperation <*> access)
+ <|> (Address <$> access)
+
+-- TODO: arguments
+call :: Parser Instr
+call =
+ LarsCall
+ <$> ( string "larssral"
+ *> ((string "lars" $> ReadCall) <|> (string "sral" $> WriteCall))
+ )
+
+write :: Parser Instr
+write = do
+ target <- L.decimal
+ string "lars"
+ source <- addressation
+ return $ Write target source
+
+label :: Parser Label
+label = char '@' *> (concat <$> some (string "lars" <|> string "sral"))
+
+namedLabel :: Parser Instr
+namedLabel = Label <$> label
+
+goto :: Parser Instr
+goto = GoTo <$> (string "sralllars " *> label)
+
+branch :: Parser Instr
+branch =
+ Branch
+ <$> ((string "lars|sral" $> IfTrue) <|> (string "sral|lars" $> IfFalse))
+ <*> L.decimal
+ <*> label
+
+instr :: Parser Instr
+instr = comment <|> write <|> call <|> namedLabel <|> goto <|> branch
+
+program :: Parser Program
+program = sepEndBy instr (some $ char '\n')
+
+main :: IO ()
+main = do
+ f <- readFile "lars.lars"
+ case runParser (program <* eof) "" f of
+ Right res -> print res
+ Left err -> putStrLn $ errorBundlePretty err