| @ -0,0 +1,2 @@ | |||
| /.stack-work | |||
| /.vscode | |||
| @ -0,0 +1,30 @@ | |||
| Copyright Abigail Magalhães (c) 2021 | |||
| All rights reserved. | |||
| Redistribution and use in source and binary forms, with or without | |||
| modification, are permitted provided that the following conditions are met: | |||
| * Redistributions of source code must retain the above copyright | |||
| notice, this list of conditions and the following disclaimer. | |||
| * Redistributions in binary form must reproduce the above | |||
| copyright notice, this list of conditions and the following | |||
| disclaimer in the documentation and/or other materials provided | |||
| with the distribution. | |||
| * Neither the name of Abigail Magalhães nor the names of other | |||
| contributors may be used to endorse or promote products derived | |||
| from this software without specific prior written permission. | |||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |||
| "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |||
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |||
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |||
| OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |||
| SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |||
| LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |||
| DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |||
| THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |||
| (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |||
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |||
| @ -0,0 +1 @@ | |||
| # layout | |||
| @ -0,0 +1,2 @@ | |||
| import Distribution.Simple | |||
| main = defaultMain | |||
| @ -0,0 +1,2 @@ | |||
| cradle: | |||
| stack: | |||
| @ -0,0 +1,27 @@ | |||
| name: layout | |||
| version: 0.1.0.0 | |||
| -- synopsis: | |||
| -- description: | |||
| homepage: https://github.com/plt-hokusai/layout#readme | |||
| license: BSD3 | |||
| license-file: LICENSE | |||
| author: Abigail Magalhães | |||
| maintainer: [email protected] | |||
| copyright: 2021 Abigail Magalhães | |||
| category: Web | |||
| build-type: Simple | |||
| cabal-version: >=1.10 | |||
| extra-source-files: README.md | |||
| executable layout | |||
| hs-source-dirs: src | |||
| main-is: Main.hs | |||
| default-language: Haskell2010 | |||
| build-depends: base >= 4.7 && < 5 | |||
| , array >= 0.5 && < 0.6 | |||
| , mtl | |||
| build-tool-depends: alex:alex >= 3.2.4 && < 4.0 | |||
| , happy:happy >= 1.19.12 && < 2.0 | |||
| other-modules: Syntax, Lexer, Lexer.Support, Parser | |||
| @ -0,0 +1,130 @@ | |||
| { | |||
| module Lexer where | |||
| import Control.Monad.State | |||
| import Control.Monad.Error | |||
| import Lexer.Support | |||
| import Debug.Trace | |||
| } | |||
| %encoding "latin1" | |||
| $lower = [ a-z ] | |||
| $upper = [ A-Z ] | |||
| @ident = $lower [ $lower $upper _ ' ]* | |||
| :- | |||
| [\ \t]+ ; | |||
| <0> "in" { token TkIn } | |||
| <0> "let" { layoutKw TkLet } | |||
| <0> "where" { layoutKw TkWhere } | |||
| <0> @ident { emit TkIdent } | |||
| <0> \\ { token TkBackslash } | |||
| <0> "->" { token TkArrow } | |||
| <0> \= { token TkEqual } | |||
| <0> \( { token TkLParen } | |||
| <0> \) { token TkRParen } | |||
| <0> \{ { token TkOpen } | |||
| <0> \} { token TkClose } | |||
| <0> "--" .* \n { \_ -> pushStartCode newline *> scan } | |||
| <0> \n { \_ -> pushStartCode newline *> scan } | |||
| <layout> { | |||
| -- Skip comments and whitespace | |||
| "--" .* \n ; | |||
| \n ; | |||
| \{ { openBrace } | |||
| () { startLayout } | |||
| } | |||
| <empty_layout> () { emptyLayout } | |||
| <newline> { | |||
| \n ; | |||
| "--" .* \n ; | |||
| () { offsideRule } | |||
| } | |||
| <eof> () { doEOF } | |||
| { | |||
| handleEOF = pushStartCode eof *> scan | |||
| doEOF _ = do | |||
| t <- Lexer.Support.layout | |||
| case t of | |||
| Nothing -> do | |||
| popStartCode | |||
| pure TkEOF | |||
| _ -> do | |||
| popLayout | |||
| pure TkVClose | |||
| scan :: Lexer Token | |||
| scan = do | |||
| input@(Input _ _ _ string) <- gets lexerInput | |||
| startcode <- startCode | |||
| case alexScan input startcode of | |||
| AlexEOF -> handleEOF | |||
| AlexError (Input _ _ _ inp) -> | |||
| throwError $ "Lexical error: " ++ show (head inp) | |||
| AlexSkip input' _ -> do | |||
| modify' $ \s -> s { lexerInput = input' } | |||
| scan | |||
| AlexToken input' tokl action -> do | |||
| modify' $ \s -> s { lexerInput = input' } | |||
| action (take tokl string) | |||
| layoutKw t _ = do | |||
| pushStartCode Lexer.layout | |||
| pure t | |||
| openBrace _ = do | |||
| popStartCode | |||
| pushLayout ExplicitLayout | |||
| pure TkOpen | |||
| startLayout _ = do | |||
| popStartCode | |||
| reference <- Lexer.Support.layout | |||
| col <- gets (inpColumn . lexerInput) | |||
| if Just (LayoutColumn col) <= reference | |||
| then pushStartCode empty_layout | |||
| else pushLayout (LayoutColumn col) | |||
| pure TkVOpen | |||
| emptyLayout _ = do | |||
| popStartCode | |||
| pushStartCode newline | |||
| pure TkVClose | |||
| offsideRule _ = do | |||
| context <- Lexer.Support.layout | |||
| col <- gets (inpColumn . lexerInput) | |||
| let continue = popStartCode *> scan | |||
| case context of | |||
| Just (LayoutColumn col') -> do | |||
| case col `compare` col' of | |||
| EQ -> do | |||
| popStartCode | |||
| pure TkVSemi | |||
| GT -> continue | |||
| LT -> do | |||
| popLayout | |||
| pure TkVClose | |||
| _ -> continue | |||
| } | |||
| @ -0,0 +1,114 @@ | |||
| {-# LANGUAGE DeriveFunctor #-} | |||
| {-# LANGUAGE GeneralizedNewtypeDeriving #-} | |||
| module Lexer.Support where | |||
| import Data.Word | |||
| import Data.List (uncons) | |||
| import Data.Char (ord) | |||
| import Control.Monad.State | |||
| import Data.List.NonEmpty (NonEmpty ((:|))) | |||
| import qualified Data.List.NonEmpty as NE | |||
| import Control.Monad.Except | |||
| data Token | |||
| = TkIdent String -- identifiers | |||
| -- Keywords | |||
| | TkLet | TkIn | TkWhere | |||
| -- Punctuation | |||
| | TkEqual | TkOpen | TkSemi | TkClose | |||
| | TkLParen | TkRParen | |||
| | TkBackslash | TkArrow | |||
| -- Layout punctuation | |||
| | TkVOpen | TkVSemi | TkVClose | |||
| | TkEOF | |||
| deriving (Eq, Show) | |||
| data AlexInput | |||
| = Input { inpLine :: {-# UNPACK #-} !Int | |||
| , inpColumn :: {-# UNPACK #-} !Int | |||
| , inpLast :: {-# UNPACK #-} !Char | |||
| , inpStream :: String | |||
| } | |||
| deriving (Eq, Show) | |||
| alexPrevInputChar :: AlexInput -> Char | |||
| alexPrevInputChar = inpLast | |||
| alexGetByte :: AlexInput -> Maybe (Word8, AlexInput) | |||
| alexGetByte inp@Input{inpStream = str} = advance <$> uncons str where | |||
| advance ('\n', rest) = | |||
| ( fromIntegral (ord '\n') | |||
| , Input { inpLine = inpLine inp + 1 | |||
| , inpColumn = 0 | |||
| , inpLast = '\n' | |||
| , inpStream = rest } | |||
| ) | |||
| advance (c, rest) = | |||
| ( fromIntegral (ord c) | |||
| , Input { inpLine = inpLine inp | |||
| , inpColumn = inpColumn inp + 1 | |||
| , inpLast = c | |||
| , inpStream = rest } | |||
| ) | |||
| newtype Lexer a = Lexer { _getLexer :: StateT LexerState (Either String) a } | |||
| deriving (Functor, Applicative, Monad, MonadState LexerState, MonadError String) | |||
| data Layout = ExplicitLayout | LayoutColumn Int | |||
| deriving (Eq, Show, Ord) | |||
| data LexerState | |||
| = LS { lexerInput :: {-# UNPACK #-} !AlexInput | |||
| , lexerStartCodes :: {-# UNPACK #-} !(NonEmpty Int) | |||
| , lexerLayout :: [Layout] | |||
| } | |||
| deriving (Eq, Show) | |||
| startCode :: Lexer Int | |||
| startCode = gets (NE.head . lexerStartCodes) | |||
| pushStartCode :: Int -> Lexer () | |||
| pushStartCode i = modify' $ \st -> | |||
| st { lexerStartCodes = NE.cons i (lexerStartCodes st) | |||
| } | |||
| popStartCode :: Lexer () | |||
| popStartCode = modify' $ \st -> | |||
| st { lexerStartCodes = | |||
| case lexerStartCodes st of | |||
| _ :| [] -> 0 :| [] | |||
| _ :| (x:xs) -> x :| xs | |||
| } | |||
| layout :: Lexer (Maybe Layout) | |||
| layout = gets (fmap fst . uncons . lexerLayout) | |||
| pushLayout :: Layout -> Lexer () | |||
| pushLayout i = modify' $ \st -> | |||
| st { lexerLayout = i:lexerLayout st } | |||
| popLayout :: Lexer () | |||
| popLayout = modify' $ \st -> | |||
| st { lexerLayout = | |||
| case lexerLayout st of | |||
| _:xs -> xs | |||
| [] -> [] | |||
| } | |||
| initState :: String -> LexerState | |||
| initState str = LS { lexerInput = Input 0 1 '\n' str | |||
| , lexerStartCodes = 0 :| [] | |||
| , lexerLayout = [] | |||
| } | |||
| emit :: (String -> Token) -> String -> Lexer Token | |||
| emit = (pure .) | |||
| token :: Token -> String -> Lexer Token | |||
| token = const . pure | |||
| runLexer :: Lexer a -> String -> Either String a | |||
| runLexer act s = fst <$> runStateT (_getLexer act) (initState s) | |||
| @ -0,0 +1,19 @@ | |||
| module Main where | |||
| import Lexer.Support | |||
| import Lexer | |||
| import Debug.Trace (traceM) | |||
| import Control.Monad.RWS | |||
| main :: IO () | |||
| main = do | |||
| putStrLn "hello world" | |||
| lexAll :: Lexer () | |||
| lexAll = do | |||
| tok <- scan | |||
| case tok of | |||
| TkEOF -> pure () | |||
| x -> do | |||
| traceM (show x) | |||
| lexAll | |||
| @ -0,0 +1,81 @@ | |||
| { | |||
| module Parser where | |||
| import Control.Monad.Error | |||
| import Lexer.Support | |||
| import Syntax | |||
| import Lexer (scan) | |||
| } | |||
| %name parseExpr Expr | |||
| %name parseDecl Decl | |||
| %tokentype { Token } | |||
| %monad { Lexer } | |||
| %lexer { lexer } { TkEOF } | |||
| %errorhandlertype explist | |||
| %error { parseError } | |||
| %token | |||
| VAR { TkIdent $$ } | |||
| 'let' { TkLet } | |||
| 'in' { TkIn } | |||
| 'where' { TkWhere } | |||
| '=' { TkEqual } | |||
| '{' { TkOpen } | |||
| ';' { TkSemi } | |||
| '}' { TkClose } | |||
| '\\' { TkBackslash } | |||
| '->' { TkArrow } | |||
| '(' { TkLParen } | |||
| ')' { TkRParen } | |||
| OPEN { TkVOpen } | |||
| SEMI { TkVSemi } | |||
| CLOSE { TkVClose } | |||
| %% | |||
| Atom :: { Expr } | |||
| : VAR { Var $1 } | |||
| | '(' Expr ')' { $2 } | |||
| Expr :: { Expr } | |||
| : '\\' VAR '->' Expr { Lam $2 $4 } | |||
| | 'let' DeclBlock 'in' Expr { Let $2 $4 } | |||
| | FuncExpr { $1 } | |||
| FuncExpr :: { Expr } | |||
| : FuncExpr Atom { App $1 $2 } | |||
| | Atom { $1 } | |||
| DeclBlock :: { [Decl] } | |||
| : '{' DeclListSemi '}' { $2 } | |||
| | OPEN DeclListSEMI Close { $2 } | |||
| DeclListSemi :: { [Decl] } | |||
| : Decl ';' DeclListSemi { $1:$3 } | |||
| | Decl { [$1] } | |||
| | {- empty -} { [] } | |||
| DeclListSEMI :: { [Decl] } | |||
| : Decl SEMI DeclListSemi { $1:$3 } | |||
| | Decl { [$1] } | |||
| | {- empty -} { [] } | |||
| Close | |||
| : CLOSE { () } | |||
| | error {% popLayout } | |||
| Decl | |||
| : VAR '=' Expr { Decl $1 $3 Nothing } | |||
| | VAR '=' Expr 'where' DeclBlock { Decl $1 $3 (Just $5) } | |||
| { | |||
| lexer cont = scan >>= cont | |||
| parseError = throwError . show | |||
| } | |||
| @ -0,0 +1,17 @@ | |||
| module Syntax (Expr(..), Decl(..), Program) where | |||
| data Expr | |||
| = Var String | |||
| | App Expr Expr | |||
| | Lam String Expr | |||
| | Let [Decl] Expr | |||
| deriving (Eq, Show) | |||
| data Decl | |||
| = Decl { declName :: String | |||
| , declRhs :: Expr | |||
| , declWhere :: Maybe [Decl] | |||
| } | |||
| deriving (Eq, Show) | |||
| type Program = [Decl] | |||
| @ -0,0 +1,67 @@ | |||
| # This file was automatically generated by 'stack init' | |||
| # | |||
| # Some commonly used options have been documented as comments in this file. | |||
| # For advanced use and comprehensive documentation of the format, please see: | |||
| # https://docs.haskellstack.org/en/stable/yaml_configuration/ | |||
| # Resolver to choose a 'specific' stackage snapshot or a compiler version. | |||
| # A snapshot resolver dictates the compiler version and the set of packages | |||
| # to be used for project dependencies. For example: | |||
| # | |||
| # resolver: lts-3.5 | |||
| # resolver: nightly-2015-09-21 | |||
| # resolver: ghc-7.10.2 | |||
| # | |||
| # The location of a snapshot can be provided as a file or url. Stack assumes | |||
| # a snapshot provided as a file might change, whereas a url resource does not. | |||
| # | |||
| # resolver: ./custom-snapshot.yaml | |||
| # resolver: https://example.com/snapshots/2018-01-01.yaml | |||
| resolver: | |||
| url: https://raw.githubusercontent.com/commercialhaskell/stackage-snapshots/master/lts/18/4.yaml | |||
| # User packages to be built. | |||
| # Various formats can be used as shown in the example below. | |||
| # | |||
| # packages: | |||
| # - some-directory | |||
| # - https://example.com/foo/bar/baz-0.0.2.tar.gz | |||
| # subdirs: | |||
| # - auto-update | |||
| # - wai | |||
| packages: | |||
| - . | |||
| # Dependency packages to be pulled from upstream that are not in the resolver. | |||
| # These entries can reference officially published versions as well as | |||
| # forks / in-progress versions pinned to a git hash. For example: | |||
| # | |||
| # extra-deps: | |||
| # - acme-missiles-0.3 | |||
| # - git: https://github.com/commercialhaskell/stack.git | |||
| # commit: e7b331f14bcffb8367cd58fbfc8b40ec7642100a | |||
| # | |||
| # extra-deps: [] | |||
| # Override default flag values for local packages and extra-deps | |||
| # flags: {} | |||
| # Extra package databases containing global packages | |||
| # extra-package-dbs: [] | |||
| # Control whether we use the GHC we find on the path | |||
| # system-ghc: true | |||
| # | |||
| # Require a specific version of stack, using version ranges | |||
| # require-stack-version: -any # Default | |||
| # require-stack-version: ">=2.7" | |||
| # | |||
| # Override the architecture used by stack, especially useful on Windows | |||
| # arch: i386 | |||
| # arch: x86_64 | |||
| # | |||
| # Extra directories used by stack for building | |||
| # extra-include-dirs: [/path/to/dir] | |||
| # extra-lib-dirs: [/path/to/dir] | |||
| # | |||
| # Allow a newer minor version of GHC than the snapshot specifies | |||
| # compiler-check: newer-minor | |||
| @ -0,0 +1,13 @@ | |||
| # This file was autogenerated by Stack. | |||
| # You should not edit this file by hand. | |||
| # For more information, please see the documentation at: | |||
| # https://docs.haskellstack.org/en/stable/lock_files | |||
| packages: [] | |||
| snapshots: | |||
| - completed: | |||
| size: 585817 | |||
| url: https://raw.githubusercontent.com/commercialhaskell/stackage-snapshots/master/lts/18/4.yaml | |||
| sha256: ea3a318eafa9e9cc56bfbe46099fd0d54d32641ab7bbe1d182ed8f5de39f804c | |||
| original: | |||
| url: https://raw.githubusercontent.com/commercialhaskell/stackage-snapshots/master/lts/18/4.yaml | |||
| @ -0,0 +1,14 @@ | |||
| Right | |||
| (Decl { declName = "foo" | |||
| , declRhs = | |||
| Let [ Decl { declName = "x" | |||
| , declRhs = | |||
| Let | |||
| [ Decl {declName = "y", declRhs = Var "z", declWhere = Nothing} ] | |||
| (Var "y") | |||
| , declWhere = Nothing | |||
| } | |||
| ] | |||
| (Var "x") | |||
| , declWhere = Nothing | |||
| }) | |||