import Data.Char (toUpper) import Data.List.Split (splitOn) import System.Environment (getArgs) preamble = "@256\nD=A\n@SP\nM=D\n" epilogue = "(END)\n@END\n0;JMP" parse' :: [Char] -> [[Char]] -> Int -> [Char] parse' op [] n | op `elem` ["add", "sub", "and", "or"] = pop "M" ++ pop ('M':(f op):"D") ++ push "D" | op `elem` ["eq", "gt", "lt"] = pop "M" ++ pop "M-D" ++ ifThenElse op n | otherwise = pop "M" ++ push (f op:"D") where f "add" = '+'; f "sub" = '-'; f "and" = '&'; f "or" = '|'; f "neg" = '-'; f "not" = '!' parse' "push" ["constant", x] _ = "@" ++ x ++ "\nD=A\n" ++ push "D" push :: [Char] -> [Char] push xs = "@SP\nA=M\nM=" ++ xs ++ "\n@SP\nM=M+1\n" ifThenElse :: [Char] -> Int -> [Char] ifThenElse cond n = "@" ++ cond' ++ show n ++ "\nD;J" ++ cond' ++ "\n" ++ push "0" ++ "@ENDIF" ++ cond' ++ show n ++ "\n0;JMP\n(" ++ cond' ++ show n ++ ")\n" ++ push "-1" ++ "(ENDIF" ++ cond' ++ show n ++ ")\n" where cond' = toUpper <$> cond pop :: [Char] -> [Char] pop xs = "@SP\nAM=M-1\nD=" ++ xs ++ "\n" parseline :: [[Char]] -> Int -> [Char] -> [Char] parseline [] _ acc = acc parseline (line:lines) n acc = parseline lines (n + 1) (acc ++ parse' cmd target n) where cmd:target = words line stripJunk :: [Char] -> [[Char]] stripJunk = filter (not . isEmptyLine) . fmap (head . splitOn "//") . lines . replCrWithNl isEmptyLine :: [Char] -> Bool isEmptyLine = null . filter (not . flip elem " \t") parseCode :: [Char] -> [Char] parseCode xs = preamble ++ (parseline (stripJunk xs) 0 "") ++ epilogue replCrWithNl :: [Char] -> [Char] replCrWithNl = fmap cr2nl where cr2nl '\r' = '\n' cr2nl c = c --input = "push constant 17\npush constant 17\neq" main = do args <- getArgs let filename = head args code <- readFile $ filename writeFile (head (splitOn "." filename) ++ ".asm") (parseCode code) --}