Microsoft MVP성태의 닷넷 이야기
C#에서 C++ DLL호출 ('PInvokeStackImbalance') [링크 복사], [링크+제목 복사],
조회: 11087
글쓴 사람
궁금하당 (abj7777 at naver.com)
홈페이지
첨부 파일

안녕하세요. 구글링해도 도저히 모르겠어서 올리게 됐습니다.

여기도 관련 글들이 많이 있더군요.
우선 현상은 다음과 같습니다.
호출함수중 유독 구조체 자료형을 사용한 한놈에서 문제가 생겼는데요.

uCAN_SendCANTxFrmae(tx);
라인에서 아래 에러가 뜹니다. (빌드는 성공)
관리 디버깅 도우미 'PInvokeStackImbalance' : 'PInvoke 함수 'WindowsFormsApp2!WindowsFormsApp2.Form1::uCAN_SendCANTxFrmae'에 대한 호출 결과 스택이 불안정하게 되었습니다. 관리되는 PInvoke 시그니처와 관리되지 않는 대상 시그니처가 일치하지 않기 때문인 것 같습니다. 호출 규칙 및 PInvoke 시그니처의 매개 변수와 관리되지 않는 대상 시그니처가 일치하는지 확인하십시오.'

찾아보니

1.
[DllImport("uCANDLL.dll")] 여기서
[DllImport("uCANDLL.dll", SetLastError = true, CharSet = CharSet.Ansi, CallingConvention = CallingConvention.StdCall)]
을 추가해봐라.
.StdCall을 Cdecl등으로 바꿔봐라.

-> 실패

2.
Intptr 및 Marshal.PtrToStructure를 사용해봐라

-> 실패

3. struct를 class로 바꿔봐라

-> 실패

---------------
방법을 알려주시면 정말 감사하겠습니다. 아래는 사용중인 Visual Studio 정보에요.

Microsoft Visual Studio Community 2019
버전 16.1.2
VisualStudio.16.Release/16.1.2+29001.49
Microsoft .NET Framework
버전 4.8.03752

설치된 버전: Community

Visual C++ 2019 00435-60000-00000-AA559
Microsoft Visual C++ 2019

Application Insights Tools for Visual Studio 패키지 9.1.00429.1
Application Insights Tools for Visual Studio

ASP.NET and Web Tools 2019 16.1.429.50124
ASP.NET and Web Tools 2019

Azure App Service 도구 v3.0.0 16.1.429.50124
Azure App Service 도구 v3.0.0

C# 도구 3.1.1-beta4-19281-06+58a4b1e79aea28115e66b06f850c83a3f1fcb6d3
IDE에서 사용되는 C# 구성 요소입니다. 프로젝트 형식 및 설정에 따라 다른 버전의 컴파일러를 사용할 수 있습니다.

F# 4.6용 Visual F# Tools 10.4 16.1.0-beta.19253.3+42526fe359672a05fd562dc16a91a43d0fe047a7
F# 4.6용 Microsoft Visual F# Tools 10.4

IntelliCode 확장 1.0
IntelliCode Visual Studio 확장 세부 정보

Microsoft JVM Debugger 1.0
Provides support for connecting the Visual Studio debugger to JDWP compatible Java Virtual Machines

Microsoft MI-Based Debugger 1.0
Provides support for connecting Visual Studio to MI compatible debuggers

Microsoft Visual C++ Wizards 1.0
Microsoft Visual C++ Wizards

Microsoft Visual Studio VC 패키지 1.0
Microsoft Visual Studio VC 패키지

NuGet 패키지 관리자 5.1.0
NuGet Package Manager in Visual Studio. For more information about NuGet, visit https://docs.nuget.org/

ProjectServicesPackage Extension 1.0
ProjectServicesPackage Visual Studio Extension Detailed Info

ResourcePackage 확장 1.0
ResourcePackage Visual Studio 확장 세부 정보

