using System; using System.ComponentModel; using System.Collections.Generic; using System.IO; using CUETools.Codecs; using CUETools.Processor; namespace CUETools.Converter { class Program { static void Usage() { Console.Error.WriteLine("Usage : CUETools.Converter.exe [options] "); Console.Error.WriteLine(); Console.Error.WriteLine("Options:"); Console.Error.WriteLine(); Console.Error.WriteLine(" --decoder Use non-default decoder."); Console.Error.WriteLine(" --decoder-option "); Console.Error.WriteLine(" --encoder Use non-default encoder."); Console.Error.WriteLine(" --encoder-format Use encoder format different from file extension."); Console.Error.WriteLine(" --lossy Use lossy encoder/mode."); Console.Error.WriteLine(" --lossless Use lossless encoder/mode (default)."); Console.Error.WriteLine(" --ignore-chunk-sizes Ignore WAV length (for pipe input)"); Console.Error.WriteLine(" -p # Padding bytes."); Console.Error.WriteLine(" -m Encoder mode (0..8 for flac, V0..V9 for mp3, etc)"); Console.Error.WriteLine(); } public static AudioEncoderSettingsViewModel GetEncoder(CUEToolsCodecsConfig config, CUEToolsFormat fmt, bool lossless, string chosenEncoder) { AudioEncoderSettingsViewModel tmpEncoder; return chosenEncoder != null ? (config.encodersViewModel.TryGetValue(fmt.extension, lossless, chosenEncoder, out tmpEncoder) ? tmpEncoder : null) : (lossless ? fmt.encoderLossless : fmt.encoderLossy); } public static IAudioSource GetAudioSource(CUEToolsCodecsConfig config, string path, string chosenDecoder, bool ignore_chunk_sizes, Dictionary decoderOptions) { if (path == "-") return new Codecs.WAV.AudioDecoder(new Codecs.WAV.DecoderSettings() { IgnoreChunkSizes = true }, "", Console.OpenStandardInput()); string extension = Path.GetExtension(path).ToLower(); Stream IO = null; if (extension == ".bin") return new Codecs.WAV.AudioDecoder(new Codecs.WAV.DecoderSettings(), path, IO, AudioPCMConfig.RedBook); CUEToolsFormat fmt; if (!extension.StartsWith(".") || !config.formats.TryGetValue(extension.Substring(1), out fmt)) throw new Exception("Unsupported audio type: " + path); var decoder = fmt.decoder; if (chosenDecoder != null && !config.decodersViewModel.TryGetValue(fmt.extension, chosenDecoder, out decoder)) throw new Exception("Unknown audio decoder " + chosenDecoder + " or unsupported audio type " + fmt.extension); if (decoder == null) throw new Exception("Unsupported audio type: " + path); var settings = decoder.Settings.Clone(); foreach (var decOpt in decoderOptions) { var property = TypeDescriptor.GetProperties(settings).Find(decOpt.Key, true); if (property == null) throw new Exception($"{settings.Name} {settings.Extension} decoder settings object (of type {settings.GetType().FullName}) doesn't have a property named {decOpt.Key}."); property.SetValue(settings, TypeDescriptor.GetConverter(property.PropertyType).ConvertFromString(decOpt.Value)); } try { object src = Activator.CreateInstance(decoder.Settings.DecoderType, settings, path, IO); if (src == null || !(src is IAudioSource)) throw new Exception("Unsupported audio type: " + path + ": " + decoder.Settings.DecoderType.FullName); return src as IAudioSource; } catch (System.Reflection.TargetInvocationException ex) { if (ex.InnerException == null) throw ex; throw ex.InnerException; } } static int Main(string[] args) { bool ok = true; string sourceFile = null, destFile = null; int padding = 8192; string encoderMode = null; string decoderName = null; string encoderName = null; string encoderFormat = null; bool ignore_chunk_sizes = false; AudioEncoderType audioEncoderType = AudioEncoderType.NoAudio; var decoderOptions = new Dictionary(); for (int arg = 0; arg < args.Length; arg++) { if (args[arg].Length == 0) ok = false; else if (args[arg] == "--ignore-chunk-sizes") ignore_chunk_sizes = true; else if (args[arg] == "--decoder" && ++arg < args.Length) decoderName = args[arg]; else if (args[arg] == "--decoder-option" && arg + 2 < args.Length) { var optionName = args[++arg]; var optionValue = args[++arg]; decoderOptions.Add(optionName, optionValue); } else if (args[arg] == "--encoder" && ++arg < args.Length) encoderName = args[arg]; else if (args[arg] == "--encoder-format" && ++arg < args.Length) encoderFormat = args[arg]; else if ((args[arg] == "-p" || args[arg] == "--padding") && ++arg < args.Length) ok = int.TryParse(args[arg], out padding); else if ((args[arg] == "-m" || args[arg] == "--mode") && ++arg < args.Length) encoderMode = args[arg]; else if (args[arg] == "--lossy") audioEncoderType = AudioEncoderType.Lossy; else if (args[arg] == "--lossless") audioEncoderType = AudioEncoderType.Lossless; else if ((args[arg][0] != '-' || args[arg] == "-") && sourceFile == null) sourceFile = args[arg]; else if ((args[arg][0] != '-' || args[arg] == "-") && sourceFile != null && destFile == null) destFile = args[arg]; else ok = false; if (!ok) break; } Console.Error.WriteLine("CUETools.Converter, Copyright (C) 2009-2020 Grigory Chudov."); Console.Error.WriteLine("This is free software under the GNU GPLv3+ license; There is NO WARRANTY, to"); Console.Error.WriteLine("the extent permitted by law. for details."); if (!ok || sourceFile == null || destFile == null) { Usage(); return 22; } if (destFile != "-" && destFile != "nul" && File.Exists(destFile)) { Console.Error.WriteLine("Error: file already exists."); return 17; } DateTime start = DateTime.Now; TimeSpan lastPrint = TimeSpan.FromMilliseconds(0); var config = new CUEConfigAdvanced(); config.Init(); #if !DEBUG try #endif { IAudioSource audioSource = null; IAudioDest audioDest = null; TagLib.UserDefined.AdditionalFileTypes.Config = config; TagLib.File sourceInfo = sourceFile == "-" ? null : TagLib.File.Create(new TagLib.File.LocalFileAbstraction(sourceFile)); #if !DEBUG try #endif { audioSource = Program.GetAudioSource(config, sourceFile, decoderName, ignore_chunk_sizes, decoderOptions); AudioBuffer buff = new AudioBuffer(audioSource, 0x10000); Console.Error.WriteLine("Filename : {0}", sourceFile); Console.Error.WriteLine("File Info : {0}kHz; {1} channel; {2} bit; {3}", audioSource.PCM.SampleRate, audioSource.PCM.ChannelCount, audioSource.PCM.BitsPerSample, audioSource.Duration); CUEToolsFormat fmt; if (encoderFormat == null) { if (destFile == "-" || destFile == "nul") encoderFormat = "wav"; else { string extension = Path.GetExtension(destFile).ToLower(); if (!extension.StartsWith(".")) throw new Exception("Unknown encoder format: " + destFile); encoderFormat = extension.Substring(1); } } if (!config.formats.TryGetValue(encoderFormat, out fmt)) throw new Exception("Unsupported encoder format: " + encoderFormat); AudioEncoderSettingsViewModel encoder = audioEncoderType == AudioEncoderType.Lossless ? Program.GetEncoder(config, fmt, true, encoderName) : audioEncoderType == AudioEncoderType.Lossy ? Program.GetEncoder(config, fmt, false, encoderName) : Program.GetEncoder(config, fmt, true, encoderName) ?? Program.GetEncoder(config, fmt, false, encoderName); if (encoder == null) { var lst = new List(config.encodersViewModel).FindAll( e => e.Extension == fmt.extension && (audioEncoderType == AudioEncoderType.NoAudio || audioEncoderType == (e.Lossless ? AudioEncoderType.Lossless : AudioEncoderType.Lossy))). ConvertAll(e => e.Name + (e.Lossless ? " (lossless)" : " (lossy)")); throw new Exception("Encoders available for format " + fmt.extension + ": " + (lst.Count == 0 ? "none" : string.Join(", ", lst.ToArray()))); } var settings = encoder.Settings.Clone(); settings.PCM = audioSource.PCM; settings.Padding = padding; settings.EncoderMode = encoderMode ?? settings.EncoderMode; object o = null; try { o = destFile == "-" ? Activator.CreateInstance(settings.EncoderType, settings, "", Console.OpenStandardOutput()) : destFile == "nul" ? Activator.CreateInstance(settings.EncoderType, settings, "", new NullStream()) : Activator.CreateInstance(settings.EncoderType, settings, destFile, null); } catch (System.Reflection.TargetInvocationException ex) { throw ex.InnerException; } if (o == null || !(o is IAudioDest)) throw new Exception("Unsupported audio type: " + destFile + ": " + settings.EncoderType.FullName); audioDest = o as IAudioDest; audioDest.FinalSampleCount = audioSource.Length; bool keepRunning = true; Console.CancelKeyPress += delegate(object sender, ConsoleCancelEventArgs e) { keepRunning = false; if (e.SpecialKey == ConsoleSpecialKey.ControlC) e.Cancel = true; else audioDest.Delete(); }; while (audioSource.Read(buff, -1) != 0) { audioDest.Write(buff); TimeSpan elapsed = DateTime.Now - start; if ((elapsed - lastPrint).TotalMilliseconds > 60) { var duration = audioSource.Duration; var position = TimeSpan.FromSeconds((double)audioSource.Position / audioSource.PCM.SampleRate); if (duration == TimeSpan.Zero && sourceInfo != null) duration = sourceInfo.Properties.Duration; if (duration < position) duration = position; if (duration < TimeSpan.FromSeconds(1)) duration = TimeSpan.FromSeconds(1); Console.Error.Write("\rProgress : {0:00}%; {1:0.00}x; {2}/{3}", 100.0 * position.TotalSeconds / duration.TotalSeconds, position.TotalSeconds / elapsed.TotalSeconds, elapsed, TimeSpan.FromSeconds(elapsed.TotalSeconds / position.TotalSeconds * duration.TotalSeconds) ); lastPrint = elapsed; } if (!keepRunning) throw new Exception("Aborted"); } TimeSpan totalElapsed = DateTime.Now - start; Console.Error.Write("\r \r"); Console.Error.WriteLine("Results : {0:0.00}x; {1}", audioSource.Position / totalElapsed.TotalSeconds / audioSource.PCM.SampleRate, totalElapsed ); } #if !DEBUG catch (Exception ex) { if (audioSource != null) audioSource.Close(); if (audioDest != null) audioDest.Delete(); throw ex; } #endif audioSource.Close(); audioDest.Close(); if (sourceFile != "-" && destFile != "-" && destFile != "nul") { TagLib.File destInfo = TagLib.File.Create(new TagLib.File.LocalFileAbstraction(destFile)); if (Tagging.UpdateTags(destInfo, Tagging.Analyze(sourceInfo), config, false)) { sourceInfo.Tag.CopyTo(destInfo.Tag, true); destInfo.Tag.Pictures = sourceInfo.Tag.Pictures; destInfo.Save(); } } } #if !DEBUG catch (Exception ex) { Console.Error.Write("\r \r"); Console.Error.WriteLine("Error : {0}", ex.Message); return 1; //Console.WriteLine("{0}", ex.StackTrace); } #endif return 0; } } }