capstone_finish

Dependencies:   BufferedSerial motor_sn7544

Committer:
Jeonghoon
Date:
Tue Aug 13 05:53:22 2019 +0000
Revision:
3:2a3664dc6634
Child:
10:ca4e4062701a
with ROS;

Who changed what in which revision?

UserRevisionLine numberNew contents of line
Jeonghoon 3:2a3664dc6634 1 /*
Jeonghoon 3:2a3664dc6634 2 * Software License Agreement (BSD License)
Jeonghoon 3:2a3664dc6634 3 *
Jeonghoon 3:2a3664dc6634 4 * Copyright (c) 2011, Willow Garage, Inc.
Jeonghoon 3:2a3664dc6634 5 * All rights reserved.
Jeonghoon 3:2a3664dc6634 6 *
Jeonghoon 3:2a3664dc6634 7 * Redistribution and use in source and binary forms, with or without
Jeonghoon 3:2a3664dc6634 8 * modification, are permitted provided that the following conditions
Jeonghoon 3:2a3664dc6634 9 * are met:
Jeonghoon 3:2a3664dc6634 10 *
Jeonghoon 3:2a3664dc6634 11 * * Redistributions of source code must retain the above copyright
Jeonghoon 3:2a3664dc6634 12 * notice, this list of conditions and the following disclaimer.
Jeonghoon 3:2a3664dc6634 13 * * Redistributions in binary form must reproduce the above
Jeonghoon 3:2a3664dc6634 14 * copyright notice, this list of conditions and the following
Jeonghoon 3:2a3664dc6634 15 * disclaimer in the documentation and/or other materials provided
Jeonghoon 3:2a3664dc6634 16 * with the distribution.
Jeonghoon 3:2a3664dc6634 17 * * Neither the name of Willow Garage, Inc. nor the names of its
Jeonghoon 3:2a3664dc6634 18 * contributors may be used to endorse or promote prducts derived
Jeonghoon 3:2a3664dc6634 19 * from this software without specific prior written permission.
Jeonghoon 3:2a3664dc6634 20 *
Jeonghoon 3:2a3664dc6634 21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
Jeonghoon 3:2a3664dc6634 22 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
Jeonghoon 3:2a3664dc6634 23 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
Jeonghoon 3:2a3664dc6634 24 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
Jeonghoon 3:2a3664dc6634 25 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
Jeonghoon 3:2a3664dc6634 26 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
Jeonghoon 3:2a3664dc6634 27 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
Jeonghoon 3:2a3664dc6634 28 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
Jeonghoon 3:2a3664dc6634 29 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
Jeonghoon 3:2a3664dc6634 30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
Jeonghoon 3:2a3664dc6634 31 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
Jeonghoon 3:2a3664dc6634 32 * POSSIBILITY OF SUCH DAMAGE.
Jeonghoon 3:2a3664dc6634 33 */
Jeonghoon 3:2a3664dc6634 34
Jeonghoon 3:2a3664dc6634 35 #ifndef ROS_NODE_HANDLE_H_
Jeonghoon 3:2a3664dc6634 36 #define ROS_NODE_HANDLE_H_
Jeonghoon 3:2a3664dc6634 37
Jeonghoon 3:2a3664dc6634 38 #include <stdint.h>
Jeonghoon 3:2a3664dc6634 39
Jeonghoon 3:2a3664dc6634 40 #include "std_msgs/Time.h"
Jeonghoon 3:2a3664dc6634 41 #include "rosserial_msgs/TopicInfo.h"
Jeonghoon 3:2a3664dc6634 42 #include "rosserial_msgs/Log.h"
Jeonghoon 3:2a3664dc6634 43 #include "rosserial_msgs/RequestParam.h"
Jeonghoon 3:2a3664dc6634 44
Jeonghoon 3:2a3664dc6634 45 #define SYNC_SECONDS 5
Jeonghoon 3:2a3664dc6634 46
Jeonghoon 3:2a3664dc6634 47 #define MODE_FIRST_FF 0
Jeonghoon 3:2a3664dc6634 48 /*
Jeonghoon 3:2a3664dc6634 49 * The second sync byte is a protocol version. It's value is 0xff for the first
Jeonghoon 3:2a3664dc6634 50 * version of the rosserial protocol (used up to hydro), 0xfe for the second version
Jeonghoon 3:2a3664dc6634 51 * (introduced in hydro), 0xfd for the next, and so on. Its purpose is to enable
Jeonghoon 3:2a3664dc6634 52 * detection of mismatched protocol versions (e.g. hydro rosserial_python with groovy
Jeonghoon 3:2a3664dc6634 53 * rosserial_arduino. It must be changed in both this file and in
Jeonghoon 3:2a3664dc6634 54 * rosserial_python/src/rosserial_python/SerialClient.py
Jeonghoon 3:2a3664dc6634 55 */
Jeonghoon 3:2a3664dc6634 56 #define MODE_PROTOCOL_VER 1
Jeonghoon 3:2a3664dc6634 57 #define PROTOCOL_VER1 0xff // through groovy
Jeonghoon 3:2a3664dc6634 58 #define PROTOCOL_VER2 0xfe // in hydro
Jeonghoon 3:2a3664dc6634 59 #define PROTOCOL_VER PROTOCOL_VER2
Jeonghoon 3:2a3664dc6634 60 #define MODE_SIZE_L 2
Jeonghoon 3:2a3664dc6634 61 #define MODE_SIZE_H 3
Jeonghoon 3:2a3664dc6634 62 #define MODE_SIZE_CHECKSUM 4 // checksum for msg size received from size L and H
Jeonghoon 3:2a3664dc6634 63 #define MODE_TOPIC_L 5 // waiting for topic id
Jeonghoon 3:2a3664dc6634 64 #define MODE_TOPIC_H 6
Jeonghoon 3:2a3664dc6634 65 #define MODE_MESSAGE 7
Jeonghoon 3:2a3664dc6634 66 #define MODE_MSG_CHECKSUM 8 // checksum for msg and topic id
Jeonghoon 3:2a3664dc6634 67
Jeonghoon 3:2a3664dc6634 68
Jeonghoon 3:2a3664dc6634 69 #define MSG_TIMEOUT 20 //20 milliseconds to recieve all of message data
Jeonghoon 3:2a3664dc6634 70
Jeonghoon 3:2a3664dc6634 71 #include "ros/msg.h"
Jeonghoon 3:2a3664dc6634 72
Jeonghoon 3:2a3664dc6634 73 namespace ros {
Jeonghoon 3:2a3664dc6634 74
Jeonghoon 3:2a3664dc6634 75 class NodeHandleBase_{
Jeonghoon 3:2a3664dc6634 76 public:
Jeonghoon 3:2a3664dc6634 77 virtual int publish(int id, const Msg* msg)=0;
Jeonghoon 3:2a3664dc6634 78 virtual int spinOnce()=0;
Jeonghoon 3:2a3664dc6634 79 virtual bool connected()=0;
Jeonghoon 3:2a3664dc6634 80 };
Jeonghoon 3:2a3664dc6634 81 }
Jeonghoon 3:2a3664dc6634 82
Jeonghoon 3:2a3664dc6634 83 #include "ros/publisher.h"
Jeonghoon 3:2a3664dc6634 84 #include "ros/subscriber.h"
Jeonghoon 3:2a3664dc6634 85 #include "ros/service_server.h"
Jeonghoon 3:2a3664dc6634 86 #include "ros/service_client.h"
Jeonghoon 3:2a3664dc6634 87
Jeonghoon 3:2a3664dc6634 88 namespace ros {
Jeonghoon 3:2a3664dc6634 89
Jeonghoon 3:2a3664dc6634 90 using rosserial_msgs::TopicInfo;
Jeonghoon 3:2a3664dc6634 91
Jeonghoon 3:2a3664dc6634 92 /* Node Handle */
Jeonghoon 3:2a3664dc6634 93 template<class Hardware,
Jeonghoon 3:2a3664dc6634 94 int MAX_SUBSCRIBERS=25,
Jeonghoon 3:2a3664dc6634 95 int MAX_PUBLISHERS=25,
Jeonghoon 3:2a3664dc6634 96 int INPUT_SIZE=512,
Jeonghoon 3:2a3664dc6634 97 int OUTPUT_SIZE=512>
Jeonghoon 3:2a3664dc6634 98 class NodeHandle_ : public NodeHandleBase_
Jeonghoon 3:2a3664dc6634 99 {
Jeonghoon 3:2a3664dc6634 100 protected:
Jeonghoon 3:2a3664dc6634 101 Hardware hardware_;
Jeonghoon 3:2a3664dc6634 102
Jeonghoon 3:2a3664dc6634 103 /* time used for syncing */
Jeonghoon 3:2a3664dc6634 104 uint32_t rt_time;
Jeonghoon 3:2a3664dc6634 105
Jeonghoon 3:2a3664dc6634 106 /* used for computing current time */
Jeonghoon 3:2a3664dc6634 107 uint32_t sec_offset, nsec_offset;
Jeonghoon 3:2a3664dc6634 108
Jeonghoon 3:2a3664dc6634 109 uint8_t message_in[INPUT_SIZE];
Jeonghoon 3:2a3664dc6634 110 uint8_t message_out[OUTPUT_SIZE];
Jeonghoon 3:2a3664dc6634 111
Jeonghoon 3:2a3664dc6634 112 Publisher * publishers[MAX_PUBLISHERS];
Jeonghoon 3:2a3664dc6634 113 Subscriber_ * subscribers[MAX_SUBSCRIBERS];
Jeonghoon 3:2a3664dc6634 114
Jeonghoon 3:2a3664dc6634 115 /*
Jeonghoon 3:2a3664dc6634 116 * Setup Functions
Jeonghoon 3:2a3664dc6634 117 */
Jeonghoon 3:2a3664dc6634 118 public:
Jeonghoon 3:2a3664dc6634 119 NodeHandle_() : configured_(false) {
Jeonghoon 3:2a3664dc6634 120
Jeonghoon 3:2a3664dc6634 121 for(unsigned int i=0; i< MAX_PUBLISHERS; i++)
Jeonghoon 3:2a3664dc6634 122 publishers[i] = 0;
Jeonghoon 3:2a3664dc6634 123
Jeonghoon 3:2a3664dc6634 124 for(unsigned int i=0; i< MAX_SUBSCRIBERS; i++)
Jeonghoon 3:2a3664dc6634 125 subscribers[i] = 0;
Jeonghoon 3:2a3664dc6634 126
Jeonghoon 3:2a3664dc6634 127 for(unsigned int i=0; i< INPUT_SIZE; i++)
Jeonghoon 3:2a3664dc6634 128 message_in[i] = 0;
Jeonghoon 3:2a3664dc6634 129
Jeonghoon 3:2a3664dc6634 130 for(unsigned int i=0; i< OUTPUT_SIZE; i++)
Jeonghoon 3:2a3664dc6634 131 message_out[i] = 0;
Jeonghoon 3:2a3664dc6634 132
Jeonghoon 3:2a3664dc6634 133 req_param_resp.ints_length = 0;
Jeonghoon 3:2a3664dc6634 134 req_param_resp.ints = NULL;
Jeonghoon 3:2a3664dc6634 135 req_param_resp.floats_length = 0;
Jeonghoon 3:2a3664dc6634 136 req_param_resp.floats = NULL;
Jeonghoon 3:2a3664dc6634 137 req_param_resp.ints_length = 0;
Jeonghoon 3:2a3664dc6634 138 req_param_resp.ints = NULL;
Jeonghoon 3:2a3664dc6634 139 }
Jeonghoon 3:2a3664dc6634 140
Jeonghoon 3:2a3664dc6634 141 Hardware* getHardware(){
Jeonghoon 3:2a3664dc6634 142 return &hardware_;
Jeonghoon 3:2a3664dc6634 143 }
Jeonghoon 3:2a3664dc6634 144
Jeonghoon 3:2a3664dc6634 145 /* Start serial, initialize buffers */
Jeonghoon 3:2a3664dc6634 146 void initNode(){
Jeonghoon 3:2a3664dc6634 147 hardware_.init();
Jeonghoon 3:2a3664dc6634 148 mode_ = 0;
Jeonghoon 3:2a3664dc6634 149 bytes_ = 0;
Jeonghoon 3:2a3664dc6634 150 index_ = 0;
Jeonghoon 3:2a3664dc6634 151 topic_ = 0;
Jeonghoon 3:2a3664dc6634 152 };
Jeonghoon 3:2a3664dc6634 153
Jeonghoon 3:2a3664dc6634 154 /* Start a named port, which may be network server IP, initialize buffers */
Jeonghoon 3:2a3664dc6634 155 void initNode(char *portName){
Jeonghoon 3:2a3664dc6634 156 hardware_.init(portName);
Jeonghoon 3:2a3664dc6634 157 mode_ = 0;
Jeonghoon 3:2a3664dc6634 158 bytes_ = 0;
Jeonghoon 3:2a3664dc6634 159 index_ = 0;
Jeonghoon 3:2a3664dc6634 160 topic_ = 0;
Jeonghoon 3:2a3664dc6634 161 };
Jeonghoon 3:2a3664dc6634 162
Jeonghoon 3:2a3664dc6634 163 protected:
Jeonghoon 3:2a3664dc6634 164 //State machine variables for spinOnce
Jeonghoon 3:2a3664dc6634 165 int mode_;
Jeonghoon 3:2a3664dc6634 166 int bytes_;
Jeonghoon 3:2a3664dc6634 167 int topic_;
Jeonghoon 3:2a3664dc6634 168 int index_;
Jeonghoon 3:2a3664dc6634 169 int checksum_;
Jeonghoon 3:2a3664dc6634 170
Jeonghoon 3:2a3664dc6634 171 bool configured_;
Jeonghoon 3:2a3664dc6634 172
Jeonghoon 3:2a3664dc6634 173 /* used for syncing the time */
Jeonghoon 3:2a3664dc6634 174 uint32_t last_sync_time;
Jeonghoon 3:2a3664dc6634 175 uint32_t last_sync_receive_time;
Jeonghoon 3:2a3664dc6634 176 uint32_t last_msg_timeout_time;
Jeonghoon 3:2a3664dc6634 177
Jeonghoon 3:2a3664dc6634 178 public:
Jeonghoon 3:2a3664dc6634 179 /* This function goes in your loop() function, it handles
Jeonghoon 3:2a3664dc6634 180 * serial input and callbacks for subscribers.
Jeonghoon 3:2a3664dc6634 181 */
Jeonghoon 3:2a3664dc6634 182
Jeonghoon 3:2a3664dc6634 183
Jeonghoon 3:2a3664dc6634 184 virtual int spinOnce(){
Jeonghoon 3:2a3664dc6634 185
Jeonghoon 3:2a3664dc6634 186 /* restart if timed out */
Jeonghoon 3:2a3664dc6634 187 uint32_t c_time = hardware_.time();
Jeonghoon 3:2a3664dc6634 188 if( (c_time - last_sync_receive_time) > (SYNC_SECONDS*2200) ){
Jeonghoon 3:2a3664dc6634 189 configured_ = false;
Jeonghoon 3:2a3664dc6634 190 }
Jeonghoon 3:2a3664dc6634 191
Jeonghoon 3:2a3664dc6634 192 /* reset if message has timed out */
Jeonghoon 3:2a3664dc6634 193 if ( mode_ != MODE_FIRST_FF){
Jeonghoon 3:2a3664dc6634 194 if (c_time > last_msg_timeout_time){
Jeonghoon 3:2a3664dc6634 195 mode_ = MODE_FIRST_FF;
Jeonghoon 3:2a3664dc6634 196 }
Jeonghoon 3:2a3664dc6634 197 }
Jeonghoon 3:2a3664dc6634 198
Jeonghoon 3:2a3664dc6634 199 /* while available buffer, read data */
Jeonghoon 3:2a3664dc6634 200 while( true )
Jeonghoon 3:2a3664dc6634 201 {
Jeonghoon 3:2a3664dc6634 202 int data = hardware_.read();
Jeonghoon 3:2a3664dc6634 203 if( data < 0 )
Jeonghoon 3:2a3664dc6634 204 break;
Jeonghoon 3:2a3664dc6634 205 checksum_ += data;
Jeonghoon 3:2a3664dc6634 206 if( mode_ == MODE_MESSAGE ){ /* message data being recieved */
Jeonghoon 3:2a3664dc6634 207 message_in[index_++] = data;
Jeonghoon 3:2a3664dc6634 208 bytes_--;
Jeonghoon 3:2a3664dc6634 209 if(bytes_ == 0) /* is message complete? if so, checksum */
Jeonghoon 3:2a3664dc6634 210 mode_ = MODE_MSG_CHECKSUM;
Jeonghoon 3:2a3664dc6634 211 }else if( mode_ == MODE_FIRST_FF ){
Jeonghoon 3:2a3664dc6634 212 if(data == 0xff){
Jeonghoon 3:2a3664dc6634 213 mode_++;
Jeonghoon 3:2a3664dc6634 214 last_msg_timeout_time = c_time + MSG_TIMEOUT;
Jeonghoon 3:2a3664dc6634 215 }
Jeonghoon 3:2a3664dc6634 216 else if( hardware_.time() - c_time > (SYNC_SECONDS)){
Jeonghoon 3:2a3664dc6634 217 /* We have been stuck in spinOnce too long, return error */
Jeonghoon 3:2a3664dc6634 218 configured_=false;
Jeonghoon 3:2a3664dc6634 219 return -2;
Jeonghoon 3:2a3664dc6634 220 }
Jeonghoon 3:2a3664dc6634 221 }else if( mode_ == MODE_PROTOCOL_VER ){
Jeonghoon 3:2a3664dc6634 222 if(data == PROTOCOL_VER){
Jeonghoon 3:2a3664dc6634 223 mode_++;
Jeonghoon 3:2a3664dc6634 224 }else{
Jeonghoon 3:2a3664dc6634 225 mode_ = MODE_FIRST_FF;
Jeonghoon 3:2a3664dc6634 226 if (configured_ == false)
Jeonghoon 3:2a3664dc6634 227 requestSyncTime(); /* send a msg back showing our protocol version */
Jeonghoon 3:2a3664dc6634 228 }
Jeonghoon 3:2a3664dc6634 229 }else if( mode_ == MODE_SIZE_L ){ /* bottom half of message size */
Jeonghoon 3:2a3664dc6634 230 bytes_ = data;
Jeonghoon 3:2a3664dc6634 231 index_ = 0;
Jeonghoon 3:2a3664dc6634 232 mode_++;
Jeonghoon 3:2a3664dc6634 233 checksum_ = data; /* first byte for calculating size checksum */
Jeonghoon 3:2a3664dc6634 234 }else if( mode_ == MODE_SIZE_H ){ /* top half of message size */
Jeonghoon 3:2a3664dc6634 235 bytes_ += data<<8;
Jeonghoon 3:2a3664dc6634 236 mode_++;
Jeonghoon 3:2a3664dc6634 237 }else if( mode_ == MODE_SIZE_CHECKSUM ){
Jeonghoon 3:2a3664dc6634 238 if( (checksum_%256) == 255)
Jeonghoon 3:2a3664dc6634 239 mode_++;
Jeonghoon 3:2a3664dc6634 240 else
Jeonghoon 3:2a3664dc6634 241 mode_ = MODE_FIRST_FF; /* Abandon the frame if the msg len is wrong */
Jeonghoon 3:2a3664dc6634 242 }else if( mode_ == MODE_TOPIC_L ){ /* bottom half of topic id */
Jeonghoon 3:2a3664dc6634 243 topic_ = data;
Jeonghoon 3:2a3664dc6634 244 mode_++;
Jeonghoon 3:2a3664dc6634 245 checksum_ = data; /* first byte included in checksum */
Jeonghoon 3:2a3664dc6634 246 }else if( mode_ == MODE_TOPIC_H ){ /* top half of topic id */
Jeonghoon 3:2a3664dc6634 247 topic_ += data<<8;
Jeonghoon 3:2a3664dc6634 248 mode_ = MODE_MESSAGE;
Jeonghoon 3:2a3664dc6634 249 if(bytes_ == 0)
Jeonghoon 3:2a3664dc6634 250 mode_ = MODE_MSG_CHECKSUM;
Jeonghoon 3:2a3664dc6634 251 }else if( mode_ == MODE_MSG_CHECKSUM ){ /* do checksum */
Jeonghoon 3:2a3664dc6634 252 mode_ = MODE_FIRST_FF;
Jeonghoon 3:2a3664dc6634 253 if( (checksum_%256) == 255){
Jeonghoon 3:2a3664dc6634 254 if(topic_ == TopicInfo::ID_PUBLISHER){
Jeonghoon 3:2a3664dc6634 255 requestSyncTime();
Jeonghoon 3:2a3664dc6634 256 negotiateTopics();
Jeonghoon 3:2a3664dc6634 257 last_sync_time = c_time;
Jeonghoon 3:2a3664dc6634 258 last_sync_receive_time = c_time;
Jeonghoon 3:2a3664dc6634 259 return -1;
Jeonghoon 3:2a3664dc6634 260 }else if(topic_ == TopicInfo::ID_TIME){
Jeonghoon 3:2a3664dc6634 261 syncTime(message_in);
Jeonghoon 3:2a3664dc6634 262 }else if (topic_ == TopicInfo::ID_PARAMETER_REQUEST){
Jeonghoon 3:2a3664dc6634 263 req_param_resp.deserialize(message_in);
Jeonghoon 3:2a3664dc6634 264 param_recieved= true;
Jeonghoon 3:2a3664dc6634 265 }else if(topic_ == TopicInfo::ID_TX_STOP){
Jeonghoon 3:2a3664dc6634 266 configured_ = false;
Jeonghoon 3:2a3664dc6634 267 }else{
Jeonghoon 3:2a3664dc6634 268 if(subscribers[topic_-100])
Jeonghoon 3:2a3664dc6634 269 subscribers[topic_-100]->callback( message_in );
Jeonghoon 3:2a3664dc6634 270 }
Jeonghoon 3:2a3664dc6634 271 }
Jeonghoon 3:2a3664dc6634 272 }
Jeonghoon 3:2a3664dc6634 273 }
Jeonghoon 3:2a3664dc6634 274
Jeonghoon 3:2a3664dc6634 275 /* occasionally sync time */
Jeonghoon 3:2a3664dc6634 276 if( configured_ && ((c_time-last_sync_time) > (SYNC_SECONDS*500) )){
Jeonghoon 3:2a3664dc6634 277 requestSyncTime();
Jeonghoon 3:2a3664dc6634 278 last_sync_time = c_time;
Jeonghoon 3:2a3664dc6634 279 }
Jeonghoon 3:2a3664dc6634 280
Jeonghoon 3:2a3664dc6634 281 return 0;
Jeonghoon 3:2a3664dc6634 282 }
Jeonghoon 3:2a3664dc6634 283
Jeonghoon 3:2a3664dc6634 284
Jeonghoon 3:2a3664dc6634 285 /* Are we connected to the PC? */
Jeonghoon 3:2a3664dc6634 286 virtual bool connected() {
Jeonghoon 3:2a3664dc6634 287 return configured_;
Jeonghoon 3:2a3664dc6634 288 };
Jeonghoon 3:2a3664dc6634 289
Jeonghoon 3:2a3664dc6634 290 /********************************************************************
Jeonghoon 3:2a3664dc6634 291 * Time functions
Jeonghoon 3:2a3664dc6634 292 */
Jeonghoon 3:2a3664dc6634 293
Jeonghoon 3:2a3664dc6634 294 void requestSyncTime()
Jeonghoon 3:2a3664dc6634 295 {
Jeonghoon 3:2a3664dc6634 296 std_msgs::Time t;
Jeonghoon 3:2a3664dc6634 297 publish(TopicInfo::ID_TIME, &t);
Jeonghoon 3:2a3664dc6634 298 rt_time = hardware_.time();
Jeonghoon 3:2a3664dc6634 299 }
Jeonghoon 3:2a3664dc6634 300
Jeonghoon 3:2a3664dc6634 301 void syncTime(uint8_t * data)
Jeonghoon 3:2a3664dc6634 302 {
Jeonghoon 3:2a3664dc6634 303 std_msgs::Time t;
Jeonghoon 3:2a3664dc6634 304 uint32_t offset = hardware_.time() - rt_time;
Jeonghoon 3:2a3664dc6634 305
Jeonghoon 3:2a3664dc6634 306 t.deserialize(data);
Jeonghoon 3:2a3664dc6634 307 t.data.sec += offset/1000;
Jeonghoon 3:2a3664dc6634 308 t.data.nsec += (offset%1000)*1000000UL;
Jeonghoon 3:2a3664dc6634 309
Jeonghoon 3:2a3664dc6634 310 this->setNow(t.data);
Jeonghoon 3:2a3664dc6634 311 last_sync_receive_time = hardware_.time();
Jeonghoon 3:2a3664dc6634 312 }
Jeonghoon 3:2a3664dc6634 313
Jeonghoon 3:2a3664dc6634 314 Time now()
Jeonghoon 3:2a3664dc6634 315 {
Jeonghoon 3:2a3664dc6634 316 uint32_t ms = hardware_.time();
Jeonghoon 3:2a3664dc6634 317 Time current_time;
Jeonghoon 3:2a3664dc6634 318 current_time.sec = ms/1000 + sec_offset;
Jeonghoon 3:2a3664dc6634 319 current_time.nsec = (ms%1000)*1000000UL + nsec_offset;
Jeonghoon 3:2a3664dc6634 320 normalizeSecNSec(current_time.sec, current_time.nsec);
Jeonghoon 3:2a3664dc6634 321 return current_time;
Jeonghoon 3:2a3664dc6634 322 }
Jeonghoon 3:2a3664dc6634 323
Jeonghoon 3:2a3664dc6634 324 void setNow( Time & new_now )
Jeonghoon 3:2a3664dc6634 325 {
Jeonghoon 3:2a3664dc6634 326 uint32_t ms = hardware_.time();
Jeonghoon 3:2a3664dc6634 327 sec_offset = new_now.sec - ms/1000 - 1;
Jeonghoon 3:2a3664dc6634 328 nsec_offset = new_now.nsec - (ms%1000)*1000000UL + 1000000000UL;
Jeonghoon 3:2a3664dc6634 329 normalizeSecNSec(sec_offset, nsec_offset);
Jeonghoon 3:2a3664dc6634 330 }
Jeonghoon 3:2a3664dc6634 331
Jeonghoon 3:2a3664dc6634 332 /********************************************************************
Jeonghoon 3:2a3664dc6634 333 * Topic Management
Jeonghoon 3:2a3664dc6634 334 */
Jeonghoon 3:2a3664dc6634 335
Jeonghoon 3:2a3664dc6634 336 /* Register a new publisher */
Jeonghoon 3:2a3664dc6634 337 bool advertise(Publisher & p)
Jeonghoon 3:2a3664dc6634 338 {
Jeonghoon 3:2a3664dc6634 339 for(int i = 0; i < MAX_PUBLISHERS; i++){
Jeonghoon 3:2a3664dc6634 340 if(publishers[i] == 0){ // empty slot
Jeonghoon 3:2a3664dc6634 341 publishers[i] = &p;
Jeonghoon 3:2a3664dc6634 342 p.id_ = i+100+MAX_SUBSCRIBERS;
Jeonghoon 3:2a3664dc6634 343 p.nh_ = this;
Jeonghoon 3:2a3664dc6634 344 return true;
Jeonghoon 3:2a3664dc6634 345 }
Jeonghoon 3:2a3664dc6634 346 }
Jeonghoon 3:2a3664dc6634 347 return false;
Jeonghoon 3:2a3664dc6634 348 }
Jeonghoon 3:2a3664dc6634 349
Jeonghoon 3:2a3664dc6634 350 /* Register a new subscriber */
Jeonghoon 3:2a3664dc6634 351 template<typename SubscriberT>
Jeonghoon 3:2a3664dc6634 352 bool subscribe(SubscriberT& s){
Jeonghoon 3:2a3664dc6634 353 for(int i = 0; i < MAX_SUBSCRIBERS; i++){
Jeonghoon 3:2a3664dc6634 354 if(subscribers[i] == 0){ // empty slot
Jeonghoon 3:2a3664dc6634 355 subscribers[i] = static_cast<Subscriber_*>(&s);
Jeonghoon 3:2a3664dc6634 356 s.id_ = i+100;
Jeonghoon 3:2a3664dc6634 357 return true;
Jeonghoon 3:2a3664dc6634 358 }
Jeonghoon 3:2a3664dc6634 359 }
Jeonghoon 3:2a3664dc6634 360 return false;
Jeonghoon 3:2a3664dc6634 361 }
Jeonghoon 3:2a3664dc6634 362
Jeonghoon 3:2a3664dc6634 363 /* Register a new Service Server */
Jeonghoon 3:2a3664dc6634 364 template<typename MReq, typename MRes>
Jeonghoon 3:2a3664dc6634 365 bool advertiseService(ServiceServer<MReq,MRes>& srv){
Jeonghoon 3:2a3664dc6634 366 bool v = advertise(srv.pub);
Jeonghoon 3:2a3664dc6634 367 for(int i = 0; i < MAX_SUBSCRIBERS; i++){
Jeonghoon 3:2a3664dc6634 368 if(subscribers[i] == 0){ // empty slot
Jeonghoon 3:2a3664dc6634 369 subscribers[i] = static_cast<Subscriber_*>(&srv);
Jeonghoon 3:2a3664dc6634 370 srv.id_ = i+100;
Jeonghoon 3:2a3664dc6634 371 return v;
Jeonghoon 3:2a3664dc6634 372 }
Jeonghoon 3:2a3664dc6634 373 }
Jeonghoon 3:2a3664dc6634 374 return false;
Jeonghoon 3:2a3664dc6634 375 }
Jeonghoon 3:2a3664dc6634 376
Jeonghoon 3:2a3664dc6634 377 /* Register a new Service Client */
Jeonghoon 3:2a3664dc6634 378 template<typename MReq, typename MRes>
Jeonghoon 3:2a3664dc6634 379 bool serviceClient(ServiceClient<MReq, MRes>& srv){
Jeonghoon 3:2a3664dc6634 380 bool v = advertise(srv.pub);
Jeonghoon 3:2a3664dc6634 381 for(int i = 0; i < MAX_SUBSCRIBERS; i++){
Jeonghoon 3:2a3664dc6634 382 if(subscribers[i] == 0){ // empty slot
Jeonghoon 3:2a3664dc6634 383 subscribers[i] = static_cast<Subscriber_*>(&srv);
Jeonghoon 3:2a3664dc6634 384 srv.id_ = i+100;
Jeonghoon 3:2a3664dc6634 385 return v;
Jeonghoon 3:2a3664dc6634 386 }
Jeonghoon 3:2a3664dc6634 387 }
Jeonghoon 3:2a3664dc6634 388 return false;
Jeonghoon 3:2a3664dc6634 389 }
Jeonghoon 3:2a3664dc6634 390
Jeonghoon 3:2a3664dc6634 391 void negotiateTopics()
Jeonghoon 3:2a3664dc6634 392 {
Jeonghoon 3:2a3664dc6634 393 rosserial_msgs::TopicInfo ti;
Jeonghoon 3:2a3664dc6634 394 int i;
Jeonghoon 3:2a3664dc6634 395 for(i = 0; i < MAX_PUBLISHERS; i++)
Jeonghoon 3:2a3664dc6634 396 {
Jeonghoon 3:2a3664dc6634 397 if(publishers[i] != 0) // non-empty slot
Jeonghoon 3:2a3664dc6634 398 {
Jeonghoon 3:2a3664dc6634 399 ti.topic_id = publishers[i]->id_;
Jeonghoon 3:2a3664dc6634 400 ti.topic_name = (char *) publishers[i]->topic_;
Jeonghoon 3:2a3664dc6634 401 ti.message_type = (char *) publishers[i]->msg_->getType();
Jeonghoon 3:2a3664dc6634 402 ti.md5sum = (char *) publishers[i]->msg_->getMD5();
Jeonghoon 3:2a3664dc6634 403 ti.buffer_size = OUTPUT_SIZE;
Jeonghoon 3:2a3664dc6634 404 publish( publishers[i]->getEndpointType(), &ti );
Jeonghoon 3:2a3664dc6634 405 }
Jeonghoon 3:2a3664dc6634 406 }
Jeonghoon 3:2a3664dc6634 407 for(i = 0; i < MAX_SUBSCRIBERS; i++)
Jeonghoon 3:2a3664dc6634 408 {
Jeonghoon 3:2a3664dc6634 409 if(subscribers[i] != 0) // non-empty slot
Jeonghoon 3:2a3664dc6634 410 {
Jeonghoon 3:2a3664dc6634 411 ti.topic_id = subscribers[i]->id_;
Jeonghoon 3:2a3664dc6634 412 ti.topic_name = (char *) subscribers[i]->topic_;
Jeonghoon 3:2a3664dc6634 413 ti.message_type = (char *) subscribers[i]->getMsgType();
Jeonghoon 3:2a3664dc6634 414 ti.md5sum = (char *) subscribers[i]->getMsgMD5();
Jeonghoon 3:2a3664dc6634 415 ti.buffer_size = INPUT_SIZE;
Jeonghoon 3:2a3664dc6634 416 publish( subscribers[i]->getEndpointType(), &ti );
Jeonghoon 3:2a3664dc6634 417 }
Jeonghoon 3:2a3664dc6634 418 }
Jeonghoon 3:2a3664dc6634 419 configured_ = true;
Jeonghoon 3:2a3664dc6634 420 }
Jeonghoon 3:2a3664dc6634 421
Jeonghoon 3:2a3664dc6634 422 virtual int publish(int id, const Msg * msg)
Jeonghoon 3:2a3664dc6634 423 {
Jeonghoon 3:2a3664dc6634 424 if(id >= 100 && !configured_)
Jeonghoon 3:2a3664dc6634 425 return 0;
Jeonghoon 3:2a3664dc6634 426
Jeonghoon 3:2a3664dc6634 427 /* serialize message */
Jeonghoon 3:2a3664dc6634 428 uint16_t l = msg->serialize(message_out+7);
Jeonghoon 3:2a3664dc6634 429
Jeonghoon 3:2a3664dc6634 430 /* setup the header */
Jeonghoon 3:2a3664dc6634 431 message_out[0] = 0xff;
Jeonghoon 3:2a3664dc6634 432 message_out[1] = PROTOCOL_VER;
Jeonghoon 3:2a3664dc6634 433 message_out[2] = (uint8_t) ((uint16_t)l&255);
Jeonghoon 3:2a3664dc6634 434 message_out[3] = (uint8_t) ((uint16_t)l>>8);
Jeonghoon 3:2a3664dc6634 435 message_out[4] = 255 - ((message_out[2] + message_out[3])%256);
Jeonghoon 3:2a3664dc6634 436 message_out[5] = (uint8_t) ((int16_t)id&255);
Jeonghoon 3:2a3664dc6634 437 message_out[6] = (uint8_t) ((int16_t)id>>8);
Jeonghoon 3:2a3664dc6634 438
Jeonghoon 3:2a3664dc6634 439 /* calculate checksum */
Jeonghoon 3:2a3664dc6634 440 int chk = 0;
Jeonghoon 3:2a3664dc6634 441 for(int i =5; i<l+7; i++)
Jeonghoon 3:2a3664dc6634 442 chk += message_out[i];
Jeonghoon 3:2a3664dc6634 443 l += 7;
Jeonghoon 3:2a3664dc6634 444 message_out[l++] = 255 - (chk%256);
Jeonghoon 3:2a3664dc6634 445
Jeonghoon 3:2a3664dc6634 446 if( l <= OUTPUT_SIZE ){
Jeonghoon 3:2a3664dc6634 447 hardware_.write(message_out, l);
Jeonghoon 3:2a3664dc6634 448 return l;
Jeonghoon 3:2a3664dc6634 449 }else{
Jeonghoon 3:2a3664dc6634 450 logerror("Message from device dropped: message larger than buffer.");
Jeonghoon 3:2a3664dc6634 451 return -1;
Jeonghoon 3:2a3664dc6634 452 }
Jeonghoon 3:2a3664dc6634 453 }
Jeonghoon 3:2a3664dc6634 454
Jeonghoon 3:2a3664dc6634 455 /********************************************************************
Jeonghoon 3:2a3664dc6634 456 * Logging
Jeonghoon 3:2a3664dc6634 457 */
Jeonghoon 3:2a3664dc6634 458
Jeonghoon 3:2a3664dc6634 459 private:
Jeonghoon 3:2a3664dc6634 460 void log(char byte, const char * msg){
Jeonghoon 3:2a3664dc6634 461 rosserial_msgs::Log l;
Jeonghoon 3:2a3664dc6634 462 l.level= byte;
Jeonghoon 3:2a3664dc6634 463 l.msg = (char*)msg;
Jeonghoon 3:2a3664dc6634 464 publish(rosserial_msgs::TopicInfo::ID_LOG, &l);
Jeonghoon 3:2a3664dc6634 465 }
Jeonghoon 3:2a3664dc6634 466
Jeonghoon 3:2a3664dc6634 467 public:
Jeonghoon 3:2a3664dc6634 468 void logdebug(const char* msg){
Jeonghoon 3:2a3664dc6634 469 log(rosserial_msgs::Log::ROSDEBUG, msg);
Jeonghoon 3:2a3664dc6634 470 }
Jeonghoon 3:2a3664dc6634 471 void loginfo(const char * msg){
Jeonghoon 3:2a3664dc6634 472 log(rosserial_msgs::Log::INFO, msg);
Jeonghoon 3:2a3664dc6634 473 }
Jeonghoon 3:2a3664dc6634 474 void logwarn(const char *msg){
Jeonghoon 3:2a3664dc6634 475 log(rosserial_msgs::Log::WARN, msg);
Jeonghoon 3:2a3664dc6634 476 }
Jeonghoon 3:2a3664dc6634 477 void logerror(const char*msg){
Jeonghoon 3:2a3664dc6634 478 log(rosserial_msgs::Log::ERROR, msg);
Jeonghoon 3:2a3664dc6634 479 }
Jeonghoon 3:2a3664dc6634 480 void logfatal(const char*msg){
Jeonghoon 3:2a3664dc6634 481 log(rosserial_msgs::Log::FATAL, msg);
Jeonghoon 3:2a3664dc6634 482 }
Jeonghoon 3:2a3664dc6634 483
Jeonghoon 3:2a3664dc6634 484 /********************************************************************
Jeonghoon 3:2a3664dc6634 485 * Parameters
Jeonghoon 3:2a3664dc6634 486 */
Jeonghoon 3:2a3664dc6634 487
Jeonghoon 3:2a3664dc6634 488 private:
Jeonghoon 3:2a3664dc6634 489 bool param_recieved;
Jeonghoon 3:2a3664dc6634 490 rosserial_msgs::RequestParamResponse req_param_resp;
Jeonghoon 3:2a3664dc6634 491
Jeonghoon 3:2a3664dc6634 492 bool requestParam(const char * name, int time_out = 1000){
Jeonghoon 3:2a3664dc6634 493 param_recieved = false;
Jeonghoon 3:2a3664dc6634 494 rosserial_msgs::RequestParamRequest req;
Jeonghoon 3:2a3664dc6634 495 req.name = (char*)name;
Jeonghoon 3:2a3664dc6634 496 publish(TopicInfo::ID_PARAMETER_REQUEST, &req);
Jeonghoon 3:2a3664dc6634 497 uint16_t end_time = hardware_.time() + time_out;
Jeonghoon 3:2a3664dc6634 498 while(!param_recieved ){
Jeonghoon 3:2a3664dc6634 499 spinOnce();
Jeonghoon 3:2a3664dc6634 500 if (hardware_.time() > end_time) return false;
Jeonghoon 3:2a3664dc6634 501 }
Jeonghoon 3:2a3664dc6634 502 return true;
Jeonghoon 3:2a3664dc6634 503 }
Jeonghoon 3:2a3664dc6634 504
Jeonghoon 3:2a3664dc6634 505 public:
Jeonghoon 3:2a3664dc6634 506 bool getParam(const char* name, int* param, int length =1){
Jeonghoon 3:2a3664dc6634 507 if (requestParam(name) ){
Jeonghoon 3:2a3664dc6634 508 if (length == req_param_resp.ints_length){
Jeonghoon 3:2a3664dc6634 509 //copy it over
Jeonghoon 3:2a3664dc6634 510 for(int i=0; i<length; i++)
Jeonghoon 3:2a3664dc6634 511 param[i] = req_param_resp.ints[i];
Jeonghoon 3:2a3664dc6634 512 return true;
Jeonghoon 3:2a3664dc6634 513 }
Jeonghoon 3:2a3664dc6634 514 }
Jeonghoon 3:2a3664dc6634 515 return false;
Jeonghoon 3:2a3664dc6634 516 }
Jeonghoon 3:2a3664dc6634 517 bool getParam(const char* name, float* param, int length=1){
Jeonghoon 3:2a3664dc6634 518 if (requestParam(name) ){
Jeonghoon 3:2a3664dc6634 519 if (length == req_param_resp.floats_length){
Jeonghoon 3:2a3664dc6634 520 //copy it over
Jeonghoon 3:2a3664dc6634 521 for(int i=0; i<length; i++)
Jeonghoon 3:2a3664dc6634 522 param[i] = req_param_resp.floats[i];
Jeonghoon 3:2a3664dc6634 523 return true;
Jeonghoon 3:2a3664dc6634 524 }
Jeonghoon 3:2a3664dc6634 525 }
Jeonghoon 3:2a3664dc6634 526 return false;
Jeonghoon 3:2a3664dc6634 527 }
Jeonghoon 3:2a3664dc6634 528 bool getParam(const char* name, char** param, int length=1){
Jeonghoon 3:2a3664dc6634 529 if (requestParam(name) ){
Jeonghoon 3:2a3664dc6634 530 if (length == req_param_resp.strings_length){
Jeonghoon 3:2a3664dc6634 531 //copy it over
Jeonghoon 3:2a3664dc6634 532 for(int i=0; i<length; i++)
Jeonghoon 3:2a3664dc6634 533 strcpy(param[i],req_param_resp.strings[i]);
Jeonghoon 3:2a3664dc6634 534 return true;
Jeonghoon 3:2a3664dc6634 535 }
Jeonghoon 3:2a3664dc6634 536 }
Jeonghoon 3:2a3664dc6634 537 return false;
Jeonghoon 3:2a3664dc6634 538 }
Jeonghoon 3:2a3664dc6634 539 };
Jeonghoon 3:2a3664dc6634 540
Jeonghoon 3:2a3664dc6634 541 }
Jeonghoon 3:2a3664dc6634 542
Jeonghoon 3:2a3664dc6634 543 #endif