-
Notifications
You must be signed in to change notification settings - Fork 6k
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
The dispose pattern docs should not use an IntPtr.Zero
handle
#44201
Comments
@IEvangelist @gewarren I want to fix this since it keeps biting people, but I need your input before I go ahead. Copying the whole sample here for convenience: public class BaseClassWithSafeHandle : IDisposable
{
private bool _disposedValue;
private SafeHandle? _safeHandle = new SafeFileHandle(IntPtr.Zero, true);
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
protected virtual void Dispose(bool disposing)
{
if (!_disposedValue)
{
if (disposing)
{
_safeHandle?.Dispose();
_safeHandle = null;
}
_disposedValue = true;
}
}
} As explained above, - private SafeHandle? _safeHandle = new SafeFileHandle(IntPtr.Zero, true);
+ private SafeHandle? _safeHandle = File.OpenHandle(Path.GetTempFileName(), FileMode.CreateNew) but I would be unhappy with the result since it would be unclear why are we even showing safe handles instead of demonstrating higher-level API-s (eg. My plan to address this would be the following: (1) change Any thoughts? Would you be fine if I PR this change? |
@antonfirsov I'm good with that. Thank you! |
File.OpenHandle is only .NET 6+, so the sample wouldn't be viable for .NET Standard or .NET Framework TFMs. Changing it to |
@bartonjs my suggestion in #44201 (comment) is about actually demonstarting Thoughts? |
That'd be fine by me. It'll make the SafeHandle/Dispose sample bigger since it'll need to define its own version of SafeAllocHandle (e.g.https://source.dot.net/#Microsoft.AspNetCore.Cryptography.Internal/SafeHandles/LocalAllocHandle.cs,21); but since it won't require "new" API the sample is both informative and applies to all TFMs. |
Describe the issue or suggestion
The following doc was likely created in .NET Framework times, where the
BaseClassWithSafeHandle
code example creates and closes anIntPTr.Zero
handle, which is NOP on Windows: https://learn.microsoft.com/en-us/dotnet/standard/garbage-collection/implementing-dispose#implement-the-dispose-patternThe problem is that on Unix
FD 0
corresponds tostdnin
which means that disposingBaseClassWithSafeHandle
will close the standard input. What's even worse, that it will free upFD 0
for usage in other places, eg.Socket
meaning that creating and disposingBaseClassWithSafeHandle
several times may close the handle under an existing socket leading to unwanted side effects.There were cases where users left the dummy code from the sample in their codebase, eventually leading to hard-to-diagnose production errors, see dotnet/runtime#56750, dotnet/runtime#64305, fabian-blum/AspNetCore.Identity.LiteDB#14.
We should change this example.
Document Details
⚠ Do not edit this section. It is required for learn.microsoft.com ➟ GitHub issue linking.
The text was updated successfully, but these errors were encountered: