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


COM 개체로 인해 IE 7 비스타 버전이 종료될 때 오류 화면이 뜬다면?

My Toolbar or BHO is Causing IE7 on Vista to Crash on Close. Help!
; https://docs.microsoft.com/en-us/archive/blogs/tonyschr/my-toolbar-or-bho-is-causing-ie7-on-vista-to-crash-on-close-help

코딩에서도, "형식 안정성"을 중요하게 여기는 것처럼, 점차로 마이크로소프트의 전반적인 제품군에서 "오류임에도 불구하고 덮어주던 방식"을 없애는 방향으로 가고 있는 것 같습니다. 위의 기사에서 언급하는 문제도 그와 같은 식이라고 볼 수 있겠지요.

CoInitialize() / CoUninitialize()의 쌍이 맞지 않는 경우에 IE 6까지는 그러한 실수를 감안하여 문제가 없도록 되어 있었는데, IE 7부터는 문제가 발생하게 되었다고 합니다. 만약, 여러분들의 프로그램에서 - 사실 별도의 스레드를 만들지 않는 한 CoInitialize() / CoUninitialize()를 호출할 일은 거의 없지만, 만약 사용하고 있다면 반드시 그 부분에 대한 호출 쌍이 맞는지 확인하셔야 할 것입니다.

그러면서, "IInitializeSpy" 인터페이스를 소개해 주고 있습니다. 어허... 제가 그동안 많이 무심했습니다. 이런 인터페이스가 있었는 줄 처음 알았으니까요. (Windows XP SP1부터 지원되었다고 합니다.) 이 인터페이스는 CoInitialize / CoUninitialize 메서드들이 호출될 때마다 Pre/Post 콜백 함수를 불려지는 것을 목적으로 정의된 것입니다.

테스트 삼아서 간단한 예제를 한번 작성해 보았습니다. (참고로, 위의 기사를 쓴 사람은 절대로 ActiveX 개발자들이 자신들의 CoInitialize() / CoUninitialize() 호출 쌍을 보정하기 위해 이 인터페이스를 구현하지 말라고 당부하고 있습니다.)

참고로, 아래의 예제에 대해 컴파일 가능한 VC++ 8 프로젝트파일은 첨부파일에 넣어두었습니다.

#include "stdafx.h"
#include <windows.h>
#include <objbase.h>
#include <objidl.h>

#pragma comment(lib, "ole32.lib")

class CInitializeSpy : public IInitializeSpy
{
public:
	CInitializeSpy()
	{
		dwCount = 0;
	}

	STDMETHOD(PreInitialize)(DWORD dwCoInit,DWORD dwCurThreadAptRefs)
	{
		::OutputDebugStr(L"PreInitialize\r\n");
		return S_OK;
	}

	STDMETHOD(PostInitialize)(HRESULT hrCoInit,DWORD dwCoInit,DWORD dwNewThreadAptRefs)
	{
		::OutputDebugStr(L"PostInitialize\r\n");
		return S_OK;
	}

	STDMETHOD(PreUninitialize)(DWORD dwCurThreadAptRefs)
	{
		::OutputDebugStr(L"PreUninitialize\r\n");
		return S_OK;
	}

	STDMETHOD(PostUninitialize)(DWORD dwNewThreadAptRefs)
	{
		::OutputDebugStr(L"PostUninitialize\r\n");
		return S_OK;
	}

	STDMETHOD(QueryInterface)(REFIID riid, void **ppvObject)
	{
		bool qied = false;

		if ( riid == IID_IUnknown)
		{
			*ppvObject = this;
			qied = true;
		}

		if ( riid == IID_IInitializeSpy)
		{
			*ppvObject = this;
			qied = true;
		}

		if ( qied == true )
		{
			dwCount ++;
			return S_OK;
		}

		return E_NOINTERFACE;
	}

	virtual ULONG STDMETHODCALLTYPE AddRef( void)
	{
		dwCount ++;
		return dwCount;
	}

	virtual ULONG STDMETHODCALLTYPE Release( void)
	{
		dwCount --;

		if ( dwCount == 0 )
		{
			delete this;
		}

		return dwCount;
	}

	DWORD dwCount;
};

int _tmain(int argc, _TCHAR* argv[])
{
	CInitializeSpy *pSpy = new CInitializeSpy();
	ULARGE_INTEGER ulCookie;

	CoRegisterInitializeSpy(pSpy, &ulCookie);

	CoInitialize(NULL);
	CoUninitialize();

	CoRevokeInitializeSpy(ulCookie);
	return 0;
}




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







[최초 등록일: ]
[최종 수정일: 7/9/2021]

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

비밀번호

댓글 작성자
 




... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
13178정성태12/1/202216512Windows: 215. Win32 API 금지된 함수 - IsBadXxxPtr 유의 함수들이 안전하지 않은 이유파일 다운로드1
13177정성태11/30/202217201오류 유형: 829. uwsgi 설치 시 fatal error: Python.h: No such file or directory
13176정성태11/29/202213023오류 유형: 828. gunicorn - ModuleNotFoundError: No module named 'flask'
13175정성태11/29/202218063오류 유형: 827. Python - ImportError: cannot import name 'html5lib' from 'pip._vendor'
13174정성태11/28/202214404.NET Framework: 2073. C# - VMMap처럼 스택 메모리의 reserve/guard/commit 상태 출력파일 다운로드1
13173정성태11/27/202215420.NET Framework: 2072. 닷넷 응용 프로그램의 스레드 스택 크기 변경
13172정성태11/25/202215475.NET Framework: 2071. 닷넷에서 ESP/RSP 레지스터 값을 구하는 방법파일 다운로드1
13171정성태11/25/202214355Windows: 214. 윈도우 - 스레드 스택의 "red zone"
13170정성태11/24/202216846Windows: 213. 윈도우 - 싱글 스레드는 컨텍스트 스위칭이 없을까요?
13169정성태11/23/202218319Windows: 212. 윈도우의 Protected Process (Light) 보안 [1]파일 다운로드2
13168정성태11/22/202214239제니퍼 .NET: 31. 제니퍼 닷넷 적용 사례 (9) - DB 서비스에 부하가 걸렸다?!
13167정성태11/21/202215466.NET Framework: 2070. .NET 7 - Console.ReadKey와 리눅스의 터미널 타입
13166정성태11/20/202216251개발 환경 구성: 651. Windows 사용자 경험으로 WSL 환경에 dotnet 런타임/SDK 설치 방법
13165정성태11/18/202213851개발 환경 구성: 650. Azure - "scm" 프로세스와 엮인 서비스 모음
13164정성태11/18/202216952개발 환경 구성: 649. Azure - 비주얼 스튜디오를 이용한 AppService 원격 디버그 방법
13163정성태11/17/202216977개발 환경 구성: 648. 비주얼 스튜디오에서 안드로이드 기기 인식하는 방법
13162정성태11/15/202218228.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일 [1]
13161정성태11/14/202216340.NET Framework: 2068. C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성) [4]
13160정성태11/11/202217458.NET Framework: 2067. C# - PublishSingleFile 적용 시 native/managed 모듈 통합 옵션
13159정성태11/10/202219723.NET Framework: 2066. C# - PublishSingleFile과 관련된 옵션 [3]
13158정성태11/9/202215358오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/202216517.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/202219630.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/202216806디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/202218007.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/202218705.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
... 31  32  [33]  34  35  36  37  38  39  40  41  42  43  44  45  ...