Compare commits
5 Commits
2270a297cc
...
f10e78ff41
Author | SHA1 | Date | |
---|---|---|---|
f10e78ff41 | |||
1a85b60d85 | |||
c748e6cb75 | |||
fe53ada328 | |||
8c6e6a5828 |
@ -24,10 +24,7 @@ typedef struct Siggen {
|
||||
SignalType signaltype;
|
||||
d fs; // Sampling frequency [Hz]
|
||||
d level_amp;
|
||||
void* private;
|
||||
|
||||
char private_data[PRIVATE_SIZE];
|
||||
|
||||
} Siggen;
|
||||
|
||||
typedef struct {
|
||||
@ -35,6 +32,16 @@ typedef struct {
|
||||
d omg;
|
||||
} SinewaveSpecific;
|
||||
|
||||
typedef struct {
|
||||
d fl;
|
||||
d fu;
|
||||
d Ts;
|
||||
d phase;
|
||||
d tau;
|
||||
bool pos;
|
||||
us flags;
|
||||
} SweepSpecific;
|
||||
|
||||
typedef struct {
|
||||
d V1, V2, S;
|
||||
int phase;
|
||||
@ -51,7 +58,6 @@ Siggen* Siggen_create(SignalType type, const d fs,const d level_dB) {
|
||||
Siggen* siggen = a_malloc(sizeof(Siggen));
|
||||
siggen->signaltype = type;
|
||||
siggen->fs = fs;
|
||||
siggen->private = NULL;
|
||||
siggen->level_amp = level_amp(level_dB);
|
||||
|
||||
feTRACE(15);
|
||||
@ -61,23 +67,24 @@ Siggen* Siggen_create(SignalType type, const d fs,const d level_dB) {
|
||||
Siggen* Siggen_Sinewave_create(const d fs, const d freq,const d level_dB) {
|
||||
fsTRACE(15);
|
||||
|
||||
Siggen* sinesiggen = Siggen_create(SINEWAVE, fs, level_dB);
|
||||
sinesiggen->private = sinesiggen->private_data;
|
||||
SinewaveSpecific* sp = (SinewaveSpecific*) sinesiggen->private;
|
||||
Siggen* sine = Siggen_create(SINEWAVE, fs, level_dB);
|
||||
dbgassert(sizeof(SinewaveSpecific) <= sizeof(sine->private_data),
|
||||
"Allocated memory too small");
|
||||
SinewaveSpecific* sp = (SinewaveSpecific*) sine->private_data;
|
||||
sp->curtime = 0;
|
||||
sp->omg = 2*number_pi*freq;
|
||||
|
||||
feTRACE(15);
|
||||
return sinesiggen;
|
||||
return sine;
|
||||
}
|
||||
|
||||
Siggen* Siggen_Whitenoise_create(const d fs, const d level_dB) {
|
||||
fsTRACE(15);
|
||||
|
||||
Siggen* whitenoise = Siggen_create(WHITENOISE, fs, level_dB);
|
||||
whitenoise->private = whitenoise->private_data;
|
||||
dbgassert(sizeof(WhitenoiseSpecific) <= sizeof(whitenoise->private_data), "Allocated memory too small");
|
||||
WhitenoiseSpecific* wn = whitenoise->private;
|
||||
dbgassert(sizeof(WhitenoiseSpecific) <= sizeof(whitenoise->private_data),
|
||||
"Allocated memory too small");
|
||||
WhitenoiseSpecific* wn = (WhitenoiseSpecific*) whitenoise->private_data;
|
||||
wn->phase = 0;
|
||||
wn->V1 = 0;
|
||||
wn->V2 = 0;
|
||||
@ -87,12 +94,50 @@ Siggen* Siggen_Whitenoise_create(const d fs, const d level_dB) {
|
||||
return whitenoise;
|
||||
}
|
||||
|
||||
|
||||
Siggen* Siggen_Sweep_create(const d fs,const d fl,const d fu,
|
||||
const d Ts, const us flags, const d level_dB) {
|
||||
fsTRACE(15);
|
||||
|
||||
Siggen* sweep = Siggen_create(SWEEP, fs, level_dB);
|
||||
dbgassert(sizeof(SweepSpecific) <= sizeof(sweep->private_data),
|
||||
"Allocated memory too small");
|
||||
// Set pointer to inplace data storage
|
||||
SweepSpecific* sp = (SweepSpecific*) sweep->private_data;
|
||||
if(fl < 0 || fu < 0 || Ts <= 0) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
sp->flags = flags;
|
||||
|
||||
sp->fl = fl;
|
||||
sp->fu = fu;
|
||||
sp->Ts = Ts;
|
||||
sp->phase = 0;
|
||||
sp->pos = flags & SWEEP_FLAG_BACKWARD ? false: true;
|
||||
if(flags & SWEEP_FLAG_BACKWARD) {
|
||||
sp->tau = Ts;
|
||||
} else {
|
||||
sp->tau = 0;
|
||||
}
|
||||
/* sp->pos = false; */
|
||||
/* sp->tau = Ts/2; */
|
||||
|
||||
feTRACE(15);
|
||||
return sweep;
|
||||
}
|
||||
|
||||
void Siggen_free(Siggen* siggen) {
|
||||
fsTRACE(15);
|
||||
assertvalidptr(siggen);
|
||||
|
||||
if(siggen->signaltype == SWEEP) {
|
||||
switch(siggen->signaltype) {
|
||||
case SWEEP:
|
||||
/* Sweep specific stuff here */
|
||||
break;
|
||||
case SINEWAVE:
|
||||
/* Sweep specific stuff here */
|
||||
break;
|
||||
}
|
||||
|
||||
a_free(siggen);
|
||||
@ -114,6 +159,81 @@ static void Sinewave_genSignal(Siggen* siggen, SinewaveSpecific* sine, vd* sampl
|
||||
feTRACE(15);
|
||||
}
|
||||
|
||||
static void Sweep_genSignal(Siggen* siggen, SweepSpecific* sweep,
|
||||
vd* samples) {
|
||||
fsTRACE(15);
|
||||
assertvalidptr(sweep);
|
||||
|
||||
const d fl = sweep->fl;
|
||||
const d fu = sweep->fu;
|
||||
const d deltat = 1/siggen->fs;
|
||||
const d Ts = sweep->Ts;
|
||||
|
||||
const d Thalf = Ts/2;
|
||||
|
||||
dVARTRACE(20, deltat);
|
||||
|
||||
// Load state
|
||||
d tau = sweep->tau;
|
||||
bool pos = sweep->pos;
|
||||
|
||||
// Obtain flags and expand
|
||||
us flags = sweep->flags;
|
||||
bool forward_sweep = flags & SWEEP_FLAG_FORWARD;
|
||||
bool backward_sweep = flags & SWEEP_FLAG_BACKWARD;
|
||||
dbgassert(!(forward_sweep && backward_sweep), "Both forward and backward flag set");
|
||||
|
||||
d k, Treverse;
|
||||
if(forward_sweep || backward_sweep) {
|
||||
k = (fu - fl)/Ts;
|
||||
Treverse = Ts;
|
||||
}
|
||||
else {
|
||||
k = (fu - fl)/Thalf;
|
||||
Treverse = Ts/2;
|
||||
}
|
||||
|
||||
|
||||
/* const d k = 0; */
|
||||
|
||||
d phase = sweep->phase;
|
||||
d curfreq;
|
||||
for(us i =0; i< samples->n_rows; i++) {
|
||||
|
||||
curfreq = fl + k*tau;
|
||||
phase = phase + 2*number_pi*curfreq*deltat;
|
||||
|
||||
// Subtract some to avoid possible overflow. Don't know whether such a
|
||||
// thing really happens
|
||||
if(phase > 2*number_pi)
|
||||
phase = phase - 2*number_pi;
|
||||
|
||||
if(pos) {
|
||||
tau = tau + deltat;
|
||||
if(tau >= Treverse) {
|
||||
if(forward_sweep) { tau = 0; }
|
||||
else if(backward_sweep) { dbgassert(false, "BUG"); }
|
||||
else { pos = false; }
|
||||
}
|
||||
|
||||
} else {
|
||||
/* dbgassert(false, "cannot get here"); */
|
||||
tau = tau - deltat;
|
||||
if(tau <= 0) {
|
||||
if(backward_sweep) { tau = Treverse; }
|
||||
else if(forward_sweep) { dbgassert(false, "BUG"); }
|
||||
else { pos = true; }
|
||||
}
|
||||
}
|
||||
setvecval(samples, i, siggen->level_amp*d_sin(phase));
|
||||
}
|
||||
// Store state
|
||||
sweep->phase = phase;
|
||||
sweep->pos = pos;
|
||||
sweep->tau = tau;
|
||||
feTRACE(15);
|
||||
}
|
||||
|
||||
static void Whitenoise_genSignal(Siggen* siggen, WhitenoiseSpecific* wn, vd* samples) {
|
||||
fsTRACE(15);
|
||||
d X;
|
||||
@ -155,19 +275,23 @@ void Siggen_genSignal(Siggen* siggen,vd* samples) {
|
||||
fsTRACE(15);
|
||||
assertvalidptr(siggen);
|
||||
assert_vx(samples);
|
||||
d fs = siggen->fs;
|
||||
|
||||
switch(siggen->signaltype) {
|
||||
case SINEWAVE:
|
||||
Sinewave_genSignal(siggen, (SinewaveSpecific*) siggen->private,
|
||||
Sinewave_genSignal(siggen,
|
||||
(SinewaveSpecific*) siggen->private_data,
|
||||
samples);
|
||||
|
||||
break;
|
||||
case WHITENOISE:
|
||||
Whitenoise_genSignal(siggen, (WhitenoiseSpecific*) siggen->private,
|
||||
Whitenoise_genSignal(siggen,
|
||||
(WhitenoiseSpecific*) siggen->private_data,
|
||||
samples);
|
||||
break;
|
||||
case SWEEP:
|
||||
Sweep_genSignal(siggen,
|
||||
(SweepSpecific*) siggen->private_data,
|
||||
samples);
|
||||
break;
|
||||
default:
|
||||
dbgassert(false, "Not implementend signal type");
|
||||
|
@ -37,8 +37,30 @@ Siggen* Siggen_Whitenoise_create(const d fs, const d level_dB);
|
||||
*/
|
||||
Siggen* Siggen_Pinknoise_create(const us fs,const d level_dB);
|
||||
|
||||
/* Siggen* Siggen_ForwardSweep_create(const d fs,; */
|
||||
/* Siggen* Siggen_(const d fs,; */
|
||||
// Define this flag to repeat a forward sweep only, or backward only. If not
|
||||
// set, we do a continuous sweep
|
||||
#define SWEEP_FLAG_FORWARD 1
|
||||
#define SWEEP_FLAG_BACKWARD 2
|
||||
|
||||
// Types of sweeps
|
||||
#define SWEEP_FLAG_LINEAR 4
|
||||
#define SWEEP_FLAG_EXPONENTIAL 8
|
||||
#define SWEEP_FLAG_HYPERBOLIC 16
|
||||
|
||||
/**
|
||||
* Create a forward sweep
|
||||
*
|
||||
* @param[in] fs: Sampling frequency [Hz]
|
||||
* @param[in] fl: Lower frequency [Hz]
|
||||
* @param[in] fl: Upper frequency [Hz]
|
||||
* @param[in] Ts: Sweep period [s]
|
||||
* @param[in] sweep_flags: Sweep period [s]
|
||||
* @param[in] level: Relative level in [dB], should be between -inf and 0
|
||||
* @return Siggen* handle
|
||||
*/
|
||||
Siggen* Siggen_Sweep_create(const d fs,const d fl,const d fu,
|
||||
const d Ts, const us sweep_flags,
|
||||
const d level);
|
||||
|
||||
/**
|
||||
* Obtain a new piece of signal
|
||||
|
@ -1,4 +1,4 @@
|
||||
#define TRACERPLUS 10
|
||||
#define TRACERPLUS (-5)
|
||||
#include "lasp_sosfilterbank.h"
|
||||
|
||||
|
||||
|
@ -52,7 +52,6 @@ class FilterBankDesigner:
|
||||
|
||||
Returns:
|
||||
True if filter is norm-compliant, False if not
|
||||
|
||||
"""
|
||||
# Skip zero-frequency
|
||||
if np.isclose(freq[0], 0):
|
||||
@ -62,7 +61,8 @@ class FilterBankDesigner:
|
||||
|
||||
# Interpolate limites to frequency array as given
|
||||
llim_full = np.interp(freq, freqlim, llim, left=-np.inf, right=-np.inf)
|
||||
ulim_full = np.interp(freq, freqlim, ulim, left=ulim[0], right=ulim[-1])
|
||||
ulim_full = np.interp(freq, freqlim, ulim,
|
||||
left=ulim[0], right=ulim[-1])
|
||||
|
||||
return bool(np.all(llim_full <= h_dB) and
|
||||
np.all(ulim_full >= h_dB))
|
||||
@ -161,6 +161,128 @@ class FilterBankDesigner:
|
||||
raise ValueError(
|
||||
f'Could not find an x-value corresponding to {nom_txt}.')
|
||||
|
||||
def getxs(self, nom_txt_start, nom_txt_end):
|
||||
"""Returns a list of all filter designators, for given start end end
|
||||
nominal frequencies.
|
||||
|
||||
Args:
|
||||
nom_txt_start: Start frequency band, i.e. '31.5'
|
||||
nom_txt_end: End frequency band, i.e. '10k'
|
||||
|
||||
Returns:
|
||||
[x0, x1, ..]
|
||||
"""
|
||||
xstart = self.nominal_txt_tox(nom_txt_start)
|
||||
xend = self.nominal_txt_tox(nom_txt_end)
|
||||
return list(range(xstart, xend+1))
|
||||
|
||||
def getNarrowBandFromOctaveBand(self, xl, xu,
|
||||
levels_in_bands, npoints=500,
|
||||
method='flat',
|
||||
scale='lin'):
|
||||
"""Create a narrow band spectrum based on a spectrum in (fractional)
|
||||
octave bands. The result is create such that the total energy in each
|
||||
frequency band is constant. The latter can be checked by working it
|
||||
back to (fractional) octave bands, which is doen using the function
|
||||
`getOctaveBandsFromNarrowBand`. Note that the resulting narrow band has
|
||||
units of *power*, not power spectral density. Input should be levels in
|
||||
**deciBells**.
|
||||
|
||||
Args:
|
||||
xl: Band designator of lowest band
|
||||
xu: Band designator of highest band
|
||||
levels_in_bands: levels in dB for each band, should have length
|
||||
(xu+1 - xl)
|
||||
npoints: Number of discrete frequency points in linear frequency
|
||||
array.
|
||||
method: 'flat' to give the same power for all frequencies in a
|
||||
certain band.
|
||||
scale: 'lin' for a linear frequency distribution. 'log' for a
|
||||
logspace. Use 'log' with caution and only if you really know what
|
||||
you are doing!'
|
||||
|
||||
Returns:
|
||||
freq, levels_dB. Where levels_dB is an array of narrow band levels
|
||||
"""
|
||||
# Lowest frequency of all frequencies
|
||||
fll = self.fl(xl)
|
||||
# Highest frequency of all frequencies
|
||||
fuu = self.fu(xu)
|
||||
|
||||
if scale == 'lin':
|
||||
freq = np.linspace(fll, fuu, npoints)
|
||||
elif scale == 'log':
|
||||
freq = np.logspace(np.log10(fll), np.log10(fuu), npoints)
|
||||
else:
|
||||
raise ValueError(f'Invalid scale parameter: {scale}')
|
||||
|
||||
levels_narrow = np.empty_like(freq)
|
||||
|
||||
if method == 'flat':
|
||||
for i, x in enumerate(range(xl, xu + 1)):
|
||||
fl = self.fl(x)
|
||||
fu = self.fu(x)
|
||||
# Find the indices in the frequency array which correspond to the
|
||||
# frequency band x
|
||||
if x != xu:
|
||||
indices_cur = np.where((freq >= fl) & (freq < fu))
|
||||
else:
|
||||
indices_cur = np.where((freq >= fl) & (freq <= fu))
|
||||
|
||||
power_cur = 10**(levels_in_bands[i] / 10)
|
||||
power_narrow = power_cur / indices_cur[0].size
|
||||
level_narrow = 10*np.log10(power_narrow)
|
||||
levels_narrow[indices_cur] = level_narrow
|
||||
|
||||
return freq, levels_narrow
|
||||
else:
|
||||
raise ValueError('Unimplemented interpolation method')
|
||||
|
||||
def getOctaveBandFromNarrowBand(self, freq, levels_narrow):
|
||||
"""Put all level results in a certain frequency band (`binning`), by
|
||||
summing up the power.
|
||||
|
||||
Args:
|
||||
freq: Narrow-band frequency array
|
||||
levels_narrow: Narrow band power levels, should be power, not *PSD*
|
||||
|
||||
Returns:
|
||||
(fm, levels_binned, nom_txt)
|
||||
"""
|
||||
# Find lower frequency xl
|
||||
for x in self.xs:
|
||||
xl = x
|
||||
fl = self.fl(x)
|
||||
if self.fl(x+1) > freq[0]:
|
||||
break
|
||||
|
||||
# Find upper frequency xu
|
||||
for x in self.xs:
|
||||
xu = x
|
||||
fu = self.fu(xu)
|
||||
if fu >= freq[-1]:
|
||||
break
|
||||
|
||||
freq_in_bands = []
|
||||
levels_in_bands = []
|
||||
nom_txt = []
|
||||
|
||||
for x in range(xl, xu+1):
|
||||
fl = self.fl(x)
|
||||
fu = self.fu(x)
|
||||
if x != xu:
|
||||
indices_cur = np.where((freq >= fl) & (freq < fu))
|
||||
else:
|
||||
indices_cur = np.where((freq >= fl) & (freq <= fu))
|
||||
|
||||
power_cur = np.sum(10**(levels_narrow[indices_cur] / 10))
|
||||
levels_in_bands.append(10*np.log10(power_cur))
|
||||
|
||||
nom_txt.append(self.nominal_txt(x))
|
||||
freq_in_bands.append(self.fm(x))
|
||||
|
||||
return np.array(freq_in_bands), np.array(levels_in_bands), nom_txt
|
||||
|
||||
|
||||
class OctaveBankDesigner(FilterBankDesigner):
|
||||
"""Octave band filter designer."""
|
||||
@ -204,9 +326,11 @@ class OctaveBankDesigner(FilterBankDesigner):
|
||||
mininf = -1e300
|
||||
|
||||
if filter_class == 1:
|
||||
lower_limits_pos = [-0.3, -0.4, -0.6, -1.3, -5.0, -5.0] + 4*[mininf]
|
||||
lower_limits_pos = [-0.3, -0.4, -
|
||||
0.6, -1.3, -5.0, -5.0] + 4*[mininf]
|
||||
elif filter_class == 0:
|
||||
lower_limits_pos = [-0.15, -0.2, -0.4, -1.1, -4.5, -4.5] + 4*[mininf]
|
||||
lower_limits_pos = [-0.15, -0.2, -
|
||||
0.4, -1.1, -4.5, -4.5] + 4*[mininf]
|
||||
lower_limits_neg = lower_limits_pos[:]
|
||||
lower_limits_neg.reverse()
|
||||
lower_limits = np.asarray(lower_limits_neg[:-1] + lower_limits_pos)
|
||||
@ -223,7 +347,6 @@ class OctaveBankDesigner(FilterBankDesigner):
|
||||
|
||||
return freqs, lower_limits, upper_limits
|
||||
|
||||
|
||||
def nominal_txt(self, x):
|
||||
"""Returns textual repressentation of corresponding to the nominal
|
||||
frequency."""
|
||||
|
@ -1,86 +0,0 @@
|
||||
#!/usr/bin/python
|
||||
"""
|
||||
Bin narrow band power in octave/third octave band data
|
||||
"""
|
||||
from lasp.filter.bandpass_fir import (OctaveBankDesigner,
|
||||
ThirdOctaveBankDesigner)
|
||||
import numpy as np
|
||||
import warnings
|
||||
|
||||
|
||||
def binPower(freq, narrow_power, band=3, start_band=-16, stop_band=13):
|
||||
"""
|
||||
Apply binning to narrow band frequency domain power results
|
||||
|
||||
Args:
|
||||
freq: Array of frequency indices
|
||||
narrow_power: narrow-band power values in units of [W] or [Pa^2]
|
||||
band: 1, or 3
|
||||
|
||||
Returns:
|
||||
( ['25', '31.5', '40', '50', ... ],
|
||||
[float(power_25), float(power_31p5), ...]) putting NAN values where
|
||||
inaccurate.
|
||||
"""
|
||||
|
||||
if band == 3:
|
||||
designer = ThirdOctaveBankDesigner()
|
||||
elif band == 1:
|
||||
designer = OctaveBankDesigner()
|
||||
else:
|
||||
raise ValueError("Parameter 'Band' should be either '1', or '3'")
|
||||
|
||||
freq = np.copy(freq)
|
||||
narrow_power = np.copy(narrow_power)
|
||||
|
||||
|
||||
# Exact midband, lower and upper frequency limit of each band
|
||||
fm = [designer.fm(x) for x in range(start_band, stop_band+1)]
|
||||
fl = [designer.fl(x) for x in range(start_band, stop_band+1)]
|
||||
fu = [designer.fu(x) for x in range(start_band, stop_band+1)]
|
||||
fex = [designer.nominal_txt(x) for x in range(start_band, stop_band+1)]
|
||||
# print(fl)
|
||||
binned_power = np.zeros(len(fm), dtype=float)
|
||||
## Start: linear interpolation between bins while Parseval is conserved
|
||||
|
||||
# current frequency resolution
|
||||
df_old = freq[1]-freq[0]
|
||||
|
||||
# preferred new frequency resolution
|
||||
df_new = .1
|
||||
|
||||
# ratio of resolutions
|
||||
|
||||
ratio = int(df_old/df_new)
|
||||
# calculate new frequency bins
|
||||
freq_new = np.linspace(freq[0],freq[-1],(len(freq)-1)*ratio+1)
|
||||
|
||||
# calculate the new bin data
|
||||
interp_power = np.interp(freq_new, freq, narrow_power)/ratio
|
||||
|
||||
# adapt first and last bin values so that Parseval still holds
|
||||
interp_power[0] = binned_power[0]*(1.+1./ratio)/2.
|
||||
interp_power[-1] = binned_power[-1]*(1.+1./ratio)/2.
|
||||
|
||||
# check if Parseval still holds
|
||||
# print(np.sum(y, axis=0))
|
||||
# print(np.sum(y_new, axis=0))
|
||||
|
||||
## Stop: linear interpolation between bins while Parseval is conserved
|
||||
binned_power = np.zeros(len(fm), dtype=float)
|
||||
for k in range(len(fm)):
|
||||
# print(k)
|
||||
# find the bins which are in the corresponding band
|
||||
bins = (fl[k] <= freq_new) & (freq_new < fu[k])
|
||||
# print(bins)
|
||||
# sum the output values of these bins to obtain the band value
|
||||
binned_power[k] = np.sum(interp_power[bins], axis=0)
|
||||
# if no frequency bin falls in a certain band, skip previous bands
|
||||
# if not any(bins):
|
||||
# binned_power[0:k+1] = np.nan
|
||||
|
||||
# check if data is valid
|
||||
if(np.isnan(binned_power).all()):
|
||||
warnings.warn('Invalid frequency array, we cannot bin these values')
|
||||
|
||||
return fm, fex, binned_power
|
@ -65,7 +65,9 @@ cdef extern from "numpy/arrayobject.h":
|
||||
cdef extern from "lasp_python.h":
|
||||
object dmat_to_ndarray(dmat*,bint transfer_ownership)
|
||||
|
||||
__all__ = ['AvPowerSpectra', 'SosFilterBank', 'FilterBank']
|
||||
__all__ = ['AvPowerSpectra', 'SosFilterBank', 'FilterBank', 'Siggen',
|
||||
'sweep_flag_forward', 'sweep_flag_backward', 'sweep_flag_linear',
|
||||
'sweep_flag_exponential', 'sweep_flag_hyperbolic']
|
||||
|
||||
setTracerLevel(15)
|
||||
cdef extern from "cblas.h":
|
||||
@ -523,13 +525,27 @@ cdef class SPLowpass:
|
||||
|
||||
cdef extern from "lasp_siggen.h":
|
||||
ctypedef struct c_Siggen "Siggen"
|
||||
us SWEEP_FLAG_FORWARD, SWEEP_FLAG_BACKWARD, SWEEP_FLAG_LINEAR
|
||||
us SWEEP_FLAG_EXPONENTIAL,SWEEP_FLAG_HYPERBOLIC
|
||||
|
||||
c_Siggen* Siggen_Whitenoise_create(d fs, d level_dB)
|
||||
c_Siggen* Siggen_Sinewave_create(d fs, d freq, d level_dB)
|
||||
c_Siggen* Siggen_Sweep_create(d fs, d fl,
|
||||
d fu, d Ts,us sweep_flags,
|
||||
d level_dB)
|
||||
void Siggen_genSignal(c_Siggen*, vd* samples) nogil
|
||||
void Siggen_free(c_Siggen*)
|
||||
|
||||
# Sweep flags
|
||||
sweep_flag_forward = SWEEP_FLAG_FORWARD
|
||||
sweep_flag_backward = SWEEP_FLAG_BACKWARD
|
||||
|
||||
sweep_flag_linear = SWEEP_FLAG_LINEAR
|
||||
sweep_flag_exponential = SWEEP_FLAG_EXPONENTIAL
|
||||
sweep_flag_hyperbolic = SWEEP_FLAG_HYPERBOLIC
|
||||
|
||||
cdef class Siggen:
|
||||
|
||||
cdef c_Siggen *_siggen
|
||||
|
||||
def __cinit__(self):
|
||||
@ -570,3 +586,17 @@ cdef class Siggen:
|
||||
siggen = Siggen()
|
||||
siggen._siggen = c_siggen
|
||||
return siggen
|
||||
|
||||
@staticmethod
|
||||
def sweep(d fs, d fl, d fu, d Ts,us sweep_flags, d level_dB):
|
||||
cdef c_Siggen* c_siggen = Siggen_Sweep_create(fs,
|
||||
fl,
|
||||
fu,
|
||||
Ts,
|
||||
sweep_flags,
|
||||
level_dB)
|
||||
if c_siggen == NULL:
|
||||
raise ValueError('Failed creating signal generator')
|
||||
siggen = Siggen()
|
||||
siggen._siggen = c_siggen
|
||||
return siggen
|
||||
|
Loading…
x
Reference in New Issue
Block a user