ResourcePackage 확장 1.0
ResourcePackage Visual Studio 확장 세부 정보

Test Adapter for Boost.Test 1.0
Enables Visual Studio's testing tools with unit tests written for Boost.Test. The use terms and Third Party Notices are available in the extension installation directory.

Test Adapter for Google Test 1.0
Google Test용으로 작성된 단위 테스트와 함께 Visual Studio의 테스트 도구를 사용합니다. 사용 약관 및 타사 고지 사항은 확장 설치 디렉터리에서 확인할 수 있습니다.

Visual Basic 도구 3.1.1-beta4-19281-06+58a4b1e79aea28115e66b06f850c83a3f1fcb6d3
IDE에서 사용되는 Visual Basic 구성 요소입니다. 프로젝트 형식 및 설정에 따라 다른 버전의 컴파일러를 사용할 수 있습니다.

Visual Studio Code 디버그 어댑터 호스트 패키지 1.0
Visual Studio에서 Visual Studio Code 디버그 어댑터를 호스트하기 위한 Interop 계층

Visual Studio Tools for CMake 1.0
Visual Studio Tools for CMake

Visual Studio Tools for CMake 1.0
Visual Studio Tools for CMake

일반적인 Azure Tools 1.10
Azure Mobile Services 및 Microsoft Azure 도구에서 사용할 일반적인 서비스를 제공합니다.











[최초 등록일: ]
[최종 수정일: 2/27/2020]


비밀번호

댓글 작성자
 



2020-02-28 09시00분
제가 사실 모든 질문에 대해 환경 구성까지 해가면서 답변할 정도로 그렇게 친절한 사람은 아닙니다. ^^ 일단, 최소 재현 프로젝트를 비주얼 스튜디오에서 F5 키로 테스트할 수 있는 유형이어야 합니다.

단지, 해당 프로젝트의 CAN_Frame 정의를 보면 byte [] 타입의 필드를 포함하고 있던데 배열은 참조형이기 때문에 당연히 C/C++의 구조체와는 다릅니다. 아마도 다음의 글들을 읽어 보시면 스스로 해결하실 수 있을 것입니다.

C# - blittable 타입이란?
; https://www.sysnet.pe.kr/2/0/11557

C# - string 배열을 담은 구조체를 직렬화하는 방법
; https://www.sysnet.pe.kr/2/0/11319

Win32 Interop - 크기가 정해지지 않은 배열을 C++에서 C#으로 전달하는 경우
; https://www.sysnet.pe.kr/2/0/737

C# 7.3 - 구조체의 고정 크기를 갖는 fixed 배열 필드에 대한 직접 접근 가능
; https://www.sysnet.pe.kr/2/0/11556
정성태
2020-02-28 09시10분
아... ^^ 구조체 정의를 넣어두셨군요. 아래와 같은 식이라면,

/*
 struct CANFrame{
    UINT8 Format; unsigned char
    UINT32 ID; unsigned int
    UINT8 DLC; unsigned char
    UINT8 Data[8]; unsigned char
    UINT64 TimeStamp; unsigned __int64
} typedef CAN_Frame;*/

다음과 같이 정의해 주면 될 것입니다.

public struct CAN_Frame
{
    public byte Format;
    public uint ID;
    public byte DLC;
    [MarshalAs(UnmanagedType.U1, SizeConst = 8)]
    public byte[] Data;
    public ulong TimeStamp;
}

구조체 정렬이 어떻게 설정되었는지는 모르겠으나, 경우에 따라 Pack으로 맞춰주어야 할 수도 있습니다.

[StructLayout(LayoutKind.Sequential, Pack = 1)]
public struct CAN_Frame
{
 ...
}
정성태
2020-02-28 11시26분
[궁금하당] 오 큰 도움이 되었고, 해결됐습니다. 친절하고 정확한 답변 정말 감사드립니다.

다만, 말씀해주신대로 했을 때,

