PowerUp is a collection or productivity utilities, disassembly and decompilation tools for multiple languages and platforms.
The project features the following utilities and tools:
- Live IDE Watcher (For C#, GO and Rust).
- .NET JIT Dissasembler.
- .NET IL Compiler.
- .NET Quick Benchmark.
- .NET Console with rich formatting.
- Others.
A watcher application that monitors source code files and IR, IL files and compiles them and later decompiles and disassembles them to produce IL,IR and X86 ASM outputs. All compilers support multiple features such as Jump Guides, X86, IR Assembly documentation, Source Maps and more.
To run the application, you need to compile Power.Up watcher and run it from the terminal by providing the correct compiler(s) and providing input and output files like so:
.\PowerUp.Watcher.exe -rs D:\01_BA\PowerUp\_code.rs D:\01_BA\PowerUp\_outRust.asm
The application supports running multiple compilers at the same time:
.\PowerUp.Watcher.exe -rs D:\01_BA\PowerUp\_code.rs D:\01_BA\PowerUp\_outRust.asm -go D:\01_BA\PowerUp\_code.go D:\01_BA\PowerUp\_outGO.asm -cs D:\01_BA\PowerUp\_code.cs D:\01_BA\PowerUp\_out6.asm D:\01_BA\PowerUp\_out6.il
C# is the only language that supports (at the moment) running code, interactive mode, benchmarking and class and struct layouts.
This demo shows the layouts feature:
C# supports multiple attributes that can be used to control compilation:
[Bench] <- Used on a method to Benchmak source code, the method needs to have no input arguments.
[Run] <- Used on a method to Run source code, the method needs to have no input arguments.
Print(X) <- This is a C# function that is used to print values when the code uses the Run attribuute
[ShowGuides] <- Used on Any to enable jump guides in the ASM outputs.
[ShowASMDocs] <- Used on Any to turn on ASM code documentation.
[ShowASMDocs(offset={X}) <- Used on Any to turn on ASM code documentation. The offset argument controls the offset from ASM code where the docs are rendered.
[ShortAddr] <- Used on Any to have short address lines in the output ASM code.
[ShortAddr(by={X})] <- Used on Any to have short address lines in the output ASM code. The additional argument is used to control the cut level.
GO Lang is not disassembled to X86 asm, but GO-specific IR is almost a 1-to-1 assembly, so most optimizations have already happened. This makes this type of IR format viable for performance analysis.
GO supports multiple comment-level attributes that can be used to control compilation:
//up:showGuides <- Used to enable jump guides in the ASM outputs.
//up:showASMDocs <- Used to turn on ASM code documentation.
//up:showASMDocs offset = {X} <- Used to enable jump guides in the ASM outputs. The offset argument decides the position of the documentation.
Rust being a LLVM based language will support default compiler flags like optimization levels and source code maps. (This feature will be also added to c# and GO with time)
Rust supports multiple comment-level attributes that can be used to control compilation:
//up:showGuides <- Used to enable jump guides in the ASM outputs.
//up:showASMDocs <- Used to turn on ASM code documentation.
//up:showASMDocs offset = {X} <- Used to enable jump guides in the ASM outputs. The offset argument decides the position of the documentation.
//up:optimization level = {X} <- Set the compilation optimization level (-C opt-level)
//up:showCode <- Show source code maps, that map assembly instructions to source code.
Note: IL Compilation is still a work in progress, and it will take some time before it's robust.
I use this application daily to create infographics: https://leveluppp.ghost.io/infographics/
If you would like to see more features, there's a Youtube video showing more features in detail:
https://www.youtube.com/watch?v=EZV_9sCrptc
To make your outputs look nice in Visual Studio Code or any other editor, you must install an X86 assembly syntax.
I use a modified version of the 13xforever X86 ASM syntax: https://github.com/13xforever/x86-assembly-textmate-bundle (I hope to be able to bundle it with this tool)
Additionally you will need Cascadia Mono for nice guides and other features.
To disassemble JIT code, you need to extract the method info using reflection and call the extension method ToAsm(). This will get native assembly code in tier0. To get a higher Tier, you need to provide the delegate that will call the method, and over time the compiler will re-JIT it to Tier1. If you want to skip this, you can compile your project with:
<TieredCompilation>false</TieredCompilation>
Example:
typeof(Program)
.GetMethod("DecompilationTestMethod")
.ToAsm()
.Print();
Example for Tier1:
typeof(Program)
.GetMethod("DecompilationTestMethod")
.ToAsm(() => p.DecompilationTestMethod(10))
.tier1
.Print();
There are no guarantees that Tier1 will always be available; Very short and fast methods tend not to be promoted (I'm working on a fix). Code that contains loops is always optimized so you should not bother with this method and switch back to the simpler version.
You can write IL Code as a string and compile it to a type; later, it can be fed to the JIT Decompiler. Native IL compilation is an interesting research topic since you can emit instructions that are not used or used in a completely different context.
var il = @"
.method public hidebysig instance int32 X (int32 x)
{
.maxstack 8
IL_0000: ldarg.1
IL_0001: ldc.i4.s 97
IL_0003: callvirt instance int32 [System.Private.CoreLib]System.String::IndexOf(char)
IL_0008: ret
}
";
ILCompiler compiler = new ILCompiler();
var type = compiler.Compile(il);
var asm = type.CompiledType.ToAsm();
asm.Print();
To create a valid benchmark that tests various examples and it's accurate, one needs to go to great lengths, and there are multiple libraries for that already. This Quick Benchmark is suitable for running only on a single method and test performance; it's handy for live demos, slides, videos, etc.
Quick Benchmark provides a run count and a warmup count. It does all of the needed operations to provide valid measurements like computing the mean and standard deviation and provide memory usage.
There's also a JIT Dissasembler that tracks the generated assembly code of the tested method across JIT compilation tiers.
To run the Benchmark, you need to call the "Measure" method like so:
double sum = 0;
QuickBenchmark.Measure(() => {
for (int i = 0; i < 10_000_000; i++)
sum *= i;
});