-
Notifications
You must be signed in to change notification settings - Fork 2
/
synthesisDriver.py
373 lines (330 loc) · 15.4 KB
/
synthesisDriver.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
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
"""
Eric Gelphman
UC San Diego Department of Electrical and Computer Engineering
Last Updated March 22, 2020 Version 1.1.5
Driver program for lattice filter synthesis, if __name__ == "__main__"
is here
Required Packages/Files:
-sys
-designFilter
-latticeFilterSynthesis
-numpy
-scipy.signal: freqz (no need to import whole package)
-matplotlib: pyplot (no need to import whole package)
"""
import sys
import designFilter as dF
import latticeFilterSynthesis as lfs
import numpy as np
from scipy.signal import freqz
import matplotlib.pyplot as plt
def calcUnitDelayLength(lamda_start, lamda_end, n_g):
"""
Function to calculate the unit delay length of the filter
Parameters: lamda_start: smallest wavelength in range of interest, in nm
lamda_end: longest wavelength in rane og interest, in nm
n_g: group index
Return: L_U: Unit delay length L_U in um
"""
denom = 2.0*((1.0/lamda_start)-(1.0/lamda_end))*n_g
L_U = 1.0/denom
L_U = L_U*(1E-03)
return L_U
def determineWavelengthEndpoints(center_wvlengths, band_wvlength):
"""
Function to obtain the endpoints of the wavelength interval of interest given the center wavelengths
Parameters: center_wavelengths: ndarray of center wavelengths (for now, at most 2) in format of
[longer wavelength, shorter wavelength]
band_wvlength: Length, in nm, of pass/stop bands
Return: nd array [lambda0, lambda1] where
lambda0: longest wavelength (smallest frequency) in interval of interest
lambda1: smallest wavelength (largest frequency) in interval of interest
"""
lambda0 = 1565#nm
lambda1 = 1520#nm
PI = np.pi
n_bands = center_wvlengths.size
if n_bands == 1:
lambda0 = center_wvlengths[0] + (band_wvlength/2.0) + 23
lambda1 = center_wvlengths[0] - (band_wvlength/2.0) - 23
else:
lambda0 = center_wvlengths[0] + (band_wvlength/2.0) + 10
lambda1 = center_wvlengths[1] - (band_wvlength/2.0) - 10
return np.array([lambda0, lambda1])
def receiveFilterParametersFile(filename):
"""
Function to recieve the necessary parameters from an input file to design the filter
Parameters: filename: Name of input file
Return: Formatted list params containing needed values:
params[0]: Group index n_g
params[1]: l_c in um
params[2]: l_c in um
params[3]: L_2 in um
params[4]: 'Pass' or 'Stop', indicating filter type
params[5]: Number of pass or stop bands
params[6]: First (longer) center wavelength
params[7]: Second (shorter) center wavelength
params[8]: Pass/Stop band width, in nm
params[9]: Max. stopband attenuation, in dB
params[10]: Filter order
"""
file1 = open(filename, 'r+')
params = []
text = file1.readlines()
if len(text) < 8:
print("Error! Not Enough Input Parameters!")
exit()
params.append(float(text[0].split('//')[0]))#Group index n_g
line1 = text[1].split('//')[0].split(',')
params.append(float(line1[0]))#l_c
params.append(float(line1[1]))#l_end
params.append(float(line1[2]))#L_2
line2 = text[2].split('//')[0]
params.append(line2)#Pass or Stop, indicating filter type
num_bands = int(text[3].split('//')[0])#Number of bands (1 or 2 for now)
params.append(num_bands)
if num_bands == 1:
params.append(float(text[4].split('//')[0]))
elif num_bands == 2:
line4 = text[4].split('//')[0].split(',')
params.append(float(line4[0]))#First center wavelength
params.append(float(line4[1]))#Second center wavelength
params.append(float(text[5].split('//')[0]))#Pass/Stop band width (in nm)
params.append(float(text[6].split('//')[0]))#Max. stopband attenuation
params.append(int(text[7].split('//')[0]))#Filter order
return params
def processFilterParametersFile(params):
"""
Function to determine necessary values to plug into the filter design functions given the
input parameters. This is to be called after recieveFilterParamtersFile.
Ex.: params = recieveFilterParametersFile(filename)
L_U, lengths, bands, max_order, filter_type, endpoints, atten, center_wvlengths, band_wvlength = processFilterParametersFile(params)
Paramters: params: Formatted list of values read from file
Return: L_U: Unit delay length in um
lengths: nd array [lc, lend, L2] these are lengths needed for layout
bands: List of passbands of filter
N: Order of filter
filter_type: 'Pass' or 'Stop'
endpoints: nd array of endpoints of desired wavelength interval [lambda_0, lambda_1] lambda_1
corresponds to larger frequency
atten: Max. stopband attenuation in dB
center_wvlengths: nd array [lamda_0, lamda_1] lamda_0 is the longer wavelength, in nm, of centers of pass/stop bands
band_wvlength: Width, in nm, of pass/stop bands
"""
PI = np.pi
lc = params[1]
lend = params[2]
L2 = params[3]
lengths = np.array([lc,lend,L2])
filter_type = params[4]#Pass or Stop
n_bands = params[5]
center_wvlengths = np.zeros(1)#Wavelength, in nm, of centers of pass/stop bands
N = 0#filter order
L_U = band_wvlength = atten = 0.0#L_U, pass/stop band wavelength in nanometers, attenuation in dB
endpoints = np.zeros(2)
if n_bands == 1:
band_wvlength = params[7]
endpoints = determineWavelengthEndpoints(np.array([params[6]]), band_wvlength)
L_U = calcUnitDelayLength(endpoints[1], endpoints[0], params[0])
center_wvlengths[0] = params[6]
atten = params[8]
N = params[9]
else:
band_wvlength = params[8]
endpoints = determineWavelengthEndpoints(np.array([params[6],params[7]]), band_wvlength)
L_U = calcUnitDelayLength(endpoints[1], endpoints[0], params[0])
center_wvlengths = np.array([params[6], params[7]])
atten = params[9]
N = params[10]
bands = dF.obtainBandsFromCenterWvlengths(center_wvlengths, endpoints, band_wvlength, filter_type)
return L_U, lengths, bands, N, filter_type, endpoints, atten, center_wvlengths, band_wvlength
def processFilterParametersTable(tableFile):
"""
Function to process data held in a file that contains a table of the wavelengths and attenuation(in dB) at those wavelengths
Parameters: tableFile: Name of file that holds the table
Return: L_U: Unit delay length in um
lengths: nd array [lc, lend, L2] these are lengths needed for layout
bands: List of passbands of filter
N: Order of filter
filter_type: 'Pass' or 'Stop'
endpoints: nd array of endpoints of desired wavelength interval [lambda_0, lambda_1] lambda_1
corresponds to larger frequency
atten: Max. stopband attenuation in dB
center_wvlengths: nd array [lamda_0, lamda_1] lamda_0 is the longer wavelength, in nm, of centers of pass/stop bands
band_wvlength: Width, in nm, of pass/stop bands
"""
file1 = open(tableFile, 'r+')
text = file1.readlines()
n_g = float(text[0].split('//')[0])
line1 = text[1].split('//')[0].split(',')
lc = float(line1[0])
lend = float(line1[1])
L2 = float(line1[2])
lengths = np.array([lc, lend, L2])
filter_type = text[2].split('//')[0]
num_bands = int(text[3].split('//')[0])
center_wvlengths = np.array([0.0])
if num_bands == 1:
center_wvlengths[0] = float(text[4].split('//')[0])
elif num_bands == 2:
vals = text[4].split('//')[0].split(',')
center_wvlengths = np.array([float(vals[0]), float(vals[1])])
band_wvlength = float(text[5].split('//')[0])
atten = float(text[6].split('//')[0])
N = int(text[7].split('//')[0])
lambda1 = float(text[len(text)-1].strip('\n').split(',')[0])#Longest wavelength/lowest frequency
lambda0 = float(text[8].strip('\n').split(',')[0])#Shortest wavelength/highest frequency
table = np.zeros((len(text)-8,2))
last_idx = len(text)-1
for ii in range(8, len(text)):
line = text[ii].strip('\n').split(',')#Format is wavlength(nm),magnitude(dB)
table[last_idx-ii][0] = dF.convertToNormalizedFrequency(lambda1, lambda0, float(line[0]))
table[last_idx-ii][1] = 10**(float(line[1])/20.0)#Convert from dB to linear scale
endpoints = np.array([lambda1, lambda0])
bands = dF.obtainBandsFromCenterWvlengths(center_wvlengths, np.array([lambda1, lambda0]), band_wvlength, filter_type)
L_U = calcUnitDelayLength(lambda0, lambda1, n_g)
return L_U, lengths, bands, N, filter_type, endpoints, atten, center_wvlengths, band_wvlength, table
def writeLayoutParametersToFile(kappalcs, phis, L_U, L_2, filename, order, method):
"""
Function to write the layout paramters to a file
Parameters: kappalcs:
phis: List of phase terms phi_n for each stage
L_U: Unit delay length in um
L_2: Length of section that doesn't include delay in the MZI
filename: Name of output file
order: Order of filter
method: Method used to design filter 'Kaiser' or 'Parks-McClellan'
Return: None
"""
file1 = open(filename, 'r+')#open file
file1.write("% Filter Design Method: " + method + " Filter Order: " + str(order)+ "\n")
file1.write("% Unit Delay Length: " + str(L_U) + " um" + "\n")
alpha = 1E-04 # db per um
gamma = 10**((-alpha*L_2)/20.0)
file1.write("% Coupler Length L_2: " + str(L_2) + " um" + "\n")
file1.write("% Loss Coefficient Gamma per Stage: " + str(gamma) + "\n")
for ii in range(len(kappalcs)):#Write parameters to file
if ii != 0:
file1.write(f"{kappalcs[ii][0]:.6e},{kappalcs[ii][1]:.6e},{phis[ii-1]:6f},\n")
else:
file1.write(f"{kappalcs[ii][0]:.6e},{kappalcs[ii][1]:.6e},\n")
file1.close()#close file
def graphTLambda(A_N, band_wvlength, endpoints, center_wvlengths, atten, filter_type):
"""
Function to graph the transmission of the filter as a function of wavelength
Parameters: A_N: coefficient array of polynomial A_N(z) that is the transfer function of the filter
endpoints: endpoints of wavelength interval of interest
Return: None
"""
PI = np.pi
w, h = freqz(np.flip(A_N))#Need to flip according to numpy/scipy documentation, remember the coef. of term with
#highest power of z^-1 occupies index 0 of array-see ECE 161B notes
c = 3.0E8
f1 = c/endpoints[0]
f0 = c/endpoints[1]
wvlength = np.zeros(w.size)
for ii in range(w.size):
denom = (1.0/PI)*(w[ii]*(f1-f0)) + f0
wvlength[ii] = c/denom
plt.title('MA/FIR filter frequency response')
plt.plot(np.flip(wvlength), 20*np.log10(abs(h)), color='b', label='Realized Frequency Response')
#Build plot of "Ideal" T(lambda)
band_edges = []
gain_levels = []
x_min = []
x_max = []
y_min = []
y_max = []
#Region before first pass/stop band
x_min.append(endpoints[1])
x_max.append(center_wvlengths[center_wvlengths.size-1]-(band_wvlength/2.0))
if filter_type == 'Pass':
gain_levels.append(-1*atten)
else:
gain_levels.append(0)
#Pass/Stop Bands
for ii in range(center_wvlengths.size):
band_edges.append(center_wvlengths[ii]-(band_wvlength/2.0))
x_min.append(center_wvlengths[ii]-(band_wvlength/2.0))
band_edges.append(center_wvlengths[ii]+(band_wvlength/2.0))
x_max.append(center_wvlengths[ii]+(band_wvlength/2.0))
if filter_type == 'Pass':
gain_levels.append(0)
if ii != 0:
x_min.append(center_wvlengths[ii]+(band_wvlength/2.0))
gain_levels.append(-1*atten)
x_max.append(center_wvlengths[ii-1]-(band_wvlength/2.0))
else:
gain_levels.append(-1*atten)
if ii != 0:
x_min.append(center_wvlengths[ii]+(band_wvlength/2.0))
gain_levels.append(0)
x_max.append(center_wvlengths[ii-1]-(band_wvlength/2.0))
y_min.append(-1*atten)
y_min.append(-1*atten)
y_max.append(0)
y_max.append(0)
#Region after pass/stop bands
x_min.append(center_wvlengths[0]+(band_wvlength/2.0))
x_max.append(endpoints[0])
if filter_type == 'Pass':
gain_levels.append(-1*atten)
else:
gain_levels.append(0)
plt.vlines(band_edges, y_min, y_max, colors='g', label='Ideal Frequency Response')
plt.hlines(gain_levels, x_min, x_max, colors ='g')
plt.legend()
plt.ylabel('Transmission [dB]', color='b')
plt.xlabel('Wavelength [um]')
plt.show()
def main():
num_args = len(sys.argv)
if num_args < 4 or num_args > 5:
print("Error! Not enough or too many args, format should by synthesisDriver.py [-l or -t] [inputFileName] [outputFileName] [Kaiser or Parks-McClellan if l]")
print("For a total of 4 args if t, or 5 args if l")
exit()
inputFileType = sys.argv[1]#Indicates inpur file type, either list (-l) or table (-t)
inputFileName = sys.argv[2]#Input file name
outputFileName = sys.argv[3]#Output file name
#Declare needed variables
L_U = 0.0
atten = 0.0
band_wvlength = 0.0
lengths = np.zeros(3)
bands = []
N = 0
filter_Type = ''
method = ''
endpoints = np.zeros(2)
center_wvlengths = np.zeros(1)
table = []
A_N = np.zeros(1)
t_width = 0.0
#Design the filter
if inputFileType == '-l':#Input file is in list format
params = receiveFilterParametersFile(inputFileName)
method = sys.argv[4]
L_U, lengths, bands, N, filter_type, endpoints, atten, center_wvlengths, band_wvlength = processFilterParametersFile(params)
del table
bands = dF.obtainBandsFromCenterWvlengths(center_wvlengths, endpoints, band_wvlength, filter_type)
A_N, order, t_width = dF.designFIRFilter(N, bands, atten, filter_type, method)#Design filter
elif inputFileType == '-t':#Input file is in table format
L_U, lengths, bands, N, filter_type, endpoints, atten, center_wvlengths, band_wvlength, table = processFilterParametersTable(inputFileName)
method = 'Kaiser'
bands = dF.obtainBandsFromCenterWvlengths(center_wvlengths, endpoints, band_wvlength, filter_type)
A_N, order, t_width = dF.designFIRFilter(N, bands, atten, filter_type, method, table)#Design filter
if A_N[0] > 0.0:
A_N = -1.0*A_N#Synthesis algorithm works best if leading coefficient is negative
A_z = np.poly1d(A_N)
if filter_type == 'Stop':
passbands = dF.obtainPassbandsFromStopbands(bands, t_width)
bands = passbands
#Synthesize the filter using the lattice architecture
fac = lengths[2]*1E-04#Convert alpha to dB/um as L_2 is in um
gamma = 10**(-1*fac/20)#Loss coefficient per stage, depends on length L_2 of MZI stage
kappalcs, phis, B_N = lfs.synthesizeFIRLattice(A_N, order, gamma)
#Write layout parameters to the file, graph filter frequency response
writeLayoutParametersToFile(kappalcs, phis, L_U, 25.0, outputFileName, order, method)
graphTLambda(A_N, band_wvlength, endpoints, center_wvlengths, atten, filter_type)
if __name__ == '__main__':
main()