直接使用匯編編寫 .NET Standard 庫

前言

Common Language Runtime(CLR)是一個很強大的運行時,它接收 Common Intermediate Language(CIL) 的輸入並最終產生機器代碼並執行。CIL 在 CLR 上至關於 ASM 彙編代碼的存在。git

CLR 之上的語言 C#、F#、VB.NET 等語言的類型系統當然設計得不錯,可是有的時候咱們須要一些操做繞過類型系統的檢查,或者有的時候語言自己並不能知足咱們的需求。github

須要使用 CIL 的常見場景:編程

  1. 咱們須要繞過類型系統,在類型系統上面 「開洞」。
  2. 咱們須要優化程序的性能,直接使用 CIL 編程能夠如同使用匯編同樣徹底的控制程序的邏輯,對程序進行人肉優化。
  3. 直接利用 C#、F# 等語言編譯成的 CIL 有其獨特的模式,容易被反編譯軟件從 CIL 還原爲源代碼,而若是直接採用 CIL 編程則很容易避開編譯器生成代碼的固有模式,使得代碼無需進行任何混淆便可讓全部反編譯器失效。

須要注意:CLR 的 JIT 部分優化依賴於 CIL 的特定模式,直接採用 CIL 進行編程而不利用 C# 等語言的編譯器生成特定模式的 CIL 可能致使優化失效,如向量化、模式匹配緩存和常數時間優化等,所以在直接使用 CIL 進行編程時最好對 CLR 的 JIT 有必定了解,以規避潛在的性能問題,JIT 的源代碼在 https://github.com/dotnet/runtime/tree/master/src/coreclr/src/jitjson

準備工做

首先咱們建立一個 .NET Standard 項目:api

mkdir MyILProject
cd MyILProject
dotnet new classlib

而後建立 global.jsonnuget.config 文件用於配置 SDK:緩存

dotnet new global
dotnet new nuget

global.json 的內容修改成以下,添加 IL SDK 來源:bash

{
  "msbuild-sdks": {
    "Microsoft.NET.Sdk.IL": "3.0.0-preview-27318-01"
  }
}

而後打開 nuget.config,將內容修改以下,添加 .net core myget 源:ide

<?xml version="1.0" encoding="utf-8"?>
<configuration>
  <packageSources>
    <add key="dotnet-core" value="https://dotnet.myget.org/F/dotnet-core/api/v3/index.json" />
  </packageSources>
</configuration>

以前建立的爲 C# 類庫項目,可是咱們此時須要的是 IL 類庫項目,所以將 MyILProject.csproj 文件重命名爲 MyILProject.ilproj工具

打開 MyILProject.ilproj 文件,引入 IL SDK,並添加一系列的屬性(如:輸出類型、優化選項、工具鏈等):性能

<Project Sdk="Microsoft.NET.Sdk.IL">

  <PropertyGroup>
    <OutputType>Library</OutputType>
    <TargetFramework>netstandard2.1</TargetFramework>
    <DebugOptimization>IMPL</DebugOptimization>
    <DebugOptimization Condition="'$(Configuration)' == 'Release'">OPT</DebugOptimization>
    <MicrosoftNetCoreIlasmPackageVersion>3.0.0-preview-27318-01</MicrosoftNetCoreIlasmPackageVersion>
  </PropertyGroup>
  
</Project>

至此,萬事俱備

第一個文件

咱們刪除掉原有的 C# 代碼文件 Class1.cs,建立代碼文件 Class1.il,並添加如下 CIL 代碼並保存:

.assembly MyILProject
{
    .ver 1:0:0:0
}

.module MyILProject.dll

.class public auto ansi sealed MyILProject.Class1
  extends [System]System.Object
{
  .method public hidebysig static int32 Hello(int32) cil managed
  {
    .maxstack 4
    
    ldstr "Hello World!"
    call void [System.Console]System.Console::WriteLine(string)

    ldarg.0
    ret
  }
}

以上代碼中,.assembly 標識了程序集名稱,.module 標識了模塊名稱,通常來講這兩個名字和項目名稱保持一致。

而後咱們建立了一個 class Class1,位於 MyILProject 這個 namespace 下,該 classpublic sealed 的,且繼承自 System.Object

最後咱們添加了一個靜態方法 int Hello(int),該方法調用 System.Console.WriteLine 輸出字符串 Hello world!,而後加載參數的值後返回該值。

測試代碼

咱們在上級目錄建立一個測試項目試試:

cd ..
mkdir Test
cd Test
dotnet new console
dotnet add reference ../MyILProject

