|  |  | @ -162,20 +162,6 @@ Panda *usb_connect() { | 
			
		
	
		
		
			
				
					
					|  |  |  |   return panda.release(); |  |  |  |   return panda.release(); | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | // must be called before threads or with mutex
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | static Panda *usb_retry_connect() { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   LOGW("attempting to connect"); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (!do_exit) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     Panda *panda = usb_connect(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     if (panda) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |       LOGW("connected to board"); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |       return panda; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     util::sleep_for(100); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   }; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   return nullptr; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | void can_recv(Panda *panda, PubMaster &pm) { |  |  |  | void can_recv(Panda *panda, PubMaster &pm) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   kj::Array<capnp::word> can_data; |  |  |  |   kj::Array<capnp::word> can_data; | 
			
		
	
		
		
			
				
					
					|  |  |  |   panda->can_receive(can_data); |  |  |  |   panda->can_receive(can_data); | 
			
		
	
	
		
		
			
				
					|  |  | @ -248,24 +234,19 @@ void can_recv_thread(Panda *panda) { | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  | } |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  | void panda_state_thread(Panda *&panda, bool spoofing_started) { |  |  |  | void send_empty_panda_state(PubMaster *pm) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   LOGD("start panda state thread"); |  |  |  |   MessageBuilder msg; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |   PubMaster pm({"pandaState"}); |  |  |  |   auto pandaState  = msg.initEvent().initPandaState(); | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   pandaState.setPandaType(cereal::PandaState::PandaType::UNKNOWN); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   pm->send("pandaState", msg); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | void panda_state_thread(PubMaster *pm, Panda *panda, bool spoofing_started) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   LOGD("start panda state thread"); | 
			
		
	
		
		
			
				
					
					|  |  |  |   uint32_t no_ignition_cnt = 0; |  |  |  |   uint32_t no_ignition_cnt = 0; | 
			
		
	
		
		
			
				
					
					|  |  |  |   bool ignition_last = false; |  |  |  |   bool ignition_last = false; | 
			
		
	
		
		
			
				
					
					|  |  |  |   Params params = Params(); |  |  |  |   Params params = Params(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   // Broadcast empty pandaState message when panda is not yet connected
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (!do_exit && !panda) { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     MessageBuilder msg; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     auto pandaState  = msg.initEvent().initPandaState(); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     pandaState.setPandaType(cereal::PandaState::PandaType::UNKNOWN); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     pm.send("pandaState", msg); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     util::sleep_for(500); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   // run at 2hz
 |  |  |  |   // run at 2hz
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (!do_exit && panda->connected) { |  |  |  |   while (!do_exit && panda->connected) { | 
			
		
	
		
		
			
				
					
					|  |  |  |     health_t pandaState = panda->get_state(); |  |  |  |     health_t pandaState = panda->get_state(); | 
			
		
	
	
		
		
			
				
					|  |  | @ -386,7 +367,7 @@ void panda_state_thread(Panda *&panda, bool spoofing_started) { | 
			
		
	
		
		
			
				
					
					|  |  |  |         i++; |  |  |  |         i++; | 
			
		
	
		
		
			
				
					
					|  |  |  |       } |  |  |  |       } | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  |     pm.send("pandaState", msg); |  |  |  |     pm->send("pandaState", msg); | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |     panda->send_heartbeat(); |  |  |  |     panda->send_heartbeat(); | 
			
		
	
		
		
			
				
					
					|  |  |  |     util::sleep_for(500); |  |  |  |     util::sleep_for(500); | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
	
		
		
			
				
					|  |  | @ -556,19 +537,27 @@ int main() { | 
			
		
	
		
		
			
				
					
					|  |  |  |   err = set_core_affinity(Hardware::TICI() ? 4 : 3); |  |  |  |   err = set_core_affinity(Hardware::TICI() ? 4 : 3); | 
			
		
	
		
		
			
				
					
					|  |  |  |   LOG("set affinity returns %d", err); |  |  |  |   LOG("set affinity returns %d", err); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   LOGW("attempting to connect"); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |   PubMaster pm({"pandaState"}); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   while (!do_exit) { |  |  |  |   while (!do_exit) { | 
			
		
	
		
		
			
				
					
					|  |  |  |     Panda *panda = nullptr; |  |  |  |     Panda *panda = usb_connect(); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     std::vector<std::thread> threads; |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     threads.emplace_back(panda_state_thread, std::ref(panda), getenv("STARTED") != nullptr); |  |  |  |     // Send empty pandaState and try again
 | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     if (panda == nullptr) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       send_empty_panda_state(&pm); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       util::sleep_for(500); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       continue; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     // connect to the board
 |  |  |  |     LOGW("connected to board"); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     panda = usb_retry_connect(); |  |  |  | 
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     if (panda != nullptr) { |  |  |  |     std::vector<std::thread> threads; | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     threads.emplace_back(panda_state_thread, &pm, panda, getenv("STARTED") != nullptr); | 
			
		
	
		
		
			
				
					
					|  |  |  |     threads.emplace_back(can_send_thread, panda, getenv("FAKESEND") != nullptr); |  |  |  |     threads.emplace_back(can_send_thread, panda, getenv("FAKESEND") != nullptr); | 
			
		
	
		
		
			
				
					
					|  |  |  |     threads.emplace_back(can_recv_thread, panda); |  |  |  |     threads.emplace_back(can_recv_thread, panda); | 
			
		
	
		
		
			
				
					
					|  |  |  |     threads.emplace_back(hardware_control_thread, panda); |  |  |  |     threads.emplace_back(hardware_control_thread, panda); | 
			
		
	
		
		
			
				
					
					|  |  |  |     threads.emplace_back(pigeon_thread, panda); |  |  |  |     threads.emplace_back(pigeon_thread, panda); | 
			
		
	
		
		
			
				
					
					|  |  |  |     } |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     for (auto &t : threads) t.join(); |  |  |  |     for (auto &t : threads) t.join(); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | 
 |