ardupilot/libraries/AP_Scripting/drivers/mount-viewpro-driver.lua

835 lines
33 KiB
Lua
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

-- mount-viewpro-driver.lua: Viewpro mount/gimbal driver
--[[
How to use
Connect gimbal UART to one of the autopilot's serial ports
Set SERIALx_PROTOCOL = 28 (Scripting) where "x" corresponds to the serial port connected to the gimbal
Set SCR_ENABLE = 1 to enable scripting and reboot the autopilot
Set MNT1_TYPE = 9 (Scripting) to enable the mount/gimbal scripting driver
Set CAM1_TYPE = 7 (Scripting) to enable the camera scripting driver
Set RCx_OPTION = 300 (Scripting1) to allow real-time selection of the video feed and camera control
Copy this script to the autopilot's SD card in the APM/scripts directory and reboot the autopilot
Set VIEP_CAM_SWLOW, VIEP_CAM_SWMID, VIEP_CAM_SWHIGH to which cameras are controlled by the auxiliary switch
0: No change in camera selection
1: EO1
2: IR thermal
3: EO1 + IR Picture-in-picture
4: IR + EO1 Picture-in-picture
5: Fusion
6: IR1 13mm
7: IR2 52mm
Set VIEP_ZOOM_SPEED to control speed of zoom (value between 0 and 7)
Set VIEP_ZOOM_MAX to the maximum zoom. E.g. for a camera with 20x zoom, set to 20
Optionally set VIEP_DEBUG = 1 or 2 to increase level of debug output to the GCS
Packet format
byte description notes
0~2 header 0x55 0xAA 0xDC
3 body length bit0~5: body length, n=all bytes from byte3 to checksum, min=4, max=63. bits6~7: frame counter
4 frame id
5~n+1 data 1st byte is command id?
n+2 checksum XOR of byte3 to n+1 (inclusive)
--]]
-- parameters
local PARAM_TABLE_KEY = 39
assert(param:add_table(PARAM_TABLE_KEY, "VIEP_", 6), "could not add param table")
assert(param:add_param(PARAM_TABLE_KEY, 1, "DEBUG", 0), "could not add VIEP_DEBUG param")
assert(param:add_param(PARAM_TABLE_KEY, 2, "CAM_SWLOW", 1), "could not add VIEP_CAM_SWLOW param")
assert(param:add_param(PARAM_TABLE_KEY, 3, "CAM_SWMID", 2), "could not add VIEP_CAM_SWMID param")
assert(param:add_param(PARAM_TABLE_KEY, 4, "CAM_SWHIGH", 6), "could not add VIEP_CAM_CAM_SWHIGH param")
assert(param:add_param(PARAM_TABLE_KEY, 5, "ZOOM_SPEED", 7), "could not add VIEP_ZOOM_SPEED param")
assert(param:add_param(PARAM_TABLE_KEY, 6, "ZOOM_MAX", 20), "could not add VIEP_ZOOM_MAX param")
-- bind parameters to variables
local MNT1_TYPE = Parameter("MNT1_TYPE") -- should be 9:Scripting
local CAM1_TYPE = Parameter("CAM1_TYPE") -- should be 7:Scripting
--[[
// @Param: VIEP_DEBUG
// @DisplayName: ViewPro debug
// @Description: ViewPro debug
// @Values: 0:Disabled, 1:Enabled, 2:Enabled including attitude reporting
// @User: Advanced
--]]
local VIEP_DEBUG = Parameter("VIEP_DEBUG") -- debug level. 0:disabled 1:enabled 2:enabled with attitude reporting
--[[
// @Param: VIEP_CAM_SWLOW
// @DisplayName: ViewPro Camera For Switch Low
// @Description: Camera selection when switch is in low position
// @Values: 0:No change in camera selection, 1:EO1, 2:IR thermal, 3:EO1 + IR Picture-in-picture, 4:IR + EO1 Picture-in-picture, 5:Fusion, 6:IR1 13mm, 7:IR2 52mm
// @User: Standard
--]]
local VIEP_CAM_SWLOW = Parameter("VIEP_CAM_SWLOW") -- RC swith low position's camera selection
--[[
// @Param: VIEP_CAM_SWMID
// @DisplayName: ViewPro Camera For Switch Mid
// @Description: Camera selection when switch is in middle position
// @Values: 0:No change in camera selection, 1:EO1, 2:IR thermal, 3:EO1 + IR Picture-in-picture, 4:IR + EO1 Picture-in-picture, 5:Fusion, 6:IR1 13mm, 7:IR2 52mm
// @User: Standard
--]]
local VIEP_CAM_SWMID = Parameter("VIEP_CAM_SWMID") -- RC swith middle position's camera selection
--[[
// @Param: VIEP_CAM_SWHIGH
// @DisplayName: ViewPro Camera For Switch High
// @Description: Camera selection when switch is in high position
// @Values: 0:No change in camera selection, 1:EO1, 2:IR thermal, 3:EO1 + IR Picture-in-picture, 4:IR + EO1 Picture-in-picture, 5:Fusion, 6:IR1 13mm, 7:IR2 52mm
// @User: Standard
--]]
local VIEP_CAM_SWHIGH = Parameter("VIEP_CAM_SWHIGH") -- RC swith high position's camera selection
--[[
// @Param: VIEP_ZOOM_SPEED
// @DisplayName: ViewPro Zoom Speed
// @Description: ViewPro Zoom Speed. Higher numbers result in faster zooming
// @Range: 0 7
// @User: Standard
--]]
local VIEP_ZOOM_SPEED = Parameter("VIEP_ZOOM_SPEED") -- zoom speed from 0 (slow) to 7 (fast)
--[[
// @Param: VIEP_ZOOM_MAX
// @DisplayName: ViewPro Zoom Times Max
// @Description: ViewPro Zoom Times Max
// @Range: 0 30
// @User: Standard
--]]
local VIEP_ZOOM_MAX = Parameter("VIEP_ZOOM_MAX") -- zoom times max
-- global definitions
local CAM_SELECT_RC_OPTION = 300 -- rc channel option used to control which camera/video is used. RCx_OPTION = 300 (scripting1)
local MAV_SEVERITY = {EMERGENCY=0, ALERT=1, CRITICAL=2, ERROR=3, WARNING=4, NOTICE=5, INFO=6, DEBUG=7}
local INIT_INTERVAL_MS = 3000 -- attempt to initialise the gimbal at this interval
local UPDATE_INTERVAL_MS = 100 -- update at 10hz
local MOUNT_INSTANCE = 0 -- always control the first mount/gimbal
local CAMERA_INSTANCE = 0 -- always use the first camera
-- packet parsing definitions
local HEADER1 = 0x55 -- 1st header byte
local HEADER2 = 0xAA -- 2nd header byte
local HEADER3 = 0xDC -- 3rd header byte
local PACKET_LENGTH_MIN = 4 -- serial packet minimum length. used for sanity checks
local PACKET_LENGTH_MAX = 63 -- serial packet maximum length. used for sanity checks
-- parsing state definitions
local PARSE_STATE_WAITING_FOR_HEADER1 = 0
local PARSE_STATE_WAITING_FOR_HEADER2 = 1
local PARSE_STATE_WAITING_FOR_HEADER3 = 2
local PARSE_STATE_WAITING_FOR_LENGTH = 3
local PARSE_STATE_WAITING_FOR_FRAMEID = 4
local PARSE_STATE_WAITING_FOR_DATA = 5
-- received FrameId
local T1_F1_B1_D1_FRAMEID = 0x40 -- includes roll, pitch, yaw angles
-- camera operation commands
local CAM_COMMAND_NO_ACTION = 0x00
local CAM_COMMAND_STOP_FOCUS_AND_ZOOM = 0x01
local CAM_COMMAND_ZOOM_OUT = 0x08
local CAM_COMMAND_ZOOM_IN = 0x09
local CAM_COMMAND_FOCUS_PLUS = 0x0A
local CAM_COMMAND_FOCUS_MINUS = 0x0B
local CAM_COMMAND_TAKE_PICTURE = 0x13
local CAM_COMMAND_START_RECORD = 0x14
local CAM_COMMAND_STOP_RECORD = 0x15
local CAM_COMMAND_AUTO_FOCUS = 0x19
local CAM_COMMAND_MANUAL_FOCUS = 0x1A
-- tracking commands
local TRACK_COMMAND_STOP = 0x01
local TRACK_COMMAND_START = 0x03
local TRACK_COMMAND2_SET_POINT = 0x0A
local TRACK_COMMAND2_SET_RECT_TOPLEFT = 0x0B
local TRACK_COMMAND2_SET_RECT_BOTTOMRIGHT = 0x0C
-- camera control2 commands
local CAM_COMMAND2_SET_EO_ZOOM = 0x53
-- hardcoded outgoing messages
local HEARTBEAT_MSG = {0x55,0xAA,0xDC,0x44,0x00,0x00,0x44}
-- local variables and definitions
local uart -- uart object connected to mount
local initialised = false -- true once connection to gimbal has been initialised
local parse_state = PARSE_STATE_WAITING_FOR_HEADER1 -- parse state
local parse_expected_crc = 0 -- incoming messages expected crc. this is checked against actual crc received
local parse_length = 0 -- incoming message parsed length
local parse_frameid = 0 -- incoming message command id
local parse_data_buff = {} -- data buffer holding roll, pitch and yaw angles from gimbal
local parse_data_bytes_recv = 0 -- count of the number of bytes received in the message so far
local last_frame_counter = 0 -- last frame counter sent to gimbal. always between 0 and 3
local cam_choice = 0 -- last camera choice (see VIEP_CAM_SWLOW/MID/HIGH parameters)
local cam_pic_count = 0 -- last picture count. used to detect trigger pic
local cam_rec_video = false -- last record video state. used to detect record video
local cam_zoom_type = 0 -- last zoom type 1:Rate 2:Pct
local cam_zoom_value = 0 -- last zoom value. If rate, zoom out = -1, hold = 0, zoom in = 1. If Pct then value from 0 to 100
local cam_focus_type = 0 -- last focus type 1:Rate, 2:Pct, 4:Auto
local cam_focus_value = 0 -- last focus value. If Rate then focus in = -1, focus hold = 0, focus out = 1
local last_tracking_type = 0 -- last recorded tracking type (0:None, 1:Point, 2:Rectangle)
local last_tracking_p1x = 0 -- last recorded tracking point1 (used for center or top-left)
local last_tracking_p1y = 0 -- last recorded tracking point1 (used for center or top-left)
local last_tracking_p2x = 0 -- last recorded tracking point2 (bottom-right)
local last_tracking_p2y = 0 -- last recorded tracking point2 (bottom-right)
local tracking_active = false -- true when tracking is active (rate and angle controls are disabled)
-- parsing status reporting variables
local last_print_ms = 0 -- system time that debug output was last printed
local bytes_read = 0 -- number of bytes read from gimbal
local bytes_written = 0 -- number of bytes written to gimbal
local bytes_error = 0 -- number of bytes read that could not be parsed
local msg_ignored = 0 -- number of ignored messages (because frame id does not match)
-- debug variables
local last_test_send_md = 0 -- system time that a test message was last sent
local debug_buff = {} -- debug buffer to display bytes from gimbal
-- get lowbyte of a number
function lowbyte(num)
return num & 0xFF
end
-- get highbyte of a number
function highbyte(num)
return (num >> 8) & 0xFF
end
-- get uint16 from two bytes
function uint16_value(hbyte, lbyte)
return ((hbyte & 0xFF) << 8) | (lbyte & 0xFF)
end
-- get int16 from two bytes
function int16_value(hbyte, lbyte)
local uret = uint16_value(hbyte, lbyte)
if uret <= 0x8000 then
return uret
else
return uret - 0x10000
end
end
-- wrap yaw angle in degrees to value between 0 and 360
function wrap_360(angle)
local res = math.fmod(angle, 360.0)
if res < 0 then
res = res + 360.0
end
return res
end
-- wrap yaw angle in degrees to value between -180 and +180
function wrap_180(angle_deg)
local res = wrap_360(angle_deg)
if res > 180 then
res = res - 360
end
return res
end
-- calculate crc from existing crc value and new byte
function calc_crc(orig_crc, b)
local crc = (orig_crc ~ b) & 0xFF
return crc
end
-- find and initialise serial port connected to gimbal
function init()
-- check mount parameter
if MNT1_TYPE:get() ~= 9 then
gcs:send_text(MAV_SEVERITY.CRITICAL, "ViewPro: set MNT1_TYPE=9")
do return end
end
-- check cam type parametr
if CAM1_TYPE:get() ~= 7 then
gcs:send_text(MAV_SEVERITY.CRITICAL, "ViewPro: set CAM1_TYPE=7")
do return end
end
-- find and init first instance of SERIALx_PROTOCOL = 28 (Scripting)
uart = serial:find_serial(0)
if uart == nil then
gcs:send_text(3, "ViewPro: no SERIALx_PROTOCOL = 28") -- MAV_SEVERITY_ERR
else
uart:begin(115200)
uart:set_flow_control(0)
initialised = true
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: started")
end
end
-- send hard coded message
function send_msg(msg)
for i=1,#msg do
uart:write(msg[i])
-- debug
bytes_written = bytes_written + 1
end
end
-- parse test message
function parse_test_msg(msg)
for i=1,#msg do
parse_byte(msg[i])
end
end
-- reading incoming packets from gimbal
function read_incoming_packets()
local n_bytes = uart:available()
while n_bytes > 0 do
n_bytes = n_bytes - 1
parse_byte(uart:read())
end
end
-- parse a single byte from gimbal
function parse_byte(b)
-- record num bytes for reporting
bytes_read = bytes_read + 1
-- debug
if VIEP_DEBUG:get() > 1 then
debug_buff[#debug_buff+1] = b
if #debug_buff >= 10 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: %x %x %x %x %x %x %x %x %x %x", debug_buff[1], debug_buff[2], debug_buff[3], debug_buff[4], debug_buff[5], debug_buff[6], debug_buff[7], debug_buff[8], debug_buff[9], debug_buff[10]))
debug_buff = {}
end
end
-- waiting for header1
if parse_state == PARSE_STATE_WAITING_FOR_HEADER1 then
if b == HEADER1 then
parse_state = PARSE_STATE_WAITING_FOR_HEADER2
parse_expected_crc = 0
parse_data_bytes_recv = 0
do return end
end
bytes_error = bytes_error + 1
end
-- waiting for header2
if parse_state == PARSE_STATE_WAITING_FOR_HEADER2 then
if b == HEADER2 then
parse_state = PARSE_STATE_WAITING_FOR_HEADER3
else
-- unexpected byte so reset parsing state
parse_state = PARSE_STATE_WAITING_FOR_HEADER1
bytes_error = bytes_error + 1
end
do return end
end
-- waiting for header3
if parse_state == PARSE_STATE_WAITING_FOR_HEADER3 then
if b == HEADER3 then
parse_state = PARSE_STATE_WAITING_FOR_LENGTH
else
-- unexpected byte so reset parsing state
parse_state = PARSE_STATE_WAITING_FOR_HEADER1
bytes_error = bytes_error + 1
end
do return end
end
-- waiting for length
if parse_state == PARSE_STATE_WAITING_FOR_LENGTH then
parse_expected_crc = calc_crc(parse_expected_crc, b)
parse_length = b & 0x3F
if parse_length >= PACKET_LENGTH_MIN and parse_length <= PACKET_LENGTH_MAX then
parse_state = PARSE_STATE_WAITING_FOR_FRAMEID
else
-- unexpected length
parse_state = PARSE_STATE_WAITING_FOR_HEADER1
bytes_error = bytes_error + 1
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.ERROR, string.format("ViewPro: invalid len:%d", parse_length))
end
end
do return end
end
-- waiting for command id
if parse_state == PARSE_STATE_WAITING_FOR_FRAMEID then
parse_expected_crc = calc_crc(parse_expected_crc, b)
parse_frameid = b
parse_state = PARSE_STATE_WAITING_FOR_DATA
do return end
end
-- waiting for data
if parse_state == PARSE_STATE_WAITING_FOR_DATA then
-- check for crc
if parse_data_bytes_recv >= parse_length - 3 then
if b == parse_expected_crc then
-- crc matched, process packet
local processed = false
-- T1_F1_B1_D1
if parse_frameid == T1_F1_B1_D1_FRAMEID then
processed = true
-- T1 holds target info including target lean angles
-- F1 holds tracker sensor status (which camera, tracking vs lost)
-- B1 section holds actual lean angles
-- D1 section holds camera status including zoom level
local servo_status = (parse_data_buff[24] & 0xF0 >> 4)
local roll_deg = int16_value(parse_data_buff[24] & 0x0F, parse_data_buff[25]) * (180.0/4095.0) - 90.0
local yaw_deg = int16_value(parse_data_buff[26], parse_data_buff[27]) * (360.0 / 65536.0)
local pitch_deg = int16_value(parse_data_buff[28], parse_data_buff[29]) * (360.0 / 65536.0)
mount:set_attitude_euler(MOUNT_INSTANCE, roll_deg, pitch_deg, yaw_deg)
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: r:%f p:%f y:%f ss:%x", roll_deg, pitch_deg, yaw_deg, servo_status))
end
end
if not processed then
msg_ignored = msg_ignored + 1
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: ignored frameid:%x", parse_frameid))
end
end
else
-- crc mismatch
bytes_error = bytes_error + 1
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: crc exp:%x got:%x", parse_expected_crc, b))
end
end
parse_state = PARSE_STATE_WAITING_FOR_HEADER1
do return end
end
-- add latest byte to crc and buffer
parse_expected_crc = calc_crc(parse_expected_crc, b)
parse_data_bytes_recv = parse_data_bytes_recv + 1
parse_data_buff[parse_data_bytes_recv] = b
end
end
-- write a byte to the uart and update the checksum
function write_byte(b, checksum)
if b == nil or checksum == nil then
gcs:send_text(3, "ViewPro: failed to write byte") -- MAV_SEVERITY_ERR
return
end
local byte_to_write = b & 0xFF
uart:write(byte_to_write)
bytes_written = bytes_written + 1
local checksum_ret = (checksum ~ byte_to_write) & 0xFF
return checksum_ret
end
-- calculate length and frame count byte
-- length is all bytes after the header including CRC
function calc_length_and_frame_byte(length)
-- increment frame counter
last_frame_counter = last_frame_counter + 1 & 0x03
return (last_frame_counter << 6 | length & 0x3F) & 0xFF
end
-- send target angles (in degrees) to gimbal
-- yaw_angle_deg is always a body-frame angle
function send_target_angles(pitch_angle_deg, yaw_angle_deg)
-- prepare A1 message, FrameId 0x1A, CmdId 0x0B
local length_and_frame_counter = calc_length_and_frame_byte(0x0C)
local pitch_angle_output = math.floor((-pitch_angle_deg / 360.0 * 65536.0) + 0.5)
local yaw_angle_output = math.floor((yaw_angle_deg / 360.0 * 65536.0) + 0.5)
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x1A, checksum) -- 0x1A: A1 FrameId
checksum = write_byte(0x0B, checksum) -- 0x0B: absolute angle
checksum = write_byte(highbyte(yaw_angle_output), checksum) -- yaw angle MSB
checksum = write_byte(lowbyte(yaw_angle_output), checksum) -- yaw angle LSB
checksum = write_byte(highbyte(pitch_angle_output), checksum) -- pitch angle MSB
checksum = write_byte(lowbyte(pitch_angle_output), checksum) -- pitch angle LSB
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
write_byte(checksum, 0) -- checksum
end
-- send target rates (in deg/sec) to gimbal
function send_target_rates(pitch_rate_degs, yaw_rate_degs)
-- prepare A1 message, FrameId 0x1A, CmdId 0x01
local length_and_frame_counter = calc_length_and_frame_byte(0x0C)
local pitch_rate_output = math.floor((-pitch_rate_degs * 100.0) + 0.5)
local yaw_rate_output = math.floor((yaw_rate_degs * 100.0) + 0.5)
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x1A, checksum) -- 0x1A: A1 FrameId
checksum = write_byte(0x01, checksum) -- 0x01: manual rate angle
checksum = write_byte(highbyte(yaw_rate_output), checksum) -- yaw rate MSB
checksum = write_byte(lowbyte(yaw_rate_output), checksum) -- yaw rate LSB
checksum = write_byte(highbyte(pitch_rate_output), checksum) -- pitch angle MSB
checksum = write_byte(lowbyte(pitch_rate_output), checksum) -- pitch angle LSB
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
checksum = write_byte(0, checksum) -- unused
write_byte(checksum, 0) -- checksum
end
-- send camera commands
function send_camera_control(camera_choice, cam_command)
-- prepare C1 message, FrameId 0x1C
-- bits 0~2 : video choose
-- bits 3~5 : zoom speed
-- bits 6~12 : operation command
-- bits 13~15 : LRF (rangefinder)
local length_and_frame_counter = calc_length_and_frame_byte(0x05)
local video_choose = camera_choice & 0x07
local zoom_speed = 0
if cam_command == CAM_COMMAND_ZOOM_OUT or cam_command == CAM_COMMAND_ZOOM_IN then
zoom_speed = (VIEP_ZOOM_SPEED:get() & 0x07) << 3
end
local operation_cmd = ((cam_command & 0xFFFF) & 0x7F) << 6
local data_bytes = video_choose | zoom_speed | operation_cmd
-- debug
--gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: camcmd:%x msb:%x lsb:%x", cam_command, highbyte(cmd_shifted), lowbyte(cmd_shifted)))
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: cam choice:%x", video_choose))
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x1C, checksum) -- 0x1C: C1 FrameId
checksum = write_byte(highbyte(data_bytes), checksum) -- msb
checksum = write_byte(lowbyte(data_bytes), checksum) -- lsb
write_byte(checksum, 0) -- checksum
end
-- send camera commands using C2 message
function send_camera_control2(cam_command, cam_value)
-- ensure cam_value is an integer
cam_value = math.floor(cam_value + 0.5)
-- prepare C2 message, FrameId 0x2C
-- byte0 : command (0x50:set brightness, 0x51:set contrast, 0x52: set aperture, 0x53:set EO zoom, 0x54:set focus, 0x55:set ISO, 0x56:set thermal cam digital zoom)
-- byte1~2 : value related to command
local length_and_frame_counter = calc_length_and_frame_byte(0x06)
cam_command = cam_command & 0xFF
local data_bytes = cam_value & 0xFFFF
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x2C, checksum) -- 0x2C: C2 FrameId
checksum = write_byte(cam_command, checksum) -- command
checksum = write_byte(highbyte(data_bytes), checksum) -- msb
checksum = write_byte(lowbyte(data_bytes), checksum) -- lsb
write_byte(checksum, 0) -- checksum
end
-- send tracking commands using E1 message
function send_tracking_control(camera_choice, tracking_command, param2)
-- prepare E1 message, FrameId 0x1E
-- byte0 : tracking source (0x01:EO 1, 0x02: IR, 0x03:EO 2)
-- byte1 : basic command
-- 0x02: Search (Bring up the cross
-- 0x03: Turn on tracking
-- 0x04: Switch tracking point to cross position(take placed by enable tracking)
-- 0X05: AI function ON/OFF
-- 0X08: For AI. Auto track target once identified
-- 0x09: For AI. Change target when click targets identified
-- 0x0A: For AI. Not change target when click identified targets.
-- 0X21: 32× 32 small template
-- 0X22: 64× 64 medium template
-- 0X23: 128× 128 big template
-- 0X24: Self-adapt between small and medium template
-- 0X25: Self-adapt between small and big template
-- 0X26: Self-adapt between medium and big template
-- 0X28: Self-adaption between small, medium and big template
-- byte2 : parameter 2, value related to above command
local length_and_frame_counter = calc_length_and_frame_byte(0x06)
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x1E, checksum) -- 0x1E: E1 FrameId
checksum = write_byte(camera_choice, checksum) -- camera choice
checksum = write_byte(tracking_command & 0xFF, checksum) -- tracking command
checksum = write_byte(param2 & 0xFF, checksum) -- param2
write_byte(checksum, 0) -- checksum
end
-- send tracking commands using E2 message
function send_tracking_control2(tracking_command2, param1, param2)
-- prepare E2 message, FrameId 0x2E
-- byte0 : tracking source (0x01:EO 1, 0x02: IR, 0x03:EO 2)
-- byte1 : basic command
-- 0x0A: The tracking point moves to the commanded position
-- 0x0B: Rectangular tracking area, top left corner point set
-- 0x0C: Rectangular tracking area, lower right corner point set
-- byte2~3 : Tracking point yaw, 1bit=1pixel, -960~960, 0 is center, negative is left, positive is right
-- byte4~5 : Tracking point pitch, 1bit=1pixel, -540~540, 0 is center, negative is up, positive is down
param1 = math.floor(param1 + 0.5) & 0xFFFF
param2 = math.floor(param2 + 0.5) & 0xFFFF
local length_and_frame_counter = calc_length_and_frame_byte(0x08)
write_byte(HEADER1, 0)
write_byte(HEADER2, 0)
write_byte(HEADER3, 0)
local checksum = write_byte(length_and_frame_counter, 0) -- length and frame count
checksum = write_byte(0x2E, checksum) -- 0x1E: E1 FrameId
checksum = write_byte(tracking_command2 & 0xFF, checksum) -- tracking command2
checksum = write_byte(highbyte(param1), checksum) -- param1 msb
checksum = write_byte(lowbyte(param1), checksum) -- param1 lsb
checksum = write_byte(highbyte(param2), checksum) -- param2 msb
checksum = write_byte(lowbyte(param2), checksum) -- param2 lsb
write_byte(checksum, 0) -- checksum
end
-- return camera selection according to RC switch position and VIEW_CAM_SWxxx parameter
-- used in C1 message's "video choose" to specify which cameras should be controlled
function get_camera_choice()
local cam_switch_pos = rc:get_aux_cached(CAM_SELECT_RC_OPTION)
if cam_switch_pos == 0 then
return VIEP_CAM_SWLOW:get()
end
if cam_switch_pos == 1 then
return VIEP_CAM_SWMID:get()
end
return VIEP_CAM_SWHIGH:get()
end
-- check for changes in camera state and send messages to gimbal if required
function check_camera_state()
-- check for change in camera
local curr_cam_choice = get_camera_choice()
if cam_choice ~= curr_cam_choice then
cam_choice = curr_cam_choice
send_camera_control(cam_choice, CAM_COMMAND_NO_ACTION)
end
-- get latest camera state from AP driver
local cam_state = camera:get_state(CAMERA_INSTANCE)
if not cam_state then
return
end
-- check for take picture
if cam_state:take_pic_incr() and cam_state:take_pic_incr() ~= cam_pic_count then
cam_pic_count = cam_state:take_pic_incr()
send_camera_control(cam_choice, CAM_COMMAND_TAKE_PICTURE)
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: took pic %u", pic_count))
end
end
-- check for start/stop recording video
if cam_state:recording_video() ~= cam_rec_video then
cam_rec_video = cam_state:recording_video()
if cam_rec_video > 0 then
send_camera_control(cam_choice, CAM_COMMAND_START_RECORD)
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: start recording")
else
send_camera_control(cam_choice, CAM_COMMAND_STOP_RECORD)
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: stop recording")
end
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: rec video:" .. tostring(cam_rec_video))
end
end
-- check zoom
-- zoom out = -1, hold = 0, zoom in = 1
local zoom_type_changed = cam_state:zoom_type() and (cam_state:zoom_type() ~= cam_zoom_type)
local zoom_value_changed = cam_state:zoom_value() and (cam_state:zoom_value() ~= cam_zoom_value)
if (zoom_type_changed or zoom_value_changed) then
cam_zoom_type = cam_state:zoom_type()
cam_zoom_value = cam_state:zoom_value()
-- zoom rate
if cam_zoom_type == 1 then
if cam_zoom_value < 0 then
send_camera_control(cam_choice, CAM_COMMAND_ZOOM_OUT)
elseif cam_zoom_value > 0 then
send_camera_control(cam_choice, CAM_COMMAND_ZOOM_IN)
else
send_camera_control(cam_choice, CAM_COMMAND_STOP_FOCUS_AND_ZOOM)
end
end
-- zoom percent
if cam_zoom_type == 2 then
-- convert zoom percentage (in the range 0 to 100) to value in the range of 0 to VIEW_ZOOM_MAX * 10
send_camera_control2(CAM_COMMAND2_SET_EO_ZOOM, VIEP_ZOOM_MAX:get() * cam_zoom_value * 0.1)
end
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: zoom type:" .. tostring(cam_zoom_type) .. " value:" .. tostring(cam_zoom_value))
end
end
-- check manual focus
-- focus in = -1, focus hold = 0, focus out = 1
local focus_type_changed = cam_state:focus_type() and (cam_state:focus_type() ~= cam_focus_type)
local focus_value_changed = cam_state:focus_value() and (cam_state:focus_value() ~= cam_focus_value)
if (focus_type_changed or focus_value_changed) then
cam_focus_type = cam_state:focus_type()
cam_focus_value = cam_state:focus_value()
-- focus rate
if cam_focus_type == 1 then
if cam_focus_value < 0 then
send_camera_control(cam_choice, CAM_COMMAND_MANUAL_FOCUS)
send_camera_control(cam_choice, CAM_COMMAND_FOCUS_MINUS)
elseif cam_focus_value == 0 then
send_camera_control(cam_choice, CAM_COMMAND_STOP_FOCUS_AND_ZOOM)
elseif cam_focus_value > 0 then
send_camera_control(cam_choice, CAM_COMMAND_MANUAL_FOCUS)
send_camera_control(cam_choice, CAM_COMMAND_FOCUS_PLUS)
end
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: focus:" .. tostring(cam_focus_step))
end
end
-- check auto focus
if cam_focus_type == 4 then
send_camera_control(cam_choice, CAM_COMMAND_AUTO_FOCUS)
if VIEP_DEBUG:get() > 0 then
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: auto focus:" .. tostring(cam_autofocus))
end
end
end
end
-- check for changes in tracking state and send messages to gimbal if required
function check_tracking_state()
-- get latest camera state from AP driver
local cam_state = camera:get_state(CAMERA_INSTANCE)
if not cam_state then
return
end
-- get current camera choice
local curr_cam_choice = get_camera_choice()
-- check for change in tracking state
local tracking_type_changed = cam_state:tracking_type() ~= last_tracking_type
local tracking_type_p1_changed = (cam_state:tracking_p1():x() ~= last_tracking_p1x) or (cam_state:tracking_p1():y() ~= last_tracking_p1y)
local tracking_type_p2_changed = (cam_state:tracking_p2():x() ~= last_tracking_p2x) or (cam_state:tracking_p2():y() ~= last_tracking_p2y)
last_tracking_type = cam_state:tracking_type()
last_tracking_p1x = cam_state:tracking_p1():x()
last_tracking_p1y = cam_state:tracking_p1():y()
last_tracking_p2x = cam_state:tracking_p2():x()
last_tracking_p2y = cam_state:tracking_p2():y()
if (last_tracking_type == 0) and tracking_type_changed then
-- turn off tracking
send_tracking_control(curr_cam_choice, TRACK_COMMAND_STOP, 0)
tracking_active = false
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: tracking OFF")
end
if (last_tracking_type == 1) and (tracking_type_changed or tracking_type_p1_changed) then
-- turn tracking point on
local yaw_value = (last_tracking_p1x - 0.5) * 960
local pitch_value = (last_tracking_p1y - 0.5) * 540
send_tracking_control(curr_cam_choice, TRACK_COMMAND_START, 0)
send_tracking_control2(TRACK_COMMAND2_SET_POINT, yaw_value, pitch_value)
tracking_active = true
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: tracking point ON")
end
if (last_tracking_type == 2) and (tracking_type_changed or tracking_type_p1_changed or tracking_type_p2_changed) then
-- turn tracking rectangle on
local yaw_value1 = (last_tracking_p1x - 0.5) * 960
local pitch_value1 = (last_tracking_p1y - 0.5) * 540
local yaw_value2 = (last_tracking_p2x - 0.5) * 960
local pitch_value2 = (last_tracking_p2y - 0.5) * 540
send_tracking_control(curr_cam_choice, TRACK_COMMAND_START, 0)
send_tracking_control2(TRACK_COMMAND2_SET_RECT_TOPLEFT, yaw_value1, pitch_value1)
send_tracking_control2(TRACK_COMMAND2_SET_RECT_BOTTOMRIGHT, yaw_value2, pitch_value2)
tracking_active = true
gcs:send_text(MAV_SEVERITY.INFO, "ViewPro: tracking rectangle ON")
end
end
-- the main update function that performs a simplified version of RTL
function update()
-- get current system time
local now_ms = millis()
-- initialise connection to gimbal
if not initialised then
init()
return update, INIT_INTERVAL_MS
end
-- status reporting
if (VIEP_DEBUG:get() > 0) and (now_ms - last_print_ms > 5000) then
last_print_ms = now_ms
gcs:send_text(MAV_SEVERITY.INFO, string.format("ViewPro: r:%u w:%u err:%u ign:%u", bytes_read, bytes_written, bytes_error, msg_ignored))
end
-- consume incoming bytes
read_incoming_packets()
-- check camera state and send control messages
check_camera_state()
-- check tracking state and send tracking control messages
check_tracking_state()
-- send heartbeat, gimbal should respond with T1+F1+B1+D1
send_msg(HEARTBEAT_MSG)
-- request gimbal attitude by sending heartbeat
if now_ms - last_test_send_md > 1000 then
last_test_send_md = now_ms
send_msg(HEARTBEAT_MSG)
end
if not tracking_active then
-- send target angle to gimbal
local roll_deg, pitch_deg, yaw_deg, yaw_is_ef = mount:get_angle_target(MOUNT_INSTANCE)
if roll_deg and pitch_deg and yaw_deg then
if yaw_is_ef then
yaw_deg = wrap_180(yaw_deg - math.deg(ahrs:get_yaw()))
end
send_target_angles(pitch_deg, yaw_deg)
return update, UPDATE_INTERVAL_MS
end
-- send target rate to gimbal
local roll_degs, pitch_degs, yaw_degs, _ = mount:get_rate_target(MOUNT_INSTANCE)
if roll_degs and pitch_degs and yaw_degs then
send_target_rates(pitch_degs, yaw_degs)
return update, UPDATE_INTERVAL_MS
end
end
return update, UPDATE_INTERVAL_MS
end
return update()