Starting in the series is a general introduction to Dynamic runtime library.
The DLR adds a set of services to the CLR for better supporting dynamic languages.
Following are the key components of DLR.
- Common hosting model
- Expression Trees:shared abstract semantic tree representation
- DynamicSites,SiteBinders ,Rules:support to make it easy to generate fast dynamic code
- IdynamicMetaObjectProvider:shared dynamic type system
- Utilities :default binder, tuples, big integers, adapters for existing static types to participate in the IDynamicMetaObjectProvider protocol, support for new static types to easily participate in IDynamicMetaObjectProvider protocol
Following figure sums up all the key components of DLR
Common Hosting
ScriptRuntime: This is the starting point for hosting. The ScriptRuntime represents global script state, such as referenced assemblies and a global object (a ScriptScope). You can have more than one ScriptRuntime per .Net AppDomain. You can load assemblies from which a ScriptRuntime coordinates with ScriptEngines to make namespaces and types available to script code.
ScriptScope:This essentially represents a namespace. Hosts can bind variable names in ScriptScopes, fetch variable values, etc. Hosts can execute code within different scopes to isolate free variable resolutions.
ScriptEngine: These are the work horse and represent a language's semantics. They offer various ways to execute code and create ScriptScopes and ScriptSources. You can have a single instance of a given language's engine in a ScriptRuntime instance.
ScriptSource : These offer methods for reading and executing code in various ways from different kinds of sources.
CompiledCode:These represent code that has been compiled to execute repeatedly without having to compile it each time, and they have a default ScriptScope the code runs in. The default scope may have optimized variable storage and lookup for the code.
You can always execute the code in any ScriptScope if you need it to execute in a clean scope each time, or you want to accumulate side effects from the code in another scope.
ScriptHost: This lets you provide a custom PlatformAdaptationLayer object to override file name resolution. For example, you might only load files from a particular directory or go to a web server for files. A host communicates its sub type of ScriptHost to the DLR when it creates a ScriptRuntime. Many hosts can just use the DLR’s default ScriptHost.
ObjectOperations: These provide a large catalogue of object operations such as member access, conversions, indexing, and operations like addition. There are several introspection and tooling support services. You get ObjectOperation instances from engines, and they are bound to their engines for the semantics of the operations.
CallSites -- These provide a caching mechanism per operation or call in dynamic code. Without CallSites, every time you executed "a + b" the DLR would have to fully search for how to add a and b. With CallSites the DLR can store a rule for what to do with a and b given their characteristics (which typically means their type).
Binders -- These represent a language's semantics for performing a specific operation in a CallSite, including any metadata about the CallSite. Binders get called when there is a cache miss in a CallSite, and they inspect the operands of the CallSite to compute how to perform the operation in the CallSite. Binders communicate how to perform operations to CallSites using Expression Trees.
Rules -- These represent a single computation or binding of how to perform an operation in a CallSite. Rules have a test for when the rule is valid (typically just testing the static runtime type of the operands). Rules have a target or expression for how to perform the operation, given the characteristics of the operands guaranteed by the Rule's test. The test and target expressions come from the binders.
DynamicObject: DynamicObject has virtual methods for all the abstract operations you can perform on via IDynamicMetaObjectProvider and DynamicMetaObject. Developer can simply derive from DynamicObject and implement the desired operations , without caring about creating Expression Trees or create own DynamicMetaObject to provide binding to a CallSite. All this is automatically handled by DLR.
(Don't worry if you are not getting as to what it all means, I have another article line up to demonstrate the usage of this)
ExpandoObject : They can delegate to it or use it directly as a simple dynamic object that lets you add, set, and remove members to the object at runtime.
(Dynamic and Expando objects would very soon be dealt in a separate article)
Utilities: There are some helper classes for implementing binders, producing rules, etc.
COM Interop: The DLR provides an IDynamicMetaObjectProvider and DynamicMetaObject for COM objects to participate in the dynamic operations and CallSites. The DLR uses IDispatch to bind to COM objects.
Shared AST (Expression Trees): This is one of the core pillars of the DLR. LINQ Expression Trees have been extended to include control flow, assignment, etc. Expression Tres are used in the binders and DynamicMetaObject protocols.
LanguageContext : These are the objects that represent a language that is implemented on the DLR and support the DLR's hosting model. These are responsible for executing code, with many members that support various higher-level features in the DLR.
IDynamicMetaObjectProvider : This is the type implemented by languages and library authors who want to control the dispatching and binding logic for the DLR's interoperability protocol. It is foremost just a jumper to get DynamicMetaObjects.
DynamicMetaObject : A DynamicMetaObject has several methods representing the abstract operations such as fetching members, setting members, invoking a member, invoking the object itself, instantiating the object, or performing standard operations (addition, multiplication, etc.).
Compiler:The DLR ships an Expression Tree compiler, Expression.Compile that returns a delegate for invoking the code represented by the tree.
Utilities: There are some helper classes for implementing languages. The DLR has a GeneratorExpression that can be used in a LambdaExpression for creating what C# calls iterators.
The DLR rewrites the expression tree, which contains YieldExpressions, to a new tree that open codes the state machine for re-entering the iteration and re-establishing any dynamic state (for example, try-catch's).
There are math helpers such as BigInteger, Complex,(see my earlier article) and Tuples.
There is a GlobalVariableExpression and optimized module support for creating supporting the DLR hosting model and having faster ScriptScopes behind CompiledCode.
Next, I'll cover the runtime section first and later on move to Hosting and Launguage Implementation.
Till next time we connect..............
Happy Coding!
Note: This article is an excerpt from the DLR documents available in open source version of DLR at codeplex.com and should not be taken into consideration for copyright voilations.
Comments
Post a Comment