|
|
@ -3,6 +3,7 @@ import datetime |
|
|
|
import os |
|
|
|
import os |
|
|
|
import time |
|
|
|
import time |
|
|
|
from collections import namedtuple |
|
|
|
from collections import namedtuple |
|
|
|
|
|
|
|
from typing import Dict, Optional, Tuple |
|
|
|
|
|
|
|
|
|
|
|
import psutil |
|
|
|
import psutil |
|
|
|
from smbus2 import SMBus |
|
|
|
from smbus2 import SMBus |
|
|
@ -39,6 +40,8 @@ DAYS_NO_CONNECTIVITY_MAX = 7 # do not allow to engage after a week without inte |
|
|
|
DAYS_NO_CONNECTIVITY_PROMPT = 4 # send an offroad prompt after 4 days with no internet |
|
|
|
DAYS_NO_CONNECTIVITY_PROMPT = 4 # send an offroad prompt after 4 days with no internet |
|
|
|
DISCONNECT_TIMEOUT = 5. # wait 5 seconds before going offroad after disconnect so you get an alert |
|
|
|
DISCONNECT_TIMEOUT = 5. # wait 5 seconds before going offroad after disconnect so you get an alert |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
prev_offroad_states: Dict[str, Tuple[bool, Optional[str]]] = {} |
|
|
|
|
|
|
|
|
|
|
|
LEON = False |
|
|
|
LEON = False |
|
|
|
last_eon_fan_val = None |
|
|
|
last_eon_fan_val = None |
|
|
|
|
|
|
|
|
|
|
@ -156,6 +159,13 @@ def handle_fan_uno(max_cpu_temp, bat_temp, fan_speed, ignition): |
|
|
|
return new_speed |
|
|
|
return new_speed |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def thermald_thread(): |
|
|
|
def thermald_thread(): |
|
|
|
health_timeout = int(1000 * 2.5 * DT_TRML) # 2.5x the expected health frequency |
|
|
|
health_timeout = int(1000 * 2.5 * DT_TRML) # 2.5x the expected health frequency |
|
|
|
|
|
|
|
|
|
|
@ -164,17 +174,18 @@ def thermald_thread(): |
|
|
|
health_sock = messaging.sub_sock('health', timeout=health_timeout) |
|
|
|
health_sock = messaging.sub_sock('health', timeout=health_timeout) |
|
|
|
location_sock = messaging.sub_sock('gpsLocation') |
|
|
|
location_sock = messaging.sub_sock('gpsLocation') |
|
|
|
|
|
|
|
|
|
|
|
ignition = False |
|
|
|
|
|
|
|
fan_speed = 0 |
|
|
|
fan_speed = 0 |
|
|
|
count = 0 |
|
|
|
count = 0 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
startup_conditions = { |
|
|
|
|
|
|
|
"ignition": False, |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
off_ts = None |
|
|
|
off_ts = None |
|
|
|
started_ts = None |
|
|
|
started_ts = None |
|
|
|
started_seen = False |
|
|
|
started_seen = False |
|
|
|
thermal_status = ThermalStatus.green |
|
|
|
thermal_status = ThermalStatus.green |
|
|
|
thermal_status_prev = ThermalStatus.green |
|
|
|
|
|
|
|
usb_power = True |
|
|
|
usb_power = True |
|
|
|
usb_power_prev = True |
|
|
|
|
|
|
|
current_branch = get_git_branch() |
|
|
|
current_branch = get_git_branch() |
|
|
|
|
|
|
|
|
|
|
|
network_type = NetworkType.none |
|
|
|
network_type = NetworkType.none |
|
|
@ -183,9 +194,6 @@ def thermald_thread(): |
|
|
|
current_filter = FirstOrderFilter(0., CURRENT_TAU, DT_TRML) |
|
|
|
current_filter = FirstOrderFilter(0., CURRENT_TAU, DT_TRML) |
|
|
|
cpu_temp_filter = FirstOrderFilter(0., CPU_TEMP_TAU, DT_TRML) |
|
|
|
cpu_temp_filter = FirstOrderFilter(0., CPU_TEMP_TAU, DT_TRML) |
|
|
|
health_prev = None |
|
|
|
health_prev = None |
|
|
|
fw_version_match_prev = True |
|
|
|
|
|
|
|
current_update_alert = None |
|
|
|
|
|
|
|
time_valid_prev = True |
|
|
|
|
|
|
|
should_start_prev = False |
|
|
|
should_start_prev = False |
|
|
|
handle_fan = None |
|
|
|
handle_fan = None |
|
|
|
is_uno = False |
|
|
|
is_uno = False |
|
|
@ -210,12 +218,12 @@ def thermald_thread(): |
|
|
|
if health.health.hwType == log.HealthData.HwType.unknown: |
|
|
|
if health.health.hwType == log.HealthData.HwType.unknown: |
|
|
|
no_panda_cnt += 1 |
|
|
|
no_panda_cnt += 1 |
|
|
|
if no_panda_cnt > DISCONNECT_TIMEOUT / DT_TRML: |
|
|
|
if no_panda_cnt > DISCONNECT_TIMEOUT / DT_TRML: |
|
|
|
if ignition: |
|
|
|
if startup_conditions["ignition"]: |
|
|
|
cloudlog.error("Lost panda connection while onroad") |
|
|
|
cloudlog.error("Lost panda connection while onroad") |
|
|
|
ignition = False |
|
|
|
startup_conditions["ignition"] = False |
|
|
|
else: |
|
|
|
else: |
|
|
|
no_panda_cnt = 0 |
|
|
|
no_panda_cnt = 0 |
|
|
|
ignition = health.health.ignitionLine or health.health.ignitionCan |
|
|
|
startup_conditions["ignition"] = health.health.ignitionLine or health.health.ignitionCan |
|
|
|
|
|
|
|
|
|
|
|
# Setup fan handler on first connect to panda |
|
|
|
# Setup fan handler on first connect to panda |
|
|
|
if handle_fan is None and health.health.hwType != log.HealthData.HwType.unknown: |
|
|
|
if handle_fan is None and health.health.hwType != log.HealthData.HwType.unknown: |
|
|
@ -270,7 +278,7 @@ def thermald_thread(): |
|
|
|
bat_temp = msg.thermal.bat |
|
|
|
bat_temp = msg.thermal.bat |
|
|
|
|
|
|
|
|
|
|
|
if handle_fan is not None: |
|
|
|
if handle_fan is not None: |
|
|
|
fan_speed = handle_fan(max_cpu_temp, bat_temp, fan_speed, ignition) |
|
|
|
fan_speed = handle_fan(max_cpu_temp, bat_temp, fan_speed, startup_conditions["ignition"]) |
|
|
|
msg.thermal.fanSpeed = fan_speed |
|
|
|
msg.thermal.fanSpeed = fan_speed |
|
|
|
|
|
|
|
|
|
|
|
# If device is offroad we want to cool down before going onroad |
|
|
|
# If device is offroad we want to cool down before going onroad |
|
|
@ -302,12 +310,8 @@ def thermald_thread(): |
|
|
|
now = datetime.datetime.utcnow() |
|
|
|
now = datetime.datetime.utcnow() |
|
|
|
|
|
|
|
|
|
|
|
# show invalid date/time alert |
|
|
|
# show invalid date/time alert |
|
|
|
time_valid = now.year >= 2019 |
|
|
|
startup_conditions["time_valid"] = now.year >= 2019 |
|
|
|
if time_valid and not time_valid_prev: |
|
|
|
set_offroad_alert_if_changed("Offroad_InvalidTime", (not startup_conditions["time_valid"])) |
|
|
|
set_offroad_alert("Offroad_InvalidTime", False) |
|
|
|
|
|
|
|
if not time_valid and time_valid_prev: |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_InvalidTime", True) |
|
|
|
|
|
|
|
time_valid_prev = time_valid |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Show update prompt |
|
|
|
# Show update prompt |
|
|
|
try: |
|
|
|
try: |
|
|
@ -326,72 +330,41 @@ def thermald_thread(): |
|
|
|
else: |
|
|
|
else: |
|
|
|
extra_text = last_update_exception |
|
|
|
extra_text = last_update_exception |
|
|
|
|
|
|
|
|
|
|
|
if current_update_alert != "update" + extra_text: |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False) |
|
|
|
current_update_alert = "update" + extra_text |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeeded", False) |
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", True, extra_text=extra_text) |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_UpdateFailed", True, extra_text=extra_text) |
|
|
|
|
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_MAX and update_failed_count > 1: |
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_MAX and update_failed_count > 1: |
|
|
|
if current_update_alert != "expired": |
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False) |
|
|
|
current_update_alert = "expired" |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
set_offroad_alert("Offroad_UpdateFailed", False) |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", True) |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_ConnectivityNeeded", True) |
|
|
|
|
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_PROMPT: |
|
|
|
elif dt.days > DAYS_NO_CONNECTIVITY_PROMPT: |
|
|
|
remaining_time = str(max(DAYS_NO_CONNECTIVITY_MAX - dt.days, 0)) |
|
|
|
remaining_time = str(max(DAYS_NO_CONNECTIVITY_MAX - dt.days, 0)) |
|
|
|
if current_update_alert != "prompt" + remaining_time: |
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False) |
|
|
|
current_update_alert = "prompt" + remaining_time |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False) |
|
|
|
set_offroad_alert("Offroad_UpdateFailed", False) |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", True, extra_text=f"{remaining_time} days.") |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeeded", False) |
|
|
|
else: |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeededPrompt", True, extra_text=f"{remaining_time} days.") |
|
|
|
set_offroad_alert_if_changed("Offroad_UpdateFailed", False) |
|
|
|
elif current_update_alert is not None: |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeeded", False) |
|
|
|
current_update_alert = None |
|
|
|
set_offroad_alert_if_changed("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
set_offroad_alert("Offroad_UpdateFailed", False) |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_ConnectivityNeeded", False) |
|
|
|
startup_conditions["not_uninstalling"] = not params.get("DoUninstall") == b"1" |
|
|
|
set_offroad_alert("Offroad_ConnectivityNeededPrompt", False) |
|
|
|
startup_conditions["accepted_terms"] = params.get("HasAcceptedTerms") == terms_version |
|
|
|
|
|
|
|
|
|
|
|
do_uninstall = params.get("DoUninstall") == b"1" |
|
|
|
|
|
|
|
accepted_terms = params.get("HasAcceptedTerms") == terms_version |
|
|
|
|
|
|
|
completed_training = params.get("CompletedTrainingVersion") == training_version |
|
|
|
completed_training = params.get("CompletedTrainingVersion") == training_version |
|
|
|
|
|
|
|
|
|
|
|
panda_signature = params.get("PandaFirmware") |
|
|
|
panda_signature = params.get("PandaFirmware") |
|
|
|
fw_version_match = (panda_signature is None) or (panda_signature == FW_SIGNATURE) # don't show alert is no panda is connected (None) |
|
|
|
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"])) |
|
|
|
should_start = ignition |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# with 2% left, we killall, otherwise the phone will take a long time to boot |
|
|
|
# with 2% left, we killall, otherwise the phone will take a long time to boot |
|
|
|
should_start = should_start and msg.thermal.freeSpace > 0.02 |
|
|
|
startup_conditions["free_space"] = msg.thermal.freeSpace > 0.02 |
|
|
|
|
|
|
|
startup_conditions["completed_training"] = completed_training or (current_branch in ['dashcam', 'dashcam-staging']) |
|
|
|
# confirm we have completed training and aren't uninstalling |
|
|
|
startup_conditions["not_driver_view"] = not params.get("IsDriverViewEnabled") == b"1" |
|
|
|
should_start = should_start and accepted_terms and (not do_uninstall) and \ |
|
|
|
startup_conditions["not_taking_snapshot"] = not params.get("IsTakingSnapshot") == b"1" |
|
|
|
(completed_training or current_branch in ['dashcam', 'dashcam-staging']) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# check for firmware mismatch |
|
|
|
|
|
|
|
should_start = should_start and fw_version_match |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# check if system time is valid |
|
|
|
|
|
|
|
should_start = should_start and time_valid |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# don't start while taking snapshot |
|
|
|
|
|
|
|
if not should_start_prev: |
|
|
|
|
|
|
|
is_viewing_driver = params.get("IsDriverViewEnabled") == b"1" |
|
|
|
|
|
|
|
is_taking_snapshot = params.get("IsTakingSnapshot") == b"1" |
|
|
|
|
|
|
|
should_start = should_start and (not is_taking_snapshot) and (not is_viewing_driver) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if fw_version_match and not fw_version_match_prev: |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_PandaFirmwareMismatch", False) |
|
|
|
|
|
|
|
if not fw_version_match and fw_version_match_prev: |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_PandaFirmwareMismatch", True) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# if any CPU gets above 107 or the battery gets above 63, kill all processes |
|
|
|
# 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 |
|
|
|
# controls will warn with CPU above 95 or battery above 60 |
|
|
|
if thermal_status >= ThermalStatus.danger: |
|
|
|
startup_conditions["device_temp_good"] = thermal_status < ThermalStatus.danger |
|
|
|
should_start = False |
|
|
|
set_offroad_alert_if_changed("Offroad_TemperatureTooHigh", (not startup_conditions["device_temp_good"])) |
|
|
|
if thermal_status_prev < ThermalStatus.danger: |
|
|
|
should_start = all(startup_conditions.values()) |
|
|
|
set_offroad_alert("Offroad_TemperatureTooHigh", True) |
|
|
|
|
|
|
|
else: |
|
|
|
|
|
|
|
if thermal_status_prev >= ThermalStatus.danger: |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_TemperatureTooHigh", False) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if should_start: |
|
|
|
if should_start: |
|
|
|
if not should_start_prev: |
|
|
|
if not should_start_prev: |
|
|
@ -403,6 +376,8 @@ def thermald_thread(): |
|
|
|
started_seen = True |
|
|
|
started_seen = True |
|
|
|
os.system('echo performance > /sys/class/devfreq/soc:qcom,cpubw/governor') |
|
|
|
os.system('echo performance > /sys/class/devfreq/soc:qcom,cpubw/governor') |
|
|
|
else: |
|
|
|
else: |
|
|
|
|
|
|
|
if startup_conditions["ignition"]: |
|
|
|
|
|
|
|
cloudlog.event("Startup blocked", startup_conditions=startup_conditions) |
|
|
|
if should_start_prev or (count == 0): |
|
|
|
if should_start_prev or (count == 0): |
|
|
|
put_nonblocking("IsOffroad", "1") |
|
|
|
put_nonblocking("IsOffroad", "1") |
|
|
|
|
|
|
|
|
|
|
@ -433,14 +408,8 @@ def thermald_thread(): |
|
|
|
msg.thermal.thermalStatus = thermal_status |
|
|
|
msg.thermal.thermalStatus = thermal_status |
|
|
|
thermal_sock.send(msg.to_bytes()) |
|
|
|
thermal_sock.send(msg.to_bytes()) |
|
|
|
|
|
|
|
|
|
|
|
if usb_power_prev and not usb_power: |
|
|
|
set_offroad_alert_if_changed("Offroad_ChargeDisabled", (not usb_power)) |
|
|
|
set_offroad_alert("Offroad_ChargeDisabled", True) |
|
|
|
|
|
|
|
elif usb_power and not usb_power_prev: |
|
|
|
|
|
|
|
set_offroad_alert("Offroad_ChargeDisabled", False) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
thermal_status_prev = thermal_status |
|
|
|
|
|
|
|
usb_power_prev = usb_power |
|
|
|
|
|
|
|
fw_version_match_prev = fw_version_match |
|
|
|
|
|
|
|
should_start_prev = should_start |
|
|
|
should_start_prev = should_start |
|
|
|
|
|
|
|
|
|
|
|
# report to server once per minute |
|
|
|
# report to server once per minute |
|
|
|