Microsoft MVP성태의 닷넷 이야기
C/C++: 165. CLion으로 만든 Rust Win32 DLL을 C#과 연동 [링크 복사], [링크+제목 복사],
조회: 11565
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

CLion으로 만든 Rust Win32 DLL을 C#과 연동

우선, CLion에서 rust를 사용하기 위해서는,

CLion - Rust
; https://www.jetbrains.com/help/clion/rust-support.html#install

Rustup
; https://rustup.rs/

rustup-init.exe를 다운로드해 실행부터 해줍니다. 그럼 %USERPROFILE%\.cargo\bin 디렉터리에 rust 환경이 구축되는데요, 따라서 편의상 환경변수 PATH 설정에 해당 경로를 추가하는 것을 권장합니다.

그다음 CLion의 Plugins를 통해 "Rust" 확장을 설치하면 준비는 끝입니다. 참고로, Visual Studio와 함께 C++ 구성요소를 미리 설치해 두는 것이 좋습니다. 왜냐하면 나중에 Rust 프로젝트를 처음 생성하는 시점에 시스템의 C/C++ 컴파일러를 설정하는 옵션이 뜰 텐데, 그때 Visual C++ 환경을 선택해 주면 됩니다.




문법 공부는 천천히 시간 나는 대로 아래의 사이트를 통해 하시고, ^^

The Rust Programming Language (번역본)
; https://doc.rust-kr.org/

곧바로 실습을 위해 CLion으로 다음과 같이 Rust Library 프로젝트를 생성하면,

rust_dll_prj_1.png

기본적으로 ./src/lib.rs 파일이 생성되고 아래의 내용을 갖습니다.

pub fn add(left: usize, right: usize) -> usize {
    left + right
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn it_works() {
        let result = add(2, 2);
        assert_eq!(result, 4);
    }
}

(CLion을 통해 생성한 프로젝트 구조는 명령행에서 "cargo new testapp2 --lib"를 실행한 것과 같습니다.)

CLion은 라이브러리 프로젝트에 대해 빌드 Configuration 설정을 "Test" 유형으로만 생성하기 때문에 이 상태에서는 "Build" 메뉴를 선택해도 rlib 파일을 생성하지 않습니다.

따라서 CLion의 "Terminal" 화면에서 "cargo build"를 입력해 빌드해야 하는데요,

PS F:\rust_prj\testapp2> cargo build
   Compiling testapp2 v0.1.0 (F:\rust_prj\testapp2)
    Finished dev [unoptimized + debuginfo] target(s) in 0.27s      

그럼, ./target/debug 디렉터리에 "libtestapp2.rlib" 파일이 생성됩니다.




