#include "AP_Networking_Config.h" #if AP_NETWORKING_BACKEND_PPP #include "AP_Networking_PPP.h" #include #include #include #include #include #include #include extern const AP_HAL::HAL& hal; #if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS /* uint32_t timestamp in smallest available units */ uint32_t sys_jiffies(void) { return AP_HAL::micros(); } #endif #if LWIP_TCPIP_CORE_LOCKING #define LWIP_TCPIP_LOCK() sys_lock_tcpip_core() #define LWIP_TCPIP_UNLOCK() sys_unlock_tcpip_core() #else #define LWIP_TCPIP_LOCK() #define LWIP_TCPIP_UNLOCK() #endif /* output some data to the uart */ uint32_t AP_Networking_PPP::ppp_output_cb(ppp_pcb *pcb, const void *data, uint32_t len, void *ctx) { auto &driver = *(AP_Networking_PPP *)ctx; LWIP_UNUSED_ARG(pcb); uint32_t remaining = len; uint8_t *ptr = const_cast((const uint8_t *)data); while (remaining > 0) { auto n = driver.uart->write(ptr, remaining); if (n > 0) { remaining -= n; ptr += n; } } return len; } /* callback on link status change */ void AP_Networking_PPP::ppp_status_callback(struct ppp_pcb_s *pcb, int code, void *ctx) { auto &driver = *(AP_Networking_PPP *)ctx; struct netif *pppif = ppp_netif(pcb); const char *msg = nullptr; switch (code) { case PPPERR_NONE: { // got addresses driver.activeSettings.ip = ntohl(netif_ip4_addr(pppif)->addr); driver.activeSettings.gw = ntohl(netif_ip4_gw(pppif)->addr); driver.activeSettings.nm = ntohl(netif_ip4_netmask(pppif)->addr); driver.activeSettings.last_change_ms = AP_HAL::millis(); break; } case PPPERR_PARAM: { /* Invalid parameter. */ msg = "PPPERR_PARAM"; break; } case PPPERR_OPEN: { /* Unable to open PPP session. */ msg = "PPPERR_OPEN"; break; } case PPPERR_DEVICE: { /* Invalid I/O device for PPP. */ msg = "PPPERR_DEVICE"; break; } case PPPERR_ALLOC: { /* Unable to allocate resources. */ msg = "PPPERR_ALLOC"; break; } case PPPERR_USER: { /* User interrupt. */ msg = "PPPERR_USER"; break; } case PPPERR_CONNECT: { /* Connection lost. */ msg = "PPPERR_CONNECT"; break; } case PPPERR_AUTHFAIL: { /* Failed authentication challenge. */ msg = "PPPERR_AUTHFAIL"; break; } case PPPERR_PROTOCOL: { /* Failed to meet protocol. */ msg = "PPPERR_PROTOCOL"; break; } case PPPERR_PEERDEAD: { /* Connection timeout */ msg = "PPPERR_PEERDEAD"; break; } case PPPERR_IDLETIMEOUT: { /* Idle Timeout */ msg = "PPPERR_IDLETIMEOUT"; break; } case PPPERR_CONNECTTIME: { /* Max connect time reached */ msg = "PPPERR_CONNECTTIME"; break; } case PPPERR_LOOPBACK: { /* Loopback detected */ msg = "PPPERR_LOOPBACK"; break; } } if (msg != nullptr) { GCS_SEND_TEXT(MAV_SEVERITY_INFO, "PPP: %s", msg); } } /* initialise PPP network backend using LWIP */ bool AP_Networking_PPP::init() { uart = AP::serialmanager().find_serial(AP_SerialManager::SerialProtocol_PPP, 0); if (uart == nullptr) { return false; } uart->set_unbuffered_writes(true); pppif = new netif; if (pppif == nullptr) { return false; } // initialise TCP/IP thread LWIP_TCPIP_LOCK(); tcpip_init(NULL, NULL); LWIP_TCPIP_UNLOCK(); hal.scheduler->delay(100); // create ppp connection LWIP_TCPIP_LOCK(); ppp = pppos_create(pppif, ppp_output_cb, ppp_status_callback, this); if (ppp == nullptr) { GCS_SEND_TEXT(MAV_SEVERITY_INFO, "PPP: failed to create link"); return false; } LWIP_TCPIP_UNLOCK(); GCS_SEND_TEXT(MAV_SEVERITY_INFO, "PPP: started"); hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&AP_Networking_PPP::ppp_loop, void), "ppp", 2048, AP_HAL::Scheduler::PRIORITY_UART, 0); return true; } /* main loop for PPP */ void AP_Networking_PPP::ppp_loop(void) { while (!hal.scheduler->is_system_initialized()) { hal.scheduler->delay_microseconds(1000); } // connect and set as default route LWIP_TCPIP_LOCK(); ppp_connect(ppp, 0); netif_set_default(pppif); LWIP_TCPIP_UNLOCK(); // ensure this thread owns the uart uart->begin(0); while (true) { uint8_t buf[1024]; auto n = uart->read(buf, sizeof(buf)); if (n > 0) { LWIP_TCPIP_LOCK(); pppos_input(ppp, buf, n); LWIP_TCPIP_UNLOCK(); hal.scheduler->delay_microseconds(100); } else { hal.scheduler->delay_microseconds(1000); } } } #endif // AP_NETWORKING_BACKEND_PPP