欧美一区二区三区老妇人-欧美做爰猛烈大尺度电-99久久夜色精品国产亚洲a-亚洲福利视频一区二区

聊一聊被 .NET程序員 遺忘的 COM 組件

一:背景

1.講故事

最近遇到了好幾起和 COM 相關(guān)的Dump,由于對(duì) COM 整體運(yùn)作不是很了解,所以分析此類dump還是比較頭疼的,比如下面這個(gè)經(jīng)典的 COM 調(diào)用棧。

創(chuàng)新互聯(lián)公司專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務(wù),包含不限于做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)、馬村網(wǎng)絡(luò)推廣、成都小程序開發(fā)、馬村網(wǎng)絡(luò)營(yíng)銷、馬村企業(yè)策劃、馬村品牌公關(guān)、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運(yùn)營(yíng)等,從售前售中售后,我們都將竭誠(chéng)為您服務(wù),您的肯定,是我們最大的嘉獎(jiǎng);創(chuàng)新互聯(lián)公司為所有大學(xué)生創(chuàng)業(yè)者提供馬村建站搭建服務(wù),24小時(shí)服務(wù)熱線:18982081108,官方網(wǎng)址:chinadenli.net


0:044> ~~[138c]s
win32u!NtUserMessageCall+0x14:
00007ffc`5c891184 c3              ret
0:061> k
 # Child-SP          RetAddr               Call Site
00 0000008c`00ffec68 00007ffc`5f21bfbe     win32u!NtUserMessageCall+0x14
01 0000008c`00ffec70 00007ffc`5f21be38     user32!SendMessageWorker+0x11e
02 0000008c`00ffed10 00007ffc`124fd4af     user32!SendMessageW+0xf8
03 0000008c`00ffed70 00007ffc`125e943b     xxx!DllUnregisterServer+0x3029f
04 0000008c`00ffeda0 00007ffc`125e9685     xxx!DllUnregisterServer+0x11c22b
05 0000008c`00ffede0 00007ffc`600b50e7     xxx!DllUnregisterServer+0x11c475
06 0000008c`00ffee20 00007ffc`60093ccd     ntdll!LdrpCallInitRoutine+0x6f
07 0000008c`00ffee90 00007ffc`60092eef     ntdll!LdrpProcessDetachNode+0xf5
08 0000008c`00ffef60 00007ffc`600ae319     ntdll!LdrpUnloadNode+0x3f
09 0000008c`00ffefb0 00007ffc`600ae293     ntdll!LdrpDecrementModuleLoadCountEx+0x71
0a 0000008c`00ffefe0 00007ffc`5cd7c00e     ntdll!LdrUnloadDll+0x93
0b 0000008c`00fff010 00007ffc`5d47cf78     KERNELBASE!FreeLibrary+0x1e
0c 0000008c`00fff040 00007ffc`5d447aa3     combase!CClassCache::CDllPathEntry::CFinishObject::Finish+0x28 [onecore\com\combase\objact\dllcache.cxx @ 3420] 
0d 0000008c`00fff070 00007ffc`5d4471a9     combase!CClassCache::CFinishComposite::Finish+0x4b [onecore\com\combase\objact\dllcache.cxx @ 3530] 
0e 0000008c`00fff0a0 00007ffc`5d3f1499     combase!CClassCache::FreeUnused+0xdd [onecore\com\combase\objact\dllcache.cxx @ 6547] 
0f 0000008c`00fff650 00007ffc`5d3f13c7     combase!CoFreeUnusedLibrariesEx+0x89 [onecore\com\combase\objact\dllapi.cxx @ 117] 
10 (Inline Function) --------`--------     combase!CoFreeUnusedLibraries+0xa [onecore\com\combase\objact\dllapi.cxx @ 74] 
11 0000008c`00fff690 00007ffc`6008a019     combase!CDllHost::MTADllUnloadCallback+0x17 [onecore\com\combase\objact\dllhost.cxx @ 929] 
12 0000008c`00fff6c0 00007ffc`6008bec4     ntdll!TppTimerpExecuteCallback+0xa9
13 0000008c`00fff710 00007ffc`5f167e94     ntdll!TppWorkerThread+0x644
14 0000008c`00fffa00 00007ffc`600d7ad1     kernel32!BaseThreadInitThunk+0x14
15 0000008c`00fffa30 00000000`00000000     ntdll!RtlUserThreadStart+0x21

為了做一個(gè)簡(jiǎn)單的梳理,我們搭建一個(gè)簡(jiǎn)單的多語(yǔ)言 COM 互操作。

二:COM 多語(yǔ)言互操作

1. 背景

可能很多新生代的程序員都不知道 COM ,最多也只聽過(guò)這個(gè)名詞,其實(shí)在 Windows 上有海量的 COM 組件,這些組件信息都是注冊(cè)在 HKEY_CLASSES_ROOT\CLSID 節(jié)點(diǎn)目錄,截圖如下:

這個(gè)和微服務(wù)中的 注冊(cè)中心 是一個(gè)道理,這一篇我們用 C# 寫一個(gè)COM組件,用 C++ 去調(diào)用。

2. C# 寫一個(gè) COM 組件

寫一個(gè) .NET Framework 4.8 下的 32bit FlyCom 組件,一個(gè)接口,一個(gè)實(shí)現(xiàn)類,具體原理后續(xù)再分析,先搭建嘗嘗鮮, C# 代碼如下:


