Microsoft MVP성태의 닷넷 이야기
LPBOOL Win32 마샬링 질문이 있습니다. [링크 복사], [링크+제목 복사],
조회: 7470
글쓴 사람
수 (keystroke915 at naver.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

안녕하세요,
C# - CopyFileEx API 사용 예제 코드
; https://www.sysnet.pe.kr/2/0/12447

이글을 보면서 예제를 실행해보면서 궁금증이 있어 질문드립니다.

CopyFileEx 함수가,
BOOL CopyFileEx(
LPCWSTR lpExistingFileName, // pointer to name of an existing file
LPCWSTR lpNewFileName, // pointer to filename to copy to
LPPROGRESS_ROUTINE lpProgressRoutine, // pointer to the callback function
LPVOID lpData, // to be passed to the callback function
LPBOOL pbCancel, // flag that can be used to cancel the operation
DWORD dwCopyFlags // flags that specify how the file is copied
);

이런 구조로 되어있는데요,
pbCancel 즉 LPBOOL형을 호출하실때 예제에서는 ref Int32로 작성하셨는데
제가 궁금증이 생겨서

bool bCancel = false;
unsafe
{
    CopyFileEx(srcFile, dstFile, &CopyProgressRoutineCallback, IntPtr.Zero, &bCancel, CopyFileFlags.COPY_FILE_RESTARTABLE);
}
//물론 위의 DllImport에서는 bool* 로 번경하였습니다.

bool bCancel = false;
unsafe
{
    CopyFileEx(srcFile, dstFile, &CopyProgressRoutineCallback, IntPtr.Zero, ref bCancel, CopyFileFlags.COPY_FILE_RESTARTABLE);
}
// 이번에는 DllImport에서 [MarshalAs(UnmanagedType.Bool)]ref bool pbCancel 로 인수 데이터형을 변경하였습니다.

이렇게 변경해보았는데도, 예제는 작동하는듯 했습니다.

제가 여기서 질문드리고 싶은 점은
1. 이렇게 세가지방법 모두 유효한 호출인가요?
2. 만약 잘못된점이 있다면 어디가 잘못되었는지, 또는 다른 대체 방법이 있는지
3. 혹시 이전에 설명 또는 언급한 자료가 있으실 경우 참조 자료

제가 이런 쪽 이슈를 어떤 키워드로 검색해야할지 감이 잡히지 않아서 부탁드리게 되었습니다.
중복된 이슈가 있으면 찾아볼수 있는 참조만 알려주셔도 감사합니다.
혹시나 이 게시판 이용할때 어긴점이 있으면 수정하도록 하겠습니다.


[연관 글]






[최초 등록일: ]
[최종 수정일: 12/11/2020]


비밀번호

댓글 작성자
 



2020-12-11 05시29분
현재 interop 관련한 공식 문서는 다음의 글을 참고하세요.

Native interoperability best practices
; https://docs.microsoft.com/en-us/dotnet/standard/native-interop/best-practices

"Boolean parameters and fields" 절에 설명이 나옵니다.

참고로, Win32 API의 경우 BOOL은 int의 typedef입니다. 그래서 사실 size로 본다면 Win32 API의 BOOL을 닷넷에서 int로 다루는 것이 맞습니다. 하지만, 닷넷의 bool은 1-byte지만 CLR의 마샬링 처리에서 4-byte의 BOOL로 처리하기 때문에 문제가 없습니다.

그런 것 외에, 1-byte와 4-byte의 크기 차이가 인자 전달의 바이트 정렬에 영향이 없다는 점도 있습니다. 왜냐하면, 메서드의 인자 전달이 IntPtr.Szie 단위로 전달이 되기 때문입니다. 즉, 1바이트를 전달해도 결국 레지스터나 스택의 전달 공간은 인자 1개 당 4(또는 8)바이트가 소비되므로 (매개변수 간에는) 별다른 간섭 없이 동작을 합니다.
정성태
2020-12-11 05시47분
[수] 앗... 설명 가만히 보니 결국 마샬링단계에서 동일해지겠군요.
답변 주셔서 감사합니다.
어떻게보면 좀 바보같은 질문이기도 했네요 ㅎㅎ
[guest]
2020-12-14 10시05분
[승준] 대문자 BOOL은 인트형인데.
약간의 문제가 발생하는 애들이 소문자 bool입니다.
이건 따로 마샬링 처리를 해줘야 되더군요.
[DllImport(@"test.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Unicode, SetLastError = true)]
[return: MarshalAs(UnmanagedType.I1)]
public static extern bool GetTest();
[DllImport(@"SJPlayerCore.dll", CallingConvention = CallingConvention.Cdecl, CharSet = CharSet.Unicode, SetLastError = true)]
public static extern void SetTest([MarshalAs(UnmanagedType.I1)] bool test);
요런식이요.
[guest]
2020-12-14 10시54분
@승준 (C++ 11 표준에서, C99 표준에서) 1-byte bool 타입이 C/C++에도 추가되면서 기존 typedef int BOOL로 임시처리한 것과 크기 면에서 다르긴 합니다.

C/C++ 컴파일러가 bool을 (예를 들어 리턴 값인 경우) ax 또는 rax 전체 바이트에 "mov(zx) ax, 1" 또는 "mov(zx) ax, 0"으로 true/false를 반환하는 경우에는 마샬링 처리가 필요 없을 것입니다. 반면, 1byte라는 점을 고려해 "mov al, 1" 또는 "mov al, 0" 식으로 처리해 준다면 상위 3바이트 영역이 쓰레기 값으로 채워져 있는 경우 닷넷의 bool 마샬링이 4바이트를 처리한다는 점에서 다른 값을 반환할 수 있습니다.

사실 위의 내용도 덧글에서 소개한 "Native interoperability best practices" 문서에서 다음과 같이 간략하게 언급은 하고 있습니다.

However, the _Bool, and bool types in C and C++ are a single byte. This can lead to hard to track down bugs as half the return value will be discarded, which will only potentially change the result.

좀 더 자세한 사항은 아래의 문서를 참고하라고 링크도 걸어주고. ^^

Customizing boolean field marshaling
; https://docs.microsoft.com/en-us/dotnet/standard/native-interop/customize-struct-marshaling#customizing-boolean-field-marshaling
정성태
2020-12-15 01시45분
다음의 글로 정리했으니 참고하세요.

C# - bool / BOOL / VARIANT_BOOL에 대한 Interop
; https://www.sysnet.pe.kr/2/0/12453
정성태
2020-12-15 08시58분
[수] 흥미로운 댓글과 새 글 감사합니다. 덕분에 많이 알아가네요
[guest]

... 16  17  18  19  20  21  22  23  24  25  26  [27]  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
5219티지레몬9/9/20199695c# PCB 자동화 프로그램(윈도우 폼 위주로 작업) 제작 준비 [3]
5218민성9/9/20197732안녕하세요 WPF에서 xaml 안에 다른 xaml을 넣고 싶습니다. [1]파일 다운로드1
5216WPF9/8/20198824WPF에서 XAML Islands를 사용하여 Win2D를 사용하니 그래픽 품질이 저하됩니다. [2]파일 다운로드1
5215허송세월9/5/20198630중복실행 방지 관련 문의 [2]파일 다운로드1
5214Jang...9/4/20198258[DB 테이블의 데이터 변경에 대한 알림 처리] SQL-Server말고 MySQL은 불가능하겠죠? [1]
5213진우8/31/20197507c# 람다 변수 캡쳐 문의 [2]
5212심성보8/29/20199177Clipboard내 여러개의 이미지를 PictureBox로 불러오는 문제 [2]
5211최휘철8/24/20198356CLR20r3 관련된 윈도우 오류입니다. ㅠㅠ 도와주세요. / 아래글 관련하여 관련 파일 올려 드려요^^ [1]파일 다운로드1
5210최휘철8/23/201912261CLR20r3 관련된 윈도우 오류입니다. ㅠㅠ 도와주세요. [5]
5209세퉁8/21/20197894폰트 파일 속성 값을 가져오는 방법 질문 드립니다. [2]파일 다운로드1
5208홍길동8/19/20198722DebugDiag에서 .Net의 Stack Trace를 Windbg에서는 어떻게 볼 수 있나요? [3]
5207민성8/16/20197074네 소스 전체를 올리도록 하겠습니다. [2]파일 다운로드1
5206민성8/14/20197207전 재현 가능하다고 봤는데 다시올리도록 하겠습니다. [1]
5205miny...8/14/20198042안녕하세요 .WPF ListBox시 체크박스가 있는데 체크박스에서 체크가 되었는지 알수 있는 방법이 있을까요? [1]
5204영민8/8/201910809안녕하세요 디버깅시 콘솔창을 띠어서 볼수가 없나요? [7]
5202민성8/6/20197546WPF에서 <Application.Resources에 xaml에 있는 icon 값을 저장하고 xaml에 불러다 사용하고 싶은데요 [1]
5201김대훈8/3/20197151상속시 생성자에 대해 질문드립니다 [3]
5200농상7/30/20199847foreach로 데이터 변경 [2]
5190오리다람7/20/20197255질문드립니다. [3]
5189진우7/19/20197823C# 스레드풀 코어별 실행 문의 [2]
5188황태관7/19/20197016비주얼베이직 2019 실행 할때 마다.. [3]
5187플하7/19/20199831UWP 관련 궁금한 사항에 대해서 [1]
5186김대훈7/14/20198446박싱과 언박싱에 대해 [2]
5185농상7/13/20197318Nullable에 대해서 [1]
5184김대훈7/4/20197171저자님의 책을 다 본후에는 [2]
51837/2/20197884.NET Compact Freamwork 컨트롤러 더블버퍼링 [1]
... 16  17  18  19  20  21  22  23  24  25  26  [27]  28  29  30  ...