public unsafe AudioMetadata ReadMetadata(Stream stream) { OggStream oggStream = null; SafeNativeMethods.VorbisCommentInit(out var vorbisComment); #if NETSTANDARD2_0 var buffer = ArrayPool <byte> .Shared.Rent(4096); #else Span <byte> buffer = stackalloc byte[4096]; #endif try { using (var sync = new OggSync()) using (var decoder = new VorbisDecoder()) { OggPage page; do { // Read from the buffer into a page while (!sync.PageOut(out page)) { #if NETSTANDARD2_0 var bytesRead = stream.Read(buffer, 0, buffer.Length); #else var bytesRead = stream.Read(buffer); #endif if (bytesRead == 0) { throw new AudioInvalidException("No Ogg stream was found."); } var nativeBuffer = new Span <byte>(sync.Buffer(bytesRead).ToPointer(), bytesRead); #if NETSTANDARD2_0 buffer.AsSpan().Slice(0, bytesRead).CopyTo(nativeBuffer); #else buffer.Slice(0, bytesRead).CopyTo(nativeBuffer); #endif sync.Wrote(bytesRead); } if (oggStream == null) { oggStream = new OggStream(SafeNativeMethods.OggPageSerialNo(page)); } oggStream.PageIn(page); while (oggStream.PacketOut(out var packet)) { decoder.HeaderIn(vorbisComment, packet); if (packet.PacketNumber == 1) { return(new VorbisCommentToMetadataAdapter(vorbisComment)); } } } while (!SafeNativeMethods.OggPageEos(page)); throw new AudioInvalidException("The end of the Ogg stream was reached without finding a header."); } } finally { #if NETSTANDARD2_0 ArrayPool <byte> .Shared.Return(buffer); #endif SafeNativeMethods.VorbisCommentClear(ref vorbisComment); oggStream?.Dispose(); } }
public unsafe AudioInfo ReadAudioInfo(Stream stream) { OggStream?oggStream = null; SafeNativeMethods.VorbisCommentInit(out var vorbisComment); #if NETSTANDARD2_0 var buffer = ArrayPool <byte> .Shared.Rent(4096); #else Span <byte> buffer = stackalloc byte[4096]; #endif try { using (var sync = new OggSync()) using (var decoder = new VorbisDecoder()) { OggPage page; do { // Read from the buffer into a page while (!sync.PageOut(out page)) { #if NETSTANDARD2_0 var bytesRead = stream.Read(buffer, 0, buffer.Length); #else var bytesRead = stream.Read(buffer); #endif if (bytesRead == 0) { throw new AudioInvalidException("No Ogg stream was found."); } var nativeBuffer = new Span <byte>(sync.Buffer(bytesRead).ToPointer(), bytesRead); #if NETSTANDARD2_0 buffer.AsSpan().Slice(0, bytesRead).CopyTo(nativeBuffer); #else buffer.Slice(0, bytesRead).CopyTo(nativeBuffer); #endif sync.Wrote(bytesRead); } oggStream ??= new(SafeNativeMethods.OggPageSerialNo(page)); oggStream.PageIn(page); while (oggStream.PacketOut(out var packet)) { if (!SafeNativeMethods.VorbisSynthesisIdHeader(packet)) { throw new AudioUnsupportedException("Not a Vorbis stream."); } decoder.HeaderIn(vorbisComment, packet); var info = decoder.GetInfo(); return(AudioInfo.CreateForLossy( "Vorbis", info.Channels, #if WINDOWS info.Rate, GetFinalGranulePosition(oggStream.SerialNumber, stream), Math.Max(info.BitRateNominal, 0))); #else (int)info.Rate, GetFinalGranulePosition((int)oggStream.SerialNumber, stream), Math.Max((int)info.BitRateNominal, 0)); #endif } } while (!SafeNativeMethods.OggPageEos(page)); throw new AudioInvalidException("The end of the Ogg stream was reached without finding a header."); } } finally { #if NETSTANDARD2_0 ArrayPool <byte> .Shared.Return(buffer); #endif SafeNativeMethods.VorbisCommentClear(ref vorbisComment); oggStream?.Dispose(); } }
public unsafe void WriteMetadata(Stream stream, AudioMetadata metadata, SettingDictionary settings) { using (var tempStream = new TempFileStream()) { OggStream inputOggStream = null; OggStream outputOggStream = null; #if NETSTANDARD2_0 var buffer = ArrayPool <byte> .Shared.Rent(4096); #else Span <byte> buffer = stackalloc byte[4096]; #endif try { using (var sync = new OggSync()) { var headerWritten = false; OggPage page; var pagesWritten = 0u; do { // Read from the buffer into a page while (!sync.PageOut(out page)) { #if NETSTANDARD2_0 var bytesRead = stream.Read(buffer, 0, buffer.Length); #else var bytesRead = stream.Read(buffer); #endif if (bytesRead == 0) { throw new AudioInvalidException("No Ogg stream was found."); } var nativeBuffer = new Span <byte>(sync.Buffer(bytesRead).ToPointer(), bytesRead); #if NETSTANDARD2_0 buffer.AsSpan().Slice(0, bytesRead).CopyTo(nativeBuffer); #else buffer.Slice(0, bytesRead).CopyTo(nativeBuffer); #endif sync.Wrote(bytesRead); } if (inputOggStream == null) { inputOggStream = new OggStream(SafeNativeMethods.OggPageSerialNo(page)); } if (outputOggStream == null) { outputOggStream = new OggStream(inputOggStream.SerialNumber); } // Write new header page(s) using a modified comment packet if (!headerWritten) { inputOggStream.PageIn(page); while (inputOggStream.PacketOut(out var packet)) { switch (packet.PacketNumber) { case 0: outputOggStream.PacketIn(packet); break; // Substitute the new comment packet case 1: using (var adapter = new MetadataToVorbisCommentAdapter(metadata)) { adapter.HeaderOut(out var commentPacket); outputOggStream.PacketIn(commentPacket); } break; // Flush at the end of the header case 2: outputOggStream.PacketIn(packet); while (outputOggStream.Flush(out var outPage)) { WritePage(outPage, tempStream); pagesWritten++; } headerWritten = true; break; default: throw new AudioInvalidException("Missing header packet."); } } } else { // Copy the existing data pages verbatim, with updated sequence numbers UpdateSequenceNumber(ref page, pagesWritten); WritePage(page, tempStream); pagesWritten++; } } while (!SafeNativeMethods.OggPageEos(page)); // Once the end of the input is reached, overwrite the original file and return stream.Position = 0; stream.SetLength(tempStream.Length); tempStream.Position = 0; tempStream.CopyTo(stream); } } finally { #if NETSTANDARD2_0 ArrayPool <byte> .Shared.Return(buffer); #endif inputOggStream?.Dispose(); outputOggStream?.Dispose(); } } }