2018-07-31 11:44:34 +02:00
|
|
|
/*
|
2019-12-22 12:17:43 +01:00
|
|
|
* Copyright (c) 2018-2019 Intel Corporation.
|
2018-07-31 11:44:34 +02:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2020-03-25 17:12:19 +01:00
|
|
|
#define DT_DRV_COMPAT intel_e1000
|
|
|
|
|
2018-07-31 11:44:34 +02:00
|
|
|
#define LOG_MODULE_NAME eth_e1000
|
|
|
|
#define LOG_LEVEL CONFIG_ETHERNET_LOG_LEVEL
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/logging/log.h>
|
2018-07-31 11:44:34 +02:00
|
|
|
LOG_MODULE_REGISTER(LOG_MODULE_NAME);
|
|
|
|
|
2020-01-08 12:25:50 +01:00
|
|
|
#include <sys/types.h>
|
includes: prefer <zephyr/kernel.h> over <zephyr/zephyr.h>
As of today <zephyr/zephyr.h> is 100% equivalent to <zephyr/kernel.h>.
This patch proposes to then include <zephyr/kernel.h> instead of
<zephyr/zephyr.h> since it is more clear that you are including the
Kernel APIs and (probably) nothing else. <zephyr/zephyr.h> sounds like a
catch-all header that may be confusing. Most applications need to
include a bunch of other things to compile, e.g. driver headers or
subsystem headers like BT, logging, etc.
The idea of a catch-all header in Zephyr is probably not feasible
anyway. Reason is that Zephyr is not a library, like it could be for
example `libpython`. Zephyr provides many utilities nowadays: a kernel,
drivers, subsystems, etc and things will likely grow. A catch-all header
would be massive, difficult to keep up-to-date. It is also likely that
an application will only build a small subset. Note that subsystem-level
headers may use a catch-all approach to make things easier, though.
NOTE: This patch is **NOT** removing the header, just removing its usage
in-tree. I'd advocate for its deprecation (add a #warning on it), but I
understand many people will have concerns.
Signed-off-by: Gerard Marull-Paretas <gerard.marull@nordicsemi.no>
2022-08-25 09:58:46 +02:00
|
|
|
#include <zephyr/kernel.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/net/ethernet.h>
|
2018-12-05 14:52:59 +01:00
|
|
|
#include <ethernet/eth_stats.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/pcie/pcie.h>
|
2022-10-17 10:24:11 +02:00
|
|
|
#include <zephyr/irq.h>
|
2018-07-31 11:44:34 +02:00
|
|
|
#include "eth_e1000_priv.h"
|
|
|
|
|
2020-05-15 14:29:51 +02:00
|
|
|
#if defined(CONFIG_ETH_E1000_PTP_CLOCK)
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/ptp_clock.h>
|
2020-05-15 14:29:51 +02:00
|
|
|
|
2022-09-06 12:22:33 +02:00
|
|
|
#define PTP_INST_NODEID(n) DT_INST_CHILD(n, ptp)
|
2020-05-15 14:29:51 +02:00
|
|
|
#endif
|
|
|
|
|
2019-12-22 12:33:44 +01:00
|
|
|
#if defined(CONFIG_ETH_E1000_VERBOSE_DEBUG)
|
|
|
|
#define hexdump(_buf, _len, fmt, args...) \
|
|
|
|
({ \
|
|
|
|
const size_t STR_SIZE = 80; \
|
|
|
|
char _str[STR_SIZE]; \
|
|
|
|
\
|
|
|
|
snprintk(_str, STR_SIZE, "%s: " fmt, __func__, ## args); \
|
|
|
|
\
|
2022-06-20 07:43:37 +02:00
|
|
|
LOG_HEXDUMP_DBG(_buf, _len, _str); \
|
2019-12-22 12:33:44 +01:00
|
|
|
})
|
|
|
|
#else
|
|
|
|
#define hexdump(args...)
|
|
|
|
#endif
|
|
|
|
|
2018-07-31 11:44:34 +02:00
|
|
|
static const char *e1000_reg_to_string(enum e1000_reg_t r)
|
|
|
|
{
|
|
|
|
#define _(_x) case _x: return #_x
|
|
|
|
switch (r) {
|
|
|
|
_(CTRL);
|
|
|
|
_(ICR);
|
|
|
|
_(ICS);
|
|
|
|
_(IMS);
|
|
|
|
_(RCTL);
|
|
|
|
_(TCTL);
|
|
|
|
_(RDBAL);
|
|
|
|
_(RDBAH);
|
|
|
|
_(RDLEN);
|
|
|
|
_(RDH);
|
|
|
|
_(RDT);
|
|
|
|
_(TDBAL);
|
|
|
|
_(TDBAH);
|
|
|
|
_(TDLEN);
|
|
|
|
_(TDH);
|
|
|
|
_(TDT);
|
|
|
|
_(RAL);
|
|
|
|
_(RAH);
|
|
|
|
}
|
|
|
|
#undef _
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_ERR("Unsupported register: 0x%x", r);
|
2018-07-31 11:44:34 +02:00
|
|
|
k_oops();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
static struct net_if *get_iface(struct e1000_dev *ctx, uint16_t vlan_tag)
|
|
|
|
{
|
|
|
|
#if defined(CONFIG_NET_VLAN)
|
|
|
|
struct net_if *iface;
|
|
|
|
|
|
|
|
iface = net_eth_get_vlan_iface(ctx->iface, vlan_tag);
|
|
|
|
if (!iface) {
|
|
|
|
return ctx->iface;
|
|
|
|
}
|
|
|
|
|
|
|
|
return iface;
|
|
|
|
#else
|
|
|
|
ARG_UNUSED(vlan_tag);
|
|
|
|
|
|
|
|
return ctx->iface;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static enum ethernet_hw_caps e1000_caps(const struct device *dev)
|
2018-07-31 11:44:34 +02:00
|
|
|
{
|
2020-05-11 09:49:55 +02:00
|
|
|
return
|
|
|
|
#if IS_ENABLED(CONFIG_NET_VLAN)
|
|
|
|
ETHERNET_HW_VLAN |
|
2020-05-15 14:29:51 +02:00
|
|
|
#endif
|
|
|
|
#if IS_ENABLED(CONFIG_ETH_E1000_PTP_CLOCK)
|
|
|
|
ETHERNET_PTP |
|
2020-05-11 09:49:55 +02:00
|
|
|
#endif
|
|
|
|
ETHERNET_LINK_10BASE_T | ETHERNET_LINK_100BASE_T |
|
2021-06-04 10:07:46 +02:00
|
|
|
ETHERNET_LINK_1000BASE_T |
|
|
|
|
/* The driver does not really support TXTIME atm but mark
|
|
|
|
* it to support it so that we can test the txtime sample.
|
|
|
|
*/
|
|
|
|
ETHERNET_TXTIME;
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
|
2020-05-15 14:29:51 +02:00
|
|
|
#if defined(CONFIG_ETH_E1000_PTP_CLOCK)
|
|
|
|
static const struct device *e1000_get_ptp_clock(const struct device *dev)
|
|
|
|
{
|
|
|
|
struct e1000_dev *ctx = dev->data;
|
|
|
|
|
|
|
|
return ctx->ptp_clock;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-12-22 12:17:43 +01:00
|
|
|
static int e1000_tx(struct e1000_dev *dev, void *buf, size_t len)
|
2018-07-31 11:44:34 +02:00
|
|
|
{
|
2019-12-22 12:33:44 +01:00
|
|
|
hexdump(buf, len, "%zu byte(s)", len);
|
|
|
|
|
2019-12-22 12:17:43 +01:00
|
|
|
dev->tx.addr = POINTER_TO_INT(buf);
|
|
|
|
dev->tx.len = len;
|
2018-07-31 11:44:34 +02:00
|
|
|
dev->tx.cmd = TDESC_EOP | TDESC_RS;
|
|
|
|
|
|
|
|
iow32(dev, TDT, 1);
|
|
|
|
|
|
|
|
while (!(dev->tx.sta)) {
|
|
|
|
k_yield();
|
|
|
|
}
|
|
|
|
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_DBG("tx.sta: 0x%02hx", dev->tx.sta);
|
2018-07-31 11:44:34 +02:00
|
|
|
|
|
|
|
return (dev->tx.sta & TDESC_STA_DD) ? 0 : -EIO;
|
|
|
|
}
|
|
|
|
|
2021-03-22 15:28:25 +01:00
|
|
|
static int e1000_send(const struct device *ddev, struct net_pkt *pkt)
|
2018-07-31 11:44:34 +02:00
|
|
|
{
|
2021-03-22 15:28:25 +01:00
|
|
|
struct e1000_dev *dev = ddev->data;
|
2018-11-27 20:16:42 +01:00
|
|
|
size_t len = net_pkt_get_len(pkt);
|
|
|
|
|
2019-02-20 09:40:48 +01:00
|
|
|
if (net_pkt_read(pkt, dev->txb, len)) {
|
2018-11-27 20:16:42 +01:00
|
|
|
return -EIO;
|
|
|
|
}
|
2018-07-31 11:44:34 +02:00
|
|
|
|
2018-06-26 14:51:05 +02:00
|
|
|
return e1000_tx(dev, dev->txb, len);
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static struct net_pkt *e1000_rx(struct e1000_dev *dev)
|
|
|
|
{
|
|
|
|
struct net_pkt *pkt = NULL;
|
2019-12-22 12:20:27 +01:00
|
|
|
void *buf;
|
|
|
|
ssize_t len;
|
2018-07-31 11:44:34 +02:00
|
|
|
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_DBG("rx.sta: 0x%02hx", dev->rx.sta);
|
2018-07-31 11:44:34 +02:00
|
|
|
|
|
|
|
if (!(dev->rx.sta & RDESC_STA_DD)) {
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_ERR("RX descriptor not ready");
|
2018-07-31 11:44:34 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
buf = INT_TO_POINTER((uint32_t)dev->rx.addr);
|
2019-12-22 12:20:27 +01:00
|
|
|
len = dev->rx.len - 4;
|
|
|
|
|
2019-12-22 12:26:33 +01:00
|
|
|
if (len <= 0) {
|
|
|
|
LOG_ERR("Invalid RX descriptor length: %hu", dev->rx.len);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-22 12:33:44 +01:00
|
|
|
hexdump(buf, len, "%zd byte(s)", len);
|
|
|
|
|
2019-12-22 12:20:27 +01:00
|
|
|
pkt = net_pkt_rx_alloc_with_buffer(dev->iface, len, AF_UNSPEC, 0,
|
|
|
|
K_NO_WAIT);
|
2018-07-31 11:44:34 +02:00
|
|
|
if (!pkt) {
|
2018-11-27 20:16:42 +01:00
|
|
|
LOG_ERR("Out of buffers");
|
2018-07-31 11:44:34 +02:00
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2019-12-22 12:20:27 +01:00
|
|
|
if (net_pkt_write(pkt, buf, len)) {
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_ERR("Out of memory for received frame");
|
2018-07-31 11:44:34 +02:00
|
|
|
net_pkt_unref(pkt);
|
|
|
|
pkt = NULL;
|
|
|
|
}
|
2018-11-27 20:16:42 +01:00
|
|
|
|
2018-07-31 11:44:34 +02:00
|
|
|
out:
|
|
|
|
return pkt;
|
|
|
|
}
|
|
|
|
|
2021-03-22 15:28:25 +01:00
|
|
|
static void e1000_isr(const struct device *ddev)
|
2018-07-31 11:44:34 +02:00
|
|
|
{
|
2021-03-22 15:28:25 +01:00
|
|
|
struct e1000_dev *dev = ddev->data;
|
2020-05-27 18:26:57 +02:00
|
|
|
uint32_t icr = ior32(dev, ICR); /* Cleared upon read */
|
2020-05-11 09:49:55 +02:00
|
|
|
uint16_t vlan_tag = NET_VLAN_TAG_UNSPEC;
|
2018-07-31 11:44:34 +02:00
|
|
|
|
|
|
|
icr &= ~(ICR_TXDW | ICR_TXQE);
|
|
|
|
|
|
|
|
if (icr & ICR_RXO) {
|
|
|
|
struct net_pkt *pkt = e1000_rx(dev);
|
|
|
|
|
|
|
|
icr &= ~ICR_RXO;
|
|
|
|
|
|
|
|
if (pkt) {
|
2020-05-11 09:49:55 +02:00
|
|
|
#if defined(CONFIG_NET_VLAN)
|
|
|
|
struct net_eth_hdr *hdr = NET_ETH_HDR(pkt);
|
|
|
|
|
|
|
|
if (ntohs(hdr->type) == NET_ETH_PTYPE_VLAN) {
|
|
|
|
struct net_eth_vlan_hdr *hdr_vlan =
|
|
|
|
(struct net_eth_vlan_hdr *)
|
|
|
|
NET_ETH_HDR(pkt);
|
|
|
|
|
|
|
|
net_pkt_set_vlan_tci(
|
|
|
|
pkt, ntohs(hdr_vlan->vlan.tci));
|
|
|
|
vlan_tag = net_pkt_vlan_tag(pkt);
|
|
|
|
|
|
|
|
#if CONFIG_NET_TC_RX_COUNT > 1
|
|
|
|
enum net_priority prio;
|
|
|
|
|
|
|
|
prio = net_vlan2priority(
|
|
|
|
net_pkt_vlan_priority(pkt));
|
|
|
|
net_pkt_set_priority(pkt, prio);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_NET_VLAN */
|
|
|
|
|
|
|
|
net_recv_data(get_iface(dev, vlan_tag), pkt);
|
2018-12-05 14:52:59 +01:00
|
|
|
} else {
|
2020-05-11 09:49:55 +02:00
|
|
|
eth_stats_update_errors_rx(get_iface(dev, vlan_tag));
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (icr) {
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_ERR("Unhandled interrupt, ICR: 0x%x", icr);
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-19 00:47:36 +02:00
|
|
|
#define PCI_VENDOR_ID_INTEL 0x8086
|
|
|
|
#define PCI_DEVICE_ID_I82540EM 0x100e
|
|
|
|
|
2021-03-22 15:28:25 +01:00
|
|
|
int e1000_probe(const struct device *ddev)
|
2018-07-31 11:44:34 +02:00
|
|
|
{
|
2022-11-01 16:36:22 +01:00
|
|
|
const pcie_bdf_t bdf = PCIE_BDF(0, 2, 0);
|
2021-03-22 15:28:25 +01:00
|
|
|
struct e1000_dev *dev = ddev->data;
|
2020-05-11 09:49:55 +02:00
|
|
|
uint32_t ral, rah;
|
2022-10-21 10:23:42 +02:00
|
|
|
struct pcie_bar mbar;
|
2019-04-19 00:47:36 +02:00
|
|
|
|
2020-06-26 21:03:17 +02:00
|
|
|
if (!pcie_probe(bdf, PCIE_ID(PCI_VENDOR_ID_INTEL,
|
|
|
|
PCI_DEVICE_ID_I82540EM))) {
|
|
|
|
return -ENODEV;
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
|
2021-08-18 13:20:59 +02:00
|
|
|
pcie_probe_mbar(bdf, 0, &mbar);
|
2020-06-26 21:03:17 +02:00
|
|
|
pcie_set_cmd(bdf, PCIE_CONF_CMDSTAT_MEM |
|
|
|
|
PCIE_CONF_CMDSTAT_MASTER, true);
|
|
|
|
|
2020-11-05 00:15:25 +01:00
|
|
|
device_map(&dev->address, mbar.phys_addr, mbar.size,
|
2020-06-26 21:03:17 +02:00
|
|
|
K_MEM_CACHE_NONE);
|
|
|
|
|
2018-07-31 11:44:34 +02:00
|
|
|
/* Setup TX descriptor */
|
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
iow32(dev, TDBAL, (uint32_t) &dev->tx);
|
2018-07-31 11:44:34 +02:00
|
|
|
iow32(dev, TDBAH, 0);
|
|
|
|
iow32(dev, TDLEN, 1*16);
|
|
|
|
|
|
|
|
iow32(dev, TDH, 0);
|
|
|
|
iow32(dev, TDT, 0);
|
|
|
|
|
|
|
|
iow32(dev, TCTL, TCTL_EN);
|
|
|
|
|
|
|
|
/* Setup RX descriptor */
|
|
|
|
|
|
|
|
dev->rx.addr = POINTER_TO_INT(dev->rxb);
|
|
|
|
dev->rx.len = sizeof(dev->rxb);
|
|
|
|
|
2020-05-27 18:26:57 +02:00
|
|
|
iow32(dev, RDBAL, (uint32_t) &dev->rx);
|
2018-07-31 11:44:34 +02:00
|
|
|
iow32(dev, RDBAH, 0);
|
|
|
|
iow32(dev, RDLEN, 1*16);
|
|
|
|
|
|
|
|
iow32(dev, RDH, 0);
|
|
|
|
iow32(dev, RDT, 1);
|
|
|
|
|
|
|
|
iow32(dev, IMS, IMS_RXO);
|
|
|
|
|
|
|
|
ral = ior32(dev, RAL);
|
|
|
|
rah = ior32(dev, RAH);
|
|
|
|
|
|
|
|
memcpy(dev->mac, &ral, 4);
|
|
|
|
memcpy(dev->mac + 4, &rah, 2);
|
|
|
|
|
2020-06-26 21:03:17 +02:00
|
|
|
return 0;
|
2020-05-11 09:49:55 +02:00
|
|
|
}
|
2018-11-12 12:29:29 +01:00
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
static void e1000_iface_init(struct net_if *iface)
|
|
|
|
{
|
2020-05-28 21:23:02 +02:00
|
|
|
struct e1000_dev *dev = net_if_get_device(iface)->data;
|
2018-07-31 11:44:34 +02:00
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
/* For VLAN, this value is only used to get the correct L2 driver.
|
|
|
|
* The iface pointer in device context should contain the main
|
|
|
|
* interface if the VLANs are enabled.
|
|
|
|
*/
|
|
|
|
if (dev->iface == NULL) {
|
|
|
|
dev->iface = iface;
|
|
|
|
|
|
|
|
/* Do the phy link up only once */
|
|
|
|
IRQ_CONNECT(DT_INST_IRQN(0),
|
2020-03-25 17:12:19 +01:00
|
|
|
DT_INST_IRQ(0, priority),
|
2020-12-16 18:17:24 +01:00
|
|
|
e1000_isr, DEVICE_DT_INST_GET(0),
|
2020-03-25 17:12:19 +01:00
|
|
|
DT_INST_IRQ(0, sense));
|
2018-07-31 11:44:34 +02:00
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
irq_enable(DT_INST_IRQN(0));
|
|
|
|
iow32(dev, CTRL, CTRL_SLU); /* Set link up */
|
|
|
|
iow32(dev, RCTL, RCTL_EN | RCTL_MPE);
|
|
|
|
}
|
2018-07-31 11:44:34 +02:00
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
ethernet_init(iface);
|
2018-11-12 13:31:15 +01:00
|
|
|
|
2020-05-11 09:49:55 +02:00
|
|
|
net_if_set_link_addr(iface, dev->mac, sizeof(dev->mac),
|
|
|
|
NET_LINK_ETHERNET);
|
2018-11-12 13:31:15 +01:00
|
|
|
|
2018-11-12 13:39:34 +01:00
|
|
|
LOG_DBG("done");
|
2018-07-31 11:44:34 +02:00
|
|
|
}
|
|
|
|
|
2019-04-19 00:47:36 +02:00
|
|
|
static struct e1000_dev e1000_dev;
|
2018-07-31 11:44:34 +02:00
|
|
|
|
|
|
|
static const struct ethernet_api e1000_api = {
|
2020-05-11 09:49:55 +02:00
|
|
|
.iface_api.init = e1000_iface_init,
|
2020-05-15 14:29:51 +02:00
|
|
|
#if defined(CONFIG_ETH_E1000_PTP_CLOCK)
|
|
|
|
.get_ptp_clock = e1000_get_ptp_clock,
|
|
|
|
#endif
|
2018-07-31 11:44:34 +02:00
|
|
|
.get_capabilities = e1000_caps,
|
2018-06-26 14:51:05 +02:00
|
|
|
.send = e1000_send,
|
2018-07-31 11:44:34 +02:00
|
|
|
};
|
|
|
|
|
2020-12-16 18:17:24 +01:00
|
|
|
ETH_NET_DEVICE_DT_INST_DEFINE(0,
|
2020-05-11 09:49:55 +02:00
|
|
|
e1000_probe,
|
2021-04-28 10:43:35 +02:00
|
|
|
NULL,
|
2020-05-11 09:49:55 +02:00
|
|
|
&e1000_dev,
|
|
|
|
NULL,
|
|
|
|
CONFIG_ETH_INIT_PRIORITY,
|
|
|
|
&e1000_api,
|
|
|
|
NET_ETH_MTU);
|
2020-05-15 14:29:51 +02:00
|
|
|
|
|
|
|
#if defined(CONFIG_ETH_E1000_PTP_CLOCK)
|
|
|
|
struct ptp_context {
|
|
|
|
struct e1000_dev *eth_context;
|
|
|
|
|
|
|
|
/* Simulate the clock. This is only for testing.
|
|
|
|
* The value is in nanoseconds
|
|
|
|
*/
|
|
|
|
uint64_t clock_time;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct ptp_context ptp_e1000_context;
|
|
|
|
|
|
|
|
static int ptp_clock_e1000_set(const struct device *dev,
|
|
|
|
struct net_ptp_time *tm)
|
|
|
|
{
|
|
|
|
struct ptp_context *ptp_context = dev->data;
|
|
|
|
|
|
|
|
/* TODO: Set the clock real value here */
|
|
|
|
ptp_context->clock_time = tm->second * NSEC_PER_SEC + tm->nanosecond;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ptp_clock_e1000_get(const struct device *dev,
|
|
|
|
struct net_ptp_time *tm)
|
|
|
|
{
|
|
|
|
struct ptp_context *ptp_context = dev->data;
|
|
|
|
|
|
|
|
/* TODO: Get the clock value */
|
|
|
|
tm->second = ptp_context->clock_time / NSEC_PER_SEC;
|
|
|
|
tm->nanosecond = ptp_context->clock_time - tm->second * NSEC_PER_SEC;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ptp_clock_e1000_adjust(const struct device *dev, int increment)
|
|
|
|
{
|
|
|
|
ARG_UNUSED(dev);
|
|
|
|
ARG_UNUSED(increment);
|
|
|
|
|
|
|
|
/* TODO: Implement clock adjustment */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-01-27 09:49:20 +01:00
|
|
|
static int ptp_clock_e1000_rate_adjust(const struct device *dev, double ratio)
|
2020-05-15 14:29:51 +02:00
|
|
|
{
|
|
|
|
const int hw_inc = NSEC_PER_SEC / CONFIG_ETH_E1000_PTP_CLOCK_SRC_HZ;
|
|
|
|
struct ptp_context *ptp_context = dev->data;
|
|
|
|
struct e1000_dev *context = ptp_context->eth_context;
|
|
|
|
int corr;
|
|
|
|
int32_t mul;
|
|
|
|
float val;
|
|
|
|
|
|
|
|
/* No change needed. */
|
2021-10-03 22:53:14 +02:00
|
|
|
if (ratio == 1.0f) {
|
2020-05-15 14:29:51 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ratio *= context->clk_ratio;
|
|
|
|
|
|
|
|
/* Limit possible ratio. */
|
2021-10-03 22:53:14 +02:00
|
|
|
if ((ratio > 1.0f + 1.0f/(2 * hw_inc)) ||
|
|
|
|
(ratio < 1.0f - 1.0f/(2 * hw_inc))) {
|
2020-05-15 14:29:51 +02:00
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Save new ratio. */
|
|
|
|
context->clk_ratio = ratio;
|
|
|
|
|
2021-10-03 22:53:14 +02:00
|
|
|
if (ratio < 1.0f) {
|
2020-05-15 14:29:51 +02:00
|
|
|
corr = hw_inc - 1;
|
2021-10-03 22:53:14 +02:00
|
|
|
val = 1.0f / (hw_inc * (1.0f - ratio));
|
|
|
|
} else if (ratio > 1.0f) {
|
2020-05-15 14:29:51 +02:00
|
|
|
corr = hw_inc + 1;
|
2021-10-03 22:53:14 +02:00
|
|
|
val = 1.0f / (hw_inc * (ratio - 1.0f));
|
2020-05-15 14:29:51 +02:00
|
|
|
} else {
|
|
|
|
val = 0;
|
|
|
|
corr = hw_inc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (val >= INT32_MAX) {
|
|
|
|
/* Value is too high.
|
|
|
|
* It is not possible to adjust the rate of the clock.
|
|
|
|
*/
|
|
|
|
mul = 0;
|
|
|
|
} else {
|
|
|
|
mul = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: Adjust the clock here */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct ptp_clock_driver_api api = {
|
|
|
|
.set = ptp_clock_e1000_set,
|
|
|
|
.get = ptp_clock_e1000_get,
|
|
|
|
.adjust = ptp_clock_e1000_adjust,
|
|
|
|
.rate_adjust = ptp_clock_e1000_rate_adjust,
|
|
|
|
};
|
|
|
|
|
|
|
|
static int ptp_e1000_init(const struct device *port)
|
|
|
|
{
|
2022-08-22 10:36:10 +02:00
|
|
|
const struct device *const eth_dev = DEVICE_DT_INST_GET(0);
|
2020-05-15 14:29:51 +02:00
|
|
|
struct e1000_dev *context = eth_dev->data;
|
|
|
|
struct ptp_context *ptp_context = port->data;
|
|
|
|
|
|
|
|
context->ptp_clock = port;
|
|
|
|
ptp_context->eth_context = context;
|
|
|
|
|
|
|
|
ptp_context->clock_time = k_ticks_to_ns_floor64(k_uptime_ticks());
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DEVICE_DEFINE(e1000_ptp_clock, PTP_CLOCK_NAME, ptp_e1000_init,
|
2021-04-28 10:43:35 +02:00
|
|
|
NULL, &ptp_e1000_context, NULL, POST_KERNEL,
|
2020-05-15 14:29:51 +02:00
|
|
|
CONFIG_APPLICATION_INIT_PRIORITY, &api);
|
|
|
|
|
|
|
|
#endif /* CONFIG_ETH_E1000_PTP_CLOCK */
|