Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 2개 있습니다.)

C# - Unhandled exception. System.Runtime.InteropServices.COMException (0x800080A5)

해당 오류를 재현하기 위해, 지난 글에서 실습한 (.NET 8로 만든) C# COM DLL을,

.NET Core 3/5+ 기반의 COM Server를 기존의 regasm처럼 등록하는 방법
; https://www.sysnet.pe.kr/2/0/13459

using System.Runtime.InteropServices;
using System;
using System.Windows.Forms;

namespace ClassLibrary1
{
    [ComVisible(true)]
    [Guid("23172f2f-a3d3-4180-97ae-7805f74a5a46")]
    [InterfaceType(ComInterfaceType.InterfaceIsDual)]
    public interface IMyNetCode
    {
        void ShowDialog();
    }

    [ComVisible(true)]
    [Guid("41AC8568-9230-4E63-B7C5-CAAD997EE207")]
    public class MyNetCode : IMyNetCode
    {
        public void ShowDialog()
        {
            string platform = (IntPtr.Size == 4) ? "x86" : "x64";
            MessageBox.Show($" {RuntimeEnvironment.GetSystemVersion()}: {platform}");
        }
    }
}

동일한 .NET 8 C# Console Application에서 활성화시켜 실행하면,

using ClassLibrary1;
using System.Runtime.InteropServices;

namespace ConsoleApp1
{
    internal class Program
    {
        static void Main(string[] args)
        {
            Guid guid = new Guid("{41AC8568-9230-4E63-B7C5-CAAD997EE207}");
            Type type = Type.GetTypeFromCLSID(guid);

            object comObject = Activator.CreateInstance(type); // 예외 발생
        }
    }
}

이런 오류가 발생합니다.

C:\temp\net45_from_net80\x64\Debug> ConsoleApp1.exe
Unhandled exception. System.Runtime.InteropServices.COMException (0x800080A5): Retrieving the COM class factory for component with CLSID {41AC8568-9230-4E63-B7C5-CAAD997EE207} failed due to the following error: 800080a5 0x800080A5.
   at System.RuntimeTypeHandle.AllocateComObject(Void* pClassFactory)
   at System.RuntimeType.CreateInstanceDefaultCtor(Boolean publicOnly, Boolean wrapExceptions)
   at ConsoleApp1.Program.Main(String[] args) in C:\temp\net45_from_net80\ConsoleApp1\Program.cs:line 16

오류 코드가 0x800080A5 (-2147450715)인데, 검색해도 잘 안 나오는 꽤나 낯선 오류입니다. ^^;




원인은 프로젝트 설정에서 기인합니다. C# COM DLL 측에서 Windows Forms의 MessageBox.Show를 사용했기 때문에 해당 프로젝트는 다음과 같이 구축한 상태인데요,

<Project Sdk="Microsoft.NET.Sdk">
    <PropertyGroup>
        <TargetFramework>net8.0-windows</TargetFramework>
        <OutputType>Library</OutputType>
        <UseWindowsForms>true</UseWindowsForms>

        <EnableComHosting>true</EnableComHosting>
        <EnableRegFreeCom>true</EnableRegFreeCom>

        <BaseOutputPath>..\x64</BaseOutputPath>
        <AppendTargetFrameworkToOutputPath>false</AppendTargetFrameworkToOutputPath>
        <Platform>x64</Platform>

    </PropertyGroup>
    <PropertyGroup>
        <SignAssembly>true</SignAssembly>
    </PropertyGroup>
    <PropertyGroup>
        <AssemblyOriginatorKeyFile>test.snk</AssemblyOriginatorKeyFile>
    </PropertyGroup>

    <Target Name="PostBuild" AfterTargets="PostBuildEvent">
        <Exec Command="dscom tlbexport &quot;$(TargetPath)&quot;" />
    </Target>

    <ItemGroup Condition="Exists('$(AssemblyName).tlb')">
        <ComHostTypeLibrary Include="$(AssemblyName).tlb" Id="1"/>
    </ItemGroup>
</Project>

따라서 호출 측에서도, 비록 Windows Forms 관련 코드를 직접적으로 사용하지 않았지만 마찬가지로 UseWindowsForms 옵션을 추가해야 합니다.

<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
      <TargetFramework>net8.0-windows</TargetFramework>
      <OutputType>Exe</OutputType>
      <UseWindowsForms>true</UseWindowsForms>

      <BaseOutputPath>..\x64</BaseOutputPath>
      <AppendTargetFrameworkToOutputPath>false</AppendTargetFrameworkToOutputPath>
      <Platform>x64</Platform>
      
      <ApplicationManifest>app.manifest</ApplicationManifest>

  </PropertyGroup>

</Project>

이후 다시 실행하면, 정상적으로 실행이 됩니다. 그런데, 여기서 재미있는 건, COM Client 역할을 하는 측의 프로젝트를 그냥 일반적인 .NET Framework Console Application으로 만들거나, 단순히 C/C++ 클라이언트로 만든 경우에는 저런 설정과 무관하게 잘 실행이 된다는 점입니다.




한 가지 더 재미있는 점이 있는데요, COM Server와 클라이언트의 TargetFramework을 아래와 같이 바꾸면,

