Skip to content

Commit fd05617

Browse files
committed
camtrack: onboard_controller: add globals for update rates
Signed-off-by: Rhys Mainwaring <[email protected]>
1 parent d9587c1 commit fd05617

File tree

1 file changed

+73
-42
lines changed

1 file changed

+73
-42
lines changed

MAVProxy/modules/mavproxy_camtrack/onboard_controller.py

+73-42
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,12 @@
7373
gi.require_version("Gst", "1.0")
7474
from gi.repository import Gst
7575

76+
# TODO: move update rate constants to CLI options
77+
MAIN_LOOP_RATE = 30.0
78+
GIMBAL_CONTROL_LOOP_RATE = 30.0
79+
CAMERA_SEND_IMAGE_STATUS_RATE = 5.0
80+
MAVLINK_RECV_RATE = 1000.0
81+
7682

7783
class CameraCapFlags(Enum):
7884
"""
@@ -259,7 +265,7 @@ def _mavlink_recv_task(self):
259265
"""
260266
Task to receive a mavlink message and forwward to controllers.
261267
"""
262-
update_rate = 1000.0 # Hz
268+
update_rate = MAVLINK_RECV_RATE # Hz
263269
update_period = 1.0 / update_rate
264270

265271
while True:
@@ -281,14 +287,6 @@ def __process_message():
281287
sleep_time = max(0.0, update_period - elapsed_time)
282288
time.sleep(sleep_time)
283289

284-
def _create_tracker(self, name):
285-
if name == "CSTR":
286-
return TrackerCSTR()
287-
elif name == "KCF":
288-
return TrackerKCF()
289-
else:
290-
raise Exception(f"Invalid tracker name: {name}")
291-
292290
def run(self):
293291
"""
294292
Run the onboard controller.
@@ -334,12 +332,12 @@ def run(self):
334332

335333
# Create tracker
336334
print(f"Using tracker: {self._tracker_name}")
337-
tracker = self._create_tracker(self._tracker_name)
335+
tracker = TrackerFactory.create_tracker(self._tracker_name)
338336
tracking_rect = None
339337
tracking_rect_changed = True
340338

341339
# TODO: how to ensure consistency of frame updates with GCS?
342-
update_rate = 50.0 # Hz
340+
update_rate = MAIN_LOOP_RATE # Hz
343341
update_period = 1.0 / update_rate
344342

345343
# TODO: consolidate common code - also used in GUI
@@ -890,7 +888,7 @@ def _mavlink_task(self):
890888
sysid = self._sysid
891889
cmpid = self._cmpid
892890

893-
update_rate = 1000.0 # Hz
891+
update_rate = MAVLINK_RECV_RATE # Hz
894892
update_period = 1.0 / update_rate
895893

896894
while True:
@@ -927,7 +925,7 @@ def _send_status_task(self):
927925
# TODO: stop sending image status when tracking stopped
928926
# TODO: set streaming rate using MAV_CMD_SET_MESSAGE_INTERVAL
929927

930-
update_rate = 20.0 # Hz
928+
update_rate = CAMERA_SEND_IMAGE_STATUS_RATE # Hz
931929
update_period = 1.0 / update_rate
932930

933931
while True:
@@ -977,9 +975,9 @@ def __init__(self, connection, enable_graphs=False):
977975

978976
# Pitch controller for centring gimbal
979977
self._pit_controller = AC_PID_Basic(
980-
initial_p=2.0,
981-
initial_i=0.0,
982-
initial_d=0.0,
978+
initial_p=1.0,
979+
initial_i=0.01,
980+
initial_d=0.01,
983981
initial_ff=0.0,
984982
initial_imax=1.0,
985983
initial_filt_E_hz=0.0,
@@ -988,9 +986,9 @@ def __init__(self, connection, enable_graphs=False):
988986

989987
# Yaw controller for centring gimbal
990988
self._yaw_controller = AC_PID_Basic(
991-
initial_p=2.0,
992-
initial_i=0.0,
993-
initial_d=0.0,
989+
initial_p=1.0,
990+
initial_i=0.01,
991+
initial_d=0.01,
994992
initial_ff=0.0,
995993
initial_imax=1.0,
996994
initial_filt_E_hz=0.0,
@@ -999,9 +997,9 @@ def __init__(self, connection, enable_graphs=False):
999997

1000998
# Gimbal pitch controller for tracking
1001999
self._pit_track_controller = AC_PID_Basic(
1002-
initial_p=2.0,
1003-
initial_i=0.2,
1004-
initial_d=0.01,
1000+
initial_p=1.0,
1001+
initial_i=0.01,
1002+
initial_d=0.02,
10051003
initial_ff=0.0,
10061004
initial_imax=1.0,
10071005
initial_filt_E_hz=0.0,
@@ -1010,9 +1008,9 @@ def __init__(self, connection, enable_graphs=False):
10101008

10111009
# Gimbal yaw controller for tracking
10121010
self._yaw_track_controller = AC_PID_Basic(
1013-
initial_p=2.0,
1014-
initial_i=0.2,
1015-
initial_d=0.01,
1011+
initial_p=1.0,
1012+
initial_i=0.01,
1013+
initial_d=0.02,
10161014
initial_ff=0.0,
10171015
initial_imax=1.0,
10181016
initial_filt_E_hz=0.0,
@@ -1057,7 +1055,7 @@ def reset(self):
10571055
def mavlink_packet(self, msg):
10581056
self._control_in_queue.put(msg)
10591057

1060-
def _send_gimbal_manager_pitch_yaw_angles(self, pitch, yaw, pitch_rate, yaw_rate):
1058+
def _send_gimbal_manager_pitch_yaw_angle(self, pitch, yaw):
10611059
"""
10621060
Send a mavlink message to set the gimbal pitch and yaw (radians).
10631061
"""
@@ -1068,6 +1066,22 @@ def _send_gimbal_manager_pitch_yaw_angles(self, pitch, yaw, pitch_rate, yaw_rate
10681066
0,
10691067
pitch,
10701068
yaw,
1069+
float("nan"),
1070+
float("nan"),
1071+
)
1072+
self._connection.mav.send(msg)
1073+
1074+
def _send_gimbal_manager_pitch_yaw_rate(self, pitch_rate, yaw_rate):
1075+
"""
1076+
Send a mavlink message to set the gimbal pitch and yaw (radians).
1077+
"""
1078+
msg = self._connection.mav.gimbal_manager_set_pitchyaw_encode(
1079+
self._connection.target_system,
1080+
self._connection.target_component,
1081+
0,
1082+
0,
1083+
float("nan"),
1084+
float("nan"),
10711085
pitch_rate,
10721086
yaw_rate,
10731087
)
@@ -1082,6 +1096,9 @@ def _control_task(self):
10821096
10831097
When not tracking, return the gimbal to its neutral position.
10841098
"""
1099+
update_rate = GIMBAL_CONTROL_LOOP_RATE
1100+
update_period = 1.0 / update_rate
1101+
10851102
if self._enable_graphs:
10861103
self._add_livegraphs()
10871104

@@ -1139,12 +1156,11 @@ def _control_task(self):
11391156
# f"Out: {yaw_pid_info.out:.2f}"
11401157
# )
11411158

1142-
self._send_gimbal_manager_pitch_yaw_angles(
1143-
float("nan"),
1144-
float("nan"),
1145-
pit_rate_rads,
1146-
yaw_rate_rads,
1147-
)
1159+
# NOTE: Set pitch and yaw rates to help determine PID gains for tracking
1160+
self._send_gimbal_manager_pitch_yaw_rate(pit_rate_rads, yaw_rate_rads)
1161+
1162+
# NOTE: Set pitch and yaw angles directly (no PID controller needed)
1163+
# self._send_gimbal_manager_pitch_yaw_angle(pit_tgt_rad, yaw_tgt_rad)
11481164

11491165
if self._enable_graphs:
11501166
self._update_livegraphs(pit_pid_info, yaw_pid_info)
@@ -1167,18 +1183,11 @@ def _control_task(self):
11671183
pit_pid_info = self._pit_track_controller.pid_info
11681184
yaw_pid_info = self._yaw_track_controller.pid_info
11691185

1170-
self._send_gimbal_manager_pitch_yaw_angles(
1171-
float("nan"),
1172-
float("nan"),
1173-
pit_rate_rads,
1174-
yaw_rate_rads,
1175-
)
1186+
self._send_gimbal_manager_pitch_yaw_rate(pit_rate_rads, yaw_rate_rads)
11761187

11771188
if self._enable_graphs:
11781189
self._update_livegraphs(pit_pid_info, yaw_pid_info)
11791190

1180-
# Update at 50Hz
1181-
update_period = 0.02
11821191
elapsed_time = time.time() - start_time
11831192
sleep_time = max(0.0, update_period - elapsed_time)
11841193
time.sleep(sleep_time)
@@ -1187,7 +1196,7 @@ def _mavlink_task(self):
11871196
"""
11881197
Process mavlink messages relevant to gimbal management.
11891198
"""
1190-
update_rate = 1000.0
1199+
update_rate = MAVLINK_RECV_RATE
11911200
update_period = 1.0 / update_rate
11921201

11931202
while True:
@@ -1335,6 +1344,22 @@ def _create(self):
13351344
return cv2.legacy.TrackerKCF_create()
13361345

13371346

1347+
class TrackerFactory:
1348+
1349+
@staticmethod
1350+
def choices():
1351+
return ["CSTR", "KCF"]
1352+
1353+
@staticmethod
1354+
def create_tracker(name):
1355+
if name == "CSTR":
1356+
return TrackerCSTR()
1357+
elif name == "KCF":
1358+
return TrackerKCF()
1359+
else:
1360+
raise Exception(f"Invalid tracker name: {name}")
1361+
1362+
13381363
if __name__ == "__main__":
13391364
import os
13401365
import sys
@@ -1352,7 +1377,13 @@ def _create(self):
13521377
type=int,
13531378
help="source component id",
13541379
)
1355-
parser.add_argument("--tracker-name", default="CSTR", type=str, help="tracker name")
1380+
parser.add_argument(
1381+
"--tracker-name",
1382+
choices=TrackerFactory.choices(),
1383+
default="CSTR",
1384+
type=str,
1385+
help="tracker name",
1386+
)
13561387
parser.add_argument(
13571388
"--enable-graphs",
13581389
action="store_const",

0 commit comments

Comments
 (0)