API
 
Loading...
Searching...
No Matches
picoMotorCtrl.hpp
Go to the documentation of this file.
1/** \file picoMotorCtrl.hpp
2 * \brief The MagAO-X Pico Motor Controller header file
3 *
4 * \ingroup picoMotorCtrl_files
5 */
6
7#ifndef picoMotorCtrl_hpp
8#define picoMotorCtrl_hpp
9
10#include <map>
11
12#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
13#include "../../magaox_git_version.h"
14
15/** \defgroup picoMotorCtrl
16 * \brief The Pico Motor Controller application.
17 *
18 * Controls a multi-channel Newport pico motor controller. Each motor gets its own thread.
19 *
20 * <a href="../handbook/operating/software/apps/picoMotorCtrl.html">Application Documentation</a>
21 *
22 * \ingroup apps
23 *
24 */
25
26/** \defgroup picoMotorCtrl_files
27 * \ingroup picoMotorCtrl
28 */
29
30
31
32namespace MagAOX
33{
34namespace app
35{
36
37int splitResponse( int & address,
38 std::string & response,
39 const std::string & fullResponse
40 )
41{
42 size_t carrot = fullResponse.find('>');
43
44 if(carrot == std::string::npos)
45 {
46 address = 1;
47 response = fullResponse;
48 return 0;
49 }
50
51 if(carrot == 0)
52 {
53 address = 0;
54 response = "";
55 return -1;
56 }
57
58 if(carrot == fullResponse.size()-1)
59 {
60 address = 0;
61 response = "";
62 return -2;
63 }
64
65 try
66 {
67 address = std::stoi( fullResponse.substr(0,carrot));
68 response = fullResponse.substr(carrot+1);
69 }
70 catch(...)
71 {
72 address = 0;
73 response = "";
74 return -3;
75 }
76
77 return 0;
78
79}
80
81/** MagAO-X application to control a multi-channel Newport Picomotor Controller.
82 *
83 * \todo need to recognize signals in tty polls and not return errors, etc.
84 * \todo need to implement an onDisconnect() to update values to unknown indicators.
85 * \todo need a frequency-dependent max amp facility.
86 * \todo convert to ioDevice
87 * \todo need telnet device, with optional username/password.
88 *
89 */
90class picoMotorCtrl : public MagAOXApp<>, public dev::ioDevice, public dev::telemeter<picoMotorCtrl>
91{
92
93 friend class dev::telemeter<picoMotorCtrl>;
94
96
97 typedef long posT;
98
100 {
101 picoMotorCtrl * m_parent {nullptr}; ///< A pointer to this for thread starting.
102
103 std::string m_name; ///< The name of this channel, from the config section
104
105 int m_address {1}; ///< The controller address, default is 1
106
107 int m_channel {-1}; ///< The number of this channel, where the motor is plugged in
108
109 int m_type {3}; ///< The motor type of this channel, default is 3
110
111 std::vector<std::string> m_presetNames;
112 std::vector<posT> m_presetPositions;
113
114 posT m_currCounts {0}; ///< The current counts, the cumulative position
115
116 bool m_doMove {false}; ///< Flag indicating that a move is requested.
117 bool m_moving {false}; ///< Flag to indicate that we are actually moving
118
119 pcf::IndiProperty m_property;
120 pcf::IndiProperty m_indiP_presetName;
121
122 std::thread * m_thread {nullptr}; /**< Thread for managing this channel. A pointer to allow
123 copying, but must be deleted in d'tor of parent.*/
124
125 bool m_threadInit {true}; ///< Thread initialization flag.
126
127 pid_t m_threadID {0}; ///< The ID of the thread.
128
129 pcf::IndiProperty m_threadProp; ///< The property to hold the thread details.
130
131 explicit motorChannel( picoMotorCtrl * p /**< [in] The parent point to set */) : m_parent(p)
132 {
133 m_thread = new std::thread;
134 }
135
136 motorChannel( picoMotorCtrl * p, ///< [in] The parent point to set
137 const std::string & n, ///< [in] The name of this channel
138 int add, ///< [in] The controller address
139 int ch, ///< [in] The number of this channel
140 int type ///< [in] The motor type of this channel
141 ) : m_parent(p), m_name(n), m_address(add), m_channel(ch), m_type(type)
142 {
143 m_thread = new std::thread;
144 }
145
147 {
148 m_parent = mc.m_parent;
149 m_name = mc.m_name;
150 m_address = mc.m_address;
151 m_channel = mc.m_channel;
152 m_type = mc.m_type;
153 m_presetNames = mc.m_presetNames;
154 m_presetPositions = mc.m_presetPositions;
155 m_currCounts = mc.m_currCounts;
156 m_doMove = mc.m_doMove;
157 m_moving = mc.m_moving;
158 m_property = mc.m_property;
159 m_indiP_presetName = mc.m_indiP_presetName;
160 m_thread = mc.m_thread;
161 m_threadInit = mc.m_threadInit;
162 m_threadID =mc.m_threadID;
163 m_threadProp = mc.m_threadProp;
164 }
165 };
166
167 typedef std::map<std::string, motorChannel> channelMapT;
168
169 /** \name Configurable Parameters
170 * @{
171 */
172
173 std::string m_deviceAddr; ///< The device address
174 std::string m_devicePort {"23"}; ///< The device port
175
176 int m_nChannels {4}; ///< The number of motor channels total on the hardware. Number of attached motors inferred from config.
177
178 ///@}
179
180 std::vector<int> m_addresses; ///< The unique controller addresses.
181
182 channelMapT m_channels; ///< Map of motor names to channel.
183
184 tty::telnetConn m_telnetConn; ///< The telnet connection manager
185
186 ///Mutex for locking telnet communications.
187 std::mutex m_telnetMutex;
188
189 public:
190
191 /// Default c'tor.
193
194 /// D'tor, declared and defined for noexcept.
196
197 /// Setup the configuration system (called by MagAOXApp::setup())
198 virtual void setupConfig();
199
200 /// Implementation of loadConfig logic, separated for testing.
201 /** This is called by loadConfig().
202 */
203 int loadConfigImpl( mx::app::appConfigurator & _config /**< [in] an application configuration from which to load values*/);
204
205 /// load the configuration system results (called by MagAOXApp::setup())
206 virtual void loadConfig();
207
208 /// Startup functions
209 /** Setsup the INDI vars.
210 *
211 */
212 virtual int appStartup();
213
214 /// Implementation of the FSM
215 /**
216 * \returns 0 on no critical error
217 * \returns -1 on an error requiring shutdown
218 */
219 virtual int appLogic();
220
221 /// Implementation of the on-power-off FSM logic
222 virtual int onPowerOff();
223
224 /// Implementation of the while-powered-off FSM
226
227 /// Do any needed shutdown tasks.
228 virtual int appShutdown();
229
230 /// Read the current channel counts from disk at startup
231 /** Reads the counts from the file with the specified name in this apps sys directory.
232 * Returns the file contents as a posT.
233 */
234 posT readChannelCounts(const std::string & chName);
235
236 int writeChannelCounts( const std::string & chName,
238 );
239
240 /// Channel thread starter function
241 static void channelThreadStart( motorChannel * mc /**< [in] the channel to start controlling */);
242
243 /// Channel thread execution function
244 /** Runs until m_shutdown is true.
245 */
247
248/** \name INDI
249 * @{
250 */
252
253 //declare our properties
254 std::vector<pcf::IndiProperty> m_indiP_counts;
255
256
257public:
258 /// The static callback function to be registered for relative position requests
259 /** Dispatches to the handler, which then signals the relavent thread.
260 *
261 * \returns 0 on success.
262 * \returns -1 on error.
263 */
264 static int st_newCallBack_picopos( void * app, ///< [in] a pointer to this, will be static_cast-ed to this
265 const pcf::IndiProperty &ipRecv ///< [in] the INDI property sent with the the new property request.
266 );
267
268 /// The handler function for relative position requests, called by the static callback
269 /** Signals the relavent thread.
270 *
271 * \returns 0 on success.
272 * \returns -1 on error.
273 */
274 int newCallBack_picopos( const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/);
275
276 /// The static callback function to be registered for position presets
277 /** Dispatches to the handler, which then signals the relavent thread.
278 *
279 * \returns 0 on success.
280 * \returns -1 on error.
281 */
282 static int st_newCallBack_presetName( void * app, ///< [in] a pointer to this, will be static_cast-ed to this
283 const pcf::IndiProperty &ipRecv ///< [in] the INDI property sent with the the new property request.
284 );
285
286 /// The handler function for position presets, called by the static callback
287 /** Signals the relavent thread.
288 *
289 * \returns 0 on success.
290 * \returns -1 on error.
291 */
292 int newCallBack_presetName( const pcf::IndiProperty &ipRecv /**< [in] the INDI property sent with the the new property request.*/);
293 ///@}
294
295 /** \name Telemeter Interface
296 *
297 * @{
298 */
299 int checkRecordTimes();
300
302
303 int recordPico( bool force = false );
304 ///@}
305
306};
307
314
316{
317 //Wait for each channel thread to exit, then delete it.
318 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
319 {
320 if(it->second.m_thread != nullptr)
321 {
322 if(it->second.m_thread->joinable()) it->second.m_thread->join();
323 delete it->second.m_thread;
324 }
325 }
326}
327
328
330{
331 config.add("device.address", "", "device.address", argType::Required, "device", "address", false, "string", "The controller IP address.");
332 config.add("device.nChannels", "", "device.nChannels", argType::Required, "device", "nChannels", false, "int", "Number of motoro channels. Default is 4.");
333
335
336 TELEMETER_SETUP_CONFIG( config );
337
338}
339
340#define PICOMOTORCTRL_E_NOMOTORS (-5)
341#define PICOMOTORCTRL_E_BADCHANNEL (-6)
342#define PICOMOTORCTRL_E_DUPMOTOR (-7)
343#define PICOMOTORCTRL_E_INDIREG (-20)
344
345int picoMotorCtrl::loadConfigImpl( mx::app::appConfigurator & _config )
346{
347 //Standard config parsing
348 _config(m_deviceAddr, "device.address");
349 _config(m_nChannels, "device.nChannels");
350
351
352 // Parse the unused config options to look for motors
353 std::vector<std::string> sections;
354
355 _config.unusedSections(sections);
356
357 if( sections.size() == 0 )
358 {
359 log<text_log>("No motors found in config.", logPrio::LOG_CRITICAL);
360
362 }
363
364 //Now see if any unused sections have a channel keyword
365 for(size_t i=0;i<sections.size(); ++i)
366 {
367 int channel = -1;
368 _config.configUnused(channel, mx::app::iniFile::makeKey(sections[i], "channel" ) );
369 if( channel == -1 )
370 {
371 //not a channel
372 continue;
373 }
374
376 {
377 log<text_log>("Bad channel specificiation: " + sections[i] + " channel: " + std::to_string(channel), logPrio::LOG_CRITICAL);
378
380 }
381
382 int address = 1;
383 _config.configUnused(address, mx::app::iniFile::makeKey(sections[i], "address" ) );
384
385 if(address < 1)
386 {
387 log<text_log>("Bad channel specificiation: " + sections[i] + " address: " + std::to_string(address), logPrio::LOG_CRITICAL);
388
390 }
391
392 int type = 3;
393 _config.configUnused(type, mx::app::iniFile::makeKey(sections[i], "type" ) );
394
395 if(type < 1)
396 {
397 log<text_log>("Bad motor type specificiation: " + sections[i] + " type: " + std::to_string(type), logPrio::LOG_CRITICAL);
398
400 }
401
402 //Ok, valid channel. Insert into map and check for duplicates.
403 std::pair<channelMapT::iterator, bool> insert = m_channels.insert(std::pair<std::string, motorChannel>(sections[i], motorChannel(this,sections[i], address, channel, type)));
404
405 if(insert.second == false)
406 {
407 log<text_log>("Duplicate motor specificiation: " + sections[i] + " " + std::to_string(channel), logPrio::LOG_CRITICAL);
409 }
410 else
411 {
412 _config.configUnused(insert.first->second.m_presetNames, mx::app::iniFile::makeKey(sections[i], "names" ));
413 _config.configUnused(insert.first->second.m_presetPositions, mx::app::iniFile::makeKey(sections[i], "positions" ));
414 }
415
416 ///\todo extend to include address
417 log<pico_channel>({sections[i], (uint8_t) channel});
418
419 bool found = false;
420 for(size_t n = 0; n < m_addresses.size(); ++n)
421 {
422 if(address == m_addresses[n])
423 {
424 found = true;
425 break;
426 }
427 }
428
429 if(!found)
430 {
431 m_addresses.push_back(address);
432 }
433 }
434
435 TELEMETER_LOAD_CONFIG( config );
436
437 return 0;
438}
439
441{
442 if( loadConfigImpl(config) < 0)
443 {
444 log<text_log>("Error during config", logPrio::LOG_CRITICAL);
445 m_shutdown = true;
446 }
447
448 if(dev::ioDevice::loadConfig(config) < 0)
449 {
450 log<text_log>("Error during ioDevice config", logPrio::LOG_CRITICAL);
451 m_shutdown = true;
452 }
453
454
455
456}
457
459{
460 ///\todo read state from disk to get current counts.
461
462 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
463 {
464 it->second.m_currCounts = readChannelCounts(it->second.m_name);
465
466
467 createStandardIndiNumber( it->second.m_property, it->first+"_pos", std::numeric_limits<posT>::lowest(), std::numeric_limits<posT>::max(), static_cast<posT>(1), "%d", "Position", it->first);
468 it->second.m_property["current"].set(it->second.m_currCounts);
469 it->second.m_property["target"].set(it->second.m_currCounts);
470 it->second.m_property.setState(INDI_IDLE);
471
472 if( registerIndiPropertyNew( it->second.m_property, st_newCallBack_picopos) < 0)
473 {
474 #ifndef PICOMOTORCTRL_TEST_NOLOG
476 #endif
478 }
479
480 if(it->second.m_presetNames.size() > 0)
481 {
482 if(createStandardIndiSelectionSw( it->second.m_indiP_presetName, it->first, it->second.m_presetNames) < 0)
483 {
485 return -1;
486 }
487 if( registerIndiPropertyNew( it->second.m_indiP_presetName, st_newCallBack_presetName) < 0)
488 {
490 return -1;
491 }
492 }
493
494 //Here we start each channel thread, with 0 R/T prio.
495 threadStart( *it->second.m_thread, it->second.m_threadInit, it->second.m_threadID, it->second.m_threadProp, 0, "", it->second.m_name, &it->second, channelThreadStart);
496 }
497
498 //Install empty signal handler for USR1, which is used to interrupt sleeps in the channel threads.
499 struct sigaction act;
500 sigset_t set;
501
502 act.sa_sigaction = &sigUsr1Handler;
503 act.sa_flags = SA_SIGINFO;
504 sigemptyset(&set);
505 act.sa_mask = set;
506
507 errno = 0;
508 if( sigaction(SIGUSR1, &act, 0) < 0 )
509 {
510 std::string logss = "Setting handler for SIGUSR1 failed. Errno says: ";
511 logss += strerror(errno);
512
514
515 return -1;
516 }
517
519
520 return 0;
521}
522
524{
525 if( state() == stateCodes::POWERON)
526 {
527 if(!powerOnWaitElapsed()) return 0;
528
530 }
531
533 {
535
536 if(rv == 0)
537 {
540 }
541 else
542 {
543 if(powerState() != 1 || powerStateTarget() != 1) return 0;
544
545 if(!stateLogged())
546 {
547 log<text_log>("Failed to connect on " + m_deviceAddr + ":" + m_devicePort);
548 }
549
550 return 0;
551 }
552
553 }
554
556 {
557
558 std::unique_lock<std::mutex> lock(m_telnetMutex);
559
560 //First check the address 1 controller
561 int rv = m_telnetConn.write("*IDN?\r\n", m_writeTimeout);
562 if(rv != TTY_E_NOERROR)
563 {
564 if(powerState() != 1 || powerStateTarget() != 1) return 0;
567 return 0;
568 }
569
570 rv = m_telnetConn.read("\r\n", m_readTimeout, true);
571 if(rv != TTY_E_NOERROR)
572 {
573 if(powerState() != 1 || powerStateTarget() != 1) return 0;
576 return 0;
577 }
578
579 if(m_telnetConn.m_strRead.find("New_Focus") != std::string::npos)
580 {
581 log<text_log>("Connected to " + m_telnetConn.m_strRead + " at address 1");
582 }
583 else
584 {
585 if(powerState() != 1 || powerStateTarget() != 1) return 0;
586 log<software_error>({__FILE__, __LINE__, "wrong response to IDN query at address 1"});
588 return 0;
589 }
590
591 //Now do a controller scan
592 rv = m_telnetConn.write("SC1\r\n", m_writeTimeout); //Will adjust addresses.
593 if(rv != TTY_E_NOERROR)
594 {
595 if(powerState() != 1 || powerStateTarget() != 1) return 0;
598 return 0;
599 }
600
601 //And now do a motor scan
602 rv = m_telnetConn.write("MC\r\n", m_writeTimeout);
603 if(rv != TTY_E_NOERROR)
604 {
605 if(powerState() != 1 || powerStateTarget() != 1) return 0;
608 return 0;
609 }
610
611 sleep(2); //Give time for controller scan to finish
612
613 for(size_t n = 0; n < m_addresses.size(); ++n)
614 {
615 if(m_addresses[n] == 1) continue; //already done.
616
617 std::string addprefix = std::to_string(m_addresses[n]) + ">";
618
619 int rv = m_telnetConn.write(addprefix + "*IDN?\r\n", m_writeTimeout);
620 if(rv != TTY_E_NOERROR)
621 {
622 if(powerState() != 1 || powerStateTarget() != 1) return 0;
625 return 0;
626 }
627
628 rv = m_telnetConn.read("\r\n", m_readTimeout, true);
629 if(rv != TTY_E_NOERROR)
630 {
631 if(powerState() != 1 || powerStateTarget() != 1) return 0;
634 return 0;
635 }
636
637 int add;
638 std::string resp;
639
641 if(rv != 0)
642 {
643 if(powerState() != 1 || powerStateTarget() != 1) return 0;
644 log<software_error>({__FILE__, __LINE__, "splitResponse returned " + std::to_string(rv)});
646 return 0;
647 }
648
649 if(add != m_addresses[n])
650 {
651 if(powerState() != 1 || powerStateTarget() != 1) return 0;
652 log<software_error>({__FILE__, __LINE__, "address did not match in response"});
654 return 0;
655 }
656
657 if(resp.find("New_Focus") != std::string::npos)
658 {
659 log<text_log>("Connected to " + resp + " at address " + std::to_string(m_addresses[n]));
660 }
661 else
662 {
663 if(powerState() != 1 || powerStateTarget() != 1) return 0;
664 log<software_error>({__FILE__, __LINE__, "wrong response to IDN query at address " + std::to_string(m_addresses[n])});
666 return 0;
667 }
668
669 //Now do a motor scan
671 if(rv != TTY_E_NOERROR)
672 {
673 if(powerState() != 1 || powerStateTarget() != 1) return 0;
676 return 0;
677 }
678 }
679
680 sleep(2); //This is to give time for motor scans to finish
681
682 //Now check for each motor attached
683 for(auto it=m_channels.begin(); it!=m_channels.end();++it)
684 {
685 std::string query = std::to_string(it->second.m_address) + ">" + std::to_string(it->second.m_channel) + "QM?";
686
688 if(rv != TTY_E_NOERROR)
689 {
690 if(powerState() != 1 || powerStateTarget() != 1) return 0;
693 return 0;
694 }
695
696 rv = m_telnetConn.read("\r\n", m_readTimeout, true);
697 if(rv != TTY_E_NOERROR)
698 {
699 if(powerState() != 1 || powerStateTarget() != 1) return 0;
702 return 0;
703 }
704
705 int add;
706 std::string resp;
707
709 if(rv != 0)
710 {
711 if(powerState() != 1 || powerStateTarget() != 1) return 0;
712 log<software_error>({__FILE__, __LINE__, "splitResponse returned " + std::to_string(rv)});
714 return 0;
715 }
716
717 if(add != it->second.m_address)
718 {
719 if(powerState() != 1 || powerStateTarget() != 1) return 0;
720 log<software_error>({__FILE__, __LINE__, "address did not match in response"});
722 return 0;
723 }
724
725 int moType = std::stoi(resp);
726 if(moType == 0)
727 {
728 if(powerState() != 1 || powerStateTarget() != 1) return 0;
729 log<text_log>("No motor connected on channel " + std::to_string(it->second.m_address) + "." + std::to_string(it->second.m_channel) + " [" + it->second.m_name + "]", logPrio::LOG_CRITICAL);
731 return -1;
732 }
733 else if (moType != it->second.m_type)
734 {
735 if(powerState() != 1 || powerStateTarget() != 1) return 0;
736 std::string msg = "Wrong motor type connected on channel " + std::to_string(it->second.m_address) + ".";
737 msg += std::to_string(it->second.m_channel) + " [" + it->second.m_name + "] ";
738 msg += "expected " + std::to_string(it->second.m_type) + " ";
739 msg += "got " + std::to_string(moType);
740
743 return -1;
744 }
745 }
746
748
749 return 0;
750 }
751
753 {
754 //check connection
755 {
756 std::unique_lock<std::mutex> lock(m_telnetMutex);
757
758 int rv = m_telnetConn.write("*IDN?\r\n", m_writeTimeout);
759 if(rv != TTY_E_NOERROR)
760 {
761 if(powerState() != 1 || powerStateTarget() != 1) return 0;
764 return 0;
765 }
766
767 rv = m_telnetConn.read("\r\n", m_readTimeout, true);
768 if(rv != TTY_E_NOERROR)
769 {
770 if(powerState() != 1 || powerStateTarget() != 1) return 0;
773 return 0;
774 }
775
776 if(m_telnetConn.m_strRead.find("New_Focus") == std::string::npos)
777 {
778 if(powerState() != 1 || powerStateTarget() != 1) return 0;
779
780 log<software_error>({__FILE__, __LINE__, "wrong response to IDN query"});
782 return 0;
783 }
784 }
785
786 //Now check state of motors
787 bool anymoving = false;
788
789 //This is where we'd check for moving
790 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
791 {
792 std::unique_lock<std::mutex> lock(m_telnetMutex);
793
794 std::string query = std::to_string(it->second.m_address) + ">" + std::to_string(it->second.m_channel) + "MD?";
795
796 int rv = m_telnetConn.write(query + "\r\n", m_writeTimeout);
797 if(rv != TTY_E_NOERROR)
798 {
799 if(powerState() != 1 || powerStateTarget() != 1) return 0;
802 return 0;
803 }
804
805 rv = m_telnetConn.read("\r\n", m_readTimeout, true);
806 if(rv != TTY_E_NOERROR)
807 {
808 if(powerState() != 1 || powerStateTarget() != 1) return 0;
811 return 0;
812 }
813
814 int add;
815 std::string resp;
816
818 if(rv != 0)
819 {
820 if(powerState() != 1 || powerStateTarget() != 1) return 0;
821 log<software_error>({__FILE__, __LINE__, "splitResponse returned " + std::to_string(rv)});
823 return 0;
824 }
825
826 if(add != it->second.m_address)
827 {
828 if(powerState() != 1 || powerStateTarget() != 1) return 0;
829 log<software_error>({__FILE__, __LINE__, "address did not match in response"});
831 return 0;
832 }
833
834 //The check for moving here. With power off detection
835 if(std::stoi(resp) == 0)
836 {
837 anymoving = true;
838 it->second.m_moving = true;
839 }
840 else
841 {
842 it->second.m_moving = false;
843 }
844
845 if(it->second.m_moving == false && it->second.m_doMove == true)
846 {
847 it->second.m_currCounts = it->second.m_property["target"].get<long>();
848 log<text_log>("moved " + it->second.m_name + " to " + std::to_string(it->second.m_currCounts) + " counts");
849 it->second.m_doMove = false;
850 recordPico(true);
851 }
852 }
853
854 if(anymoving == false) state(stateCodes::READY);
856
857 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
858 {
859 std::unique_lock<std::mutex> lock(m_indiMutex);
860 if(it->second.m_moving) updateIfChanged(it->second.m_property, "current", it->second.m_currCounts, INDI_BUSY);
861 else updateIfChanged(it->second.m_property, "current", it->second.m_currCounts, INDI_IDLE);
862
863 for(size_t n=0; n < it->second.m_presetNames.size(); ++n)
864 {
865 bool changed = false;
866 if( it->second.m_currCounts == it->second.m_presetPositions[n])
867 {
868 if(it->second.m_indiP_presetName[it->second.m_presetNames[n]] == pcf::IndiElement::Off) changed = true;
869 it->second.m_indiP_presetName[it->second.m_presetNames[n]] = pcf::IndiElement::On;
870 }
871 else
872 {
873 if(it->second.m_indiP_presetName[it->second.m_presetNames[n]] == pcf::IndiElement::On) changed = true;
874 it->second.m_indiP_presetName[it->second.m_presetNames[n]] = pcf::IndiElement::Off;
875 }
876
877 if(changed) m_indiDriver->sendSetProperty(it->second.m_indiP_presetName);
878 }
879
880 if(writeChannelCounts(it->second.m_name, it->second.m_currCounts) < 0)
881 {
883 }
884 }
885
887
888 return 0;
889 }
890
891
892 return 0;
893}
894
896{
897 return 0;
898}
899
901{
902 return 0;
903}
904
906{
907 //Shutdown and join the threads
908 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
909 {
910 if(it->second.m_thread->joinable())
911 {
912 pthread_kill(it->second.m_thread->native_handle(), SIGUSR1);
913 try
914 {
915 it->second.m_thread->join(); //this will throw if it was already joined
916 }
917 catch(...)
918 {
919 }
920 }
921 }
922
924
925 return 0;
926}
927
929{
930 std::string statusDir = sysPath;
931 statusDir += "/";
933
934 std::string fileName = statusDir + "/" + chName;
935
936 std::ifstream posIn;
937 posIn.open( fileName );
938
939 if(!posIn.good())
940 {
941 log<text_log>("no position file for " + chName + " found. initializing to 0.");
942 return 0;
943 }
944
945 long pos;
946 posIn >> pos;
947
948 posIn.close();
949
950 log<text_log>("initializing " + chName + " to " + std::to_string(pos));
951
952 return pos;
953}
954
957 )
958{
959 std::string statusDir = sysPath;
960 statusDir += "/";
962
963 std::string fileName = statusDir + "/" + chName;
964
966
967 std::ofstream posOut;
968 posOut.open( fileName );
969
970 if(!posOut.good())
971 {
972 log<text_log>("could not open counts file for " + chName + " -- can not store position.", logPrio::LOG_ERROR);
973 return -1;
974 }
975
976 posOut << counts;
977
978 posOut.close();
979
980 return 0;
981}
982
984{
985 mc->m_parent->channelThreadExec(mc);
986}
987
989{
990 //Get the thread PID immediately so the caller can return.
991 mc->m_threadID = syscall(SYS_gettid);
992
993 //Wait for initialization to complete.
994 while( mc->m_threadInit == true && m_shutdown == 0)
995 {
996 sleep(1);
997 }
998
999 //Now begin checking for state change request.
1000 while(!m_shutdown)
1001 {
1002 //If told to move and not moving, start a move
1003 if(mc->m_doMove && !mc->m_moving && (state() == stateCodes::READY || state() == stateCodes::OPERATING))
1004 {
1005 long dr = mc->m_property["target"].get<long>() - mc->m_currCounts;
1006
1007 recordPico(true);
1008 std::unique_lock<std::mutex> lock(m_telnetMutex);
1010 mc->m_moving = true;
1011 log<text_log>("moving " + mc->m_name + " by " + std::to_string(dr) + " counts");
1012
1013 std::string comm = std::to_string(mc->m_address) + ">" + std::to_string(mc->m_channel) + "PR" + std::to_string(dr);
1014
1015 int rv = m_telnetConn.write(comm + "\r\n", m_writeTimeout);
1016 if(rv != TTY_E_NOERROR)
1017 {
1018 if(powerState() != 1 || powerStateTarget() != 1) //about to get POWEROFF
1019 {
1020 sleep(1);
1021 continue;
1022 }
1025 }
1026 }
1027 else if( !(state() == stateCodes::READY || state() == stateCodes::OPERATING))
1028 {
1029 mc->m_doMove = false; //In case a move is requested when not able to move
1030 }
1031
1032 sleep(1);
1033 }
1034
1035
1036}
1037
1038
1040 const pcf::IndiProperty &ipRecv
1041 )
1042{
1043 return static_cast<picoMotorCtrl*>(app)->newCallBack_picopos(ipRecv);
1044}
1045
1046int picoMotorCtrl::newCallBack_picopos( const pcf::IndiProperty &ipRecv )
1047{
1048
1049 //Search for the channel
1050 std::string propName = ipRecv.getName();
1051 size_t nend = propName.rfind("_pos");
1052
1053 if(nend == std::string::npos)
1054 {
1055 log<software_error>({__FILE__, __LINE__, "Channel without _pos received"});
1056 return -1;
1057 }
1058
1059 std::string chName = propName.substr(0, nend);
1060 channelMapT::iterator it = m_channels.find(chName);
1061
1062 if(it == m_channels.end())
1063 {
1064 log<software_error>({__FILE__, __LINE__, "Unknown channel name received"});
1065 return -1;
1066 }
1067
1068 if(it->second.m_doMove == true)
1069 {
1070 log<text_log>("channel " + it->second.m_name + " is already moving", logPrio::LOG_WARNING);
1071 return 0;
1072 }
1073
1074 //Set the target element, and the doMove flag, and then signal the thread.
1075 {//scope for mutex
1076 std::unique_lock<std::mutex> lock(m_indiMutex);
1077
1078 long counts; //not actually used
1079 if(indiTargetUpdate( it->second.m_property, counts, ipRecv, true) < 0)
1080 {
1081 return log<software_error,-1>({__FILE__,__LINE__});
1082 }
1083 }
1084
1085 it->second.m_doMove= true;
1086
1087 pthread_kill(it->second.m_thread->native_handle(), SIGUSR1);
1088
1089 return 0;
1090}
1091
1093 const pcf::IndiProperty &ipRecv
1094 )
1095{
1096 return static_cast<picoMotorCtrl*>(app)->newCallBack_presetName (ipRecv);
1097}
1098
1099int picoMotorCtrl::newCallBack_presetName( const pcf::IndiProperty &ipRecv )
1100{
1101 channelMapT::iterator it = m_channels.find(ipRecv.getName());
1102
1103 if(it == m_channels.end())
1104 {
1105 log<software_error>({__FILE__, __LINE__, "Unknown channel name received"});
1106 return -1;
1107 }
1108
1109 if(it->second.m_doMove == true)
1110 {
1111 log<text_log>("channel " + it->second.m_name + " is already moving", logPrio::LOG_WARNING);
1112 return 0;
1113 }
1114
1115 long counts = -1e10;
1116
1117 size_t i;
1118 for(i=0; i< it->second.m_presetNames.size(); ++i)
1119 {
1120 if(!ipRecv.find(it->second.m_presetNames[i])) continue;
1121
1122 if(ipRecv[it->second.m_presetNames[i]].getSwitchState() == pcf::IndiElement::On)
1123 {
1124 if(counts != -1e10)
1125 {
1126 log<text_log>("More than one preset selected", logPrio::LOG_ERROR);
1127 return -1;
1128 }
1129
1130 counts = it->second.m_presetPositions[i];
1131 std::cerr << "selected: " << it->second.m_presetNames[i] << " " << counts << "\n";
1132 }
1133 }
1134
1135 //Set the target element, and the doMove flag, and then signal the thread.
1136 {//scope for mutex
1137 std::unique_lock<std::mutex> lock(m_indiMutex);
1138
1139 it->second.m_property["target"].set(counts);
1140 }
1141
1142 it->second.m_doMove= true;
1143
1144 pthread_kill(it->second.m_thread->native_handle(), SIGUSR1);
1145
1146 return 0;
1147}
1148
1153
1155{
1156 return recordPico(true);
1157}
1158
1160{
1161 static std::vector<int64_t> lastpos(m_nChannels, std::numeric_limits<long>::max());
1162
1163 bool changed = false;
1164 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
1165 {
1166 if(it->second.m_currCounts != lastpos[it->second.m_channel-1]) changed = true;
1167 }
1168
1169 if( changed || force )
1170 {
1171 for(channelMapT::iterator it = m_channels.begin(); it != m_channels.end(); ++ it)
1172 {
1173 lastpos[it->second.m_channel-1] = it->second.m_currCounts;
1174 }
1175
1177 }
1178
1179 return 0;
1180}
1181
1182} //namespace app
1183} //namespace MagAOX
1184
1185#endif //picoMotorCtrl_hpp
Internal class to manage setuid privilege escalation with RAII.
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.
std::string m_configName
The name of the configuration file (minus .conf).
Definition MagAOXApp.hpp:83
stateCodes::stateCodeT state()
Get the current state code.
int registerIndiPropertyNew(pcf::IndiProperty &prop, int(*)(void *, const pcf::IndiProperty &))
Register an INDI property which is exposed for others to request a New Property for.
int createStandardIndiNumber(pcf::IndiProperty &prop, const std::string &name, const T &min, const T &max, const T &step, const std::string &format, const std::string &label="", const std::string &group="")
Create a standard R/W INDI Number property with target and current elements.
int powerState()
Returns the current power state.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
indiDriver< MagAOXApp > * m_indiDriver
The INDI driver wrapper. Constructed and initialized by execute, which starts and stops communication...
int powerStateTarget()
Returns the target power state.
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.
std::mutex m_indiMutex
Mutex for locking INDI communications.
int createStandardIndiSelectionSw(pcf::IndiProperty &prop, const std::string &name, const std::vector< std::string > &elements, const std::vector< std::string > &elementLabels, const std::string &label="", const std::string &group="")
Create a standard R/W INDI selection (one of many) switch with vector of elements and element labels.
int threadStart(std::thread &thrd, bool &thrdInit, pid_t &tpid, pcf::IndiProperty &thProp, int thrdPrio, const std::string &cpuset, const std::string &thrdName, thisPtr *thrdThis, Function &&thrdStart)
Start a thread, using this class's privileges to set priority, etc.
std::string sysPath
The path to the system directory, for PID file, etc.
Definition MagAOXApp.hpp:91
int indiTargetUpdate(pcf::IndiProperty &localProperty, T &localTarget, const pcf::IndiProperty &remoteProperty, bool setBusy=true)
Get the target element value from an new property.
int newCallBack_picopos(const pcf::IndiProperty &ipRecv)
The handler function for relative position requests, called by the static callback.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
int m_nChannels
The number of motor channels total on the hardware. Number of attached motors inferred from config.
virtual int onPowerOff()
Implementation of the on-power-off FSM logic.
tty::telnetConn m_telnetConn
The telnet connection manager.
static int st_newCallBack_presetName(void *app, const pcf::IndiProperty &ipRecv)
The static callback function to be registered for position presets.
std::map< std::string, motorChannel > channelMapT
~picoMotorCtrl() noexcept
D'tor, declared and defined for noexcept.
static void channelThreadStart(motorChannel *mc)
Channel thread starter function.
posT readChannelCounts(const std::string &chName)
Read the current channel counts from disk at startup.
std::string m_deviceAddr
The device address.
virtual int appStartup()
Startup functions.
virtual void loadConfig()
load the configuration system results (called by MagAOXApp::setup())
virtual int appLogic()
Implementation of the FSM.
int recordPico(bool force=false)
dev::telemeter< picoMotorCtrl > telemeterT
virtual int whilePowerOff()
Implementation of the while-powered-off FSM.
std::vector< int > m_addresses
The unique controller addresses.
int recordTelem(const telem_pico *)
static int st_newCallBack_picopos(void *app, const pcf::IndiProperty &ipRecv)
The static callback function to be registered for relative position requests.
std::string m_devicePort
The device port.
virtual int appShutdown()
Do any needed shutdown tasks.
std::mutex m_telnetMutex
Mutex for locking telnet communications.
int writeChannelCounts(const std::string &chName, posT counts)
virtual void setupConfig()
Setup the configuration system (called by MagAOXApp::setup())
void channelThreadExec(motorChannel *mc)
Channel thread execution function.
int newCallBack_presetName(const pcf::IndiProperty &ipRecv)
The handler function for position presets, called by the static callback.
channelMapT m_channels
Map of motor names to channel.
std::vector< pcf::IndiProperty > m_indiP_counts
@ OPERATING
The device is operating, other than homing.
@ FAILURE
The application has failed, should be used when m_shutdown is set for an error.
@ 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.
std::string ttyErrorString(int ec)
Get a text explanation of a TTY_E_ error code.
Definition ttyErrors.cpp:15
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_BUSY
Definition indiUtils.hpp:29
std::stringstream msg
int splitResponse(int &address, std::string &response, const std::string &fullResponse)
void sigUsr1Handler(int signum, siginfo_t *siginf, void *ucont)
Empty signal handler. SIGUSR1 is used to interrupt sleep in various threads.
Definition MagAOXApp.cpp:42
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
static constexpr logPrioT LOG_WARNING
A condition has occurred which may become an error, but the process continues.
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
#define PICOMOTORCTRL_E_INDIREG
#define PICOMOTORCTRL_E_NOMOTORS
#define PICOMOTORCTRL_E_BADCHANNEL
#define PICOMOTORCTRL_E_DUPMOTOR
An input/output capable device.
Definition ioDevice.hpp:27
unsigned m_writeTimeout
The write timeout [msec].
Definition ioDevice.hpp:29
int loadConfig(mx::app::appConfigurator &config)
Load the device section from an application configurator.
Definition ioDevice.cpp:28
int setupConfig(mx::app::appConfigurator &config)
Setup an application configurator for the device section.
Definition ioDevice.cpp:20
unsigned m_readTimeout
The read timeout [msec].
Definition ioDevice.hpp:28
A device base class which saves telemetry.
Definition telemeter.hpp:69
int checkRecordTimes(const telT &tel, telTs... tels)
Check the time of the last record for each telemetry type and make an entry if needed.
picoMotorCtrl * m_parent
A pointer to this for thread starting.
int m_address
The controller address, default is 1.
int m_channel
The number of this channel, where the motor is plugged in.
std::vector< std::string > m_presetNames
bool m_doMove
Flag indicating that a move is requested.
pcf::IndiProperty m_threadProp
The property to hold the thread details.
posT m_currCounts
The current counts, the cumulative position.
bool m_threadInit
Thread initialization flag.
motorChannel(picoMotorCtrl *p, const std::string &n, int add, int ch, int type)
std::string m_name
The name of this channel, from the config section.
bool m_moving
Flag to indicate that we are actually moving.
int m_type
The motor type of this channel, default is 3.
Software ERR log entry.
Log entry recording CPU temperatures.
A Telnet connection manager, wrapping libtelnet.
int write(const std::string &buffWrite, int timeoutWrite)
Write to a telnet connection.
std::string m_strRead
The accumulated string read from the device.
int noLogin()
Set flags as if we're logged in, used when device doesn't require it.
std::string m_prompt
The device's prompt, used for detecting end of transmission.
int read(const std::string &eot, int timeoutRead, bool clear=true)
Read from a telnet connection, until end-of-transmission string is read.
int connect(const std::string &host, const std::string &port)
Connect to the device.
#define TELEMETER_APP_LOGIC
Call telemeter::appLogic with error checking.
#define TELEMETER_LOAD_CONFIG(cfig)
Call telemeter::loadConfig with error checking.
#define TELEMETER_APP_STARTUP
Call telemeter::appStartup with error checking.
#define TELEMETER_SETUP_CONFIG(cfig)
Call telemeter::setupConfig with error checking.
#define TELEMETER_APP_SHUTDOWN
Call telemeter::appShutdown with error checking.
#define TTY_E_NOERROR
Definition ttyErrors.hpp:15