UR10-Sentry/Robot/UR/URModbusServer.py

166 lines
5.8 KiB
Python
Raw Normal View History

2024-07-19 18:35:10 +00:00
from Communication.ModbusTCP import ModbusTCP
import time, math, struct
# The robot controller acts as a Modbus TCP server (port 502),
# clients can establish connections to it and send standard MODBUS requests to it.
# - Note that some Modbus device manufacturers use the terms Master (client) and Slave (server).
# Typically the external IO device is going to be a server and the robot is going to behave as the client
# (requesting and consuming messages from the server). Master=client and Slave=server.
# However, note that the UR controller can be both a server and a client
# - Note that all values are unsigned, if you want to convert to signed integers,
# program "if (val > 32768): val = val - 65535".
#
# - The MODBUS Server has 0-based addressing.
# Be aware that some other devices are 1-based (e.g. Anybus X-gateways), then just add one to the address
# on that device. (e.g. address 3 on the robot will be address 4 on the Anybus X-gateway)
class URModbusServer:
"""Give read and write access to data in the robot controller for other devices
An interface for communicating with the modbus TCP server (port 502) on the UR.
Defines functions for retrieving information from the controller.
Information will be re-requested if an error occurs.
All information will be formatted to human readable information.
"""
def __init__(self, host):
"""
:param host: IP address to connect with
"""
self.modbusTCP = ModbusTCP(host, 502)
def get_tcp_position(self):
"""
Connects with the Modbus server to requests Cartesian data of the TCP
:return: Readable cartesian data of TCP, vector in mm, axis in radials
"""
packet = self.modbusTCP.read_holding_registers(400, quantity=6)
if packet is None:
time.sleep(0.5)
2024-07-23 16:33:20 +00:00
print("[TCP] Modbus Error: retrying")
2024-07-19 18:35:10 +00:00
return self.get_tcp_position()
else:
x = self._format(packet[9:11]) / 10
y = self._format(packet[11:13]) / 10
z = self._format(packet[13:15]) / 10
rx = self._format(packet[15:17]) / 1000
ry = self._format(packet[17:19]) / 1000
rz = self._format(packet[19:21]) / 1000
return x, y, z, rx, ry, rz
def get_joint_angles(self, tries=0) -> tuple:
"""
Connects with the Modbus server to requests the angles of each joint, in radians
:return: Readable angle values of each joint in radials
"""
2024-07-25 22:17:36 +00:00
if tries>50:
raise ModbusError("[Angles] Modbus Error: Failed")
2024-07-25 22:17:36 +00:00
#return 0, 0, 0, 0, 0, 0
2024-07-19 18:35:10 +00:00
packet = self.modbusTCP.read_holding_registers(270, quantity=6)
2024-07-23 16:33:20 +00:00
time.sleep(0.001)
2024-07-19 18:35:10 +00:00
packet_signs = self.modbusTCP.read_holding_registers(320, quantity=6)
if (packet is None) or (packet_signs is None):
2024-07-19 21:37:24 +00:00
time.sleep(0.01)
2024-07-23 16:33:20 +00:00
print(f"[Angles] Modbus Error #{tries}: retrying")
2024-07-19 18:35:10 +00:00
return self.get_joint_angles(tries+1)
else:
base = self._format_sign(packet[9:11], packet_signs[9:11])
shoulder = self._format_sign(packet[11:13], packet_signs[11:13])
elbow = self._format_sign(packet[13:15], packet_signs[13:15])
wrist_1 = self._format_sign(packet[15:17], packet_signs[15:17])
wrist_2 = self._format_sign(packet[17:19], packet_signs[17:19])
wrist_3 = self._format_sign(packet[19:21], packet_signs[19:21])
return base, shoulder, elbow, wrist_1, wrist_2, wrist_3
def get_joint_angles_degrees(self, tries=0):
angles = self.get_joint_angles(tries)
return tuple([round(math.degrees(angle), 3) for angle in angles])
def get_joint_speeds(self, tries=0):
"""
2024-07-19 19:48:42 +00:00
Connects with the Modbus server to requests the speed of each joint, in radians per second
:return: Readable angle speeds of each joint in radians per second
2024-07-19 18:35:10 +00:00
"""
2024-07-25 22:17:36 +00:00
if tries>50:
#print("[Speeds] Modbus Error: Failed")
raise ModbusError("[Speeds] Modbus Error: Failed")
2024-07-19 18:35:10 +00:00
packet = self.modbusTCP.read_holding_registers(280, quantity=6)
if (packet is None):
2024-07-19 21:37:24 +00:00
time.sleep(0.01)
2024-07-23 16:33:20 +00:00
print(f"[Speeds] Modbus Error #{tries}: retrying")
2024-07-19 18:35:10 +00:00
return self.get_joint_speeds(tries+1)
else:
base = self._format(packet[9:11]) / 1000
shoulder = self._format(packet[11:13]) / 1000
elbow = self._format(packet[13:15]) / 1000
wrist_1 = self._format(packet[15:17]) / 1000
wrist_2 = self._format(packet[17:19]) / 1000
wrist_3 = self._format(packet[19:21]) / 1000
return base, shoulder, elbow, wrist_1, wrist_2, wrist_3
@staticmethod
def _format(d):
"""Formats signed integers to unsigned float
:param d: signed integer to format
:return: unsigned float
"""
d = d.hex()
d_i = int(d, 16)
d_f = 0
if d_i < 32768:
d_f = float(d_i)
if d_i > 32767:
d_i = 65535 - d_i
d_f = float(d_i) * -1
return d_f
@staticmethod
def _format_2(d):
"""Formats signed integers to signed float
:param d: signed integer to format
:return: signed float
"""
d = d.hex()
d_i = int(d, 16)
d_f = float(d_i)
return d_f
@staticmethod
def _format_sign(d, sign):
"""
:param d:
:param sign:
:return:
"""
sign = sign.hex()
sign = int(sign, 16)
d = d.hex()
d_i = int(d, 16)
d_f = float(d_i) / 1000
if sign == 0:
return d_f
else:
d_f = round(d_f - math.radians(360), 3)
return d_f
class ModbusError(Exception):
pass