[vlc-commits] spatialaudio: add an Ambisonics audio renderer

Adrien Maglo git at videolan.org
Wed Jul 19 18:58:17 CEST 2017


vlc | branch: master | Adrien Maglo <magsoft at videolan.org> | Tue Apr 18 15:41:24 2017 +0200| [8cd7cc76481169e338a1f67efe3edb7eb432d407] | committer: Thomas Guillem

spatialaudio: add an Ambisonics audio renderer

This module converts ambisonics channels to physical ones.

This module contains also a binauralizer submodule: It applies a head-related
transfer function to each physical channel in order to simulate a 3D sound with
a headphone.

Signed-off-by: Thomas Guillem <thomas at gllm.fr>

> http://git.videolan.org/gitweb.cgi/vlc.git/?a=commit;h=8cd7cc76481169e338a1f67efe3edb7eb432d407
---

 configure.ac                                       |   5 +
 modules/MODULES_LIST                               |   1 +
 modules/audio_filter/Makefile.am                   |   9 +
 .../audio_filter/channel_mixer/spatialaudio.cpp    | 518 +++++++++++++++++++++
 4 files changed, 533 insertions(+)

diff --git a/configure.ac b/configure.ac
index 8ef7bbcd96..fe93da279e 100644
--- a/configure.ac
+++ b/configure.ac
@@ -2705,6 +2705,11 @@ dnl
 PKG_ENABLE_MODULES_VLC([OPUS], [], [ogg opus >= 1.0.3], [Opus support], [auto])
 
 dnl
+dnl  Ambisonic channel mixer and binauralizer plugin
+dnl
+PKG_ENABLE_MODULES_VLC([SPATIALAUDIO], [], [spatialaudio], [Ambisonic channel mixer and binauralizer], [auto])
+
+dnl
 dnl  theora decoder plugin
 dnl
 PKG_ENABLE_MODULES_VLC([THEORA], [], [ogg theoradec >= 1.0 theoraenc], [experimental theora codec], [auto])
diff --git a/modules/MODULES_LIST b/modules/MODULES_LIST
index 510cc4af36..d4c37a0b0e 100644
--- a/modules/MODULES_LIST
+++ b/modules/MODULES_LIST
@@ -358,6 +358,7 @@ $Id$
  * smf: Standard MIDI file demuxer
  * sndio: OpenBSD sndio audio output
  * soxr: SoX Resampler library audio filter
+ * spatialaudio: Ambisonics to binaural and array of speaker converter
  * spatializer: A spatializer audio filter
  * spdif: S/PDIF audio pass-through decoder
  * speex: a speex audio decoder/packetizer using the libspeex library
diff --git a/modules/audio_filter/Makefile.am b/modules/audio_filter/Makefile.am
index b935045927..1b8a2aba38 100644
--- a/modules/audio_filter/Makefile.am
+++ b/modules/audio_filter/Makefile.am
@@ -75,6 +75,15 @@ audio_filter_LTLIBRARIES += \
 	libsimple_channel_mixer_plugin.la \
 	libtrivial_channel_mixer_plugin.la
 
+# Spatial audio: ambisonics / binaural
+libspatialaudio_plugin_la_SOURCES = \
+	audio_filter/channel_mixer/spatialaudio.cpp
+libspatialaudio_plugin_la_CXXFLAGS = $(AM_CXXFLAGS) $(SPATIALAUDIO_CFLAGS)
+libspatialaudio_plugin_la_LIBADD = $(SPATIALAUDIO_LIBS)
+libspatialaudio_plugin_la_LDFLAGS = $(AM_LDFLAGS) -rpath '$(audio_filterdir)'
+EXTRA_LTLIBRARIES += libspatialaudio_plugin.la
+audio_filter_LTLIBRARIES += $(LTLIBspatialaudio)
+
 # Converters
 libaudio_format_plugin_la_SOURCES = audio_filter/converter/format.c
 libaudio_format_plugin_la_CPPFLAGS = $(AM_CPPFLAGS)
diff --git a/modules/audio_filter/channel_mixer/spatialaudio.cpp b/modules/audio_filter/channel_mixer/spatialaudio.cpp
new file mode 100644
index 0000000000..12ebad5556
--- /dev/null
+++ b/modules/audio_filter/channel_mixer/spatialaudio.cpp
@@ -0,0 +1,518 @@
+/*****************************************************************************
+ * spatialaudio.cpp : Ambisonics audio renderer and binauralizer filter
+ *****************************************************************************
+ * Copyright © 2017 VLC authors and VideoLAN
+ *
+ * Authors: Adrien Maglo <magsoft at videolan.org>
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation; either version 2.1 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * along with this program; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Preamble
+ *****************************************************************************/
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <assert.h>
+
+#include <vlc_common.h>
+#include <vlc_plugin.h>
+#include <vlc_aout.h>
+#include <vlc_filter.h>
+#include <vlc_vout.h>
+#include <vlc_keys.h>
+
+#include <new>
+#include <vector>
+#include <sstream>
+
+#include <spatialaudio/Ambisonics.h>
+#include <spatialaudio/SpeakersBinauralizer.h>
+
+#define CFG_PREFIX "spatialaudio-"
+
+#define DEFAULT_HRTF_PATH "hrtfs" DIR_SEP "dodeca_and_7channel_FHK_HRTF.sofa"
+
+#define HRTF_FILE_TEXT N_("HRTF SOFA file for the binauralization")
+#define HRTF_FILE_LONGTEXT N_("To use a custom HRTF (Head-related transfer function)" \
+                              "in the SOFA format.")
+
+#define HEADPHONES_TEXT N_("Headphones mode (binaural)")
+#define HEADPHONES_LONGTEXT N_("If the output is stereo, render ambisonics " \
+                               "with the binaural decoder.")
+
+static int OpenBinauralizer(vlc_object_t *p_this);
+static int Open( vlc_object_t * );
+static void Close( vlc_object_t * );
+
+vlc_module_begin()
+    set_shortname("Spatialaudio")
+    set_description(N_("Ambisonics renderer and binauralizer"))
+    set_capability("audio renderer", 1)
+    set_category(CAT_AUDIO)
+    set_subcategory(SUBCAT_AUDIO_MISC)
+    set_callbacks(Open, Close)
+    add_bool(CFG_PREFIX "headphones", false,
+             HEADPHONES_TEXT, HEADPHONES_LONGTEXT, true)
+    add_loadfile("hrtf-file", NULL,
+                 HRTF_FILE_TEXT, HRTF_FILE_LONGTEXT, true)
+    add_shortcut("ambisonics")
+
+    add_submodule()
+    set_shortname(N_("Binauralizer"))
+    set_capability("audio filter", 0)
+    set_callbacks(OpenBinauralizer, Close)
+    add_shortcut("binauralizer")
+vlc_module_end()
+
+#define AMB_BLOCK_TIME_LEN 1024
+
+struct filter_sys_t
+{
+    filter_sys_t()
+        : speakers(NULL)
+        , i_inputPTS(0)
+        , inBuf(NULL)
+        , outBuf(NULL)
+    {}
+    ~filter_sys_t()
+    {
+        delete[] speakers;
+        if (inBuf != NULL)
+            for (unsigned i = 0; i < i_inputNb; ++i)
+                free(inBuf[i]);
+        free(inBuf);
+
+        if (outBuf != NULL)
+            for (unsigned i = 0; i < i_outputNb; ++i)
+                free(outBuf[i]);
+        free(outBuf);
+    }
+
+    enum
+    {
+        AMBISONICS_DECODER, // Ambisonics decoding module
+        AMBISONICS_BINAURAL_DECODER, // Ambisonics decoding module using binaural
+        BINAURALIZER // Binauralizer module
+    } mode;
+
+    CAmbisonicBinauralizer binauralDecoder;
+    SpeakersBinauralizer binauralizer;
+    CAmbisonicDecoder speakerDecoder;
+    CAmbisonicProcessor processor;
+    CAmbisonicZoomer zoomer;
+
+    CAmbisonicSpeaker *speakers;
+
+    std::vector<float> inputSamples;
+    mtime_t i_inputPTS;
+    unsigned i_rate;
+    unsigned i_order;
+
+    float** inBuf;
+    float** outBuf;
+    unsigned i_inputNb;
+    unsigned i_outputNb;
+
+    /* View point. */
+    float f_teta;
+    float f_phi;
+    float f_roll;
+    float f_zoom;
+};
+
+static std::string getHRTFPath(filter_t *p_filter)
+{
+    std::string HRTFPath;
+
+    char *userHRTFPath = var_InheritString(p_filter, "hrtf-file");
+
+    if (userHRTFPath != NULL)
+    {
+        HRTFPath = std::string(userHRTFPath);
+        free(userHRTFPath);
+    }
+    else
+    {
+        char *dataDir = config_GetDataDir();
+        if (dataDir != NULL)
+        {
+            std::stringstream ss;
+            ss << std::string(dataDir) << DIR_SEP << DEFAULT_HRTF_PATH;
+            HRTFPath = ss.str();
+            free(dataDir);
+        }
+    }
+
+    return HRTFPath;
+}
+
+static block_t *Mix( filter_t *p_filter, block_t *p_buf )
+{
+    filter_sys_t *p_sys = p_filter->p_sys;
+
+    const size_t i_prevSize = p_sys->inputSamples.size();
+    p_sys->inputSamples.resize(i_prevSize + p_buf->i_nb_samples * p_sys->i_inputNb);
+    memcpy((char*)(p_sys->inputSamples.data() + i_prevSize), (char*)p_buf->p_buffer, p_buf->i_buffer);
+
+    const size_t i_inputBlockSize = sizeof(float) * p_sys->i_inputNb * AMB_BLOCK_TIME_LEN;
+    const size_t i_outputBlockSize = sizeof(float) * p_sys->i_outputNb * AMB_BLOCK_TIME_LEN;
+    const size_t i_nbBlocks = p_sys->inputSamples.size() * sizeof(float) / i_inputBlockSize;
+
+    block_t *p_out_buf = block_Alloc(i_outputBlockSize * i_nbBlocks);
+    if (unlikely(p_out_buf == NULL))
+    {
+        block_Release(p_buf);
+        return NULL;
+    }
+
+    p_out_buf->i_nb_samples = i_nbBlocks * AMB_BLOCK_TIME_LEN;
+    if (p_sys->i_inputPTS == 0)
+        p_out_buf->i_pts = p_buf->i_pts;
+    else
+        p_out_buf->i_pts = p_sys->i_inputPTS;
+    p_out_buf->i_dts = p_out_buf->i_pts;
+    p_out_buf->i_length = p_out_buf->i_nb_samples * 1000000 / p_sys->i_rate;
+
+    float *p_dest = (float *)p_out_buf->p_buffer;
+    const float *p_src = (float *)p_sys->inputSamples.data();
+
+    for (unsigned b = 0; b < i_nbBlocks; ++b)
+    {
+        for (unsigned i = 0; i < p_sys->i_inputNb; ++i)
+        {
+            for (unsigned j = 0; j < AMB_BLOCK_TIME_LEN; ++j)
+            {
+                float val = p_src[(b * AMB_BLOCK_TIME_LEN + j) * p_sys->i_inputNb + i];
+                p_sys->inBuf[i][j] = val;
+            }
+        }
+
+        // Compute
+        switch (p_sys->mode)
+        {
+            case filter_sys_t::BINAURALIZER:
+                p_sys->binauralizer.Process(p_sys->inBuf, p_sys->outBuf);
+                break;
+            case filter_sys_t::AMBISONICS_DECODER:
+            case filter_sys_t::AMBISONICS_BINAURAL_DECODER:
+            {
+                CBFormat inData;
+                inData.Configure(p_sys->i_order, true, AMB_BLOCK_TIME_LEN);
+
+                for (unsigned i = 0; i < p_sys->i_inputNb; ++i)
+                    inData.InsertStream(p_sys->inBuf[i], i, AMB_BLOCK_TIME_LEN);
+
+                Orientation ori(p_sys->f_teta, p_sys->f_phi, p_sys->f_roll);
+                p_sys->processor.SetOrientation(ori);
+                p_sys->processor.Refresh();
+                p_sys->processor.Process(&inData, inData.GetSampleCount());
+
+                p_sys->zoomer.SetZoom(p_sys->f_zoom);
+                p_sys->zoomer.Refresh();
+                p_sys->zoomer.Process(&inData, inData.GetSampleCount());
+
+                if (p_sys->mode == filter_sys_t::AMBISONICS_DECODER)
+                    p_sys->speakerDecoder.Process(&inData, inData.GetSampleCount(), p_sys->outBuf);
+                else
+                    p_sys->binauralDecoder.Process(&inData, p_sys->outBuf);
+                break;
+            }
+            default:
+                vlc_assert_unreachable();
+        }
+
+        // Interleave the results.
+        for (unsigned i = 0; i < p_sys->i_outputNb; ++i)
+            for (unsigned j = 0; j < AMB_BLOCK_TIME_LEN; ++j)
+                p_dest[(b * AMB_BLOCK_TIME_LEN + j) * p_sys->i_outputNb + i] = p_sys->outBuf[i][j];
+    }
+
+    p_sys->inputSamples.erase(p_sys->inputSamples.begin(),
+                              p_sys->inputSamples.begin() + i_inputBlockSize * i_nbBlocks / sizeof(float));
+
+    assert(p_sys->inputSamples.size() < i_inputBlockSize);
+
+    p_sys->i_inputPTS = p_out_buf->i_pts + p_out_buf->i_length;
+
+    block_Release(p_buf);
+    return p_out_buf;
+}
+
+static void Flush( filter_t *p_filter )
+{
+    filter_sys_t *p_sys = p_filter->p_sys;
+    p_sys->inputSamples.clear();
+    p_sys->i_inputPTS = 0;
+}
+
+static void ChangeViewpoint( filter_t *p_filter, const vlc_viewpoint_t *p_vp)
+{
+    filter_sys_t *p_sys = (filter_sys_t *)p_filter->p_sys;
+
+#define RAD(d) ((float) ((d) * M_PI / 180.f))
+    p_sys->f_teta = -RAD(p_vp->yaw);
+    p_sys->f_phi = RAD(p_vp->pitch);
+    p_sys->f_roll = RAD(p_vp->roll);
+
+    if (p_vp->fov >= FIELD_OF_VIEW_DEGREES_DEFAULT)
+        p_sys->f_zoom = 0.f; // no unzoom as it does not really make sense.
+    else
+        p_sys->f_zoom = (FIELD_OF_VIEW_DEGREES_DEFAULT - p_vp->fov) / (FIELD_OF_VIEW_DEGREES_DEFAULT - FIELD_OF_VIEW_DEGREES_MIN);
+#undef RAD
+}
+
+static int allocateBuffers(filter_sys_t *p_sys)
+{
+    p_sys->inBuf = (float**)calloc(p_sys->i_inputNb, sizeof(float*));
+    if (p_sys->inBuf == NULL)
+        return VLC_ENOMEM;
+
+    for (unsigned i = 0; i < p_sys->i_inputNb; ++i)
+    {
+        p_sys->inBuf[i] = (float *)malloc(AMB_BLOCK_TIME_LEN * sizeof(float));
+        if (p_sys->inBuf[i] == NULL)
+            return VLC_ENOMEM;
+    }
+
+    p_sys->outBuf = (float**)calloc(p_sys->i_outputNb, sizeof(float*));
+    if (p_sys->outBuf == NULL)
+        return VLC_ENOMEM;
+
+    for (unsigned i = 0; i < p_sys->i_outputNb; ++i)
+    {
+        p_sys->outBuf[i] = (float *)malloc(AMB_BLOCK_TIME_LEN * sizeof(float));
+        if (p_sys->outBuf[i] == NULL)
+            return VLC_ENOMEM;
+    }
+
+    return VLC_SUCCESS;
+}
+
+static int OpenBinauralizer(vlc_object_t *p_this)
+{
+    filter_t *p_filter = (filter_t *)p_this;
+    audio_format_t *infmt = &p_filter->fmt_in.audio;
+    audio_format_t *outfmt = &p_filter->fmt_out.audio;
+
+    filter_sys_t *p_sys;
+    p_sys = p_filter->p_sys = (filter_sys_t*)new(std::nothrow)filter_sys_t();
+    if (p_sys == NULL)
+        return VLC_ENOMEM;
+
+    p_sys->mode = filter_sys_t::BINAURALIZER;
+    p_sys->i_rate = p_filter->fmt_in.audio.i_rate;
+    p_sys->i_inputNb = p_filter->fmt_in.audio.i_channels;
+    p_sys->i_outputNb = 2;
+
+    if (allocateBuffers(p_sys) != VLC_SUCCESS)
+    {
+        delete p_sys;
+        return VLC_ENOMEM;
+    }
+
+    unsigned s = 0;
+    p_sys->speakers = new CAmbisonicSpeaker[infmt->i_channels]();
+
+    p_sys->speakers[s++].SetPosition({DegreesToRadians(30), 0.f, 1.f});
+    p_sys->speakers[s++].SetPosition({DegreesToRadians(-30), 0.f, 1.f});
+
+    if ((infmt->i_physical_channels & AOUT_CHANS_MIDDLE) == AOUT_CHANS_MIDDLE)
+    {
+        /* Middle */
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(110), 0.f, 1.f});
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(-110), 0.f, 1.f});
+    }
+
+    if ((infmt->i_physical_channels & AOUT_CHANS_REAR) == AOUT_CHANS_REAR)
+    {
+        /* Rear */
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(145), 0.f, 1.f});
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(-145), 0.f, 1.f});
+    }
+
+    if ((infmt->i_physical_channels & AOUT_CHAN_CENTER) == AOUT_CHAN_CENTER)
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(0), 0.f, 1.f});
+
+    if ((infmt->i_physical_channels & AOUT_CHAN_LFE) == AOUT_CHAN_LFE)
+        p_sys->speakers[s++].SetPosition({DegreesToRadians(0), 0.f, 0.5f});
+
+    std::string HRTFPath = getHRTFPath(p_filter);
+    msg_Dbg(p_filter, "Using the HRTF file: %s", HRTFPath.c_str());
+
+    unsigned i_tailLength = 0;
+    if (!p_sys->binauralizer.Configure(p_sys->i_rate, AMB_BLOCK_TIME_LEN,
+                                       p_sys->speakers, infmt->i_channels, i_tailLength,
+                                       HRTFPath))
+    {
+        msg_Err(p_filter, "Error creating the binauralizer.");
+        delete p_sys;
+        return VLC_EGENERIC;
+    }
+    p_sys->binauralizer.Reset();
+
+    outfmt->i_format = infmt->i_format = VLC_CODEC_FL32;
+    outfmt->i_physical_channels = AOUT_CHANS_STEREO;
+    aout_FormatPrepare(infmt);
+    aout_FormatPrepare(outfmt);
+
+    p_filter->pf_audio_filter = Mix;
+    p_filter->pf_flush = Flush;
+    p_filter->pf_change_viewpoint = ChangeViewpoint;
+
+    return VLC_SUCCESS;
+}
+
+static int Open(vlc_object_t *p_this)
+{
+    filter_t *p_filter = (filter_t *)p_this;
+    audio_format_t *infmt = &p_filter->fmt_in.audio;
+    audio_format_t *outfmt = &p_filter->fmt_out.audio;
+
+    assert(infmt->channel_type != outfmt->channel_type);
+
+    if (infmt->channel_type != AUDIO_CHANNEL_TYPE_AMBISONICS)
+        return VLC_EGENERIC;
+
+    if (infmt->i_format != VLC_CODEC_FL32 || outfmt->i_format != VLC_CODEC_FL32)
+        return VLC_EGENERIC;
+
+    filter_sys_t *p_sys;
+    p_sys = p_filter->p_sys = (filter_sys_t*)new(std::nothrow)filter_sys_t();
+    if (p_sys == NULL)
+        return VLC_ENOMEM;
+
+    p_sys->f_teta = 0.f;
+    p_sys->f_phi = 0.f;
+    p_sys->f_roll = 0.f;
+    p_sys->f_zoom = 0.f;
+    p_sys->i_rate = p_filter->fmt_in.audio.i_rate;
+    p_sys->i_inputNb = p_filter->fmt_in.audio.i_channels;
+    p_sys->i_outputNb = p_filter->fmt_out.audio.i_channels;
+
+    if (allocateBuffers(p_sys) != VLC_SUCCESS)
+    {
+        delete p_sys;
+        return VLC_ENOMEM;
+    }
+
+    p_sys->i_order = sqrt(infmt->i_channels) - 1;
+
+    if (p_sys->i_order < 1)
+    {
+        msg_Err(p_filter, "Invalid number of Ambisonics channels");
+        delete p_sys;
+        return VLC_EGENERIC;
+    }
+
+    msg_Dbg(p_filter, "Order: %d %d", p_sys->i_order, infmt->i_channels);
+
+    static const char *const options[] = { "headphones", NULL };
+    config_ChainParse(p_filter, CFG_PREFIX, options, p_filter->p_cfg);
+
+    unsigned i_tailLength = 0;
+    if (p_filter->fmt_out.audio.i_channels == 2
+     && var_InheritBool(p_filter, CFG_PREFIX "headphones"))
+    {
+        p_sys->mode = filter_sys_t::AMBISONICS_BINAURAL_DECODER;
+
+        std::string HRTFPath = getHRTFPath(p_filter);
+        msg_Dbg(p_filter, "Using the HRTF file: %s", HRTFPath.c_str());
+
+        if (!p_sys->binauralDecoder.Configure(p_sys->i_order, true,
+                p_sys->i_rate, AMB_BLOCK_TIME_LEN, i_tailLength,
+                HRTFPath))
+        {
+            msg_Err(p_filter, "Error creating the binaural decoder.");
+            delete p_sys;
+            return VLC_EGENERIC;
+        }
+        p_sys->binauralDecoder.Reset();
+    }
+    else
+    {
+        p_sys->mode = filter_sys_t::AMBISONICS_DECODER;
+
+        unsigned i_nbChannels = aout_FormatNbChannels(&p_filter->fmt_out.audio);
+        if (!p_sys->speakerDecoder.Configure(p_sys->i_order, true,
+            kAmblib_CustomSpeakerSetUp, i_nbChannels))
+        {
+            msg_Err(p_filter, "Error creating the Ambisonics decoder.");
+            delete p_sys;
+            return VLC_EGENERIC;
+        }
+
+        /* Speaker setup, inspired from:
+         * https://www.dolby.com/us/en/guide/surround-sound-speaker-setup/7-1-setup.html
+         * The position must follow the order of pi_vlc_chan_order_wg4 */
+        unsigned s = 0;
+
+        p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(30), 0.f, 1.f});
+        p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(-30), 0.f, 1.f});
+
+        if ((outfmt->i_physical_channels & AOUT_CHANS_MIDDLE) == AOUT_CHANS_MIDDLE)
+        {
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(110), 0.f, 1.f});
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(-110), 0.f, 1.f});
+        }
+
+        if ((outfmt->i_physical_channels & AOUT_CHANS_REAR) == AOUT_CHANS_REAR)
+        {
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(145), 0.f, 1.f});
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(-145), 0.f, 1.f});
+        }
+
+        if ((outfmt->i_physical_channels & AOUT_CHAN_CENTER) == AOUT_CHAN_CENTER)
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(0), 0.f, 1.f});
+
+        if ((outfmt->i_physical_channels & AOUT_CHAN_LFE) == AOUT_CHAN_LFE)
+            p_sys->speakerDecoder.SetPosition(s++, {DegreesToRadians(0), 0.f, 0.5f});
+
+        /* Check we have setup the right number of speaker. */
+        assert(s == i_nbChannels);
+
+        p_sys->speakerDecoder.Refresh();
+    }
+
+    if (!p_sys->processor.Configure(p_sys->i_order, true, AMB_BLOCK_TIME_LEN, 0))
+    {
+        msg_Err(p_filter, "Error creating the ambisonic processor.");
+        delete p_sys;
+        return VLC_EGENERIC;
+    }
+
+    if (!p_sys->zoomer.Configure(p_sys->i_order, true, 0))
+    {
+        msg_Err(p_filter, "Error creating the ambisonic zoomer.");
+        delete p_sys;
+        return VLC_EGENERIC;
+    }
+
+    p_filter->pf_audio_filter = Mix;
+    p_filter->pf_flush = Flush;
+    p_filter->pf_change_viewpoint = ChangeViewpoint;
+
+    return VLC_SUCCESS;
+}
+
+static void Close(vlc_object_t *p_this)
+{
+    filter_t *p_filter = (filter_t *)p_this;
+
+    delete p_filter->p_sys;
+}



More information about the vlc-commits mailing list