API
 
Loading...
Searching...
No Matches
xt1121Ctrl.hpp
Go to the documentation of this file.
1/** \file xt1121Ctrl.hpp
2 * \brief The MagAO-X Acromag XT 1121digital I/O controller.
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup xt1121Ctrl_files
7 */
8
9#ifndef xt1121Ctrl_hpp
10#define xt1121Ctrl_hpp
11
12
13#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
14#include "../../magaox_git_version.h"
15
16#include "xtChannels.hpp"
17
18namespace MagAOX
19{
20namespace app
21{
22
23
24/** \defgroup xt1121Ctrl Acromag xt1121Controller
25 * \brief Control of an Acromag xt1121digital I/O module
26 *
27 * <a href="../handbook/operating/software/apps/xt1121Ctrl.html">Application Documentation</a>
28 *
29 * \ingroup apps
30 *
31 */
32
33/** \defgroup xt1121Ctrl_files Acromag xt1121Controller Files
34 * \ingroup xt1121Ctrl
35 */
36
37/** MagAO-X application to control an Acromag xt1121digital i/o module
38 *
39 * \ingroup xt1121Ctrl
40 *
41 */
42class xt1121Ctrl : public MagAOXApp<>, public xt1121Channels
43{
44
45protected:
46
47 /** \name configurable parameters
48 *@{
49 */
50
51 std::string m_address; ///< The I.P. address of the device
52
53 uint16_t m_port {502}; ///< The port to use. Default is 502 for modbus.
54
55 ///@}
56
57 modbus * m_mb {nullptr}; ///< The modbus protocol communication object
58
59public:
60
61 ///Default c'tor
62 xt1121Ctrl();
63
64 ///Destructor
66
67 /// Setup the configuration system (called by MagAOXApp::setup())
68 virtual void setupConfig();
69
70 /// load the configuration system results (called by MagAOXApp::setup())
71 virtual void loadConfig();
72
73 /// Startup functions
74 /** Sets up the INDI vars.
75 *
76 */
77 virtual int appStartup();
78
79 /// Implementation of the FSM for the Siglent SDG
80 virtual int appLogic();
81
82 /// Implementation of the on-power-off FSM logic
83 virtual int onPowerOff();
84
85 /// Implementation of the while-powered-off FSM
87
88 /// Do any needed shutdown tasks. Currently nothing in this app.
89 virtual int appShutdown();
90
91 /// Get the current state of the outlets.
92 /**
93 * \returns 0 on success
94 * \returns -1 on error
95 */
96 int getState();
97
98 //INDI:
100 //declare our properties
117
118public:
119
120 /// Callback worker to actually set or clear a channel and send it to the device
121 /** Contains the target/current logic, and calls the xtChannels::setRegisters
122 * function, and then the modbus write_registers.
123 *
124 * \returns 0 on success
125 * \returns -1 on error
126 */
127 int channelSetCallback( size_t chNo, ///< [in] The channel number to set
128 pcf::IndiProperty & ipToSet, ///< [in] The corresponding local INDI property
129 const pcf::IndiProperty & ipRecv ///< [in] The received INDI property
130 );
131
148};
149
150inline
152{
153 m_powerMgtEnabled = true;
154 m_powerOnWait = 2; //set default.
155 return;
156}
157
158inline
160{
161 if(m_mb)
162 {
163 delete m_mb;
164 };
165
166 return;
167}
168
169inline
171{
172 config.add("device.address", "", "device.address", argType::Required, "device", "address", true, "string", "The device I.P. address.");
173 config.add("device.port", "", "device.port", argType::Required, "device", "port", true, "int", "The device port. Default is 502.");
174 //config.add("device.powerOnWait", "", "device.powerOnWait", argType::Required, "device", "powerOnWait", false, "int", "Time after power-on to begin attempting connections [sec]. Default is 2 sec.");
175 config.add("device.inputOnly", "", "device.inputOnly", argType::Required, "device", "inputOnly", false, "vector<int>", "List of channels which are input-only.");
176
177}
178
179
180///\todo mxlib loadConfig needs to return int to propagate errors!
181
182inline
184{
185 config(m_address, "device.address");
186 config(m_port, "device.port");
187
188 std::vector<int> ino;
189 config(ino, "device.inputOnly");
190
191 for(size_t i=0; i< ino.size();++i)
192 {
193 if(setInputOnly(ino[i]) != 0)
194 {
195 log<text_log>("Error setting channel " + std::to_string(i) + " to input only.", logPrio::LOG_ERROR);
196 }
197 }
198}
199
200
201
202inline
204{
205 // set up the INDI properties
206 REG_INDI_NEWPROP(m_indiP_ch00, "ch00", pcf::IndiProperty::Number);
207 m_indiP_ch00.add (pcf::IndiElement("current"));
208 m_indiP_ch00["current"].set(-1);
209 m_indiP_ch00.add (pcf::IndiElement("target"));
210
211 REG_INDI_NEWPROP(m_indiP_ch01, "ch01", pcf::IndiProperty::Number);
212 m_indiP_ch01.add (pcf::IndiElement("current"));
213 m_indiP_ch01["current"].set(-1);
214 m_indiP_ch01.add (pcf::IndiElement("target"));
215
216 REG_INDI_NEWPROP(m_indiP_ch02, "ch02", pcf::IndiProperty::Number);
217 m_indiP_ch02.add (pcf::IndiElement("current"));
218 m_indiP_ch02["current"].set(-1);
219 m_indiP_ch02.add (pcf::IndiElement("target"));
220
221 REG_INDI_NEWPROP(m_indiP_ch03, "ch03", pcf::IndiProperty::Number);
222 m_indiP_ch03.add (pcf::IndiElement("current"));
223 m_indiP_ch03["current"].set(-1);
224 m_indiP_ch03.add (pcf::IndiElement("target"));
225
226 REG_INDI_NEWPROP(m_indiP_ch04, "ch04", pcf::IndiProperty::Number);
227 m_indiP_ch04.add (pcf::IndiElement("current"));
228 m_indiP_ch04["current"].set(-1);
229 m_indiP_ch04.add (pcf::IndiElement("target"));
230
231 REG_INDI_NEWPROP(m_indiP_ch05, "ch05", pcf::IndiProperty::Number);
232 m_indiP_ch05.add (pcf::IndiElement("current"));
233 m_indiP_ch05["current"].set(-1);
234 m_indiP_ch05.add (pcf::IndiElement("target"));
235
236 REG_INDI_NEWPROP(m_indiP_ch06, "ch06", pcf::IndiProperty::Number);
237 m_indiP_ch06.add (pcf::IndiElement("current"));
238 m_indiP_ch06["current"].set(-1);
239 m_indiP_ch06.add (pcf::IndiElement("target"));
240
241 REG_INDI_NEWPROP(m_indiP_ch07, "ch07", pcf::IndiProperty::Number);
242 m_indiP_ch07.add (pcf::IndiElement("current"));
243 m_indiP_ch07["current"].set(-1);
244 m_indiP_ch07.add (pcf::IndiElement("target"));
245
246 REG_INDI_NEWPROP(m_indiP_ch08, "ch08", pcf::IndiProperty::Number);
247 m_indiP_ch08.add (pcf::IndiElement("current"));
248 m_indiP_ch08["current"].set(-1);
249 m_indiP_ch08.add (pcf::IndiElement("target"));
250
251 REG_INDI_NEWPROP(m_indiP_ch09, "ch09", pcf::IndiProperty::Number);
252 m_indiP_ch09.add (pcf::IndiElement("current"));
253 m_indiP_ch09["current"].set(-1);
254 m_indiP_ch09.add (pcf::IndiElement("target"));
255
256 REG_INDI_NEWPROP(m_indiP_ch10, "ch10", pcf::IndiProperty::Number);
257 m_indiP_ch10.add (pcf::IndiElement("current"));
258 m_indiP_ch10["current"].set(-1);
259 m_indiP_ch10.add (pcf::IndiElement("target"));
260
261 REG_INDI_NEWPROP(m_indiP_ch11, "ch11", pcf::IndiProperty::Number);
262 m_indiP_ch11.add (pcf::IndiElement("current"));
263 m_indiP_ch11["current"].set(-1);
264 m_indiP_ch11.add (pcf::IndiElement("target"));
265
266 REG_INDI_NEWPROP(m_indiP_ch12, "ch12", pcf::IndiProperty::Number);
267 m_indiP_ch12.add (pcf::IndiElement("current"));
268 m_indiP_ch12["current"].set(-1);
269 m_indiP_ch12.add (pcf::IndiElement("target"));
270
271 REG_INDI_NEWPROP(m_indiP_ch13, "ch13", pcf::IndiProperty::Number);
272 m_indiP_ch13.add (pcf::IndiElement("current"));
273 m_indiP_ch13["current"].set(-1);
274 m_indiP_ch13.add (pcf::IndiElement("target"));
275
276 REG_INDI_NEWPROP(m_indiP_ch14, "ch14", pcf::IndiProperty::Number);
277 m_indiP_ch14.add (pcf::IndiElement("current"));
278 m_indiP_ch14["current"].set(-1);
279 m_indiP_ch14.add (pcf::IndiElement("target"));
280
281 REG_INDI_NEWPROP(m_indiP_ch15, "ch15", pcf::IndiProperty::Number);
282 m_indiP_ch15.add (pcf::IndiElement("current"));
283 m_indiP_ch15["current"].set(-1);
284 m_indiP_ch15.add (pcf::IndiElement("target"));
285
286
287 return 0;
288
289}
290
291
292
293inline
295{
296 if( state() == stateCodes::POWERON )
297 {
298 if(!powerOnWaitElapsed())
299 {
300 return 0;
301 }
302
304 }
305
307 {
308 std::string response;
309
310 //Might have gotten here because of a power off.
311 if(m_powerState == 0) return 0;
312
313 m_mb = new(std::nothrow) modbus(m_address, m_port);
314
315 if(m_mb == nullptr)
316 {
317 return log<software_critical, -1>({__FILE__, __LINE__, "allocation failure"});
318 }
319
321
322 if( m_mb->modbus_connect() == false)
323 {
324 if(!stateLogged())
325 {
326 log<text_log>("connect failed at " + m_address + ":" + std::to_string(m_port));
327 }
328 delete m_mb;
329 m_mb = nullptr;
330 return 0;
331 }
332
334 log<text_log>("connected to " + m_address + ":" + std::to_string(m_port));
335 }
336
338 {
339 //Get a lock
340 std::unique_lock<std::mutex> lock(m_indiMutex);
341
342 if( getState() == 0 )
343 {
345 return 0;
346 }
347 else
348 {
351 }
352 }
353
355 {
356 //Get a lock if we can
357 std::unique_lock<std::mutex> lock(m_indiMutex, std::try_to_lock);
358
359 //but don't wait for it, just go back around.
360 if(!lock.owns_lock()) return 0;
361
362 if(getState() < 0)
363 {
364 if(m_powerState == 0) return 0;
365
367 return 0;
368 }
369
370 return 0;
371 }
372
373 //Fall through check?
374
375 return 0;
376
377}
378
379inline
381{
382 std::lock_guard<std::mutex> lock(m_indiMutex);
383
384 updateIfChanged(m_indiP_ch00, "current", -1);
385 updateIfChanged(m_indiP_ch00, "target", -1);
386
387 updateIfChanged(m_indiP_ch01, "current", -1);
388 updateIfChanged(m_indiP_ch01, "target", -1);
389
390 updateIfChanged(m_indiP_ch02, "current", -1);
391 updateIfChanged(m_indiP_ch02, "target", -1);
392
393 updateIfChanged(m_indiP_ch03, "current", -1);
394 updateIfChanged(m_indiP_ch03, "target", -1);
395
396 updateIfChanged(m_indiP_ch04, "current", -1);
397 updateIfChanged(m_indiP_ch04, "target", -1);
398
399 updateIfChanged(m_indiP_ch05, "current", -1);
400 updateIfChanged(m_indiP_ch05, "target", -1);
401
402 updateIfChanged(m_indiP_ch06, "current", -1);
403 updateIfChanged(m_indiP_ch06, "target", -1);
404
405 updateIfChanged(m_indiP_ch07, "current", -1);
406 updateIfChanged(m_indiP_ch07, "target", -1);
407
408 updateIfChanged(m_indiP_ch08, "current", -1);
409 updateIfChanged(m_indiP_ch08, "target", -1);
410
411 updateIfChanged(m_indiP_ch09, "current", -1);
412 updateIfChanged(m_indiP_ch09, "target", -1);
413
414 updateIfChanged(m_indiP_ch10, "current", -1);
415 updateIfChanged(m_indiP_ch10, "target", -1);
416
417 updateIfChanged(m_indiP_ch11, "current", -1);
418 updateIfChanged(m_indiP_ch11, "target", -1);
419
420 updateIfChanged(m_indiP_ch12, "current", -1);
421 updateIfChanged(m_indiP_ch12, "target", -1);
422
423 updateIfChanged(m_indiP_ch13, "current", -1);
424 updateIfChanged(m_indiP_ch13, "target", -1);
425
426 updateIfChanged(m_indiP_ch14, "current", -1);
427 updateIfChanged(m_indiP_ch14, "target", -1);
428
429 updateIfChanged(m_indiP_ch15, "current", -1);
430 updateIfChanged(m_indiP_ch15, "target", -1);
431
432
433 return 0;
434}
435
436inline
438{
439 return 0;
440}
441
442inline
444{
445 if(m_mb) m_mb->modbus_close();
446 return 0;
447}
448
449
450
451inline
453{
455
456 try
457 {
458 ///\todo this hangs if power goes off during call
460 }
461 catch(std::exception & e)
462 {
463 if(m_powerState == 0) return 0; //due to power off
464
465 return log<software_error,-1>({__FILE__, __LINE__, std::string("Exception caught: ") + e.what()});
466 }
467
468 if( readRegisters(input_regs) !=0 )
469 {
470 return log<software_error,-1>({__FILE__, __LINE__});
471 }
472
473 updateIfChanged(m_indiP_ch00, "current", channel(0));
474 updateIfChanged(m_indiP_ch01, "current", channel(1));
475 updateIfChanged(m_indiP_ch02, "current", channel(2));
476 updateIfChanged(m_indiP_ch03, "current", channel(3));
477 updateIfChanged(m_indiP_ch04, "current", channel(4));
478 updateIfChanged(m_indiP_ch05, "current", channel(5));
479 updateIfChanged(m_indiP_ch06, "current", channel(6));
480 updateIfChanged(m_indiP_ch07, "current", channel(7));
481 updateIfChanged(m_indiP_ch08, "current", channel(8));
482 updateIfChanged(m_indiP_ch09, "current", channel(9));
483 updateIfChanged(m_indiP_ch10, "current", channel(10));
484 updateIfChanged(m_indiP_ch11, "current", channel(11));
485 updateIfChanged(m_indiP_ch12, "current", channel(12));
486 updateIfChanged(m_indiP_ch13, "current", channel(13));
487 updateIfChanged(m_indiP_ch14, "current", channel(14));
488 updateIfChanged(m_indiP_ch15, "current", channel(15));
489
490 return 0;
491
492
493}
494
495
497 pcf::IndiProperty & ipToSet,
498 const pcf::IndiProperty & ipRecv
499 )
500{
501 int current = -1, target = -1;
502
503 if(ipRecv.find("current"))
504 {
505 current = ipRecv["current"].get<unsigned>();
506 }
507
508 if(ipRecv.find("target"))
509 {
510 target = ipRecv["target"].get<unsigned>();
511 }
512
513 if(target == -1) target = current;
514
515 if(target < 0) return 0;
516
517 //Lock the mutex, waiting if necessary
518 std::unique_lock<std::mutex> lock(m_indiMutex);
519
520 if(target == 0) clearChannel(chNo);
521 else setChannel(chNo);
522
523 target = channel(chNo); //This checks for inputOnly
524
525 updateIfChanged(ipToSet, "target", target);
526
528
529 if( setRegisters(input_regs) !=0 )
530 {
531 return log<software_error,-1>({__FILE__, __LINE__});
532 }
533
534 try
535 {
537 }
538 catch(std::exception & e)
539 {
540 if(m_powerState == 0) return 0; //due to power off
541
542 return log<software_error,-1>({__FILE__, __LINE__, std::string("Exception caught: ") + e.what()});
543 }
544
545
546 log<text_log>("Set channel " + std::to_string(chNo) + " to " + std::to_string(target));
547
548 return 0;
549
550}
551
553{
555
557
558 return -1;
559}
560
562{
564
566
567 return -1;
568}
569
571{
573
575
576 return -1;
577}
578
580{
582
584
585 return -1;
586}
587
589{
591
593
594 return -1;
595}
596
598{
600
602
603 return -1;
604}
605
607{
609
611
612 return -1;
613}
614
616{
618
620
621 return -1;
622}
623
625{
627
629
630 return -1;
631}
632
634{
636
638
639 return -1;
640}
641
643{
645
647
648 return -1;
649}
650
652{
654
656
657 return -1;
658}
659
661{
663
665
666 return -1;
667}
668
670{
672
674
675 return -1;
676}
677
679{
681
683
684 return -1;
685}
686
688{
690
692
693 return -1;
694}
695
696
697}//namespace app
698} //namespace MagAOX
699#endif
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
void updateIfChanged(pcf::IndiProperty &p, const std::string &el, const T &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI property element value if it has changed.
stateCodes::stateCodeT state()
Get the current state code.
int m_powerState
Current power state, 1=On, 0=Off, -1=Unk.
int stateLogged()
Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change,...
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
bool powerOnWaitElapsed()
This method tests whether the power on wait time has elapsed.
unsigned long m_powerOnWait
Time in sec to wait for device to boot after power on.
std::mutex m_indiMutex
Mutex for locking INDI communications.
virtual void setupConfig()
Setup the configuration system (called by MagAOXApp::setup())
pcf::IndiProperty m_indiP_ch11
virtual void loadConfig()
load the configuration system results (called by MagAOXApp::setup())
virtual int appLogic()
Implementation of the FSM for the Siglent SDG.
pcf::IndiProperty m_indiP_ch02
pcf::IndiProperty m_indiP_ch14
virtual int appStartup()
Startup functions.
~xt1121Ctrl() noexcept
Destructor.
virtual int appShutdown()
Do any needed shutdown tasks. Currently nothing in this app.
pcf::IndiProperty m_indiP_ch07
std::string m_address
The I.P. address of the device.
uint16_t m_port
The port to use. Default is 502 for modbus.
pcf::IndiProperty m_indiP_ch10
int getState()
Get the current state of the outlets.
pcf::IndiProperty m_indiP_ch00
pcf::IndiProperty m_indiP_ch08
pcf::IndiProperty m_indiP_ch06
pcf::IndiProperty m_indiP_ch15
pcf::IndiProperty m_indiP_ch01
pcf::IndiProperty m_indiP_ch05
xt1121Ctrl()
Default c'tor.
pcf::IndiProperty m_indiP_ch09
virtual int whilePowerOff()
Implementation of the while-powered-off FSM.
modbus * m_mb
The modbus protocol communication object.
virtual int onPowerOff()
Implementation of the on-power-off FSM logic.
pcf::IndiProperty m_indiP_ch12
pcf::IndiProperty m_indiP_ch03
pcf::IndiProperty m_indiP_ch13
pcf::IndiProperty m_indiP_ch04
int channelSetCallback(size_t chNo, pcf::IndiProperty &ipToSet, const pcf::IndiProperty &ipRecv)
Callback worker to actually set or clear a channel and send it to the device.
Modbus Operator Class.
Definition modbus.hpp:63
bool modbus_connect()
Definition modbus.cpp:19
void modbus_read_input_registers(int address, int amount, uint16_t *buffer)
Definition modbus.cpp:131
void modbus_set_slave_id(int id)
Definition modbus.cpp:15
void modbus_close()
Definition modbus.cpp:49
void modbus_write_registers(int address, int amount, uint16_t *value)
Definition modbus.cpp:254
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define REG_INDI_NEWPROP(prop, propName, type)
Register a NEW INDI property with the class, using the standard callback name.
#define INDI_NEWCALLBACK_DECL(class, prop)
Declare the callback for a new property request, and declare and define the static wrapper.
@ OPERATING
The device is operating, other than homing.
@ ERROR
The application has encountered an error, from which it is recovering (with or without intervention)
@ READY
The device is ready for operation, but is not operating.
@ CONNECTED
The application has connected to the device or service.
@ NOTCONNECTED
The application is not connected to the device or service.
@ POWERON
The device power is on.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
Software CRITICAL log entry.
Software ERR log entry.
Utility class for managing Acromag xt12XX digital I/O channels.
int setRegisters(uint16_t registers[numRegisters])
Set registers based on current channel states.
int setChannel(size_t chNo)
Set a channel to true.
static constexpr int numRegisters
The number of registers needed for the number of channels.
int channel(size_t chNo)
Gets the current state of a channel.
int setInputOnly(size_t chNo)
Set a channel to be input only.
int clearChannel(size_t chNo)
Set a channel to false.
int readRegisters(uint16_t registers[numRegisters])
Read channel states from the registers.
Utility class for managing Acromag xt12XX digital I/O channels.