00001
00030 #ifndef SICK_LIDAR
00031 #define SICK_LIDAR
00032
00033
00034 #include <new>
00035 #include <string>
00036 #include <iomanip>
00037 #include <iostream>
00038 #include <fcntl.h>
00039 #include <pthread.h>
00040 #include <arpa/inet.h>
00041 #include <sys/time.h>
00042 #include <unistd.h>
00043 #include "SickException.hh"
00044
00045
00046 namespace SickToolbox {
00047
00052 template < class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00053 class SickLIDAR {
00054
00055 public:
00056
00058 SickLIDAR( );
00059
00061 bool IsInitialized() { return _sick_initialized; }
00062
00064 virtual ~SickLIDAR( );
00065
00066 protected:
00067
00069 int _sick_fd;
00070
00072 bool _sick_initialized;
00073
00075 SICK_MONITOR_CLASS *_sick_buffer_monitor;
00076
00078 bool _sick_monitor_running;
00079
00081 virtual void _setupConnection( ) = 0;
00082
00084 virtual void _teardownConnection( ) = 0;
00085
00087 void _startListening( ) throw( SickThreadException );
00088
00090 void _stopListening( ) throw( SickThreadException );
00091
00093 bool _monitorRunning( ) const { return _sick_monitor_running; }
00094
00096 void _setBlockingIO( ) const throw ( SickIOException );
00097
00099 void _setNonBlockingIO( ) const throw ( SickIOException );
00100
00102 void _sendMessage( const SICK_MSG_CLASS &sick_message, const unsigned int byte_interval ) const
00103 throw( SickIOException );
00104
00106 void _recvMessage( SICK_MSG_CLASS &sick_message, const unsigned int timeout_value ) const throw ( SickTimeoutException );
00107
00109 void _recvMessage( SICK_MSG_CLASS &sick_message,
00110 const uint8_t * const byte_sequence,
00111 const unsigned int byte_sequence_length,
00112 const unsigned int timeout_value ) const throw ( SickTimeoutException );
00113
00115 double _computeElapsedTime( const struct timeval &beg_time, const struct timeval &end_time ) const { return ((end_time.tv_sec*1e6)+(end_time.tv_usec))-((beg_time.tv_sec*1e6)+beg_time.tv_usec); }
00116
00118 virtual void _sendMessageAndGetReply( const SICK_MSG_CLASS &send_message,
00119 SICK_MSG_CLASS &recv_message,
00120 const uint8_t * const byte_sequence,
00121 const unsigned int byte_sequence_length,
00122 const unsigned int byte_interval,
00123 const unsigned int timeout_value,
00124 const unsigned int num_tries ) throw( SickTimeoutException, SickIOException);
00125
00126 };
00127
00131 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00132 SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::SickLIDAR( ) :
00133 _sick_fd(0), _sick_initialized(false), _sick_buffer_monitor(NULL), _sick_monitor_running(false) {
00134
00135 try {
00136
00137 _sick_buffer_monitor = new SICK_MONITOR_CLASS;
00138 }
00139 catch ( std::bad_alloc &allocation_exception ) {
00140 std::cerr << "SickLIDAR::SickLIDAR: Allocation error - " << allocation_exception.what() << std::endl;
00141 }
00142
00143 }
00144
00148 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00149 SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::~SickLIDAR( ) {
00150
00151
00152 if (_sick_buffer_monitor) {
00153 delete _sick_buffer_monitor;
00154 }
00155
00156 }
00157
00161 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00162 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_startListening( ) throw( SickThreadException ) {
00163
00164
00165 try {
00166 _sick_buffer_monitor->StartMonitor(_sick_fd);
00167 }
00168
00169
00170 catch(SickThreadException &sick_thread_exception) {
00171 std::cerr << sick_thread_exception.what() << std::endl;
00172 throw;
00173 }
00174
00175
00176 catch(...) {
00177 std::cerr << "SickLIDAR::_startListening: Unknown exception!!!" << std::endl;
00178 throw;
00179 }
00180
00181
00182 _sick_monitor_running = true;
00183
00184 }
00185
00189 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00190 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_stopListening( ) throw( SickThreadException ) {
00191
00192
00193 try {
00194 _sick_buffer_monitor->StopMonitor();
00195 }
00196
00197
00198 catch(SickThreadException &sick_thread_exception) {
00199 std::cerr << sick_thread_exception.what() << std::endl;
00200 throw;
00201 }
00202
00203
00204 catch(...) {
00205 std::cerr << "SickLIDAR::_stopListening: Unknown exception!!!" << std::endl;
00206 throw;
00207 }
00208
00209
00210 _sick_monitor_running = false;
00211
00212 }
00213
00217 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00218 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_setBlockingIO( ) const throw( SickIOException ) {
00219
00220
00221 int fd_flags = 0;
00222 if((fd_flags = fcntl(_sick_fd,F_GETFL)) < 0) {
00223 throw SickIOException("SickLIDAR::_setNonBlocking: fcntl failed!");
00224 }
00225
00226
00227 if(fcntl(_sick_fd,F_SETFL,fd_flags & (~O_NONBLOCK)) < 0) {
00228 throw SickIOException("SickLIDAR::_setNonBlocking: fcntl failed!");
00229 }
00230
00231 }
00232
00236 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00237 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_setNonBlockingIO( ) const throw ( SickIOException ) {
00238
00239
00240 int fd_flags = 0;
00241 if((fd_flags = fcntl(_sick_fd,F_GETFL)) < 0) {
00242 throw SickIOException("SickLIDAR::_setNonBlockingIO: fcntl failed!");
00243 }
00244
00245
00246 if(fcntl(_sick_fd,F_SETFL,fd_flags | O_NONBLOCK) < 0) {
00247 throw SickIOException("SickLIDAR::_setNonBlockingIO: fcntl failed!");
00248 }
00249
00250 }
00251
00257 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00258 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_sendMessage( const SICK_MSG_CLASS &sick_message, const unsigned int byte_interval ) const
00259 throw( SickIOException ) {
00260
00261 uint8_t message_buffer[SICK_MSG_CLASS::MESSAGE_MAX_LENGTH] = {0};
00262
00263
00264 sick_message.GetMessage(message_buffer);
00265 unsigned int message_length = sick_message.GetMessageLength();
00266
00267
00268 if (byte_interval == 0) {
00269
00270
00271 if ((unsigned int)write(_sick_fd,message_buffer,message_length) != message_length) {
00272 throw SickIOException("SickLIDAR::_sendMessage: write() failed!");
00273 }
00274
00275 }
00276 else {
00277
00278
00279 for (unsigned int i = 0; i < message_length; i++) {
00280
00281
00282 if (write(_sick_fd,&message_buffer[i],1) != 1) {
00283 throw SickIOException("SickLIDAR::_sendMessage: write() failed!");
00284 }
00285
00286
00287 usleep(byte_interval);
00288 }
00289
00290 }
00291
00292 }
00293
00300 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00301 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_recvMessage( SICK_MSG_CLASS &sick_message,
00302 const unsigned int timeout_value ) const throw ( SickTimeoutException ) {
00303
00304
00305 struct timeval beg_time, end_time;
00306
00307
00308 gettimeofday(&beg_time,NULL);
00309
00310
00311 while(!_sick_buffer_monitor->GetNextMessageFromMonitor(sick_message)) {
00312
00313
00314 usleep(1000);
00315
00316
00317 gettimeofday(&end_time,NULL);
00318 if (_computeElapsedTime(beg_time,end_time) > timeout_value) {
00319 throw SickTimeoutException("SickLIDAR::_recvMessage: Timeout occurred!");
00320 }
00321
00322 }
00323
00324 }
00325
00336 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00337 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_recvMessage( SICK_MSG_CLASS &sick_message,
00338 const uint8_t * const byte_sequence,
00339 const unsigned int byte_sequence_length,
00340 const unsigned int timeout_value ) const throw( SickTimeoutException ) {
00341
00342
00343 uint8_t payload_buffer[SICK_MSG_CLASS::MESSAGE_PAYLOAD_MAX_LENGTH];
00344
00345
00346 struct timeval beg_time, end_time;
00347
00348
00349 SICK_MSG_CLASS curr_message;
00350
00351
00352 gettimeofday(&beg_time,NULL);
00353
00354
00355 for(;;) {
00356
00357
00358 unsigned int i = 0;
00359 if (_sick_buffer_monitor->GetNextMessageFromMonitor(curr_message)) {
00360
00361
00362 curr_message.GetPayloadSubregion(payload_buffer,0,byte_sequence_length-1);
00363
00364
00365 for (i=0; (i < byte_sequence_length) && (payload_buffer[i] == byte_sequence[i]); i++);
00366
00367
00368 if (i == byte_sequence_length) {
00369 sick_message = curr_message;
00370 break;
00371 }
00372
00373 }
00374
00375
00376 usleep(1000);
00377
00378
00379 gettimeofday(&end_time,NULL);
00380 if (_computeElapsedTime(beg_time,end_time) > timeout_value) {
00381 throw SickTimeoutException();
00382 }
00383
00384 }
00385
00386 }
00387
00395 template< class SICK_MONITOR_CLASS, class SICK_MSG_CLASS >
00396 void SickLIDAR< SICK_MONITOR_CLASS, SICK_MSG_CLASS >::_sendMessageAndGetReply( const SICK_MSG_CLASS &send_message,
00397 SICK_MSG_CLASS &recv_message,
00398 const uint8_t * const byte_sequence,
00399 const unsigned int byte_sequence_length,
00400 const unsigned int byte_interval,
00401 const unsigned int timeout_value,
00402 const unsigned int num_tries )
00403 throw( SickTimeoutException, SickIOException ) {
00404
00405
00406 for(unsigned int i = 0; i < num_tries; i++) {
00407
00408 try {
00409
00410
00411 _sendMessage(send_message,byte_interval);
00412
00413
00414 _recvMessage(recv_message,byte_sequence,byte_sequence_length,timeout_value);
00415
00416
00417 break;
00418
00419 }
00420
00421
00422 catch (SickTimeoutException &sick_timeout) {
00423
00424
00425 if (i == num_tries - 1) {
00426 throw SickTimeoutException("SickLIDAR::_sendMessageAndGetReply: Attempted max number of tries w/o success!");
00427 }
00428
00429
00430 std::cerr << sick_timeout.what() << " " << num_tries - i - 1 << " tries remaining" << std::endl;
00431
00432 }
00433
00434
00435 catch (SickIOException &sick_io_error) {
00436 std::cerr << sick_io_error.what() << std::endl;
00437 throw;
00438 }
00439
00440
00441 catch (...) {
00442 std::cerr << "SickLIDAR::_sendMessageAndGetReply: Unknown exception!!!" << std::endl;
00443 throw;
00444 }
00445
00446 }
00447
00448 }
00449
00450 }
00451
00452 #endif