Microsoft MVP성태의 닷넷 이야기
VC++: 155. CComPtr/CComQIPtr과 Conformance mode 옵션의 충돌 [링크 복사], [링크+제목 복사],
조회: 15106
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 4개 있습니다.)

CComPtr/CComQIPtr과 Conformance mode 옵션의 충돌

별다른 생성자를 추가하지 않은 경우, 클래스의 형변환은 상위 클래스로는 가능해도 하위 클래스로는 불가능합니다. 간단하게 테스트를 해볼까요?

class MyType
{
public:
    int f = 100;

    MyType() { }
};

class MyDervied : public MyType
{
public:
    bool result = true;

    MyDervied() { f = 200; }
};

위와 같은 클래스 구조에서, 다음과 같이 코드를 사용하면,

MyDervied test;
MyType t = test;

printf("test = %d, t = %d\n", t.f, test.f); // t = 200, test = 200

정상적으로 컴파일 및 빌드가 됩니다. 하지만, 반대로 부모 인스턴스를 자식 인스턴스에 대입하려는 시도는,

MyType t;
MyDervied test = t; // Error C2440 'initializing': cannot convert from 'MyType' to 'MyDervied'

보는 바와 같이 오류가 발생합니다.




위의 코드를 개선해 볼까요? 우선 자식을 부모 클래스에 대입하는 경우는 비록 컴파일은 가능하지만 Visual Studio의 코드 편집기 상에서 "t = test" 코드에 밑줄이 표시되면서 "C26437: Do not slice (es.63)" 경고를 알려줍니다. 이 경고를 없애려면, 명시적으로 부모 타입에 생성자를 하나 추가해 주면 됩니다.

class MyType
{
public:
    int f = 100;

    MyType() { }

    MyType(MyType& inst)
    {
        this->f = inst.f;
    }
};

그리고, 부모 인스턴스를 자식 인스턴스에 대입하는 것도 역시 자식 인스턴스에서 부모 인스턴스를 받아들이는 생성자를 정의해 주면 됩니다.

class MyDervied : public MyType
{
public:
    bool result = true;

    MyDervied() { this->f = 200; }

    MyDervied(MyType& inst)
    {
        this->f = inst.f;
    }
};

MyType t;
MyDervied test = t; // 정상적으로 컴파일

일단, 여기까지는 다들 아시는 그대로입니다.




이 상황에서 template을 사용하면 어떻게 될까요?

template <class T>
class MyType
{
public:
    int f = 100;

    MyType() { }

    MyType(MyType& inst)
    {
       this->f = inst.f;
    }
};

template <class T>
class MyDervied : public MyType<T>
{
public:
    bool result = true;

    MyDervied() { this->f = 200; }

    MyDervied(MyType<T>& inst)
    {
        this->f = inst.f;
    }
};

일단, 위와 같은 경우 다음과 같은 식으로는 잘 컴파일이 됩니다.

{
    MyDervied<int> test;
    MyType<int> t = test;

    printf("t = %d, test = %d\n", t.f, test.f);
}

{
    MyType<int> t;
    MyDervied<int> test = t;

    printf("t = %d, test = %d\n", t.f, test.f);
}

하지만, 만약 템플릿 타입 인자가 다르다면 어떻게 될까요?

{
    MyDervied<int> test;
    MyType<long> t = test; // Error C2440 'initializing': cannot convert from 'MyDervied' to 'MyType'
}

이런 경우, 컴파일러는 타입이 다르기 때문에 당연히 오류가 발생합니다.




그런데, 저 템플릿 타입이 달라도 컴파일이 되고 싶은 경우가 있습니다. 다소 특수한 사례를 들어서, 템플릿 인자 간의 상속 관계가 있는 경우 형변환을 허용하고 싶다면 다음과 같은 식으로 코딩을 할 수 있습니다.

class Root { };

class Node : public Root { };

class Terminal : public Root { };

template <class T>
class MyType
{
public:
    MyType() { }

    operator T& () { return p; }

    T p;
};

template <class T>
class MyDerived : public MyType<T>
{
public:
    MyDerived() { }

    MyDerived(Root& inst)
    {
    }
};

위의 코드는 이렇게 사용하는 것이 가능합니다. (자식 타입으로 형변환하고 있습니다.)

{
    MyType<Terminal> t;
    MyDerived<Node> test = t;
}

