2005-09-03 04:45:56 +00:00
|
|
|
{-# OPTIONS -fglasgow-exts #-}
|
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-- |
|
|
|
|
-- Module : Data.Dynamic
|
|
|
|
-- Copyright : (c) The University of Glasgow 2001
|
|
|
|
-- License : BSD-style (see the file libraries/base/LICENSE)
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-- Maintainer : libraries@haskell.org
|
|
|
|
-- Stability : experimental
|
|
|
|
-- Portability : portable
|
|
|
|
--
|
|
|
|
-- The Dynamic interface provides basic support for dynamic types.
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-- Operations for injecting values of arbitrary type into
|
|
|
|
-- a dynamically typed value, Dynamic, are provided, together
|
|
|
|
-- with operations for converting dynamic values into a concrete
|
|
|
|
-- (monomorphic) type.
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-----------------------------------------------------------------------------
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 03:24:30 +00:00
|
|
|
module AltData.Dynamic (
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
-- Module Data.Typeable re-exported for convenience
|
2005-05-19 03:24:30 +00:00
|
|
|
module AltData.Typeable,
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
-- * The @Dynamic@ type
|
|
|
|
Dynamic, -- abstract, instance of: Show, Typeable
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
-- * Converting to and from @Dynamic@
|
|
|
|
toDyn, -- :: Typeable a => a -> Dynamic
|
|
|
|
fromDyn, -- :: Typeable a => Dynamic -> a -> a
|
|
|
|
fromDynamic, -- :: Typeable a => Dynamic -> Maybe a
|
|
|
|
|
2005-07-06 05:22:54 +00:00
|
|
|
#if __GLASGOW_HASKELL__ >= 603
|
2005-05-19 02:09:02 +00:00
|
|
|
-- * Applying functions of dynamic type
|
|
|
|
dynApply,
|
|
|
|
dynApp,
|
|
|
|
dynTypeRep
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-07-06 05:22:54 +00:00
|
|
|
#endif
|
2005-05-19 02:09:02 +00:00
|
|
|
) where
|
2005-04-24 08:51:33 +00:00
|
|
|
|
|
|
|
import AltData.Typeable
|
|
|
|
import Data.Maybe
|
|
|
|
|
2005-05-19 03:24:30 +00:00
|
|
|
import System.IO.Unsafe (unsafePerformIO)
|
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
import GHC.Base
|
|
|
|
import GHC.Show
|
2005-07-06 05:22:54 +00:00
|
|
|
#if __GLASGOW_HASKELL__ >= 603
|
2005-05-19 02:09:02 +00:00
|
|
|
import GHC.Err
|
2005-07-06 05:22:54 +00:00
|
|
|
#endif
|
2005-05-19 02:09:02 +00:00
|
|
|
|
|
|
|
unsafeCoerce :: a -> b
|
|
|
|
unsafeCoerce = unsafeCoerce#
|
|
|
|
|
|
|
|
-------------------------------------------------------------
|
|
|
|
--
|
|
|
|
-- The type Dynamic
|
|
|
|
--
|
|
|
|
-------------------------------------------------------------
|
|
|
|
|
|
|
|
{-|
|
|
|
|
A value of type 'Dynamic' is an object encapsulated together with its type.
|
|
|
|
|
|
|
|
A 'Dynamic' may only represent a monomorphic value; an attempt to
|
|
|
|
create a value of type 'Dynamic' from a polymorphically-typed
|
|
|
|
expression will result in an ambiguity error (see 'toDyn').
|
|
|
|
|
|
|
|
'Show'ing a value of type 'Dynamic' returns a pretty-printed representation
|
|
|
|
of the object\'s type; useful for debugging.
|
|
|
|
-}
|
2005-04-24 08:51:33 +00:00
|
|
|
data Dynamic = Dynamic TypeRep Obj
|
|
|
|
|
2005-05-19 03:24:30 +00:00
|
|
|
instance Typeable Dynamic where
|
|
|
|
#if __GLASGOW_HASKELL__ >= 603
|
|
|
|
typeOf _ = mkTyConApp (mkTyCon "AltData.Dynamic") []
|
|
|
|
#else
|
|
|
|
typeOf _ = mkAppTy (mkTyCon "AltData.Dynamic") []
|
|
|
|
#endif
|
2005-04-24 08:51:33 +00:00
|
|
|
|
|
|
|
instance Show Dynamic where
|
|
|
|
-- the instance just prints the type representation.
|
|
|
|
showsPrec _ (Dynamic t _) =
|
2005-05-19 02:09:02 +00:00
|
|
|
showString "<<" .
|
2005-04-24 08:51:33 +00:00
|
|
|
showsPrec 0 t .
|
2005-05-19 02:09:02 +00:00
|
|
|
showString ">>"
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
type Obj = forall a . a
|
|
|
|
-- Dummy type to hold the dynamically typed value.
|
|
|
|
--
|
|
|
|
-- In GHC's new eval/apply execution model this type must
|
|
|
|
-- be polymorphic. It can't be a constructor, because then
|
|
|
|
-- GHC will use the constructor convention when evaluating it,
|
|
|
|
-- and this will go wrong if the object is really a function. On
|
|
|
|
-- the other hand, if we use a polymorphic type, GHC will use
|
|
|
|
-- a fallback convention for evaluating it that works for all types.
|
|
|
|
-- (using a function type here would also work).
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
-- | Converts an arbitrary value into an object of type 'Dynamic'.
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-- The type of the object must be an instance of 'Typeable', which
|
|
|
|
-- ensures that only monomorphically-typed objects may be converted to
|
|
|
|
-- 'Dynamic'. To convert a polymorphic object into 'Dynamic', give it
|
|
|
|
-- a monomorphic type signature. For example:
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
-- > toDyn (id :: Int -> Int)
|
2005-04-24 08:51:33 +00:00
|
|
|
--
|
2005-05-19 02:09:02 +00:00
|
|
|
toDyn :: Typeable a => a -> Dynamic
|
|
|
|
toDyn v = Dynamic (typeOf v) (unsafeCoerce v)
|
|
|
|
|
|
|
|
-- | Converts a 'Dynamic' object back into an ordinary Haskell value of
|
|
|
|
-- the correct type. See also 'fromDynamic'.
|
|
|
|
fromDyn :: Typeable a
|
|
|
|
=> Dynamic -- ^ the dynamically-typed object
|
|
|
|
-> a -- ^ a default value
|
|
|
|
-> a -- ^ returns: the value of the first argument, if
|
|
|
|
-- it has the correct type, otherwise the value of
|
|
|
|
-- the second argument.
|
|
|
|
fromDyn (Dynamic t v) def
|
|
|
|
| typeOf def == t = unsafeCoerce v
|
|
|
|
| otherwise = def
|
|
|
|
|
|
|
|
-- | Converts a 'Dynamic' object back into an ordinary Haskell value of
|
|
|
|
-- the correct type. See also 'fromDyn'.
|
|
|
|
fromDynamic
|
|
|
|
:: Typeable a
|
|
|
|
=> Dynamic -- ^ the dynamically-typed object
|
|
|
|
-> Maybe a -- ^ returns: @'Just' a@, if the dynamically-typed
|
|
|
|
-- object has the correct type (and @a@ is its value),
|
|
|
|
-- or 'Nothing' otherwise.
|
|
|
|
fromDynamic (Dynamic t v) =
|
|
|
|
case unsafeCoerce v of
|
2005-04-24 08:51:33 +00:00
|
|
|
r | t == typeOf r -> Just r
|
2005-05-19 03:24:30 +00:00
|
|
|
| otherwise -> unsafePerformIO (putStrLn $
|
|
|
|
"Couldn't match `" ++show(typeOf r) ++
|
|
|
|
"' against `" ++show t ++"'"++
|
|
|
|
"\n\tExpected type: " ++show(typeOf r) ++
|
|
|
|
"\n\tInferred type: " ++show t
|
|
|
|
) `seq` Nothing
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-07-06 05:22:54 +00:00
|
|
|
#if __GLASGOW_HASKELL__ >= 603
|
|
|
|
|
2005-04-24 08:51:33 +00:00
|
|
|
-- (f::(a->b)) `dynApply` (x::a) = (f a)::b
|
|
|
|
dynApply :: Dynamic -> Dynamic -> Maybe Dynamic
|
|
|
|
dynApply (Dynamic t1 f) (Dynamic t2 x) =
|
|
|
|
case funResultTy t1 t2 of
|
2005-05-19 02:09:02 +00:00
|
|
|
Just t3 -> Just (Dynamic t3 ((unsafeCoerce f) x))
|
2005-04-24 08:51:33 +00:00
|
|
|
Nothing -> Nothing
|
|
|
|
|
2005-07-06 05:22:54 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
dynApp :: Dynamic -> Dynamic -> Dynamic
|
|
|
|
dynApp f x = case dynApply f x of
|
|
|
|
Just r -> r
|
|
|
|
Nothing -> error ("Type error in dynamic application.\n" ++
|
|
|
|
"Can't apply function " ++ show f ++
|
|
|
|
" to argument " ++ show x)
|
2005-04-24 08:51:33 +00:00
|
|
|
|
2005-05-19 02:09:02 +00:00
|
|
|
dynTypeRep :: Dynamic -> TypeRep
|
|
|
|
dynTypeRep (Dynamic tr _) = tr
|
2005-07-06 05:22:54 +00:00
|
|
|
|
|
|
|
#endif
|