forked from telegraphic/21cmSense
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgenerate_array.py
153 lines (128 loc) · 5.04 KB
/
generate_array.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
"""
# generate_array.py
Utilities for generating antenna array geometries. Use to
"""
import aipy as a
import numpy as np
import pylab as plt
import os
class AntennaArray(a.pol.AntennaArray):
def __init__(self, *args, **kwargs):
a.pol.AntennaArray.__init__(self, *args, **kwargs)
self.array_params = {}
def get_ant_params(self, ant_prms={'*':'*'}):
prms = a.fit.AntennaArray.get_params(self, ant_prms)
for k in ant_prms:
top_pos = np.dot(self._eq2zen, self[int(k)].pos)
if ant_prms[k] == '*':
prms[k].update({'top_x':top_pos[0], 'top_y':top_pos[1], 'top_z':top_pos[2]})
else:
for val in ant_prms[k]:
if val == 'top_x': prms[k]['top_x'] = top_pos[0]
elif val == 'top_y': prms[k]['top_y'] = top_pos[1]
elif val == 'top_z': prms[k]['top_z'] = top_pos[2]
return prms
def set_ant_params(self, prms):
changed = a.fit.AntennaArray.set_params(self, prms)
for i, ant in enumerate(self):
ant_changed = False
top_pos = np.dot(self._eq2zen, ant.pos)
try:
top_pos[0] = prms[str(i)]['top_x']
ant_changed = True
except(KeyError): pass
try:
top_pos[1] = prms[str(i)]['top_y']
ant_changed = True
except(KeyError): pass
try:
top_pos[2] = prms[str(i)]['top_z']
ant_changed = True
except(KeyError): pass
if ant_changed: ant.pos = np.dot(np.linalg.inv(self._eq2zen), top_pos)
changed |= ant_changed
return changed
def get_arr_params(self):
return self.array_params
def set_arr_params(self, prms):
for param in prms:
self.array_params[param] = prms[param]
return self.array_params
def view(self):
ant_pos = []
for ant in self.ants:
ant_pos.append(ant.pos)
plot_antenna_array(ant_pos)
def generate_hexagon(n_side, l, scale_factor):
""" Generate antennas configured in an hexagonal array.
Parameters:
nside (int): Number of antennas on each side of the hexagon
l (float): Spacing between antennas in meters
scale_fac (float): scale factor between rows of antennas
Returns:
antpos: a numpy array with shape (n_ant, 3), giving x,y,z
coords of antennas in meters
"""
dl = l * scale_factor #close packed hex
antpos = []
cen_y, cen_z = 0, 0
for row in np.arange(n_side):
for cen_x in np.arange((2 * n_side - 1) - row):
dx = row / 2.0
antpos.append(((cen_x + dx) * l, row * dl, cen_z))
if row != 0:
antpos.append(((cen_x + dx) * l, -row * dl, cen_z))
return np.array(antpos)
def plot_antenna_array(antpos):
""" Plot an antenna array (2D X-Y positions). """
antpos = np.array(antpos)
plt.plot(antpos[:,0], antpos[:, 1], 'o', c='#333333')
plt.xlabel('x-pos [m]')
plt.ylabel('y-pos [m]')
#plt.xlim(np.min(antpos[:, 0]) - 2, np.max(antpos[:, 0]) + 2)
#plt.ylim(np.min(antpos[:, 1]) - 2, np.max(antpos[:, 1]) + 2)
plt.minorticks_on()
plt.savefig("figures/antenna-positions.pdf")
plt.show()
def save_antenna_array(antpos, filename_out, directory_out="array_geometries"):
""" Save an antenna array geometry to a CSV file.
Args:
antpos: Numpy array of antenna xyz positions, shape (n_ant, 3)
filename_out: Name of output file, e.g. lwa_ovro.txt
directory_out: Output directory. Defaults to array_geometries/
"""
fout = os.path.join(directory_out, filename_out)
np.savetxt(fout, antpos, header="antpos_x antpos_y antpos_z")
def load_antenna_array(filename, directory="array_geometries"):
"""Load an antenna array from file."""
fin = os.path.join(directory, filename)
ant_pos = np.genfromtxt(fin)
return ant_pos
def generate_aa(arr_pos, ant_pos, params):
"""Return the AntennaArray to be used for simulation."""
antennas = []
n_ants = len(ant_pos)
for i in range(n_ants):
beam = a.fit.Beam(np.array([1.0])) # Beam is computed at 100 MHz. Beam object is used for UVW calcs
# Which are returned in wavelength (I think)
antennas.append(a.fit.Antenna(0, 0, 0, beam))
aa = AntennaArray(arr_pos, antennas)
p = {}
for i in range(n_ants):
top_pos = ant_pos[i]
p[str(i)] = {'top_x':top_pos[0], 'top_y':top_pos[1], 'top_z':top_pos[2]}
aa.set_ant_params(p)
aa.set_arr_params(params)
return aa
if __name__ == "__main__":
ant_pos = generate_hexagon(7, 10, 1.0)
#plot_antenna_array(antpos)
save_antenna_array(ant_pos, "brawl127.txt")
ovro = ('37.240391', '-118.281667', 1184)
#Set other array parameters here
params = {
'name': 'BRAWL',
'Trx': 500 * 1e3 # receiver temp in mK, T_sky is taken care of later
}
aa = generate_aa(ovro, ant_pos, params)
aa.view()