385 lines
11 KiB
C++
385 lines
11 KiB
C++
|
|
#include "AP_Networking_Config.h"
|
|
|
|
#if AP_NETWORKING_BACKEND_CHIBIOS
|
|
|
|
#include "AP_Networking_ChibiOS.h"
|
|
#include <GCS_MAVLink/GCS.h>
|
|
|
|
#include <lwip/udp.h>
|
|
#include <lwip/ip_addr.h>
|
|
#include <lwip/tcpip.h>
|
|
#include <lwip/netifapi.h>
|
|
#if LWIP_DHCP
|
|
#include <lwip/dhcp.h>
|
|
#endif
|
|
#include <hal.h>
|
|
#include "../../modules/ChibiOS/os/various/evtimer.h"
|
|
#include <AP_HAL_ChibiOS/hwdef/common/stm32_util.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
#ifndef STM32_ETH_BUFFERS_EXTERN
|
|
#error "Must use external ethernet buffers"
|
|
#endif
|
|
|
|
/*
|
|
these are referenced as globals inside lwip
|
|
*/
|
|
stm32_eth_rx_descriptor_t *__eth_rd;
|
|
stm32_eth_tx_descriptor_t *__eth_td;
|
|
uint32_t *__eth_rb[STM32_MAC_RECEIVE_BUFFERS];
|
|
uint32_t *__eth_tb[STM32_MAC_TRANSMIT_BUFFERS];
|
|
|
|
#define LWIP_SEND_TIMEOUT_MS 50
|
|
#define LWIP_NETIF_MTU 1500
|
|
#define LWIP_LINK_POLL_INTERVAL TIME_S2I(5)
|
|
|
|
#define PERIODIC_TIMER_ID 1
|
|
#define FRAME_RECEIVED_ID 2
|
|
|
|
/*
|
|
allocate buffers for LWIP
|
|
*/
|
|
bool AP_Networking_ChibiOS::allocate_buffers()
|
|
{
|
|
#define AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE ((((STM32_MAC_BUFFERS_SIZE - 1) | 3) + 1) / 4) // typically == 381
|
|
// check total size of buffers
|
|
const uint32_t total_size = sizeof(stm32_eth_rx_descriptor_t)*STM32_MAC_RECEIVE_BUFFERS +
|
|
sizeof(stm32_eth_tx_descriptor_t)*STM32_MAC_TRANSMIT_BUFFERS +
|
|
sizeof(uint32_t)*STM32_MAC_RECEIVE_BUFFERS*AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE +
|
|
sizeof(uint32_t)*STM32_MAC_TRANSMIT_BUFFERS*AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE; // typically == 9240
|
|
|
|
// ensure that we allocate 32-bit aligned memory, and mark it non-cacheable
|
|
uint32_t size = 1;
|
|
uint8_t rasr = 0;
|
|
// find size closest to power of 2
|
|
while (size < total_size) {
|
|
size = size << 1;
|
|
rasr++;
|
|
}
|
|
void *mem = malloc_eth_safe(size);
|
|
if (mem == nullptr) {
|
|
return false;
|
|
}
|
|
// ensure our memory is aligned
|
|
// ref. Cortex-M7 peripherals PM0253, section 4.6.4 MPU region base address register
|
|
if (((uint32_t)mem) % size) {
|
|
AP_HAL::panic("Bad alignment of ETH memory");
|
|
}
|
|
|
|
// for total_size == 9240, size should be 16384 and (rasr-1) should be 13 (MPU_RASR_SIZE_16K)
|
|
const uint32_t rasr_size = MPU_RASR_SIZE(rasr-1);
|
|
|
|
// set up MPU region for buffers
|
|
mpuConfigureRegion(STM32_NOCACHE_MPU_REGION_ETH,
|
|
(uint32_t)mem,
|
|
MPU_RASR_ATTR_AP_RW_RW |
|
|
MPU_RASR_ATTR_NON_CACHEABLE |
|
|
MPU_RASR_ATTR_S |
|
|
rasr_size |
|
|
MPU_RASR_ENABLE);
|
|
mpuEnable(MPU_CTRL_PRIVDEFENA);
|
|
SCB_CleanInvalidateDCache();
|
|
|
|
// assign buffers
|
|
__eth_rd = (stm32_eth_rx_descriptor_t *)mem;
|
|
__eth_td = (stm32_eth_tx_descriptor_t *)&__eth_rd[STM32_MAC_RECEIVE_BUFFERS];
|
|
__eth_rb[0] = (uint32_t*)&__eth_td[STM32_MAC_TRANSMIT_BUFFERS];
|
|
for (uint16_t i = 1; i < STM32_MAC_RECEIVE_BUFFERS; i++) {
|
|
__eth_rb[i] = &(__eth_rb[i-1][AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE]);
|
|
}
|
|
__eth_tb[0] = &(__eth_rb[STM32_MAC_RECEIVE_BUFFERS-1][AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE]);
|
|
for (uint16_t i = 1; i < STM32_MAC_TRANSMIT_BUFFERS; i++) {
|
|
__eth_tb[i] = &(__eth_tb[i-1][AP_NETWORKING_EXTERN_MAC_BUFFER_SIZE]);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
initialise ChibiOS network backend using LWIP
|
|
*/
|
|
bool AP_Networking_ChibiOS::init()
|
|
{
|
|
#ifdef HAL_GPIO_ETH_ENABLE
|
|
hal.gpio->pinMode(HAL_GPIO_ETH_ENABLE, HAL_GPIO_OUTPUT);
|
|
hal.gpio->write(HAL_GPIO_ETH_ENABLE, frontend.param.enabled ? 1 : 0);
|
|
#endif
|
|
|
|
if (!allocate_buffers()) {
|
|
GCS_SEND_TEXT(MAV_SEVERITY_ERROR, "NET: Failed to allocate buffers");
|
|
return false;
|
|
}
|
|
|
|
if (!macInit()) {
|
|
GCS_SEND_TEXT(MAV_SEVERITY_ERROR, "NET: macInit failed");
|
|
return false;
|
|
}
|
|
|
|
#if LWIP_IGMP
|
|
if (ETH != nullptr) {
|
|
// enbale "permit multicast" so we can receive multicast packets
|
|
ETH->MACPFR |= ETH_MACPFR_PM;
|
|
}
|
|
#endif
|
|
|
|
thisif = new netif;
|
|
if (thisif == nullptr) {
|
|
return false;
|
|
}
|
|
|
|
if (!hal.scheduler->thread_create(FUNCTOR_BIND_MEMBER(&AP_Networking_ChibiOS::thread, void),
|
|
"network",
|
|
2048, AP_HAL::Scheduler::PRIORITY_NET, 0)) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void AP_Networking_ChibiOS::link_up_cb(void *p)
|
|
{
|
|
auto *driver = (AP_Networking_ChibiOS *)p;
|
|
#if LWIP_DHCP
|
|
if (driver->frontend.get_dhcp_enabled()) {
|
|
dhcp_start(driver->thisif);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void AP_Networking_ChibiOS::link_down_cb(void *p)
|
|
{
|
|
auto *driver = (AP_Networking_ChibiOS *)p;
|
|
#if LWIP_DHCP
|
|
if (driver->frontend.get_dhcp_enabled()) {
|
|
dhcp_stop(driver->thisif);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* This function does the actual transmission of the packet. The packet is
|
|
* contained in the pbuf that is passed to the function. This pbuf
|
|
* might be chained.
|
|
*
|
|
* @param netif the lwip network interface structure for this ethernetif
|
|
* @param p the MAC packet to send (e.g. IP packet including MAC addresses and type)
|
|
* @return ERR_OK if the packet could be sent
|
|
* an err_t value if the packet couldn't be sent
|
|
*
|
|
* @note Returning ERR_MEM here if a DMA queue of your MAC is full can lead to
|
|
* strange results. You might consider waiting for space in the DMA queue
|
|
* to become available since the stack doesn't retry to send a packet
|
|
* dropped because of memory failure (except for the TCP timers).
|
|
*/
|
|
int8_t AP_Networking_ChibiOS::low_level_output(struct netif *netif, struct pbuf *p)
|
|
{
|
|
struct pbuf *q;
|
|
MACTransmitDescriptor td;
|
|
(void)netif;
|
|
|
|
if (macWaitTransmitDescriptor(ÐD1, &td, TIME_MS2I(LWIP_SEND_TIMEOUT_MS)) != MSG_OK) {
|
|
return ERR_TIMEOUT;
|
|
}
|
|
|
|
#if ETH_PAD_SIZE
|
|
pbuf_header(p, -ETH_PAD_SIZE); /* drop the padding word */
|
|
#endif
|
|
|
|
/* Iterates through the pbuf chain. */
|
|
for(q = p; q != NULL; q = q->next) {
|
|
macWriteTransmitDescriptor(&td, (uint8_t *)q->payload, (size_t)q->len);
|
|
}
|
|
macReleaseTransmitDescriptorX(&td);
|
|
|
|
#if ETH_PAD_SIZE
|
|
pbuf_header(p, ETH_PAD_SIZE); /* reclaim the padding word */
|
|
#endif
|
|
|
|
return ERR_OK;
|
|
}
|
|
|
|
/*
|
|
* Receives a frame.
|
|
* Allocates a pbuf and transfers the bytes of the incoming
|
|
* packet from the interface into the pbuf.
|
|
*
|
|
* @param netif the lwip network interface structure for this ethernetif
|
|
* @return a pbuf filled with the received packet (including MAC header)
|
|
* NULL on memory error
|
|
*/
|
|
bool AP_Networking_ChibiOS::low_level_input(struct netif *netif, struct pbuf **pbuf)
|
|
{
|
|
MACReceiveDescriptor rd;
|
|
struct pbuf *q;
|
|
u16_t len;
|
|
|
|
(void)netif;
|
|
|
|
if (macWaitReceiveDescriptor(ÐD1, &rd, TIME_IMMEDIATE) != MSG_OK) {
|
|
return false;
|
|
}
|
|
|
|
len = (u16_t)rd.size;
|
|
|
|
#if ETH_PAD_SIZE
|
|
len += ETH_PAD_SIZE; /* allow room for Ethernet padding */
|
|
#endif
|
|
|
|
/* We allocate a pbuf chain of pbufs from the pool. */
|
|
*pbuf = pbuf_alloc(PBUF_RAW, len, PBUF_POOL);
|
|
|
|
if (*pbuf != nullptr) {
|
|
#if ETH_PAD_SIZE
|
|
pbuf_header(*pbuf, -ETH_PAD_SIZE); /* drop the padding word */
|
|
#endif
|
|
|
|
/* Iterates through the pbuf chain. */
|
|
for(q = *pbuf; q != NULL; q = q->next) {
|
|
macReadReceiveDescriptor(&rd, (uint8_t *)q->payload, (size_t)q->len);
|
|
}
|
|
macReleaseReceiveDescriptorX(&rd);
|
|
|
|
#if ETH_PAD_SIZE
|
|
pbuf_header(*pbuf, ETH_PAD_SIZE); /* reclaim the padding word */
|
|
#endif
|
|
} else {
|
|
macReleaseReceiveDescriptorX(&rd); // Drop packet
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
int8_t AP_Networking_ChibiOS::ethernetif_init(struct netif *netif)
|
|
{
|
|
netif->state = NULL;
|
|
netif->name[0] = 'm';
|
|
netif->name[1] = 's';
|
|
netif->output = etharp_output;
|
|
netif->linkoutput = low_level_output;
|
|
|
|
/* set MAC hardware address length */
|
|
netif->hwaddr_len = ETHARP_HWADDR_LEN;
|
|
|
|
/* maximum transfer unit */
|
|
netif->mtu = LWIP_NETIF_MTU;
|
|
|
|
/* device capabilities */
|
|
netif->flags = NETIF_FLAG_BROADCAST | NETIF_FLAG_ETHARP;
|
|
|
|
#if LWIP_IGMP
|
|
// also enable multicast
|
|
netif->flags |= NETIF_FLAG_IGMP;
|
|
#endif
|
|
|
|
return ERR_OK;
|
|
}
|
|
|
|
/*
|
|
networking thread
|
|
*/
|
|
void AP_Networking_ChibiOS::thread()
|
|
{
|
|
while (!hal.scheduler->is_system_initialized()) {
|
|
hal.scheduler->delay_microseconds(1000);
|
|
}
|
|
|
|
/* start tcpip thread */
|
|
tcpip_init(NULL, NULL);
|
|
|
|
frontend.param.macaddr.get_address(thisif->hwaddr);
|
|
|
|
struct {
|
|
ip4_addr_t ip, gateway, netmask;
|
|
} addr {};
|
|
|
|
if (!frontend.get_dhcp_enabled()) {
|
|
addr.ip.addr = htonl(frontend.get_ip_param());
|
|
addr.gateway.addr = htonl(frontend.get_gateway_param());
|
|
addr.netmask.addr = htonl(frontend.get_netmask_param());
|
|
}
|
|
|
|
const MACConfig mac_config = {thisif->hwaddr};
|
|
macStart(ÐD1, &mac_config);
|
|
|
|
/* Add interface. */
|
|
auto result = netifapi_netif_add(thisif, &addr.ip, &addr.netmask, &addr.gateway, NULL, ethernetif_init, tcpip_input);
|
|
if (result != ERR_OK) {
|
|
AP_HAL::panic("Failed to initialise netif");
|
|
}
|
|
|
|
netifapi_netif_set_default(thisif);
|
|
netifapi_netif_set_up(thisif);
|
|
|
|
/* Setup event sources.*/
|
|
event_timer_t evt;
|
|
event_listener_t el0, el1;
|
|
|
|
evtObjectInit(&evt, LWIP_LINK_POLL_INTERVAL);
|
|
evtStart(&evt);
|
|
chEvtRegisterMask(&evt.et_es, &el0, PERIODIC_TIMER_ID);
|
|
chEvtRegisterMaskWithFlags(macGetEventSource(ÐD1), &el1,
|
|
FRAME_RECEIVED_ID, MAC_FLAGS_RX);
|
|
chEvtAddEvents(PERIODIC_TIMER_ID | FRAME_RECEIVED_ID);
|
|
|
|
while (true) {
|
|
eventmask_t mask = chEvtWaitAny(ALL_EVENTS);
|
|
if (mask & PERIODIC_TIMER_ID) {
|
|
bool current_link_status = macPollLinkStatus(ÐD1);
|
|
if (current_link_status != netif_is_link_up(thisif)) {
|
|
if (current_link_status) {
|
|
tcpip_callback_with_block((tcpip_callback_fn) netif_set_link_up, thisif, 0);
|
|
tcpip_callback_with_block(link_up_cb, this, 0);
|
|
}
|
|
else {
|
|
tcpip_callback_with_block((tcpip_callback_fn) netif_set_link_down, thisif, 0);
|
|
tcpip_callback_with_block(link_down_cb, this, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (mask & FRAME_RECEIVED_ID) {
|
|
struct pbuf *p;
|
|
while (low_level_input(thisif, &p)) {
|
|
if (p != NULL) {
|
|
struct eth_hdr *ethhdr = (struct eth_hdr *)p->payload;
|
|
switch (htons(ethhdr->type)) {
|
|
/* IP or ARP packet? */
|
|
case ETHTYPE_IP:
|
|
case ETHTYPE_ARP:
|
|
/* full packet send to tcpip_thread to process */
|
|
if (thisif->input(p, thisif) == ERR_OK) {
|
|
break;
|
|
}
|
|
/* Falls through */
|
|
default:
|
|
pbuf_free(p);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
update called at 10Hz
|
|
*/
|
|
void AP_Networking_ChibiOS::update()
|
|
{
|
|
const uint32_t ip = ntohl(thisif->ip_addr.addr);
|
|
const uint32_t nm = ntohl(thisif->netmask.addr);
|
|
const uint32_t gw = ntohl(thisif->gw.addr);
|
|
|
|
if (ip != activeSettings.ip ||
|
|
nm != activeSettings.nm ||
|
|
gw != activeSettings.gw) {
|
|
activeSettings.ip = ip;
|
|
activeSettings.gw = gw;
|
|
activeSettings.nm = nm;
|
|
activeSettings.last_change_ms = AP_HAL::millis();
|
|
}
|
|
}
|
|
|
|
#endif // AP_NETWORKING_BACKEND_CHIBIOS
|
|
|