{-# LANGUAGE BlockArguments #-}
{-# LANGUAGE TemplateHaskell #-}
module PlutusTx.Optimize.SpaceTime (peel, unroll) where
import Prelude
import Language.Haskell.TH.Syntax qualified as TH
import PlutusTx.Function (fix)
peel
:: forall a b
. Int
-> (TH.Code TH.Q (a -> b) -> TH.Code TH.Q (a -> b))
-> TH.Code TH.Q (a -> b)
peel :: forall a b.
Int -> (Code Q (a -> b) -> Code Q (a -> b)) -> Code Q (a -> b)
peel Int
0 Code Q (a -> b) -> Code Q (a -> b)
f = [||((a -> b) -> a -> b) -> a -> b
forall a b. ((a -> b) -> a -> b) -> a -> b
fix \a -> b
self -> $$(Code Q (a -> b) -> Code Q (a -> b)
f [||a -> b
self||])||]
peel Int
n Code Q (a -> b) -> Code Q (a -> b)
f
| Int
n Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
> Int
0 = Code Q (a -> b) -> Code Q (a -> b)
f (Int -> (Code Q (a -> b) -> Code Q (a -> b)) -> Code Q (a -> b)
forall a b.
Int -> (Code Q (a -> b) -> Code Q (a -> b)) -> Code Q (a -> b)
peel (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) Code Q (a -> b) -> Code Q (a -> b)
f)
| Bool
otherwise = [Char] -> Code Q (a -> b)
forall a. HasCallStack => [Char] -> a
error ([Char] -> Code Q (a -> b)) -> [Char] -> Code Q (a -> b)
forall a b. (a -> b) -> a -> b
$ [Char]
"PlutusTx.Optimize.SpaceTime.peel: negative n: " [Char] -> [Char] -> [Char]
forall a. Semigroup a => a -> a -> a
<> Int -> [Char]
forall a. Show a => a -> [Char]
show Int
n
unroll
:: forall a b
. Int
-> (TH.Code TH.Q (a -> b) -> TH.Code TH.Q (a -> b))
-> TH.Code TH.Q (a -> b)
unroll :: forall a b.
Int -> (Code Q (a -> b) -> Code Q (a -> b)) -> Code Q (a -> b)
unroll Int
n Code Q (a -> b) -> Code Q (a -> b)
f = [||((a -> b) -> a -> b) -> a -> b
forall a b. ((a -> b) -> a -> b) -> a -> b
fix \a -> b
self -> $$(Int
-> (Code Q (a -> b) -> Code Q (a -> b))
-> Code Q (a -> b)
-> Code Q (a -> b)
forall a. Int -> (a -> a) -> a -> a
nTimes Int
n Code Q (a -> b) -> Code Q (a -> b)
f [||a -> b
self||])||]
nTimes :: Int -> (a -> a) -> (a -> a)
nTimes :: forall a. Int -> (a -> a) -> a -> a
nTimes Int
0 a -> a
_ = a -> a
forall a. a -> a
id
nTimes Int
1 a -> a
f = a -> a
f
nTimes Int
n a -> a
f = a -> a
f (a -> a) -> (a -> a) -> a -> a
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Int -> (a -> a) -> a -> a
forall a. Int -> (a -> a) -> a -> a
nTimes (Int
n Int -> Int -> Int
forall a. Num a => a -> a -> a
- Int
1) a -> a
f