-
Notifications
You must be signed in to change notification settings - Fork 11
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add motor enable 1.0x #140
Changes from 18 commits
08f4e11
f811619
4197beb
317d486
d955e1f
9806e15
9c2f3a1
51ba6d4
878c51e
b095daf
5c85bbb
becb57a
afba6bc
cbc14d9
d64f308
5dbc69b
b596190
1f0a26e
88736e7
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7,6 +7,7 @@ | |
|
||
from geometry_msgs.msg import Twist | ||
from std_msgs.msg import Bool | ||
from std_srvs.srv import SetBool, SetBoolRequest, SetBoolResponse | ||
from std_srvs.srv import Trigger, TriggerRequest, TriggerResponse | ||
|
||
from panther_msgs.msg import DriverState, IOState | ||
|
@@ -17,7 +18,8 @@ def __init__(self, name: str) -> None: | |
self._node_name = name | ||
rospy.init_node(self._node_name, anonymous=False) | ||
|
||
self._lock = Lock() | ||
self._e_stop_lock = Lock() | ||
self._motors_lock = Lock() | ||
|
||
line_names = { | ||
'MOTOR_ON': False, # Used to enable motors | ||
|
@@ -41,8 +43,10 @@ def __init__(self, name: str) -> None: | |
) | ||
|
||
self._e_stop_state = not self._lines['STAGE2_INPUT'].get_value() | ||
self._last_motor_state = self._lines['STAGE2_INPUT'].get_value() | ||
self._cmd_vel_msg_time = rospy.get_time() | ||
self._can_net_err = True | ||
self._motor_enabled = True | ||
|
||
# ------------------------------- | ||
# Publishers | ||
|
@@ -55,7 +59,7 @@ def __init__(self, name: str) -> None: | |
self._e_stop_state_pub.publish(self._e_stop_state) | ||
|
||
self._io_state = IOState() | ||
self._io_state.motor_on = self._lines['STAGE2_INPUT'].get_value() | ||
self._io_state.motor_on = self._last_motor_state | ||
self._io_state.aux_power = False | ||
self._io_state.charger_connected = False | ||
self._io_state.fan = False | ||
|
@@ -83,6 +87,17 @@ def __init__(self, name: str) -> None: | |
self._e_stop_trigger_server = rospy.Service( | ||
'hardware/e_stop_trigger', Trigger, self._e_stop_trigger_cb | ||
) | ||
self._motor_enable_server = rospy.Service( | ||
'hardware/motor_enable', SetBool, self._motor_enable_cb | ||
) | ||
|
||
# ------------------------------- | ||
# Service clients | ||
# ------------------------------- | ||
|
||
self._reset_roboteq_script_client = rospy.ServiceProxy( | ||
'driver/reset_roboteq_script', Trigger | ||
) | ||
|
||
# ------------------------------- | ||
# Timers | ||
|
@@ -103,20 +118,20 @@ def __del__(self): | |
self._chip.close() | ||
|
||
def _cmd_vel_cb(self, *args) -> None: | ||
with self._lock: | ||
with self._e_stop_lock: | ||
self._cmd_vel_msg_time = rospy.get_time() | ||
|
||
def _motor_controllers_state_cb(self, msg: DriverState) -> None: | ||
with self._lock: | ||
with self._e_stop_lock: | ||
self._can_net_err = any( | ||
{msg.rear.fault_flag.can_net_err, msg.front.fault_flag.can_net_err} | ||
) | ||
|
||
def _e_stop_reset_cb(self, req: TriggerRequest) -> TriggerResponse: | ||
with self._lock: | ||
with self._e_stop_lock: | ||
if not self._e_stop_state: | ||
return TriggerResponse(True, 'E-STOP is not active, reset is not needed') | ||
|
||
if rospy.get_time() - self._cmd_vel_msg_time <= 2.0: | ||
return TriggerResponse( | ||
False, | ||
|
@@ -134,19 +149,50 @@ def _e_stop_reset_cb(self, req: TriggerRequest) -> TriggerResponse: | |
return TriggerResponse(True, 'E-STOP reset successful') | ||
|
||
def _e_stop_trigger_cb(self, req: TriggerRequest) -> TriggerResponse: | ||
with self._lock: | ||
with self._e_stop_lock: | ||
if self._e_stop_state: | ||
return TriggerResponse(True, 'E-SROP already triggered') | ||
|
||
self._e_stop_state = True | ||
self._e_stop_state_pub.publish(self._e_stop_state) | ||
return TriggerResponse(True, 'E-SROP triggered successful') | ||
|
||
def _motor_enable_cb(self, req: SetBoolRequest) -> SetBoolResponse: | ||
with self._motors_lock: | ||
if not self._lines['STAGE2_INPUT'].get_value(): | ||
self._motor_enabled = False | ||
return SetBoolResponse( | ||
not req.data, | ||
f'Motors are {"already " if not req.data else ""}disabled. ' | ||
+ '(Main switch set to Stage 1)', | ||
) | ||
|
||
if self._motor_enabled == req.data: | ||
return SetBoolResponse( | ||
True, f'Motors are already {"enabled" if self._motor_enabled else "disabled"}' | ||
) | ||
|
||
self._motor_enabled = req.data | ||
self._lines['MOTOR_ON'].set_value(req.data) | ||
self._publish_motor_state(req.data) | ||
return SetBoolResponse(True, f'Motors {"enabled" if req.data else "disabled"}') | ||
|
||
def _set_motor_state_timer_cb(self, *args) -> None: | ||
motor_state = self._lines['STAGE2_INPUT'].get_value() | ||
self._lines['MOTOR_ON'].set_value(motor_state) | ||
if self._io_state.motor_on != motor_state: | ||
self._io_state.motor_on = motor_state | ||
with self._motors_lock: | ||
motor_state = self._lines['STAGE2_INPUT'].get_value() | ||
# if switch changes from off to on overwrite service value | ||
if not self._last_motor_state and motor_state: | ||
self._motor_enabled = True | ||
|
||
self._last_motor_state = motor_state | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Taught with C++ practice, I think that it would not be a stupid approach to use static variables in such situations. The Example def my_function(self):
if not hasattr(self.my_function, "counter"):
self.my_function.counter = 0
self.my_function.counter += 1
print(f"Counter: {self.my_function.counter}") There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't see the point. This makes the code less readable. We would have to make an strong assumption the class is a singleton. Even though it would work I do not think assigning variables to a function rather than object is a good idea |
||
|
||
state_to_set = motor_state and self._motor_enabled | ||
self._lines['MOTOR_ON'].set_value(state_to_set) | ||
self._publish_motor_state(state_to_set) | ||
|
||
def _publish_motor_state(self, val: bool) -> None: | ||
if self._io_state.motor_on != val: | ||
self._io_state.motor_on = val | ||
Kotochleb marked this conversation as resolved.
Show resolved
Hide resolved
|
||
self._io_state_pub.publish(self._io_state) | ||
|
||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
self._lines['STAGE2_INPUT'].get_value()
bardzo często się pojawia w kodzie proponuje stworzyć lambdeis_stage2()
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It appears 4 times. If we had to add lambda, we would have to do this for both pins and also port it to
power_borad_node.py
to keep the same coding style