잘 보시면, t 인스턴스는 MyType의 T&() 연산자로 인해 Terminal 타입에 해당하는 인스턴스를 반환합니다. 그리고 MyDerived의 생성자는 Root& 타입을 인자로 정의했으므로 Terminal 타입의 인스턴스를 받을 수 있습니다.

여기서 재미있는 것은, 저 코드가 Visual C++의 "Conformance mode"를 적용하는 경우에는 컴파일 오류가 발생한다는 점입니다. 실제로 프로젝트 속성 창에서 "Configuration Properties" / "C/C++" / "Language" 범주로 들어가 "Conformance mode" 옵션을 "Yes (/permissive-)"로 켜면,

atl_cpp_conformance_mode_0.png

이후 "test = t;" 코드에서 다음과 같은 컴파일 오류가 발생합니다.

Error C2440 'initializing': cannot convert from 'MyType<Terminal>' to 'MyDerived<Node>'

즉, 무조건 템플릿 타입 인자가 같아야 한다는 것을 강제하는 것입니다. 하지만 템플릿 타입을 같게 만들어도 여전히 컴파일 오류가 발생합니다.

{
    MyType<Terminal> t;
    MyDerived<Terminal> test = t; // Error C2440 'initializing': cannot convert from 'MyType<T>' to 'MyDerived<Terminal>'
}

왜냐하면, 부모 클래스의 인스턴스를 자식 클래스의 인스턴스로 대입하려 했기 때문입니다. 따라서 다시 원론적으로 돌아와서, 상속받은 타입에서 부모 클래스의 인스턴스로 형변환하는 것만을 허용하게 된 것입니다.

{
    MyDerived<Terminal> test;
    MyType<Terminal> t = test;
}

즉, 이것은 템플릿을 사용하지 않던 원래의 클래스 타입과 동일한 규칙이 적용되는 것이어서 이제는 별다른 생성자도 필요하지 않습니다. 따라서, 위와 같은 코드는 그냥 다음과 같이 기본 클래스를 정의해도 가능한 수준입니다.

template <class T>
class MyType
{
public:
    MyType() { }
};

template <class T>
class MyDerived : public MyType<T>
{
public:
    MyDerived() { }
};




아니... 그나저나 도대체 저런 이상한 형태의 클래스 상속을 누가 사용한단 말입니까? ^^; 자, 서론이 길었군요. 바로 저런 식으로 사용하는 사례가 CComPtr과 CComQIPtr입니다.

이 타입들을 축약하면 다음과 같이 정의할 수 있습니다.

#include <d3d11.h>
#include <Unknwnbase.h>

template <class T>
class CComPtr
{
public:
    CComPtr() { p = nullptr; }

    CComPtr(T* lp) { p = lp; }

    operator T* () { return p; }

    T* p;
};

template <class T>
class CComQIPtr : public CComPtr<T>
{
public:
    CComQIPtr() { }

    CComQIPtr(IUnknown* lp) { }
};

보시면, CComQIPtr 생성자에서 IUnknown* 인자를 받아들이고, CComPtr에서는 T*() 연산자 오버로드로 포인터를 반환하고 있습니다. 즉, IUnknown을 상속한 타입을 템플릿 인자로 지정한 CComPtr 인스턴스는 CComQIPtr에 이렇게 대입하는 것이 가능합니다.

// IDXGIResource와 ID3D11Texture2D는 서로 다른 타입이지만, IUnknown을 상속
{
    CComPtr<IDXGIResource> pUnk;
    CComQIPtr<ID3D11Texture2D> pDisp = pUnk;
}

어차피 내부에서 IUnknown의 QueryInterface 함수를 호출하기 때문에 저렇게 사용하는 것은 어떠한 버그도 발생시키지 않습니다. 하지만, 저 코드 역시 마찬가지로 "Conformance mode"를 켜면 이렇게 오류가 발생합니다.

Error C2440 'initializing': cannot convert from 'CComPtr<IDXGIResource>' to 'CComQIPtr<ID3D11Texture2D>'




이 상황을 해결하려면 간단하게는 "Conformance" 모드를 끄면 됩니다. 하지만, 근래의 C++ 표준 준수를 위한 분위기를 고려한다면 좋은 선택이 아닙니다.

그렇다면, 다시 원론적으로 문제를 해결하는 수밖에는 없습니다. 일단, 대입은 안 되므로 다음과 같은 식으로 QueryInterface를 풀어쓰면서 스마트 포인터는 나름대로 그와 연동해 사용하는 것입니다.

