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