diff --git a/doc/plugins.rst b/doc/plugins.rst index 29d864699..021b5f43f 100644 --- a/doc/plugins.rst +++ b/doc/plugins.rst @@ -588,6 +588,10 @@ Encodes into `FLAC `_ (lossless). - Description * - **compression** - Sets the libFLAC compression level. The levels range from 0 (fastest, least compression) to 8 (slowest, most compression). + * - **oggflac yes|no** + - Configures if the stream should be Ogg FLAC versus native FLAC. Defaults to "no" (use native FLAC). + * - **oggchaining yes|no** + - Configures if the stream should use Ogg Chaining for in-stream metadata. Defaults to "no". Setting this to "yes" also enables Ogg FLAC. lame ---- diff --git a/src/encoder/plugins/FlacEncoderPlugin.cxx b/src/encoder/plugins/FlacEncoderPlugin.cxx index f2c514178..352434f41 100644 --- a/src/encoder/plugins/FlacEncoderPlugin.cxx +++ b/src/encoder/plugins/FlacEncoderPlugin.cxx @@ -23,8 +23,11 @@ #include "pcm/Buffer.hxx" #include "util/DynamicFifoBuffer.hxx" #include "util/RuntimeError.hxx" +#include "util/Serial.hxx" +#include "util/StringUtil.hxx" #include +#include #if !defined(FLAC_API_VERSION_CURRENT) || FLAC_API_VERSION_CURRENT <= 7 #error libFLAC is too old @@ -34,6 +37,7 @@ class FlacEncoder final : public Encoder { const AudioFormat audio_format; FLAC__StreamEncoder *const fse; + const unsigned compression; PcmBuffer expand_buffer; @@ -44,7 +48,7 @@ class FlacEncoder final : public Encoder { DynamicFifoBuffer output_buffer; public: - FlacEncoder(AudioFormat _audio_format, FLAC__StreamEncoder *_fse); + FlacEncoder(AudioFormat _audio_format, FLAC__StreamEncoder *_fse, unsigned _compression, bool _oggflac, bool _oggchaining); ~FlacEncoder() noexcept override { FLAC__stream_encoder_delete(fse); @@ -56,9 +60,14 @@ public: } void Flush() override { + } + + void PreTag() override { (void) FLAC__stream_encoder_finish(fse); } + void SendTag(const Tag &tag) override; + void Write(const void *data, size_t length) override; size_t Read(void *dest, size_t length) noexcept override { @@ -80,6 +89,8 @@ private: class PreparedFlacEncoder final : public PreparedEncoder { const unsigned compression; + const bool oggchaining; + const bool oggflac; public: explicit PreparedFlacEncoder(const ConfigBlock &block); @@ -88,12 +99,16 @@ public: Encoder *Open(AudioFormat &audio_format) override; [[nodiscard]] const char *GetMimeType() const noexcept override { - return "audio/flac"; + if(oggflac) + return "audio/ogg"; + return "audio/flac"; } }; PreparedFlacEncoder::PreparedFlacEncoder(const ConfigBlock &block) - :compression(block.GetBlockValue("compression", 5U)) + :compression(block.GetBlockValue("compression", 5U)), + oggchaining(block.GetBlockValue("oggchaining",false)), + oggflac(block.GetBlockValue("oggflac",false) || oggchaining) { } @@ -105,8 +120,23 @@ flac_encoder_init(const ConfigBlock &block) static void flac_encoder_setup(FLAC__StreamEncoder *fse, unsigned compression, - const AudioFormat &audio_format, unsigned bits_per_sample) + const AudioFormat &audio_format) { + unsigned bits_per_sample; + + switch (audio_format.format) { + case SampleFormat::S8: + bits_per_sample = 8; + break; + + case SampleFormat::S16: + bits_per_sample = 16; + break; + + default: + bits_per_sample = 24; + } + if (!FLAC__stream_encoder_set_compression_level(fse, compression)) throw FormatRuntimeError("error setting flac compression to %d", compression); @@ -123,16 +153,26 @@ flac_encoder_setup(FLAC__StreamEncoder *fse, unsigned compression, audio_format.sample_rate)) throw FormatRuntimeError("error setting flac sample rate to %d", audio_format.sample_rate); + + if (!FLAC__stream_encoder_set_ogg_serial_number(fse, + GenerateSerial())) + throw FormatRuntimeError("error setting ogg serial number"); } -FlacEncoder::FlacEncoder(AudioFormat _audio_format, FLAC__StreamEncoder *_fse) - :Encoder(false), +FlacEncoder::FlacEncoder(AudioFormat _audio_format, FLAC__StreamEncoder *_fse, unsigned _compression, bool _oggflac, bool _oggchaining) + :Encoder(_oggchaining), audio_format(_audio_format), fse(_fse), + compression(_compression), output_buffer(8192) { /* this immediately outputs data through callback */ - auto init_status = + auto init_status = _oggflac ? + FLAC__stream_encoder_init_ogg_stream(fse, + nullptr, WriteCallback, + nullptr, nullptr, nullptr, + this) + : FLAC__stream_encoder_init_stream(fse, WriteCallback, nullptr, nullptr, nullptr, @@ -146,24 +186,17 @@ FlacEncoder::FlacEncoder(AudioFormat _audio_format, FLAC__StreamEncoder *_fse) Encoder * PreparedFlacEncoder::Open(AudioFormat &audio_format) { - unsigned bits_per_sample; - - /* FIXME: flac should support 32bit as well */ switch (audio_format.format) { case SampleFormat::S8: - bits_per_sample = 8; break; case SampleFormat::S16: - bits_per_sample = 16; break; case SampleFormat::S24_P32: - bits_per_sample = 24; break; default: - bits_per_sample = 24; audio_format.format = SampleFormat::S24_P32; } @@ -173,16 +206,46 @@ PreparedFlacEncoder::Open(AudioFormat &audio_format) throw std::runtime_error("FLAC__stream_encoder_new() failed"); try { - flac_encoder_setup(fse, compression, - audio_format, bits_per_sample); + flac_encoder_setup(fse, compression, audio_format); } catch (...) { FLAC__stream_encoder_delete(fse); throw; } - return new FlacEncoder(audio_format, fse); + return new FlacEncoder(audio_format, fse, compression, oggflac, oggchaining); } +void +FlacEncoder::SendTag(const Tag &tag) +{ + /* re-initialize encoder since flac_encoder_finish resets everything */ + flac_encoder_setup(fse, compression, audio_format); + + FLAC__StreamMetadata *metadata = FLAC__metadata_object_new(FLAC__METADATA_TYPE_VORBIS_COMMENT); + FLAC__StreamMetadata_VorbisComment_Entry entry; + + for (const auto &item : tag) { + char name[64]; + ToUpperASCII(name, tag_item_names[item.type], sizeof(name)); + FLAC__metadata_object_vorbiscomment_entry_from_name_value_pair(&entry, name, item.value); + FLAC__metadata_object_vorbiscomment_append_comment(metadata, entry, false); + } + + FLAC__stream_encoder_set_metadata(fse,&metadata,1); + + auto init_status = FLAC__stream_encoder_init_ogg_stream(fse, + nullptr, WriteCallback, + nullptr, nullptr, nullptr, + this); + + FLAC__metadata_object_delete(metadata); + + if (init_status != FLAC__STREAM_ENCODER_INIT_STATUS_OK) + throw FormatRuntimeError("failed to initialize encoder: %s\n", + FLAC__StreamEncoderInitStatusString[init_status]); +} + + static inline void pcm8_to_flac(int32_t *out, const int8_t *in, unsigned num_samples) noexcept { diff --git a/src/encoder/plugins/OggEncoder.hxx b/src/encoder/plugins/OggEncoder.hxx index aa668bcb9..39b62d822 100644 --- a/src/encoder/plugins/OggEncoder.hxx +++ b/src/encoder/plugins/OggEncoder.hxx @@ -23,7 +23,7 @@ #include "../EncoderAPI.hxx" #include "lib/xiph/OggStreamState.hxx" #include "lib/xiph/OggPage.hxx" -#include "lib/xiph/OggSerial.hxx" +#include "util/Serial.hxx" #include @@ -42,7 +42,7 @@ protected: public: OggEncoder(bool _implements_tag) :Encoder(_implements_tag), - stream(GenerateOggSerial()) { + stream(GenerateSerial()) { } /* virtual methods from class Encoder */ diff --git a/src/encoder/plugins/OpusEncoderPlugin.cxx b/src/encoder/plugins/OpusEncoderPlugin.cxx index d579d1e50..717e5f998 100644 --- a/src/encoder/plugins/OpusEncoderPlugin.cxx +++ b/src/encoder/plugins/OpusEncoderPlugin.cxx @@ -405,7 +405,7 @@ OpusEncoder::PreTag() void OpusEncoder::SendTag(const Tag &tag) { - stream.Reinitialize(GenerateOggSerial()); + stream.Reinitialize(GenerateSerial()); opus_encoder_ctl(enc, OPUS_GET_LOOKAHEAD(&lookahead)); GenerateHeaders(&tag); } diff --git a/src/encoder/plugins/VorbisEncoderPlugin.cxx b/src/encoder/plugins/VorbisEncoderPlugin.cxx index 4acf1ec05..c460ab94e 100644 --- a/src/encoder/plugins/VorbisEncoderPlugin.cxx +++ b/src/encoder/plugins/VorbisEncoderPlugin.cxx @@ -225,7 +225,7 @@ VorbisEncoder::SendTag(const Tag &tag) /* reset ogg_stream_state and begin a new stream */ - stream.Reinitialize(GenerateOggSerial()); + stream.Reinitialize(GenerateSerial()); /* send that vorbis_comment to the ogg_stream_state */ diff --git a/src/lib/xiph/meson.build b/src/lib/xiph/meson.build index d57d68d8f..4666d28fc 100644 --- a/src/lib/xiph/meson.build +++ b/src/lib/xiph/meson.build @@ -72,7 +72,6 @@ if libogg_dep.found() ogg = static_library( 'ogg', 'OggVisitor.cxx', - 'OggSerial.cxx', 'OggSyncState.cxx', 'OggFind.cxx', 'OggPacket.cxx', diff --git a/src/lib/xiph/OggSerial.cxx b/src/util/Serial.cxx similarity index 86% rename from src/lib/xiph/OggSerial.cxx rename to src/util/Serial.cxx index 6f5ba3f84..c91fe62da 100644 --- a/src/lib/xiph/OggSerial.cxx +++ b/src/util/Serial.cxx @@ -17,18 +17,18 @@ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ -#include "OggSerial.hxx" -#include "util/Compiler.h" +#include "Serial.hxx" +#include "Compiler.h" #include #include -static std::atomic_uint next_ogg_serial; +static std::atomic_uint next_serial; int -GenerateOggSerial() noexcept +GenerateSerial() noexcept { - unsigned serial = ++next_ogg_serial; + unsigned serial = ++next_serial; if (gcc_unlikely(serial < 16)) { /* first-time initialization: seed with a clock value, which is random enough for our use */ @@ -38,7 +38,7 @@ GenerateOggSerial() noexcept const auto now = steady_clock::now().time_since_epoch(); const auto now_ms = duration_cast(now); const unsigned seed = now_ms.count(); - next_ogg_serial = serial = seed; + next_serial = serial = seed; } return serial; diff --git a/src/lib/xiph/OggSerial.hxx b/src/util/Serial.hxx similarity index 86% rename from src/lib/xiph/OggSerial.hxx rename to src/util/Serial.hxx index bc6298139..2e8723e7f 100644 --- a/src/lib/xiph/OggSerial.hxx +++ b/src/util/Serial.hxx @@ -17,13 +17,13 @@ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ -#ifndef MPD_OGG_SERIAL_HXX -#define MPD_OGG_SERIAL_HXX +#ifndef MPD_SERIAL_HXX +#define MPD_SERIAL_HXX /** - * Generate the next pseudo-random Ogg serial. + * Generate the next pseudo-random serial. */ int -GenerateOggSerial() noexcept; +GenerateSerial() noexcept; #endif diff --git a/src/util/meson.build b/src/util/meson.build index dc37588cb..953296e5c 100644 --- a/src/util/meson.build +++ b/src/util/meson.build @@ -29,6 +29,7 @@ util = static_library( 'ByteReverse.cxx', 'format.c', 'BitReverse.cxx', + 'Serial.cxx', include_directories: inc, )