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

.NET COM+ 를 Managed/Native 클라이언트에서 각각 호출했을 때의 콜 스택 비교


우선, 테스트 환경을 구성해 볼까요? ^^

C#으로 서버 유형의 COM+ 코드를 다음과 같이 만들어 줍니다.

[GuidAttribute("0C0445BE-A12E-4DE4-A022-156EF491989D")]
[ClassInterface(ClassInterfaceType.AutoDual)]
public class MyClass : System.EnterpriseServices.ServicedComponent
{
    public object DoIt()
    {
        System.Diagnostics.Trace.WriteLine(Environment.StackTrace);
        return "test";
    }
}

[assembly: ApplicationAccessControl(false)]
[assembly: ApplicationActivation(ActivationOption.Server)]
[assembly: ApplicationName("TestRDS")]
[assembly: ApplicationID("727FC170-1D80-4e89-84CC-22AAB10A6F24")]
[assembly: ComVisible(true)]

그리고, C# 으로 해당 COM+ 를 호출하는 코드를 (관리자 권한으로) 불러주면,

class Program
{
    static void Main(string[] args)
    {
        TestCOMObj.MyClass mc = new TestCOMObj.MyClass();
        Console.WriteLine(mc.DoIt());
    }
}

COM+ 구성요소 서비스에 새롭게 COM+ 응용 프로그램이 생성된 것을 볼 수 있습니다.

자, 이제 Visual Studio 의 "Attach to process..." 기능을 이용해서 실행 중인 COM+ exe 개체에 디버거를 연결한 후, DoIt 메서드에 중단점을 잡고 C# 클라이언트를 다시 호출해 주면 다음과 같은 콜 스택이 확인됩니다.

TestCOMObj.dll!TestCOMObj.MyClass.DoIt() Line 18    C#
[Native to Managed Transition]  
[Managed to Native Transition]  
mscorlib.dll!System.Runtime.Remoting.Messaging.StackBuilderSink.PrivateProcessMessage(...[생략]...) + 0x1b bytes  
mscorlib.dll!System.Runtime.Remoting.Messaging.StackBuilderSink.SyncProcessMessage(...[생략]...) + 0x26a bytes    
mscorlib.dll!System.Runtime.Remoting.RemotingServices.ExecuteMessage(...[생략]...) + 0x63 bytes   
System.EnterpriseServices.dll!System.EnterpriseServices.ServicedComponentProxy.LocalInvoke(...[생략]...) + 0x1b3 bytes    
System.EnterpriseServices.dll!System.EnterpriseServices.ServicedComponentProxy.Invoke(...[생략]...) + 0x55 bytes  
System.EnterpriseServices.dll!System.EnterpriseServices.ServicedComponent.RemoteDispatchHelper(...[생략]...) + 0x38 bytes 
System.EnterpriseServices.dll!System.EnterpriseServices.ServicedComponent.System.EnterpriseServices.IRemoteDispatch.RemoteDispatchNotAutoDone(string s) + 0x14 bytes    
[Native to Managed Transition]  

다음으로, C/C++ 클라이언트를 만들고,

#include "stdafx.h"

#import "..\\..\\ConsoleApplication1\\ConsoleApplication1\\bin\\Debug\\TestCOMObj.tlb" no_namespace raw_interfaces_only

int _tmain(int argc, _TCHAR* argv[])
{
    CoInitialize(NULL);
    {
        _MyClassPtr ptr;
        ptr.CreateInstance(__uuidof(MyClass));

        VARIANT vtResult;
        ptr->DoIt(&vtResult);

    }

    CoUninitialize();

    return 0;
}

이제 호출을 해보면, C# COM+ 측의 콜스택이 어떻게 나올까요? 제 경우에는, 별반 다르지 않게 콜 스택이 나올 거라고 생각했는데, 그게 아니었습니다.

complus_from_nativecall_1.png

보시는 것처럼, System.EnterpriseServices, System.Runtime.Remoting.Messaging 클래스가 전혀 관여하지 않고 Native 클라이언트로부터 온 호출이 직접 전달되고 있습니다.

즉, .NET COM+ 개체 자체의 GC 등으로 인한 영향을 제외한다면 Native 클라이언트로부터 .NET COM+ 개체를 호출하는 경우 오버헤드가 전혀 없음을 알 수 있습니다.




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







[최초 등록일: ]
[최종 수정일: 8/8/2018]

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

비밀번호

댓글 작성자
 



2012-10-14 11시18분
[ryujh] 안녕하세요.

native 클라이언트는 .net 의 존재(?)를 모르니까 COM+로 접근하여 개체를 호출하니까 이해가 가는데

COM+ 개체에서 디버깅을 보니 의문점이 있습니다.

개체가 실행시 어트리뷰트 설정대로 COM+ 에 등록해주니까 실행 후에 DoIt 안에서는 아무것도 안하니까 오버헤드가 없는걸로 보입니다.

DoIt 안에서 ServicedComponent 클래스 멤버를 사용하면 오버헤드가 걸리지 않는지요?

주제대로 호출했을 때만 비교한다면 의문점은 없지만 궁금해서 글남깁니다.


[guest]
2012-10-15 12시35분
"ryujh"님... 예를 들어, MyClass 안에 Test() 라는 메서드를 추가하고 그것을 DoIt 안에서 호출하면 오버헤드가 걸리지 않을까?.... 라는 질문인가요? 글쎄요... 구체적으로 어떤 오버헤드를 말씀하시는 건지 잘 모르겠습니다. DoIt에서 Test 메서드를 부르면 DoIt과 Test메서드 사이에 콜스택이 더 있을 거라는 예상인가요? 아니면 Test메서드를 불렀기 때문에 Native에서 호출하는 경우 예전에는 보이지 않았던 Native와 Managed사이의 콜스택이 더 있을 거라는 예상인가요?

