FERS 0.1.0
The Flexible Extensible Radar Simulator
Loading...
Searching...
No Matches
fers_signal::FmcwChirpSignal Class Referencefinal

FMCW linear chirp signal implementation. More...

#include "radar_signal.h"

+ Inheritance diagram for fers_signal::FmcwChirpSignal:
+ Collaboration diagram for fers_signal::FmcwChirpSignal:

Public Member Functions

 FmcwChirpSignal (RealType chirp_bandwidth, RealType chirp_duration, RealType chirp_period, RealType start_frequency_offset=0.0, std::optional< std::size_t > chirp_count=std::nullopt, FmcwChirpDirection direction=FmcwChirpDirection::Up)
 Constructs an FMCW chirp signal with timing and sweep parameters.
 
 ~FmcwChirpSignal () override=default
 
 FmcwChirpSignal (const FmcwChirpSignal &) noexcept=delete
 
FmcwChirpSignaloperator= (const FmcwChirpSignal &) noexcept=delete
 
 FmcwChirpSignal (FmcwChirpSignal &&) noexcept=delete
 
FmcwChirpSignaloperator= (FmcwChirpSignal &&) noexcept=delete
 
RealType getChirpBandwidth () const noexcept
 Gets the chirp bandwidth in hertz.
 
RealType getChirpDuration () const noexcept
 Gets the chirp duration in seconds.
 
RealType getChirpPeriod () const noexcept
 Gets the chirp period in seconds.
 
RealType getStartFrequencyOffset () const noexcept
 Gets the start frequency offset relative to carrier in hertz.
 
const std::optional< std::size_t > & getChirpCount () const noexcept
 Gets the optional finite chirp count.
 
RealType getChirpRate () const noexcept
 Gets the chirp rate in hertz per second.
 
RealType getSignedChirpRate () const noexcept
 Gets the signed chirp rate in hertz per second.
 
FmcwChirpDirection getDirection () const noexcept
 Gets the FMCW sweep direction.
 
bool isDownChirp () const noexcept
 Returns true when this chirp sweeps downward.
 
bool isTriangle () const noexcept
 Returns false for linear chirps; triangles override this shape predicate.
 
std::optional< std::size_t > activeChirpIndexAt (RealType time_since_segment_start) const noexcept
 Returns the active chirp index for a time since the segment start.
 
bool isActiveAt (RealType time_since_segment_start) const noexcept
 Returns true when the signal is inside an active chirp at the specified time.
 
RealType basebandPhaseForChirpTime (RealType chirp_time) const noexcept
 Computes baseband phase for a time inside a chirp.
 
std::optional< RealTypeinstantaneousBasebandPhase (RealType time_since_segment_start) const noexcept
 Computes instantaneous baseband phase at a time since segment start.
 
std::vector< ComplexTyperender (const std::vector< interp::InterpPoint > &points, unsigned &size, RealType fracWinDelay) const override
 Renders an FMCW waveform from interpolation points.
 
bool isFmcwFamily () const noexcept override
 Returns true when this signal belongs to the FMCW waveform family.
 
void clear () noexcept
 Clears the internal signal data.
 
void load (std::span< const ComplexType > inData, unsigned samples, RealType sampleRate)
 Loads complex radar waveform data.
 
RealType getRate () const noexcept
 Gets the sample rate of the signal.
 
unsigned getSampleCount () const noexcept
 Gets the number of native samples held by this signal.
 
virtual std::vector< ComplexTyperenderSlice (const std::vector< interp::InterpPoint > &points, RealType outputStartTime, RealType outputSampleRate, std::size_t sampleCount, RealType fracWinDelay) const
 Renders a bounded absolute-time slice on the requested output grid.
 

Detailed Description

FMCW linear chirp signal implementation.

Definition at line 307 of file radar_signal.h.

Constructor & Destructor Documentation

◆ FmcwChirpSignal() [1/3]

fers_signal::FmcwChirpSignal::FmcwChirpSignal ( RealType  chirp_bandwidth,
RealType  chirp_duration,
RealType  chirp_period,
RealType  start_frequency_offset = 0.0,
std::optional< std::size_t >  chirp_count = std::nullopt,
FmcwChirpDirection  direction = FmcwChirpDirection::Up 
)

