Can I get KnownNat n to imply KnownNat (n * 3), etc?

后端 未结 3 1171
灰色年华
灰色年华 2021-01-11 18:14

I\'m working with data types of this shape, using V from linear:

type Foo n = V (n * 3) Double -> Double

Havin

相关标签:
3条回答
  • 2021-01-11 18:38

    I post another answer as it is more direct, editing the previous won't make sense.

    In fact using the trick (popularised if not invented by Edward Kmett), from reflections reifyNat:

    {-# LANGUAGE GADTs #-}
    {-# LANGUAGE DataKinds #-}
    {-# LANGUAGE KindSignatures #-}
    {-# LANGUAGE TypeOperators #-}
    {-# LANGUAGE ScopedTypeVariables #-}
    {-# LANGUAGE FlexibleContexts #-}
    import GHC.TypeLits
    import Data.Proxy
    import Unsafe.Coerce
    
    newtype MagicNat3 r = MagicNat3 (forall (n :: Nat). KnownNat (n * 3) => Proxy n -> r)
    
    trickValue :: Integer -> Integer
    trickValue = (*3)
    
    -- No type-level garantee that the function will be called with (n * 3)
    -- you have to believe us
    trick :: forall a n. KnownNat n => Proxy n -> (forall m. KnownNat (m * 3) => Proxy m -> a) -> a
    trick p f = unsafeCoerce (MagicNat3 f :: MagicNat3 a) (trickValue (natVal p)) Proxy
    
    test :: forall m. KnownNat (m * 3) => Proxy m -> Integer
    test _ = natVal (Proxy :: Proxy (m * 3))
    

    So when you run it:

    λ *Main > :t trick (Proxy :: Proxy 4) test :: Integer
    trick (Proxy :: Proxy 4) test :: Integer :: Integer
    λ *Main > trick (Proxy :: Proxy 4) test :: Integer
    12
    

    The trick is based on the fact that in GHC the one member class dictionaries (like KnownNat) are represented by the member itself. In KnownNat situation it turns out to be Integer. So we just unsafeCoerce it there. Universal quantification makes it sound from the outside.

    0 讨论(0)
  • 2021-01-11 18:45

    Your question isn't very descriptive, so I'll try my best to feel blanks:

    Let's assume that Blah n is Proxy n.

    I also assume that reflectNat is a way to call universally quantified (over typelevel Nat) function, using term-level natural number.

    I don't know better way than writing your own reflectNat providing that

    {-# LANGUAGE RankNTypes #-}
    {-# LANGUAGE GADTs #-}
    {-# LANGUAGE DataKinds #-}
    {-# LANGUAGE KindSignatures #-}
    {-# LANGUAGE TypeOperators #-}
    {-# LANGUAGE ScopedTypeVariables #-}
    {-# LANGUAGE FlexibleContexts #-}
    import GHC.TypeLits
    import Data.Proxy
    
    data Vec a (n :: Nat) where
      Nil  :: Vec a 0
      Cons :: a -> Vec a n -> Vec a (1 + n)
    
    vecToList :: Vec a n -> [a]
    vecToList Nil = []
    vecToList (Cons h t) = h : vecToList t
    
    repl :: forall n a. KnownNat n => Proxy n -> a -> Vec a n
    repl p x = undefined -- this is a bit tricky with Nat from GHC.TypeLits, but possible
    
    foo :: forall (n :: Nat). KnownNat (1 + n) => Proxy n -> Vec Bool (1 + n)
    foo _ = repl (Proxy :: Proxy (1 + n)) True
    
    -- Here we have to write our own version of 'reflectNat' providing right 'KnownNat' instances
    -- so we can call `foo`
    reflectNat :: Integer -> (forall n. KnownNat (1 + n) => Proxy (n :: Nat) -> a) -> a
    reflectNat = undefined
    
    test :: [Bool]
    test = reflectNat 5 $ \p -> vecToList (foo p)
    

    Alternatively, using singletons you can use SomeSing. Then types will be different

    reflectNat :: Integer -> (forall (n :: Nat). SomeSing (n :: Nat) -> a) -> a
    

    I.e. instead of magic dict KnownNat you have concrete singleton value. Thus in foo you'd need to construct SomeSing (1 + n) explicitly, given SomeSing n -- which is quite simple.

    In run-time both KnownNat dictionary and SomeSing value will be passed around carring the number value, and explicit is IMHO better in this situation.p)

    0 讨论(0)
  • 2021-01-11 18:50

    So, three months later, I have been going back and forth on good ways to accomplish this, but I finally settled on an actual very succinct trick that doesn't require any throwaway newtypes; it involves using a Dict from the constraints library; you could easily write a:

    natDict :: KnownNat n => Proxy n -> Dict (KnownNat n)
    natDict _ = Dict
    
    triple :: KnownNat n => Proxy n -> Dict (KnownNat (n * 3))
    triple p = reifyNat (natVal p * 3) $
                 \p3 -> unsafeCoerce (natDict p3)
    

    And once you get Dict (KnownNat (n * 3), you can pattern match on it to get the (n * 3) instance in scope:

    case triple (Proxy :: Proxy n) of
      Dict -> -- KnownNat (n * 3) is in scope
    

    You can actually set these up as generic, too:

    addNats :: (KnownNat n, KnownNat m) => Proxy n -> Proxy m -> Dict (KnownNat (n * m))
    addNats px py = reifyNat (natVal px + natVal py) $
                      \pz -> unsafeCoerce (natDict pz)
    

    Or, you can make them operators and you can use them to "combine" Dicts:

    infixl 6 %+
    infixl 7 %*
    (%+) :: Dict (KnownNat n) -> Dict (KnownNat m) -> Dict (KnownNat (n + m))
    (%*) :: Dict (KnownNat n) -> Dict (KnownNat m) -> Dict (KnownNat (n * m))
    

    And you can do things like:

    case d1 %* d2 %+ d3 of
      Dict -> -- in here, KnownNat (n1 * n2 + n3) is in scope
    

    I've wrapped this up in a nice library, typelits-witnesses that I've been using. Thank you all for your help!

    0 讨论(0)
提交回复
热议问题