2020-05-09 09:08:11 +08:00
|
|
|
#!/usr/bin/env python3
|
2020-01-18 03:03:22 +08:00
|
|
|
import datetime
|
2020-08-24 20:17:41 +08:00
|
|
|
import os
|
2020-08-19 02:52:01 +08:00
|
|
|
import time
|
2020-10-15 21:47:38 +08:00
|
|
|
from typing import Dict, Optional, Tuple
|
2020-08-24 20:17:41 +08:00
|
|
|
|
|
|
|
import psutil
|
2020-01-18 03:03:22 +08:00
|
|
|
from smbus2 import SMBus
|
2020-08-24 20:17:41 +08:00
|
|
|
|
|
|
|
import cereal.messaging as messaging
|
2020-01-18 03:03:22 +08:00
|
|
|
from cereal import log
|
|
|
|
from common.filter_simple import FirstOrderFilter
|
2020-08-24 20:17:41 +08:00
|
|
|
from common.numpy_fast import clip, interp
|
2020-10-21 05:04:18 +08:00
|
|
|
from common.params import Params
|
2020-08-24 20:17:41 +08:00
|
|
|
from common.realtime import DT_TRML, sec_since_boot
|
2020-08-12 07:23:57 +08:00
|
|
|
from selfdrive.controls.lib.alertmanager import set_offroad_alert
|
2021-02-26 13:10:10 +08:00
|
|
|
from selfdrive.hardware import EON, TICI, HARDWARE
|
2020-01-18 03:03:22 +08:00
|
|
|
from selfdrive.loggerd.config import get_available_percent
|
|
|
|
from selfdrive.pandad import get_expected_signature
|
2020-08-24 20:17:41 +08:00
|
|
|
from selfdrive.swaglog import cloudlog
|
2020-12-02 22:20:57 +08:00
|
|
|
from selfdrive.thermald.power_monitoring import PowerMonitoring
|
2020-08-24 20:17:41 +08:00
|
|
|
from selfdrive.version import get_git_branch, terms_version, training_version
|
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
FW_SIGNATURE = get_expected_signature()
|
|
|
|
|
2021-03-02 04:14:29 +08:00
|
|
|
DISABLE_LTE_ONROAD = os.path.exists("/persist/disable_lte_onroad") or TICI
|
2021-02-26 13:10:10 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
ThermalStatus = log.DeviceState.ThermalStatus
|
|
|
|
NetworkType = log.DeviceState.NetworkType
|
|
|
|
NetworkStrength = log.DeviceState.NetworkStrength
|
2020-01-18 03:03:22 +08:00
|
|
|
CURRENT_TAU = 15. # 15s time constant
|
2020-04-02 06:58:40 +08:00
|
|
|
CPU_TEMP_TAU = 5. # 5s time constant
|
2020-01-18 03:03:22 +08:00
|
|
|
DAYS_NO_CONNECTIVITY_MAX = 7 # do not allow to engage after a week without internet
|
|
|
|
DAYS_NO_CONNECTIVITY_PROMPT = 4 # send an offroad prompt after 4 days with no internet
|
2020-04-21 05:15:01 +08:00
|
|
|
DISCONNECT_TIMEOUT = 5. # wait 5 seconds before going offroad after disconnect so you get an alert
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-10-15 21:47:38 +08:00
|
|
|
prev_offroad_states: Dict[str, Tuple[bool, Optional[str]]] = {}
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
LEON = False
|
|
|
|
last_eon_fan_val = None
|
|
|
|
|
2020-08-24 20:17:41 +08:00
|
|
|
def read_tz(x):
|
|
|
|
if x is None:
|
2020-02-29 12:27:41 +08:00
|
|
|
return 0
|
2020-08-24 20:17:41 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
try:
|
2020-12-30 14:32:03 +08:00
|
|
|
with open(f"/sys/devices/virtual/thermal/thermal_zone{x}/temp") as f:
|
2020-08-24 20:17:41 +08:00
|
|
|
return int(f.read())
|
2020-01-18 03:03:22 +08:00
|
|
|
except FileNotFoundError:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2020-08-24 20:17:41 +08:00
|
|
|
def read_thermal(thermal_config):
|
2021-02-17 13:39:32 +08:00
|
|
|
dat = messaging.new_message('deviceState')
|
|
|
|
dat.deviceState.cpuTempC = [read_tz(z) / thermal_config.cpu[1] for z in thermal_config.cpu[0]]
|
|
|
|
dat.deviceState.gpuTempC = [read_tz(z) / thermal_config.gpu[1] for z in thermal_config.gpu[0]]
|
|
|
|
dat.deviceState.memoryTempC = read_tz(thermal_config.mem[0]) / thermal_config.mem[1]
|
|
|
|
dat.deviceState.ambientTempC = read_tz(thermal_config.ambient[0]) / thermal_config.ambient[1]
|
|
|
|
dat.deviceState.batteryTempC = read_tz(thermal_config.bat[0]) / thermal_config.bat[1]
|
2020-01-18 03:03:22 +08:00
|
|
|
return dat
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
def setup_eon_fan():
|
|
|
|
global LEON
|
|
|
|
|
|
|
|
os.system("echo 2 > /sys/module/dwc3_msm/parameters/otg_switch")
|
|
|
|
|
|
|
|
bus = SMBus(7, force=True)
|
|
|
|
try:
|
|
|
|
bus.write_byte_data(0x21, 0x10, 0xf) # mask all interrupts
|
|
|
|
bus.write_byte_data(0x21, 0x03, 0x1) # set drive current and global interrupt disable
|
|
|
|
bus.write_byte_data(0x21, 0x02, 0x2) # needed?
|
|
|
|
bus.write_byte_data(0x21, 0x04, 0x4) # manual override source
|
|
|
|
except IOError:
|
|
|
|
print("LEON detected")
|
|
|
|
LEON = True
|
|
|
|
bus.close()
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
def set_eon_fan(val):
|
|
|
|
global LEON, last_eon_fan_val
|
|
|
|
|
|
|
|
if last_eon_fan_val is None or last_eon_fan_val != val:
|
|
|
|
bus = SMBus(7, force=True)
|
|
|
|
if LEON:
|
|
|
|
try:
|
|
|
|
i = [0x1, 0x3 | 0, 0x3 | 0x08, 0x3 | 0x10][val]
|
|
|
|
bus.write_i2c_block_data(0x3d, 0, [i])
|
|
|
|
except IOError:
|
|
|
|
# tusb320
|
|
|
|
if val == 0:
|
|
|
|
bus.write_i2c_block_data(0x67, 0xa, [0])
|
|
|
|
#bus.write_i2c_block_data(0x67, 0x45, [1<<2])
|
|
|
|
else:
|
|
|
|
#bus.write_i2c_block_data(0x67, 0x45, [0])
|
|
|
|
bus.write_i2c_block_data(0x67, 0xa, [0x20])
|
2020-05-31 11:14:58 +08:00
|
|
|
bus.write_i2c_block_data(0x67, 0x8, [(val - 1) << 6])
|
2020-01-18 03:03:22 +08:00
|
|
|
else:
|
|
|
|
bus.write_byte_data(0x21, 0x04, 0x2)
|
|
|
|
bus.write_byte_data(0x21, 0x03, (val*2)+1)
|
|
|
|
bus.write_byte_data(0x21, 0x04, 0x4)
|
|
|
|
bus.close()
|
|
|
|
last_eon_fan_val = val
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
# temp thresholds to control fan speed - high hysteresis
|
|
|
|
_TEMP_THRS_H = [50., 65., 80., 10000]
|
|
|
|
# temp thresholds to control fan speed - low hysteresis
|
|
|
|
_TEMP_THRS_L = [42.5, 57.5, 72.5, 10000]
|
|
|
|
# fan speed options
|
|
|
|
_FAN_SPEEDS = [0, 16384, 32768, 65535]
|
|
|
|
# max fan speed only allowed if battery is hot
|
2020-09-26 13:50:55 +08:00
|
|
|
_BAT_TEMP_THRESHOLD = 45.
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
|
2020-01-24 08:41:29 +08:00
|
|
|
def handle_fan_eon(max_cpu_temp, bat_temp, fan_speed, ignition):
|
2020-01-18 03:03:22 +08:00
|
|
|
new_speed_h = next(speed for speed, temp_h in zip(_FAN_SPEEDS, _TEMP_THRS_H) if temp_h > max_cpu_temp)
|
|
|
|
new_speed_l = next(speed for speed, temp_l in zip(_FAN_SPEEDS, _TEMP_THRS_L) if temp_l > max_cpu_temp)
|
|
|
|
|
|
|
|
if new_speed_h > fan_speed:
|
|
|
|
# update speed if using the high thresholds results in fan speed increment
|
|
|
|
fan_speed = new_speed_h
|
|
|
|
elif new_speed_l < fan_speed:
|
|
|
|
# update speed if using the low thresholds results in fan speed decrement
|
|
|
|
fan_speed = new_speed_l
|
|
|
|
|
2020-09-26 13:50:55 +08:00
|
|
|
if bat_temp < _BAT_TEMP_THRESHOLD:
|
2020-01-18 03:03:22 +08:00
|
|
|
# no max fan speed unless battery is hot
|
|
|
|
fan_speed = min(fan_speed, _FAN_SPEEDS[-2])
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
set_eon_fan(fan_speed // 16384)
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
return fan_speed
|
|
|
|
|
2020-01-24 08:41:29 +08:00
|
|
|
|
|
|
|
def handle_fan_uno(max_cpu_temp, bat_temp, fan_speed, ignition):
|
2020-02-12 06:03:40 +08:00
|
|
|
new_speed = int(interp(max_cpu_temp, [40.0, 80.0], [0, 80]))
|
2020-01-24 08:41:29 +08:00
|
|
|
|
|
|
|
if not ignition:
|
|
|
|
new_speed = min(30, new_speed)
|
|
|
|
|
|
|
|
return new_speed
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
|
2020-10-15 21:47:38 +08:00
|
|
|
def set_offroad_alert_if_changed(offroad_alert: str, show_alert: bool, extra_text: Optional[str]=None):
|
|
|
|
if prev_offroad_states.get(offroad_alert, None) == (show_alert, extra_text):
|
|
|
|
return
|
|
|
|
prev_offroad_states[offroad_alert] = (show_alert, extra_text)
|
|
|
|
set_offroad_alert(offroad_alert, show_alert, extra_text)
|
|
|
|
|
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
def thermald_thread():
|
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
pm = messaging.PubMaster(['deviceState'])
|
2020-12-30 14:32:03 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
pandaState_timeout = int(1000 * 2.5 * DT_TRML) # 2.5x the expected pandaState frequency
|
|
|
|
pandaState_sock = messaging.sub_sock('pandaState', timeout=pandaState_timeout)
|
2021-02-11 08:39:39 +08:00
|
|
|
location_sock = messaging.sub_sock('gpsLocationExternal')
|
2021-02-24 22:25:06 +08:00
|
|
|
managerState_sock = messaging.sub_sock('managerState', conflate=True)
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
fan_speed = 0
|
|
|
|
count = 0
|
|
|
|
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions = {
|
|
|
|
"ignition": False,
|
|
|
|
}
|
2020-10-21 21:14:29 +08:00
|
|
|
startup_conditions_prev = startup_conditions.copy()
|
2020-10-15 21:47:38 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
off_ts = None
|
|
|
|
started_ts = None
|
|
|
|
started_seen = False
|
|
|
|
thermal_status = ThermalStatus.green
|
|
|
|
usb_power = True
|
2020-08-12 07:23:57 +08:00
|
|
|
current_branch = get_git_branch()
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-02-04 03:28:38 +08:00
|
|
|
network_type = NetworkType.none
|
2020-03-08 03:14:04 +08:00
|
|
|
network_strength = NetworkStrength.unknown
|
2020-02-04 03:28:38 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
current_filter = FirstOrderFilter(0., CURRENT_TAU, DT_TRML)
|
2020-04-02 06:58:40 +08:00
|
|
|
cpu_temp_filter = FirstOrderFilter(0., CPU_TEMP_TAU, DT_TRML)
|
2021-02-17 13:39:32 +08:00
|
|
|
pandaState_prev = None
|
2020-01-18 03:03:22 +08:00
|
|
|
should_start_prev = False
|
2020-04-08 10:18:14 +08:00
|
|
|
handle_fan = None
|
|
|
|
is_uno = False
|
2021-02-24 22:25:06 +08:00
|
|
|
ui_running_prev = False
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
params = Params()
|
2020-12-30 14:32:03 +08:00
|
|
|
power_monitor = PowerMonitoring()
|
2020-04-21 05:15:01 +08:00
|
|
|
no_panda_cnt = 0
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2021-02-24 00:04:10 +08:00
|
|
|
thermal_config = HARDWARE.get_thermal_config()
|
2020-08-24 20:17:41 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
while 1:
|
2021-02-17 13:39:32 +08:00
|
|
|
pandaState = messaging.recv_sock(pandaState_sock, wait=True)
|
2020-08-24 20:17:41 +08:00
|
|
|
msg = read_thermal(thermal_config)
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
if pandaState is not None:
|
|
|
|
usb_power = pandaState.pandaState.usbPowerMode != log.PandaState.UsbPowerMode.client
|
2020-04-21 05:15:01 +08:00
|
|
|
|
|
|
|
# If we lose connection to the panda, wait 5 seconds before going offroad
|
2021-02-17 13:39:32 +08:00
|
|
|
if pandaState.pandaState.pandaType == log.PandaState.PandaType.unknown:
|
2020-04-21 05:15:01 +08:00
|
|
|
no_panda_cnt += 1
|
|
|
|
if no_panda_cnt > DISCONNECT_TIMEOUT / DT_TRML:
|
2020-10-15 21:47:38 +08:00
|
|
|
if startup_conditions["ignition"]:
|
2020-04-21 05:15:01 +08:00
|
|
|
cloudlog.error("Lost panda connection while onroad")
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions["ignition"] = False
|
2020-04-21 05:15:01 +08:00
|
|
|
else:
|
|
|
|
no_panda_cnt = 0
|
2021-02-17 13:39:32 +08:00
|
|
|
startup_conditions["ignition"] = pandaState.pandaState.ignitionLine or pandaState.pandaState.ignitionCan
|
2020-04-07 07:01:42 +08:00
|
|
|
|
2020-04-08 10:18:14 +08:00
|
|
|
# Setup fan handler on first connect to panda
|
2021-02-17 13:39:32 +08:00
|
|
|
if handle_fan is None and pandaState.pandaState.pandaType != log.PandaState.PandaType.unknown:
|
|
|
|
is_uno = pandaState.pandaState.pandaType == log.PandaState.PandaType.uno
|
2020-04-08 10:18:14 +08:00
|
|
|
|
2020-08-24 20:17:41 +08:00
|
|
|
if (not EON) or is_uno:
|
2020-04-08 10:18:14 +08:00
|
|
|
cloudlog.info("Setting up UNO fan handler")
|
|
|
|
handle_fan = handle_fan_uno
|
|
|
|
else:
|
|
|
|
cloudlog.info("Setting up EON fan handler")
|
|
|
|
setup_eon_fan()
|
|
|
|
handle_fan = handle_fan_eon
|
|
|
|
|
2020-04-07 07:01:42 +08:00
|
|
|
# Handle disconnect
|
2021-02-17 13:39:32 +08:00
|
|
|
if pandaState_prev is not None:
|
|
|
|
if pandaState.pandaState.pandaType == log.PandaState.PandaType.unknown and \
|
|
|
|
pandaState_prev.pandaState.pandaType != log.PandaState.PandaType.unknown:
|
2020-04-07 07:01:42 +08:00
|
|
|
params.panda_disconnect()
|
2021-02-17 13:39:32 +08:00
|
|
|
pandaState_prev = pandaState
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-02-18 02:38:39 +08:00
|
|
|
# get_network_type is an expensive call. update every 10s
|
|
|
|
if (count % int(10. / DT_TRML)) == 0:
|
2020-02-18 02:37:38 +08:00
|
|
|
try:
|
2020-08-26 20:57:17 +08:00
|
|
|
network_type = HARDWARE.get_network_type()
|
|
|
|
network_strength = HARDWARE.get_network_strength(network_type)
|
2020-02-22 07:02:24 +08:00
|
|
|
except Exception:
|
2020-03-08 03:14:04 +08:00
|
|
|
cloudlog.exception("Error getting network status")
|
2020-02-04 03:28:38 +08:00
|
|
|
|
2021-02-20 04:19:29 +08:00
|
|
|
msg.deviceState.freeSpacePercent = get_available_percent(default=100.0)
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.memoryUsagePercent = int(round(psutil.virtual_memory().percent))
|
|
|
|
msg.deviceState.cpuUsagePercent = int(round(psutil.cpu_percent()))
|
|
|
|
msg.deviceState.networkType = network_type
|
|
|
|
msg.deviceState.networkStrength = network_strength
|
|
|
|
msg.deviceState.batteryPercent = HARDWARE.get_battery_capacity()
|
|
|
|
msg.deviceState.batteryStatus = HARDWARE.get_battery_status()
|
|
|
|
msg.deviceState.batteryCurrent = HARDWARE.get_battery_current()
|
|
|
|
msg.deviceState.batteryVoltage = HARDWARE.get_battery_voltage()
|
|
|
|
msg.deviceState.usbOnline = HARDWARE.get_usb_present()
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# Fake battery levels on uno for frame
|
2020-08-24 20:17:41 +08:00
|
|
|
if (not EON) or is_uno:
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.batteryPercent = 100
|
|
|
|
msg.deviceState.batteryStatus = "Charging"
|
|
|
|
msg.deviceState.batteryTempC = 0
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
current_filter.update(msg.deviceState.batteryCurrent / 1e6)
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# TODO: add car battery voltage check
|
2021-02-17 13:39:32 +08:00
|
|
|
max_cpu_temp = cpu_temp_filter.update(max(msg.deviceState.cpuTempC))
|
|
|
|
max_comp_temp = max(max_cpu_temp, msg.deviceState.memoryTempC, max(msg.deviceState.gpuTempC))
|
|
|
|
bat_temp = msg.deviceState.batteryTempC
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-04-08 10:18:14 +08:00
|
|
|
if handle_fan is not None:
|
2020-10-15 21:47:38 +08:00
|
|
|
fan_speed = handle_fan(max_cpu_temp, bat_temp, fan_speed, startup_conditions["ignition"])
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.fanSpeedPercentDesired = fan_speed
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-06-13 06:03:32 +08:00
|
|
|
# If device is offroad we want to cool down before going onroad
|
|
|
|
# since going onroad increases load and can make temps go over 107
|
|
|
|
# We only do this if there is a relay that prevents the car from faulting
|
2020-09-01 17:25:04 +08:00
|
|
|
is_offroad_for_5_min = (started_ts is None) and ((not started_seen) or (off_ts is None) or (sec_since_boot() - off_ts > 60 * 5))
|
2020-11-04 11:56:25 +08:00
|
|
|
if max_cpu_temp > 107. or bat_temp >= 63. or (is_offroad_for_5_min and max_cpu_temp > 70.0):
|
2020-01-18 03:03:22 +08:00
|
|
|
# onroad not allowed
|
|
|
|
thermal_status = ThermalStatus.danger
|
2020-06-24 10:02:42 +08:00
|
|
|
elif max_comp_temp > 96.0 or bat_temp > 60.:
|
2020-01-18 03:03:22 +08:00
|
|
|
# hysteresis between onroad not allowed and engage not allowed
|
|
|
|
thermal_status = clip(thermal_status, ThermalStatus.red, ThermalStatus.danger)
|
2020-06-24 10:02:42 +08:00
|
|
|
elif max_cpu_temp > 94.0:
|
2020-01-18 03:03:22 +08:00
|
|
|
# hysteresis between engage not allowed and uploader not allowed
|
|
|
|
thermal_status = clip(thermal_status, ThermalStatus.yellow, ThermalStatus.red)
|
|
|
|
elif max_cpu_temp > 80.0:
|
|
|
|
# uploader not allowed
|
|
|
|
thermal_status = ThermalStatus.yellow
|
|
|
|
elif max_cpu_temp > 75.0:
|
|
|
|
# hysteresis between uploader not allowed and all good
|
|
|
|
thermal_status = clip(thermal_status, ThermalStatus.green, ThermalStatus.yellow)
|
2021-01-17 06:23:06 +08:00
|
|
|
else:
|
|
|
|
thermal_status = ThermalStatus.green # default to good condition
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# **** starting logic ****
|
|
|
|
|
|
|
|
# Check for last update time and display alerts if needed
|
2020-03-27 10:49:58 +08:00
|
|
|
now = datetime.datetime.utcnow()
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# show invalid date/time alert
|
2020-11-20 18:47:39 +08:00
|
|
|
startup_conditions["time_valid"] = (now.year > 2020) or (now.year == 2020 and now.month >= 10)
|
2020-10-15 21:47:38 +08:00
|
|
|
set_offroad_alert_if_changed("Offroad_InvalidTime", (not startup_conditions["time_valid"]))
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# Show update prompt
|
|
|
|
try:
|
|
|
|
last_update = datetime.datetime.fromisoformat(params.get("LastUpdateTime", encoding='utf8'))
|
|
|
|
except (TypeError, ValueError):
|
|
|
|
last_update = now
|
|
|
|
dt = now - last_update
|
|
|
|
|
2020-02-12 06:59:37 +08:00
|
|
|
update_failed_count = params.get("UpdateFailedCount")
|
|
|
|
update_failed_count = 0 if update_failed_count is None else int(update_failed_count)
|
2020-08-12 07:23:57 +08:00
|
|
|
last_update_exception = params.get("LastUpdateException", encoding='utf8')
|
2020-02-12 06:59:37 +08:00
|
|
|
|
2020-08-12 07:23:57 +08:00
|
|
|
if update_failed_count > 15 and last_update_exception is not None:
|
|
|
|
if current_branch in ["release2", "dashcam"]:
|
|
|
|
extra_text = "Ensure the software is correctly installed"
|
|
|
|
else:
|
|
|
|
extra_text = last_update_exception
|
|
|
|
|
2020-10-15 21:47:38 +08:00
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", True, extra_text=extra_text)
|
2020-08-12 07:23:57 +08:00
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_MAX and update_failed_count > 1:
|
2020-10-15 21:47:38 +08:00
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", True)
|
2020-01-18 03:03:22 +08:00
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_PROMPT:
|
2020-02-12 06:59:37 +08:00
|
|
|
remaining_time = str(max(DAYS_NO_CONNECTIVITY_MAX - dt.days, 0))
|
2020-10-15 21:47:38 +08:00
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", True, extra_text=f"{remaining_time} days.")
|
|
|
|
else:
|
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False)
|
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False)
|
|
|
|
|
2020-12-22 07:13:20 +08:00
|
|
|
startup_conditions["up_to_date"] = params.get("Offroad_ConnectivityNeeded") is None or params.get("DisableUpdates") == b"1"
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions["not_uninstalling"] = not params.get("DoUninstall") == b"1"
|
|
|
|
startup_conditions["accepted_terms"] = params.get("HasAcceptedTerms") == terms_version
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
panda_signature = params.get("PandaFirmware")
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions["fw_version_match"] = (panda_signature is None) or (panda_signature == FW_SIGNATURE) # don't show alert is no panda is connected (None)
|
|
|
|
set_offroad_alert_if_changed("Offroad_PandaFirmwareMismatch", (not startup_conditions["fw_version_match"]))
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# with 2% left, we killall, otherwise the phone will take a long time to boot
|
2021-02-20 04:19:29 +08:00
|
|
|
startup_conditions["free_space"] = msg.deviceState.freeSpacePercent > 2
|
2020-11-04 11:56:25 +08:00
|
|
|
startup_conditions["completed_training"] = params.get("CompletedTrainingVersion") == training_version or \
|
|
|
|
(current_branch in ['dashcam', 'dashcam-staging'])
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions["not_driver_view"] = not params.get("IsDriverViewEnabled") == b"1"
|
|
|
|
startup_conditions["not_taking_snapshot"] = not params.get("IsTakingSnapshot") == b"1"
|
2020-01-18 03:03:22 +08:00
|
|
|
# if any CPU gets above 107 or the battery gets above 63, kill all processes
|
|
|
|
# controls will warn with CPU above 95 or battery above 60
|
2020-10-15 21:47:38 +08:00
|
|
|
startup_conditions["device_temp_good"] = thermal_status < ThermalStatus.danger
|
|
|
|
set_offroad_alert_if_changed("Offroad_TemperatureTooHigh", (not startup_conditions["device_temp_good"]))
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
startup_conditions["hardware_supported"] = pandaState is not None and pandaState.pandaState.pandaType not in [log.PandaState.PandaType.whitePanda,
|
|
|
|
log.PandaState.PandaType.greyPanda]
|
|
|
|
set_offroad_alert_if_changed("Offroad_HardwareUnsupported", pandaState is not None and not startup_conditions["hardware_supported"])
|
2020-11-04 11:56:25 +08:00
|
|
|
|
2020-12-30 14:32:03 +08:00
|
|
|
# Handle offroad/onroad transition
|
2021-02-02 13:14:02 +08:00
|
|
|
should_start = all(startup_conditions.values())
|
2020-01-18 03:03:22 +08:00
|
|
|
if should_start:
|
|
|
|
if not should_start_prev:
|
|
|
|
params.delete("IsOffroad")
|
2021-02-26 13:10:10 +08:00
|
|
|
if TICI and DISABLE_LTE_ONROAD:
|
|
|
|
os.system("sudo systemctl stop --no-block lte")
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
off_ts = None
|
|
|
|
if started_ts is None:
|
|
|
|
started_ts = sec_since_boot()
|
|
|
|
started_seen = True
|
|
|
|
else:
|
2020-10-21 21:14:29 +08:00
|
|
|
if startup_conditions["ignition"] and (startup_conditions != startup_conditions_prev):
|
2020-10-15 21:47:38 +08:00
|
|
|
cloudlog.event("Startup blocked", startup_conditions=startup_conditions)
|
2020-12-30 14:32:03 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
if should_start_prev or (count == 0):
|
2020-10-21 05:04:18 +08:00
|
|
|
params.put("IsOffroad", "1")
|
2021-02-26 13:10:10 +08:00
|
|
|
if TICI and DISABLE_LTE_ONROAD:
|
|
|
|
os.system("sudo systemctl start --no-block lte")
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
started_ts = None
|
|
|
|
if off_ts is None:
|
|
|
|
off_ts = sec_since_boot()
|
|
|
|
|
2020-03-11 13:18:48 +08:00
|
|
|
# Offroad power monitoring
|
2021-02-17 13:39:32 +08:00
|
|
|
power_monitor.calculate(pandaState)
|
|
|
|
msg.deviceState.offroadPowerUsageUwh = power_monitor.get_power_used()
|
|
|
|
msg.deviceState.carBatteryCapacityUwh = max(0, power_monitor.get_car_battery_capacity())
|
2020-08-17 17:56:27 +08:00
|
|
|
|
|
|
|
# Check if we need to disable charging (handled by boardd)
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.chargingDisabled = power_monitor.should_disable_charging(pandaState, off_ts)
|
2020-08-17 17:56:27 +08:00
|
|
|
|
|
|
|
# Check if we need to shut down
|
2021-02-17 13:39:32 +08:00
|
|
|
if power_monitor.should_shutdown(pandaState, off_ts, started_seen, LEON):
|
2020-08-19 02:52:01 +08:00
|
|
|
cloudlog.info(f"shutting device down, offroad since {off_ts}")
|
|
|
|
# TODO: add function for blocking cloudlog instead of sleep
|
|
|
|
time.sleep(10)
|
2021-02-24 00:04:10 +08:00
|
|
|
HARDWARE.shutdown()
|
2020-03-11 13:18:48 +08:00
|
|
|
|
2021-02-24 22:25:06 +08:00
|
|
|
# If UI has crashed, set the brightness to reasonable non-zero value
|
|
|
|
manager_state = messaging.recv_one_or_none(managerState_sock)
|
|
|
|
if manager_state is not None:
|
|
|
|
ui_running = "ui" in (p.name for p in manager_state.managerState.processes if p.running)
|
|
|
|
if ui_running_prev and not ui_running:
|
|
|
|
HARDWARE.set_screen_brightness(20)
|
|
|
|
ui_running_prev = ui_running
|
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.chargingError = current_filter.x > 0. and msg.deviceState.batteryPercent < 90 # if current is positive, then battery is being discharged
|
|
|
|
msg.deviceState.started = started_ts is not None
|
|
|
|
msg.deviceState.startedMonoTime = int(1e9*(started_ts or 0))
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2021-02-17 13:39:32 +08:00
|
|
|
msg.deviceState.thermalStatus = thermal_status
|
|
|
|
pm.send("deviceState", msg)
|
2020-01-18 03:03:22 +08:00
|
|
|
|
2020-10-15 21:47:38 +08:00
|
|
|
set_offroad_alert_if_changed("Offroad_ChargeDisabled", (not usb_power))
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
should_start_prev = should_start
|
2020-10-21 21:14:29 +08:00
|
|
|
startup_conditions_prev = startup_conditions.copy()
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
# report to server once per minute
|
|
|
|
if (count % int(60. / DT_TRML)) == 0:
|
2020-12-30 14:32:03 +08:00
|
|
|
location = messaging.recv_sock(location_sock)
|
2020-01-18 03:03:22 +08:00
|
|
|
cloudlog.event("STATUS_PACKET",
|
2020-03-13 02:55:33 +08:00
|
|
|
count=count,
|
2021-02-17 13:39:32 +08:00
|
|
|
pandaState=(pandaState.to_dict() if pandaState else None),
|
2021-02-11 08:39:39 +08:00
|
|
|
location=(location.gpsLocationExternal.to_dict() if location else None),
|
2021-02-17 13:39:32 +08:00
|
|
|
deviceState=msg.to_dict())
|
2020-01-18 03:03:22 +08:00
|
|
|
|
|
|
|
count += 1
|
|
|
|
|
|
|
|
|
2020-02-21 08:22:25 +08:00
|
|
|
def main():
|
2020-01-18 03:03:22 +08:00
|
|
|
thermald_thread()
|
|
|
|
|
2020-03-13 02:55:33 +08:00
|
|
|
|
2020-01-18 03:03:22 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|