Typesystem Debugging
For debugging typesystem MPS provides Typesystem Trace - an integrated visual tool that gives you insight into the evaluation process that happens inside the typesystem engine.
We prepared a dedicated sample language for you to easily experiment with the typesystem. Open the Expressions sample project that comes bundled with MPS and should be available among the sample projects in the user home folder.
data:image/s3,"s3://crabby-images/610ad/610ad28cf259f6e623054332bae4d5bf35fbe2ef" alt="tt11.png tt11.png"
The language to experiment with is a simplified expression language with several types, four arithmetic operations (+, -, *, /), assignment (:=), two types of variable declarations and a variable reference. The editor is very basic with almost no customization, so editing the expressions will perhaps be quite rough. Nevertheless, we expect you to inspect the existing samples and debug their types more than writing new code, so the lack of smooth editing should not be an issue.
data:image/s3,"s3://crabby-images/87918/87918849fb54d375adc9246454c2774cd850ec5e" alt="TT1.png TT1.png"
The language can be embedded into Java thanks to the SimpleMathWrapper concept, but no interaction between the language and BaseLanguage is possible.
The expression language supports six types, organized by subtyping rules into two branches:
Element -> Number -> Float -> Long -> Int
Element -> Bool
If you open the Simple example class, you can place the caret to any part of the expression or select a valid expression block. As soon as you press CtrlShift0P, you'll see the type of the selected node in a pop-up dialog. As an alternative path, you can right-click any node and select Language Debug | Show Type in the context menu.
data:image/s3,"s3://crabby-images/71e25/71e25e74626bad51ad7455b5692289a662e114a9" alt="TT2.png TT2.png"
data:image/s3,"s3://crabby-images/8b2e7/8b2e71b6f4702a10ffb65b3e84014cc6fb7bb288" alt="TT3.png TT3.png"
The Main sample class will give you a more involved example showing how Type-inference correctly propagates the suitable type to variables:
data:image/s3,"s3://crabby-images/45dfe/45dfed1bc01d39460bcb897bf91ebc891f6dd88f" alt="tt4.png tt4.png"
Just check the calculated types for yourself.
The TypeError sample class shows a simple example of a type error. Just uncomment the code Ctrl0/ and check the reported error:
data:image/s3,"s3://crabby-images/80075/80075b121b761002a5af368d805db2eba06dcdd2" alt="tt9.png tt9.png"
Since this variable declaration declares its type explicitly to be an Int, while the initializer is of type Float, the type-system reports an error. You may check the status bar at the bottom or hover over the incorrect piece of code.
When you press CtrlShift0X or navigate through the pop-up menu, you get the Type-system Trace panel displayed on the right hand-side.
data:image/s3,"s3://crabby-images/8eba3/8eba3c58b8776815f1f61909bd652f390e784d58" alt="TT6.png TT6.png"
The Trace shows in Panel 2 all steps (i.e. type system rules) that the type-system engine executed. The steps are ordered top-to-bottom in the order in which they were performed. When you have Button 1 selected, the Panel 2 highlights the steps that directly or indirectly influence the type of the node selected in the editor (Panel 1). Panel 3 details the step selected in Panel 2 - it describes what changes have been made to the type-system engine's state in the step. The actual state of the engine's working memory is displayed in Panel 4.
The Simple sample class is probably the easiest one to start experimenting with. The types get resolved in six steps, following the typesystem rules specified in the language. F3 (Go to Source Node) will navigate you to the node, which is being affected by the current rule.
data:image/s3,"s3://crabby-images/5a16e/5a16e08b3e2c085ce7fc32bfb9a0d2320cccbac1" alt="tt7.png tt7.png"
The type of a variable declaration has to be a super-type of the type of the initializer. The aValue variable is assigned the a type-system variable, the initializer expression is assigned the b type-system variable and a>=b (b sub-type or equal type to a) is added into the working memory.
Following the type-system rule for Arithmetic Expressions, b has to be a sub-type of Number, the value 10 is assigned the c variable, 1.3F is assigned the d variable and a when-concrete handler is added to wait for c to be calculated.
Following the rules for float constants d is solved as Float.
Following the rules for integer constants c is solved as Int. This triggers the when-concrete handler registered in step 2 and registered another when-concrete handler to wait for d. Since d has already been resolved to Float, the handler triggered and resolves b (the whole arithmetic expression) as Float. This also solves the earlier equation (step 2) that b<=Number.
Now a can be resolved as Float, which also solves the step 1 equation that a>=b.
Thanks for your feedback!