Microsoft MVP성태의 닷넷 이야기
C/C++: 165. CLion으로 만든 Rust Win32 DLL을 C#과 연동 [링크 복사], [링크+제목 복사],
조회: 11599
글쓴 사람
정성태 (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분
정성태

... 166  167  168  169  170  171  172  173  174  175  176  177  [178]  179  180  ...
NoWriterDateCnt.TitleFile(s)
536정성태9/12/200732369.NET Framework: 97. WCF : netTcpBinding에서의 각종 Timeout 값 설명 [11]
535정성태9/11/200729862.NET Framework: 96. WCF - PerSession에서의 클라이언트 연결 관리 [5]
534정성태9/3/200725329개발 환경 구성: 29. VHD 파일 크기 줄이기
533정성태9/2/200728038개발 환경 구성: 28. CA 서비스 - 사용자 정의 템플릿 유형 추가
532정성태9/2/200730567개발 환경 구성: 27. AD CA에서 Code Signing 인증서 유형 추가 방법
531정성태9/2/200726330.NET Framework: 95. WCF에서의 DataTable 사용
530정성태9/1/200722893.NET Framework: 94. WCF 예외에 대한 시행착오
529정성태8/31/200725747.NET Framework: 93. WCF - DataContract와 KnownType 특성 [1]
528정성태8/30/200720391오류 유형: 47. VPC - 네트워크 어댑터 MAC 주소 중복 오류
527정성태8/30/200730456Team Foundation Server: 20. 잠긴 파일을 강제로 해제 [2]
526정성태8/29/200720361오류 유형: 46. VS.NET 2008 - ASP.NET 디버깅 : Strong name validation failed.
525정성태8/27/200722601VS.NET IDE: 54. VS.NET 2008 - 새롭게 도입되는 XSD Schema Designer
524정성태8/23/200740098오류 유형: 45. 요청한 작업은, 사용자가 매핑한 구역이 열려 있는...
523정성태8/16/200722812VS.NET IDE: 53. VS.NET 2008 - 서비스 참조 시 기존 데이터 컨테이너 DLL 사용
522정성태8/13/200726406VS.NET IDE: 52. VS.NET 2008 - WCF를 위한 디버깅 환경 개선
521정성태8/8/200726390.NET Framework: 92. XmlSerializer 생성자의 실행 속도를 올리는 방법 - 두 번째 이야기 [3]
520정성태8/7/200721599VS.NET IDE: 51. Visual Studio 2008 베타 2 설치
519정성태7/27/200727983오류 유형: 44. System.BadImageFormatException [2]
518정성태7/26/200728999오류 유형: 43. System.ComponentModel.LicenseException [1]
517정성태7/19/200717351개발 환경 구성: 26. VPC - 일반 사용자 계정으로 구동
516정성태7/19/200720519오류 유형: 42. TFS - Error loading menu: Index was outside the bounds of the array [2]
515정성태7/18/200728171오류 유형: 41. SSL 서버 자격 증명을 만드는 동안 심각한 오류가 발생했습니다.
514정성태7/14/200720868Team Foundation Server: 19. Orcas에서 개선되는 TFS 기능들
513정성태7/4/200731840.NET Framework: 91. Foreground Thread / Background Thread [1]
512정성태6/27/200721739오류 유형: 40. error PRJ0050: Failed to register output.
511정성태6/25/200729822.NET Framework: 90. XmlSerializer 생성자의 실행 속도를 올리는 방법 [2]
... 166  167  168  169  170  171  172  173  174  175  176  177  [178]  179  180  ...