forked from jbtronics/DS1054_BodePlotter
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbode.py
255 lines (196 loc) · 8.92 KB
/
bode.py
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
# bode.py
# Program to plot bode diagrams using a DS1054Z and a jds6600
# Jan Böhmer (c) 2019
# published under MIT license. See file "LICENSE" for full license text
# from jds6600 import * 1st try
import fygen
import numpy as np
import time
from ds1054z import DS1054Z
import argparse
import dft
import matplotlib.pyplot as plt
import scipy.signal
parser = argparse.ArgumentParser(description="This program plots Bode Diagrams of a DUT using an JDS6600 and Rigol DS1054Z")
parser.add_argument('MIN_FREQ', metavar='min', type=float, help="The minimum frequency for which should be tested")
parser.add_argument('MAX_FREQ', metavar='max', type=float, help="The maximum frequency for which should be tested")
parser.add_argument('COUNT', metavar='N', nargs="?", default=50, type=int, help='The number of frequencies for which should be probed')
parser.add_argument("--awg_port", dest="AWG_PORT", default="/dev/ttyUSB0", help="The serial port where the AWG is connected to")
parser.add_argument("--ds_ip", default="auto", dest="OSC_IP", help="The IP address of the DS1054Z. Set to auto, to auto discover the oscilloscope via Zeroconf")
parser.add_argument("--linear", dest="LINEAR", action="store_true", help="Set this flag to use a linear scale")
parser.add_argument("--awg_voltage", dest="VOLTAGE", default=5, type=float, help="The amplitude of the signal used for the generator")
parser.add_argument("--step_time", dest="TIMEOUT", default=0.00, type=float, help="The pause between to measurements in ms.")
parser.add_argument("--phase", dest="PHASE", action="store_true", help="Set this flag if you want to plot the Phase diagram too")
parser.add_argument("--no_smoothing", dest="SMOOTH", action="store_false", help="Set this to disable the smoothing of the data with a Savitzky–Golay filter")
parser.add_argument("--use_manual_settings", dest="MANUAL_SETTINGS", action="store_true", help="When this option is set, the options on the oscilloscope for voltage and time base are not changed by this program.")
parser.add_argument("--output", dest="file", type=argparse.FileType("w"), help="Write the measured data to the given CSV file.")
parser.add_argument("--no_plots", dest="PLOTS", action="store_false", help="When this option is set no plots are shown. Useful in combination with --output")
parser.add_argument("--normalize", dest="NORMALIZE", action="store_true", help="Set this option if you dont want to get the absolute voltage levels on the output, but the value normalized on the input level.")
parser.add_argument("--use_dft", dest="DFT", action="store_true", help="Use Discrete Fourier Transform on raw data; more accurate but slower.")
args = parser.parse_args()
if args.OSC_IP == "auto":
import ds1054z.discovery
results = ds1054z.discovery.discover_devices()
if not results:
print("No Devices found! Try specifying the IP Address manually.")
exit()
OSC_IP = results[0].ip
print("Found Oscilloscope! Using IP Address " + OSC_IP)
else:
OSC_IP = args.OSC_IP
DEFAULT_PORT = args.AWG_PORT
MIN_FREQ = args.MIN_FREQ
MAX_FREQ = args.MAX_FREQ
STEP_COUNT = args.COUNT
# Do some validity checs
if MIN_FREQ < 0 or MAX_FREQ < 0:
exit("Frequencies has to be greater 0!")
if MIN_FREQ >= MAX_FREQ:
exit("MAX_FREQ has to be greater then min frequency")
if STEP_COUNT <= 0:
exit("The step count has to be positive")
TIMEOUT = args.TIMEOUT
AWG_CHANNEL = 0 # //channel 1
AWG_VOLT = args.VOLTAGE
print("Init AWG")
# awg = jds6600(DEFAULT_PORT)
awg = fygen.FYGen(DEFAULT_PORT)
# AWG_MAX_FREQ = awg.getinfo_devicetype()
AWG_MODEL = awg.get_model()
AWG_MAX_FREQ = float(AWG_MODEL[7:9])
print("Maximum Generator Frequency: %d MHz"% AWG_MAX_FREQ)
if MAX_FREQ > AWG_MAX_FREQ * 1e6:
exit("Your MAX_FREQ is higher than your AWG can achieve!")
# We use sine for sweep
# awg.setwaveform(AWG_CHANNEL, "sine")
awg.set(AWG_CHANNEL, enable=True, wave='sin')
# Init scope
scope = DS1054Z(OSC_IP)
# Set some options for the oscilloscope
if not args.MANUAL_SETTINGS:
# Center vertically
scope.set_channel_offset(1, 0)
scope.set_channel_offset(2, 0)
# Display one period in 2 divs
period = (1/MIN_FREQ) / 2
scope.timebase_scale = period
scope.run()
# Set the sensitivity according to the selected voltage
scope.set_channel_scale(1, args.VOLTAGE / 3, use_closest_match=True)
# Be a bit more pessimistic for the default voltage, because we run into problems if it is too confident
scope.set_channel_scale(2, args.VOLTAGE / 2, use_closest_match=True)
freqs = np.linspace(MIN_FREQ, MAX_FREQ, num=STEP_COUNT)
if not args.LINEAR:
freqs = np.logspace(np.log10(MIN_FREQ), np.log10(MAX_FREQ), num=STEP_COUNT)
else:
freqs = np.linspace(MIN_FREQ, MAX_FREQ, num=STEP_COUNT)
# Set amplitude
awg.set(AWG_CHANNEL, volts=AWG_VOLT, enable=True)
volts = list()
phases = list()
# We have to wait a bit before we measure the first value
awg.set(AWG_CHANNEL, freq_hz=float(freqs[0]), enable=True)
time.sleep(0.05)
if not args.MANUAL_SETTINGS:# initialize voltage reading to see if scope is set in correct vertical scale, in case vout is bigger than vin
scope.display_channel(1, enable=True)
scope.display_channel(2, enable=True)
volt = scope.get_channel_measurement(2, 'vpp')
vscalelist = [0.001, 0.002, 0.005, 0.01, 0.02, 0.05, 0.1, 0.2, 0.5, 1, 2, 5, 10]
scopevscale = scope.get_channel_scale(2)
index = vscalelist.index(scopevscale)
while volt is None: # increase voltage scale until vpp is read
#print("vscale ", vscalelist[index])
scope.set_channel_scale(2, vscalelist[index] , use_closest_match=True)
time.sleep(1)
volt = scope.get_channel_measurement(2, 'vpp')
print("vpp: ", volt)
if index < 9:
index = index + 3
else:
index = 12
for freq in freqs:
awg.set(AWG_CHANNEL, freq_hz=float(freq), enable=True)
time.sleep(TIMEOUT)
if args.DFT:
volt0, volt, phase = dft.measure_with_dft(scope, freq)
else:
volt0 = scope.get_channel_measurement(1, 'vpp')
volt = scope.get_channel_measurement(2, 'vpp')
phase = scope.get_channel_measurement('CHAN1, CHAN2', 'rphase')
if phase:
phase = -1*phase
if not args.NORMALIZE:
volts.append(volt)
else:
if volt0 < 0.01:
print("Input voltage is very low, check your connections and retry")
exit()
else:
volts.append(volt/volt0)
# Use a better timebase
if not args.MANUAL_SETTINGS:
# Display one period in 2 divs
period = (1/freq) / 2
scope.timebase_scale = period
# Use better voltage scale for next time
if volt:
scope.set_channel_scale(2, volt / 2, use_closest_match=True)
else:
scope.set_channel_scale(2, AWG_VOLT / 2, use_closest_match=True)
print(freq, " volt0: ", volt0, " volt: ", volt, " phase: ", phase)
# Write data to file if needed
if args.file:
if args.PHASE:
args.file.write("Frequency in Hz; Amplitude in V; Phase in Degree\n")
else:
args.file.write("Frequency in Hz; Amplitude in V\n")
for n in range(0, len(freqs)):
if volts[n]:
volt = volts[n]
else:
volt = float("nan")
if args.PHASE:
if phases[n]:
phase = phases[n]
else:
phase = phases[n]
args.file.write("%f;%f;%f \n"%(freqs[n], volt, phase))
else:
args.file.write("%f;%f \n"%(freqs[n], volt))
args.file.close()
# Plot graphics
if not args.PLOTS:
exit()
plt.plot(freqs, volts, label="Measured data")
if args.SMOOTH:
try:
yhat = scipy.signal.savgol_filter(volts, 9, 3) # window size 51, polynomial order 3
plt.plot(freqs, yhat, "--", color="red", label="Smoothed data")
except:
print("Error during smoothing amplitude data")
plt.title("Amplitude diagram (N=%d)"%STEP_COUNT)
plt.xlabel("Frequency [Hz]")
plt.ylabel("Voltage Peak-Peak [V]")
plt.legend()
# Set log x axis
if not args.LINEAR:
plt.xscale("log")
plt.show()
if args.PHASE:
try:
plt.plot(freqs, phases)
plt.title("Phase diagram (N=%d)"%STEP_COUNT)
plt.ylabel("Phase [°]")
plt.xlabel("Frequency [Hz]")
except:
print("Phase was not correctly measured, check your connections")
if args.SMOOTH:
try:
yhat = scipy.signal.savgol_filter(phases, 9, 3) # window size 51, polynomial order 3
plt.plot(freqs, yhat, "--", color="red", label="Smoothed data")
except:
print("Error during smoothing phase data")
# Set log x axis
if not args.LINEAR:
plt.xscale("log")
plt.show()