title | authors | intro | types | categories | published | updated | ||||
---|---|---|---|---|---|---|---|---|---|---|
A man, the Roslyn compiler and an Azure VM. A peek into the .NET Compiler |
|
I am in Bristol to talk about the Microsoft C# compiler. |
|
|
2017/02/13 18:30:00 |
2017/02/13 19:30:00 |
I'm heading down to Bristol for a meetup at dot net southwest where I will be giving the talk "A man, the Roslyn compiler and an Azure VM. A peek into the .NET Compiler"
Stuff from the event can be found here:
- Slides Roslyn
Such a cool name. Such promise. Alas the fun police at Microsoft decided that project Rosyln, the wonderfully awesomely name for the the new .net compiler would infact be converted to "The Microsoft .net Compiler Platform".
During the talk. I make a joke about Roslyn and it's similarity to Roxanne by the police. This joke never works. I will increment the following number everytime this joke fall flat on it's face: 5.5
The .NET Compiler Platform is a major update of the C# and VB compilers. But, “Roslyn” is not just about compilers but an open platform that provides an API that provides an object model on top of the C# and VB compilers and language services so anyone can extend the development experience enriching the IDE with refactoring, code analysis and custom diagnostics. It enables building code analysis tools with the same APIs that are used by Visual Studio.
Previously the .net compiler was a black box. Rosyln opens the black box. And inside there are some pretty interesting things.
The original compilers (csc.exe and vbc.exe) are written in Native code. Roslyn os written in managed code, The advantage of building them in managed code is that users can reference the real compilers as libraries from .NET applications (no wrappers needed).
Building a compiler like Roslyn is a big job, it's taken years I estimate the repro contains 4.6 million lines of code and has over 191 contributors. Unless someone is a really ranty commenter, that's a lot of naughts and ones.
So it does beg the question, Why Bother. Well, as a company we have lots of people that use C# and in turn lots of people who would benefit from a more open compiler.
For starters there are the 10s of developers that build the language features in Visual Studio, they now have a clean shared platform to build on top of, they can build new features faster. Then there are the thousands of developers that are building developer productivity. Like the people at DevExpress behind Code Rush which is built on top of Roslyn.
Finally there are 100,000 of VB and C# developers that will benefit from better tooling inside of Visual Studio and the add ons that partners can now build. They may also wish to take advantage of the new SDK to create code fix and diagnostic tools that can now easily be built.
You can get the SDK from the following location: .net compiler platform SDK
There are a couple of things that can trip you up with this.
The first is that you will also need the Visual Studio SDK installed if you want to create code fixes and analysers. Some places give out of date instructions on how to install this. You now install it either when installing VS (by selecting to add the SDK) or if you already have VS 2107 installed you will need to go to File New Project > Open the Installer and select to install the SDK. (In 2015, I think you get to the installer in Program Features > Change).
When you have both SDK's you should now have an project option called "Analyzer with Code Fix" in Visual C# > Extensibility. If you don't you should check that the .net framework selected is above 4.6. If you still do not see the project type, toggle the extension on and off in Tools > Extensions and Updates > Disable. You will have to restart VS between disabling the extension and enabling it.
The first IDE to use Rosyln was Web Matrix (The lightweight editor that's now been super seeded by VS Code) and there was a preview for Visual Studio 2013. However Visual Studio 2015 is the main IDE that Roslyn is used and of course 2017 is also using Roslyn. Roslyn is also on of the foundational building blocks that made Visual Studio for Mac possible.
I've never built a compiler but I have been reliably informed that you can think of the .net compiler as being split into 5 parts. Parser, Symbols, Metadata Import, Binder and IL Emitter. Each of those parts has an API associated with it: Syntax Tree API, Symbol API, Binding and Flow Analysis API, Emit API.
These API's are what Visual Studio will call to implement some of it's productivity capabilities. For example, the Syntax Tree API is used to colourise the source code in the editor or provide collapsible outlining.
There are two layers of APIs. The Compiler, and the workspace. At a compiler level you can take a.net file and figure stuff out about it. Compile a Syntax tree and query the code. However, code if often spead about over a solution and contains multiple projects. To reason over that you will need to use the workspace API's this will ensure that you can think about your code holistically and have more capability since Rosyln will have a better understanding of how your code interconnects.
If you think about your C# code as files then the first step of compiling it is to convert all those strings of text into a Syntax tree that represents the .net code. In Roslyn a Syntax tree is immutable (it can't be changed). To make modifications you will need to copy it and generate a brand new tree.
It's resilient in that even if your c# is a total uncompilable mess, Rosyln isn't going to fail. It will construct a tree and show the errors where is finds them. This is important since Rosyln is running with every keypress you take in Visual Studio. During editing a C# file there will be many times that a C# file is without braces or just plain wrong. Since Rosyln is used to show you where the code is wrong it needs to be able to construct a tree in any situation.
If you ToString a Syntax tree the C# code generated will be an exact replica of the code that went into creating it. Including the line spaces, tabs and all the other Triva that isn't code, but is crucial to reconstruct the originating file.
It might seem like a huge task to construct a new Syntax tree for each and every change you make to any file in your project. However, in the near decade that Roslyn has been under development the team have tuned performance to ensure it is very efficient. I'm not a good enough C# developer to fully understand, however, I have been told that the Roslyn code base is a bit of a master class in managed language performance. I guess this makes sense since the folk writing the C# also understand intimately how it's being compiled to Intermediate Language.
When the C# file is parsed and converted into a tree each element is converted into objects. The bit that make up the important blocks of code are called nodes. When you query a node it contains lots of stuff but at it's core it's the basic structure of the code.
The next type of objects are Syntax Tokens these are nested if you like inside of the nodes and represent all of the C# grammer and information.
All the other stuff that isn't actaully important for compilation but is important to maintain consistency with the original file is called Syntax Triva. This is all the white space, tabs and other elements.
When a project is compiled. There maybe multiple references and Syntax trees, ultimately these will all be brought together and IL will be emitted.
It's hard to construct a mental model of a Syntax Tree just by looking at C# but to query things you will need to understand how it is structured.
Luckily the Roslyn team built a Syntax Tree Viewer so you could get a visualisation at any time. View > Other Windows > Syntax Visualiser.
First up I am going to create a very simple console application. I will need to install the Nuget Package Microsoft.CodeAnalysis. I open up the Package Manager Windows and Type:
Install-Package Microsoft.CodeAnalysis
It can take some time for the code package to install and it's important to remember your console app will need to be targeting .net framework 4.6 and greater.
In the void main I write
var tree = CSharpSyntaxTree.ParseText(@"
class bb {
void b(){
}
}
");
Console.Write(tree);
Console.ReadLine();
You can check to see if the code contains any errors by checking the diagnostics:
var error = tree.GetDiagnostics().Where(n => n.Severity == Microsoft.CodeAnalysis.DiagnosticSeverity.Error).FirstOrDefault();
Console.Write(error);
If I remove a brace and then run the code again. It should now log an error.
To query the tree a little further I use a combination of Linq and the API:
var root = tree.GetRoot();
var method = root.DescendantNodes().OfType().First();
Console.Write(method);
I could also find some code in the method. Fist I will add the following to the Method:
try{
}
catch(Exception ex){
}
I then need to search for the Try Catch instead using TryStatmentSyntax
var trystatement = root.DescendantNodes().OfType().First();
Each block type has different properties, a try has a black for example.
var block = trystatement.Block;
Console.WriteLine(block);
In the try if I add something like a comment, then this will be printed:
try{
// Whats up Bristol
}
We can modify the code and add a return type to the code for example:
var method = root.DescendantNodes().OfType().First();
var trivia = SyntaxFactory.ParseTrailingTrivia(" ");
var returnType = SyntaxFactory.ParseTypeName("string").WithTrailingTrivia(trivia);
var newmethod = method.WithReturnType(returnType);
Console.Write("New: " + newmethod);
You can find the code for the example here: github.com/thebeebs/RoslynCodeFixForColour
I also want to search Properties and so I will add SymbolKind.Property to the array of RegisterSymbolAction
context.RegisterSymbolAction(AnalyzeSymbol, SymbolKind.NamedType, SymbolKind.Property);
Check to see if the Symbol contains color
private static void AnalyzeSymbol(SymbolAnalysisContext context)
{
var symbol = context.Symbol;
if (symbol.Name.ToLower().Contains("color"))
{
var diagnostic = Diagnostic.Create(Rule, symbol.Locations[0], symbol.Name);
context.ReportDiagnostic(diagnostic);
}
}
Thats the code diagnostic taken care of. Next to correct the code fix.
I'm now looking for properties and well as Type Declarations
// Find the type declaration identified by the diagnostic.
SyntaxNode declaration = root.FindToken(diagnosticSpan.Start).Parent.AncestorsAndSelf().OfType().FirstOrDefault();
if (declaration == null) {
declaration = root.FindToken(diagnosticSpan.Start).Parent.AncestorsAndSelf().OfType().FirstOrDefault();
}
Change the signature and name of the CodeFix:
private async Task CorrectSpelling(Document document, SyntaxNode node, CancellationToken cancellationToken)
And Finally I need to make sure the code fix can handle both properties and Type Declarations.
string identifierToken;
if (node.IsKind(SyntaxKind.PropertyDeclaration))
{
identifierToken = ((PropertyDeclarationSyntax)node).Identifier.Text;
}
else
{
identifierToken = ((TypeDeclarationSyntax)node).Identifier.Text;
}
var newName = identifierToken.Replace("olor", "olour");