Constructs an FMCW chirp signal with timing and sweep parameters.

Definition at line 55 of file radar_signal.cpp.

57 :
58 _chirp_bandwidth(chirp_bandwidth), _chirp_duration(chirp_duration), _chirp_period(chirp_period),
59 _start_frequency_offset(start_frequency_offset), _chirp_count(chirp_count),
60 _chirp_rate(chirp_bandwidth / chirp_duration), _direction(direction)
61 {
62 }
math::Vec3 max

◆ ~FmcwChirpSignal()

fers_signal::FmcwChirpSignal::~FmcwChirpSignal ( )
overridedefault

◆ FmcwChirpSignal() [2/3]

fers_signal::FmcwChirpSignal::FmcwChirpSignal ( const FmcwChirpSignal )
deletenoexcept

◆ FmcwChirpSignal() [3/3]

fers_signal::FmcwChirpSignal::FmcwChirpSignal ( FmcwChirpSignal &&  )
deletenoexcept

Member Function Documentation

◆ activeChirpIndexAt()

std::optional< std::size_t > fers_signal::FmcwChirpSignal::activeChirpIndexAt ( RealType  time_since_segment_start) const
noexcept

Returns the active chirp index for a time since the segment start.

Definition at line 65 of file radar_signal.cpp.

66 {
68 {
69 return std::nullopt;
70 }
71
72 const auto chirp_index = static_cast<std::size_t>(std::floor(time_since_segment_start / _chirp_period));
73 if (_chirp_count.has_value() && chirp_index >= *_chirp_count)
74 {
75 return std::nullopt;
76 }
77
78 const RealType chirp_time = time_since_segment_start - static_cast<RealType>(chirp_index) * _chirp_period;
79 // Exact arithmetic cannot make this negative, but floating-point boundary rounding can.
80 if (chirp_time < 0.0 || chirp_time >= _chirp_duration)
81 {
82 return std::nullopt;
83 }
84
85 return chirp_index;
86 }
double RealType
Type for real numbers.
Definition config.h:27

References max.

◆ basebandPhaseForChirpTime()

RealType fers_signal::FmcwChirpSignal::basebandPhaseForChirpTime ( RealType  chirp_time) const
noexcept

Computes baseband phase for a time inside a chirp.

Definition at line 101 of file radar_signal.cpp.

102 {
103 return 2.0 * PI * _start_frequency_offset * chirp_time + PI * getSignedChirpRate() * chirp_time * chirp_time;
104 }
RealType getSignedChirpRate() const noexcept
Gets the signed chirp rate in hertz per second.
constexpr RealType PI
Mathematical constant π (pi).
Definition config.h:43

References max, and PI.

◆ clear()

void fers_signal::Signal::clear ( )
noexceptinherited

Clears the internal signal data.

Definition at line 232 of file radar_signal.cpp.

233 {
234 _size = 0;
235 _rate = 0;
236 }

Referenced by fers_signal::Signal::load().

+ Here is the caller graph for this function:

◆ getChirpBandwidth()

RealType fers_signal::FmcwChirpSignal::getChirpBandwidth ( ) const
noexcept

Gets the chirp bandwidth in hertz.

Definition at line 326 of file radar_signal.h.

326{ return _chirp_bandwidth; }

◆ getChirpCount()

const std::optional< std::size_t > & fers_signal::FmcwChirpSignal::getChirpCount ( ) const
noexcept

Gets the optional finite chirp count.

Definition at line 338 of file radar_signal.h.

338{ return _chirp_count; }

◆ getChirpDuration()

RealType fers_signal::FmcwChirpSignal::getChirpDuration ( ) const
noexcept

Gets the chirp duration in seconds.

Definition at line 329 of file radar_signal.h.

329{ return _chirp_duration; }

Referenced by serial::fmcw_validation::validateSchedule().

+ Here is the caller graph for this function:

◆ getChirpPeriod()

RealType fers_signal::FmcwChirpSignal::getChirpPeriod ( ) const
noexcept

Gets the chirp period in seconds.

Definition at line 332 of file radar_signal.h.

332{ return _chirp_period; }

Referenced by serial::fmcw_validation::validateSchedule().

+ Here is the caller graph for this function:

◆ getChirpRate()

RealType fers_signal::FmcwChirpSignal::getChirpRate ( ) const
noexcept

