229 self.log.debug(f
"I was configured! See? {self.config=}")
231 fsm = properties.TextVector(name=
'fsm')
232 fsm.add_element(DefText(name=
'state', _value=StateCodes.INITIALIZED.name))
233 self.add_property(fsm)
235 sv = properties.SwitchVector(
237 rule=constants.SwitchRule.ONE_OF_MANY,
238 perm=constants.PropertyPerm.READ_WRITE,
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))
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
251 nv.add_element(DefNumber(
252 name=
'target', label=
'Number of frames', format=
'%i',
253 min=1, max=150, step=1, _value=1
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
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
268 nv = properties.NumberVector(name=
'ctrl_mtx')
269 nv.add_element(DefNumber(
270 name=
'm00', label=
'm00', format=
'%.4f',
271 min=-10.00, max=10.00, step=0.0001, _value=-0.2231
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
279 self.client.get_properties(
'adctrack')
280 self.client.get_properties(
'fwsci1')
282 self.log.info(
"Found camera: {:s}".format(self.
config.camera.shmim))
287 indi_client=self.client,
301 if self.client[
'adctrack.deltaADC1.current'] != 0:
309 self.properties[
'fsm'][
'state'] = StateCodes.READY.name
310 self.update_property(self.properties[
'fsm'])
313 target_list = [
'idle',
'adcLoop',
'oneshot',
'calibrate']
314 for key
in target_list:
315 if existing_property[key] == constants.SwitchState.ON:
318 if current_state
not in new_message:
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]
327 self.properties[
'fsm'][
'state'] = StateCodes.READY.name
329 self.log.debug(
'State changed to idle')
330 elif key ==
'adcLoop':
331 self.
_state = States.CLOSED_LOOP
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
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
343 self.properties[
'fsm'][
'state'] = StateCodes.OPERATING.name
344 self.log.debug(
'State changed to calibration')
346 self.update_property(existing_property)
347 self.update_property(self.properties[
'fsm'])
410 if self.
_state == States.CLOSED_LOOP:
412 transpose = img.shaped.T
413 img = transpose.ravel()
414 img = self.
ADC.filter_image(img)
416 self.
ADC.set_psf(img)
418 angles = self.
ADC.find_speckle_angles2()
419 pair_angles = self.
ADC.speckle_pairs(angles)
420 self.log.debug(f
'angle offsets: {angles}')
422 error = self.
ADC.calculate_command(pair_angles)
425 if np.all(np.abs(self.
_command)) < 2:
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')
431 elif self.
_state == States.ONESHOT:
433 transpose = img.shaped.T
434 img = transpose.ravel()
436 img = self.
ADC.crop_image(img,extent=self.
_extent)
437 self.
ADC.set_psf(img)
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}')
447 angles = self.
ADC.find_speckle_angles2()
448 pair_angles = self.
ADC.speckle_pairs(angles)
449 self.log.debug(f
'angle offsets: {angles}')
451 self.
_command = self.
ADC.calculate_command(pair_angles)
453 self.log.info(f
'One-shot ADC correction calculated a command of: {self._command}')
455 if np.all(np.abs(self.
_command)) < 5:
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')
463 elif self.
_state == States.CALIB:
464 sweep_angles = np.linspace(-3,3,26)
465 diff_pointing_pairs = np.zeros((len(sweep_angles),2))
467 for i, orientation
in enumerate(sweep_angles):
468 self.log.debug(f
'Step {i:d}')
473 transpose = img.shaped.T
474 img = transpose.ravel()
476 img = self.
ADC.crop_image(img,extent=self.
_extent)
477 self.
ADC.set_psf(img)
479 angles = self.
ADC.find_speckle_angles2()
480 pointing_pair = self.
ADC.speckle_pairs(angles)
481 diff_pointing_pairs[i,] = pointing_pair
490 b1[j] , a1[j] = np.polyfit(sweep_angles,diff_pointing_pairs[:,j],deg=1)
492 response = np.matrix([b1])
493 new_control_mtx = np.linalg.pinv(response)
497 self.log.info(f
'calibration updated control matrix to: {self._control_mtx}')