diff options
Diffstat (limited to 'src/Language/Fiddle/Ast')
-rw-r--r-- | src/Language/Fiddle/Ast/FileInterface.hs | 33 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/Generic.hs | 5 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/Instances.hs | 228 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/Kinds.hs | 25 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/Stage.hs | 43 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/SyntaxTree.hs | 537 | ||||
-rw-r--r-- | src/Language/Fiddle/Ast/Internal/SyntaxTreeKinds.hs | 1 |
7 files changed, 872 insertions, 0 deletions
diff --git a/src/Language/Fiddle/Ast/FileInterface.hs b/src/Language/Fiddle/Ast/FileInterface.hs new file mode 100644 index 0000000..d29fc9d --- /dev/null +++ b/src/Language/Fiddle/Ast/FileInterface.hs @@ -0,0 +1,33 @@ +module Language.Fiddle.Ast.FileInterface where + +-- Definitions for file interfaces. These interfaces contain an abstract +-- representation of the symbols and information exported by a fiddle file. +-- These interfaces are also serializable, and when compiling a fiddle file, all +-- the import statements should supply an fdi (fiddle interface) file to speed +-- up subsequent compilations. + +-- import Data.Text +-- +-- data ObjectType = ObjectType +-- { objectTypeSize :: Word32 +-- } +-- +-- data Metatype +-- = Object +-- { objectLocation :: Word64, +-- objectType :: Text +-- } +-- | Type +-- { typeSizeBytes :: Word32 +-- } +-- +-- data Element a = Element +-- { elementFullyQualifiedSymbol :: Text, +-- elementDocumentation :: Maybe Text, +-- elementMetatype :: Metatype, +-- elementAnnotation :: a +-- } +-- +-- data FileInterface a = FiddleInterface +-- { exportedElements :: [Element a] +-- } diff --git a/src/Language/Fiddle/Ast/Internal/Generic.hs b/src/Language/Fiddle/Ast/Internal/Generic.hs new file mode 100644 index 0000000..cb075cc --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/Generic.hs @@ -0,0 +1,5 @@ +{-# LANGUAGE TypeOperators #-} +module Language.Fiddle.Ast.Internal.Generic where + +import GHC.Generics +import GHC.TypeError as TypeError diff --git a/src/Language/Fiddle/Ast/Internal/Instances.hs b/src/Language/Fiddle/Ast/Internal/Instances.hs new file mode 100644 index 0000000..c8c606c --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/Instances.hs @@ -0,0 +1,228 @@ +{-# LANGUAGE ConstraintKinds #-} +{-# LANGUAGE DefaultSignatures #-} +{-# LANGUAGE FlexibleContexts #-} +{-# LANGUAGE IncoherentInstances #-} +{-# LANGUAGE TypeOperators #-} +{-# LANGUAGE UndecidableInstances #-} + +module Language.Fiddle.Ast.Internal.Instances where + +import Data.Functor.Identity +import Data.Kind (Type) +import Data.Typeable +import GHC.Generics +import GHC.TypeError as TypeError +import Language.Fiddle.Ast.Internal.Kinds +import Language.Fiddle.Ast.Internal.Stage + +-- Type class to easily switch between stages if there is no difference in the +-- syntax tree structure between these stages. Can make things much cleaner and +-- avoids the boilerplate and bugprone-ness of needing to rote copy everything. +class + (Functor f) => + EasySwitchStage + (t :: Stage -> SynTree) + (f :: Type -> Type) + (fromStage :: Stage) + (toStage :: Stage) + where + switchStage :: t fromStage f a -> t toStage f a + default switchStage :: + ( Generic (t fromStage f a), + Generic (t toStage f a), + GEasySwitchStage (Rep (t fromStage f a)) (Rep (t toStage f a)) + ) => + t fromStage f a -> + t toStage f a + switchStage t = to $ gSwitchStage (from t) + +-- Class for walking a syntax tree under the context of a monad and modifying +-- the different parts of the SynTree type.. +class Alter (t :: SynTree) where + alter :: + forall f1 f2 a1 a2 m. + (Monad m, Traversable f1) => + (forall z. f1 z -> m (f2 z)) -> + (a1 -> m a2) -> + t f1 a1 -> + m (t f2 a2) + default alter :: + forall f1 f2 a1 a2 m. + ( Generic (t f1 a1), + Generic (t f2 a2), + Traversable f1, + GAlter t f1 f2 a1 a2 (Rep (t f1 a1)) (Rep (t f2 a2)), + Monad m + ) => + ( forall z. + f1 z -> + m (f2 z) + ) -> + (a1 -> m a2) -> + t f1 a1 -> + m (t f2 a2) + alter ffn fn t = to <$> galter (proxyOf t) ffn fn (from t) + +-- A syntax tree object is annotated if it has an annotation 'a' as the last +-- element. +class Annotated (t :: SynTree) where + annot :: t f a -> a + default annot :: (Generic (t f a), GAnnot a (Rep (t f a))) => t f a -> a + annot t = gannot (from t) + +-- Generic implementations of common typeclass for SyntaxTrees. +-- +-- This is where we try to hide the pig behind the curtain. +-- +-- +-- --------------------------------------------------------------- +-- +-- +-- +class GAlter t f1 f2 a1 a2 r1 r2 where + galter :: + forall proxy x m. + (Monad m, Traversable f1) => + proxy t -> + (forall z. f1 z -> m (f2 z)) -> + (a1 -> m a2) -> + r1 x -> + m (r2 x) + +{- Altering a record with type a1 will apply the mapping function and produce a + record with type a2 -} +instance GAlter t f1 f2 a1 a2 (Rec0 a1) (Rec0 a2) where + galter _ _ fn k1 = K1 <$> fn (unK1 k1) + +{- Base-Case. Altering unrelated leaf types will do nothing. -} +instance GAlter t f1 f2 a1 a2 (Rec0 u1) (Rec0 u1) where + galter _ _ _ = return + +{- Recursive case. Call alter on sub-structures. -} +instance + (Alter u) => + GAlter t f1 f2 a1 a2 (Rec0 (u f1 a1)) (Rec0 (u f2 a2)) + where + galter _ ffn fn k1 = + K1 <$> alter ffn fn (unK1 k1) + +{- Recursive case. Called when there are list of substructures that need to be + recused. -} +instance + (Alter u, Traversable l) => + GAlter t f1 f2 a1 a2 (Rec0 (l (u f1 a1))) (Rec0 (l (u f2 a2))) + where + galter _ ffn fn k1 = + K1 <$> mapM (alter ffn fn) (unK1 k1) + +-- instance GAlter t f1 f2 a1 a2 (Rec0 (f1 z)) (Rec0 (f2 z)) where +-- galter _ ffn _ k1 = K1 <$> ffn (unK1 k1) + +{- Generic altering. Descends into the function and alters whatever is inside + the functor and then transforms the functor using the ffn function. -} +instance + (Traversable f1, Alter u) => + GAlter t f1 f2 a1 a2 (Rec0 (f1 (u f1 a1))) (Rec0 (f2 (u f2 a2))) + where + galter proxy ffn fn k1 = do + newK <- mapM (alter ffn fn) (unK1 k1) + K1 <$> ffn newK + +instance + ( GAlter t f1 f2 a1 a2 l1 l2, + GAlter t f1 f2 a1 a2 r1 r2 + ) => + GAlter t f1 f2 a1 a2 (l1 :*: r1) (l2 :*: r2) + where + galter proxy ffn fn (a :*: b) = do + a' <- galter proxy ffn fn a + b' <- galter proxy ffn fn b + return (a' :*: b') + +instance + ( GAlter t f1 f2 a1 a2 l1 l2, + GAlter t f1 f2 a1 a2 r1 r2 + ) => + GAlter t f1 f2 a1 a2 (l1 :+: r1) (l2 :+: r2) + where + galter proxy ffn fn (R1 r) = R1 <$> galter proxy ffn fn r + galter proxy ffn fn (L1 l) = L1 <$> galter proxy ffn fn l + +instance + (GAlter t f1 f2 a1 a2 r1 r2) => + GAlter t f1 f2 a1 a2 (M1 i c r1) (M1 i c r2) + where + galter proxy ffn fn (M1 a) = M1 <$> galter proxy ffn fn a + +class GAnnot a r where + gannot :: r x -> a + +instance GAnnot a (Rec0 a) where + gannot = unK1 + +instance (GAnnot a r) => GAnnot a (l :*: r) where + gannot (_ :*: r) = gannot r + +instance (GAnnot a r, GAnnot a l) => GAnnot a (l :+: r) where + gannot (R1 r) = gannot r + gannot (L1 l) = gannot l + +instance (GAnnot a r) => GAnnot a (M1 i c r) where + gannot (M1 a) = gannot a + +instance + (GEasySwitchStage s1 s2) => + GEasySwitchStage (M1 i c s1) (M1 i c s2) + where + gSwitchStage (M1 a) = M1 (gSwitchStage a) + +instance + (GEasySwitchStage l1 l2, GEasySwitchStage r1 r2) => + GEasySwitchStage (l1 :+: r1) (l2 :+: r2) + where + gSwitchStage (R1 r) = R1 $ gSwitchStage r + gSwitchStage (L1 l) = L1 $ gSwitchStage l + +instance + (GEasySwitchStage l1 l2, GEasySwitchStage r1 r2) => + (GEasySwitchStage (l1 :*: r1) (l2 :*: r2)) + where + gSwitchStage (l :*: r) = gSwitchStage l :*: gSwitchStage r + +instance + (EasySwitchStage t f fs ts) => + (GEasySwitchStage (Rec0 (t fs f a)) (Rec0 (t ts f a))) + where + gSwitchStage (K1 val) = K1 (switchStage val) + +instance + ( EasySwitchStage t f fs ts, + Functor func + ) => + (GEasySwitchStage (Rec0 (func (t fs f a))) (Rec0 (func (t ts f a)))) + where + gSwitchStage (K1 val) = K1 (switchStage <$> val) + +instance (GEasySwitchStage (Rec0 a) (Rec0 a)) where + gSwitchStage = id + +instance + ( TypeError + ( TypeError.Text "Unable to match type " + :<>: TypeError.ShowType a + :<>: TypeError.Text " with " + :<>: TypeError.ShowType b + ) + ) => + (GEasySwitchStage (Rec0 a) (Rec0 b)) + where + gSwitchStage = error "Cannot be called" + +class GEasySwitchStage r1 r2 where + gSwitchStage :: r1 x -> r2 x + +proxyOf :: t f a -> Proxy t +proxyOf _ = Proxy + +instance (Alter t, Traversable f) => Functor (t f) where + fmap f t = runIdentity (alter return (return . f) t) diff --git a/src/Language/Fiddle/Ast/Internal/Kinds.hs b/src/Language/Fiddle/Ast/Internal/Kinds.hs new file mode 100644 index 0000000..1368c84 --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/Kinds.hs @@ -0,0 +1,25 @@ + +module Language.Fiddle.Ast.Internal.Kinds where + +import Data.Kind (Type) +import Language.Fiddle.Ast.Internal.Stage + +-- The Kind types should have for SyntaxTrees. +-- +-- A synatx tree kind should take as parameters, an applicative functor for the +-- first argument. This applicative functor allows errors and warnings to be +-- contained to their deferred regions during compilation, so more errors may be +-- reported an once. +-- +-- The second argument is the annotation type. Every syntax tree element has an +-- annotation. When initially parsing, this annotation is a list of comments +-- that came before the element and the source position of the element. +-- +-- SynTree types are not dependent on the compilation stage. They're the same +-- regardless of the compilation stage. +type SynTree = (Type -> Type) -> Type -> Type + +-- A syntax tree which is dependent on the stage in compilation. As compilations +-- progress through stages, types in the syntax tree may change as things become +-- more and more machine readable. +type StagedSynTree = Stage -> SynTree diff --git a/src/Language/Fiddle/Ast/Internal/Stage.hs b/src/Language/Fiddle/Ast/Internal/Stage.hs new file mode 100644 index 0000000..cfcd0e6 --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/Stage.hs @@ -0,0 +1,43 @@ +{-# LANGUAGE ConstraintKinds #-} +{-# LANGUAGE FlexibleContexts #-} +{-# LANGUAGE IncoherentInstances #-} +{-# LANGUAGE TypeOperators #-} +{-# LANGUAGE UndecidableInstances #-} +module Language.Fiddle.Ast.Internal.Stage where + +import Data.Type.Bool +import Data.Type.Equality +import Data.Typeable +import qualified GHC.TypeError as TypeError +import GHC.TypeLits + +-- Stage of compilation. Parts of the AST maybe un unavailable with other stages +-- as compilation simplifies the AST. +data Stage = Stage1 | Stage2 | Stage3 + deriving (Typeable) + +-- Returns the stage before the given stage. +type family PreviousStage (s :: Stage) :: Stage where + PreviousStage Stage1 = TypeError (TypeError.Text "No Prior stage to Stage1") + PreviousStage s = NumberToStage (StageToNumber s - 1) + +-- Returns the stage after the give stage. +type family NextStage (s :: Stage) :: Stage where + NextStage s = NumberToStage (StageToNumber s + 1) + +-- Converts a stage to a number. +type family StageToNumber (s :: Stage) :: Natural where + StageToNumber Stage1 = 1 + StageToNumber Stage2 = 2 + StageToNumber Stage3 = 3 + +-- Converts a number to a stage. +type family NumberToStage (n :: Natural) :: Stage where + NumberToStage 1 = Stage1 + NumberToStage 2 = Stage2 + NumberToStage 3 = Stage3 + NumberToStage n = TypeError (TypeError.Text "Number in NumberToStage.") + +-- Type-level constraint to determine if a stage is less than some natural +-- ordinal. Used to bound parts of the AST in multiple stages. +type StageLessThan stage (n :: Natural) = (CmpNat (StageToNumber stage) n == LT) diff --git a/src/Language/Fiddle/Ast/Internal/SyntaxTree.hs b/src/Language/Fiddle/Ast/Internal/SyntaxTree.hs new file mode 100644 index 0000000..48852ee --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/SyntaxTree.hs @@ -0,0 +1,537 @@ +{-# LANGUAGE ConstraintKinds #-} +{-# LANGUAGE DeriveAnyClass #-} +{-# LANGUAGE DeriveGeneric #-} +{-# LANGUAGE FlexibleContexts #-} +{-# LANGUAGE IncoherentInstances #-} +{-# LANGUAGE TypeOperators #-} +{-# LANGUAGE UndecidableInstances #-} + +module Language.Fiddle.Ast.Internal.SyntaxTree + ( -- Type Families + NumberType, + ImportType, + -- Witness Types + Witness (..), + WitnessType, + -- AST Types + Name (..), + Directive (..), + DirectiveBody (..), + DirectiveElement (..), + DirectiveExpression (..), + Directed (..), + FiddleUnit (..), + Identifier (..), + Expression (..), + ImportStatement (..), + ImportList (..), + FiddleDecl (..), + ObjTypeBody (..), + ObjType (..), + ObjTypeDecl (..), + Modifier (..), + ModifierKeyword (..), + DeferredRegisterBody (..), + BodyType (..), + RegisterBody (..), + RegisterBitsDecl (..), + RegisterBitsTypeRef (..), + AnonymousBitsType (..), + BitType (..), + EnumBody (..), + EnumConstantDecl (..), + PackageBody (..), + -- Helper Functions + mapDirected, + mapDirectedM, + asDirected, + undirected, + -- Utility Functions + squeeze, + ) +where + +import Data.Coerce +import Data.Functor.Identity +import Data.Kind (Type) +import Data.List.NonEmpty +import Data.Proxy +import Data.Text (Text) +import Data.Traversable +import Data.Type.Bool +import Data.Type.Equality +import Data.Typeable +import Data.Void (Void, absurd) +import GHC.Generics +import GHC.TypeLits +import Language.Fiddle.Ast.Internal.Generic +import Language.Fiddle.Ast.Internal.Instances +import Language.Fiddle.Ast.Internal.Kinds +import Language.Fiddle.Ast.Internal.Stage + +-- The Type of number during each stage of compilation. When in the first stage, +-- numbers are just strings like anything else. In later stages, numbers get +-- parsed into actual integers. This makes it easier to process later. +type family NumberType (a :: Stage) :: Type where + NumberType Stage1 = Text + NumberType Stage2 = Integer + NumberType s = NumberType (PreviousStage s) + +-- The type that represents an import statement. In the early stages of +-- compilation, this is just a string representing the import path, but in later +-- stages of compilation, this actually gets replaced by an abstract +-- representation of the imported material. +type family ImportType (stage :: Stage) :: SynTree where + ImportType Stage1 = ImportStatement + ImportType Stage2 = ImportStatement + ImportType Stage3 = ImportStatement + +-- A way to disable or enable a subtree type based on a type-level boolean. +-- +-- This is used over GADT's specific parameterization to allow for deriving +-- generics and reduce boilerplate. +-- +-- This is a wrapper type to allow defining instances. +data Witness (s :: Bool) where + Witness :: (WitnessType s) -> Witness s + +-- | If the type level expression is true, the witness type is the Unit type, if +-- it is false, the witness type is Void, thus making whatever it touches +-- uninhabitable. This is how parts of the AST is enabled or disabled during +-- compilation. +type family WitnessType (s :: Bool) where + WitnessType True = () + WitnessType False = Void + +-- A Name is multiple identifiers separated by dots. It's the way of namespaces +-- to different packages. +data Name :: SynTree where + Name :: NonEmpty (Identifier f a) -> a -> Name f a + deriving (Generic, Annotated, Alter, Typeable) + +-- Syntax tree fo the directive sublanguage. Directives can be on many elements +-- and provide the compiler with additional information about the emitted code. +-- +-- The directive subtree by design does not depend on the compilation stage. +-- This is because the directive sublanguage should pass directly to the backend +-- compilation stage. +data Directive :: SynTree where + Directive :: f (DirectiveBody f a) -> a -> Directive f a + deriving (Generic, Annotated, Alter, Typeable) + +-- A directive body has multiple directive elements. +data DirectiveBody :: SynTree where + DirectiveBody :: [DirectiveElement f a] -> a -> DirectiveBody f a + deriving (Generic, Annotated, Alter, Typeable) + +-- Element in the directive. +data DirectiveElement :: SynTree where + -- A directive can just be a key. Where the mere presence of the key has + -- semantic value. + DirectiveElementKey :: + -- Which backend is this directive intended for? + Maybe (Identifier f a) -> + Identifier f a -> + a -> + DirectiveElement f a + -- A directive can be more complex too. It can have an optional backend + -- specificer, a key and a value. + DirectiveElementKeyValue :: + -- Which backend is this directive intendend for? + Maybe (Identifier f a) -> + -- The key for this directive. + Identifier f a -> + -- The value for this directive. + DirectiveExpression f a -> + a -> + DirectiveElement f a + deriving (Generic, Annotated, Alter, Typeable) + +-- Expressions which can be found in the directive. +data DirectiveExpression f a where + DirectiveString :: Text -> a -> DirectiveExpression f a + DirectiveNumber :: Text -> a -> DirectiveExpression f a + deriving (Generic, Annotated, Alter, Typeable) + +-- A type, which wraps another syntax tree, but tacks on an array of directives. +-- that apply to the subtree. +data Directed t stage f a where + Directed :: [Directive f a] -> t stage f a -> a -> Directed t stage f a + deriving (Generic, Annotated, Alter, Typeable) + +-- Apply a function to the underlying subtree in a Directed type. +mapDirected :: (t s f a -> t' s' f a) -> Directed t s f a -> Directed t' s' f a +mapDirected fn (Directed dr tfa a) = Directed dr (fn tfa) a + +-- Apply a monadic function to the underlying subtree in a Directed type. +mapDirectedM :: + (Monad m) => (t s f a -> m (t' s' f a)) -> Directed t s f a -> m (Directed t' s' f a) +mapDirectedM fn (Directed dr tfa a) = Directed dr <$> fn tfa <*> pure a + +asDirected :: (Annotated (t s)) => t s f a -> Directed t s f a +asDirected tfa = Directed [] tfa (annot tfa) + +undirected :: Directed t s f a -> t s f a +undirected (Directed _ tfa _) = tfa + +-- Root of the parse tree. Just contains a list of declarations. +data FiddleUnit (stage :: Stage) (f :: Type -> Type) a where + FiddleUnit :: [Directed FiddleDecl stage f a] -> a -> FiddleUnit stage f a + deriving (Generic, Annotated, Typeable) + +deriving instance (Alter (ImportType stage)) => Alter (FiddleUnit stage) + +-- Just an identifier. +data Identifier f a = Identifier !Text a + deriving (Generic, Annotated, Alter, Typeable) + +-- Expression. +data Expression (s :: Stage) :: SynTree where + -- Just a string. Parsing the number comes in stage2. + LitNum :: NumberType stage -> a -> Expression stage f a + Var :: Identifier f a -> a -> Expression stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data ImportStatement f a where + ImportStatement :: Text -> Maybe (ImportList f a) -> a -> ImportStatement f a + deriving (Generic, Annotated, Alter, Typeable) + +data ImportList f a where + ImportList :: [Identifier f a] -> a -> ImportList f a + deriving (Generic, Annotated, Alter, Typeable) + +-- Top-level declarations. +data FiddleDecl :: StagedSynTree where + {- + - An option is a key/value pair. + - option <ident> <ident>; + -} + OptionDecl :: + Identifier f a -> + Identifier f a -> + a -> + FiddleDecl stage f a + ImportDecl :: + ImportType stage f a -> + a -> + FiddleDecl stage f a + UsingDecl :: + Name f a -> a -> FiddleDecl stage f a + {- Package Statement. Package Name, Package body -} + PackageDecl :: + Name f a -> + f (PackageBody stage f a) -> + a -> + FiddleDecl stage f a + {- location <identifier> = <expr>. -} + LocationDecl :: + Identifier f a -> + Expression stage f a -> + a -> + FiddleDecl stage f a + {- bits <identifier> : <type> -} + BitsDecl :: + Identifier f a -> + BitType stage f a -> + a -> + FiddleDecl stage f a + {- objtype <identifier> : <type> -} + ObjTypeDecl :: + Identifier f a -> + f (ObjTypeBody stage f a) -> + a -> + FiddleDecl stage f a + {- object <ident> at <expr> : <type> -} + ObjectDecl :: + Identifier f a -> + Expression stage f a -> + ObjType stage f a -> + a -> + FiddleDecl stage f a + deriving (Generic, Annotated, Typeable) + +deriving instance (Alter (ImportType stage)) => Alter (FiddleDecl stage) + +data ObjTypeBody (stage :: Stage) (f :: Type -> Type) a where + ObjTypeBody :: + BodyType f a -> + [Directed ObjTypeDecl stage f a] -> + a -> + ObjTypeBody stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data ObjType stage f a where + -- { <body> } + -- Anonymous types are only allowed in stage1. Stage2 should have them be + -- de-anonymized. + AnonymousObjType :: + (Witness (stage == Stage1)) -> + f (ObjTypeBody stage f a) -> + a -> + ObjType stage f a + -- <type>[<expr>] + ArrayObjType :: ObjType stage f a -> Expression stage f a -> a -> ObjType stage f a + -- <identifier> + ReferencedObjType :: Name f a -> a -> ObjType stage f a + deriving (Typeable, Generic, Alter, Annotated, Typeable) + +data ObjTypeDecl stage f a where + {- assert_pos(<expr>) -} + AssertPosStatement :: + Witness (StageLessThan stage 3) -> + Expression stage f a -> + a -> + ObjTypeDecl stage f a + {- reg <ident>(<expr>) : <regtype> -} + RegisterDecl :: + Maybe (Modifier f a) -> + Maybe (Identifier f a) -> + Expression stage f a -> + Maybe (RegisterBody stage f a) -> + a -> + ObjTypeDecl stage f a + {- reserved(n); -} + ReservedDecl :: Expression stage f a -> a -> ObjTypeDecl stage f a + {- <struct|union> { subfields } <name>; -} + TypeSubStructure :: + f (ObjTypeBody stage f a) -> + Maybe (Identifier f a) -> + a -> + ObjTypeDecl stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data Modifier f a where + ModifierKeyword :: ModifierKeyword -> a -> Modifier f a + deriving (Generic, Annotated, Alter, Typeable) + +data ModifierKeyword = Rw | Ro | Wo deriving (Eq, Ord, Show, Read, Typeable) + +data DeferredRegisterBody stage f a where + DeferredRegisterBody :: + [Directed RegisterBitsDecl stage f a] -> + a -> + DeferredRegisterBody stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data BodyType (f :: Type -> Type) a where + Union :: a -> BodyType f a + Struct :: a -> BodyType f a + deriving (Generic, Annotated, Alter, Typeable) + +data RegisterBody stage f a where + RegisterBody :: + BodyType f a -> + f (DeferredRegisterBody stage f a) -> + a -> + RegisterBody stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data RegisterBitsDecl stage f a where + -- reserved(<expr>) + ReservedBits :: Expression stage f a -> a -> RegisterBitsDecl stage f a + -- <modifer> <ident> : <type> + DefinedBits :: + Maybe (Modifier f a) -> + Identifier f a -> + RegisterBitsTypeRef stage f a -> + a -> + RegisterBitsDecl stage f a + BitsSubStructure :: + RegisterBody stage f a -> + Maybe (Identifier f a) -> + a -> + RegisterBitsDecl stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data RegisterBitsTypeRef stage f a where + -- <type>[<expr>] + RegisterBitsArray :: + RegisterBitsTypeRef stage f a -> + Expression stage f a -> + a -> + RegisterBitsTypeRef stage f a + {- Reference to a type. -} + RegisterBitsReference :: Name f a -> a -> RegisterBitsTypeRef stage f a + {- enum(<expr>) { <body> } + Anonymous types are only allowed in stage1. + Stage2 should de-anonymize these type. -} + RegisterBitsAnonymousType :: + Witness (stage == Stage1) -> + AnonymousBitsType stage f a -> + a -> + RegisterBitsTypeRef stage f a + {- (<expr>) + - + - The expression is just bits ... i.e. an integer. + -} + RegisterBitsJustBits :: + Expression stage f a -> + a -> + RegisterBitsTypeRef stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data AnonymousBitsType stage f a where + -- enum(<expr>) { <body> } + AnonymousEnumBody :: + Expression stage f a -> + f (EnumBody stage f a) -> + a -> + AnonymousBitsType stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data BitType (stage :: Stage) (f :: Type -> Type) a where + -- enum(<expr>) { <body> } + EnumBitType :: + Expression stage f a -> + f (EnumBody stage f a) -> + a -> + BitType stage f a + -- (<expr>) + RawBits :: Expression stage f a -> a -> BitType stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data EnumBody (stage :: Stage) (f :: Type -> Type) a where + -- <decl>, + EnumBody :: [Directed EnumConstantDecl stage f a] -> a -> EnumBody stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data EnumConstantDecl stage f a where + -- <ident> = <expr> + EnumConstantDecl :: + Identifier f a -> + Expression stage f a -> + a -> + EnumConstantDecl stage f a + -- reserved = <expr> + EnumConstantReserved :: + Expression stage f a -> + a -> + EnumConstantDecl stage f a + deriving (Generic, Annotated, Alter, Typeable) + +data PackageBody (stage :: Stage) (f :: Type -> Type) a where + {- The body of a package -} + PackageBody :: [Directed FiddleDecl stage f a] -> a -> PackageBody stage f a + deriving (Generic, Annotated, Typeable) + +deriving instance (Alter (ImportType stage)) => Alter (PackageBody stage) + +squeeze :: (Alter t, Traversable f, Monad f) => t f a -> f (t Identity a) +squeeze = alter (fmap Identity) return + +-- Expression involves NumberType, so we add the constraint: +deriving instance (Functor f, NumberType s ~ NumberType s') => EasySwitchStage Expression f s s' + +-- FiddleDecl includes both NumberType and ImportType, so we need both constraints: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s', + ImportType s ~ ImportType s' + ) => + EasySwitchStage FiddleDecl f s s' + +-- ObjType includes NumberType, so we add the constraint: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage ObjType f s s' + +-- ObjTypeBody doesn't have any special type families, so no additional constraints: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage ObjTypeBody f s s' + +-- ObjTypeDecl doesn't have special type families, so no additional constraints: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage ObjTypeDecl f s s' + +-- DeferredRegisterBody doesn't have special type families: +deriving instance + ( Functor f, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage DeferredRegisterBody f s s' + +-- RegisterBody depends on NumberType, so we add that constraint: +deriving instance + ( Functor f, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage RegisterBody f s s' + +-- RegisterBitsDecl depends on NumberType: +deriving instance + ( Functor f, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage RegisterBitsDecl f s s' + +-- RegisterBitsTypeRef depends on NumberType: +deriving instance + ( Functor f, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s' + ) => + EasySwitchStage RegisterBitsTypeRef f s s' + +-- AnonymousBitsType depends on NumberType: +deriving instance + (Functor f, NumberType s ~ NumberType s') => + EasySwitchStage AnonymousBitsType f s s' + +-- BitType depends on NumberType: +deriving instance + (Functor f, NumberType s ~ NumberType s') => + EasySwitchStage BitType f s s' + +-- EnumBody doesn't depend on any type families: +deriving instance + (Functor f, NumberType s ~ NumberType s') => + EasySwitchStage EnumBody f s s' + +-- EnumConstantDecl depends on NumberType: +deriving instance + (Functor f, NumberType s ~ NumberType s') => + EasySwitchStage EnumConstantDecl f s s' + +-- PackageBody includes both NumberType and ImportType: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s', + ImportType s ~ ImportType s' + ) => + EasySwitchStage PackageBody f s s' + +-- FiddleUnit includes NumberType and ImportType, so we need both constraints: +deriving instance + ( Functor f, + StageLessThan s 3 ~ StageLessThan s' 3, + (s == Stage1) ~ (s' == Stage1), + NumberType s ~ NumberType s', + ImportType s ~ ImportType s' + ) => + EasySwitchStage FiddleUnit f s s' + +-- Directed depends on its underlying AST type: +deriving instance + (EasySwitchStage t f s s') => + EasySwitchStage (Directed t) f s s' diff --git a/src/Language/Fiddle/Ast/Internal/SyntaxTreeKinds.hs b/src/Language/Fiddle/Ast/Internal/SyntaxTreeKinds.hs new file mode 100644 index 0000000..bfdfb13 --- /dev/null +++ b/src/Language/Fiddle/Ast/Internal/SyntaxTreeKinds.hs @@ -0,0 +1 @@ +module Language.Fiddle.Ast.Internal.SyntaxTreeKinds where |