前言

.Net运行模型,无非就两个过程。一个是调用入口函数,另外一个就是编译入口函数。前者主调用,后者主编译。

概括

一:入口函数:RunMainInternal

所有的.Net程序,包括控制台,Web,窗体等等入口都是Main。而调用这个Main的就是RunMainInternal,从这个函数的名字就可以看出,它的Run就表示是运行Main函数的,而后面的Internal表示是内部的,而非外部所能调用,当然你用一些非常规手段,这个函数还是可以外部的调用的,那是后话。

二:通过LLDB看下它的堆栈

* thread #1, name = 'corerun', stop reason = step over
* frame #0: 0x00007ffff6d53cdf libcoreclr.so`RunMainInternal
frame #1: 0x00007ffff6d53c99 libcoreclr.so`RunMain
frame #2: 0x00007ffff6d4ec76 libcoreclr.so`RunMain
frame #3: 0x00007ffff6d4ea22 libcoreclr.so`RunMain
frame #4: 0x00007ffff6d4f030 libcoreclr.so`Assembly::ExecuteMainMethod
frame #5: 0x00007ffff6dc2b9c libcoreclr.so`CorHost2::ExecuteAssembly
frame #6: 0x00007ffff6d1a6c1 libcoreclr.so`::coreclr_execute_assembly
frame #7: 0x000055555556f9f9 corerun`run(config=0x00007fffffffdbc0)
frame #9: 0x00007ffff7829d90 libc.so.6`__libc_start_call_main
frame #10: 0x00007ffff7829e40 libc.so.6`__libc_start_main_impl
frame #11: 0x000055555556c0e5 corerun`_start + 37

可以看到它在Linux平台下面,是通过Glibc OR Newlibc调用的。而它的前面一个函数就是RunMain,这个通过它名字也可以看到它就是运行Main函数的。

三:编译函数PreStubWorker

这个函数顾名思义,预桩工作。就是插桩的意思。每次编译一个函数,都会经过它。它会调用RyuJIT把函数编译成机器码,然后再返回到这个被编译函数的函数头地址运行。

部分代码:

extern "C" PCODE STDCALL PreStubWorker(TransitionBlock* pTransitionBlock, MethodDesc* pMD)
{
PCODE pbRetVal = NULL; BEGIN_PRESERVE_LAST_ERROR; STATIC_CONTRACT_THROWS;
STATIC_CONTRACT_GC_TRIGGERS;
STATIC_CONTRACT_MODE_ANY;
STATIC_CONTRACT_ENTRY_POINT;
// 此处省略一万字
}

可以看到它只是带了一个MethodDesc,也就是函数的描述结构体。通过它可以找到Module,MSIL,然后进行一个编译。

四:看下它的堆栈

 thread #1, name = 'corerun', stop reason = breakpoint 2.1
* frame #0: 0x00007ffff6ef55d0 libcoreclr.so`::PreStubWorker
frame #2: 0x00007ffff731e14f libcoreclr.so`CallDescrWorkerInternal
frame #3: 0x00007ffff6faae98 libcoreclr.so`CallDescrWorkerWithHandler
frame #4: 0x00007ffff6fabb8d libcoreclr.so`MethodDescCallSite::CallTargetWorker
frame #5: 0x00007ffff6d2e4b3 libcoreclr.so`MethodDescCallSite::Call
frame #6: 0x00007ffff6d53f6f libcoreclr.so`RunMainInternal
frame #7: 0x00007ffff6d53c99 libcoreclr.so`RunMain
frame #8: 0x00007ffff6d4ec76 libcoreclr.so`RunMain
frame #9: 0x00007ffff6d4ea22 libcoreclr.so`RunMain
frame #10: 0x00007ffff6d4f030 libcoreclr.so`Assembly::ExecuteMainMethod
frame #11: 0x00007ffff6dc2b9c libcoreclr.so`CorHost2::ExecuteAssembly
frame #12: 0x00007ffff6d1a6c1 libcoreclr.so`::coreclr_execute_assembly
frame #13: 0x000055555556f9f9 corerun`run(config=0x00007fffffffdbc0) argv=0x00007fffffffddb8) at corerun.cpp:624:21
frame #15: 0x00007ffff7829d90 libc.so.6`__libc_start_call_main
frame #16: 0x00007ffff7829e40 libc.so.6`__libc_start_main_impl
frame #17: 0x000055555556c0e5 corerun`_start + 37

它刚好在RunMainInternal的后面,说明它是在被调用之后,才会进行编译。

结尾

作者:江湖评谈

