aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/Reducer/HigherOrder.hs
blob: 1699c87bbf149a714c43f18b6bfef99da03d445a (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
-- MIT License, Copyright (c) 2024 Marvin Borner
-- Slightly modified version of Tromp's AIT/Lambda.lhs reducer implementation
module Reducer.HigherOrder
  ( reduce
  ) where

import           Helper

data HigherOrder = HigherOrderBruijn Int | HigherOrderAbstraction (HigherOrder -> HigherOrder) | HigherOrderApplication HigherOrder HigherOrder
data NamedTerm = NamedVariable Int | NamedAbstraction Int NamedTerm | NamedApplication NamedTerm NamedTerm

(!?) :: [a] -> Int -> Maybe a
(!?) []       _ = Nothing
(!?) (x : _ ) 0 = Just x
(!?) (_ : xs) i = xs !? (i - 1)

app :: HigherOrder -> HigherOrder -> HigherOrder
app (HigherOrderAbstraction f) = f
app f                          = HigherOrderApplication f

eval :: Expression -> HigherOrder
eval = go []
 where
  go env (Bruijn x) = case env !? x of
    Just v -> v
    _      -> HigherOrderBruijn x
  go env (Abstraction e    ) = HigherOrderAbstraction $ \x -> go (x : env) e
  go env (Application e1 e2) = app (go env e1) (go env e2)
  go _   _                   = invalidProgramState

toNamedTerm :: HigherOrder -> NamedTerm
toNamedTerm = go 0
 where
  go _ (HigherOrderBruijn i) = NamedVariable i
  go d (HigherOrderAbstraction f) =
    NamedAbstraction d $ go (d + 1) (f (HigherOrderBruijn d))
  go d (HigherOrderApplication e1 e2) = NamedApplication (go d e1) (go d e2)

resolveExpression :: NamedTerm -> Expression
resolveExpression = resolve []
 where
  resolve vs (NamedVariable i) = Bruijn $ case vs !? i of
    Just v -> v
    _      -> i
  resolve vs (NamedAbstraction v t) = Abstraction $ resolve (v : vs) t
  resolve vs (NamedApplication l r) = Application (resolve vs l) (resolve vs r)

reduce :: Expression -> Expression
reduce = resolveExpression . toNamedTerm . eval