namespace FlyCom
{
    [Guid("31A3CED7-B4F1-4D59-881A-EA1D7ABCC4CF")]
    public interface BaseFly
    {
        [DispId(1)]
        string Show(string str);
    }

    [Guid("270C3ED3-053D-4324-9176-9C3FA2BE58A7")]
    [ProgId("FlyCom.Show")]
    public class Fly : BaseFly
    {
        public string Show(string str)
        {
            return $"str={str}, length={str.Length}";
        }
    }
}

這里簡(jiǎn)單說(shuō)一下:

  1. Guid

一個(gè)是接口(BaseFly) 的唯一碼,即 IID 信息, 一個(gè)是 COM組件的 唯一碼,叫做 CLSID。

  1. ProgId

因?yàn)?GUID 不方便記憶,所以給這個(gè) COM組件 取一個(gè)別名叫 FlyCom.Show 。

  1. DispId

這個(gè)是為了遵循 COM多語(yǔ)言互通下的 vtable調(diào)用標(biāo)準(zhǔn),表示第一個(gè)接口方法是 Show,后續(xù)再聊。

有了代碼,接下來(lái)還要做三個(gè)配置。

  • 對(duì) COM 的可見(jiàn)性

修改 AssemblyInfo.cs 中的 ComVisible = true,參考如下:


// Setting ComVisible to false makes the types in this assembly not visible
// to COM components.  If you need to access a type in this assembly from
// COM, set the ComVisible attribute to true on that type.
[assembly: ComVisible(true)]

  • 生成簽名

一般來(lái)說(shuō),將 com 放到 注冊(cè)表,最好都生成一個(gè)強(qiáng)簽名,否則會(huì)有警告提示。

  • 注冊(cè) com 互操作

在屬性面板中,選擇 Build 選項(xiàng)卡,選中 Register for COM interop 選項(xiàng)即可。

3. 注冊(cè) COM 到注冊(cè)表

要將 com組件 放到注冊(cè)表,需要使用注冊(cè)表編輯工具 regasm。


Microsoft Windows [版本 10.0.19042.746]
(c) 2020 Microsoft Corporation. 保留所有權(quán)利。

C:\Users\Administrator>cd /d C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools\x64

C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools\x64>C:\Windows\Microsoft.NET\Framework\v4.0.30319\regasm.exe D:\net6\ConsoleApp1\FlyCom\bin\Debug\FlyCom.dll /tlb:FlyCom.tlb  /CodeBase
Microsoft .NET Framework 程序集注冊(cè)實(shí)用工具版本 4.8.4084.0
(適用于 Microsoft .NET Framework 版本 4.8.4084.0)
版權(quán)所有 (C) Microsoft Corporation。保留所有權(quán)利。

成功注冊(cè)了類型
成功注冊(cè)了導(dǎo)出到“D:\net6\ConsoleApp1\FlyCom\bin\Debug\FlyCom.tlb”的程序集和類型庫(kù)

C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools\x64>

從輸出中可以看到已成功注冊(cè),并且生成了一個(gè) FlyCom.tlb 代理文件,接下來(lái)可以到注冊(cè)表中驗(yàn)證一下 GUID=270C3ED3-053D-4324-9176-9C3FA2BE58A7 注冊(cè)項(xiàng)以及別名為 FlyCom.Show 的注冊(cè)項(xiàng)。


4. 使用 C++ 調(diào)用

要想 C++ 調(diào)用 C# 寫的 COM 組件,就像 RPC 調(diào)用一樣,直接自動(dòng)生成的代理文件即可,將 FlyCom.tlb 復(fù)制到 根目錄,并且將程序改成 Win32 位,截圖如下:

接下來(lái)就是完整的 C++ 代碼。


#include <Windows.h>
#include <string.h>
#include <iostream>

#import "FlyCom.tlb" named_guids raw_interface_only

using namespace std;

int main()
{
	CoInitialize(NULL);

	FlyCom::BaseFlyPtr ptr;

	ptr.CreateInstance("FlyCom.Show");

	wchar_t* c = ptr->Show(L"hello world");

	wprintf(L"%s", c);

	getchar();
}

將程序跑起來(lái)后,真的很完美。

從 C++ 調(diào)用 COM 的流程圖可以很清楚的看到,這是面向接口編程的方式,非常完美。

三:COM 多語(yǔ)言互通原理

1. 架構(gòu)圖

千言萬(wàn)語(yǔ)不及一張圖。

這就是 COM 能夠?qū)崿F(xiàn)多語(yǔ)言互通的規(guī)范,熟悉 C++ 的朋友肯定知道 vtable ,C++ 能夠?qū)崿F(xiàn)多態(tài),全靠這玩意,COM 也是用了 vtable 這套模式,所以諸如 JAVA,C#,VBS 必須在二進(jìn)制層面將代碼組織成上圖這種形式,才能實(shí)現(xiàn) COM 的互通。

所以在 C# 中你看到的 DispId 特性就是為了按照 vtable 方式進(jìn)行組織,對(duì)于 ole32 和 combase 這些 COM 運(yùn)行環(huán)境的基石,我們后續(xù)用 windbg 來(lái)解讀一下,這一篇就先到這里,希望對(duì)你有幫助。

分享題目:聊一聊被 .NET程序員 遺忘的 COM 組件
分享網(wǎng)址:http://chinadenli.net/article42/dsoidec.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站制作、品牌網(wǎng)站建設(shè)網(wǎng)站導(dǎo)航、網(wǎng)站維護(hù)用戶體驗(yàn)、定制網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

成都定制網(wǎng)站建設(shè)