Gets the chirp rate in hertz per second.

Definition at line 341 of file radar_signal.h.

341{ return _chirp_rate; }

◆ getDirection()

FmcwChirpDirection fers_signal::FmcwChirpSignal::getDirection ( ) const
noexcept

Gets the FMCW sweep direction.

Definition at line 350 of file radar_signal.h.

350{ return _direction; }

◆ getRate()

RealType fers_signal::Signal::getRate ( ) const
noexceptinherited

Gets the sample rate of the signal.

Returns
The sample rate of the signal.

Definition at line 86 of file radar_signal.h.

86{ return _rate; }

◆ getSampleCount()

unsigned fers_signal::Signal::getSampleCount ( ) const
noexceptinherited

Gets the number of native samples held by this signal.

Definition at line 89 of file radar_signal.h.

89{ return _size; }

◆ getSignedChirpRate()

RealType fers_signal::FmcwChirpSignal::getSignedChirpRate ( ) const
noexcept

Gets the signed chirp rate in hertz per second.

Definition at line 344 of file radar_signal.h.

345 {
346 return isDownChirp() ? -_chirp_rate : _chirp_rate;
347 }
bool isDownChirp() const noexcept
Returns true when this chirp sweeps downward.

◆ getStartFrequencyOffset()

RealType fers_signal::FmcwChirpSignal::getStartFrequencyOffset ( ) const
noexcept

Gets the start frequency offset relative to carrier in hertz.

Definition at line 335 of file radar_signal.h.

335{ return _start_frequency_offset; }

◆ instantaneousBasebandPhase()

std::optional< RealType > fers_signal::FmcwChirpSignal::instantaneousBasebandPhase ( RealType  time_since_segment_start) const
noexcept

Computes instantaneous baseband phase at a time since segment start.

Definition at line 89 of file radar_signal.cpp.

90 {
92 if (!chirp_index.has_value())
93 {
94 return std::nullopt;
95 }
96
97 const RealType chirp_time = time_since_segment_start - static_cast<RealType>(*chirp_index) * _chirp_period;
99 }
RealType basebandPhaseForChirpTime(RealType chirp_time) const noexcept
Computes baseband phase for a time inside a chirp.
std::optional< std::size_t > activeChirpIndexAt(RealType time_since_segment_start) const noexcept
Returns the active chirp index for a time since the segment start.

References max.

◆ isActiveAt()

bool fers_signal::FmcwChirpSignal::isActiveAt ( RealType  time_since_segment_start) const
noexcept

Returns true when the signal is inside an active chirp at the specified time.

Definition at line 362 of file radar_signal.h.

363 {
365 }

References max.

◆ isDownChirp()

bool fers_signal::FmcwChirpSignal::isDownChirp ( ) const
noexcept

Returns true when this chirp sweeps downward.

Definition at line 353 of file radar_signal.h.

353{ return _direction == FmcwChirpDirection::Down; }
@ Down
Instantaneous baseband frequency decreases over the chirp.

References fers_signal::Down.

◆ isFmcwFamily()

bool fers_signal::FmcwChirpSignal::isFmcwFamily ( ) const
overridevirtualnoexcept

Returns true when this signal belongs to the FMCW waveform family.

Reimplemented from fers_signal::Signal.

Definition at line 378 of file radar_signal.h.

378{ return true; }

◆ isTriangle()

bool fers_signal::FmcwChirpSignal::isTriangle ( ) const
noexcept

Returns false for linear chirps; triangles override this shape predicate.

Definition at line 356 of file radar_signal.h.

356{ return false; }

◆ load()

void fers_signal::Signal::load ( std::span< const ComplexType inData,
unsigned  samples,
RealType  sampleRate 
)
inherited

Loads complex radar waveform data.

Parameters
inDataThe input span of complex signal data.
samplesThe number of samples in the input data.
sampleRateThe sample rate of the input data.

Definition at line 238 of file radar_signal.cpp.

