API
 
Loading...
Searching...
No Matches
adcCtrl.py
Go to the documentation of this file.
1import sys
2import logging
3from enum import Enum
4import time
5import numpy as np
6
7import xconf
8
9from magaox.indi.device import XDevice, BaseConfig
10from magaox.camera import XCam
11from magaox.constants import StateCodes
12
13from purepyindi2 import device, properties, constants
14from purepyindi2.messages import DefNumber, DefSwitch, DefLight, DefText
15
16import hcipy as hp
17from scipy.optimize import minimize
18
20 def __init__(self,wavelength=656E-9,bandwidth=100E-9,grating_angle=-28,grating_freq=47):
21 self.wavelength = wavelength
22 self.bandwidth = bandwidth
23 self.grating_angle = grating_angle
24 self.grating_freq = grating_freq
25 self.normalized_wavelength = wavelength / 656E-9 #normalizing the wavelengths to the ha values
26 self.normalized_bandwidth = bandwidth / 656E-9
27 self.maxiter = 6
28 self.control_mtx = np.matrix([[0,0],])
29 self.current_speckle = None
30
31 def set_measurement(self, data):
32 self.data = data
33
34 def make_gaussian(self, mu_x, mu_y, sigma_x,sigma_y, orientation):
35 def func(grid):
36 new_grid = grid.shifted([-mu_x, -mu_y]).rotated(orientation)
37 x = new_grid.x / sigma_x
38 y = new_grid.y / sigma_y
39 r2 = x**2 + y**2
40 return hp.Field(np.exp(-0.5 * r2), grid)
41 return func
42
43 def satellite_spot(self, amplitude, mu_x, mu_y, sigma_x, sigma_y, orientation, background):
44 def func(grid):
45 return hp.Field(amplitude * self.make_gaussian(mu_x, mu_y, sigma_x, sigma_y, orientation)(grid) + background, grid)
46 return func
47
48 def cost(self, theta):
49 fit = self.satellite_spot(*theta)(self.data.grid)
50 j = np.sum( (self.data - fit)**2)
51
52 aspect_ratio = np.abs(theta[3] / theta[4])
53
54 ##boundary condition for the aspect ratio
55 if self.current_speckle == 0 or self.current_speckle == 2:
56 if aspect_ratio >= 1:
57 j+= 1E3 * aspect_ratio **2
58 elif self.current_speckle ==1 or self.current_speckle == 3:
59 if aspect_ratio <= 1:
60 j+= 1E3 * 1/aspect_ratio**2
61
62 return j
63
64 def fit(self,theta_est):
65 fitting = minimize(self.cost,theta_est,options={'maxiter':self.maxiter})
66 return fitting
67
69 M00 = np.sum(self.data)
70 M10 = np.sum(self.data * self.data.grid.x)
71 M01 = np.sum(self.data * self.data.grid.y)
72
73 centroid = [M10/M00,M01/M00]
74 return centroid
75
76 def estimate_angle(self):
77 M00 = np.sum(self.data)
78 M10 = np.sum(self.data * self.data.grid.x)
79 M01 = np.sum(self.data * self.data.grid.y)
80
81 M20 = np.sum(self.data * self.data.grid.x**2)
82 M02 = np.sum(self.data * self.data.grid.y**2)
83 M11 = np.sum(self.data * self.data.grid.y * self.data.grid.x)
84
85 mu10 = M10 / M00
86 mu01 = M01 / M00
87 mu20 = M20 / M00 - mu10**2
88 mu02 = M02 / M00 - mu01**2
89 mu11 = M11 / M00 - mu10 * mu01
90 angle = (1/2 * np.arctan2(2 * mu11, mu20 - mu02))
91
92 return angle
93
94 def find_speckle(self, image,speckle_number):
95 '''speckles are indexed from the top right going counter clockwise'''
96 grating_freq = self.grating_freq
97 grating_angle = self.grating_angle
98 corners = np.array([[0, grating_freq * self.normalized_wavelength],[grating_freq * self.normalized_wavelength,0],[0, -grating_freq * self.normalized_wavelength],[-grating_freq * self.normalized_wavelength,0]])
99 sizes = np.array([[8,20],[20,8],[8,20],[20,8]])
100 #sizes = np.array([[16,25],[25,16],[16,25],[25,16]])
101
102 rect = hp.make_rotated_aperture(hp.make_rectangular_aperture(size=sizes[speckle_number], center=corners[speckle_number]), np.deg2rad(-grating_angle))(image.grid)
103 speckle_img = rect * image
104 return speckle_img
105
106 def set_psf(self,psf):
107 self.psf = psf
108
110
111 speckle_angles = np.zeros(4)
112 sig_x = [0.8,3.5,0.8,3.5]
113 sig_y = [3.5,0.8,3.5,0.8]
114
115 for i in range(4):
116 img = self.find_speckle(self.psf,i)
117 self.current_speckle = i
118 self.set_measurement(img)
119
120 sigma_x = sig_x[i]
121 sigma_y = sig_y[i]
122 #orientation = np.radians(28)
123
124 orientation = self.estimate_angle()
125 if self.current_speckle == 0 or self.current_speckle ==2:
126 orientation = np.pi/2 - orientation
127 elif self.current_speckle == 1 or self.current_speckle == 3:
128 orientation = -orientation
129
130 amplitude = self.data.max()
131 centroid = self.estimate_centroid()
132 mu_x = centroid[0]
133 mu_y = centroid[1]
134 background = 0
135
136 theta_est = np.array([amplitude, mu_x, mu_y, sigma_x, sigma_y, orientation, background])
137 fit = self.fit(theta_est)
138 speckle_angles[i] = np.degrees(fit.x[5])
139
140 self.current_speckle = None
141 speckle_angles = np.array(speckle_angles).T
142
143 return speckle_angles
144
145 def speckle_pairs(self, speckle_angles):
146 pair02 = speckle_angles[0] - speckle_angles[2]
147 pair13 = speckle_angles[1] - speckle_angles[3]
148 return np.array([pair02,pair13])
149
150 def calculate_command(self,speckle_angles):
151 predicted_disp = self.control_mtx * np.matrix(speckle_angles).T
152 predicted_disp = np.array(predicted_disp)
153 return -predicted_disp
154
155 def clear(self):
156 self.psf = None
157 self.data = None
158
159 def set_control_mtx(self,matrix):
160 self.control_mtx = matrix
161
162 '''this is stuff specifically for working with the real calibration datacubes'''
163 def window_field(self,data, center, width, height):
164 indx = data.grid.closest_to(center)
165 y_ind, x_ind = np.unravel_index(indx, data.shaped.shape)
166 cutout = data.shaped[(y_ind-height//2):(y_ind + height//2), (x_ind-width//2):(x_ind+width//2)]
167 sub_grid = hp.make_pupil_grid([width, height], [width * data.grid.delta[0], height * data.grid.delta[1]])
168 return hp.Field(cutout.ravel(), sub_grid)
169
170 def crop_image(self, image,extent=400,mask_diam=60):
171 #cutout a centered PSF
172 img = image/image.max()
173
174 img_subtracted = img >0.1
175 center_of_intensity = np.array([sum(img_subtracted*img_subtracted.grid.x)/sum(img_subtracted),sum(img_subtracted*img_subtracted.grid.y)/sum(img_subtracted)])
176 mask_ap = hp.make_circular_aperture(mask_diam,center_of_intensity)
177 mask = mask_ap(img_subtracted.grid)
178 mask = abs(mask - 1)
179 masked_img = mask * img
180
181 img = masked_img
182 img = self.window_field(img,[center_of_intensity[0],center_of_intensity[1]],extent,extent)
183 img /= img.max()
184
185 bk = np.median(img)
186 img -= bk
187 img = hp.Field([x if x>0 else 0 for x in img],img.grid)
188
189 return img
190
191 def filter_image(self,img,low_freq = 0.01,high_freq=1):
192
193 ff = hp.FourierFilter(img.grid, hp.make_circular_aperture(2 * np.pi * low_freq))
194 filtered_img= np.real(ff.forward(img + 0j))
195 img = img - filtered_img
196
197 ff2 = hp.FourierFilter(img.grid, hp.make_circular_aperture(2 * np.pi * high_freq))
198 filtered_img = np.real(ff2.forward(img + 0j))
199 img = filtered_img
200 filtered_subtracted = img
201
202 return filtered_subtracted
203
204
205@xconf.config
207 """
208 """
209 shmim : str = xconf.field(help="Name of the camera device (specifically, the associated shmim, if different)")
210 dark_shmim : str = xconf.field(help="Name of the dark frame shmim associated with this camera device")
211
212@xconf.config
213class AdcCtrlConfig(BaseConfig):
214 """ Active ADC control
215 """
216 camera : CameraConfig = xconf.field(help="Camera to use")
217 sleep_interval_sec : float = xconf.field(default=0.25, help="Sleep interval between loop() calls")
218
219class States(Enum):
220 IDLE = 0
221 CLOSED_LOOP = 1
222 ONESHOT = 2
223 CALIB = 3
224
225class adcCtrl(XDevice):
226 config: AdcCtrlConfig
227
228 def setup(self):
229 self.log.debug(f"I was configured! See? {self.config=}")
230
231 fsm = properties.TextVector(name='fsm')
232 fsm.add_element(DefText(name='state', _value=StateCodes.INITIALIZED.name))
233 self.add_property(fsm)
234
235 sv = properties.SwitchVector(
236 name='state',
237 rule=constants.SwitchRule.ONE_OF_MANY,
238 perm=constants.PropertyPerm.READ_WRITE,
239 )
240 sv.add_element(DefSwitch(name="idle", _value=constants.SwitchState.ON))
241 sv.add_element(DefSwitch(name="adcLoop", _value=constants.SwitchState.OFF))
242 sv.add_element(DefSwitch(name="oneshot", _value=constants.SwitchState.OFF))
243 sv.add_element(DefSwitch(name="calibrate", _value=constants.SwitchState.OFF))
244 self.add_property(sv, callback=self.handle_state)
245
246 nv = properties.NumberVector(name='n_avg')
247 nv.add_element(DefNumber(
248 name='current', label='Number of frames', format='%i',
249 min=1, max=150, step=1, _value=1
250 ))
251 nv.add_element(DefNumber(
252 name='target', label='Number of frames', format='%i',
253 min=1, max=150, step=1, _value=1
254 ))
255 self.add_property(nv, callback=self.handle_n_avg)
256
257 nv = properties.NumberVector(name='gain')
258 nv.add_element(DefNumber(
259 name='current', label='ADC Loop Gain', format='%.2f',
260 min=0.00, max=1.00, step=0.01, _value=0.10
261 ))
262 nv.add_element(DefNumber(
263 name='target', label='ADC Loop Gain', format='%.2f',
264 min=0.00, max=1.00, step=0.01, _value=0.10
265 ))
266 self.add_property(nv, callback=self.handle_gain)
267
268 nv = properties.NumberVector(name='ctrl_mtx')
269 nv.add_element(DefNumber( #first element
270 name='m00', label='m00', format='%.4f',
271 min=-10.00, max=10.00, step=0.0001, _value=-0.2231 #default from matrix calc in november
272 ))
273 nv.add_element(DefNumber(
274 name='m01', label='m01', format='%.4f',
275 min=-10.00, max=10.00, step=0.0001, _value=-0.2298 #default from matrix calc in november
276 ))
277 self.add_property(nv, callback=self.handle_ctrl_mtx)
278
279 self.client.get_properties('adctrack')
280 self.client.get_properties('fwsci1')
281
282 self.log.info("Found camera: {:s}".format(self.config.camera.shmim))
283 self.camera = XCam(
284 self.config.camera.shmim,
285 pixel_size=6.0/21.0,
286 use_hcipy=True,
287 indi_client=self.client,
288 )
289
290 self._state = States.IDLE
291
292 #self._loop_counter = 0
293 self._n_avg = 1
294 self._gain = 0.5
295 self._command = 0
296 self._control_mtx = np.array([-0.22312707, -0.22983197])
297 self._extent = 400
298 self.delta_1 = 0
299 self.delta_2 = 0
300
301 if self.client['adctrack.deltaADC1.current'] != 0:
302 self.set_command(0,0)
303 self.send_command()
304
305 self.update_wavelength()
306 self.ADC = AdcFitter(wavelength=self._center_wavelength)
307 self.ADC.set_control_mtx(self._control_mtx)
308
309 self.properties['fsm']['state'] = StateCodes.READY.name
310 self.update_property(self.properties['fsm'])
311
312 def handle_state(self, existing_property, new_message):
313 target_list = ['idle', 'adcLoop', 'oneshot','calibrate']
314 for key in target_list:
315 if existing_property[key] == constants.SwitchState.ON:
316 current_state = key
317
318 if current_state not in new_message:
319
320 for key in target_list:
321 existing_property[key] = constants.SwitchState.OFF
322 if key in new_message:
323 existing_property[key] = new_message[key]
324
325 if key == 'idle':
326 self._state = States.IDLE
327 self.properties['fsm']['state'] = StateCodes.READY.name
328 self._command = 0
329 self.log.debug('State changed to idle')
330 elif key == 'adcLoop':
331 self._state = States.CLOSED_LOOP
332 self.update_wavelength()
333 self.properties['fsm']['state'] = StateCodes.OPERATING.name
334 self.log.debug('State changed to closed-loop')
335 elif key == 'oneshot':
336 self._state = States.ONESHOT
337 self.update_wavelength()
338 self.properties['fsm']['state'] = StateCodes.OPERATING.name
339 self.log.debug('State changed to oneshot')
340 elif key == 'calibrate':
341 self._state = States.CALIB
342 self.update_wavelength()
343 self.properties['fsm']['state'] = StateCodes.OPERATING.name
344 self.log.debug('State changed to calibration')
345
346 self.update_property(existing_property)
347 self.update_property(self.properties['fsm'])
348
349 def handle_n_avg(self, existing_property, new_message):
350 if 'target' in new_message and new_message['target'] != existing_property['current']:
351 existing_property['current'] = new_message['target']
352 existing_property['target'] = new_message['target']
353 self._n_avg = int(new_message['target'])
354 self.log.debug(f'now averaging over {self._n_avg} frames')
355 self.update_property(existing_property)
356
357 def handle_gain(self, existing_property, new_message):
358 if 'target' in new_message and new_message['target'] != existing_property['current']:
359 existing_property['current'] = new_message['target']
360 existing_property['target'] = new_message['target']
361 self._gain = float(new_message['target'])
362 self.log.debug(f'loop gain changed to {self._gain}')
363 self.update_property(existing_property)
364
365 def handle_ctrl_mtx(self, existing_property, new_message):
366 pass
367
369 if self.client['fwsci1.filterName.i'] == constants.SwitchState.ON:
370 self._center_wavelength = 762E-9
371 self._extent = 400
372 elif self.client['fwsci1.filterName.z'] == constants.SwitchState.ON:
373 self._center_wavelength = 908E-9
374 self._extent = 480
375 else:
376 self._center_wavelength = 656E-9
377 self._extent = 400
378 self.log.debug(f'using center wavelength {self._center_wavelength*1E9} nm')
379
381 self._command = 0
382 self.properties['state']['oneshot'] = constants.SwitchState.OFF
383 self.properties['state']['adcLoop'] = constants.SwitchState.OFF
384 self.properties['state']['calibrate'] = constants.SwitchState.OFF
385 self.properties['state']['idle'] = constants.SwitchState.ON
386 self.update_property(self.properties['state'])
387 self._state = States.IDLE
388
389 def set_command(self, d1, d2):
390 self.delta_1 = d1
391 self.delta_2 = d2
392
393 def send_command(self):
394 self.client['adctrack.deltaADC1.target'] = self.delta_1 + self.delta_2
395 self.client['adctrack.deltaADC2.target'] = self.delta_1 - self.delta_2
396
397 do_check = True
398 tolerance = 0.05
399 while do_check:
400
401 current_1 = self.client['adctrack.deltaADC1.current']
402 current_2 = self.client['adctrack.deltaADC2.current']
403
404 if abs(current_1 - self.delta_1 - self.delta_2) < tolerance and abs(current_2 - self.delta_1 + self.delta_2) < tolerance:
405 do_check = False
406
407 time.sleep(0.05)
408
409 def loop(self):
410 if self._state == States.CLOSED_LOOP:
411 img = self.camera.grab_stack(self._n_avg)
412 transpose = img.shaped.T
413 img = transpose.ravel()
414 img = self.ADC.filter_image(img)
415 img = self.ADC.crop_image(img,self._extent)
416 self.ADC.set_psf(img)
417
418 angles = self.ADC.find_speckle_angles2()
419 pair_angles = self.ADC.speckle_pairs(angles)
420 self.log.debug(f'angle offsets: {angles}')
421
422 error = self.ADC.calculate_command(pair_angles)
423 self._command = self._command + self._gain * error
424
425 if np.all(np.abs(self._command)) < 2: #setting a threshold so the prisms don't do anything crazy
426 self.set_command(np.squeeze(self._command),0)
427 self.send_command()
428 self.log.debug(f'ADC command sent: {self._command}')
429 else: self.log.info(f'ADC command {self._command} exceeds acceptable threshold and was not sent')
430
431 elif self._state == States.ONESHOT:
432 img = self.camera.grab_stack(self._n_avg)
433 transpose = img.shaped.T
434 img = transpose.ravel()
435 #img = self.ADC.filter_image(img)
436 img = self.ADC.crop_image(img,extent=self._extent)
437 self.ADC.set_psf(img)
438
439 self.log.info(f'image intensity:{np.sum(img):.2f}')
440 self.log.info(f'min intensity: {np.min(img):.2f}')
441 self.log.info(f'max intensity: {np.max(img):.2f}')
442 just_speckle = self.ADC.find_speckle(img,3)
443 self.log.info(f'speckle 3 intensity: {np.sum(just_speckle):.6f}')
444 center_of_intensity = np.array([sum(img*img.grid.x)/sum(img),sum(img*img.grid.y)/sum(img)])
445 self.log.info(f'center of intensity: {center_of_intensity}')
446
447 angles = self.ADC.find_speckle_angles2()
448 pair_angles = self.ADC.speckle_pairs(angles)
449 self.log.debug(f'angle offsets: {angles}')
450
451 self._command = self.ADC.calculate_command(pair_angles)
452
453 self.log.info(f'One-shot ADC correction calculated a command of: {self._command}')
454
455 if np.all(np.abs(self._command)) < 5: #setting a threshold so the prisms don't do anything crazy
456 self.set_command(np.squeeze(self._command),0)
457 self.send_command()
458 self.log.debug(f'ADC command sent: {self._command}')
459 else: self.log.info(f'ADC command {self._command} exceeds acceptable threshold and was not sent')
460
461 self.transition_to_idle()
462
463 elif self._state == States.CALIB:
464 sweep_angles = np.linspace(-3,3,26)
465 diff_pointing_pairs = np.zeros((len(sweep_angles),2))
466
467 for i, orientation in enumerate(sweep_angles):
468 self.log.debug(f'Step {i:d}')
469 self.set_command(orientation, 0)
470 self.send_command()
471
472 img = self.camera.grab_stack(self._n_avg)
473 transpose = img.shaped.T
474 img = transpose.ravel()
475 #img = self.ADC.filter_image(img)
476 img = self.ADC.crop_image(img,extent=self._extent)
477 self.ADC.set_psf(img)
478
479 angles = self.ADC.find_speckle_angles2()
480 pointing_pair = self.ADC.speckle_pairs(angles)
481 diff_pointing_pairs[i,] = pointing_pair
482
483 self.set_command(0,0)
484 self.send_command()
485
486 a1 = np.zeros(2)
487 b1 = np.zeros(2)
488
489 for j in range(2):
490 b1[j] , a1[j] = np.polyfit(sweep_angles,diff_pointing_pairs[:,j],deg=1)
491
492 response = np.matrix([b1])
493 new_control_mtx = np.linalg.pinv(response)
494
495 self._control_mtx = new_control_mtx.T
496 self.ADC.set_control_mtx(self._control_mtx)
497 self.log.info(f'calibration updated control matrix to: {self._control_mtx}')
498
499 # self.properties['ctrl_mtx']['m00'] = self._control_mtx[0][0]
500 # self.properties['ctrl_mtx']['m01'] = self._control_mtx[0][1]
501 # self.update_property(self.properties['ctrl_mtx'])
502
503 self.transition_to_idle()
504
505
506
507
508
509
make_gaussian(self, mu_x, mu_y, sigma_x, sigma_y, orientation)
Definition adcCtrl.py:34
filter_image(self, img, low_freq=0.01, high_freq=1)
Definition adcCtrl.py:191
find_speckle_angles2(self)
Definition adcCtrl.py:109
__init__(self, wavelength=656E-9, bandwidth=100E-9, grating_angle=-28, grating_freq=47)
Definition adcCtrl.py:20
estimate_angle(self)
Definition adcCtrl.py:76
set_control_mtx(self, matrix)
Definition adcCtrl.py:159
current_speckle
boundary condition for the aspect ratio
Definition adcCtrl.py:29
cost(self, theta)
Definition adcCtrl.py:48
speckle_pairs(self, speckle_angles)
Definition adcCtrl.py:145
window_field(self, data, center, width, height)
Definition adcCtrl.py:163
set_measurement(self, data)
Definition adcCtrl.py:31
fit(self, theta_est)
Definition adcCtrl.py:64
crop_image(self, image, extent=400, mask_diam=60)
Definition adcCtrl.py:170
estimate_centroid(self)
Definition adcCtrl.py:68
set_psf(self, psf)
Definition adcCtrl.py:106
satellite_spot(self, amplitude, mu_x, mu_y, sigma_x, sigma_y, orientation, background)
Definition adcCtrl.py:43
find_speckle(self, image, speckle_number)
Definition adcCtrl.py:94
calculate_command(self, speckle_angles)
Definition adcCtrl.py:150
handle_gain(self, existing_property, new_message)
Definition adcCtrl.py:357
transition_to_idle(self)
Definition adcCtrl.py:380
send_command(self)
Definition adcCtrl.py:393
handle_ctrl_mtx(self, existing_property, new_message)
Definition adcCtrl.py:365
handle_state(self, existing_property, new_message)
Definition adcCtrl.py:312
update_wavelength(self)
Definition adcCtrl.py:368
AdcCtrlConfig config
Definition adcCtrl.py:226
handle_n_avg(self, existing_property, new_message)
Definition adcCtrl.py:349
set_command(self, d1, d2)
Definition adcCtrl.py:389