Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

서로 다른 프로세스에서 WM_DROPFILES 메시지를 전송하는 방법

아래의 답변을 하면서,

drag&drop 관련해서 문의 드립니다.
; https://www.sysnet.pe.kr/3/0/4756

테스트하다가 알게 된 사실을 하나 공유합니다. ^^

OLE Drag&Drop이 아닌, 예전의 WM_DROPFILES Win32 윈도우 메시지를 이용해 (예를 들어 탐색기 같은) 외부 응용 프로그램에서 끌어다 놓기 했을 때 그 정보를 다음과 같이 받아올 수 있습니다.

BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
   // ...[생략]...

   DragAcceptFiles(hWnd, TRUE);

   // ...[생략]...

   return TRUE;
}

LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
    // ...[생략]...
	
	case WM_DROPFILES:
	{
		int fileCount = DragQueryFile((HDROP)wParam, -1, nullptr, 0);
		DWORD dwError = ::GetLastError();
		DragFinish((HDROP)wParam); 
		char buf[1024] = { 0 };
		if (fileCount != 0)
		{
			sprintf(buf, "FileCount: %d", fileCount);
		}
		else
		{
			sprintf(buf, "Error: %d", dwError);
		}

		MessageBoxA(nullptr, buf, nullptr, MB_OK);
	}
	break;
	
	// ...[생략]...
    }
    return 0;
}

실제로 위와 같이 Win32 프로그램을 만든 후 윈도우 탐색기에서 파일을 끌어다 놓으면 DragQueryFile의 호출 결과로 1이라는 결과가 나옵니다.




WM_DROPFILES가 Win32 Message이다 보니, 이를 SendMessage로 보내는 것도 가능합니다. 이에 대해서는 웹상의 많은 자료들이 다음과 같은 코드로 알려주고 있습니다.

char filename[] = "c:\\temp\\test.wav";

POINT point;
point.x = 0;
point.y = 0;

int cbBuffer = sizeof(DROPFILES) + strlen(filename) + 2;
			
if (HGLOBAL hGlobal = GlobalAlloc(GHND | GMEM_SHARE | GMEM_ZEROINIT, cbBuffer))
{
	DROPFILES * df = (DROPFILES *)GlobalLock(hGlobal);
	df->pFiles = sizeof(DROPFILES);
	df->pt = point;
	df->fNC = FALSE;
	df->fWide = FALSE;

	strcpy(reinterpret_cast<char *>(df + 1), filename);

	GlobalUnlock(hGlobal);

	SendMessage(hWnd, WM_DROPFILES, (WPARAM)hGlobal, 0);

	GlobalFree(hGlobal);
}

실제로 위와 같은 코드는 같은 EXE 프로세스 내에서 SendMessage를 한 경우에는 잘 동작합니다. 하지만, 다른 EXE 프로세스로 보낼 때는 상대 측의 WM_DROPFILES에 있는 DragQueryFile 함수가 0을 반환하고 오류 코드는 6 (The handle is invalid.)으로 나옵니다.

왜냐하면, 윈도우 시스템은 WM_DROPFILES 메시지의 경우 자동으로 마샬링을 하지는 않기 때문입니다. 설령, 마샬링을 한다 해도 현재의 WM_DROPFILES 처리 방식에서 보자면 넘어온 HGLOBAL 주소 값이 프로세스 간에 마샬링 된 주소와 동일할 것이라는 보장이 없으므로 역시 오류가 발생하는 것은 마찬가지입니다.

그럼, 도대체 어떤 수수께끼가 있는 것일까요?

이에 대해서는 WM_DROPFILES를 허용하기 위해 UAC의 UIPI 보안을 설정하는 코드에서 힌트를 얻을 수 있습니다.

[Tip] UAC 때문에 WM_DROPFILES 메시지를 받을수 없는 경우에 대한 처리 
; http://www.tipssoft.com/bulletin/board.php?bo_table=FAQ&wr_id=1571

위의 글에 보면, WM_DROPFILES뿐만 아니라 WM_COPYGLOBALDATA도 Message Filter에 추가해야 하는 것을 알 수 있습니다.

ChangeWindowMessageFilter(WM_COPYGLOBALDATA, MSGFLT_ADD);
ChangeWindowMessageFilter(WM_DROPFILES, MSGFLT_ADD);

유추가 되지요? ^^

