import acoular as ac
import numpy as np
import matplotlib.pyplot as plt
Drone auralization example
How to use Acoular for simulating a drone flyby end exporting a stereo wav file
Introduction
In addition to its capabilities to detect and analyze sources with implementations of various array methods, Acoular has several tools for synthesizing signals and simulating measurements. This can be helpful when designing an experiment and pre-testing measurement setups and algorithms without having to actually set up the hardware.
In this post, we’ll be using these tools to simulate the flyby of a multicopter drone as it would be experienced by an arbitrary observer. Instead of the typical microphone array with dozens of microphones, the output will be restricted to just two channels, which shall represent the “ears” of a person.
For applications of the simulation workflow in an array context, see also here or here.
In the following, we will explore how to:
- implement an Acoular-compatible class for generating an artificial multicopter signal
- simulate a source which radiates this signal, has a dipole characteristic and flies along a predefined trajectory
- export a stereo wav file with a simple auralization of that flyby
First, we import Acoular, NumPy, and Matplotlib’s plotting functionality
Generating the drone signal
As Acoular does not feature a specific way to generate drone signals, we define our own signal class for doing that. It is derived from the existing SignalGenerator
class.
Signals of multicopter drones are usually very recognizable, often featuring strong tonal components caused by the rotors. In reality, the radiated noise depends on a lot of factors, which we will not consider in detail here. For now, it shall be enough to assume the signal to be dependent on the number of rotors (with respective rotational speeds) and the number of blades per rotor. From that, the blade passing frequencies are calculated as dominant components and nicely mixed with some higher harmonics and broadband noise.
The new class shall be called DroneSignalGenerator
and its interface will allow setting a list of “revolutions per minute” values for the rotors, implicitly setting the numbers of rotors, and the already mentioned number of blades per rotor.
# import traits.api to enforce data types in object parameters
from traits.api import List, Int
class DroneSignalGenerator( ac.SignalGenerator ):
"""
Class for generating a synthetic multicopter drone signal.
This is just a basic example class for demonstration purposes
with only few settable and some arbitrary fixed parameters.
It is not intended to create perfectly realistic signals.
"""
# List with rotor speeds (for each rotor independently)
# Default: 1 rotor, 15000 rpm
= List([15000,])
rpm_list
# Number of blades per rotor
# Default: 2
= Int(2)
num_blades_per_rotor
def signal( self ):
"""
function that returns the full signal
"""
# initialize a random generator for noise generation
= np.random.default_rng(seed = 42)
rng # use 1/f² broadband noise as basis for the signal
= rng.standard_normal(self.numsamples) # normal distributed values
wn = np.fft.rfft(wn) # transform to freq domain
wnf /= (np.linspace(0.1,1,len(wnf))*5)**2 # spectrum ~ 1/f²
wnf = np.fft.irfft(wnf) # transform to time domain
sig
# vector with all time instances
= np.arange(self.numsamples, dtype=float) / self.sample_freq
t
# iterate over all rotors
for rpm in self.rpm_list:
= rpm / 60 # rotor speed in Hz
f_base
# randomly set phase of rotor
= rng.uniform() * 2*np.pi
phase
# calculate higher harmonics up to 50 times the rotor speed
for n in np.arange(50)+1:
# if we're looking at a blade passing frequency, make it louder
if n % self.num_blades_per_rotor == 0:
= 1
amp else:
= 0.2
amp
# exponentially decrease amplitude for higher freqs with arbitrary factor
*= np.exp(-n/10)
amp
# add harmonic signal component to existing signal
+= amp * np.sin(2*np.pi*n * f_base * t + phase)
sig
# return signal normalized to given RMS value
return sig * self.rms / np.std(sig)
Now we use the newly-defined class to create an object with parameters for a specific drone signal. We choose to have a quadcopter with four rotors. As the simulation should feature the drone flying by, it makes sense for two of the rotors running at a higher speed than the other two, so as to tilt a quadcopter to let it fly in one direction. Moreover, all rotors are chosen to have slightly different rotational speeds to make it a little more realistic (15010 rpm / 14962 rpm and 13536 rpm / 13007 rpm).
The sample_freq
and numsamples
traits that we set here are inherited from the SignalGenerator
base class. We use a standard sampling frequency for audio signals of 44’100 Hz here and generate a signal of a little above 10 seconds. The strength of the signal can be set via the (also inherited) rms
trait, but we just use the default value of 1.0 here, so don’t have to specifically set it.
# length of signal
= 10.5 # s
t_msm # sampling frequency
= 44100 # Hz
f_sample
= DroneSignalGenerator(rpm_list = [15010,14962,13536,13007],
drone_signal = 2,
num_blades_per_rotor = f_sample,
sample_freq = f_sample*t_msm)
numsamples
# If you're running the example in an interactive environment, you might want
# to listen to the pure signal by uncommenting the two following lines:
#from IPython.display import Audio
#display(Audio(drone_signal.signal(),rate = f_sample))
The spectrum (PSD) of our signal looks like this:
1,(10,3))
plt.figure(
plt.psd(drone_signal.signal(), = f_sample,
Fs = 4096)
NFFT plt.show()
Microphone positions
In addition to the signal, we need characteristics of the source itself, its movement, the environment, and the way it is observed. We start with the “observer”, i.e. a microphone array with only two microphones where the ears of a standing person might be located. The observer’s head is looking mostly towards positive \(y\), but also slightly to the side from which the drone will come later.
= ac.MicGeom()
m = np.array([[-0.07, 0.07], # x positions, all values in m
m.mpos_tot -0.03, 0.03], # y
[1.7 , 1.7]]) # z [
Flight path
Next, the flight path is defined. A trajectory in Acoular is defined via its “waypoints” with corresponding times. We can set as many waypoints as we want as a dictionary.
Let’s say that our drone flies about 10 m above ground, from left to right, and a little in front of the observer. The flight speed is 16 m/s in \(x\) direction. The absolute speed will be a little higher, since we slightly and randomly vary the \(z\) and \(y\) positions from one waypoint to the next.
= 16 # m/s
flight_speed
# 11 seconds trajectory, which is a little more than we have signal for
= np.arange(12)
ts
# initialize a random generator for path deviations
= np.random.default_rng(seed = 23)
rng
# Set one waypoint each second,
= { t : ((t-5.5)*flight_speed, # vary
waypoints 6 + rng.uniform(-0.2,0.2), # randomly vary y position up to ±0.2 m around 6 m
10 + rng.uniform(-0.3,0.3)) # randomly vary z position up to ±0.3 m around 10 m height
for t in ts }
= ac.Trajectory(points = waypoints) traj
Let’s plot the trajectory together with the observer positions, as viewed from above.
2,(10,3))
plt.figure(
# plot observer
0,:], m.mpos[1,:], 'rx', label = 'observer')
plt.plot(m.mpos[
# plot trajectory
= np.linspace(0,11,100)
times = traj.location(times)
xt, yt, zt = 'trajectory')
plt.plot(xt, yt, label
# plot the predefined waypoints
= zip(*traj.points.values())
xwp, ywp, zwp '>', label = 'traj. waypoints')
plt.plot(xwp, ywp,
'$x$ / m')
plt.xlabel('$y$ / m')
plt.ylabel(
plt.legend()'equal')
plt.axis( plt.show()
Compared to the length of the flight path, the two observer microphones are positioned rather close to each other, so they appear almost as one position in the plot.
Drone directivity
Now we define an actual source. Many drones exhibit a strong directivity, so using a MovingPointSourceDipole
seems a good choice here. If we don’t explicitly specify otherwise, the dipole lobes will be oriented along the z axis, which is what we want in this case. For calculating the observed sound pressure, i.e. for taking into account the sound travel path source to the observer, the object needs to know what kind of environment it will be existing in. In our case, we just define a resting fluid with a speed of sound of 343 m/s.
# We'll keep the environment simple for now: just air at standard conditions with speed of sound 343 m/s
= ac.Environment(c=343.)
e
# Define point source
= ac.MovingPointSourceDipole(signal = drone_signal, # the signal of the source
p = traj, # set trajectory
trajectory = True, # take into account convective amplification
conv_amp = m, # set the "array" with which to measure the sound field
mics = 0.5, # observation starts 0.5 seconds after signal starts at drone
start = e) # the environment the source is moving in env
With this, we defined our source. But for a little more realism, let’s add a “mirror source” to simulate ground reflections. The properties of this source are exactly the same as for the original source, except for the \(z\) coordinate, which is inversed here. After the definition of our mirror source, both source are combined into a joint sound field description via a SourceMixer
object.
# Copy the waypoints from the original source into a new trajectory, but with inverted z
= { time : (x, y, -z) for time, (x, y, z) in waypoints.items() }
waypoints_reflection = ac.Trajectory(points = waypoints_reflection)
traj_reflection
# Define a mirror source with the mirrored trajectory
= ac.MovingPointSourceDipole(signal = drone_signal, # the same signal as above
p_reflection = traj_reflection, # set trajectory of mirror source
trajectory = True,
conv_amp = m,
mics = 0.5,
start = e)
env
# Mix the original source and the mirror source
= ac.SourceMixer( sources = [p, p_reflection] ) drone_above_ground
Export a WAV file
Now we export a wav file of our simulation, so that we can listen to it with any audio player software. Because of the lazy evaluation paradigm, all the code above should run rather quickly, since no serious calculations were done up to now. This will change here, as the source signal is propagated sample-per-sample to generate the observed sound pressure time signals. The export may take a few minutes, so don’t be alarmed if you don’t immediately get the output.
Before letting the data stream be exported as wav file, however, we divert it through a TimeCache
object, which already writes the data on the disk (in Acoular-compatible format). The reason for this is that we do not need to recalculate everything if we happen to need the exact same data at a later point in time.
# Write data stream onto disk for later re-use. This step is not necessary if runtime isn't an issue.
= ac.TimeCache(source = drone_above_ground)
cached_signals
# Prepare wav output.
# If you don't need caching, you can directly put "source = drone_above_ground" here.
= ac.WriteWAV(name = 'drone_flyby_with_ground_reflection.wav',
output = cached_signals,
source = [0,1]) # export both channels as stereo
channels
# Start the actual export
output.save()
That’s it, we did a simple drone auralization using the tools available in Acoular. If you listen to the output file with properly connected stereo headphones, you should hear something that resembles a quadcopter moving from left to right.
Spectrogram
Finally, let’s visualize the transient signal of one channel in a spectrogram. We use Acoular’s return_result
function on the output object to get the whole signal track at once instead of from a block-wise yielding generator.
3,(10,5))
plt.figure(0],
plt.specgram(ac.tools.return_result(output)[:,= f_sample,
Fs = 4096-256,
noverlap = 4096,
NFFT =-100,
vmin=-50)
vmax0,5000)
plt.ylim(
plt.colorbar()'$t$ / s')
plt.xlabel('$f$ / Hz')
plt.ylabel( plt.show()
Although this simulation is comparativly simple and far from realistic, the spectrogram already exhibits many of the features observed in actual measurement data of drone flybys: the frequency shift due to the Doppler effect, higher levels when the drone is close to the observer, and interference patterns due to ground reflections.
#hide
4,(7,5))
plt.figure(0],
plt.specgram(ac.tools.return_result(output)[:,= f_sample,
Fs = 4096-256,
noverlap = 4096,
NFFT =-100,
vmin=-50)
vmax0,5000)
plt.ylim(
plt.colorbar()'$t$ / s')
plt.xlabel('$f$ / Hz')
plt.ylabel("thumb_drone_spectrogram.png", transparent=True, dpi=50) plt.savefig(