@ -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 | |||
}) |