239 {
240 clear();
241 const unsigned ratio = params::oversampleRatio();
242 const auto oversampled_samples = static_cast<std::size_t>(samples) * static_cast<std::size_t>(ratio);
243 if (oversampled_samples > std::numeric_limits<unsigned>::max())
244 {
245 throw std::overflow_error("Oversampled signal sample count exceeds unsigned range");
246 }
247 _data.resize(oversampled_samples);
248 _size = static_cast<unsigned>(oversampled_samples);
249 _rate = sampleRate * static_cast<RealType>(ratio);
250
251 if (ratio == 1)
252 {
253 std::ranges::copy(inData, _data.begin());
254 }
255 else
256 {
257 upsample(inData, samples, _data);
258 }
259 }
void clear() noexcept
Clears the internal signal data.
void upsample(const std::span< const ComplexType > in, const unsigned size, std::span< ComplexType > out)
Upsamples a complex waveform with zero-stuffing followed by Blackman FIR filtering.
unsigned oversampleRatio() noexcept
Get the oversampling ratio.
Definition parameters.h:151

References fers_signal::Signal::clear(), max, params::oversampleRatio(), and fers_signal::upsample().

+ Here is the call graph for this function:

◆ operator=() [1/2]

FmcwChirpSignal & fers_signal::FmcwChirpSignal::operator= ( const FmcwChirpSignal )
deletenoexcept

◆ operator=() [2/2]

FmcwChirpSignal & fers_signal::FmcwChirpSignal::operator= ( FmcwChirpSignal &&  )
deletenoexcept

◆ render()

std::vector< ComplexType > fers_signal::FmcwChirpSignal::render ( const std::vector< interp::InterpPoint > &  points,
unsigned size,
RealType  fracWinDelay 
) const
overridevirtual

Renders an FMCW waveform from interpolation points.

Reimplemented from fers_signal::Signal.

Definition at line 106 of file radar_signal.cpp.

108 {
109 size = 0;
110 return {};
111 }

References max.

◆ renderSlice()

std::vector< ComplexType > fers_signal::Signal::renderSlice ( const std::vector< interp::InterpPoint > &  points,
RealType  outputStartTime,
RealType  outputSampleRate,
std::size_t  sampleCount,
RealType  fracWinDelay 
) const
virtualinherited

Renders a bounded absolute-time slice on the requested output grid.

Definition at line 272 of file radar_signal.cpp.

275 {
276 auto out = std::vector<ComplexType>(sampleCount);
277 if (_size == 0 || _rate <= 0.0 || outputSampleRate <= 0.0 || points.empty())
278 {
279 return out;
280 }
281
282 const RealType timestep = 1.0 / outputSampleRate;
283 const int filt_length = static_cast<int>(params::renderFilterLength());
285
286 auto iter = points.begin();
287 auto next = points.size() > 1 ? std::next(iter) : iter;
288 const RealType idelay = std::round(_rate * iter->delay);
290
291 for (std::size_t i = 0; i < sampleCount; ++i)
292 {
293 while (sample_time > next->time && next != iter)
294 {
295 iter = next;
296 if (std::next(next) != points.end())
297 {
298 ++next;
299 }
300 else
301 {
302 break;
303 }
304 }
305
306 auto [amplitude, phase, fdelay, i_sample_unwrap] =
307 calculateWeightsAndDelays(iter, next, sample_time, idelay, fracWinDelay);
308 const RealType native_position = (sample_time - points.front().time) * _rate;
309 const auto source_index = static_cast<int>(std::floor(native_position));
312 {
313 source_fraction = 0.0;
314 }
315
317 const auto delay_unwrap = static_cast<int>(std::floor(combined_delay));
318 fdelay = combined_delay - static_cast<RealType>(delay_unwrap);
320
321 const auto& filt = interp.getFilter(fdelay);
322 const ComplexType accum =
323 performConvolution(source_index, filt.data(), filt_length, amplitude, i_sample_unwrap);
324 out[i] = std::exp(ComplexType(0.0, 1.0) * phase) * accum;
325
327 }
328
329 return out;
330 }
static InterpFilter & getInstance() noexcept
Retrieves the singleton instance of the InterpFilter class.
std::complex< RealType > ComplexType
Type for complex numbers.
Definition config.h:35
unsigned renderFilterLength() noexcept
Get the render filter length.
Definition parameters.h:139

References interp::InterpFilter::getInstance(), max, and params::renderFilterLength().

Referenced by fers_signal::Signal::render().

+ Here is the call graph for this function:
+ Here is the caller graph for this function:

The documentation for this class was generated from the following files: