-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathodriveMotorController.py
191 lines (158 loc) · 9.3 KB
/
odriveMotorController.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
# Imports - logger
import logging
import logger
# Imports - exit handling
import sys
# Imports - control
import odrive
# Start Logger
logger = logging.getLogger("DynamometerControl")
logger.setLevel(logging.DEBUG)
class odriveMotorController:
# Our controller is a bit old, this version is more relevant: # https://docs.odriverobotics.com/v/0.5.4/getting-started.html
# Latest: https://docs.odriverobotics.com/v/0.5.5/specifications.html
# https://docs.odriverobotics.com/v/0.5.5/getting-started.html
# Troubleshooting: https://docs.odriverobotics.com/v/latest/troubleshooting.html
def __init__(self):
"""
Object to control an Odrive motor
"""
# Start Logger
self.logger = logging.getLogger("DynamometerControl")
self.logger.setLevel(logging.DEBUG)
# Connect to Odrive
self.odrv0 = odrive.find_any
# Get version number
try:
self.logger.info("self.odrv0.hw_version_major:", self.odrv0.hw_version_major) # 3
self.logger.info("self.odrv0.hw_version_minor:", self.odrv0.hw_version_minor) # 6
self.logger.info("self.odrv0.hw_version_revision:", self.odrv0.hw_version_revision) # Doesn’t work
self.logger.info("self.odrv0.fw_version_major:", self.odrv0.fw_version_major) # 3
self.logger.info("self.odrv0.fw_version_minor:", self.odrv0.fw_version_minor) # 5
self.logger.info("self.odrv0.fw_version_revision:", self.odrv0.fw_version_revision) # 5
except AttributeError as e:
self.logger.fatal("Could not retrieve odrive information: " + str(e))
raise e
def configure(self):
"""
Reconfigures odrv0 and odrv0.axis0
"""
raise NotImplementedError
# Reset
self.odrv0.erase_configuration() # Error
# https://docs.odriverobotics.com/v/0.5.5/getting-started.html#setting-the-limits
self.odrv0.axis0.motor.config.current_lim = 10 # Up to 40A w/o heat sink (Watch out for motor overheat!)
self.odrv0.axis0.controller.config.vel_limit = 2
self.odrv0.axis0.motor.config.calibration_current = 2 # 10 by default, set 2 for now
# https://docs.odriverobotics.com/v/0.5.5/getting-started.html#setting-other-hardware-parameters
self.odrv0.config.enable_brake_resistor = True
self.odrv0.config.brake_resistance = -1 # 50 W Default, but needs to be in Ohms # 2 defualt, measure somewhat close
# self.odrv0.config.dc_max_negative_current = .01 # Default 10 mA (in Amps)
self.odrv0.axis0.motor.config.pole_pairs = 7
self.odrv0.axis0.motor.config.torque_constant = 8.27 / 270
self.odrv0.axis0.motor.config.motor_type = MotorType.HIGH_CURRENT
# Thermistor configured by default, do not change
#<axis>.motor.fet_thermistor.config.temp_limit_lower # Default 100
#<axis>.motor.fet_thermistor.config.temp_limit_upper # Default 120
# https://docs.odriverobotics.com/v/0.5.5/commands.html#setting-up-sensorless
# Velocity control:
#self.odrv0.axis0.config.sensorless_ramp.vel = # rad/sec # Default 400 rad → 270 rpm → 28.27
#self.odrv0.axis0.config.sensorless_ramp.accel = # rad/sec^2 # Default 200 rad/s^2 – > 28.27 /2
# These values are good for the D5065 motor (which we have (I think))
self.odrv0.axis0.controller.config.vel_gain = 0.01
self.odrv0.axis0.controller.config.vel_integrator_gain = 0.05
self.odrv0.axis0.controller.config.control_mode = CONTROL_MODE_VELOCITY_CONTROL
#self.odrv0.axis0.controller.config.vel_limit = <a value greater than axis.config.sensorless_ramp.vel / (2pi * 7)>
self.odrv0.axis0.motor.config.current_lim = 2 * self.odrv0.axis0.config.sensorless_ramp.current
self.odrv0.axis0.sensorless_estimator.config.pm_flux_linkage = 5.51328895422 / (7 * 270)
self.odrv0.axis0.config.enable_sensorless_mode = True
# https://docs.odriverobotics.com/v/0.5.5/getting-started.html#watchdog-timer
# TODO: Calibrate
# Save config
self.odrv0.save_configuration()
# TODO: Reconnect
return False
def calibrate(self, bypassVerification):
"""
Prompts for verification if bypassVerification is False and runs a full calibration. This may not be required for sensorless mode
Skips verification if bypassVerification is True
@param bypassVerification (Boolean)
"""
if not bypassVerification:
print("Ready to run a full calibration. Continue? (Y/n): ")
response = input()
logger.debug('User response to "Ready to run a full calibration. Continue? (Y/n): " ' + response)
if not response == 'Y':
logger.fatal("Motor calibration canceled by user!")
raise SystemExit("Motor calibration canceled by user!")
else:
logger.debug("Continuing...")
self.calibrate(True)
else:
logger.info("Starting calibration.")
self.odrv0.axis0.requested_state = odrive.AXIS_STATE_MOTOR_CALIBRATION
print("Press any key to save config once the motor stops or send a KeyboardInterrupt to cancel ([CTRL] + [C])")
response = input()
logger.debug('User response to "Press any key to save config once the motor stops or send a KeyboardInterrupt to cancel ([CTRL] + [C])" ' + response)
self.odrv0.save_configuration()
# TODO: Reconnect
raise NotImplementedError
# Verify config
def verifyConfig(self):
"""
Verifies the motor configuration of odrv0.asis0
"""
try:
self.logger.info("self.odrv0.axis0.motor.config.current_lim:", self.odrv0.axis0.motor.config.current_lim)
self.logger.info("self.odrv0.axis0.controller.config.vel_limit:", self.odrv0.axis0.controller.config.vel_limit)
self.logger.info("self.odrv0.axis0.motor.config.calibration_current:", self.odrv0.axis0.motor.config.calibration_current)
self.logger.info("self.odrv0.config.enable_brake_resistor:", self.odrv0.config.enable_brake_resistor)
self.logger.info("self.odrv0.config.brake_resistance:", self.odrv0.config.brake_resistance)
self.logger.info("self.odrv0.config.dc_max_negative_current:", self.odrv0.config.dc_max_negative_current)
self.logger.info("self.odrv0.axis0.motor.config.pole_pairs:", self.odrv0.axis0.motor.config.pole_pairs)
self.logger.info("self.odrv0.axis0.motor.config.torque_constant:", self.odrv0.axis0.motor.config.torque_constant)
self.logger.info("self.odrv0.axis0.motor.config.motor_type:", self.odrv0.axis0.motor.config.motor_type)
self.logger.info("self.odrv0.axis0.config.sensorless_ramp.ve", self.odrv0.axis0.config.sensorless_ramp.vel)
self.logger.info("self.odrv0.axis0.config.sensorless_ramp.accel", self.odrv0.axis0.config.sensorless_ramp.accel)
self.logger.info("self.odrv0.axis0.controller.config.vel_gain:", self.odrv0.axis0.controller.config.vel_gain)
self.logger.info("self.odrv0.axis0.controller.config.vel_integrator_gain:", self.odrv0.axis0.controller.config.vel_integrator_gain)
self.logger.info("self.odrv0.axis0.controller.config.control_mode:", self.odrv0.axis0.controller.config.control_mode)
self.logger.info("self.odrv0.axis0.motor.config.current_lim:", self.odrv0.axis0.motor.config.current_lim)
self.logger.info("self.odrv0.axis0.sensorless_estimator.config.pm_flux_linkage:", self.odrv0.axis0.sensorless_estimator.config.pm_flux_linkage)
self.logger.info("self.odrv0.axis0.config.enable_sensorless_mode:", self.odrv0.axis0.config.enable_sensorless_mode)
self.logger.info("self.odrv0.axis0.motor.is_calibrated", self.odrv0.axis0.motor.is_calibrated)
except AttributeError as e:
self.logger.error("Could not verify odrive information: " + str(e))
raise e
return False
# Control
def setVelocity(self, velocity):
"""
Updates the motor's target velocity
@param velocity in rad/s
"""
self.odrv0.axis0.config.sensorlesss_ramp = velocity
def startSensorless(self, bypassVerification=False):
"""
Prompts for verification if bypassVerification is False and starts the motor in sensorless mode assuming the target velocity has already been set
Skips verification if bypassVerification is True
@param bypassVerification (Boolean)
"""
if not bypassVerification:
print("Ready to enable the motor in sensorless mode. Continue? (Y/n): ")
response = input()
logger.debug('User response to "Ready to enable the motor in sensorless mode. Continue? (Y/n): " ' + response)
if not response == 'Y':
logger.fatal("Motor enable canceled by user!")
raise SystemExit("Motor enable canceled by user!")
else:
logger.debug("Continuing...")
self.startSensorless(True)
else:
logger.info("Starting the motor.")
self.odrv0.axis0.requested_state = odrive.AXIS_STATE_CLOSED_LOOP_CONTROL
def stop(self):
"""
Sets the motor's target velocity to 0 rad/s
"""
self.setVelocity(0)