|  |  | @ -9,9 +9,11 @@ | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <bitset> |  |  |  | #include <bitset> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <cassert> |  |  |  | #include <cassert> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <cerrno> |  |  |  | #include <cerrno> | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | #include <chrono> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <cstdint> |  |  |  | #include <cstdint> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <cstdio> |  |  |  | #include <cstdio> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <cstdlib> |  |  |  | #include <cstdlib> | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | #include <future> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <thread> |  |  |  | #include <thread> | 
			
		
	
		
		
			
				
					
					|  |  |  | #include <unordered_map> |  |  |  | #include <unordered_map> | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -34,20 +36,19 @@ | 
			
		
	
		
		
			
				
					
					|  |  |  | #define CUTOFF_IL 200 |  |  |  | #define CUTOFF_IL 200 | 
			
		
	
		
		
			
				
					
					|  |  |  | #define SATURATE_IL 1600 |  |  |  | #define SATURATE_IL 1600 | 
			
		
	
		
		
			
				
					
					|  |  |  | #define NIBBLE_TO_HEX(n) ((n) < 10 ? (n) + '0' : ((n) - 10) + 'a') |  |  |  | #define NIBBLE_TO_HEX(n) ((n) < 10 ? (n) + '0' : ((n) - 10) + 'a') | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | using namespace std::chrono_literals; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | std::atomic<bool> safety_setter_thread_running(false); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | std::atomic<bool> ignition(false); |  |  |  | std::atomic<bool> ignition(false); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | ExitHandler do_exit; |  |  |  | ExitHandler do_exit; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | std::string get_time_str(const struct tm &time) { |  |  |  | std::string get_time_str(const struct tm &time) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   char s[30] = {'\0'}; |  |  |  |   char s[30] = {'\0'}; | 
			
		
	
		
		
			
				
					
					|  |  |  |   std::strftime(s, std::size(s), "%Y-%m-%d %H:%M:%S", &time); |  |  |  |   std::strftime(s, std::size(s), "%Y-%m-%d %H:%M:%S", &time); | 
			
		
	
		
		
			
				
					
					|  |  |  |   return s; |  |  |  |   return s; | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | void safety_setter_thread(Panda *panda) { |  |  |  | bool safety_setter_thread(Panda *panda) { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |   LOGD("Starting safety setter thread"); |  |  |  |   LOGD("Starting safety setter thread"); | 
			
		
	
		
		
			
				
					
					|  |  |  |   // diagnostic only is the default, needed for VIN query
 |  |  |  |   // diagnostic only is the default, needed for VIN query
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   panda->set_safety_model(cereal::CarParams::SafetyModel::ELM327); |  |  |  |   panda->set_safety_model(cereal::CarParams::SafetyModel::ELM327); | 
			
		
	
	
		
		
			
				
					|  |  | @ -57,8 +58,7 @@ void safety_setter_thread(Panda *panda) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   // switch to SILENT when CarVin param is read
 |  |  |  |   // switch to SILENT when CarVin param is read
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (true) { |  |  |  |   while (true) { | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (do_exit || !panda->connected) { |  |  |  |     if (do_exit || !panda->connected) { | 
			
		
	
		
		
			
				
					
					|  |  |  |       safety_setter_thread_running = false; |  |  |  |       return false; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |       return; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     }; |  |  |  |     }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     std::string value_vin = p.get("CarVin"); |  |  |  |     std::string value_vin = p.get("CarVin"); | 
			
		
	
	
		
		
			
				
					|  |  | @ -78,8 +78,7 @@ void safety_setter_thread(Panda *panda) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   LOGW("waiting for params to set safety model"); |  |  |  |   LOGW("waiting for params to set safety model"); | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (true) { |  |  |  |   while (true) { | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (do_exit || !panda->connected) { |  |  |  |     if (do_exit || !panda->connected) { | 
			
		
	
		
		
			
				
					
					|  |  |  |       safety_setter_thread_running = false; |  |  |  |       return false; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |       return; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     }; |  |  |  |     }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (p.getBool("ControlsReady")) { |  |  |  |     if (p.getBool("ControlsReady")) { | 
			
		
	
	
		
		
			
				
					|  |  | @ -101,8 +100,7 @@ void safety_setter_thread(Panda *panda) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   LOGW("setting safety model: %d with param %d", (int)safety_model, safety_param); |  |  |  |   LOGW("setting safety model: %d with param %d", (int)safety_model, safety_param); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   panda->set_safety_model(safety_model, safety_param); |  |  |  |   panda->set_safety_model(safety_model, safety_param); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |   return true; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   safety_setter_thread_running = false; |  |  |  |  | 
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -349,6 +347,7 @@ void send_peripheral_state(PubMaster *pm, Panda *panda) { | 
			
		
	
		
		
			
				
					
					|  |  |  | void panda_state_thread(PubMaster *pm, Panda * peripheral_panda, Panda *panda, bool spoofing_started) { |  |  |  | void panda_state_thread(PubMaster *pm, Panda * peripheral_panda, Panda *panda, bool spoofing_started) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   Params params; |  |  |  |   Params params; | 
			
		
	
		
		
			
				
					
					|  |  |  |   bool ignition_last = false; |  |  |  |   bool ignition_last = false; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   std::future<bool> safety_future; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   LOGD("start panda state thread"); |  |  |  |   LOGD("start panda state thread"); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | @ -360,10 +359,8 @@ void panda_state_thread(PubMaster *pm, Panda * peripheral_panda, Panda *panda, b | 
			
		
	
		
		
			
				
					
					|  |  |  |     // clear VIN, CarParams, and set new safety on car start
 |  |  |  |     // clear VIN, CarParams, and set new safety on car start
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (ignition && !ignition_last) { |  |  |  |     if (ignition && !ignition_last) { | 
			
		
	
		
		
			
				
					
					|  |  |  |       params.clearAll(CLEAR_ON_IGNITION_ON); |  |  |  |       params.clearAll(CLEAR_ON_IGNITION_ON); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |       if (!safety_future.valid() || safety_future.wait_for(0ms) == std::future_status::ready) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |       if (!safety_setter_thread_running) { |  |  |  |         safety_future = std::async(std::launch::async, safety_setter_thread, panda); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |         safety_setter_thread_running = true; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         std::thread(safety_setter_thread, panda).detach(); |  |  |  |  | 
			
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |       } else { |  |  |  |       } else { | 
			
		
	
		
		
			
				
					
					|  |  |  |         LOGW("Safety setter thread already running"); |  |  |  |         LOGW("Safety setter thread already running"); | 
			
		
	
		
		
			
				
					
					|  |  |  |       } |  |  |  |       } | 
			
		
	
	
		
		
			
				
					|  |  | 
 |