Microsoft MVP성태의 닷넷 이야기
C#에서 C++ DLL호출 ('PInvokeStackImbalance') [링크 복사], [링크+제목 복사],
조회: 18588
글쓴 사람
궁금하당 (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]

... 61  62  63  64  65  66  67  68  69  70  [71]  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
787궁금이2/2/200915533TFS 관련하여 질문드리고자 합니다. [2]
786맨날맑음1/30/200916749WPF를 SmartClient로 배포할순 없을까요? [2]
785정성우12/16/200815719Vista 환경에서 VB6로 개발한 어플리케이션이 IE 통해서는 런칭이 안됩니다.. [4]
783서광원11/20/200816428.NET 어셈블리를 COM에서 사용하도록 변경 했을때 배포는 어떻게 해야 하나요? [3]
782김용환10/14/200816871웹서비스에서 캐시를 사용하고 싶습니다. [3]
781이수정9/5/200814967visual studio 2008 관련 질문입니다. [2]
776신현창8/5/200815207VS2005로 실행시 갑자기 COM+ 등록이 안됩니다. [1]
775땡초7/21/200815520조언을 구하고저 합니다. [2]
774남호진7/19/200816450Visual Studio 2005나 2008에서 서버탐색기에연결된 데이터에서 정보를 끌어... [2]
773지워니7/17/200818239웹서비스로 db2의 데이터가 오질 않아요. [1]
984이영구7/16/201118144    답변글 [답변]: 웹서비스로 db2의 데이터가 오질 않아요.
772다자래7/15/200815818스레드와 이벤트를 이용해서 mssql 서버의 저장프로시져의 레코드 진행률을 표현하는 프로그래스바를 구현하려고하는데....막막 [1]
771guest6/26/200815571[WCF] Self host 만드는 방법. [2]파일 다운로드1
770조조5/27/200815158스마트클라이언트에서 stand-alone 형태를 임베디드형태로 변환시 문제.. [1]
769제영한5/20/200815511배포프로젝트에 대한 조언을 구합니다. [2]
767김형중5/6/200816319RMclock 관련 문의 입니다. [1]
766정성태4/28/200817302[데브피아 Q&A 모음] 2008-04-21 ~ 2008-04-24
764정성태4/21/200818082[데브피아 Q&A 모음] 2008-04-14 ~ 2008-04-18 [2]
763단테4/17/200815524COM 객체 생성하는 방법
765정성태4/21/200816220    답변글 [답변]: CreateInstance를 부르시는 것이 추천됩니다.
762이민지4/16/200819276ClickOnce 실행 시 Internet Explorer 7.0에서 에러 메세지 출력 [1]
761이민지4/16/200815897ClickOnce로 배포를 하다가 생긴 문제에 대해 질문드립니다. [1]
760한귀순4/15/200816770DataSet 의 designer.cs [2]
759정성태4/14/200817186[데브피아 Q&A 모음] 2008-04-04 ~ 2008-04-12
7573/28/2008147422008 Server, Vista 에서 RDS 지원? [4]
756손승휘3/24/200818664Microsoft Web Browser OCX 부분이 안전하지 않는 ACTIVEX로 IE에서 인식되어는 점 [1]
... 61  62  63  64  65  66  67  68  69  70  [71]  72  73  74  75  ...