from cereal import car
from selfdrive . car . chrysler . values import CAR
VisualAlert = car . CarControl . HUDControl . VisualAlert
AudibleAlert = car . CarControl . HUDControl . AudibleAlert
MODEL_TO_CONSTANT = {
CAR . PACIFICA_2017_HYBRID : 0 ,
CAR . PACIFICA_2018 : 0x64 ,
CAR . PACIFICA_2018_HYBRID : 0xa8 ,
CAR . PACIFICA_2019_HYBRID : 0x68 ,
CAR . JEEP_CHEROKEE : 0xa4 ,
}
def calc_checksum ( data ) :
""" This function does not want the checksum byte in the input data.
jeep chrysler canbus checksum from http : / / illmatics . com / Remote % 20 Car % 20 Hacking . pdf
"""
end_index = len ( data )
index = 0
checksum = 0xFF
temp_chk = 0 ;
bit_sum = 0 ;
if ( end_index < = index ) :
return False
for index in range ( 0 , end_index ) :
shift = 0x80
curr = data [ index ]
iterate = 8
while ( iterate > 0 ) :
iterate - = 1
bit_sum = curr & shift ;
temp_chk = checksum & 0x80
if ( bit_sum != 0 ) :
bit_sum = 0x1C
if ( temp_chk != 0 ) :
bit_sum = 1
checksum = checksum << 1
temp_chk = checksum | 1
bit_sum ^ = temp_chk
else :
if ( temp_chk != 0 ) :
bit_sum = 0x1D
checksum = checksum << 1
bit_sum ^ = checksum
checksum = bit_sum
shift = shift >> 1
return ~ checksum & 0xFF
def make_can_msg ( addr , dat ) :
return [ addr , 0 , dat , 0 ]
def create_lkas_heartbit ( car_fingerprint ) :
# LKAS_HEARTBIT (729) Lane-keeping heartbeat.
msg = ' 0000000820 ' . decode ( ' hex ' ) # 2017
return make_can_msg ( 0x2d9 , msg )
def create_lkas_hud ( packer , gear , lkas_active , hud_alert , car_fingerprint , hud_count ) :
# LKAS_HUD 0x2a6 (678) Controls what lane-keeping icon is displayed.
if hud_alert == VisualAlert . steerRequired :
msg = ' 0000000300000000 ' . decode ( ' hex ' )
return make_can_msg ( 0x2a6 , msg )
color = 0 # default values are for park or neutral
lines = 0
alerts = 0
if hud_count < ( 3 * 4 ) : # first 3 seconds, 4Hz
lines = 1
alerts = 1
elif hud_count < ( 6 * 4 ) : # next 3 seconds, 4Hz
lines = 1
alerts = 0
# CAR.PACIFICA_2018_HYBRID and CAR.PACIFICA_2019_HYBRID
# had color = 1 and lines = 1 but trying 2017 hybrid style for now.
if gear in ( ' drive ' , ' reverse ' , ' low ' ) :
if lkas_active :
color = 2 # control active, display green.
lines = 6
else :
color = 1 # control off, display white.
lines = 1
values = {
" LKAS_ICON_COLOR " : color , # byte 0, last 2 bits
" CAR_MODEL " : MODEL_TO_CONSTANT [ car_fingerprint ] , # byte 1
" LKAS_LANE_LINES " : lines , # byte 2, last 4 bits
" LKAS_ALERTS " : alerts , # byte 3, last 4 bits
}
return packer . make_can_msg ( " LKAS_HUD " , 0 , values ) # 0x2a6
def create_lkas_command ( packer , apply_steer , frame ) :
# LKAS_COMMAND (658) Lane-keeping signal to turn the wheel.
values = {
" LKAS_STEERING_TORQUE " : apply_steer ,
" LKAS_HIGH_TORQUE " : 1 ,
" COUNTER " : frame % 0x10 ,
}
dat = packer . make_can_msg ( " LKAS_COMMAND " , 0 , values ) [ 2 ]
dat = [ ord ( i ) for i in dat ] [ : - 1 ]
checksum = calc_checksum ( dat )
values [ " CHECKSUM " ] = checksum
return packer . make_can_msg ( " LKAS_COMMAND " , 0 , values )
def create_chimes ( audible_alert ) :
# '0050' nothing, chime '4f55'
if audible_alert == AudibleAlert . none :
msg = ' 0050 ' . decode ( ' hex ' )
else :
msg = ' 4f55 ' . decode ( ' hex ' )
return make_can_msg ( 0x339 , msg )
def create_wheel_buttons ( frame ) :
# WHEEL_BUTTONS (571) Message sent to cancel ACC.
start = [ 0x01 ] # acc cancel set
counter = ( frame % 10 ) << 4
dat = start + [ counter ]
dat = dat + [ calc_checksum ( dat ) ]
return make_can_msg ( 0x23b , str ( bytearray ( dat ) ) )