Skip to content

Commit

Permalink
Parse where clause into the corresponding let (fix #209)
Browse files Browse the repository at this point in the history
  • Loading branch information
ulysses4ever committed Feb 1, 2024
1 parent a04e531 commit 1a688e3
Show file tree
Hide file tree
Showing 4 changed files with 52 additions and 13 deletions.
25 changes: 25 additions & 0 deletions gibbon-compiler/examples/ParseWhere.hs
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
-- Test that the where clause parses and evaluates.
-- Three places you can see where:
-- - top-level function definition
-- - let binding
-- - case alternative

module ParseWhere where

data MyBool = B Bool

f x = y + case B True of
B _ -> z
where z = 0
where
y = x+1

g x = let
y = z
where z = -1
in x + y

gibbon_main = f 1 + g 1

main :: IO ()
main = print gibbon_main
1 change: 1 addition & 0 deletions gibbon-compiler/examples/ParseWhere.hs.ans
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
2
36 changes: 23 additions & 13 deletions gibbon-compiler/src/Gibbon/HaskellFrontend.hs
Original file line number Diff line number Diff line change
Expand Up @@ -881,7 +881,7 @@ desugarExp type_syns toplevel e =
desugarFun :: (Show a, Pretty a) => TypeSynEnv -> TopTyEnv -> TopTyEnv -> Decl a -> PassM (Var, [Var], TyScheme, Exp0)
desugarFun type_syns toplevel env decl =
case decl of
FunBind _ [Match _ fname pats (UnGuardedRhs _ bod) _where] -> do
FunBind _ [Match _ fname pats (UnGuardedRhs _ bod) mbwhere] -> do
let fname_str = nameToStr fname
fname_var = toVar (fname_str)
(vars, arg_tys,bindss) <- unzip3 <$> mapM (desugarPatWithTy type_syns) pats
Expand All @@ -893,9 +893,21 @@ desugarFun type_syns toplevel env decl =
let funty = ArrowTy arg_tys ret_ty
pure $ (ForAll [] funty)
Just ty -> pure ty
bod' <- desugarExp type_syns toplevel bod
bod' <- desugarExp type_syns toplevel (whereToLet mbwhere bod)
pure $ (fname_var, args, unCurryTopTy fun_ty, (mkLets binds bod'))
_ -> error $ "desugarFun: Found a function with multiple RHS, " ++ prettyPrint decl
_ -> error $ "desugarFun: Found a function with multiple RHS or guards (currently unsupported), "
++ prettyPrint decl

type WhereGroup l = Maybe (H.Binds l)

{- | Turn an expression depending on the given where bindings into a let-expression
with the same bindings. E.g. '(x+1, x=1)' is turned into 'let x=1 in x+1'. If the where bindings are a `Nothing`, this is the identity function.
-}
whereToLet :: WhereGroup l -> Exp l -> Exp l
whereToLet Nothing body = body
whereToLet (Just where_binds) body = case where_binds of
(BDecls l _) -> Let l where_binds body
_ -> error "whereToLet: implicit parameters in binding group"

multiArgsToOne :: [Var] -> [Ty0] -> Exp0 -> (Var, Exp0)
multiArgsToOne args tys ex =
Expand Down Expand Up @@ -1060,13 +1072,12 @@ desugarOp qop =
desugarAlt :: (Show a, Pretty a) => TypeSynEnv -> TopTyEnv -> Alt a -> PassM (DataCon, [(Var,Ty0)], Exp0)
desugarAlt type_syns toplevel alt =
case alt of
Alt _ (PApp _ qname ps) (UnGuardedRhs _ rhs) Nothing -> do
Alt _ (PApp _ qname ps) (UnGuardedRhs _ rhs) mbwhere -> do
let conName = qnameToStr qname
desugarCase ps conName rhs
Alt _ (PWildCard _) (UnGuardedRhs _ rhs) _b ->
desugarCase [] "_default" rhs
desugarCase ps conName (whereToLet mbwhere rhs)
Alt _ (PWildCard _) (UnGuardedRhs _ rhs) mbwhere ->
desugarCase [] "_default" (whereToLet mbwhere rhs)
Alt _ _ GuardedRhss{} _ -> error "desugarExp: Guarded RHS not supported in case."
Alt _ _ _ Just{} -> error "desugarExp: Where clauses not allowed in case."
Alt _ pat _ _ -> error $ "desugarExp: Unsupported pattern in case: " ++ prettyPrint pat
where
desugarCase ps conName rhs = do
Expand All @@ -1086,18 +1097,17 @@ generateBind type_syns toplevel env decl exp2 =
TypeSig{} -> pure exp2
-- 'collectTypeSynonyms' takes care of this.
TypeDecl{} -> pure exp2
PatBind _ _ _ Just{} -> error "generateBind: where clauses not allowed"
PatBind _ _ GuardedRhss{} _ -> error "generateBind: Guarded right hand side not supported."
PatBind _ (PTuple _ Boxed pats) (UnGuardedRhs _ rhs) Nothing -> do
rhs' <- desugarExp type_syns toplevel rhs
PatBind _ (PTuple _ Boxed pats) (UnGuardedRhs _ rhs) mbwhere -> do
rhs' <- desugarExp type_syns toplevel (whereToLet mbwhere rhs)
w <- gensym "tup"
ty' <- newMetaTy
let tupexp e = LetE (w,[],ty',rhs') e
binds = reverse $ zip pats [0..]
prjexp <- generateTupleProjs toplevel env binds (VarE w) exp2
pure $ tupexp prjexp
PatBind _ pat (UnGuardedRhs _ rhs) Nothing -> do
rhs' <- desugarExp type_syns toplevel rhs
PatBind _ pat (UnGuardedRhs _ rhs) mbwhere -> do
rhs' <- desugarExp type_syns toplevel (whereToLet mbwhere rhs)
w <- case pat of
PVar _ v -> pure $ toVar (nameToStr v)
PWildCard _ -> gensym "wildcard_"
Expand Down
3 changes: 3 additions & 0 deletions gibbon-compiler/tests/test-gibbon-examples.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -358,6 +358,9 @@ tests:
# gibbon3 and gibbon2: Inferlocations bug.
failing: [interp1,gibbon1,pointer]

- name: ParseWhere.hs
answer-file: examples/ParseWhere.hs.ans

## FAILING: Multiple packed outputs.
- name: TupleTest.hs
dir: examples/poly
Expand Down

0 comments on commit 1a688e3

Please sign in to comment.