434 lines
12 KiB
C
434 lines
12 KiB
C
|
/*
|
||
|
* Copyright (c) 2023 Enphase Energy
|
||
|
*
|
||
|
* SPDX-License-Identifier: Apache-2.0
|
||
|
*/
|
||
|
|
||
|
#define DT_DRV_COMPAT siemens_ivshmem_eth
|
||
|
|
||
|
#include <zephyr/drivers/virtualization/ivshmem.h>
|
||
|
#include <zephyr/logging/log.h>
|
||
|
#include <zephyr/net/ethernet.h>
|
||
|
#include <ethernet/eth_stats.h>
|
||
|
|
||
|
#include "eth.h"
|
||
|
#include "eth_ivshmem_priv.h"
|
||
|
|
||
|
LOG_MODULE_REGISTER(eth_ivshmem, CONFIG_ETHERNET_LOG_LEVEL);
|
||
|
|
||
|
#define ETH_IVSHMEM_STATE_RESET 0
|
||
|
#define ETH_IVSHMEM_STATE_INIT 1
|
||
|
#define ETH_IVSHMEM_STATE_READY 2
|
||
|
#define ETH_IVSHMEM_STATE_RUN 3
|
||
|
|
||
|
static const char * const eth_ivshmem_state_names[] = {
|
||
|
[ETH_IVSHMEM_STATE_RESET] = "RESET",
|
||
|
[ETH_IVSHMEM_STATE_INIT] = "INIT",
|
||
|
[ETH_IVSHMEM_STATE_READY] = "READY",
|
||
|
[ETH_IVSHMEM_STATE_RUN] = "RUN"
|
||
|
};
|
||
|
|
||
|
struct eth_ivshmem_dev_data {
|
||
|
struct net_if *iface;
|
||
|
|
||
|
uint32_t tx_rx_vector;
|
||
|
uint32_t peer_id;
|
||
|
uint8_t mac_addr[6];
|
||
|
struct k_poll_signal poll_signal;
|
||
|
struct eth_ivshmem_queue ivshmem_queue;
|
||
|
|
||
|
K_KERNEL_STACK_MEMBER(thread_stack, CONFIG_ETH_IVSHMEM_THREAD_STACK_SIZE);
|
||
|
struct k_thread thread;
|
||
|
bool enabled;
|
||
|
uint32_t state;
|
||
|
#if defined(CONFIG_NET_STATISTICS_ETHERNET)
|
||
|
struct net_stats_eth stats;
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
struct eth_ivshmem_cfg_data {
|
||
|
const struct device *ivshmem;
|
||
|
const char *name;
|
||
|
void (*generate_mac_addr)(uint8_t mac_addr[6]);
|
||
|
};
|
||
|
|
||
|
#if defined(CONFIG_NET_STATISTICS_ETHERNET)
|
||
|
static struct net_stats_eth *eth_ivshmem_get_stats(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
|
||
|
return &dev_data->stats;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static int eth_ivshmem_start(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
|
||
|
dev_data->enabled = true;
|
||
|
|
||
|
/* Wake up thread to check/update state */
|
||
|
k_poll_signal_raise(&dev_data->poll_signal, 0);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int eth_ivshmem_stop(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
|
||
|
dev_data->enabled = false;
|
||
|
|
||
|
/* Wake up thread to check/update state */
|
||
|
k_poll_signal_raise(&dev_data->poll_signal, 0);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static enum ethernet_hw_caps eth_ivshmem_caps(const struct device *dev)
|
||
|
{
|
||
|
ARG_UNUSED(dev);
|
||
|
return ETHERNET_LINK_10BASE_T | ETHERNET_LINK_100BASE_T | ETHERNET_LINK_1000BASE_T;
|
||
|
}
|
||
|
|
||
|
static int eth_ivshmem_send(const struct device *dev, struct net_pkt *pkt)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
const struct eth_ivshmem_cfg_data *cfg_data = dev->config;
|
||
|
size_t len = net_pkt_get_len(pkt);
|
||
|
|
||
|
void *data;
|
||
|
int res = eth_ivshmem_queue_tx_get_buff(&dev_data->ivshmem_queue, &data, len);
|
||
|
|
||
|
if (res != 0) {
|
||
|
LOG_ERR("Failed to allocate tx buffer");
|
||
|
eth_stats_update_errors_tx(dev_data->iface);
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
if (net_pkt_read(pkt, data, len)) {
|
||
|
LOG_ERR("Failed to read tx packet");
|
||
|
eth_stats_update_errors_tx(dev_data->iface);
|
||
|
return -EIO;
|
||
|
}
|
||
|
|
||
|
res = eth_ivshmem_queue_tx_commit_buff(&dev_data->ivshmem_queue);
|
||
|
if (res == 0) {
|
||
|
/* Notify peer */
|
||
|
ivshmem_int_peer(cfg_data->ivshmem, dev_data->peer_id, dev_data->tx_rx_vector);
|
||
|
}
|
||
|
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
static struct net_pkt *eth_ivshmem_rx(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
const struct eth_ivshmem_cfg_data *cfg_data = dev->config;
|
||
|
const void *rx_data;
|
||
|
size_t rx_len;
|
||
|
|
||
|
int res = eth_ivshmem_queue_rx(&dev_data->ivshmem_queue, &rx_data, &rx_len);
|
||
|
|
||
|
if (res != 0) {
|
||
|
if (res != -EWOULDBLOCK) {
|
||
|
LOG_ERR("Queue RX failed");
|
||
|
eth_stats_update_errors_rx(dev_data->iface);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
struct net_pkt *pkt = net_pkt_rx_alloc_with_buffer(
|
||
|
dev_data->iface, rx_len, AF_UNSPEC, 0, K_MSEC(100));
|
||
|
if (pkt == NULL) {
|
||
|
LOG_ERR("Failed to allocate rx buffer");
|
||
|
eth_stats_update_errors_rx(dev_data->iface);
|
||
|
goto dequeue;
|
||
|
}
|
||
|
|
||
|
if (net_pkt_write(pkt, rx_data, rx_len) != 0) {
|
||
|
LOG_ERR("Failed to write rx packet");
|
||
|
eth_stats_update_errors_rx(dev_data->iface);
|
||
|
net_pkt_unref(pkt);
|
||
|
}
|
||
|
|
||
|
dequeue:
|
||
|
if (eth_ivshmem_queue_rx_complete(&dev_data->ivshmem_queue) == 0) {
|
||
|
/* Notify peer */
|
||
|
ivshmem_int_peer(cfg_data->ivshmem, dev_data->peer_id, dev_data->tx_rx_vector);
|
||
|
}
|
||
|
|
||
|
return pkt;
|
||
|
}
|
||
|
|
||
|
static void eth_ivshmem_set_state(const struct device *dev, uint32_t state)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
const struct eth_ivshmem_cfg_data *cfg_data = dev->config;
|
||
|
|
||
|
LOG_DBG("State update: %s -> %s",
|
||
|
eth_ivshmem_state_names[dev_data->state],
|
||
|
eth_ivshmem_state_names[state]);
|
||
|
dev_data->state = state;
|
||
|
ivshmem_set_state(cfg_data->ivshmem, state);
|
||
|
}
|
||
|
|
||
|
static void eth_ivshmem_state_update(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
const struct eth_ivshmem_cfg_data *cfg_data = dev->config;
|
||
|
|
||
|
uint32_t peer_state = ivshmem_get_state(cfg_data->ivshmem, dev_data->peer_id);
|
||
|
|
||
|
switch (dev_data->state) {
|
||
|
case ETH_IVSHMEM_STATE_RESET:
|
||
|
switch (peer_state) {
|
||
|
case ETH_IVSHMEM_STATE_RESET:
|
||
|
case ETH_IVSHMEM_STATE_INIT:
|
||
|
eth_ivshmem_set_state(dev, ETH_IVSHMEM_STATE_INIT);
|
||
|
break;
|
||
|
default:
|
||
|
/* Wait for peer to reset */
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
case ETH_IVSHMEM_STATE_INIT:
|
||
|
if (dev_data->iface == NULL || peer_state == ETH_IVSHMEM_STATE_RESET) {
|
||
|
/* Peer is not ready for init */
|
||
|
break;
|
||
|
}
|
||
|
eth_ivshmem_queue_reset(&dev_data->ivshmem_queue);
|
||
|
eth_ivshmem_set_state(dev, ETH_IVSHMEM_STATE_READY);
|
||
|
break;
|
||
|
case ETH_IVSHMEM_STATE_READY:
|
||
|
case ETH_IVSHMEM_STATE_RUN:
|
||
|
switch (peer_state) {
|
||
|
case ETH_IVSHMEM_STATE_RESET:
|
||
|
net_eth_carrier_off(dev_data->iface);
|
||
|
eth_ivshmem_set_state(dev, ETH_IVSHMEM_STATE_RESET);
|
||
|
break;
|
||
|
case ETH_IVSHMEM_STATE_READY:
|
||
|
case ETH_IVSHMEM_STATE_RUN:
|
||
|
if (dev_data->enabled && dev_data->state == ETH_IVSHMEM_STATE_READY) {
|
||
|
eth_ivshmem_set_state(dev, ETH_IVSHMEM_STATE_RUN);
|
||
|
net_eth_carrier_on(dev_data->iface);
|
||
|
} else if (!dev_data->enabled && dev_data->state == ETH_IVSHMEM_STATE_RUN) {
|
||
|
net_eth_carrier_off(dev_data->iface);
|
||
|
eth_ivshmem_set_state(dev, ETH_IVSHMEM_STATE_RESET);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
FUNC_NORETURN static void eth_ivshmem_thread(void *arg1, void *arg2, void *arg3)
|
||
|
{
|
||
|
const struct device *dev = arg1;
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
struct k_poll_event poll_event;
|
||
|
|
||
|
ARG_UNUSED(arg2);
|
||
|
ARG_UNUSED(arg3);
|
||
|
|
||
|
k_poll_event_init(&poll_event,
|
||
|
K_POLL_TYPE_SIGNAL,
|
||
|
K_POLL_MODE_NOTIFY_ONLY,
|
||
|
&dev_data->poll_signal);
|
||
|
|
||
|
while (true) {
|
||
|
k_poll(&poll_event, 1, K_FOREVER);
|
||
|
poll_event.signal->signaled = 0;
|
||
|
poll_event.state = K_POLL_STATE_NOT_READY;
|
||
|
|
||
|
eth_ivshmem_state_update(dev);
|
||
|
if (dev_data->state != ETH_IVSHMEM_STATE_RUN) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
while (true) {
|
||
|
struct net_pkt *pkt = eth_ivshmem_rx(dev);
|
||
|
|
||
|
if (pkt == NULL) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (net_recv_data(dev_data->iface, pkt) < 0) {
|
||
|
/* Upper layers are not ready to receive packets */
|
||
|
net_pkt_unref(pkt);
|
||
|
}
|
||
|
|
||
|
k_yield();
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int eth_ivshmem_initialize(const struct device *dev)
|
||
|
{
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
const struct eth_ivshmem_cfg_data *cfg_data = dev->config;
|
||
|
int res;
|
||
|
|
||
|
k_poll_signal_init(&dev_data->poll_signal);
|
||
|
|
||
|
if (!device_is_ready(cfg_data->ivshmem)) {
|
||
|
LOG_ERR("ivshmem device not ready");
|
||
|
return -ENODEV;
|
||
|
}
|
||
|
|
||
|
uint16_t protocol = ivshmem_get_protocol(cfg_data->ivshmem);
|
||
|
|
||
|
if (protocol != IVSHMEM_V2_PROTO_NET) {
|
||
|
LOG_ERR("Invalid ivshmem protocol %hu", protocol);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
|
||
|
uint32_t id = ivshmem_get_id(cfg_data->ivshmem);
|
||
|
uint32_t max_peers = ivshmem_get_max_peers(cfg_data->ivshmem);
|
||
|
|
||
|
LOG_INF("ivshmem: id %u, max_peers %u", id, max_peers);
|
||
|
if (id > 1) {
|
||
|
LOG_ERR("Invalid ivshmem ID %u", id);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
if (max_peers != 2) {
|
||
|
LOG_ERR("Invalid ivshmem max peers %u", max_peers);
|
||
|
return -EINVAL;
|
||
|
}
|
||
|
dev_data->peer_id = (id == 0) ? 1 : 0;
|
||
|
|
||
|
bool tx_buffer_first = id == 0;
|
||
|
uintptr_t output_section_addr;
|
||
|
size_t output_section_size = ivshmem_get_output_mem_section(
|
||
|
cfg_data->ivshmem, 0, &output_section_addr);
|
||
|
|
||
|
res = eth_ivshmem_queue_init(
|
||
|
&dev_data->ivshmem_queue, output_section_addr,
|
||
|
output_section_size, tx_buffer_first);
|
||
|
if (res != 0) {
|
||
|
LOG_ERR("Failed to init ivshmem queue");
|
||
|
return res;
|
||
|
}
|
||
|
LOG_INF("shmem queue: desc len 0x%hX, header size 0x%X, data size 0x%X",
|
||
|
dev_data->ivshmem_queue.desc_max_len,
|
||
|
dev_data->ivshmem_queue.vring_header_size,
|
||
|
dev_data->ivshmem_queue.vring_data_max_len);
|
||
|
|
||
|
uint16_t n_vectors = ivshmem_get_vectors(cfg_data->ivshmem);
|
||
|
|
||
|
/* For simplicity, state and TX/RX vectors do the same thing */
|
||
|
ivshmem_register_handler(cfg_data->ivshmem, &dev_data->poll_signal, 0);
|
||
|
dev_data->tx_rx_vector = 0;
|
||
|
if (n_vectors == 0) {
|
||
|
LOG_ERR("Error no ivshmem ISR vectors");
|
||
|
return -EINVAL;
|
||
|
} else if (n_vectors > 1) {
|
||
|
ivshmem_register_handler(cfg_data->ivshmem, &dev_data->poll_signal, 1);
|
||
|
dev_data->tx_rx_vector = 1;
|
||
|
}
|
||
|
|
||
|
ivshmem_set_state(cfg_data->ivshmem, ETH_IVSHMEM_STATE_RESET);
|
||
|
|
||
|
cfg_data->generate_mac_addr(dev_data->mac_addr);
|
||
|
LOG_INF("MAC Address %02X:%02X:%02X:%02X:%02X:%02X",
|
||
|
dev_data->mac_addr[0], dev_data->mac_addr[1],
|
||
|
dev_data->mac_addr[2], dev_data->mac_addr[3],
|
||
|
dev_data->mac_addr[4], dev_data->mac_addr[5]);
|
||
|
|
||
|
k_tid_t tid = k_thread_create(
|
||
|
&dev_data->thread, dev_data->thread_stack,
|
||
|
K_KERNEL_STACK_SIZEOF(dev_data->thread_stack),
|
||
|
eth_ivshmem_thread,
|
||
|
(void *) dev, NULL, NULL,
|
||
|
CONFIG_ETH_IVSHMEM_THREAD_PRIORITY,
|
||
|
K_ESSENTIAL, K_NO_WAIT);
|
||
|
k_thread_name_set(tid, cfg_data->name);
|
||
|
|
||
|
ivshmem_enable_interrupts(cfg_data->ivshmem, true);
|
||
|
|
||
|
/* Wake up thread to check/update state */
|
||
|
k_poll_signal_raise(&dev_data->poll_signal, 0);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void eth_ivshmem_iface_init(struct net_if *iface)
|
||
|
{
|
||
|
const struct device *dev = net_if_get_device(iface);
|
||
|
struct eth_ivshmem_dev_data *dev_data = dev->data;
|
||
|
|
||
|
if (dev_data->iface == NULL) {
|
||
|
dev_data->iface = iface;
|
||
|
}
|
||
|
|
||
|
net_if_set_link_addr(
|
||
|
iface, dev_data->mac_addr,
|
||
|
sizeof(dev_data->mac_addr),
|
||
|
NET_LINK_ETHERNET);
|
||
|
|
||
|
ethernet_init(iface);
|
||
|
|
||
|
/* Do not start the interface until PHY link is up */
|
||
|
net_if_carrier_off(iface);
|
||
|
|
||
|
/* Wake up thread to check/update state */
|
||
|
k_poll_signal_raise(&dev_data->poll_signal, 0);
|
||
|
}
|
||
|
|
||
|
static const struct ethernet_api eth_ivshmem_api = {
|
||
|
.iface_api.init = eth_ivshmem_iface_init,
|
||
|
#if defined(CONFIG_NET_STATISTICS_ETHERNET)
|
||
|
.get_stats = eth_ivshmem_get_stats,
|
||
|
#endif
|
||
|
.start = eth_ivshmem_start,
|
||
|
.stop = eth_ivshmem_stop,
|
||
|
.get_capabilities = eth_ivshmem_caps,
|
||
|
.send = eth_ivshmem_send,
|
||
|
};
|
||
|
|
||
|
#define ETH_IVSHMEM_RANDOM_MAC_ADDR(inst) \
|
||
|
static void generate_mac_addr_##inst(uint8_t mac_addr[6]) \
|
||
|
{ \
|
||
|
uint32_t entropy = sys_rand32_get(); \
|
||
|
mac_addr[0] = (entropy >> 16) & 0xff; \
|
||
|
mac_addr[1] = (entropy >> 8) & 0xff; \
|
||
|
mac_addr[2] = (entropy >> 0) & 0xff; \
|
||
|
/* Clear multicast bit */ \
|
||
|
mac_addr[0] &= 0xFE; \
|
||
|
gen_random_mac(mac_addr, mac_addr[0], mac_addr[1], mac_addr[2]); \
|
||
|
}
|
||
|
|
||
|
#define ETH_IVSHMEM_LOCAL_MAC_ADDR(inst) \
|
||
|
static void generate_mac_addr_##inst(uint8_t mac_addr[6]) \
|
||
|
{ \
|
||
|
const uint8_t addr[6] = DT_INST_PROP(0, local_mac_address); \
|
||
|
memcpy(mac_addr, addr, sizeof(addr)); \
|
||
|
}
|
||
|
|
||
|
#define ETH_IVSHMEM_GENERATE_MAC_ADDR(inst) \
|
||
|
BUILD_ASSERT(DT_INST_PROP(inst, zephyr_random_mac_address) || \
|
||
|
NODE_HAS_VALID_MAC_ADDR(DT_DRV_INST(inst)), \
|
||
|
"eth_ivshmem requires either a fixed or random mac address"); \
|
||
|
COND_CODE_1(DT_INST_PROP(inst, zephyr_random_mac_address), \
|
||
|
(ETH_IVSHMEM_RANDOM_MAC_ADDR(inst)), \
|
||
|
(ETH_IVSHMEM_LOCAL_MAC_ADDR(inst)))
|
||
|
|
||
|
#define ETH_IVSHMEM_INIT(inst) \
|
||
|
ETH_IVSHMEM_GENERATE_MAC_ADDR(inst); \
|
||
|
static struct eth_ivshmem_dev_data eth_ivshmem_dev_##inst = {}; \
|
||
|
static const struct eth_ivshmem_cfg_data eth_ivshmem_cfg_##inst = { \
|
||
|
.ivshmem = DEVICE_DT_GET(DT_INST_PHANDLE(inst, ivshmem_v2)), \
|
||
|
.name = "ivshmem_eth" STRINGIFY(inst), \
|
||
|
.generate_mac_addr = generate_mac_addr_##inst, \
|
||
|
}; \
|
||
|
ETH_NET_DEVICE_DT_INST_DEFINE(inst, \
|
||
|
eth_ivshmem_initialize, \
|
||
|
NULL, \
|
||
|
ð_ivshmem_dev_##inst, \
|
||
|
ð_ivshmem_cfg_##inst, \
|
||
|
CONFIG_ETH_INIT_PRIORITY, \
|
||
|
ð_ivshmem_api, \
|
||
|
NET_ETH_MTU);
|
||
|
|
||
|
DT_INST_FOREACH_STATUS_OKAY(ETH_IVSHMEM_INIT);
|