System.TypeLoadException: ''CAN_Frame' 형식의 'Data' 필드를 마샬링할 수 없습니다. 관리되는/관리되지 않는 형식 조합이 잘못되었습니다. 배열 필드는 ByValArray 또는 SafeArray와 쌍이 되어야 합니다.'

라는 오류가 떠서 아래같이 바꿨더니 성공적으로 됐습니다~!

[StructLayout(LayoutKind.Sequential, Pack = 1)] --> 변경
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]

[MarshalAs(UnmanagedType.U1, SizeConst = 8)] --> 변경
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]

환경 정보는.. 구성을 부탁드리려 했다기 보단.. 혹시 디버깅에 참고가 될까해서 넣은 정보입니다..
[guest]
2020-02-28 11시29분
[궁금하당] 최종 오류안난 코드 -->

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct CAN_Frame
{
    public byte Format;
    public uint ID;
    public byte DLC;
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
    public byte[] Data;
    public ulong TimeStamp;
}
[guest]

... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
1556김응규4/22/201512465    답변글 [답변]: WCF net.tcp 관련해서 질문 드립니다. [1]파일 다운로드1
1554(삭제)4/19/201513218안드로이드 에뮬레이터에서 특정 앱 구동불가할때! [2]
1552spow...4/18/201519102마이크로세컨드 단위의 Sleep이 필요한데 찾질 못하겠습니다. [5]
1553spow...4/18/201512431    답변글 [답변]: 마이크로세컨드 단위의 Sleep이 필요한데 찾질 못하겠습니다. [1]
1551인디언기...4/16/201511788iisnode에 대해서 글 올리셨던데 질문이 있습니다 [1]
1549김선아4/15/201512230.net framework 4.5와 c++ dll에 관한 질문입니다 [1]
1550김선아4/15/201510619    답변글 [답변]: .net framework 4.5와 c++ dll에 관한 질문입니다 [2]
1547문명식4/3/201512634IWebBrowser Key Event 관련 [2]
1545나종식3/25/201511612internet explorer dns cache flush [2]
1543이용원3/23/201511588hyper-v 질문드려요! [1]
1544이용원3/24/201510738    답변글 한가지만 더 여쭤볼게요! [1]
1541임기성3/20/201517958Microsoft.Office.Core dll관련 문제 [1]
1539펜플3/17/201511283asp.net 질문입니다. [1]
1540113/17/201511281    답변글 [답변]: asp.net 질문입니다. [1]
1538열공합시닷3/13/201512069중계서버 질문드립니다 [1]
1537김재영3/11/201513155asp.net의 빌드 유휴시간 이후의 리빌드 작업을 컨트롤 할 수 있습니까? [2]
1536솔솔3/9/201511580mac address얻으려고 하는데요.. [1]
1535Stud...3/4/201512136C#엔 MFC의 theApp 과 같은 기능을 하는것이 없나요? [3]
1534김동진3/4/201511584C#에서 BHO로 Windows Explorer에서의 동작이 가능할지요 [2]
1533김동현3/3/201511875컴파일 된 프로그램에 코드 삽입(?) 관련입니다. [1]
1532반가워요3/3/201514510C# 소켓연결에 packet송수신... [8]
1531강현수3/2/201513147C# com 등록 시 GUID 변경 관련 문의 [1]
1529김민규2/15/201513028C++ 64bit/32bit [3]파일 다운로드1
1528김개똥2/14/201510516좀 관련없는 질문이지만, 정말 답답하고 찾는데 명확한 해답 찾지못해 질문 올립니다. <Windows 사용자 인증> [1]
1527방문자1/27/201511552System Timer의 Tick에서 Backgroundworker를 호출하면 스레드 문제가 있을까요? [1]
1526황상대1/26/201513398C# 마샬링 관련 질문이 있습니다. [1]
... 46  47  48  49  50  [51]  52  53  54  55  56  57  58  59  60  ...