일단, 위와 같이 생성한 rlib 파일은 러스트 전용 라이브러리로써 Win32 DLL과는 다릅니다. 따라서 (C# 등에서 이용하려면) 출력을 Win32 DLL로 바꿔야 하는데요, 이를 위해 프로젝트에 추가된 Cargo.toml 파일에 crate-type 값을 다음과 같이 설정해 줍니다.

[package]
name = "testapp2"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[lib]
crate-type = ["cdylib"]

[dependencies]

이후, CLion의 "Edit Configurations"를 이용해 DLL로 빌드하는 모드를 하나 더 추가한 다음,

rust_dll_prj_2.png

다시 빌드하면 이제 ./target/debug 디렉터리에 testapp2.dll 파일이 생성됩니다.




여기까지 하면, 비록 DLL은 생성되지만 add 함수가 외부로 노출된 상태는 아닙니다. Visual C++에서처럼 dllexport와 같은 처리를 해야 하는데요, 이를 위해 rust에서는 #[no_mangle]과 함께 extern 예약어를 적용해야 합니다.

#[no_mangle]
pub extern fn add(left: usize, right: usize) -> usize {
    left + right
}

이후 다시 빌드하고, dumpbin.exe를 이용해 DLL 정보를 확인하면,

F:\rust_prj\testapp2\target\debug> dumpbin /EXPORTS testapp2.dll
...[생략]...

    00000000 characteristics
    FFFFFFFF time date stamp
        0.00 version
           1 ordinal base
           1 number of functions
           1 number of names

    ordinal hint RVA      name

          1    0 00001000 add = add

...[생략]...


F:\rust_prj\testapp2\target\debug> dumpbin /HEADERS testapp2.dll
...[생략]...

FILE HEADER VALUES
            8664 machine (x64)
               5 number of sections
        660B6574 time date stamp Tue Apr  2 10:55:00 2024
               0 file pointer to symbol table
               0 number of symbols
              F0 size of optional header
            2022 characteristics
                   Executable
                   Application can handle large (>2GB) addresses
                   DLL

OPTIONAL HEADER VALUES
             20B magic # (PE32+)
...[생략]...

일반적인 64bit PE32+ 포맷으로 add 함수를 export 시킨 것을 확인할 수 있습니다.




Rust 컴파일러가 출력한 결과물이 64비트 DLL이니까, calling convention 고민은 하지 않아도 됩니다.

자, 그럼 C#에서 Rust DLL을 연결해 호출해 볼까요? ^^ 간단하게 Console 프로젝트를 하나 만들고, 이렇게 DllImport로 연결하면,

using System.Runtime.InteropServices;

namespace ConsoleApp1;

internal class Program
{
    [DllImport("testapp2.dll")]
    static extern uint add(uint left, uint right);

    static void Main(string[] args)
    {
        Console.WriteLine($"Hello, Rust!: {add(5, 6)}");
    }
}

빌드 후 정상적으로 add 함수가 불리는 것을 확인할 수 있습니다. Rust 문법만 공부하면 심심할 테니, 이렇게 C#과 연동해 보면 좀 더 재미가 있을 것입니다. ^^

참고로, 관련해서 아래의 문서도 좀 보시고. ^^

Creating A DLL With Rust
; https://samrambles.com/guides/window-hacking-with-rust/creating-a-dll-with-rust/index.html

rust-analyzer.vs
; https://marketplace.visualstudio.com/items?itemName=kitamstudios.RustAnalyzer




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







[최초 등록일: ]
[최종 수정일: 4/15/2024]

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

비밀번호

댓글 작성자
 



2024-05-04 05시37분
정성태

... 181  182  183  184  185  186  [187]  188  189  190  191  192  193  194  195  ...
NoWriterDateCnt.TitleFile(s)
286정성태6/23/200622407웹: 4. 웹 사이트 식별자(Identifier) 값 변경
285정성태6/20/200622653오류 유형: 9. [TFS] Report 관련 서비스를 조회할 때 rsErrorImpersonatingUser 오류 메시지 발생 [1]
284정성태6/19/200620420VS.NET IDE: 40. FxCop - IDE 에서 제공해 주는 SuppressMessage 코드
283정성태1/19/200721307Team Foundation Server: 8. 소스 세이프에서 TFS SourceControl 로 마이그레이션 [2]
279정성태12/27/200626697개발 환경 구성: 3. VS.NET 원격 디버깅 [1]
280정성태6/12/200626167    답변글 개발 환경 구성: 3.1. VS.NET 2003 원격 디버깅 설정
281정성태8/11/200627648    답변글 개발 환경 구성: 3.2. VS.NET 2005 원격 디버깅 설정
315정성태8/11/200628287        답변글 개발 환경 구성: 3.3. VS.NET 2005 원격 디버깅 설정 - ASP.NET F5 디버깅
278정성태6/11/200624853오류 유형: 8. [Outlook] 0x8004011D 에러 - "Exchange over the Internet" 환경
276정성태6/7/200618300Team Foundation Server: 7. 외부 빌드 머신 구성
287정성태6/24/200615946    답변글 Team Foundation Server: 7.1. 외부 빌드 머신 구성 - 다른 블로그 자료
275정성태6/7/200623842디버깅 기술: 4. VC++ 8.0 원격 디버깅 구성 - Side-by-Side DLL 문제.
269정성태6/6/200621030Team Foundation Server: 6. HTTPS를 통한 Team Server 접근 [1]
270정성태6/5/200617990    답변글 Team Foundation Server: 6.1. HTTPS를 통한 Team Server 접근 [1]
273정성태6/6/200620729    답변글 Team Foundation Server: 6.2. 두번째 방법 - HTTPS 를 통한 Team Server 접근 [1]
267정성태6/4/200620037Team Foundation Server: 5. 인터넷으로 Team Server 접근 [2]
266정성태6/8/200616635오류 유형: 7. [설치] mpoai9.dll 관련 오류
265정성태6/1/200624381디버깅 기술: 3. 원격 컴퓨터 디버깅 - VPC 설정
314정성태8/11/200621455    답변글 디버깅 기술: 3.1. Managed 원격 디버깅과 WinDBG 원격 디버깅
264정성태6/1/200630560오류 유형: 6. [VC++ 컴파일] already defined in ntdll.lib(ntdll.dll)
263정성태6/1/200631467디버깅 기술: 2. 커널 구조체 살펴보기 [5]
262정성태6/1/200623862오류 유형: 5. [설치] WinFX Beta2 - 설치시 문제점 해결
261정성태6/1/200620313웹: 3. IIS 6.0 - AppPool을 활용하여 실 서버(운영 서버)에서 디버깅
258정성태6/1/200628223디버깅 기술: 1. 디버깅 방법 - CLR 프로파일러 [1]파일 다운로드1
274정성태6/7/200621135    답변글 디버깅 기술: 1.1. 디버깅 방법 - CLR 프로파일러 ( on Vista )
254정성태6/1/200617592개발 환경 구성: 2. VPC에 Vista 설치하는 방법 [2]
... 181  182  183  184  185  186  [187]  188  189  190  191  192  193  194  195  ...