-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathvna_measurement_noswitch.hpp
122 lines (95 loc) · 3.04 KB
/
vna_measurement_noswitch.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
#pragma once
#include <mculib/small_function.hpp>
#include "common.hpp"
#include "sample_processor.hpp"
// implements sweep and dsp for no-switch VNAs (dedicated receiver
// for each of reference, reflected, and thru).
// given synthesizer controls and adc data feed, emit a stream
// of data points.
// nChannels is either 2 or 3. 2 => reference and reflect. 3 => reference, reflect, and thru.
template<int nChannels>
class VNAMeasurementNoSwitch {
public:
typedef complex<int32_t> complexi;
// how many periods to wait after changing synthesizer frequency
uint32_t nWaitSynth = 20;
// how many periods to average over
uint32_t nPeriods = 15;
// called when a new data point is available
small_function<void(int freqIndex, uint64_t freqHz, const VNAObservationSet& v)> emitDataPoint;
// called to change synthesizer frequency
small_function<void(uint64_t freqHz)> frequencyChanged;
// sweep params
uint64_t sweepStartHz = 0, sweepStepHz = 0;
// if sweepPoints is 1, sets frequency to startFreqHz and disables sweep
int sweepPoints = 1;
int sweepDataPointsPerFreq = 1;
VNAMeasurementNoSwitch() {}
void init() { }
void resetSweep() {
sweepParamsGeneration++;
}
public:
// state variables
// number of periods so far in this data point
uint32_t periodCounter = 0;
// number of periods left to wait
uint32_t periodCounterSynth = 0;
// number of data points since synthesizer frequency change
uint32_t dpCounterSynth = 0;
// number of frequency points since start of sweep
int sweepCurrPoint = 0;
// current data point variables
complexi currDP[3];
uint64_t currFreq;
uint32_t sweepParamsGeneration = 0;
uint32_t sweepParamsGeneration2 = 0;
void sweepAdvance() {
if(sweepParamsGeneration2 != sweepParamsGeneration) {
// reset sweep
sweepCurrPoint = 0;
sweepParamsGeneration2 = sweepParamsGeneration;
periodCounterSynth = nWaitSynth * 5;
} else {
// advance sweep
sweepCurrPoint++;
if(sweepCurrPoint >= sweepPoints) {
sweepCurrPoint = 0;
periodCounterSynth = nWaitSynth*5;
} else {
periodCounterSynth = nWaitSynth;
}
}
dpCounterSynth = 0;
currFreq = sweepStartHz + sweepStepHz*sweepCurrPoint;
frequencyChanged(currFreq);
}
// call this function when a new value is available
void processValue(int32_t* valRe, int32_t* valIm) {
if(periodCounterSynth > 0) {
periodCounterSynth--;
return;
}
for(int i=0; i<nChannels; i++)
currDP[i] += complexi{valRe[i], valIm[i]};
periodCounter++;
if(periodCounter == nPeriods) {
periodCounter = 0;
// emit new data point
VNAObservationSet value = {
to_complexf(currDP[1]),
to_complexf(currDP[0]),
nChannels >= 3 ? to_complexf(currDP[2]) : 0.};
emitDataPoint(sweepCurrPoint, currFreq, value);
currDP[0] = currDP[1] = currDP[2] = 0;
dpCounterSynth++;
if(dpCounterSynth >= sweepDataPointsPerFreq && sweepPoints > 1) {
dpCounterSynth = 0;
sweepAdvance();
}
}
}
static inline complexf to_complexf(VNAMeasurementNoSwitch::complexi value) {
return {(float) value.real(), (float) value.imag()};
}
};