Skip to content

Commit

Permalink
new Point Source 2.5D WFS handling, old one becomes legacy
Browse files Browse the repository at this point in the history
-old handling was Spors/Rabenstein, 2D to 2.5D
-new one is Delft, 3D to 2.5D
  • Loading branch information
fs446 committed Mar 15, 2019
1 parent a15b4f3 commit 45f664f
Show file tree
Hide file tree
Showing 2 changed files with 133 additions and 6 deletions.
26 changes: 26 additions & 0 deletions doc/references.bib
Original file line number Diff line number Diff line change
Expand Up @@ -70,3 +70,29 @@ @article{Borish1984
year = {1984},
doi = {10.1121/1.390983}
}
@article{Firtha2017,
author = {Gergely Firtha AND P{\'e}ter Fiala AND Frank Schultz AND
Sascha Spors},
title = {{Improved Referencing Schemes for 2.5D Wave Field Synthesis
Driving Functions}},
journal = {IEEE/ACM Trans. Audio Speech Language Process.},
volume = {25},
number = {5},
pages = {1117-1127},
year = {2017},
doi = {10.1109/TASLP.2017.2689245}
}
@phdthesis{Start1997,
author = {Evert W. Start},
title = {{Direct Sound Enhancement by Wave Field Synthesis}},
school = {Delft University of Technology},
year = {1997}
}
@phdthesis{Schultz2016,
author = {Frank Schultz},
title = {{Sound Field Synthesis for Line Source Array Applications in
Large-Scale Sound Reinforcement}},
school = {University of Rostock},
year = {2016},
doi = {10.18453/rosdok_id00001765}
}
113 changes: 107 additions & 6 deletions sfs/mono/wfs.py
Original file line number Diff line number Diff line change
Expand Up @@ -155,8 +155,101 @@ def _point(omega, x0, n0, xs, c=None):


def point_25d(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
r"""Driving function for 2.5-dimensional WFS of a virtual point source.
.. versionchanged:: 0.5
see notes, old handling of `point_25d()` is now `point_25d_legacy()`
Parameters
----------
omega : float
Angular frequency of point source.
x0 : (N, 3) array_like
Sequence of secondary source positions.
n0 : (N, 3) array_like
Sequence of normal vectors of secondary sources.
xs : (3,) array_like
Position of virtual point source.
xref : (3,) array_like, optional
Reference point xref or contour xref(x0) for amplitude correct
synthesis.
c : float, optional
Speed of sound in m/s.
omalias: float, optional
Angular frequency where spatial aliasing becomes prominent.
Returns
-------
d : (N,) numpy.ndarray
Complex weights of secondary sources.
selection : (N,) numpy.ndarray
Boolean array containing ``True`` or ``False`` depending on
whether the corresponding secondary source is "active" or not.
secondary_source_function : callable
A function that can be used to create the sound field of a
single secondary source. See `sfs.mono.synthesize()`.
Notes
-----
`point_25d()` derives 2.5D WFS from the 3D
Neumann-Rayleigh integral (i.e. the TU Delft approach), cf. e.g.
Eq. (3.10), (3.11) in :cite:`Start1997`.
Also cf. Eq. (2.137) in :cite:`Schultz2016`.
The theoretical link of `point_25d()` and `point_25d_legacy()` was
introduced as *unified WFS framework* in :cite:`Firtha2017`.
.. math::
D(\x_0,\w) = \sqrt{8 \pi \, \i\wc}
\sqrt{\frac{|\x_\text{ref}-\x_0| \cdot
|\x_0-\x_\text{s}|}{|\x_\text{ref}-\x_0| + |\x_0-\x_\text{s}|}}
\scalarprod{\frac{\x_0-\x_\text{s}}{|\x_0-\x_\text{s}|}}{\n_0}
\frac{\e{-\i\wc |\x_0-\x_\text{s}|}}{4\pi\,|\x_0-\x_\text{s}|}
Examples
--------
.. plot::
:context: close-figs
d, selection, secondary_source = sfs.mono.wfs.point_25d(
omega, array.x, array.n, xs)
normalize_gain = 4 * np.pi * np.linalg.norm(xs)
plot(normalize_gain * d, selection, secondary_source)
"""
x0 = util.asarray_of_rows(x0)
n0 = util.asarray_of_rows(n0)
xs = util.asarray_1d(xs)
xref = util.asarray_1d(xref)
k = util.wavenumber(omega, c)

ds = x0 - xs
dr = xref - x0
s = np.linalg.norm(ds, axis=1)
r = np.linalg.norm(dr, axis=1)

d = (
preeq_25d(omega, omalias, c) *
np.sqrt(8 * np.pi) *
np.sqrt((r * s) / (r + s)) *
inner1d(n0, ds) / s *
np.exp(-1j * k * s) / (4 * np.pi * s))
selection = util.source_selection_point(n0, x0, xs)
return d, selection, secondary_source_point(omega, c)


point_3d = _point


def point_25d_legacy(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
r"""Driving function for 2.5-dimensional WFS for a virtual point source.
.. versionadded:: 0.5
`point_25d()` was renamed to `point_25d_legacy()` (and a new
function with the name `point_25d()` was introduced). See notes for
further details.
Parameters
----------
omega : float
Expand All @@ -171,6 +264,8 @@ def point_25d(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
Reference point for synthesized sound field.
c : float, optional
Speed of sound.
omalias: float, optional
Angular frequency where spatial aliasing becomes prominent.
Returns
-------
Expand All @@ -185,6 +280,13 @@ def point_25d(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
Notes
-----
`point_25d_legacy()` derives 2.5D WFS from the 2D
Neumann-Rayleigh integral (i.e. the approach by Rabenstein & Spors), cf.
:cite:`Spors2008`. Also cf. Eq. (2.145)-(2.147) :cite:`Schultz2016`.
The theoretical link of `point_25d()` and `point_25d_legacy()` was
introduced as *unified WFS framework* in :cite:`Firtha2017`.
.. math::
D(\x_0,\w) = \sqrt{\i\wc |\x_\text{ref}-\x_0|}
Expand All @@ -197,9 +299,10 @@ def point_25d(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
.. plot::
:context: close-figs
d, selection, secondary_source = sfs.mono.wfs.point_25d(
d, selection, secondary_source = sfs.mono.wfs.point_25d_legacy(
omega, array.x, array.n, xs)
plot(d, selection, secondary_source)
normalize_gain = np.linalg.norm(xs)
plot(normalize_gain * d, selection, secondary_source)
"""
x0 = util.asarray_of_rows(x0)
Expand All @@ -217,9 +320,6 @@ def point_25d(omega, x0, n0, xs, xref=[0, 0, 0], c=None, omalias=None):
return d, selection, secondary_source_point(omega, c)


point_3d = _point


def _plane(omega, x0, n0, n=[0, 1, 0], c=None):
r"""Driving function for 2/3-dimensional WFS for a virtual plane wave.
Expand Down Expand Up @@ -499,7 +599,8 @@ def preeq_25d(omega, omalias, c):
H(\w) = \begin{cases}
\sqrt{\i \wc} & \text{for } \w \leq \w_\text{alias} \\
\sqrt{\i \frac{\w_\text{alias}}{c}} & \text{for } \w > \w_\text{alias}
\sqrt{\i \frac{\w_\text{alias}}{c}} &
\text{for } \w > \w_\text{alias}
\end{cases}
"""
Expand Down

0 comments on commit 45f664f

Please sign in to comment.