diff --git a/OpenMcdf.sln b/OpenMcdf.sln
index d6e3a65c..23cafb86 100644
--- a/OpenMcdf.sln
+++ b/OpenMcdf.sln
@@ -65,6 +65,8 @@ Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "OpenMcdf.Test", "sources\Te
EndProject
Project("{9A19103F-16F7-4668-BE54-9A1E7A4F7556}") = "OpenMcdf.Benchmark", "sources\Test\OpenMcdf.Benchmark\OpenMcdf.Benchmark.csproj", "{B3645D34-1E22-4BCC-8956-A8A56FA9F114}"
EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "StructuredStorage", "StructuredStorage\StructuredStorage.csproj", "{543A2F13-F9B1-4B9F-B982-EECC6F44F0E0}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@@ -103,6 +105,10 @@ Global
{B3645D34-1E22-4BCC-8956-A8A56FA9F114}.Debug|Any CPU.Build.0 = Debug|Any CPU
{B3645D34-1E22-4BCC-8956-A8A56FA9F114}.Release|Any CPU.ActiveCfg = Release|Any CPU
{B3645D34-1E22-4BCC-8956-A8A56FA9F114}.Release|Any CPU.Build.0 = Release|Any CPU
+ {543A2F13-F9B1-4B9F-B982-EECC6F44F0E0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {543A2F13-F9B1-4B9F-B982-EECC6F44F0E0}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {543A2F13-F9B1-4B9F-B982-EECC6F44F0E0}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {543A2F13-F9B1-4B9F-B982-EECC6F44F0E0}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
diff --git a/StructuredStorage/LockBytes.cs b/StructuredStorage/LockBytes.cs
new file mode 100644
index 00000000..dd514822
--- /dev/null
+++ b/StructuredStorage/LockBytes.cs
@@ -0,0 +1,42 @@
+using System.Diagnostics;
+using System.Runtime.InteropServices;
+using Windows.Win32;
+using Windows.Win32.Foundation;
+using Windows.Win32.System.Com.StructuredStorage;
+
+namespace StructuredStorage;
+
+///
+/// Encapsulates ILockBytes over an HGlobal allocation.
+///
+internal sealed class LockBytes : IDisposable
+{
+ readonly ILockBytes lockBytes;
+ private bool disposedValue;
+
+ public LockBytes(int count)
+ {
+ IntPtr hGlobal = Marshal.AllocHGlobal(count);
+ HRESULT hr = PInvoke.CreateILockBytesOnHGlobal((HGLOBAL)hGlobal, true, out lockBytes);
+ hr.ThrowOnFailure();
+ }
+
+ public void Dispose()
+ {
+ if (disposedValue)
+ return;
+
+ int count = Marshal.ReleaseComObject(lockBytes);
+ Debug.Assert(count == 0);
+
+ disposedValue = true;
+ GC.SuppressFinalize(this);
+ }
+
+ ~LockBytes()
+ {
+ Dispose();
+ }
+
+ internal ILockBytes ILockBytes => lockBytes;
+}
diff --git a/StructuredStorage/NativeMethods.json b/StructuredStorage/NativeMethods.json
new file mode 100644
index 00000000..66108bf1
--- /dev/null
+++ b/StructuredStorage/NativeMethods.json
@@ -0,0 +1,6 @@
+{
+ "$schema": "https://aka.ms/CsWin32.schema.json",
+ "wideCharOnly": true,
+ "emitSingleFile": true,
+ "public": false
+}
diff --git a/StructuredStorage/NativeMethods.txt b/StructuredStorage/NativeMethods.txt
new file mode 100644
index 00000000..9a9892f0
--- /dev/null
+++ b/StructuredStorage/NativeMethods.txt
@@ -0,0 +1,37 @@
+// Structured storage
+
+// Functions
+CreateILockBytesOnHGlobal
+PropVariantToVariant
+ReadClassStg
+ReadClassStm
+SHCreateItemFromParsingName
+SHCreateStreamOnFile
+StgCreateDocfileOnILockBytes
+StgCreateStorageEx
+StgIsStorageFile
+StgOpenStorage
+StgOpenStorageEx
+VariantClear
+VariantToPropVariant
+WriteClassStg
+WriteClassStm
+
+// Interfaces
+IEnumSTATPROPSETSTG
+IEnumSTATPROPSTG
+IEnumSTATSTG
+ILockBytes
+IPropertySetStorage
+IPropertyStorage
+IRootStorage
+IStorage
+IStream
+
+// Enumerations
+PROPSETFLAG_*
+STGTY
+
+// Constants
+STG_E_*
+PROPSETFLAG_*
diff --git a/StructuredStorage/PropertySetStorage.cs b/StructuredStorage/PropertySetStorage.cs
new file mode 100644
index 00000000..7c6df922
--- /dev/null
+++ b/StructuredStorage/PropertySetStorage.cs
@@ -0,0 +1,50 @@
+using Windows.Win32;
+using Windows.Win32.System.Com.StructuredStorage;
+
+namespace StructuredStorage;
+
+///
+/// Wraps IPropertySetStorage.
+///
+public sealed class PropertySetStorage
+{
+ ///
+ /// PROPSETFLAG constants.
+ ///
+ [Flags]
+#pragma warning disable CA1008
+ public enum Flags
+ {
+ Default = (int)PInvoke.PROPSETFLAG_DEFAULT,
+ NonSimple = (int)PInvoke.PROPSETFLAG_NONSIMPLE,
+ ANSI = (int)PInvoke.PROPSETFLAG_ANSI,
+ Unbuffered = (int)PInvoke.PROPSETFLAG_UNBUFFERED,
+ CaseSensitive = (int)PInvoke.PROPSETFLAG_CASE_SENSITIVE,
+ }
+#pragma warning restore CA1008
+
+ private readonly IPropertySetStorage propSet; // Cast of IStorage does not need disposal
+
+ internal PropertySetStorage(IStorage storage)
+ {
+ propSet = (IPropertySetStorage)storage;
+ }
+
+ public PropertyStorage Create(Guid formatID, StorageModes mode) => Create(formatID, Flags.Default, mode, Guid.Empty);
+
+ public PropertyStorage Create(Guid formatID, Flags flags = Flags.Default, StorageModes mode = StorageModes.ShareExclusive | StorageModes.AccessReadWrite) => Create(formatID, flags, mode, Guid.Empty);
+
+ public unsafe PropertyStorage Create(Guid formatID, Flags flags, StorageModes mode, Guid classID)
+ {
+ propSet.Create(&formatID, &classID, (uint)flags, (uint)mode, out IPropertyStorage stg);
+ return new(stg);
+ }
+
+ public unsafe PropertyStorage Open(Guid formatID, StorageModes mode = StorageModes.ShareExclusive | StorageModes.AccessReadWrite)
+ {
+ propSet.Open(&formatID, (uint)mode, out IPropertyStorage propStorage);
+ return new(propStorage);
+ }
+
+ public unsafe void Remove(Guid formatID) => propSet.Delete(&formatID);
+}
diff --git a/StructuredStorage/PropertyStorage.cs b/StructuredStorage/PropertyStorage.cs
new file mode 100644
index 00000000..d713aca0
--- /dev/null
+++ b/StructuredStorage/PropertyStorage.cs
@@ -0,0 +1,167 @@
+using System.Collections;
+using System.Diagnostics;
+using System.Runtime.InteropServices;
+using Windows.Win32;
+using Windows.Win32.Foundation;
+using Windows.Win32.System.Com.StructuredStorage;
+
+namespace StructuredStorage;
+
+///
+/// Enumerates STATPROPSTG elements from a PropertyStorage.
+///
+internal sealed class StatPropStgEnumerator : IEnumerator
+{
+ readonly IEnumSTATPROPSTG enumerator;
+ STATPROPSTG propStat;
+
+ public STATPROPSTG Current => propStat;
+
+ object IEnumerator.Current => propStat;
+
+ public unsafe StatPropStgEnumerator(IPropertyStorage propertyStorage)
+ {
+ propertyStorage.Enum(out enumerator);
+ }
+
+ public unsafe void Dispose()
+ {
+ FreeName();
+
+ Marshal.ReleaseComObject(enumerator);
+ }
+
+ private unsafe void FreeName()
+ {
+ Marshal.FreeCoTaskMem((nint)propStat.lpwstrName.Value);
+ propStat.lpwstrName = null;
+ }
+
+ public unsafe bool MoveNext()
+ {
+ FreeName();
+
+ fixed (STATPROPSTG* statPtr = &propStat)
+ {
+ uint fetched;
+ enumerator.Next(1, statPtr, &fetched);
+ return fetched > 0;
+ }
+ }
+
+ public void Reset()
+ {
+ FreeName();
+
+ enumerator.Reset();
+ }
+}
+
+///
+/// Creates an enumerator for STATPROPSTG elements from a PropertyStorage.
+///
+internal sealed class StatPropStgCollection : IEnumerable
+{
+ readonly IPropertyStorage propertyStorage;
+
+ public StatPropStgCollection(IPropertyStorage propertyStorage)
+ {
+ this.propertyStorage = propertyStorage;
+ }
+
+ public IEnumerator GetEnumerator() => new StatPropStgEnumerator(propertyStorage);
+
+ IEnumerator IEnumerable.GetEnumerator() => new StatPropStgEnumerator(propertyStorage);
+}
+
+///
+/// Wraps IPropertyStorage.
+///
+public sealed class PropertyStorage : IDisposable
+{
+ private readonly IPropertyStorage propertyStorage;
+ private bool disposed;
+
+ internal unsafe PropertyStorage(IPropertyStorage propertyStorage)
+ {
+ this.propertyStorage = propertyStorage;
+ StatPropStgCollection = new(propertyStorage);
+
+ STATPROPSETSTG prop;
+ this.propertyStorage.Stat(&prop);
+ }
+
+ #region IDisposable Members
+
+ public void Dispose()
+ {
+ if (disposed)
+ return;
+
+ int count = Marshal.ReleaseComObject(propertyStorage);
+ Debug.Assert(count == 0);
+
+ disposed = true;
+ }
+
+ #endregion
+
+ internal StatPropStgCollection StatPropStgCollection { get; }
+
+ public void Flush(CommitFlags flags = CommitFlags.Default) => propertyStorage.Commit((uint)flags);
+
+ public unsafe void Remove(int propertyID)
+ {
+ PROPSPEC propspec = new()
+ {
+ ulKind = PROPSPEC_KIND.PRSPEC_PROPID,
+ Anonymous = new PROPSPEC._Anonymous_e__Union()
+ {
+ propid = (uint)propertyID,
+ },
+ };
+ propertyStorage.DeleteMultiple(1, &propspec);
+ }
+
+ public void Revert() => propertyStorage.Revert();
+
+ public unsafe object? this[int propertyID]
+ {
+ get
+ {
+ PROPSPEC spec = PropVariantExtensions.CreatePropSpec(PROPSPEC_KIND.PRSPEC_PROPID, propertyID);
+
+ var variants = new PROPVARIANT[1];
+ propertyStorage.ReadMultiple(1, &spec, variants);
+ HRESULT hr = PInvoke.PropVariantToVariant(variants[0], out object variant);
+ hr.ThrowOnFailure();
+ return variant;
+ }
+
+ set
+ {
+ PROPSPEC spec = PropVariantExtensions.CreatePropSpec(PROPSPEC_KIND.PRSPEC_PROPID, propertyID);
+
+ HRESULT hr = PInvoke.VariantToPropVariant(value, out PROPVARIANT pv);
+ hr.ThrowOnFailure();
+
+ PROPVARIANT[] pvs = [pv];
+ propertyStorage.WriteMultiple(1, &spec, pvs, 2);
+ }
+ }
+}
+
+static class PropVariantExtensions
+{
+ public static PROPSPEC CreatePropSpec(PROPSPEC_KIND kind, int propertyID)
+ {
+ return new PROPSPEC
+ {
+ ulKind = kind,
+ Anonymous = new PROPSPEC._Anonymous_e__Union
+ {
+ propid = (uint)propertyID,
+ },
+ };
+ }
+}
diff --git a/StructuredStorage/Storage.cs b/StructuredStorage/Storage.cs
new file mode 100644
index 00000000..13a21832
--- /dev/null
+++ b/StructuredStorage/Storage.cs
@@ -0,0 +1,343 @@
+using System.Collections;
+using System.Diagnostics;
+using System.Runtime.InteropServices;
+using Windows.Win32;
+using Windows.Win32.Foundation;
+using Windows.Win32.Security;
+using Windows.Win32.System.Com;
+using Windows.Win32.System.Com.StructuredStorage;
+
+namespace StructuredStorage;
+
+#pragma warning disable CA1069 // Enums values should not be duplicated
+#pragma warning disable CA1724 // Type names should not match namespaces
+#pragma warning disable CA1028 // Enum storage should be Int32
+#pragma warning disable CA1008 // Enums should have zero value
+
+///
+/// STGC constants.
+///
+[Flags]
+public enum CommitFlags : uint
+{
+ Default = STGC.STGC_DEFAULT,
+ Overwrite = STGC.STGC_OVERWRITE,
+ OnlyIfCurrent = STGC.STGC_ONLYIFCURRENT,
+ DangerouslyCommitMerelyToDiskCache = STGC.STGC_DANGEROUSLYCOMMITMERELYTODISKCACHE,
+ Consolidate = STGC.STGC_CONSOLIDATE,
+}
+
+///
+/// STGM constants.
+///
+[Flags]
+public enum StorageModes : uint
+{
+ FailIfThere = STGM.STGM_FAILIFTHERE,
+ Direct = STGM.STGM_DIRECT,
+ AccessRead = STGM.STGM_READ,
+ AccessWrite = STGM.STGM_WRITE,
+ AccessReadWrite = STGM.STGM_READWRITE,
+ ShareExclusive = STGM.STGM_SHARE_EXCLUSIVE,
+ ShareDenyWrite = STGM.STGM_SHARE_DENY_WRITE,
+ ShareDenyRead = STGM.STGM_SHARE_DENY_READ,
+ ShareDenyNone = STGM.STGM_SHARE_DENY_NONE,
+ Create = STGM.STGM_CREATE,
+ Transacted = STGM.STGM_TRANSACTED,
+ Convert = STGM.STGM_CONVERT,
+ Priority = STGM.STGM_PRIORITY,
+ NoScratch = STGM.STGM_NOSCRATCH,
+ NoSnapShot = STGM.STGM_NOSNAPSHOT,
+ DirectSWMR = STGM.STGM_DIRECT_SWMR,
+ DeleteOnRelease = STGM.STGM_DELETEONRELEASE,
+ ModeSimple = STGM.STGM_SIMPLE,
+}
+
+///
+/// Enumerates STATSTG elements from a Storage.
+///
+internal sealed class StatStgEnumerator : IEnumerator
+{
+ readonly IEnumSTATSTG enumerator;
+ STATSTG stat;
+
+ public STATSTG Current => stat;
+
+ object IEnumerator.Current => stat;
+
+ public unsafe StatStgEnumerator(IStorage storage)
+ {
+ storage.EnumElements(0, null, 0, out enumerator);
+ }
+
+ public unsafe void Dispose()
+ {
+ FreeName();
+
+ Marshal.ReleaseComObject(enumerator);
+ }
+
+ private unsafe void FreeName()
+ {
+ Marshal.FreeCoTaskMem((nint)stat.pwcsName.Value);
+ stat.pwcsName = null;
+ }
+
+ public unsafe bool MoveNext()
+ {
+ FreeName();
+
+ fixed (STATSTG* statPtr = &stat)
+ {
+ uint fetched;
+ enumerator.Next(1, statPtr, &fetched);
+ return fetched > 0;
+ }
+ }
+
+ public void Reset()
+ {
+ FreeName();
+
+ enumerator.Reset();
+ }
+}
+
+///
+/// Creates an enumerator for STATSTG elements from a Storage.
+///
+internal sealed class StatStgCollection : IEnumerable
+{
+ readonly IStorage storage;
+
+ public StatStgCollection(IStorage storage)
+ {
+ this.storage = storage;
+ }
+
+ public IEnumerator GetEnumerator() => new StatStgEnumerator(storage);
+
+ IEnumerator IEnumerable.GetEnumerator() => new StatStgEnumerator(storage);
+}
+
+///
+/// Wraps a COM structured storage object.
+///
+public sealed class Storage : IDisposable
+{
+ static readonly Guid IStorageGuid = typeof(IStorage).GUID;
+
+ static STGOPTIONS DefaultOptions => new()
+ {
+ usVersion = 2,
+ reserved = 0,
+ ulSectorSize = 4096,
+ pwcsTemplateFile = null,
+ };
+
+ readonly IStorage storage;
+ readonly LockBytes? lockBytes; // Prevents garbage collection of in-memory storage
+
+ public Storage? Parent { get; }
+
+ public PropertySetStorage PropertySetStorage { get; }
+
+ internal StatStgCollection StatStgCollection { get; }
+
+ bool disposed;
+
+ // Methods
+ internal Storage(IStorage storage, Storage? parent = null, LockBytes? lockBytes = null)
+ {
+ this.storage = storage;
+ Parent = parent;
+ this.lockBytes = lockBytes;
+ PropertySetStorage = new(storage);
+ StatStgCollection = new StatStgCollection(storage);
+ }
+
+ public static unsafe Storage Create(string fileName, StorageModes modes = StorageModes.ShareExclusive | StorageModes.AccessReadWrite)
+ {
+ STGOPTIONS opts = DefaultOptions;
+ HRESULT hr = PInvoke.StgCreateStorageEx(fileName, (STGM)modes, STGFMT.STGFMT_DOCFILE, 0, &opts, (PSECURITY_DESCRIPTOR)null, IStorageGuid, out void* ptr);
+ hr.ThrowOnFailure();
+
+ var iStorage = (IStorage)Marshal.GetObjectForIUnknown((nint)ptr);
+ Marshal.Release((nint)ptr);
+ return new(iStorage);
+ }
+
+ public static Storage CreateInMemory(int capacity)
+ {
+ LockBytes lockBytes = new(capacity);
+ HRESULT hr = PInvoke.StgCreateDocfileOnILockBytes(lockBytes.ILockBytes, STGM.STGM_READWRITE | STGM.STGM_SHARE_EXCLUSIVE | STGM.STGM_CREATE, 0, out IStorage storage);
+ hr.ThrowOnFailure();
+ return new(storage, null, lockBytes);
+ }
+
+ public static unsafe Storage Open(string fileName, StorageModes modes = StorageModes.ShareExclusive | StorageModes.AccessReadWrite)
+ {
+ STGOPTIONS opts = DefaultOptions;
+ HRESULT hr = PInvoke.StgOpenStorageEx(fileName, (STGM)modes, STGFMT.STGFMT_DOCFILE, 0, &opts, (PSECURITY_DESCRIPTOR)null, IStorageGuid, out void* ptr);
+ if (hr == HRESULT.STG_E_FILENOTFOUND)
+ throw new FileNotFoundException(null, fileName);
+ if (hr == HRESULT.STG_E_FILEALREADYEXISTS)
+ hr = HRESULT.STG_E_DOCFILECORRUPT;
+ hr.ThrowOnFailure();
+
+ var iStorage = (IStorage)Marshal.GetObjectForIUnknown((nint)ptr);
+ Marshal.Release((nint)ptr);
+ return new(iStorage);
+ }
+
+ #region IDisposable Members
+
+ public void Dispose()
+ {
+ if (disposed)
+ return;
+
+ int count = Marshal.ReleaseComObject(storage);
+ Debug.Assert(count == 0);
+
+ lockBytes?.Dispose();
+
+ disposed = true;
+ }
+
+ #endregion
+
+ public unsafe Storage CreateStorage(string name, StorageModes flags = StorageModes.Create | StorageModes.ShareExclusive | StorageModes.AccessReadWrite)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (char* namePtr = name)
+ {
+ storage.CreateStorage(namePtr, (STGM)flags, 0, 0, out IStorage childStorage);
+ return new Storage(childStorage, this);
+ }
+ }
+
+ public unsafe Stream CreateStream(string name, StorageModes flags = StorageModes.Create | StorageModes.ShareExclusive | StorageModes.AccessReadWrite)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (char* namePtr = name)
+ {
+ storage.CreateStream(namePtr, (STGM)flags, 0, 0, out IStream stm);
+ return new Stream(stm, this);
+ }
+ }
+
+ internal StatStgEnumerator CreateStatStgEnumerator() => new(storage);
+
+ public void DestroyElement(string name)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ storage.DestroyElement(name);
+ }
+
+ public void DestroyElementIfExists(string name)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ if (ContainsElement(name))
+ storage.DestroyElement(name);
+ }
+
+ public bool ContainsElement(string name)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ return StatStgCollection.Any(s => s.pwcsName.AsSpan().SequenceEqual(name));
+ }
+
+ public bool ContainsStream(string name)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ return StatStgCollection.Any(s => (STGTY)s.type == STGTY.STGTY_STREAM && s.pwcsName.AsSpan().SequenceEqual(name));
+ }
+
+ public void Commit(CommitFlags flags = CommitFlags.Default)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ storage.Commit((uint)flags);
+ }
+
+ public void MoveElement(string name, Storage destination) => MoveElement(name, destination, name);
+
+ public void MoveElement(string name, Storage destination, string newName)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ storage.MoveElementTo(name, destination.storage, newName, 0);
+ }
+
+ public unsafe Storage OpenStorage(string name, StorageModes flags = StorageModes.AccessReadWrite | StorageModes.ShareExclusive)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (char* namePtr = name)
+ {
+ storage.OpenStorage(namePtr, null, (STGM)flags, null, 0, out IStorage childStorage);
+ return new Storage(childStorage, this);
+ }
+ }
+
+ public unsafe Stream OpenStream(string name, StorageModes flags = StorageModes.AccessReadWrite | StorageModes.ShareExclusive)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (char* namePtr = name)
+ {
+ storage.OpenStream(namePtr, null, (STGM)flags, 0, out IStream iStream);
+ return new Stream(iStream, this);
+ }
+ }
+
+ public Stream OpenOrCreateStream(string name, StorageModes flags = StorageModes.AccessReadWrite | StorageModes.ShareExclusive)
+ => ContainsStream(name) ? OpenStream(name, flags) : CreateStream(name, flags);
+
+ public void Revert()
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ storage.Revert();
+ }
+
+ public unsafe void SwitchToFile(string fileName)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (char* fileNamePtr = fileName)
+ {
+ if (storage is not IRootStorage rootStorage)
+ throw new InvalidOperationException("Not file storage");
+ rootStorage.SwitchToFile(fileNamePtr);
+ }
+ }
+
+ // Properties
+ public Guid Id
+ {
+ get
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ HRESULT hr = PInvoke.ReadClassStg(storage, out Guid guid);
+ hr.ThrowOnFailure();
+ return guid;
+ }
+
+ set
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ HRESULT hr = PInvoke.WriteClassStg(storage, value);
+ hr.ThrowOnFailure();
+ }
+ }
+}
diff --git a/StructuredStorage/Stream.cs b/StructuredStorage/Stream.cs
new file mode 100644
index 00000000..f49a54dc
--- /dev/null
+++ b/StructuredStorage/Stream.cs
@@ -0,0 +1,185 @@
+using System.Diagnostics;
+using System.Runtime.InteropServices;
+using Windows.Win32;
+using Windows.Win32.Foundation;
+using Windows.Win32.System.Com;
+
+namespace StructuredStorage;
+
+///
+/// Implements Stream on an COM IStream.
+///
+public sealed class Stream : System.IO.Stream
+{
+ public Storage Parent { get; }
+
+ readonly IStream stream;
+ bool disposed;
+
+ internal Stream(IStream stream, Storage parent)
+ {
+ this.stream = stream;
+ Parent = parent;
+
+ STGM mode = Stat.grfMode;
+ CanRead = mode.HasFlag(STGM.STGM_READWRITE) || !mode.HasFlag(STGM.STGM_WRITE);
+ CanWrite = mode.HasFlag(STGM.STGM_READWRITE) || mode.HasFlag(STGM.STGM_WRITE);
+ }
+
+ protected override void Dispose(bool disposing)
+ {
+ if (disposed)
+ return;
+
+ if (disposing)
+ {
+ Flush();
+
+ int count = Marshal.ReleaseComObject(stream);
+ Debug.Assert(count == 0);
+ }
+
+ disposed = true;
+
+ base.Dispose(disposing);
+ }
+
+ public override void Flush() => Flush(CommitFlags.Default);
+
+ public void Flush(CommitFlags flags)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ stream.Commit((STGC)flags);
+ }
+
+ public override int Read(byte[] buffer, int offset, int count)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ Span slice = buffer.AsSpan(offset, count);
+ return Read(slice);
+ }
+
+ public override unsafe int Read(Span buffer)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (byte* ptr = buffer)
+ {
+ uint read;
+ HRESULT hr = stream.Read(ptr, (uint)buffer.Length, &read);
+ hr.ThrowOnFailure();
+ return (int)read;
+ }
+ }
+
+ public void Revert()
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ stream.Revert();
+ }
+
+ public override unsafe long Seek(long offset, SeekOrigin origin)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ ulong pos;
+ stream.Seek(offset, origin, &pos);
+ return (long)pos;
+ }
+
+ public override void SetLength(long value)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ stream.SetSize((ulong)value);
+ }
+
+ public override void Write(byte[] buffer, int offset, int count)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ ReadOnlySpan slice = buffer.AsSpan(offset, count);
+ Write(slice);
+ }
+
+ public override unsafe void Write(ReadOnlySpan buffer)
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ fixed (byte* ptr = buffer)
+ {
+ uint written;
+ HRESULT result = stream.Write(ptr, (uint)buffer.Length, &written);
+ result.ThrowOnFailure();
+ }
+ }
+
+ // Properties
+ public override bool CanRead { get; }
+
+ public override bool CanSeek => true;
+
+ public override bool CanWrite { get; }
+
+ public override long Length
+ {
+ get
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ return (long)Stat.cbSize;
+ }
+ }
+
+ public override unsafe long Position
+ {
+ get
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ ulong pos;
+ stream.Seek(0L, SeekOrigin.Current, &pos);
+ return (long)pos;
+ }
+
+ set
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ stream.Seek(value, SeekOrigin.Begin, null);
+ }
+ }
+
+ public Guid Id
+ {
+ get
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ HRESULT hr = PInvoke.ReadClassStm(stream, out Guid guid);
+ hr.ThrowOnFailure();
+ return guid;
+ }
+
+ set
+ {
+ ObjectDisposedException.ThrowIf(disposed, this);
+
+ int hr = PInvoke.WriteClassStm(stream, value);
+ Marshal.ThrowExceptionForHR(hr);
+ }
+ }
+
+ internal unsafe STATSTG Stat
+ {
+ get
+ {
+ STATSTG stat;
+ stream.Stat(&stat, STATFLAG.STATFLAG_NONAME);
+ return stat;
+ }
+ }
+}
diff --git a/StructuredStorage/StructuredStorage.csproj b/StructuredStorage/StructuredStorage.csproj
new file mode 100644
index 00000000..40393c66
--- /dev/null
+++ b/StructuredStorage/StructuredStorage.csproj
@@ -0,0 +1,18 @@
+
+
+
+ net8.0-windows
+ enable
+ enable
+ 12.0
+ true
+
+
+
+
+ all
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+
+
+
+