질문이 잘 이해가 안됩니다. ^^
정성태
2012-10-15 12시53분
[ryujh] 급하게 쓰느라고 질문이 잘못된 것 같습니다.

예제인 DoIt 호출시 콜스택의 오버헤드가 없는 것은 이해했고

DoIt 내부에서 ServiceComponent 의 메소드를 사용한다면 .net COM+ 관련 오버헤드가 생기지 않을지요? 이겁니다.

어찌보면 오버헤드가 생긴다는 당연한 것을 질문한것 같아서 잡설이 된것 같군요.

----

그리고 class 어트리뷰트가 DoIt 호출과는 무관한 것인지요?

이상입니다.



[guest]
2012-10-15 01시29분
일단 DoIt 으로 진입하면 그 이후의 호출은 닷넷이니까, 당연히 CLR위에서의 오버헤드는 있을 것입니다. 그리고, 클래스 어트리뷰트 중에서 딱히 어떤 것이 있을까요? 트랜잭션이나 JIT 설정이라면 어차피 그것들로 인해서 닷넷의 기능이 추가되는 것이 아니고 COM+ 에게 힌트를 주는 것이기 때문에 (예상으로는 ^^) 그다지 영향은 없을 것 같습니다.
정성태

... 121  122  123  124  125  126  127  128  129  130  131  132  133  [134]  135  ...
NoWriterDateCnt.TitleFile(s)
1739정성태8/24/201427758.NET Framework: 457. 교착상태(Dead-lock) 해결 방법 - Lock Leveling [2]파일 다운로드1
1738정성태8/23/201423445.NET Framework: 456. C# - CAS를 이용한 Lock 래퍼 클래스파일 다운로드1
1737정성태8/20/201420929VS.NET IDE: 93. Visual Studio 2013 동기화 문제
1736정성태8/19/201426916VC++: 79. [부연] CAS Lock 알고리즘은 과연 빠른가? [2]파일 다운로드1
1735정성태8/19/201419419.NET Framework: 455. 닷넷 사용자 정의 예외 클래스의 최소 구현 코드 - 두 번째 이야기
1734정성태8/13/201421184오류 유형: 237. Windows Media Player cannot access the file. The file might be in use, you might not have access to the computer where the file is stored, or your proxy settings might not be correct.
1733정성태8/13/201427519.NET Framework: 454. EmptyWorkingSet Win32 API를 사용하는 C# 예제파일 다운로드1
1732정성태8/13/201435826Windows: 99. INetCache 폴더가 다르게 보이는 이유
1731정성태8/11/201428299개발 환경 구성: 235. 점(.)으로 시작하는 파일명을 탐색기에서 만드는 방법
1730정성태8/11/201423481개발 환경 구성: 234. Royal TS의 터미널(Terminal) 연결에서 한글이 깨지는 현상 해결 방법
1729정성태8/11/201419478오류 유형: 236. SqlConnection - The requested Performance Counter is not a custom counter, it has to be initialized as ReadOnly.
1728정성태8/8/201431715.NET Framework: 453. C# - 오피스 파워포인트(Powerpoint) 파일을 WinForm에서 보는 방법파일 다운로드1
1727정성태8/6/201421918오류 유형: 235. SignalR 오류 메시지 - Counter 'Messages Bus Messages Published Total' does not exist in the specified Category. [2]
1726정성태8/6/201420707오류 유형: 234. IIS Express에서 COM+ 사용 시 SecurityException - "Requested registry access is not allowed" 발생
1725정성태8/6/201422654오류 유형: 233. Visual Studio 2013 Update3 적용 후 Microsoft.VisualStudio.Web.PageInspector.Runtime 모듈에 대한 FileNotFoundException 예외 발생
1724정성태8/5/201427454.NET Framework: 452. .NET System.Threading.Thread 개체에서 Native Thread Id를 구하는 방법 - 두 번째 이야기 [1]파일 다운로드1
1723정성태7/29/201459850개발 환경 구성: 233. DirectX 9 예제 프로젝트 빌드하는 방법 [3]파일 다운로드1
1722정성태7/25/201422196오류 유형: 232. IIS 500 Internal Server Error - NTFS 암호화된 폴더에 웹 애플리케이션이 위치한 경우
1721정성태7/24/201425503.NET Framework: 451. 함수형 프로그래밍 개념 - 리스트 해석(List Comprehension)과 순수 함수 [2]
1720정성태7/23/201423462개발 환경 구성: 232. C:\WINDOWS\system32\LogFiles\HTTPERR 폴더에 로그 파일을 남기지 않는 설정
1719정성태7/22/201427350Math: 13. 동전을 여러 더미로 나누는 경우의 수 세기(Partition Number) - 두 번째 이야기파일 다운로드1
1718정성태7/19/201436799Math: 12. HTML에서 수학 관련 기호/수식을 표현하기 위한 방법 - MathJax.js [4]
1716정성태7/17/201436492개발 환경 구성: 231. PC 용 무료 안드로이드 에뮬레이터 - genymotion
1715정성태7/13/201431581기타: 47. 운영체제 종료 후에도 USB 외장 하드의 전원이 꺼지지 않는 경우 [3]
1714정성태7/11/201421573VS.NET IDE: 92. Visual Studio 2013을 지원하는 IL Support 확장 도구
1713정성태7/11/201445348Windows: 98. 윈도우 시스템 디스크 용량 확보를 위한 "Package Cache" 폴더 이동 [1]
... 121  122  123  124  125  126  127  128  129  130  131  132  133  [134]  135  ...