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

来源:https://www.cnblogs.com/hez2010/archive/2020/02/27/12374911.html
-Advertisement-
Play Games

直接使用 CIL - .NET 上的彙編語言編寫 .NET Standard 類庫 ...


前言

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

CLR 之上的語言 C#、F#、VB.NET 等語言的類型系統固然設計得不錯,但是有的時候我們需要一些操作繞過類型系統的檢查,或者有的時候語言本身並不能滿足我們的需求。

需要使用 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/jit

準備工作

首先我們創建一個 .NET Standard 項目:

mkdir MyILProject
cd MyILProject
dotnet new classlib

然後創建 global.jsonnuget.config 文件用於配置 SDK:

dotnet new global
dotnet new nuget

global.json 的內容修改為如下,添加 IL SDK 來源:

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

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

<?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


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、條件語句 if if…else… 多重if if嵌套 三目運算符 (化簡的if else) if 條件: 條件成立執行代碼1 條件成立執行代碼2 if False: print('if判斷為假,不執行') print('有減進的語句都在if里') print('沒有縮進,我不是if,可以執行') ...
  • c語言心形告白代碼實現 1.彩色告白 include include include include define U 0.1 define V 0.053 void SetColor(unsigned short ForeColor,unsigned short BackGroundColor) ...
  • go 語言中文網(每日資訊)_2020 02 27 一、Go 語言中文網 1. "如何正確看待 Google 宣佈 Fuchsia 操作系統沒有選 Go 作為終端開發語言 " 2. "Actor 還是 CSP?Go 中的併發模式還能講這麼細緻" 3. "【每日一庫】讓你的 json 可以 grep: ...
  • 行為模式:關註系統中對象之間的相互交互,研究運行時對象之間的相互通信和協作,明確對象職責 1.模板方法模式(template method) 定義了一個操作中的演算法骨架,將某些步驟延遲到子類中實現。這樣,新的子類可以在不改變一個演算法結構的前提下重新定義該演算法的某些特定步驟。 即:處理步驟父類中定義好 ...
  • 1、什麼變數 變數來源於數學,從根本上說,變數相當於是對一塊數據存儲空間的命名,程式可以通過定義一個變數來申請一塊數據存儲空間,之後可以通過引用變數名來使用這塊存儲空間。 1.1變數聲明 Go 語言變數名由字母、數字、下劃線組成,其中首個字母不能為數字。 第一種,指定變數類型,聲明後若不賦值,使用默 ...
  • 下載MyEclipse10以及破解包 "MyEclipse10:" 提取碼:020c "破解包" 提取碼:mycj 註:破解包內含有破解教程,很詳細,這裡就不多說了 MyEclipse10漢化 操作系統:win10 "漢化包" 提取碼:nbsb 操作步驟 1:將漢化包中的language的文件夾全粘 ...
  • 1 考察知識點 本題考察的知識點有以下幾個: 1. Keys 和 Scan 的區別 2. Keys 查詢的缺點 3. Scan 如何使用? 4. Scan 查詢的特點 2 解答思路 1. Keys 查詢存在的問題 2. Scan 的使用 3. Scan 的特點 3 Keys 使用相關 1)Keys ...
  • springBoot1.x升級到springBoot2.x,出現預設的mysql驅動包版本的問題,需要修改jdbc驅動類和連接字元串 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...