diff --git a/Concentus.Oggfile/Concentus.Oggfile.csproj b/Concentus.Oggfile/Concentus.Oggfile.csproj index f114cf0..a9cc75c 100644 --- a/Concentus.Oggfile/Concentus.Oggfile.csproj +++ b/Concentus.Oggfile/Concentus.Oggfile.csproj @@ -9,7 +9,7 @@ Properties Concentus.Oggfile Concentus.Oggfile - v4.6 + v4.8 512 true diff --git a/Concentus/Concentus.csproj b/Concentus/Concentus.csproj index b0dc74a..b196b31 100644 --- a/Concentus/Concentus.csproj +++ b/Concentus/Concentus.csproj @@ -9,7 +9,7 @@ Properties Concentus Concentus - v4.6 + v4.8 512 true diff --git a/Linux/TeddyBench b/Linux/TeddyBench new file mode 100755 index 0000000..360440e --- /dev/null +++ b/Linux/TeddyBench @@ -0,0 +1,22 @@ +#!/bin/sh + +# check for mono +if ! type "mono" > /dev/null +then + echo "mono could not be found. Please read the README.md for installation instructions." + exit +fi + +# check for ffmpeg +if ! type "ffmpeg" > /dev/null +then + echo "ffmpeg could not be found. Please read the README.md for installation instructions." + exit +fi + +# Some exports to avoid crashes +export MONO_MANAGED_WATCHER=disabled +export MONO_WINFORMS_XIM_STYLE=disabled + +# Start TeddyBench +mono TeddyBench.exe diff --git a/README.md b/README.md index e29975a..ac8f6bb 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,44 @@ # Tonie file tool With this tool you can dump existing files of the famous audio box or create custom ones. + +# Linux +WARNING: The `TeddyBench` Linux port has alpha status. So don't use it unless you are a developer and you are able to understand cryptic error messages. It is not fully tested but converting files should work. + +It is tested with Ubuntu 20.04. + +## Requirements +Please install the following packages + +``` +# sudo apt install mono-complete xcb ffmpeg libgdiplus libcanberra-gtk-module libcanberra-gtk3-module +``` + +## Running + +### General + +You have to run `TeddyBench` with `mono`. Using `wine` is not working. + +``` +# mono TeddyBench.exe +``` + +### Know issues +For some reasons TonieBench is running unstable with mono. So you can try the following steps. + +**A. Running with sudo (root)** + +``` +# sudo mono TeddyBench.exe +``` + +**B. Set some environment variables** + +``` +# export MONO_MANAGED_WATCHER=disabled +# export MONO_WINFORMS_XIM_STYLE=disabled +# mono TeddyBench.exe +``` +## Development +You can compile it directly under Linux with `monodevelop`. diff --git a/Teddy/App.config b/Teddy/App.config index 2d2a12d..4bfa005 100644 --- a/Teddy/App.config +++ b/Teddy/App.config @@ -1,6 +1,6 @@ - + diff --git a/Teddy/Teddy.csproj b/Teddy/Teddy.csproj index 0b8cdee..34e0f08 100644 --- a/Teddy/Teddy.csproj +++ b/Teddy/Teddy.csproj @@ -10,7 +10,7 @@ Exe Teddy Teddy - v4.6 + v4.8 512 true true diff --git a/TeddyBench/MainForm.cs b/TeddyBench/MainForm.cs index 6702425..a9df5c8 100644 --- a/TeddyBench/MainForm.cs +++ b/TeddyBench/MainForm.cs @@ -15,6 +15,7 @@ using System.Linq; using System.Net; using System.Net.Http; +using System.Runtime.InteropServices; using System.Text; using System.Text.RegularExpressions; using System.Threading; @@ -871,7 +872,17 @@ private void btnAdd_Click(object sender, EventArgs e) if (dlg.ShowDialog() == DialogResult.OK) { - AddFiles(dlg.FileNames); + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + { + // Sort in alphabetical order because the Windows.Form dialog under Linux is scrappy + var list = dlg.FileNames.ToList(); + list.Sort(); + AddFiles(list.ToArray()); + } + else + { + AddFiles(dlg.FileNames); + } } } @@ -881,13 +892,19 @@ private void AddFiles(string[] fileNames) if (ask.ShowDialog() == DialogResult.OK) { + string[] extensions; + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + extensions = new String[] {".mp3", ".ogg", ".wav", "wma", "aac"}; + else + extensions = new String[] {".mp3"}; + if (fileNames.Count() == 1) { string fileName = fileNames[0]; - if (fileName.ToLower().EndsWith(".mp3")) + if (extensions.Any(ext => fileName.ToLower().EndsWith(ext))) { - switch (MessageBox.Show("You are about to encode a single MP3, is this right?", "Encode a MP3 file", MessageBoxButtons.YesNo)) + switch (MessageBox.Show("You are about to encode an single audio file, is this right?", "Encode an audio file", MessageBoxButtons.YesNo)) { case DialogResult.No: return; @@ -916,9 +933,9 @@ private void AddFiles(string[] fileNames) } else { - if (fileNames.Where(f => !f.ToLower().EndsWith(".mp3")).Count() > 0) + if (fileNames.Where(f => !extensions.Any(ext => f.ToLower().EndsWith(ext))).Count() > 0) { - MessageBox.Show("Please select MP3 files only.", "Add file..."); + MessageBox.Show("Please select supported audio files only.", "Add file..."); return; } diff --git a/TeddyBench/TeddyBench.csproj b/TeddyBench/TeddyBench.csproj index e5807a1..937e291 100644 --- a/TeddyBench/TeddyBench.csproj +++ b/TeddyBench/TeddyBench.csproj @@ -1,5 +1,5 @@  - + diff --git a/TonieAudio/TonieAudio.cs b/TonieAudio/TonieAudio.cs index f565bf3..4f3d071 100644 --- a/TonieAudio/TonieAudio.cs +++ b/TonieAudio/TonieAudio.cs @@ -41,6 +41,9 @@ A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER using System.Runtime.InteropServices; using System.Security.Cryptography; using System.Text; +using System.Threading; +using System.Threading.Tasks; +using Xabe.FFmpeg; using static TonieFile.ProtoCoder; namespace TonieFile @@ -242,7 +245,11 @@ private void BuildFileList(string[] sources) { foreach (var source in sources) { - string item = source.Trim('"').Trim(Path.DirectorySeparatorChar); + string item; + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + item = source; + else + item = source.Trim('"').Trim(Path.DirectorySeparatorChar); if (Directory.Exists(item)) { @@ -281,9 +288,15 @@ private void BuildFileList(string[] sources) } else if (File.Exists(item)) { - if (!item.ToLower().EndsWith(".mp3")) + string[] extensions; + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + extensions = new String[] { ".mp3", ".ogg", ".wav", "wma", "aac" }; + else + extensions = new String[] { ".mp3" }; + + if (!extensions.Any(ext => item.ToLower().EndsWith(ext))) { - throw new InvalidDataException("Specified item '" + item + "' is no MP3"); + throw new InvalidDataException("Specified item '" + item + "' is no a supported audio file"); } FileList.Add(item); } @@ -439,11 +452,97 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, try { - var prefixStream = new Mp3FileReader(prefixFile); - var prefixResampled = new MediaFoundationResampler(prefixStream, outFormat); + Stream prefixResampled = new MemoryStream(); + + // Linux + string prefixTmpWavFile = ""; + + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + { + prefixTmpWavFile = Path.ChangeExtension(Path.GetTempFileName(), "wav"); + + //Console.Write(" Convert file " + sourceFile + " to WAV. Temp file " + prefixTmpWavFile); + + // The isReady flag and error test variables are not a nice solution but is working + bool isReadyPrefix = false; + String FFmpegErrorTextPrefix = ""; + + // Convert to WAV + Task.Run(async () => + { + try + { + // Convert to WAV and resample + IMediaInfo prefixInputFile = await FFmpeg.GetMediaInfo(sourceFile); + + IAudioStream prefixAudioStream = prefixInputFile.AudioStreams.First() + .SetCodec(AudioCodec.pcm_f32le) + .SetChannels(2) + .SetSampleRate(samplingRate); + + IConversionResult prefixConversionResult = await FFmpeg.Conversions.New() + .AddStream(prefixAudioStream) + .SetOutput(prefixTmpWavFile) + .Start(); + } + catch (Exception e) + { + FFmpegErrorTextPrefix = e.Message; + throw new Exception("FFmepg error " + e.Message + "\n"); + } + + isReadyPrefix = true; + }); + + while (!isReadyPrefix) + { + Thread.Sleep(200); + if (FFmpegErrorTextPrefix != "") + throw new Exception("FFmepg error: " + FFmpegErrorTextPrefix + "\n"); + } + + // Read WAV file + byte[] prefixWavBytes = File.ReadAllBytes(prefixTmpWavFile); + prefixResampled.Write(prefixWavBytes, 0, prefixWavBytes.Length); + prefixResampled.Seek(0, SeekOrigin.Begin); + + // Skip WAV header + byte[] bytes = new byte[4]; + prefixResampled.Seek(16, 0); + prefixResampled.Read(bytes, 0, 4); + int Subchunk1Size = BitConverter.ToInt32(bytes, 0); // Get FMT size + + // Skip some header information + prefixResampled.Read(buffer, 0, Subchunk1Size + 12); // Data starts at FMT size + 12 bytes + + // Read data chunk + prefixResampled.Read(bytes, 0, 4); + var str = System.Text.Encoding.Default.GetString(bytes); + if (str != "data") + throw new Exception("WAV error: Data section not found \n"); + + // Skip data length + prefixResampled.Read(bytes, 0, 4); + } + else + { + var prefixStream = new Mp3FileReader(prefixFile); + var prefixResampledTmp = new MediaFoundationResampler(prefixStream, outFormat); + + // Convert NAudioStream to System.IO.Stream + byte[] sampleByte = { 0 }; + int read; + while ((read = prefixResampledTmp.Read(sampleByte, 0, sampleByte.Length)) > 0) + { + prefixResampled.Write(sampleByte, 0, read); + } + + prefixResampled.Seek(0, SeekOrigin.Begin); + } while (true) { + bytesReturned = prefixResampled.Read(buffer, 0, buffer.Length); if (bytesReturned <= 0) @@ -454,7 +553,11 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, bool isEmpty = (buffer.Where(v => v != 0).Count() == 0); if (!isEmpty) { - float[] sampleBuffer = ConvertToFloat(buffer, bytesReturned, channels); + float[] sampleBuffer; + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + sampleBuffer = MemoryMarshal.Cast(buffer.AsSpan()).ToArray(); + else + sampleBuffer = ConvertToFloat(buffer, bytesReturned, channels); if ((outputData.Length + 0x1000 + sampleBuffer.Length) >= maxSize) { @@ -464,6 +567,9 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, } lastIndex = (uint)oggOut.PageCounter; } + + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + File.Delete(prefixTmpWavFile); } catch (Exception ex) { @@ -471,9 +577,95 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, } } + Stream streamResampled = new MemoryStream(); + + // Linux + string tmpWavFile = ""; + /* then the real audio file */ - var stream = new Mp3FileReader(sourceFile); - var streamResampled = new MediaFoundationResampler(stream, outFormat); + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + { + tmpWavFile = Path.ChangeExtension(Path.GetTempFileName(), "wav"); + + //Console.Write(" Convert file " + sourceFile + " to WAV. Temp file " + TmpWavFile); + + // The isReady flag and error test variables are not a nice solution but is working + bool isReady = false; + String FFmpegErrorText = ""; + + // Convert to WAV and resample + Task.Run(async () => + { + try + { + IMediaInfo inputFile = await FFmpeg.GetMediaInfo(sourceFile); + + IAudioStream audioStream = inputFile.AudioStreams.First() + .SetCodec(AudioCodec.pcm_f32le) + .SetChannels(2) + .SetSampleRate(samplingRate); + + IConversionResult conversionResult = await FFmpeg.Conversions.New() + .AddStream(audioStream) + .SetOutput(tmpWavFile) + .AddParameter("-map_metadata -1 -fflags +bitexact -flags:v +bitexact -flags:a +bitexact") // Remove meta data + .Start(); + } + catch (Exception e) + { + FFmpegErrorText = e.Message; + Console.WriteLine(e.Message); + throw new Exception("FFmepg error " + e.Message + "\n"); + } + + isReady = true; + }); + + while (!isReady) + { + Thread.Sleep(200); + if (FFmpegErrorText != "") + throw new Exception("FFmepg error: " + FFmpegErrorText + "\n"); + } + + // Read WAV file + byte[] wavBytes = File.ReadAllBytes(tmpWavFile); + streamResampled.Write(wavBytes, 0, wavBytes.Length); + streamResampled.Seek(0, SeekOrigin.Begin); + + // Skip WAV header + byte[] bytes = new byte[4]; + streamResampled.Seek(16, 0); + streamResampled.Read(bytes, 0, 4); + int Subchunk1Size = BitConverter.ToInt32(bytes, 0); // Get FMT size + + // Skip some header information + streamResampled.Read(buffer, 0, Subchunk1Size + 12); // Data starts at FMT size + 12 bytes + + // Read data chunk + streamResampled.Read(bytes, 0, 4); + var str = System.Text.Encoding.Default.GetString(bytes); + if(str != "data") + throw new Exception("WAV error: Data section not found \n"); + + // Skip data length + streamResampled.Read(bytes, 0, 4); + } + else + { + var stream = new Mp3FileReader(sourceFile); + var streamResampledTmp = new MediaFoundationResampler(stream, outFormat); + + // Convert NAudioStream to System.IO.Stream + byte[] sampleByte = { 0 }; + int read; + while ((read = streamResampledTmp.Read(sampleByte, 0, sampleByte.Length)) > 0) + { + streamResampled.Write(sampleByte, 0, read); + } + + streamResampled.Seek(0, SeekOrigin.Begin); + } while (true) { @@ -485,7 +677,7 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, } totalBytesRead += bytesReturned; - float progress = (float)stream.Position / stream.Length; + float progress = (float)streamResampled.Position / streamResampled.Length; if ((int)(progress * 20) != lastPct) { @@ -506,15 +698,22 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, bool isEmpty = (buffer.Where(v => v != 0).Count() == 0); if (!isEmpty) { - float[] sampleBuffer = ConvertToFloat(buffer, bytesReturned, channels); - + float[] sampleBuffer; + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + sampleBuffer = MemoryMarshal.Cast(buffer.AsSpan()).ToArray(); + else + sampleBuffer = ConvertToFloat(buffer, bytesReturned, channels); + oggOut.WriteSamples(sampleBuffer, 0, sampleBuffer.Length); } lastIndex = (uint)oggOut.PageCounter; } Console.WriteLine("]"); - stream.Close(); + if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux)) + File.Delete(tmpWavFile); + + streamResampled.Close(); } catch (OpusOggWriteStream.PaddingException e) { @@ -534,7 +733,7 @@ private void GenerateAudio(List sourceFiles, uint audioId, int bitRate, catch (Exception e) { Console.WriteLine(); - throw new Exception("Failed processing " + sourceFile); + throw new Exception("Failed processing " + sourceFile + "\n Error: " + e.Message + "\n"); } if (!warned && outputData.Length >= maxSize / 2) @@ -1008,4 +1207,4 @@ private OggPage GetOggPage(ref int offset) } } -} \ No newline at end of file +} diff --git a/TonieAudio/TonieAudio.csproj b/TonieAudio/TonieAudio.csproj index 8b82346..0ef4bd5 100644 --- a/TonieAudio/TonieAudio.csproj +++ b/TonieAudio/TonieAudio.csproj @@ -1,5 +1,5 @@  - + Debug @@ -9,7 +9,7 @@ Properties TonieAudio TonieAudio - v4.6 + v4.8 512 true @@ -18,7 +18,7 @@ true full false - bin\Debug\ + ..\TeddyBench\bin\Debug DEBUG;TRACE prompt 4 @@ -37,7 +37,6 @@ DEBUG;TRACE full x64 - 7.3 prompt MinimumRecommendedRules.ruleset @@ -47,7 +46,6 @@ true pdbonly x64 - 7.3 prompt MinimumRecommendedRules.ruleset @@ -57,7 +55,6 @@ false full x64 - 7.3 prompt MinimumRecommendedRules.ruleset true @@ -68,7 +65,6 @@ true pdbonly x64 - 7.3 prompt MinimumRecommendedRules.ruleset @@ -82,6 +78,23 @@ + + ..\packages\Xabe.FFmpeg.4.4.0\lib\netstandard2.0\Xabe.FFmpeg.dll + + + ..\packages\System.Buffers.4.5.1\lib\net461\System.Buffers.dll + + + + ..\packages\System.Numerics.Vectors.4.5.0\lib\net46\System.Numerics.Vectors.dll + + + + ..\packages\System.Runtime.CompilerServices.Unsafe.4.5.3\lib\net461\System.Runtime.CompilerServices.Unsafe.dll + + + ..\packages\System.Memory.4.5.4\lib\net461\System.Memory.dll + diff --git a/TonieAudio/packages.config b/TonieAudio/packages.config index e3f2a9a..4d0e143 100644 --- a/TonieAudio/packages.config +++ b/TonieAudio/packages.config @@ -3,4 +3,9 @@ + + + + + \ No newline at end of file