而後修改 Program.cs

using System;
using MyILProject;

namespace Test
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine(Class1.Hello(25));
        }
    }
}

運行

dotnet run

能夠看到輸出爲:

Hello world!
25

與咱們所指望的一致。

而後咱們試一下實例化 Class1

var x = new Class1();

卻發現報錯:

Program.cs(10,28): error CS1729: 'Class1' does not contain a constructor that takes 0 arguments [...\Test.csproj]

這是由於,咱們沒有爲這個類建立構造方法,那麼很簡單,咱們只須要加一個構造方法便可,要注意構造方法特有的方法名爲 .ctor

.method public hidebysig specialname rtspecialname instance void .ctor () cil managed 
{
  .maxstack 8
    
  ldarg.0
  call instance void [System.Private.CoreLib]System.Object::.ctor()
  nop
  ret
}

而後就能夠成功調用了!

添加引用

你會發現一個問題,上述代碼雖然能正常運行,可是編譯的時候卻存在警告:

Class.il(9): warning : Reference to undeclared extern assembly 'mscorlib'. Attempting autodetect [...\MyILProject.ilproj]      
Class.il(15): warning : Reference to undeclared extern assembly 'System.Console'. Attempting autodetect [...\MyILProject.ilproj]
Class.il(26): warning : Reference to undeclared extern assembly 'System.Private.CoreLib'. Attempting autodetect [...\MyILProject.ilproj]

這是由於咱們並無聲明咱們引入的庫 mscorlibSystem.ConsoleSystem.Provate.CoreLib,所幸的是,由於這些是 .NET Core SDK 中自帶的庫所以編譯器能夠自動查找並補上引用,因此沒有報錯,不然運行的時候會拋出異常: System.IO.FileNotFoundException: Could not load file or assembly xxxxx

若是想消除這些警告,咱們能夠建立一個頭文件引用這些庫,而後在 CIL 代碼文件的頭部 #include 頭文件,示例以下:

MyILProject 中新建 include 文件夾,建立一個 include.h:

.assembly extern System.Runtime
{
  .publickeytoken = ( B0 3F 5F 7F 11 D5 0A 3A )
  .ver 4:0:0:0
}

.assembly extern System.Console
{
  .publickeytoken = ( B0 3F 5F 7F 11 D5 0A 3A )
  .ver 4:0:0:0
}

.assembly extern System.Private.CoreLib
{
  .publickeytoken = ( B0 3F 5F 7F 11 D5 0A 3A )
  .ver 4:0:0:0
}

而後在 Class1.il 頭部加一行 #include "include.h" 包含該文件。

最後修改 MyILProject.ilproj,將 include 文件夾加入 INCLUDE 查找目錄(-INCLUDE=...):

<Project Sdk="Microsoft.NET.Sdk.IL">

  <PropertyGroup>
    <OutputType>Library</OutputType>
    <TargetFramework>netstandard2.1</TargetFramework>
    <DebugOptimization>IMPL</DebugOptimization>
    <DebugOptimization Condition="'$(Configuration)' == 'Release'">OPT</DebugOptimization>
    <MicrosoftNetCoreIlasmPackageVersion>3.0.0-preview-27318-01</MicrosoftNetCoreIlasmPackageVersion>
    <IlasmFlags>$(IlasmFlags) -INCLUDE=include</IlasmFlags>
  </PropertyGroup>
  
</Project>

此次咱們再次嘗試編譯,就不會報錯了。

CLI

上面的內容只簡單的使用了一些 CIL 語法,然而 CIL 也是很是強大的,包含有不少內容,具體能夠參考 Common Language Infrastructure(CLI),這部分的內容包含在標準 ECMA-355 中,截至本文發佈,最新的 CLI 標準版本是 2012 年發佈的第六版。

ECMA-355:https://www.ecma-international.org/publications/standards/Ecma-335.htm ,歡迎各位讀者前去閱讀。

應用案例

.NET BCL 中提供了一個特殊的庫:System.Runtime.CompilerServices.Unsafe,這個庫容許你無視 C# 的類型系統進行各類類型轉換等的騷操做,這是你用 C# 不管如何都不可能寫出來的,官方也知道這一點,所以該庫徹底是直接使用 CIL 編寫的,源代碼可參考:https://github.com/dotnet/runtime/blob/master/src/libraries/System.Runtime.CompilerServices.Unsafe/src/System.Runtime.CompilerServices.Unsafe.il

相關文章
相關標籤/搜索