Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 5개 있습니다.)
(시리즈 글이 6개 있습니다.)
개발 환경 구성: 300. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법
; https://www.sysnet.pe.kr/2/0/11052

.NET Framework: 828. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/11884

개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기
; https://www.sysnet.pe.kr/2/0/12118

.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)
; https://www.sysnet.pe.kr/2/0/12120

.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석
; https://www.sysnet.pe.kr/2/0/12126

.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)
; https://www.sysnet.pe.kr/2/0/12127




C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법

예전에 트윗으로 한번 소개해 드렸는데, .NET DLL에서도 Win32 C/C++에서 그랬던 것처럼 함수를 export시키는 것이 가능합니다. 아래의 글에 대한 덧글을 보면,

Is is possible to export functions from a C# DLL like in VS C++?
; http://stackoverflow.com/questions/4818850/is-is-possible-to-export-functions-from-a-c-sharp-dll-like-in-vs-c

IL 언어 자체에서도 .export 지시자를 통해 이 작업이 가능하기 때문에 ildasm.exe/ilasm.exe의 조합으로 처리할 수 있습니다. 물론 그렇게 하면 좀 복잡한데, 다행히 이 작업을 쉽게 만들어 둔 확장이 있습니다.

Unmanaged Exports 
; https://sites.google.com/site/robertgiesecke/Home/uploads/unmanagedexports

Unmanaged Exports (DllExport for .Net) 1.2.7 
; https://www.nuget.org/packages/UnmanagedExports

그래서 사용 방법이 매우 쉽습니다.

예를 하나 들어볼까요? ^^ 일단 다음과 같이 간단한 C# DLL 프로젝트를 만들고,

using System.Windows.Forms;

public class Class1
{
    public static void DllRegisterServer()
    {
        MessageBox.Show("Test is GOOD!");
    }
}

NuGet 콘솔(Tools / NuGet Package Manager / Package Manager Console)을 이용해 다음의 명령으로 Unmanaged Exports를 추가합니다.

Install-Package UnmanagedExports 

그럼 다음과 같은 노란색 경고 메시지가 눈에 띕니다.

The project 'ClassLibrary1' has no platform target. Only x86 or x64 assemblies can export functions.

export를 해야 하기 때문에 x86, x64에 대한 명시적인 구분을 해야 한다는 것입니다. 현재 DLL 프로젝트가 "AnyCPU"로 되어 있기 때문에 이를 "x86" (또는 x64)로 바꿔줍니다.

이렇게 Unmanaged Exports 도구가 설치되었으므로 이제 여러분이 원하는 C# static 메서드를 export 시킬 수 있습니다.

using System.Runtime.InteropServices;
using System.Windows.Forms;
using RGiesecke.DllExport;

public class Class1
{
    [DllExport("DllRegisterServer", CallingConvention = CallingConvention.StdCall)]
    public static void DllRegisterServer()
    {
        MessageBox.Show("Test is GOOD!");
    }
}

빌드하고, 해당 C# DLL 파일을 depends.exe 등의 도구로 확인을 해보겠습니다.

cs_export_func_1.png

오~~~! 훌륭합니다. ^^

예제에서 export시킨 DllRegisterServer는 regsvr32.exe가 호출해줄 수 있는 바로 그 함수의 signature입니다. 따라서, 명령행에서 다음과 같이 호출해 볼 수도 있습니다.

cs_export_func_2.png

(첨부한 파일은 이 글의 예제 코드를 포함합니다.)




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 9/27/2016]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2016-09-30 01시26분
[popo] 좋은 정보 공유 감사합니다! ^^
[guest]
2021-08-18 10시45분
[DevO] 혹시 C# dll export 시킨 후에 해당 function 을 C++ 에서 호출할수 있나요?
[guest]
2021-08-18 11시12분
@DevO 본문에서 regsvr32.exe로부터 실행하는 것을 보면 답이 나오지 않을까요? ^^
정성태
2022-03-03 04시39분
좋은 정보 감사합니다. 꼭 필요한 기능이 있을 때 마다 성태님 홈페이지에 해답이 있네요 ^^
이성열
2022-03-03 06시01분
음. 그런데 오래된 패키지라 그런지 VS 2019에서는 잘 안되네요 ㅠㅠ
DllExport라는 패키지로 다시 해보고 있습니다.
이성열
2022-03-03 07시18분
몇 시간 동안 구글링하면서 해보다가 안되서 다시 들어왔는데.. 연관 글에 보니까 관련 글이 몇개 더 있었군요. 미리 감사합니다^^
이성열
2022-03-03 07시25분
저도 최근에는 안 해봐서 또 어떻게 바뀌었는지 모르겠습니다. ^^; 해보시고 피드백 부탁드립니다. ^^