// 이 한 줄의 코드를,
// CComQIPtr<ID3D11Texture2D> spTextureResource = spDXGIResource;
    
// 이렇게 풀어서 적용
CComQIPtr<ID3D11Texture2D> spTextureResource;
{
    ID3D11Texture2D* pTextureResource;
    hr = spDXGIResource->QueryInterface(__uuidof(ID3D11Texture2D), (LPVOID*)&pTextureResource);
    if (hr == S_OK)
    {
        spTextureResource.Attach(pTextureResource);
    }
}

좀 멋은 없어졌지만... 뭐 그래도... ^^;




그나저나, 갑자기 저 문제가 왜 발생한 것일까요? ^^ 문서를 보면 이에 대한 설명이 나옵니다.

/permissive- (Standards conformance)
; https://learn.microsoft.com/en-us/cpp/build/reference/permissive-standards-conformance

이 기능은 Visual Studio 2017부터 15.5 버전 이후로 기본값으로 바뀌었다고 합니다. 그러니까, "DXGI를 이용한 윈도우 화면 캡처 소스 코드(Visual C++)" 글을 썼던 게 2017-12-05일이니까 아마도 그때 당시에는 /permissive+ 상태가 기본값이어서 아무런 문제가 없었던 것입니다.

하지만, 15.5 버전 패치가 적용된 이후부터는 새 프로젝트를 만들어 저 소스코드를 복사해 적용한 사용자들의 경우에는 /permissive- 옵션의 영향으로 컴파일 오류가 발생했던 것입니다.

참고로, /permissive- 옵션은 C++ 표준 옵션을 "Preview - Features from the Latest C++ Working Draft (/std:c++latest)" 또는 "/std:c++20"으로 설정하면 암시적으로 켜진다고 합니다.




그런데, 생각보다 커뮤니티의 분위기가 조용합니다. 관련해서 검색해 보면 아래의 이슈 하나가 나오는데요,

afxhtml.h fails to compile when disabling permissive mode
; https://developercommunity.visualstudio.com/t/afxhtmlh-fails-to-compile-when-disabling-permissiv/471981

재현 예제 코드를 정리하면 결국 이것과 같습니다.

#include <atlbase.h>

int main()
{
    CComPtr<IUnknown> pUnk;
    CComQIPtr<IDispatch> pDisp = pUnk;
}

// After importing a TLB, how do I convert from one type of _com_ptr_t to another?
// https://devblogs.microsoft.com/oldnewthing/20221228-00/?p=107621

이슈 대응을 보면 "Visual Studio 2019 버전 16.3"부터 해결되었다고 나옵니다. 이상하군요, 전혀 해결되지 않았는데... 혹시 제가 모르는 또 다른 옵션이 있을까요? ^^ 암튼, 근래에는 아무래도 ATL COM의 분위기가 많이 식었기 때문에 그에 따라 CComPtr과 CComQIPtr을 사용하는 경우도 많지 않은 듯합니다.




참고로, 예전에 Conformance 모드 관련해 써 놓은 글이 하나 있군요. ^^

C++의 연산자 동의어(operator synonyms), 대체 토큰
; https://www.sysnet.pe.kr/2/0/12424




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

[연관 글]






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

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

비밀번호

댓글 작성자
 



2023-04-03 11시45분
What is this [uuid(…)] in front of my C++ class declaration?
; https://devblogs.microsoft.com/oldnewthing/20230331-00/?p=107998

permissive 옵션을 켜면, Visual C++의 비표준 확장을 사용할 수 없으므로 [uuid] 속성도 컴파일 오류가 발생한다고 합니다. 대신 __declspec을 사용하면 되는데, 이것 역시 비표준이지만 단지 표준 방식(2개의 밑줄로 시작하는 식별자)으로 확장된 것이므로 허용을 한다고.

What are the duck-typing requirements of _com_ptr_t?
; https://devblogs.microsoft.com/oldnewthing/20230508-00/?p=108156

-------------------------

"/std:c++20" 옵션인 경우 permissive 옵션을 명시적으로 끈다고 해도 여전히 permissive 옵션이 적용된다고 합니다.

In my Visual Studio project, I set my Conformance mode to permissive, but it’s still not permissive
; https://devblogs.microsoft.com/oldnewthing/20240725-00/?p=110045

