00001 
00002 
00003 """
00004   ArbotiX Node: serial connection to an ArbotiX board w/ PyPose/NUKE/ROS
00005   Copyright (c) 2008-2011 Michael E. Ferguson.  All right reserved.
00006 
00007   Redistribution and use in source and binary forms, with or without
00008   modification, are permitted provided that the following conditions are met:
00009       * Redistributions of source code must retain the above copyright
00010         notice, this list of conditions and the following disclaimer.
00011       * Redistributions in binary form must reproduce the above copyright
00012         notice, this list of conditions and the following disclaimer in the
00013         documentation and/or other materials provided with the distribution.
00014       * Neither the name of Vanadium Labs LLC nor the names of its 
00015         contributors may be used to endorse or promote products derived 
00016         from this software without specific prior written permission.
00017   
00018   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
00019   ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
00020   WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
00021   DISCLAIMED. IN NO EVENT SHALL VANADIUM LABS BE LIABLE FOR ANY DIRECT, INDIRECT,
00022   INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
00023   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
00024   OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
00025   LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
00026   OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
00027   ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
00028 """
00029 
00030 import roslib; roslib.load_manifest('arbotix_python')
00031 import rospy
00032 import sys
00033 
00034 from arbotix_msgs.msg import *
00035 from arbotix_msgs.srv import *
00036 
00037 from arbotix_python.arbotix import ArbotiX
00038 from arbotix_python.diff_controller import DiffController
00039 from arbotix_python.follow_controller import FollowController
00040 from arbotix_python.servo_controller import *
00041 from arbotix_python.linear_controller import *
00042 from arbotix_python.publishers import *
00043 from arbotix_python.io import *
00044 
00045 
00046 controller_types = { "follow_controller" : FollowController,
00047                      "diff_controller"   : DiffController,
00048 
00049                      "linear_controller" : LinearControllerAbsolute,
00050                      "linear_controller_i" : LinearControllerIncremental }
00051 
00052 
00053 
00054 class ArbotixROS(ArbotiX):
00055     
00056     def __init__(self):
00057         pause = False
00058 
00059         
00060         port = rospy.get_param("~port", "/dev/ttyUSB0")
00061         baud = int(rospy.get_param("~baud", "115200"))
00062 
00063         self.rate = rospy.get_param("~rate", 100.0)
00064         self.fake = rospy.get_param("~sim", False)
00065 
00066         self.use_sync_read = rospy.get_param("~sync_read",True)      
00067         self.use_sync_write = rospy.get_param("~sync_write",True)    
00068 
00069         
00070         self.diagnostics = DiagnosticsPublisher()
00071         self.joint_state_publisher = JointStatePublisher()
00072 
00073         
00074         if not self.fake:
00075             ArbotiX.__init__(self, port, baud)        
00076             rospy.sleep(1.0)
00077             rospy.loginfo("Started ArbotiX connection on port " + port + ".")
00078         else:
00079             rospy.loginfo("ArbotiX being simulated.")
00080 
00081         
00082         self.joints = dict()
00083         for name in rospy.get_param("~joints", dict()).keys():
00084             joint_type = rospy.get_param("~joints/"+name+"/type", "dynamixel")
00085             if joint_type == "dynamixel":
00086                 self.joints[name] = DynamixelServo(self, name)
00087             elif joint_type == "hobby_servo":
00088                 self.joints[name] = HobbyServo(self, name)
00089             elif joint_type == "calibrated_linear":
00090                 self.joints[name] = LinearJoint(self, name)
00091 
00092     
00093         if len(rospy.get_param("~dynamixels", dict()).keys()) > 0:
00094             rospy.logwarn("Warning: use of dynamixels as a dictionary is deprecated and will be removed in 0.8.0")
00095             for name in rospy.get_param("~dynamixels", dict()).keys():
00096                 self.joints[name] = DynamixelServo(self, name, "~dynamixels")
00097         if len(rospy.get_param("~servos", dict()).keys()) > 0:
00098             rospy.logwarn("Warning: use of servos as a dictionary is deprecated and will be removed in 0.8.0")
00099             for name in rospy.get_param("~servos", dict()).keys():
00100                 self.joints[name] = HobbyServo(self, name, "~servos")
00101     
00102 
00103         
00104         self.controllers = [ServoController(self, "servos"), ]
00105         controllers = rospy.get_param("~controllers", dict())
00106         for name, params in controllers.items():
00107             try:
00108                 controller = controller_types[params["type"]](self, name)
00109                 self.controllers.append( controller )
00110                 pause = pause or controller.pause
00111             except Exception as e:
00112                 if type(e) == KeyError:
00113                     rospy.logerr("Unrecognized controller: " + params["type"])
00114                 else:  
00115                     rospy.logerr(str(type(e)) + str(e))
00116 
00117         
00118         if not self.fake:
00119             if rospy.has_param("~digital_servos") or rospy.has_param("~digital_sensors") or rospy.has_param("~analog_sensors"):
00120                 pause = True
00121             if pause:
00122                 while self.getDigital(1) == -1 and not rospy.is_shutdown():
00123                     rospy.loginfo("Waiting for response...")
00124                     rospy.sleep(0.25)
00125             rospy.loginfo("ArbotiX connected.")
00126 
00127         for controller in self.controllers:
00128             controller.startup()
00129 
00130         
00131         rospy.Service('~SetupAnalogIn',SetupChannel, self.analogInCb)
00132         rospy.Service('~SetupDigitalIn',SetupChannel, self.digitalInCb)
00133         rospy.Service('~SetupDigitalOut',SetupChannel, self.digitalOutCb)
00134         
00135         self.io = dict()
00136         if not self.fake:
00137             for v,t in {"digital_servos":DigitalServo,"digital_sensors":DigitalSensor,"analog_sensors":AnalogSensor}.items():
00138                 temp = rospy.get_param("~"+v,dict())
00139                 for name in temp.keys():
00140                     pin = rospy.get_param('~'+v+'/'+name+'/pin',1)
00141                     value = rospy.get_param('~'+v+'/'+name+'/value',0)
00142                     rate = rospy.get_param('~'+v+'/'+name+'/rate',10)
00143                     self.io[name] = t(name, pin, value, rate, self)
00144         
00145         r = rospy.Rate(self.rate)
00146 
00147         
00148         while not rospy.is_shutdown():
00149     
00150             
00151             for controller in self.controllers:
00152                 controller.update()
00153 
00154             
00155             for io in self.io.values():
00156                 io.update()
00157 
00158             
00159             self.joint_state_publisher.update(self.joints.values(), self.controllers)
00160             self.diagnostics.update(self.joints.values(), self.controllers)
00161 
00162             r.sleep()
00163 
00164         
00165         for controller in self.controllers:
00166             controller.shutdown()
00167 
00168     def analogInCb(self, req):
00169         
00170         if not self.fake:
00171             self.io[req.topic_name] = AnalogSensor(req.topic_name, req.pin, req.value, req.rate, self) 
00172         return SetupChannelResponse()
00173 
00174     def digitalInCb(self, req):
00175         if not self.fake:
00176             self.io[req.topic_name] = DigitalSensor(req.topic_name, req.pin, req.value, req.rate, self) 
00177         return SetupChannelResponse()
00178 
00179     def digitalOutCb(self, req):
00180         if not self.fake:
00181             self.io[req.topic_name] = DigitalServo(req.topic_name, req.pin, req.value, req.rate, self) 
00182         return SetupChannelResponse()
00183 
00184 
00185 if __name__ == "__main__":
00186     rospy.init_node('arbotix')
00187     a = ArbotixROS()
00188