[C# COM 서버 역할을 하는 ClassLibrary1]
    <TargetFramework>net7.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

[C# COM Client 역할을 하는 ConsoleApp1]
    <TargetFramework>net7.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

당연히 잘 동작하겠지만, RuntimeEnvironment.GetSystemVersion()의 결괏값이 "v4.0.30319"라고 나온다는 점입니다. .NET 8 환경에서는 정상적으로 "v8.0.0"이 나왔던 반면, .NET 7인 경우에는 마치 Desktop용의 .NET Framework을 가리키는 버전이 나오는 것입니다.

아울러, 다음과 같이 버전이 서로 일치하지 않는 경우에도,

[C# COM 서버 역할을 하는 ClassLibrary1]
    <TargetFramework>net7.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

[C# COM Client 역할을 하는 ConsoleApp1]
    <TargetFramework>net8.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

혹은 반대로 하는 경우에도,

[C# COM 서버 역할을 하는 ClassLibrary1]
    <TargetFramework>net8.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

[C# COM Client 역할을 하는 ConsoleApp1]
    <TargetFramework>net7.0-windows</TargetFramework>
    <UseWindowsForms>true</UseWindowsForms>

이유는 알 수 없지만, 동일하게 "0x800080A5" 오류가 발생합니다. (제가 테스트한 것은 .NET 5부터 한 것인데, 모두 COM 서버와 클라이언트의 프로젝트 버전이 동일해야 합니다. 아마도 이전의 .NET Core 런타임에서도 마찬가지 현상이 있을 듯합니다.) 이런 조합의 수까지 고려해 테스트를 하려면, 마이크로소프트도 참 골치 아플 것 같습니다. ^^;

(2023-12-02 업데이트) 정식 문서에 보면,
Hosting layer error and exit codes
; https://github.com/dotnet/runtime/blob/main/docs/design/features/host-error-codes.md

0x800080a5에 대해,

Error returned by hostfxr_initialize_for_runtime_config if the component being initialized requires framework which is not available or incompatible with the frameworks loaded by the runtime already in the process. For example trying to load a component which requires 3.0 into a process which is already running a 2.0 runtime.

라는 설명이 나오는군요. ^^ 결국 지난번의 결론에 해당합니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/2/2023]

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

비밀번호

댓글 작성자
 




... 91  92  93  94  95  96  [97]  98  99  100  101  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11546정성태6/12/201825979.NET Framework: 766. C# 7.2의 특징 - GC 및 메모리 복사 방지를 위한 struct 타입 개선 [9]파일 다운로드1
11545정성태6/11/201824783오류 유형: 469. .NET Core 프로젝트를 Visual Studio에서 실행 시 System.BadImageFormatException 발생하는 경우 [1]
11544정성태6/10/201824258.NET Framework: 765. C# 7.2 - 숫자 리터럴의 선행 밑줄과 뒤에 오지 않는 명명된 인수
11543정성태6/9/201823577.NET Framework: 764. C# 7.2 - private protected 접근자 추가파일 다운로드1
11542정성태6/9/201862096개발 환경 구성: 381. Azure Web App 확장 예제 - Remove Custom Headers
11541정성태6/9/201821316개발 환경 구성: 380. Azure Web App 확장 배포 방법 [1]
11540정성태6/9/201821362개발 환경 구성: 379. Azure Web App 확장 예제 제작 [2]
11539정성태6/8/201821974.NET Framework: 763. .NET Core 2.1 - Tiered Compilation 도입파일 다운로드1
11538정성태6/8/201820625.NET Framework: 762. .NET Core 2.1 - 확장 도구(Tools) 관리 [1]
11537정성태6/8/201825856.NET Framework: 761. C# - SmtpClient로 SMTP + SSL/TLS 서버를 이용하는 방법 [5]
11536정성태6/7/201822602.NET Framework: 760. Microsoft Build 2018 - The future of C# 동영상 내용 정리 [1]파일 다운로드1
11535정성태6/7/201825414.NET Framework: 759. C# - System.Span<T> 성능 [1]
11534정성태6/6/201830846.NET Framework: 758. C# 7.2 - Span<T> [6]
11533정성태6/5/201833571.NET Framework: 757. 포인터 형 매개 변수를 갖는 C++ DLL의 함수를 C#에서 호출하는 방법파일 다운로드1
11532정성태6/5/201822713.NET Framework: 756. JSON의 escape sequence 문자 처리 방식
11531정성태6/4/201827010오류 유형: 468. JSON.parse가 허용하지 않는 문자 [9]
11530정성태5/31/201827897.NET Framework: 755. C# 7.2 - 스택에만 생성할 수 있는 값 타입 지원 - "ref struct" [2]파일 다운로드1
11529정성태5/23/201825236.NET Framework: 754. 닷넷의 관리 포인터(Managed Pointer)와 System.TypedReference [6]파일 다운로드1
11528정성태5/17/201824710.NET Framework: 753. C# 7.2 - 3항 연산자에 ref 지원(conditional ref operator) [1]
11527정성태5/17/201822353오류 유형: 467. RDP 로그인 에러 - This could be due to CredSSP encryption oracle remediation.
11526정성태5/16/201822252.NET Framework: 752. C# 7.2 - 메서드의 반환값 및 로컬 변수에 ref readonly 기능 추가파일 다운로드1
11525정성태5/16/201826400.NET Framework: 751. C# 7.2 - 메서드의 매개 변수에 in 변경자 추가 [3]파일 다운로드1
11524정성태5/15/201825339.NET Framework: 750. C# 7.2 - readonly 구조체 [5]파일 다운로드1
11523정성태5/15/201822363.NET Framework: 749. C# - 값 형식의 readonly 인스턴스에 대한 메서드 호출 시 defensive copy 발생 [1]파일 다운로드1
11522정성태5/15/201820782개발 환경 구성: 378. Azure - VM 진단 설정 화면의 "This subscription is not registered with the Microsoft.Insights resource provider."
11521정성태5/15/201819135개발 환경 구성: 377. Azure - 원하는 성능 데이터로 모니터링 대시보드 구성
... 91  92  93  94  95  96  [97]  98  99  100  101  102  103  104  105  ...