즉, 탐색기에서 파일을 끌어다 놓으면 우선 HGLOBAL에 대한 데이터를 대상 프로세스에 WM_COPYGLOBALDATA를 이용해 복사한 후, 그 프로세스에서 새롭게 할당된 HGLOBAL 주소를 WM_DROPFILES에 넘기는 것으로 짐작할 수 있습니다.

문제는, WM_COPYGLOBALDATA 메시지의 사용법에 대한 공식 문서가 없다는 것입니다.

다행이라면, 검색해 보면 ^^ 사용 코드가 나옵니다.

[ros-diffs] [jimtabor] 50047: [Win32k|User32] - Finish 50030 (work by Giannis), Now PostMessage passes all the correct data based on Get/PeekMessage. Example: Post A, Get/Peek A, Translate A, Dispatch A, shoul... 
; https://www.mail-archive.com/ros-diffs@reactos.org/msg09395.html

+  Ret = SendMessageA( hWnd,
+                      WM_COPYGLOBALDATA,
+                      (WPARAM)GlobalSize((HGLOBAL)wParam), // Zero if not a handle.
+                      (LPARAM)wParam);                     // Send wParam as lParam.
+
+  if ( Ret ) return NtUserPostMessage(hWnd, Msg, (WPARAM)Ret, lParam);

사용법이 정말 간단합니다. WM_COPYGLOBALDATA는 GlobalAlloc으로 전달받은 주소의 데이터를 대상 윈도우의 프로세스로 마샬링해주고, 그에 대한 새로운 EXE 프로세스 공간의 메모리 주소 값을 반환해줍니다.

이를 이용해 최종적으로 다음과 같이 EXE 간의 WM_DROPFILES 메시지 전송을 할 수 있습니다.

char filename[] = "c:\\temp\\test.wav";

POINT point;
point.x = 0;
point.y = 0;

HWND targetWindow = FindWindow(nullptr, L"...[yours]...");
	
int cbBuffer = sizeof(DROPFILES) + strlen(filename) + 2;
			
if (HGLOBAL hGlobal = GlobalAlloc(GHND | GMEM_SHARE | GMEM_ZEROINIT, cbBuffer))
{
	DROPFILES * df = (DROPFILES *)GlobalLock(hGlobal);
	df->pFiles = sizeof(DROPFILES);
	df->pt = point;
	df->fNC = FALSE;
	df->fWide = FALSE;

	strcpy(reinterpret_cast<char *>(df + 1), filename);

	GlobalUnlock(hGlobal);

	// WM_COPYGLOBALDATA == 0x49
	int result = SendMessage(targetWindow, 0x49, (WPARAM)cbBuffer, (LPARAM)hGlobal);
	SendMessage(targetWindow, WM_DROPFILES, (WPARAM)result, 0);

	GlobalFree(hGlobal);
}

첨부한 파일은 다음과 같은 구성의 Win32Project1과 ConsoleApplication1 프로젝트를 포함합니다.

  • Win32Project1: WM_DROPFILES 메시지를 받아 처리하는 프로그램
  • ConsoleApplication1: SendMessage로 WM_DROPFILES 메시지를 Win32Project1로 전송하는 프로그램 (Help / About 메뉴로 SendMessage 코드 수행)




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







[최초 등록일: ]
[최종 수정일: 3/19/2023]

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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...
NoWriterDateCnt.TitleFile(s)
12132정성태1/28/202012239.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기)파일 다운로드1
12131정성태1/27/202012295개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/20209768VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202015326.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202010113오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
12127정성태1/25/20209990.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)파일 다운로드1
12126정성태1/25/202010816.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/20208696VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202010525VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202011990웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/20208986.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/20209567VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202010987.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202011008디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202011664개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202010638디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202011110디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202010888디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202012750디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202013374오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/20209976오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202013212디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
12110정성태1/11/202011809디버깅 기술: 154. Patch Guard로 인해 블루 스크린(BSOD)가 발생하는 사례 [5]파일 다운로드1
12109정성태1/10/20209734오류 유형: 588. Driver 프로젝트 빌드 오류 - Inf2Cat error -2: "Inf2Cat, signability test failed."
12108정성태1/10/20209741오류 유형: 587. Kernel Driver 시작 시 127(The specified procedure could not be found.) 오류 메시지 발생
12107정성태1/10/202010693.NET Framework: 877. C# - 프로세스의 모든 핸들을 열람 - 두 번째 이야기
... 46  47  48  49  50  51  52  53  54  55  56  57  58  59  [60]  ...