이건 옵션의 적용 순서에 따른 버그로 보인다고 하는데요, 따라서 permissive 옵션을 설정 창을 통해 끄지 말고 그냥 "Additional Options"에 명시하는 식으로 우회할 수 있다고 합니다.

-------------------------

Extending CComPtr for remote activation
; https://www.codeproject.com/Articles/5386186/Extending-CComPtr-for-remote-activation
정성태

... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1252정성태3/1/201225312Windows: 55. 윈도우 8 베타 설치 과정 [1]
1251정성태2/27/201229262VC++: 60. C/C++ Native 스레드 콜 스택 덤프를 얻는 공개 라이브러리 [2]파일 다운로드1
1250정성태2/27/201231373VC++: 59. C/C++ 프로젝트 빌드 속도 개선 - UnityBuild를 아세요? [3]
1249정성태2/26/201231061.NET Framework: 311. .NET 스레드 콜 스택 덤프 (5) - ICorDebug 인터페이스 사용법 [2]파일 다운로드3
1248정성태2/25/201242544.NET Framework: 310. C#의 Shift 비트 연산 정리파일 다운로드1
1247정성태2/25/201225250.NET Framework: 309. .NET 응용 프로그램에 기본 생성되는 스레드들에 대한 탐구 [1]파일 다운로드1
1246정성태2/25/201224826개발 환경 구성: 145. 한영 변환은 되지만, 정작 한글 입력이 안되는 경우
1245정성태2/25/201235538개발 환경 구성: 144. 윈도우에서도 유닉스처럼 명령행으로 원격 접속하는 방법
1244정성태2/24/201232709.NET Framework: 308. .NET System.Threading.Thread 개체에서 Native Thread Id를 구할 수 있을까? [1]파일 다운로드1
1243정성태2/23/201232662개발 환경 구성: 143. Visual Studio 2010 - .NET Framework 소스 코드 디버깅 - 두 번째 이야기 [1]
1242정성태2/20/201239514VC++: 58. API Hooking - 64비트를 고려해야 한다면? EasyHook! [7]파일 다운로드1
1241정성태2/20/201226358.NET Framework: 307. .NET 4.0 응용 프로그램을 위한 ILMerge
1240정성태2/19/201232628디버깅 기술: 48. C/C++ JNI DLL을 Visual Studio로 디버깅하는 방법 [2]
1239정성태2/19/201224275.NET Framework: 306. 컴퓨터에 실행된 프로세스 중에 닷넷 응용 프로그램임을 알 수 있는 방법 - C# [1]파일 다운로드1
1238정성태2/19/201228152.NET Framework: 305. GetPrivateProfileSection / WritePrivateProfileSection의 C# 버전파일 다운로드1
1237정성태2/18/201232472개발 환경 구성: 142. Windows Embedded POSReady 7 설치 [1]
1236정성태2/17/201228278개발 환경 구성: 141. Windows 2008 R2 RDP 라이선스 서버 설치하는 방법
1235정성태2/16/201226726.NET Framework: 304. Hyper-V의 가상 머신을 C#으로 제어하는 방법 [1]파일 다운로드1
1234정성태2/16/201227137.NET Framework: 303. 원본 파일의 공백/라인을 유지한 체 XML 파일을 저장하는 방법 [1]파일 다운로드1
1233정성태2/16/201233232.NET Framework: 302. supportedRuntime 옵션과 System.BadImageFormatException 예외 [5]
1232정성태2/9/201229142VC++: 57. 웹 브라우저에서 Flash만 빼고 다른 ActiveX를 차단할 수 있을까? [3]파일 다운로드1
1231정성태2/8/201238676VC++: 56. Win32 API 후킹 - Trampoline API Hooking [5]파일 다운로드1
1230정성태2/6/201224054개발 환경 구성: 140. 프로젝트 생성 시부터 "Enable the Visual Studio hosting process" 옵션을 끄는 방법
1229정성태2/4/201229078.NET Framework: 301. P/Invoke의 성능을 높이기 위해 C++/CLI가 선택되려면? [5]파일 다운로드1
1228정성태2/4/201278370.NET Framework: 300. C#으로 만드는 음성인식/TTS 프로그램 [47]파일 다운로드1
1227정성태2/3/201229251.NET Framework: 299. 해당 어셈블리가 Debug 빌드인지, Release 빌드인지 알아내는 방법파일 다운로드1
... 151  [152]  153  154  155  156  157  158  159  160  161  162  163  164  165  ...