-
Notifications
You must be signed in to change notification settings - Fork 93
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
COM interfaces should be *interfaces*, and structs should be marshable #26
Comments
Does this mean that CsWinRT would use |
How does the |
Simple: Edit: AFAIK other than |
Coming from #86 I'd suggest making the non-struct projection the default and not an opt-in, so people who don't have in-depth knowledge of COM don't immediately break the whole environment by violating COM rules when talking with external components. Struct based COM objects have a lot of disadvantages and are only useful if you know exactly what you're doing. For general one-off interop its usually not worth the effort, C++ has smart pointers and ATL/WRL for implementing COM objects for a reason, because its really hard to get everything right if you have to do everything yourself. Exposing the raw COM ABI as the "default experience" to C# developers looking for an interop nuget package feels like a flawed design. Disadvantages of struct based COM objects:
|
My two cents is that this is a bad idea, in general. There are certainly places where not using pointers is possible and where using "safe" alternatives like However, interop code is itself fundamentally unsafe. The correctness of it is dependent on the bindings themselves being correct and safe and in many cases they cannot because they use concepts that cannot be represented in C# or understood by the JIT/VM. In many cases trying to avoid unsafe can be more unsafe than just using unsafe directly. Some examples.... In many cases, APIs take a In several cases, APIs take or have variable-length arrays which means they fundamentally cannot ever be handled by value because the type system does not contain the full metadata required to do so (this is true even in C/C++). C# does not have good scoping rules for Likewise, there are many APIs that take |
@tannergooding: I started where you're at, which is why the first codegen I wrote uses all blittable structs and only relied on .NET marshaling for method signatures (and not on COM "interface" methods, which are implemented via C# 9 function pointers and thus cannot be marshaled). But we're getting a lot of feedback about this. Folks want to implement COM interfaces and this simply cannot be done if we declare it as a struct. But once we use interfaces, structs that refer to them become managed types and thus non-blittable. Then methods that take those structs cannot use pointers to them any more and are forced into using more of the .NET marshalling layer. So it is very much a cascade. I agree that there are APIs that simply cannot be correctly represented in the marshaling world. And the two worlds are incompatible due to the cascading effect. So my idea here is that if the switch is set to generate the .NET marshaled world, some APIs will simply be off-limits -- we will error out rather than generate them. Thoughts? |
I think this is part of what the new low level APIs for RCW and CCW is meant to support: dotnet/runtime#1845.
I think that's probably reasonable overall. That being said, one of my concerns is that there are many APIs (especially in COM) where One example of this is D3D12CreateDevice where the fourth parameter is considered optional and if it is There are also functions where you basically do a query to determine number of elements by passing a valid pointer to So it's very easy to get into a scenario where you define these "safe" COM interfaces but where one function basically prevents the overall usage of them or where you are now dependent on With regular P/Invokes, you can at least define multiple helpers that resolve to the same internal P/Invoke. However, with COM any method you define in the interface is considered part of the VTBL and thus can hinder things. You can somewhat use extension methods, but that often means you have some overload that either takes pointers or |
Yes, that's what we do today. I was hoping to get the best of both worlds at once by using pointers for the
I plan to keep generating friendly overloads on COM interfaces via extension methods, assuming the original overloads aren't friendly enough. |
Its one thing having to write unsafe code, and a completely different thing having to write the whole COM infrastructure like apartment marshaling logic manually because .NET is free threaded while COM is not. People will either forget finalizers (leaking COM objects) or forget apartments (violating COM rules, destabilzing the environment). C# is used by people not exposed to low level COM, C++ teams already recognized the code quality problems of manual reference counting and are using smart pointer classes, reintroducing manual reference counting as the default experience into .NET is simply a design mistake and not appropriate. (Maybe this library isn't intended to be the default go-to for interop, but looking at the goals of the Windows metadata project and how prominently this projection is featured there, it definitely will get a lot of visibility to people who "just" want to do interop. This happens a lot for desktop applications, including COM interop..)
I don't exactly see why these are supposed to be incompatible, COM references are just pointers with lots of rules attached and I'd expect to be able to cast them between representations (via helper methods, not necessarily a C# language cast). For classic COM interop this is already possible via the Personally I'd have expected this interop layer to coordinate with the work from CsWinRT which, the last time I looked, intended to build the new generation of the native COM interop layer outside the dotnet runtime. So I'd definitely prefer if CsWin32 picks up CsWinRT infrastructure rather than legacy .NET runtime COM marshaling. If theres any reason to still have struct based vtables for performance sensitive scenarios I believe they should be opt-in on a per-interface level and people can do the appropriate "casts" at the appropriate stage (early once vs. late before a call) like they already can do now in classic COM marshaling (IntPtr vs. interface) when the runtime treatment of COM interop is not sufficient.
This definitely happens very often, and its always been awkward in C# projections. I believe the correct projection is to a one-element array of the interface type (?) which isn't very user friendly so most people prefer writing their ComImport incorrectly to make it easier to use when they think they won't be needing to support null-pointer-to-interface scenarios. However I don't believe using structs for "literal" projections outweights the complete loss of infrastructure and the resulting destabilization and complexity increase. |
I agree that ref counting can be painful. In my own libraries I largely just have a small It definitely still isn't perfect and we don't have an allocation free way to make it such today. |
This would be the new lowlevel CCW/RCW APIs I mentioned above: dotnet/runtime#1845 I talked to him on teams a few minutes back and he basically summed it up as being for 2 scenarios:
and basically that if there is no desire to support WinRT is basically offers a more efficient lifetime mechanism. |
One more thing to consider. The prototype for our AOT friendly |
I haven't looked into how CsWinRT works, but as it interacts with WinRT whereas CsWin32 interacts with Win32, I don't expect it follows that we can do everything the same way. I also do not intend to emit code that requires .NET 6 when no one is on it yet. Folks who target net35 want to use this generator, and while I'm on the fence on how much to support that, I certainly can't give up supporting net472. |
All of the COM could be done in the same way CsWinRT supports WinRT. CsWinRT even has a limited/naive/slow .netstandard2.0 version.
Agreed. The |
@AArnott There might actually be another way to improve the interoperability for COM in this model - |
Is it slower than ordinary .NET /COM interop? If so, what would the user gain by switching to a slower WinRT system? |
AOT friendly is the key here. It is all known managed code. So it is slower than |
Crossing off one of the 4 goals and renaming this issue to scope to just the COM problem. As discussed above, the first "goal" in the issue description may not be a good goal. Someone can open another issue to track that if our friendly overloads are not enough. But remember that we're just getting started with friendly overloads--they're going to get friendlier. |
@AArnott last nightly build seems from 5 days ago (3/10/2021, 0.1.410-beta), am I missing something? would love to test this update to see if its usable for me now |
The nightly build hit a network snag. It's now available as 0.1.413-beta on the nightly feed. |
In consideration of projects that are maintained by developers that prefer to avoid pointers and the
unsafe
keyword in C#, we might offer a setting to generate code that does not use pointers. In particular, code generation would:Generate only the "friendly overloads" as the direct extern methods. Instead of getting pointer-basedextern
methods with pointer-free friendly overloads, we would generate just theextern
methods with the friendly signatures.Note that mixing the two varieties where marshalled structs and pointer-based
extern
methods coexist is a non-goal as it would be very complex to maintain and perhaps generate confusing code as well.The text was updated successfully, but these errors were encountered: