MagAO-X
Operations Applications Utilities Source
ttyIOUtils.hpp
Go to the documentation of this file.
1 /** \file ttyIOUtils.hpp
2  * \brief Utilities for i/o on a file descriptor pointing to a tty device.
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * \ingroup tty_files
6  * History:
7  * - 2018-01-15 created by JRM, starting with code imported from VisAO
8  */
9 
10 #ifndef tty_ttyIOUtils_hpp
11 #define tty_ttyIOUtils_hpp
12 
13 //#include <string.h>
14 
15 #include <unistd.h>
16 #include <fcntl.h>
17 #include <poll.h>
18 #include <termios.h>
19 
20 #include <string>
21 
22 #include <mx/timeUtils.hpp>
23 
24 #include "ttyErrors.hpp"
25 
26 #ifndef TTY_BUFFSIZE
27  #define TTY_BUFFSIZE (1024)
28 #endif
29 
30 namespace MagAOX
31 {
32 namespace tty
33 {
34 
35 /// Replace lone \\r and \\n with \\r\\n for telnet-ness.
36 /** Do it all at once instead of during char-by-char transmission
37  * cuz some devices (I'm looking at you SDG) get impatient and
38  * stop paying attention.
39  *
40  * \returns 0 on success
41  * \returns -1 on error (nothing yet)
42  *
43  */
44 int telnetCRLF( std::string & telnetStr, ///< [out] the string with \\r an \\n converted to \\r\\n
45  const std::string & inputStr ///< [in] the string to be converted
46  )
47 {
48  telnetStr.resize(inputStr.size());
49 
50  size_t N = inputStr.size();
51  size_t j = 0;
52  for(size_t i=0;i<N; ++i)
53  {
54  if(inputStr[i] != '\r' && inputStr[i] != '\n')
55  {
56  telnetStr[j] = inputStr[i];
57  }
58  else if(inputStr[i] == '\r')
59  {
60  telnetStr[j] = '\r';
61 
62  if(i < N-1)
63  {
64  if(inputStr[i+1] == '\n')
65  {
66  ++j;
67  telnetStr[j] = '\n';
68  ++i;
69  ++j; //i is incremented on continue, but j is not
70  continue;
71  }
72  }
73  telnetStr.push_back(' ');
74  ++j;
75  telnetStr[j] = '\n';
76  }
77  else if(inputStr[i] == '\n')
78  {
79  telnetStr[j] = '\r';
80  telnetStr.push_back(' ');
81  ++j;
82  telnetStr[j] = '\n';
83  }
84  ++j;
85  }
86 
87  return 0;
88 }
89 
90 
91 /// Open a file as a raw-mode tty device
92 /**
93  * \returns TTY_E_NOERROR on success.
94  * \returns TTY_E_TCGETATTR on a error from tcgetattr.
95  * \returns TTY_E_TCSETATTR on an error from tcsetattr.
96  * \returns TTY_E_SETISPEED on a cfsetispeed error.
97  * \returns TTY_E_SETOSPEED on a cfsetospeed error.
98  *
99  * \ingroup tty
100  */
101 int ttyOpenRaw( int & fileDescrip, ///< [out] the file descriptor. Set to 0 on an error.
102  std::string & deviceName, ///< [in] the device path name, e.g. /dev/ttyUSB0
103  speed_t speed ///< [in] indicates the baud rate (see http://pubs.opengroup.org/onlinepubs/7908799/xsh/termios.h.html)
104  )
105 {
106  errno = 0;
107 
108  fileDescrip = ::open( deviceName.c_str(), O_RDWR | O_NOCTTY | O_NDELAY);
109 
110 
111  struct termios termopt;
112  if( tcgetattr(fileDescrip, &termopt) < 0 )
113  {
114  close(fileDescrip);
115  fileDescrip = 0;
116  return TTY_E_TCGETATTR;
117  }
118 
119  if( cfsetispeed(&termopt, speed) < 0 )
120  {
121  close(fileDescrip);
122  fileDescrip = 0;
123  return TTY_E_SETISPEED;
124  }
125 
126  if( cfsetospeed(&termopt, speed) < 0 )
127  {
128  close(fileDescrip);
129  fileDescrip = 0;
130  return TTY_E_SETOSPEED;
131  }
132 
133  cfmakeraw(&termopt);
134 
135  if( tcsetattr(fileDescrip, TCSANOW, &termopt) < 0 )
136  {
137  close(fileDescrip);
138  fileDescrip = 0;
139  return TTY_E_TCSETATTR;
140  }
141 
142  return TTY_E_NOERROR;
143 }
144 
145 /// Check if the end of the buffer contains the end-of-transmission string
146 /**
147  * \returns true if the last N chars of buffRead are equal to eot, where N is the length of eot.
148  * \returns false otherwise.
149  *
150  * \ingroup tty
151  */
152 inline
153 bool isEndOfTrans( const std::string & strRead, ///< [in] The read buffer to check
154  const std::string & eot ///< [in] The end-of-transmission string
155  )
156 {
157  //If buffRead isn't long enough yet.
158  if(eot.size() > strRead.size()) return false;
159 
160  //Now check from back, if any don't match it's false.
161  for(size_t i=0; i < eot.size(); ++i)
162  {
163  if( strRead[strRead.size()-1-i] != eot[eot.size()-1-i] ) return false;
164  }
165 
166  return true;
167 }
168 
169 
170 /// Write to the tty console indicated by a file descriptor.
171 /**
172  *
173  * \returns TTY_E_NOERROR on success
174  * \returns TTY_E_TIMEOUTONWRITEPOLL if the poll times out.
175  * \returns TTY_E_ERRORONWRITEPOLL if an error is returned by poll.
176  * \returns TTY_E_TIMEOUTONWRITE if a timeout occurs during the write.
177  * \returns TTY_E_ERRORONWRITE if an error occurs writing to the file.
178  *
179  * \ingroup tty
180  */
181 inline
182 int ttyWrite( const std::string & buffWrite, ///< [in] The characters to write to the tty.
183  int fd, ///< [in] The file descriptor of the open tty.
184  int timeoutWrite ///< [in] The timeout in milliseconds.
185  )
186 {
187  double t0;
188  struct pollfd pfd;
189 
190  errno = 0;
191  pfd.fd = fd;
192  pfd.events = POLLOUT;
193 
194  t0 = mx::get_curr_time();
195 
196  size_t totWritten = 0;
197  while( totWritten < buffWrite.size())
198  {
199  int timeoutCurrent = timeoutWrite - (mx::get_curr_time()-t0)*1000;
200  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONWRITE;
201 
202  int rv = poll( &pfd, 1, timeoutCurrent);
203  if( rv == 0 ) return TTY_E_TIMEOUTONWRITEPOLL;
204  else if( rv < 0 ) return TTY_E_ERRORONWRITEPOLL;
205 
206  rv = write(fd, buffWrite.c_str()+totWritten, buffWrite.size()-totWritten);
207  if(rv < 0) return TTY_E_ERRORONWRITE;
208 
209  //sleep(1);
210  #ifdef TTY_DEBUG
211  std::cerr << "Wrote " << rv << " chars of " << buffWrite.size() << "\n";
212  #endif
213 
214  totWritten += rv;
215 
216  if( ( mx::get_curr_time()-t0)*1000 > timeoutWrite ) return TTY_E_TIMEOUTONWRITE;
217  }
218 
219  return TTY_E_NOERROR;
220 }
221 
222 /// Read from a tty console indicated by a file-descriptor, until an end of transmission string is read.
223 /**
224  * \returns TTY_E_NOERROR on success
225  * \returns TTY_E_TIMEOUTONREADPOLL if the poll times out.
226  * \returns TTY_E_ERRORONREADPOLL if an error is returned by poll.
227  * \returns TTY_E_TIMEOUTONREAD if a timeout occurs during the read.
228  * \returns TTY_E_ERRORONREAD if an error occurs reading from the file.
229  *
230  * \ingroup tty
231  */
232 inline
233 int ttyRead( std::string & strRead, ///< [out] The string in which to store the output.
234  const std::string & eot, ///< [in] A sequence of characters which indicates the end of transmission.
235  int fd, ///< [in] The file descriptor of the open tty.
236  int timeoutRead ///< [in] The timeout in milliseconds.
237  )
238 {
239  int rv;
240  int timeoutCurrent;
241  double t0;
242 
243  struct pollfd pfd;
244 
245  errno = 0;
246 
247  pfd.fd = fd;
248  pfd.events = POLLIN;
249 
250  strRead.clear();
251  char buffRead[TTY_BUFFSIZE];
252 
253  //Start timeout clock for reading.
254  t0 = mx::get_curr_time();
255  timeoutCurrent = timeoutRead;
256 
257  //Now read the response up to the eot.
258  strRead.clear();
259 
260  rv = poll( &pfd, 1, timeoutCurrent);
261  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
262  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
263 
264  rv = read(fd, buffRead, TTY_BUFFSIZE);
265  if( rv < 0 ) return TTY_E_ERRORONREAD;
266 
267  strRead.append( buffRead, rv);
268 
269  while( !isEndOfTrans(strRead, eot) )
270  {
271  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
272  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
273 
274  rv = poll( &pfd, 1, timeoutCurrent);
275  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
276  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
277 
278  rv = read(fd, buffRead, TTY_BUFFSIZE);
279  if( rv < 0 ) return TTY_E_ERRORONREAD;
280  buffRead[rv] ='\0';
281 
282  strRead.append( buffRead, rv);
283  #ifdef TTY_DEBUG
284  std::cerr << "ttyRead: read " << rv << " bytes. buffRead=" << buffRead << "\n";
285  #endif
286  }
287 
288 
289  return TTY_E_NOERROR;
290 
291 
292 }
293 
294 /// Write to a tty on an open file descriptor, then get the result.
295 /** The read is conducted until an end-of-transmission string is received.
296  * Echo characters are swallowed if desired.
297  *
298  * \returns TTY_E_NOERROR on success
299  * \returns TTY_E_TIMEOUTONWRITEPOLL if the poll times out.
300  * \returns TTY_E_ERRORONWRITEPOLL if an error is returned by poll.
301  * \returns TTY_E_TIMEOUTONWRITE if a timeout occurs during the write.
302  * \returns TTY_E_ERRORONWRITE if an error occurs writing to the file.
303  * \returns TTY_E_TIMEOUTONREADPOLL if the poll times out.
304  * \returns TTY_E_ERRORONREADPOLL if an error is returned by poll.
305  * \returns TTY_E_TIMEOUTONREAD if a timeout occurs during the read.
306  * \returns TTY_E_ERRORONREAD if an error occurs reading from the file.
307  *
308  * \ingroup tty
309  */
310 inline
311 int ttyWriteRead( std::string & strRead, ///< [out] The string in which to store the output.
312  const std::string & strWrite, ///< [in] The characters to write to the tty.
313  const std::string & eot, ///< [in] A sequence of characters which indicates the end of transmission.
314  bool swallowEcho, ///< [in] If true, strWrite.size() characters are read after the write
315  int fd, ///< [in] The file descriptor of the open tty.
316  int timeoutWrite, ///< [in] The write timeout in milliseconds.
317  int timeoutRead ///< [in] The read timeout in milliseconds.
318  )
319 {
320  strRead.clear();
321 
322  int rv;
323 
324  //Write First
325  rv = ttyWrite( strWrite, fd, timeoutWrite);
326  if(rv != TTY_E_NOERROR) return rv;
327 
328 
329 
330  //Now read response from console
331  int timeoutCurrent;
332  double t0;
333 
334  struct pollfd pfd;
335  pfd.fd = fd;
336  pfd.events = POLLIN;
337 
338 
339  //Start timeout clock for reading.
340  t0 = mx::get_curr_time();;
341 
342  if(swallowEcho)
343  {
344  size_t totrv = 0;
345  char buffRead[TTY_BUFFSIZE];
346 
347  //First swallow the echo.
348  while( totrv <= strWrite.size() )
349  {
350  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
351  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
352 
353  rv = poll( &pfd, 1, timeoutCurrent);
354  if( rv == 0 ) return TTY_E_TIMEOUTONREADPOLL;
355  if( rv < 0 ) return TTY_E_ERRORONREADPOLL;
356 
357  rv = read(fd, buffRead, TTY_BUFFSIZE);
358  if( rv < 0 ) return TTY_E_ERRORONREAD;
359 
360  totrv += rv;
361  }
362  }
363 
364  timeoutCurrent = timeoutRead - (mx::get_curr_time()-t0)*1000;
365  if(timeoutCurrent < 0) return TTY_E_TIMEOUTONREAD;
366 
367  //Now read the response up to the eot.
368  return ttyRead(strRead, eot, fd, timeoutCurrent);
369 }
370 
371 
372 
373 } //namespace tty
374 } //namespace MagAOX
375 
376 #endif //tty_ttyIOUtils_hpp
int ttyWrite(const std::string &buffWrite, int fd, int timeoutWrite)
Write to the tty console indicated by a file descriptor.
Definition: ttyIOUtils.hpp:182
#define TTY_E_SETISPEED
Definition: ttyErrors.hpp:15
#define TTY_E_ERRORONREAD
Definition: ttyErrors.hpp:24
#define TTY_E_TCGETATTR
Definition: ttyErrors.hpp:14
#define TTY_E_ERRORONREADPOLL
Definition: ttyErrors.hpp:23
Error numbers for the tty utilities.
#define TTY_E_ERRORONWRITE
Definition: ttyErrors.hpp:20
#define TTY_E_TCSETATTR
Definition: ttyErrors.hpp:17
int ttyOpenRaw(int &fileDescrip, std::string &deviceName, speed_t speed)
Open a file as a raw-mode tty device.
Definition: ttyIOUtils.hpp:101
#define TTY_E_ERRORONWRITEPOLL
Definition: ttyErrors.hpp:19
#define TTY_E_SETOSPEED
Definition: ttyErrors.hpp:16
int telnetCRLF(std::string &telnetStr, const std::string &inputStr)
Replace lone \r and \n with \r\n for telnet-ness.
Definition: ttyIOUtils.hpp:44
#define TTY_E_TIMEOUTONREADPOLL
Definition: ttyErrors.hpp:22
#define TTY_BUFFSIZE
Definition: ttyIOUtils.hpp:27
int ttyRead(std::string &strRead, const std::string &eot, int fd, int timeoutRead)
Read from a tty console indicated by a file-descriptor, until an end of transmission string is read...
Definition: ttyIOUtils.hpp:233
int ttyWriteRead(std::string &strRead, const std::string &strWrite, const std::string &eot, bool swallowEcho, int fd, int timeoutWrite, int timeoutRead)
Write to a tty on an open file descriptor, then get the result.
Definition: ttyIOUtils.hpp:311
#define TTY_E_TIMEOUTONWRITEPOLL
Definition: ttyErrors.hpp:18
#define TTY_E_NOERROR
Definition: ttyErrors.hpp:13
bool isEndOfTrans(const std::string &strRead, const std::string &eot)
Check if the end of the buffer contains the end-of-transmission string.
Definition: ttyIOUtils.hpp:153
#define TTY_E_TIMEOUTONWRITE
Definition: ttyErrors.hpp:21
#define TTY_E_TIMEOUTONREAD
Definition: ttyErrors.hpp:25