.Net7 CLR的调用函数和编译函数的更多相关文章

  1. C++调用C中编译过的函数要加extern "C"

    C++语言支持函数重载,C语言不支持函数重载.函数被C++编译后在库中的名字与C语言的不同.假设某个C 函数的声明如下:void foo(int x, int y);该函数被C 编译器编译后在库中的名 ...

  2. C/C++:函数的编译方式与调用约定以及extern “C”的使用

    转自:https://www.cnblogs.com/qinfengxiaoyue/archive/2013/02/04/2891908.html 函数在C++编译方式与C编译方式下的主要不同在于:由 ...

  3. 【原创】Matlab.NET混合编程技巧之直接调用Matlab内置函数

                  本博客所有文章分类的总目录:[总目录]本博客博文总目录-实时更新    Matlab和C#混合编程文章目录 :[目录]Matlab和C#混合编程文章目录 在我的上一篇文章[ ...

  4. Matlab.NET混合编程技巧之——直接调用Matlab内置函数(附源码)

    原文:[原创]Matlab.NET混合编程技巧之--直接调用Matlab内置函数(附源码) 在我的上一篇文章[原创]Matlab.NET混编技巧之——找出Matlab内置函数中,已经大概的介绍了mat ...

  5. jQuery 2.0.3 源码分析Sizzle引擎 - 编译函数(大篇幅)

    声明:本文为原创文章,如需转载,请注明来源并保留原文链接Aaron,谢谢! 从Sizzle1.8开始,这是Sizzle的分界线了,引入了编译函数机制 网上基本没有资料细说这个东东的,sizzle引入这 ...

  6. mfc 调用Windows的API函数实现同步异步串口通信(源码)

    在工业控制中,工控机(一般都基于Windows平台)经常需要与智能仪表通过串口进行通信.串口通信方便易行,应用广泛. 一般情况下,工控机和各智能仪表通过RS485总线进行通信.RS485的通信方式是半 ...

  7. 在VS2012中采用C++中调用DLL中的函数 (4)

    这两天因为需要用到VS2012来生成一个DLL代码,但是之前并没有用过DLL相关的内容,从昨天开始尝试调试DLL的文件调用,起初笔者在网络上找到了3片采用VSXXX版本进行调试的例子,相关的内容见本人 ...

  8. 在C++中调用DLL中的函数 (3)

    1.dll的优点 代码复用是提高软件开发效率的重要途径.一般而言,只要某部分代码具有通用性,就可将它构造成相对独立的功能模块并在之后的项目中重复使用.比较常见的例子是各种应用程序框架,ATL.MFC等 ...

  9. 在C++中调用DLL中的函数 (2)

    应用程序使用DLL可以采用两种方式: 一种是隐式链接,另一种是显式链接.在使用DLL之前首先要知道DLL中函数的结构信息. Visual C++6.0在VC\bin目录下提供了一个名为Dumpbin. ...

  10. 在C++中调用DLL中的函数

    如何在C++中调用DLL中的函数 应用程序使用DLL可以采用两种方式:一种是隐式链接,另一种是显式链接.在使用DLL之前首先要知道DLL中函数的结构信息.Visual C++6.0在VC\bin目录下 ...

随机推荐

  1. 学习lua-02,引入其他lua文件,调用全局方法

    local testlua = require "testlua" --testlua.compertoNum(1, 2, 4, 5, 11, 23, 543, 123, 45, ...

  2. 博弈论[leetocde913]

    class Solution { static final int MOUSE_WIN = 1; static final int CAT_WIN = 2; static final int DRAW ...

  3. java jdk8安装之后java -version失败

    将此目录下的(C:\Program Files (x86)\Common Files\Oracle\Java\javapath_target_86850671)三个.exe文件删除即可!

  4. ethcat开发记录 一

    一.方案 1.移植开源方案SOEM 2.专用芯片 二.SOEM移植 (一)硬件 stm32f407,168M PHY:LAN8720A (ii) Points to note 1, the PHY a ...

  5. 浏览器垃圾回收机制:栈垃圾回收、堆垃圾回收、新生区老生区、Scavenge算法、标记-清除算法、标记-整理算法、全停顿、增量标记

    浏览器垃圾回收机制根据数据的存储方式分为栈垃圾回收和堆垃圾回收. 栈垃圾回收的方式非常简便,当一个函数执行结束之后,JavaScript 引擎会通过向下移动 ESP 来销毁该函数保存在栈中的执行上下文 ...

  6. redis 简单安装

    参考官网,安装步骤基本照搬redis官网,其他只做简单说明https://redis.io/download/https://redis.io/docs/getting-started/install ...

  7. ceph pg修复过程

    1.通过命令查看哪些pg状态不一致 ceph pg dump|grep inconsistent 2.根据输出的pg id 进行一致性检查 ceph pg scrub 1.23 instructing ...

  8. 【分享】HMCL启动器

    HMCL-3.3.173.exe 链接:https://pan.baidu.com/s/1KpEm3K0asNhPAXjufsCGIA 提取码:93kr

  9. 求小于N的最大素数

    问题 求小于N的最大素数 分析 枚举:从可能的集合中一一列举各元素 枚举过程中需要考虑的问题: 给出解空间 减少搜索的空间 采用合适的搜索顺序 枚举关键字(枚举核心):减少规模 代码实现 1 impo ...

  10. java输入一个字符串,要求将该字符串中出现的英文字母, * 按照顺序输出,区分大小写,且大写优先

    public static void main(String[] args) { String input ="A8r4c5jaAjp#7"; //转为char[] char[] ...