-------------------------------------------

C# - .NET 7부터 UnmanagedCallersOnly 함수 export 기능을 AOT 빌드에 통합
; https://www.sysnet.pe.kr/2/0/13464
정성태

... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12352정성태10/2/202010986개발 환경 구성: 517. Hyper-V Internal 네트워크에 NAT을 이용한 인터넷 연결 제공
12351정성태10/2/202010489오류 유형: 659. Nox 실행이 안 되는 경우 - Unable to bind to the underlying transport for ...
12350정성태9/25/202014022Windows: 175. 윈도우 환경에서 클라이언트 소켓의 최대 접속 수 [2]파일 다운로드1
12349정성태9/25/20208961Linux: 32. Ubuntu 20.04 - docker를 위한 tcp 바인딩 추가
12348정성태9/25/20209661오류 유형: 658. 리눅스 docker - Got permission denied while trying to connect to the Docker daemon socket at unix:///var/run/docker.sock
12347정성태9/25/202023803Windows: 174. WSL 2의 네트워크 통신 방법 [4]
12346정성태9/25/20208896오류 유형: 657. IIS - http://localhost 방문 시 Service Unavailable 503 오류 발생
12345정성태9/25/20208616오류 유형: 656. iisreset 실행 시 "Restart attempt failed." 오류가 발생하지만 웹 서비스는 정상적인 경우파일 다운로드1
12344정성태9/25/20209800Windows: 173. 서비스 관리자에 "IIS Admin Service"가 등록되어 있지 않다면?
12343정성태9/24/202019347.NET Framework: 945. C# - 닷넷 응용 프로그램에서 메모리 누수가 발생할 수 있는 패턴 [5]
12342정성태9/24/202010675디버깅 기술: 171. windbg - 인스턴스가 살아 있어 메모리 누수가 발생하고 있는지 확인하는 방법
12341정성태9/23/20209815.NET Framework: 944. C# - 인스턴스가 살아 있어 메모리 누수가 발생하고 있는지 확인하는 방법파일 다운로드1
12340정성태9/23/20209567.NET Framework: 943. WPF - WindowsFormsHost를 담은 윈도우 생성 시 메모리 누수
12339정성태9/21/20209525오류 유형: 655. 코어 모드의 윈도우는 GUI 모드의 윈도우로 교체가 안 됩니다.
12338정성태9/21/20209053오류 유형: 654. 우분투 설치 시 "CHS: Error 2001 reading sector ..." 오류 발생
12337정성태9/21/202010379오류 유형: 653. Windows - Time zone 설정을 바꿔도 반영이 안 되는 경우
12336정성태9/21/202012871.NET Framework: 942. C# - WOL(Wake On Lan) 구현
12335정성태9/21/202022278Linux: 31. 우분투 20.04 초기 설정 - 고정 IP 및 SSH 설치
12334정성태9/21/20207722오류 유형: 652. windbg - !py 확장 명령어 실행 시 "failed to find python interpreter"
12333정성태9/20/20208166.NET Framework: 941. C# - 전위/후위 증감 연산자에 대한 오버로딩 구현 (2)
12332정성태9/18/202010194.NET Framework: 940. C# - Windows Forms ListView와 DataGridView의 예제 코드파일 다운로드1
12331정성태9/18/20209362오류 유형: 651. repadmin /syncall - 0x80090322 The target principal name is incorrect.
12330정성태9/18/202010404.NET Framework: 939. C# - 전위/후위 증감 연산자에 대한 오버로딩 구현 [2]파일 다운로드1
12329정성태9/16/202012314오류 유형: 650. ASUS 메인보드 관련 소프트웨어 설치 후 ArmouryCrate.UserSessionHelper.exe 프로세스 무한 종료 현상
12328정성태9/16/202012496VS.NET IDE: 150. TFS의 이력에서 "Get This Version"과 같은 기능을 Git으로 처리한다면?
12327정성태9/12/202010106.NET Framework: 938. C# - ICS(Internet Connection Sharing) 제어파일 다운로드1
... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...