[PATCH] Add support for Fritzbox 4040

Adrian Schmutzler mail at adrianschmutzler.de
Mi Nov 28 14:35:49 CET 2018


Hallo Christian,

> -----Original Message-----
> From: franken-dev [mailto:franken-dev-bounces at freifunk.net] On Behalf Of
> Christian Dresel
> Sent: Sonntag, 25. November 2018 10:11
> To: franken-dev at freifunk.net
> Subject: [PATCH] Add support for Fritzbox 4040
> 
> Add a new target ipq40xx
> We need the following patches to use VLANs
> https://github.com/chunkeey/LEDE-
> IPQ40XX/commit/465fd7c8e131ec49563d8d7e0502fb5e13dc2124
> https://github.com/chunkeey/LEDE-
> IPQ40XX/commit/f28a3fbebe976414aa62650c25f2298e58d3c306

das kann OpenWRT nicht selber?

Grüße

Adrian

> 
> Signed-off-by: Christian Dresel <fff at chrisi01.de>
> ---
>  bsp/ipq40xx.bsp                                    |    6 +
>  bsp/ipq40xx/.config                                |   22 +
>  .../0006-ipq40xx-ipqess-ethernet-driver.patch      | 2178
> ++++++++++++++++++++
>  .../openwrt/0007-ipqess-integration.patch          |  207 ++
>  .../openwrt/0008-Enable-QinQ-on-the-switch.patch   |  106 +
>  build_patches/openwrt/0009-enable-IPQ_ESS.patch    |   23 +
>  buildscript                                        |    6 +-
>  .../files/etc/uci-defaults/50-fff-boardname        |    3 +
>  .../fff-network/ipq40xx/network.avm_fritzbox-4040  |    8 +
>  .../fff/fff-sysupgrade/files/etc/sysupgrade.sh     |    3 +
>  10 files changed, 2559 insertions(+), 3 deletions(-)
>  create mode 100644 bsp/ipq40xx.bsp
>  create mode 100644 bsp/ipq40xx/.config
>  create mode 100644 build_patches/openwrt/0006-ipq40xx-ipqess-ethernet-
> driver.patch
>  create mode 100644 build_patches/openwrt/0007-ipqess-integration.patch
>  create mode 100644 build_patches/openwrt/0008-Enable-QinQ-on-the-
> switch.patch
>  create mode 100644 build_patches/openwrt/0009-enable-IPQ_ESS.patch
>  create mode 100644 src/packages/fff/fff-
> network/ipq40xx/network.avm_fritzbox-4040
> 
> diff --git a/bsp/ipq40xx.bsp b/bsp/ipq40xx.bsp
> new file mode 100644
> index 0000000..eac5759
> --- /dev/null
> +++ b/bsp/ipq40xx.bsp
> @@ -0,0 +1,6 @@
> +machine=ipq40xx
> +chipset=ipq40xx
> +subtarget=generic
> +target=$builddir/$machine
> +images=("openwrt-${chipset}-avm_fritzbox-4040-squashfs-sysupgrade.bin"
> +        )
> diff --git a/bsp/ipq40xx/.config b/bsp/ipq40xx/.config
> new file mode 100644
> index 0000000..b923e3d
> --- /dev/null
> +++ b/bsp/ipq40xx/.config
> @@ -0,0 +1,22 @@
> +# Generated using "./buildscript config openwrt".
> +# Do no edit manually
> +#
> +CONFIG_TARGET_ipq40xx=y
> +CONFIG_TARGET_MULTI_PROFILE=y
> +CONFIG_TARGET_DEVICE_ipq40xx_DEVICE_avm_fritzbox-4040=y
> +CONFIG_TARGET_DEVICE_ipq40xx_DEVICE_avm_fritzbox-4040=y
> +CONFIG_BUSYBOX_CUSTOM=y
> +# CONFIG_TARGET_ipq806x_DEVICE_compex_wpq864 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_linksys_ea8500 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_nec_wg2600hp is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_netgear_d7800 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_netgear_r7500 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_netgear_r7500v2 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_netgear_r7800 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_qcom_ipq8064-ap148 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_qcom_ipq8064-ap148-legacy is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_qcom_ipq8064-db149 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_tplink_c2600 is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_tplink_vr2600v is not set
> +# CONFIG_TARGET_ipq806x_DEVICE_zyxel_nbg6817 is not set
> +# CONFIG_TARGET_ipq806x_Default is not set
> \ No newline at end of file
> diff --git a/build_patches/openwrt/0006-ipq40xx-ipqess-ethernet-driver.patch
> b/build_patches/openwrt/0006-ipq40xx-ipqess-ethernet-driver.patch
> new file mode 100644
> index 0000000..41a0f66
> --- /dev/null
> +++ b/build_patches/openwrt/0006-ipq40xx-ipqess-ethernet-driver.patch
> @@ -0,0 +1,2178 @@
> +From 465fd7c8e131ec49563d8d7e0502fb5e13dc2124 Mon Sep 17 00:00:00 2001
> +From: John Crispin <john at phrozen.org>
> +Date: Sun, 18 Nov 2018 20:30:48 +0100
> +Subject: [PATCH 1/2] ipq40xx: ipqess ethernet driver
> +
> +modified John Crispin's essedma replacement driver.
> +
> +Fouled-up-by: Christian Lamparter <chunkeey at gmail.com>
> +---
> + .../drivers/net/ethernet/qualcomm/ipqess.c         | 1206
> ++++++++++++++++++++
> + .../drivers/net/ethernet/qualcomm/ipqess.h         |  568 +++++++++
> + .../drivers/net/ethernet/qualcomm/ipqess_ethtool.c |  191 ++++
> + .../linux/ipq40xx/patches-4.14/999-0-ipqess.patch  |  167 +++
> + 4 files changed, 2132 insertions(+)
> + create mode 100644 target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.c
> + create mode 100644 target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.h
> + create mode 100644 target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess_ethtool.c
> + create mode 100644 target/linux/ipq40xx/patches-4.14/999-0-ipqess.patch
> +
> +diff --git a/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.c b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.c
> +new file mode 100644
> +index 0000000000..affe403fea
> +--- /dev/null
> ++++ b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.c
> +@@ -0,0 +1,1206 @@
> ++/*
> ++ * Copyright (c) 2014 - 2017, The Linux Foundation. All rights reserved.
> ++ *
> ++ * Permission to use, copy, modify, and/or distribute this software for
> ++ * any purpose with or without fee is hereby granted, provided that the
> ++ * above copyright notice and this permission notice appear in all copies.
> ++ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
> WARRANTIES
> ++ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
> OF
> ++ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE
> LIABLE FOR
> ++ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY
> DAMAGES
> ++ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
> WHETHER IN AN
> ++ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
> ARISING OUT
> ++ * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
> SOFTWARE.
> ++ */
> ++
> ++#include <linux/if_vlan.h>
> ++#include <linux/interrupt.h>
> ++#include <linux/module.h>
> ++#include <linux/of.h>
> ++#include <linux/of_net.h>
> ++#include <linux/of_mdio.h>
> ++#include <linux/of_device.h>
> ++#include <linux/platform_device.h>
> ++#include <linux/phy.h>
> ++#include <linux/skbuff.h>
> ++#include <linux/vmalloc.h>
> ++
> ++#include <net/checksum.h>
> ++#include <net/ip6_checksum.h>
> ++
> ++#include "ipqess.h"
> ++
> ++#define IPQESS_RRD_SIZE		16
> ++#define IPQESS_NEXT_IDX(X, Y)  (((X) + 1) & ((Y) - 1))
> ++#define IPQESS_TX_DMA_BUF_LEN	0x3fff
> ++
> ++static void ipqess_w32(struct ipqess *ess, u32 reg, u32 val)
> ++{
> ++	__raw_writel(val, ess->hw_addr + reg);
> ++}
> ++
> ++static u32 ipqess_r32(struct ipqess *ess, u16 reg)
> ++{
> ++	return __raw_readl(ess->hw_addr + reg);
> ++}
> ++
> ++static void ipqess_m32(struct ipqess *ess, u32 mask, u32 val, u16 reg)
> ++{
> ++	u32 _val = ipqess_r32(ess, reg);
> ++	_val &= ~mask;
> ++	_val |= val;
> ++	ipqess_w32(ess, reg, _val);
> ++}
> ++
> ++static int ipqess_tx_ring_alloc(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		u16 idx;
> ++
> ++		ess->tx_ring[i].ess = ess;
> ++		ess->tx_ring[i].idx = i * 4;
> ++		ess->tx_ring[i].count = IPQESS_TX_RING_SIZE;
> ++		ess->tx_ring[i].nq = netdev_get_tx_queue(ess->netdev, i);
> ++
> ++		ess->tx_ring[i].buf = devm_kzalloc(&ess->pdev->dev,
> ++			sizeof(struct ipqess_buf) * IPQESS_TX_RING_SIZE,
> ++			GFP_KERNEL);
> ++		if (!ess->tx_ring[i].buf) {
> ++			netdev_err(ess->netdev, "buffer alloc of tx ring failed");
> ++			return -ENOMEM;
> ++		}
> ++
> ++		ess->tx_ring[i].hw_desc = dmam_alloc_coherent(&ess->pdev-
> >dev,
> ++			sizeof(struct ipqess_tx_desc) * IPQESS_TX_RING_SIZE,
> ++			&ess->tx_ring[i].dma, GFP_KERNEL);
> ++		if (!ess->tx_ring[i].hw_desc) {
> ++			netdev_err(ess->netdev, "descriptor allocation for tx
> ring failed");
> ++			return -ENOMEM;
> ++		}
> ++
> ++		ipqess_w32(ess, IPQESS_REG_TPD_BASE_ADDR_Q(ess-
> >tx_ring[i].idx),
> ++			 (u32)ess->tx_ring[i].dma);
> ++
> ++		idx = ipqess_r32(ess, IPQESS_REG_TPD_IDX_Q(ess-
> >tx_ring[i].idx));
> ++		idx >>= IPQESS_TPD_CONS_IDX_SHIFT;
> ++		idx &= 0xffff;
> ++		ess->tx_ring[i].head = ess->tx_ring[i].tail = idx;
> ++
> ++		ipqess_m32(ess, IPQESS_TPD_PROD_IDX_MASK <<
> IPQESS_TPD_PROD_IDX_SHIFT,
> ++			 idx, IPQESS_REG_TPD_IDX_Q(ess->tx_ring[i].idx));
> ++		ipqess_w32(ess, IPQESS_REG_TX_SW_CONS_IDX_Q(ess-
> >tx_ring[i].idx), idx);
> ++		ipqess_w32(ess, IPQESS_REG_TPD_RING_SIZE,
> IPQESS_TX_RING_SIZE);
> ++	}
> ++
> ++	ipqess_m32(ess, 0, BIT(IPQESS_LOAD_PTR_SHIFT),
> ++		 IPQESS_REG_TX_SRAM_PART);
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_tx_unmap_and_free(struct device *dev, struct ipqess_buf
> *buf)
> ++{
> ++	int len = 0;
> ++
> ++	if (buf->flags & IPQESS_DESC_SINGLE)
> ++		dma_unmap_single(dev, buf->dma,	buf->length,
> DMA_TO_DEVICE);
> ++	else if (buf->flags & IPQESS_DESC_PAGE)
> ++		dma_unmap_page(dev, buf->dma, buf->length,
> DMA_TO_DEVICE);
> ++
> ++	if (buf->flags & IPQESS_DESC_LAST) {
> ++		len = buf->skb->len;
> ++		dev_kfree_skb_any(buf->skb);
> ++	}
> ++
> ++	buf->flags = 0;
> ++
> ++	return len;
> ++}
> ++
> ++static void ipqess_tx_ring_free(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		int j;
> ++
> ++		if (ess->tx_ring[i].hw_desc)
> ++			continue;
> ++
> ++		for (j = 0; j < IPQESS_TX_RING_SIZE; j++) {
> ++			struct ipqess_buf *buf = &ess->tx_ring[i].buf[j];
> ++
> ++			ipqess_tx_unmap_and_free(&ess->pdev->dev, buf);
> ++		}
> ++
> ++		ess->tx_ring[i].buf = NULL;
> ++	}
> ++}
> ++
> ++static int ipqess_rx_buf_prepare(struct ipqess_buf *buf,
> ++	struct ipqess_rx_ring *rx_ring)
> ++{
> ++	buf->dma = dma_map_single(&rx_ring->ess->pdev->dev, buf->skb-
> >data,
> ++				  IPQESS_RX_HEAD_BUFF_SIZE,
> DMA_FROM_DEVICE);
> ++	if (dma_mapping_error(&rx_ring->ess->pdev->dev, buf->dma)) {
> ++		WARN_ONCE(0, "IPQESS DMA mapping failed for linear address
> %x", buf->dma);
> ++		dev_kfree_skb_any(buf->skb);
> ++		buf->skb = NULL;
> ++		/* TODO: which return code */
> ++		return -EINVAL;
> ++	}
> ++
> ++	buf->length = IPQESS_RX_HEAD_BUFF_SIZE;
> ++	rx_ring->hw_desc[rx_ring->head] = (void *)buf->dma;
> ++	rx_ring->head = (rx_ring->head + 1) % IPQESS_RX_RING_SIZE;
> ++
> ++	ipqess_m32(rx_ring->ess, IPQESS_RFD_PROD_IDX_BITS,
> ++		 (rx_ring->head + IPQESS_RX_RING_SIZE - 1) %
> IPQESS_RX_RING_SIZE,
> ++		 IPQESS_REG_RFD_IDX_Q(rx_ring->idx));
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_rx_buf_alloc_napi(struct ipqess_rx_ring *rx_ring)
> ++{
> ++	struct ipqess_buf *buf = &rx_ring->buf[rx_ring->head];
> ++
> ++	buf->skb = napi_alloc_skb(&rx_ring->napi_rx,
> ++		IPQESS_RX_HEAD_BUFF_SIZE);
> ++	if (!buf->skb)
> ++		return -ENOMEM;
> ++	return ipqess_rx_buf_prepare(buf, rx_ring);
> ++}
> ++
> ++static int ipqess_rx_buf_alloc(struct ipqess_rx_ring *rx_ring)
> ++{
> ++	struct ipqess_buf *buf = &rx_ring->buf[rx_ring->head];
> ++
> ++	buf->skb = netdev_alloc_skb_ip_align(rx_ring->ess->netdev,
> ++		IPQESS_RX_HEAD_BUFF_SIZE);
> ++	if (!buf->skb)
> ++		return -ENOMEM;
> ++	return ipqess_rx_buf_prepare(buf, rx_ring);
> ++}
> ++
> ++static void ipqess_refill_work(struct work_struct *work)
> ++{
> ++	struct ipqess_rx_ring *rx_ring = container_of(work,
> ++		struct ipqess_rx_ring, refill_work);
> ++	int refill = 0;
> ++
> ++	/* don't let this loop by accident. */
> ++	while (atomic_dec_and_test(&rx_ring->refill_count)) {
> ++		if (ipqess_rx_buf_alloc(rx_ring)) {
> ++			refill++;
> ++			dev_dbg(&rx_ring->ess->pdev->dev,
> ++         			"Not all buffers was reallocated");
> ++		}
> ++	}
> ++
> ++	if (refill || atomic_read(&rx_ring->refill_count)) {
> ++		atomic_add(refill, &rx_ring->refill_count);
> ++		schedule_work(&rx_ring->refill_work);
> ++	}
> ++}
> ++
> ++static int ipqess_rx_ring_alloc(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		int j;
> ++
> ++		ess->rx_ring[i].ess = ess;
> ++		ess->rx_ring[i].idx = i;
> ++
> ++		ess->rx_ring[i].buf = devm_kzalloc(&ess->pdev->dev,
> ++			sizeof(struct ipqess_buf) * IPQESS_RX_RING_SIZE,
> ++			GFP_KERNEL);
> ++		if (!ess->rx_ring[i].buf)
> ++			return -ENOMEM;
> ++
> ++		ess->rx_ring[i].hw_desc = dmam_alloc_coherent(&ess->pdev-
> >dev,
> ++			sizeof(struct ipqess_rx_desc) * IPQESS_RX_RING_SIZE,
> ++			&ess->rx_ring[i].dma, GFP_KERNEL);
> ++		if (!ess->rx_ring[i].hw_desc)
> ++			return -ENOMEM;
> ++
> ++		for (j = 0; j < IPQESS_RX_RING_SIZE; j++)
> ++			if (ipqess_rx_buf_alloc(&ess->rx_ring[i]) < 0)
> ++				return -ENOMEM;
> ++
> ++		INIT_WORK(&ess->rx_ring[i].refill_work, ipqess_refill_work);
> ++
> ++		ipqess_w32(ess, IPQESS_REG_RFD_BASE_ADDR_Q(i),
> ++			 (u32)(ess->rx_ring[i].dma));
> ++	}
> ++
> ++	ipqess_w32(ess, IPQESS_REG_RX_DESC0,
> ++		 (IPQESS_RX_HEAD_BUFF_SIZE << IPQESS_RX_BUF_SIZE_SHIFT)
> |
> ++		 (IPQESS_RX_RING_SIZE << IPQESS_RFD_RING_SIZE_SHIFT));
> ++
> ++	return 0;
> ++}
> ++
> ++static void ipqess_rx_ring_free(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	for (i = 0, i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		int j;
> ++
> ++		for (j = 0; j < IPQESS_RX_RING_SIZE; j++) {
> ++			dma_unmap_single(&ess->pdev->dev,
> ++					 ess->rx_ring[i].buf[j].dma,
> ++					 ess->rx_ring[i].buf[j].length,
> ++					 DMA_FROM_DEVICE);
> ++			dev_kfree_skb_any(ess->rx_ring[i].buf[j].skb);
> ++		}
> ++
> ++		atomic_set(&ess->rx_ring[i].refill_count, 0);
> ++		cancel_work_sync(&ess->rx_ring[i].refill_work);
> ++	}
> ++}
> ++
> ++static struct net_device_stats *ipqess_get_stats(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	uint32_t *p;
> ++	int i;
> ++	u32 stat;
> ++
> ++	spin_lock(&ess->stats_lock);
> ++	p = (uint32_t *)&(ess->ipqessstats);
> ++
> ++	for (i = 0; i < IPQESS_MAX_TX_QUEUE; i++) {
> ++		stat = ipqess_r32(ess, IPQESS_REG_TX_STAT_PKT_Q(i));
> ++		*p += stat;
> ++		p++;
> ++	}
> ++
> ++	for (i = 0; i < IPQESS_MAX_TX_QUEUE; i++) {
> ++		stat = ipqess_r32(ess, IPQESS_REG_TX_STAT_BYTE_Q(i));
> ++		*p += stat;
> ++		p++;
> ++	}
> ++
> ++	for (i = 0; i < IPQESS_MAX_RX_QUEUE; i++) {
> ++		stat = ipqess_r32(ess, IPQESS_REG_RX_STAT_PKT_Q(i));
> ++		*p += stat;
> ++		p++;
> ++	}
> ++
> ++	for (i = 0; i < IPQESS_MAX_RX_QUEUE; i++) {
> ++		stat = ipqess_r32(ess, IPQESS_REG_RX_STAT_BYTE_Q(i));
> ++		*p += stat;
> ++		p++;
> ++	}
> ++
> ++	spin_unlock(&ess->stats_lock);
> ++	return &ess->stats;
> ++}
> ++
> ++static void ipqess_adjust_link(struct net_device *netdev)
> ++{
> ++	if (netdev->phydev->link)
> ++		netif_carrier_on(netdev);
> ++	else
> ++		netif_carrier_off(netdev);
> ++}
> ++
> ++static int ipqess_phy_connect_node(struct ipqess *ess,
> ++				 struct device_node *phy_node)
> ++{
> ++	struct phy_device *phydev;
> ++	int phy_mode;
> ++
> ++	phy_mode = of_get_phy_mode(phy_node);
> ++	if (phy_mode < 0) {
> ++		dev_err(&ess->netdev->dev, "incorrect phy-mode %d\n",
> phy_mode);
> ++		return -EINVAL;
> ++	}
> ++
> ++	phydev = of_phy_connect(ess->netdev, phy_node,
> ++				ipqess_adjust_link, 0, phy_mode);
> ++	if (!phydev) {
> ++		dev_err(&ess->netdev->dev, "could not connect to PHY\n");
> ++		return -ENODEV;
> ++	}
> ++
> ++	dev_info(&ess->netdev->dev,
> ++		 "connected to PHY at %s [uid=%08x, driver=%s]\n",
> ++		 phydev_name(phydev), phydev->phy_id,
> ++		 phydev->drv->name);
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_phy_connect(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	struct device_node *np;
> ++
> ++	np = of_parse_phandle(ess->of_node, "phy-handle", 0);
> ++	if (!np && of_phy_is_fixed_link(ess->of_node))
> ++		if (!of_phy_register_fixed_link(ess->of_node))
> ++			np = of_node_get(ess->of_node);
> ++	if (!np)
> ++		return -ENODEV;
> ++
> ++	switch (of_get_phy_mode(np)) {
> ++	case PHY_INTERFACE_MODE_SGMII:
> ++		break;
> ++	default:
> ++		goto err_phy;
> ++	}
> ++
> ++	if (ipqess_phy_connect_node(ess, np))
> ++		goto err_phy;
> ++
> ++	netdev->phydev->autoneg = AUTONEG_ENABLE;
> ++	netdev->phydev->speed = 0;
> ++	netdev->phydev->duplex = 0;
> ++
> ++	if (of_phy_is_fixed_link(ess->of_node))
> ++		netdev->phydev->supported |=
> ++			SUPPORTED_Pause | SUPPORTED_Asym_Pause;
> ++
> ++	netdev->phydev->supported &= PHY_GBIT_FEATURES |
> SUPPORTED_Pause |
> ++			SUPPORTED_Asym_Pause;
> ++	netdev->phydev->advertising = netdev->phydev->supported |
> ++		ADVERTISED_Autoneg;
> ++	phy_start_aneg(netdev->phydev);
> ++	of_node_put(np);
> ++
> ++	return 0;
> ++
> ++err_phy:
> ++	if (of_phy_is_fixed_link(ess->of_node))
> ++		of_phy_deregister_fixed_link(ess->of_node);
> ++	of_node_put(np);
> ++	dev_err(&netdev->dev, "%s: invalid phy\n", __func__);
> ++
> ++	return -EINVAL;
> ++}
> ++
> ++static int ipqess_rx_poll(struct ipqess_rx_ring *rx_ring, int budget)
> ++{
> ++	u32 length = 0, num_desc, tail;
> ++	int done = 0;
> ++
> ++	tail = ipqess_r32(rx_ring->ess, IPQESS_REG_RFD_IDX_Q(rx_ring->idx));
> ++	tail >>= IPQESS_RFD_CONS_IDX_SHIFT;
> ++	tail &= IPQESS_RFD_CONS_IDX_MASK;
> ++
> ++	while (done < budget) {
> ++		struct sk_buff *skb;
> ++		struct ipqess_rx_desc *rd;
> ++
> ++		if (rx_ring->tail == tail)
> ++			break;
> ++
> ++		skb = rx_ring->buf[rx_ring->tail].skb;
> ++		dma_unmap_single(&rx_ring->ess->pdev->dev,
> ++				 rx_ring->buf[rx_ring->tail].dma,
> ++				 rx_ring->buf[rx_ring->tail].length,
> ++				 DMA_FROM_DEVICE);
> ++		rd = (struct ipqess_rx_desc *)skb->data;
> ++		num_desc = rd->rrd1 & IPQESS_RRD_NUM_RFD_MASK;
> ++		length = rd->rrd6 & IPQESS_RRD_PKT_SIZE_MASK;
> ++		rx_ring->tail = IPQESS_NEXT_IDX(rx_ring->tail,
> IPQESS_RX_RING_SIZE);
> ++
> ++		skb_reserve(skb, IPQESS_RRD_SIZE);
> ++		if (num_desc > 1) {
> ++			/* can we use build_skb here ? */
> ++			struct sk_buff *skb_prev = NULL;
> ++			int size_remaining;
> ++			int i;
> ++
> ++			skb->data_len = 0;
> ++			skb->tail += (IPQESS_RX_HEAD_BUFF_SIZE -
> IPQESS_RRD_SIZE);
> ++			skb->len = skb->truesize = length;
> ++			size_remaining = length - (IPQESS_RX_HEAD_BUFF_SIZE
> - IPQESS_RRD_SIZE);
> ++
> ++			for (i = 1; i < num_desc; i++) {
> ++				/* TODO: use build_skb ? */
> ++				struct sk_buff *skb_temp = rx_ring-
> >buf[rx_ring->tail].skb;
> ++
> ++				dma_unmap_single(&rx_ring->ess->pdev->dev,
> ++						 rx_ring->buf[rx_ring->tail].dma,
> ++						 rx_ring->buf[rx_ring-
> >tail].length,
> ++						 DMA_FROM_DEVICE);
> ++
> ++				skb_put(skb_temp, min(size_remaining,
> IPQESS_RX_HEAD_BUFF_SIZE));
> ++				if (skb_prev)
> ++					skb_prev->next = rx_ring->buf[rx_ring-
> >tail].skb;
> ++				else
> ++					skb_shinfo(skb)->frag_list = rx_ring-
> >buf[rx_ring->tail].skb;
> ++				skb_prev = rx_ring->buf[rx_ring->tail].skb;
> ++				rx_ring->buf[rx_ring->tail].skb->next = NULL;
> ++
> ++				skb->data_len += rx_ring->buf[rx_ring->tail].skb-
> >len;
> ++				size_remaining -= rx_ring->buf[rx_ring-
> >tail].skb->len;
> ++
> ++				rx_ring->tail = IPQESS_NEXT_IDX(rx_ring->tail,
> IPQESS_RX_RING_SIZE);
> ++			}
> ++
> ++		} else {
> ++			skb_put(skb, length);
> ++		}
> ++
> ++		skb->dev = rx_ring->ess->netdev;
> ++		skb->protocol = eth_type_trans(skb, rx_ring->ess->netdev);
> ++		skb_record_rx_queue(skb, rx_ring->idx);
> ++
> ++		if (rd->rrd6 & IPQESS_RRD_CSUM_FAIL_MASK)
> ++			skb_checksum_none_assert(skb);
> ++		else
> ++			skb->ip_summed = CHECKSUM_UNNECESSARY;
> ++
> ++		if (rd->rrd7 & IPQESS_RRD_CVLAN) {
> ++			__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021Q), rd-
> >rrd4);
> ++		} else if (rd->rrd1 & IPQESS_RRD_SVLAN) {
> ++			__vlan_hwaccel_put_tag(skb, htons(ETH_P_8021AD),
> rd->rrd4);
> ++		}
> ++		napi_gro_receive(&rx_ring->napi_rx, skb);
> ++
> ++		/* TODO: do we need to have these here ? */
> ++		rx_ring->ess->stats.rx_packets++;
> ++		rx_ring->ess->stats.rx_bytes += length;
> ++
> ++		done++;
> ++		while (num_desc) {
> ++			if (ipqess_rx_buf_alloc_napi(rx_ring)) {
> ++				schedule_work(&rx_ring->refill_work);
> ++				atomic_add(num_desc, &rx_ring->refill_count);
> ++				break;
> ++			}
> ++			num_desc--;
> ++		}
> ++
> ++	}
> ++
> ++	ipqess_w32(rx_ring->ess, IPQESS_REG_RX_SW_CONS_IDX_Q(rx_ring-
> >idx),
> ++		 rx_ring->tail);
> ++
> ++	return done;
> ++}
> ++
> ++static int ipqess_tx_complete(struct ipqess_tx_ring *tx_ring, int budget)
> ++{
> ++	u32 tail;
> ++	int done = 0;
> ++	int total = 0, ret;
> ++
> ++	tail = ipqess_r32(tx_ring->ess, IPQESS_REG_TPD_IDX_Q(tx_ring->idx));
> ++	tail >>= IPQESS_TPD_CONS_IDX_SHIFT;
> ++	tail &= IPQESS_TPD_CONS_IDX_MASK;
> ++
> ++	while ((tx_ring->tail != tail) && (done < budget)) {
> ++		//pr_info("freeing txq:%d tail:%d tailbuf:%p\n", tx_ring->idx,
> tx_ring->tail, &tx_ring->buf[tx_ring->tail]);
> ++		ret = ipqess_tx_unmap_and_free(&tx_ring->ess->pdev->dev,
> ++				       &tx_ring->buf[tx_ring->tail]);
> ++		tx_ring->tail = IPQESS_NEXT_IDX(tx_ring->tail, tx_ring->count);
> ++		if (ret) {
> ++			total += ret;
> ++			done++;
> ++		}
> ++	}
> ++
> ++	ipqess_w32(tx_ring->ess,
> ++		 IPQESS_REG_TX_SW_CONS_IDX_Q(tx_ring->idx),
> ++		 tx_ring->tail);
> ++
> ++	if (netif_tx_queue_stopped(tx_ring->nq)) {
> ++		printk("S %d\n", tx_ring->idx);
> ++		netif_tx_wake_queue(tx_ring->nq);
> ++	}
> ++
> ++	netdev_tx_completed_queue(tx_ring->nq, done, total);
> ++
> ++	return done;
> ++}
> ++
> ++static int ipqess_tx_napi(struct napi_struct *napi, int budget)
> ++{
> ++	struct ipqess_tx_ring *tx_ring = container_of(napi, struct
> ipqess_tx_ring,
> ++						    napi_tx);
> ++	u32 reg_data;
> ++	u32 shadow_tx_status;
> ++	int work_done = 0;
> ++	struct queue *queue = &tx_ring->ess->queue[tx_ring->idx / 4];
> ++
> ++	reg_data = ipqess_r32(tx_ring->ess, IPQESS_REG_TX_ISR);
> ++	queue->tx_status |= reg_data & BIT(tx_ring->idx);
> ++	shadow_tx_status = queue->tx_status;
> ++
> ++	work_done = ipqess_tx_complete(tx_ring, budget);
> ++
> ++	ipqess_w32(tx_ring->ess, IPQESS_REG_TX_ISR, shadow_tx_status);
> ++
> ++	if (likely(work_done < budget)) {
> ++		napi_complete(napi);
> ++		ipqess_w32(tx_ring->ess,
> IPQESS_REG_TX_INT_MASK_Q(tx_ring->idx), 0x1);
> ++	}
> ++
> ++	return work_done;
> ++}
> ++
> ++static int ipqess_rx_napi(struct napi_struct *napi, int budget)
> ++{
> ++	struct ipqess_rx_ring *rx_ring = container_of(napi, struct
> ipqess_rx_ring,
> ++						    napi_rx);
> ++	struct ipqess *ess = rx_ring->ess;
> ++	int remain_budget = budget;
> ++	int rx_done;
> ++	u32 rx_mask = BIT(rx_ring->idx << IPQESS_RX_PER_CPU_MASK_SHIFT);
> ++	u32 status;
> ++
> ++poll_again:
> ++	ipqess_w32(ess, IPQESS_REG_RX_ISR, rx_mask);
> ++	rx_done = ipqess_rx_poll(rx_ring, remain_budget);
> ++
> ++	if (rx_done == remain_budget)
> ++		return budget;
> ++
> ++	status = ipqess_r32(ess, IPQESS_REG_RX_ISR);
> ++	if (status & rx_mask) {
> ++		remain_budget -= rx_done;
> ++		goto poll_again;
> ++	}
> ++
> ++	napi_complete(napi);
> ++	ipqess_w32(ess, IPQESS_REG_RX_INT_MASK_Q(rx_ring->idx), 0x1);
> ++
> ++	return rx_done + budget - remain_budget;
> ++}
> ++
> ++static irqreturn_t ipqess_interrupt_tx(int irq, void *priv)
> ++{
> ++	struct ipqess_tx_ring *tx_ring = (struct ipqess_tx_ring *) priv;
> ++
> ++	if (likely(napi_schedule_prep(&tx_ring->napi_tx))) {
> ++		__napi_schedule(&tx_ring->napi_tx);
> ++		ipqess_w32(tx_ring->ess,
> ++			 IPQESS_REG_TX_INT_MASK_Q(tx_ring->idx),
> ++			 0x0);
> ++	}
> ++
> ++	return IRQ_HANDLED;
> ++}
> ++
> ++static irqreturn_t ipqess_interrupt_rx(int irq, void *priv)
> ++{
> ++	struct ipqess_rx_ring *rx_ring = (struct ipqess_rx_ring *) priv;
> ++
> ++	if (likely(napi_schedule_prep(&rx_ring->napi_rx))) {
> ++		__napi_schedule(&rx_ring->napi_rx);
> ++		ipqess_w32(rx_ring->ess,
> ++			 IPQESS_REG_RX_INT_MASK_Q(rx_ring->idx),
> ++			 0x0);
> ++	}
> ++
> ++	return IRQ_HANDLED;
> ++}
> ++
> ++static void ipqess_irq_enable(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	ipqess_w32(ess, IPQESS_REG_RX_ISR, 0xff);
> ++	ipqess_w32(ess, IPQESS_REG_TX_ISR, 0xffff);
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		ipqess_w32(ess, IPQESS_REG_RX_INT_MASK_Q(i), 1);
> ++		ipqess_w32(ess, IPQESS_REG_TX_INT_MASK_Q(i), 1);
> ++	}
> ++}
> ++
> ++static void ipqess_irq_disable(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		ipqess_w32(ess, IPQESS_REG_RX_INT_MASK_Q(i), 0);
> ++		ipqess_w32(ess, IPQESS_REG_TX_INT_MASK_Q(i), 0);
> ++	}
> ++}
> ++
> ++static int __init ipqess_init(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	const char *mac_addr;
> ++
> ++	mac_addr = of_get_mac_address(ess->of_node);
> ++	if (mac_addr)
> ++		ether_addr_copy(netdev->dev_addr, mac_addr);
> ++	if (!is_valid_ether_addr(netdev->dev_addr)) {
> ++		random_ether_addr(netdev->dev_addr);
> ++		dev_err(&netdev->dev, "generated random MAC address
> %pM\n",
> ++			netdev->dev_addr);
> ++		netdev->addr_assign_type = NET_ADDR_RANDOM;
> ++	}
> ++
> ++	return ipqess_phy_connect(netdev);
> ++}
> ++
> ++static void ipqess_uninit(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++
> ++	phy_disconnect(netdev->phydev);
> ++	if (of_phy_is_fixed_link(ess->of_node))
> ++		of_phy_deregister_fixed_link(ess->of_node);
> ++}
> ++
> ++static int ipqess_open(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	int i;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		napi_enable(&ess->tx_ring[i].napi_tx);
> ++		napi_enable(&ess->rx_ring[i].napi_rx);
> ++	}
> ++	ipqess_irq_enable(ess);
> ++	phy_start(ess->netdev->phydev);
> ++	netif_tx_start_all_queues(netdev);
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_stop(struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	int i;
> ++
> ++	netif_tx_stop_all_queues(netdev);
> ++	phy_stop(netdev->phydev);
> ++	ipqess_irq_disable(ess);
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		napi_disable(&ess->tx_ring[i].napi_tx);
> ++		napi_disable(&ess->rx_ring[i].napi_rx);
> ++	}
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_do_ioctl(struct net_device *netdev, struct ifreq *ifr, int cmd)
> ++{
> ++	switch (cmd) {
> ++	case SIOCGMIIPHY:
> ++	case SIOCGMIIREG:
> ++	case SIOCSMIIREG:
> ++		return phy_mii_ioctl(netdev->phydev, ifr, cmd);
> ++	default:
> ++		break;
> ++	}
> ++
> ++	return -EOPNOTSUPP;
> ++}
> ++
> ++
> ++static inline u16 ipqess_tx_desc_available(struct ipqess_tx_ring *tx_ring)
> ++{
> ++	u16 count = 0;
> ++
> ++	if (tx_ring->tail <= tx_ring->head)
> ++		count = IPQESS_TX_RING_SIZE;
> ++
> ++	count += tx_ring->tail - tx_ring->head - 1;
> ++
> ++	return count;
> ++}
> ++
> ++static inline int ipqess_cal_txd_req(struct sk_buff *skb)
> ++{
> ++	int i, nfrags;
> ++	struct skb_frag_struct *frag;
> ++
> ++	nfrags = 1;
> ++	if (skb_is_gso(skb)) {
> ++		for (i = 0; i < skb_shinfo(skb)->nr_frags; i++) {
> ++			frag = &skb_shinfo(skb)->frags[i];
> ++			nfrags += DIV_ROUND_UP(frag->size,
> IPQESS_TX_DMA_BUF_LEN);
> ++		}
> ++	} else {
> ++		nfrags += skb_shinfo(skb)->nr_frags;
> ++	}
> ++
> ++	return nfrags; // DIV_ROUND_UP(nfrags, 2);
> ++}
> ++
> ++static struct ipqess_buf *ipqess_get_tx_buffer(struct ipqess_tx_ring
> *tx_ring,
> ++					       struct ipqess_tx_desc *desc)
> ++{
> ++	return &tx_ring->buf[desc - (struct ipqess_tx_desc *)tx_ring-
> >hw_desc];
> ++}
> ++
> ++static struct ipqess_tx_desc *ipqess_tx_desc_next(struct ipqess_tx_ring
> *tx_ring)
> ++{
> ++	struct ipqess_tx_desc *desc;
> ++
> ++	desc = (&((struct ipqess_tx_desc *)(tx_ring->hw_desc))[tx_ring-
> >head]);
> ++	tx_ring->head = IPQESS_NEXT_IDX(tx_ring->head, tx_ring->count);
> ++
> ++	return desc;
> ++}
> ++
> ++static void ipqess_rollback_tx(struct ipqess *eth,
> ++			    struct ipqess_tx_desc *first_desc, int queue_id)
> ++{
> ++	struct ipqess_tx_ring *tx_ring = &eth->tx_ring[queue_id / 4];
> ++	struct ipqess_buf *buf;
> ++	struct ipqess_tx_desc *desc = NULL;
> ++	u16 start_index, index;
> ++
> ++	start_index = first_desc - (struct ipqess_tx_desc *)(tx_ring->hw_desc);
> ++
> ++	index = start_index;
> ++	while (index != tx_ring->head) {
> ++		desc = (&((struct ipqess_tx_desc *)(tx_ring->hw_desc))[index]);
> ++		buf = &tx_ring->buf[index];
> ++		ipqess_tx_unmap_and_free(&eth->pdev->dev, buf);
> ++		memset(desc, 0, sizeof(struct ipqess_tx_desc));
> ++		if (++index == tx_ring->count)
> ++			index = 0;
> ++	}
> ++	tx_ring->head = start_index;
> ++}
> ++
> ++static int ipqess_tx_map_and_fill(struct ipqess_tx_ring *tx_ring, struct
> sk_buff *skb)
> ++{
> ++	struct ipqess_buf *buf = NULL;
> ++	struct platform_device *pdev = tx_ring->ess->pdev;
> ++	struct ipqess_tx_desc *desc = NULL, *first_desc = NULL;
> ++	u32 word1 = 0, word3 = 0, lso_word1 = 0, svlan_tag = 0;
> ++	u16 len, lso_len = 0;
> ++	int i = 0;
> ++
> ++	if (skb_is_gso(skb)) {
> ++		if (skb_shinfo(skb)->gso_type & SKB_GSO_TCPV4) {
> ++			lso_word1 |= IPQESS_TPD_IPV4_EN;
> ++			ip_hdr(skb)->check = 0;
> ++			tcp_hdr(skb)->check =
> ~csum_tcpudp_magic(ip_hdr(skb)->saddr,
> ++				ip_hdr(skb)->daddr, 0, IPPROTO_TCP, 0);
> ++		} else if (skb_shinfo(skb)->gso_type & SKB_GSO_TCPV6) {
> ++			lso_word1 |= IPQESS_TPD_LSO_V2_EN;
> ++			ipv6_hdr(skb)->payload_len = 0;
> ++			tcp_hdr(skb)->check =
> ~csum_ipv6_magic(&ipv6_hdr(skb)->saddr,
> ++				&ipv6_hdr(skb)->daddr, 0, IPPROTO_TCP, 0);
> ++		}
> ++
> ++		lso_word1 |= IPQESS_TPD_LSO_EN |
> ++			     ((skb_shinfo(skb)->gso_size &
> IPQESS_TPD_MSS_MASK) << IPQESS_TPD_MSS_SHIFT) |
> ++			     (skb_transport_offset(skb) <<
> IPQESS_TPD_HDR_SHIFT);
> ++	} else if (likely(skb->ip_summed == CHECKSUM_PARTIAL)) {
> ++			u8 css, cso;
> ++			cso = skb_checksum_start_offset(skb);
> ++			css = cso + skb->csum_offset;
> ++
> ++			word1 |= (IPQESS_TPD_CUSTOM_CSUM_EN);
> ++			word1 |= (cso >> 1) << IPQESS_TPD_HDR_SHIFT;
> ++			word1 |= ((css >> 1) <<
> IPQESS_TPD_CUSTOM_CSUM_SHIFT);
> ++	}
> ++
> ++	if (skb_vlan_tag_present(skb)) {
> ++		switch (skb->vlan_proto) {
> ++		case htons(ETH_P_8021Q):
> ++			word3 |= BIT(IPQESS_TX_INS_CVLAN);
> ++			word3 |= skb_vlan_tag_get(skb) <<
> IPQESS_TX_CVLAN_TAG_SHIFT;
> ++			break;
> ++		case htons(ETH_P_8021AD):
> ++			word1 |= BIT(IPQESS_TX_INS_SVLAN);
> ++			svlan_tag = skb_vlan_tag_get(skb) <<
> IPQESS_TX_SVLAN_TAG_SHIFT;
> ++			break;
> ++		default:
> ++			dev_err(&pdev->dev, "no ctag or stag present\n");
> ++			goto vlan_tag_error;
> ++		}
> ++	}
> ++
> ++        if (skb->protocol == htons(ETH_P_PPP_SES))
> ++                word1 |= IPQESS_TPD_PPPOE_EN;
> ++
> ++	word3 |= 0x3e << IPQESS_TPD_PORT_BITMAP_SHIFT;
> ++	len = skb_headlen(skb);
> ++
> ++	first_desc = desc = ipqess_tx_desc_next(tx_ring);
> ++	if (lso_word1 & IPQESS_TPD_LSO_V2_EN) {
> ++		desc->addr = cpu_to_le16(skb->len);
> ++		desc->word1 = word1 | lso_word1;
> ++		desc->svlan_tag = svlan_tag;
> ++		desc->word3 = word3;
> ++		desc = ipqess_tx_desc_next(tx_ring);
> ++	}
> ++
> ++	buf = ipqess_get_tx_buffer(tx_ring, desc);
> ++	if (lso_word1)
> ++		buf->length = lso_len;
> ++	else
> ++		buf->length = len;
> ++	buf->dma = dma_map_single(&pdev->dev,
> ++				skb->data, len, DMA_TO_DEVICE);
> ++	if (dma_mapping_error(&pdev->dev, buf->dma))
> ++		goto dma_error;
> ++
> ++	desc->addr = cpu_to_le32(buf->dma);
> ++	desc->len  = cpu_to_le16(len);
> ++
> ++	buf->flags |= IPQESS_DESC_SINGLE;
> ++	desc->word1 = word1 | lso_word1;
> ++	desc->svlan_tag = svlan_tag;
> ++	desc->word3 = word3;
> ++
> ++	while (i < skb_shinfo(skb)->nr_frags) {
> ++		skb_frag_t *frag = &skb_shinfo(skb)->frags[i];
> ++		len = skb_frag_size(frag);
> ++		desc = ipqess_tx_desc_next(tx_ring);
> ++		buf = ipqess_get_tx_buffer(tx_ring, desc);
> ++		buf->length = len;
> ++		buf->flags |= IPQESS_DESC_PAGE;
> ++		buf->dma = skb_frag_dma_map(&pdev->dev, frag, 0, len,
> DMA_TO_DEVICE);
> ++		if (dma_mapping_error(NULL, buf->dma))
> ++			goto dma_error;
> ++
> ++		desc->addr = cpu_to_le32(buf->dma);
> ++		desc->len  = cpu_to_le16(len);
> ++		desc->svlan_tag = svlan_tag;
> ++		desc->word1 = word1 | lso_word1;
> ++		desc->word3 = word3;
> ++		i++;
> ++	}
> ++	desc->word1 |= 1 << IPQESS_TPD_EOP_SHIFT;
> ++	buf->skb = skb;
> ++	buf->flags |= IPQESS_DESC_LAST;
> ++
> ++	return 0;
> ++
> ++dma_error:
> ++	ipqess_rollback_tx(tx_ring->ess, first_desc, tx_ring->idx);
> ++	dev_err(&pdev->dev, "TX DMA map failed\n");
> ++vlan_tag_error:
> ++	return -ENOMEM;
> ++}
> ++
> ++static netdev_tx_t ipqess_xmit(struct sk_buff *skb,
> ++			     struct net_device *netdev)
> ++{
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	struct ipqess_tx_ring *tx_ring;
> ++	int tx_num;
> ++	int ret;
> ++
> ++	tx_ring = &ess->tx_ring[skb_get_queue_mapping(skb)];
> ++	tx_num = ipqess_cal_txd_req(skb);
> ++	if (ipqess_tx_desc_available(tx_ring) <= tx_num) {
> ++		printk("s %d %x\n", tx_ring->idx, ipqess_r32(tx_ring->ess,
> IPQESS_REG_TX_INT_MASK_Q(tx_ring->idx)));
> ++		netif_tx_stop_queue(tx_ring->nq);
> ++		ipqess_w32(tx_ring->ess,
> IPQESS_REG_TX_INT_MASK_Q(tx_ring->idx), 0x1);
> ++		return NETDEV_TX_BUSY;
> ++	}
> ++
> ++	ret = ipqess_tx_map_and_fill(tx_ring, skb);
> ++	if (ret) {
> ++		dev_kfree_skb_any(skb);
> ++		ess->stats.tx_errors++;
> ++		goto err_out;
> ++	}
> ++
> ++	ess->stats.tx_packets++;
> ++	ess->stats.tx_bytes += skb->len;
> ++	netdev_tx_sent_queue(tx_ring->nq, skb->len);
> ++
> ++	if (!skb->xmit_more || netif_xmit_stopped(tx_ring->nq))
> ++		ipqess_m32(ess,
> ++			 IPQESS_TPD_PROD_IDX_BITS,
> ++			 tx_ring->head,
> ++			 IPQESS_REG_TPD_IDX_Q(tx_ring->idx));
> ++
> ++err_out:
> ++	return NETDEV_TX_OK;
> ++}
> ++
> ++static int ipqess_set_mac_address(struct net_device *netdev, void *p)
> ++{
> ++	int ret = eth_mac_addr(netdev, p);
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++	const char *macaddr = netdev->dev_addr;
> ++
> ++	if (ret)
> ++		return ret;
> ++
> ++//	spin_lock_bh(&mac->hw->page_lock);
> ++	ipqess_w32(ess, IPQESS_REG_MAC_CTRL1,
> ++		 (macaddr[0] << 8) | macaddr[1]);
> ++	ipqess_w32(ess, IPQESS_REG_MAC_CTRL0,
> ++		 (macaddr[2] << 24) | (macaddr[3] << 16) |
> ++		 (macaddr[4] << 8) | macaddr[5]);
> ++//	spin_unlock_bh(&mac->hw->page_lock);
> ++
> ++	return 0;
> ++}
> ++
> ++static const struct net_device_ops ipqess_axi_netdev_ops = {
> ++	.ndo_init		= ipqess_init,
> ++	.ndo_uninit		= ipqess_uninit,
> ++	.ndo_open		= ipqess_open,
> ++	.ndo_stop		= ipqess_stop,
> ++	.ndo_do_ioctl		= ipqess_do_ioctl,
> ++	.ndo_start_xmit		= ipqess_xmit,
> ++	.ndo_get_stats		= ipqess_get_stats,
> ++	.ndo_set_mac_address	= ipqess_set_mac_address,
> ++};
> ++
> ++static void ipqess_reset(struct ipqess *ess)
> ++{
> ++	int i;
> ++
> ++	/* disable all IRQs */
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		ipqess_w32(ess, IPQESS_REG_RX_INT_MASK_Q(i), 0x0);
> ++		ipqess_w32(ess, IPQESS_REG_TX_INT_MASK_Q(i), 0x0);
> ++	}
> ++
> ++	ipqess_w32(ess, IPQESS_REG_MISC_IMR, 0);
> ++	ipqess_w32(ess, IPQESS_REG_WOL_IMR, 0);
> ++
> ++	/* clear the IRQ status registers */
> ++	ipqess_w32(ess, IPQESS_REG_RX_ISR, 0xff);
> ++	ipqess_w32(ess, IPQESS_REG_TX_ISR, 0xffff);
> ++	ipqess_w32(ess, IPQESS_REG_MISC_ISR, 0x1fff);
> ++	ipqess_w32(ess, IPQESS_REG_WOL_ISR, 0x1);
> ++	ipqess_w32(ess, IPQESS_REG_WOL_CTRL, 0);
> ++
> ++	/* disable RX and TX queues */
> ++	ipqess_m32(ess, IPQESS_RXQ_CTRL_EN, 0, IPQESS_REG_RXQ_CTRL);
> ++	ipqess_m32(ess, IPQESS_TXQ_CTRL_TXQ_EN, 0,
> IPQESS_REG_TXQ_CTRL);
> ++}
> ++
> ++static int ipqess_hw_init(struct ipqess *ess)
> ++{
> ++	int i, err;
> ++
> ++	ipqess_reset(ess);
> ++
> ++	ipqess_m32(ess, BIT(IPQESS_INTR_SW_IDX_W_TYP_SHIFT),
> ++		 IPQESS_INTR_SW_IDX_W_TYPE <<
> IPQESS_INTR_SW_IDX_W_TYP_SHIFT,
> ++		 IPQESS_REG_INTR_CTRL);
> ++
> ++	ipqess_w32(ess, IPQESS_REG_RX_ISR, 0xff);
> ++	ipqess_w32(ess, IPQESS_REG_TX_ISR, 0xffff);
> ++
> ++	/* enable IRQ delay slot */
> ++	ipqess_w32(ess, IPQESS_REG_IRQ_MODRT_TIMER_INIT,
> ++		 (IPQESS_TX_IMT << IPQESS_IRQ_MODRT_TX_TIMER_SHIFT) |
> ++		 (IPQESS_RX_IMT << IPQESS_IRQ_MODRT_RX_TIMER_SHIFT));
> ++
> ++	/* Configure the TX Queue bursting */
> ++	ipqess_w32(ess, IPQESS_REG_TXQ_CTRL,
> ++		 (IPQESS_TPD_BURST <<
> IPQESS_TXQ_NUM_TPD_BURST_SHIFT) |
> ++		 (IPQESS_TXF_BURST << IPQESS_TXQ_TXF_BURST_NUM_SHIFT)
> |
> ++		 IPQESS_TXQ_CTRL_TPD_BURST_EN);
> ++
> ++	/* Set RSS type */
> ++	ipqess_w32(ess, IPQESS_REG_RSS_TYPE,
> ++		 IPQESS_RSS_TYPE_IPV4TCP | IPQESS_RSS_TYPE_IPV6_TCP |
> ++		 IPQESS_RSS_TYPE_IPV4_UDP | IPQESS_RSS_TYPE_IPV6UDP |
> ++		 IPQESS_RSS_TYPE_IPV4 | IPQESS_RSS_TYPE_IPV6);
> ++
> ++	/* Set RFD ring burst and threshold */
> ++	ipqess_w32(ess, IPQESS_REG_RX_DESC1,
> ++		(IPQESS_RFD_BURST << IPQESS_RXQ_RFD_BURST_NUM_SHIFT)
> |
> ++		(IPQESS_RFD_THR << IPQESS_RXQ_RFD_PF_THRESH_SHIFT) |
> ++		(IPQESS_RFD_LTHR <<
> IPQESS_RXQ_RFD_LOW_THRESH_SHIFT));
> ++
> ++	/* Set Rx FIFO
> ++	 * - threshold to start to DMA data to host
> ++	 * - remove vlan bit
> ++	 */
> ++	ipqess_w32(ess, IPQESS_REG_RXQ_CTRL,
> ++		 IPQESS_FIFO_THRESH_128_BYTE |
> IPQESS_RXQ_CTRL_RMV_VLAN);
> ++
> ++	err = ipqess_rx_ring_alloc(ess);
> ++	if (err)
> ++		return err;
> ++
> ++	err = ipqess_tx_ring_alloc(ess);
> ++	if (err)
> ++		return err;
> ++
> ++	/* Disable TX FIFO low watermark and high watermark */
> ++	ipqess_w32(ess, IPQESS_REG_TXF_WATER_MARK, 0);
> ++
> ++	/* Configure RSS indirection table.
> ++	 * 128 hash will be configured in the following
> ++	 * pattern: hash{0,1,2,3} = {Q0,Q2,Q4,Q6} respectively
> ++	 * and so on
> ++	 */
> ++	for (i = 0; i < IPQESS_NUM_IDT; i++)
> ++		ipqess_w32(ess, IPQESS_REG_RSS_IDT(i),
> IPQESS_RSS_IDT_VALUE);
> ++
> ++	/* Configure load balance mapping table.
> ++	 * 4 table entry will be configured according to the
> ++	 * following pattern: load_balance{0,1,2,3} = {Q0,Q1,Q3,Q4}
> ++	 * respectively.
> ++	 */
> ++	ipqess_w32(ess, IPQESS_REG_LB_RING, IPQESS_LB_REG_VALUE);
> ++
> ++	/* Configure Virtual queue for Tx rings
> ++	 * User can also change this value runtime through
> ++	 * a sysctl
> ++	 */
> ++	ipqess_w32(ess, IPQESS_REG_VQ_CTRL0, IPQESS_VQ_REG_VALUE);
> ++	ipqess_w32(ess, IPQESS_REG_VQ_CTRL1, IPQESS_VQ_REG_VALUE);
> ++
> ++	/* Configure Max AXI Burst write size to 128 bytes*/
> ++	ipqess_w32(ess, IPQESS_REG_AXIW_CTRL_MAXWRSIZE,
> ++		 IPQESS_AXIW_MAXWRSIZE_VALUE);
> ++
> ++	/* Enable All 16 tx and 8 rx irq mask */
> ++	ipqess_m32(ess, 0, IPQESS_TXQ_CTRL_TXQ_EN,
> IPQESS_REG_TXQ_CTRL);
> ++	ipqess_m32(ess, 0, IPQESS_RXQ_CTRL_EN, IPQESS_REG_RXQ_CTRL);
> ++
> ++	return 0;
> ++}
> ++
> ++static void ipqess_cleanup(struct ipqess *ess)
> ++{
> ++	ipqess_reset(ess);
> ++	unregister_netdev(ess->netdev);
> ++
> ++	ipqess_tx_ring_free(ess);
> ++	ipqess_rx_ring_free(ess);
> ++
> ++	free_netdev(ess->netdev);
> ++}
> ++
> ++static int ipqess_axi_probe(struct platform_device *pdev)
> ++{
> ++	struct ipqess *ess;
> ++	struct net_device *netdev;
> ++	struct resource *res;
> ++	int i, err = 0;
> ++
> ++	netdev = alloc_etherdev_mqs(sizeof(struct ipqess),
> ++				    IPQESS_NETDEV_QUEUES,
> ++				    IPQESS_NETDEV_QUEUES);
> ++	if (!netdev)
> ++		return -ENODEV;
> ++
> ++	ess = netdev_priv(netdev);
> ++	memset(ess, 0, sizeof(struct ipqess));
> ++	ess->netdev = netdev;
> ++	ess->pdev = pdev;
> ++	ess->of_node = pdev->dev.of_node;
> ++	spin_lock_init(&ess->stats_lock);
> ++	SET_NETDEV_DEV(netdev, &pdev->dev);
> ++	platform_set_drvdata(pdev, netdev);
> ++
> ++	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> ++	ess->hw_addr = devm_ioremap_resource(&pdev->dev, res);
> ++	if (IS_ERR(ess->hw_addr)) {
> ++		err = PTR_ERR(ess->hw_addr);
> ++		goto err_out;
> ++	}
> ++
> ++	for (i = 0; i < IPQESS_MAX_TX_QUEUE; i++)
> ++		ess->tx_irq[i] = platform_get_irq(pdev, i);
> ++	for (i = 0; i < IPQESS_MAX_RX_QUEUE; i++)
> ++		ess->rx_irq[i] = platform_get_irq(pdev, i +
> IPQESS_MAX_TX_QUEUE);
> ++
> ++	netdev->netdev_ops = &ipqess_axi_netdev_ops;
> ++	netdev->features = NETIF_F_HW_CSUM | NETIF_F_RXCSUM |
> ++			   NETIF_F_HW_VLAN_CTAG_RX |
> ++			   NETIF_F_HW_VLAN_CTAG_TX |
> ++			   NETIF_F_TSO | NETIF_F_TSO6 |
> ++			   NETIF_F_GRO | NETIF_F_SG;
> ++	netdev->hw_features = NETIF_F_HW_CSUM | NETIF_F_RXCSUM |
> ++			      NETIF_F_HW_VLAN_CTAG_RX |
> ++			      NETIF_F_HW_VLAN_CTAG_TX | NETIF_F_SG |
> ++			      NETIF_F_TSO | NETIF_F_TSO6 |
> ++			      NETIF_F_GRO;
> ++	netdev->vlan_features = NETIF_F_HW_CSUM | NETIF_F_SG |
> ++				NETIF_F_TSO | NETIF_F_TSO6 |
> ++				NETIF_F_GRO;
> ++	netdev->wanted_features = NETIF_F_HW_CSUM | NETIF_F_SG |
> ++				  NETIF_F_TSO | NETIF_F_TSO6 |
> ++				  NETIF_F_GRO;
> ++	netdev->watchdog_timeo = 5 * HZ;
> ++	netdev->base_addr = (u32) ess->hw_addr;
> ++	netdev->max_mtu = 9000;
> ++	netdev->gso_max_segs = IPQESS_TX_RING_SIZE / 2;
> ++
> ++	ipqess_set_ethtool_ops(netdev);
> ++
> ++	err = register_netdev(netdev);
> ++	if (err)
> ++		goto err_out;
> ++
> ++	err = ipqess_hw_init(ess);
> ++	if (err)
> ++		goto err_out;
> ++
> ++	for (i = 0; i < IPQESS_NETDEV_QUEUES; i++) {
> ++		netif_napi_add(netdev,
> ++			       &ess->tx_ring[i].napi_tx,
> ++			       ipqess_tx_napi, 64);
> ++		netif_napi_add(netdev,
> ++			       &ess->rx_ring[i].napi_rx,
> ++			       ipqess_rx_napi, 64);
> ++
> ++		ess->queue[i].ess = ess;
> ++		ess->queue[i].idx = i;
> ++		err = devm_request_irq(&ess->netdev->dev,
> ++			ess->tx_irq[i << IPQESS_TX_CPU_START_SHIFT],
> ++			ipqess_interrupt_tx, 0, "ipqess TX", &ess->tx_ring[i]);
> ++		if (err)
> ++			goto err_out;
> ++
> ++		err = devm_request_irq(&ess->netdev->dev,
> ++			ess->rx_irq[i << IPQESS_RX_CPU_START_SHIFT],
> ++			ipqess_interrupt_rx, 0, "ipqess RX", &ess->rx_ring[i]);
> ++		if (err)
> ++			goto err_out;
> ++	}
> ++
> ++
> ++	return 0;
> ++
> ++err_out:
> ++	ipqess_cleanup(ess);
> ++	return err;
> ++}
> ++
> ++static int ipqess_axi_remove(struct platform_device *pdev)
> ++{
> ++	const struct net_device *netdev = platform_get_drvdata(pdev);
> ++	struct ipqess *ess = netdev_priv(netdev);
> ++
> ++	ipqess_cleanup(ess);
> ++
> ++	return 0;
> ++}
> ++
> ++static const struct of_device_id ipqess_of_mtable[] = {
> ++	{.compatible = "qcom,ess-edma" },
> ++	{}
> ++};
> ++MODULE_DEVICE_TABLE(of, ipqess_of_mtable);
> ++
> ++static struct platform_driver ipqess_axi_driver = {
> ++	.driver = {
> ++		.name    = "ess-edma",
> ++		.of_match_table = ipqess_of_mtable,
> ++	},
> ++	.probe    = ipqess_axi_probe,
> ++	.remove   = ipqess_axi_remove,
> ++};
> ++
> ++module_platform_driver(ipqess_axi_driver);
> ++
> ++MODULE_AUTHOR("Qualcomm Atheros Inc");
> ++MODULE_AUTHOR("John Crispin <john at phrozen.org>");
> ++MODULE_LICENSE("GPL");
> +diff --git a/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.h b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.h
> +new file mode 100644
> +index 0000000000..a988bbca07
> +--- /dev/null
> ++++ b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess.h
> +@@ -0,0 +1,568 @@
> ++/*
> ++ * Copyright (c) 2014 - 2016, The Linux Foundation. All rights reserved.
> ++ *
> ++ * Permission to use, copy, modify, and/or distribute this software for
> ++ * any purpose with or without fee is hereby granted, provided that the
> ++ * above copyright notice and this permission notice appear in all copies.
> ++ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
> WARRANTIES
> ++ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
> OF
> ++ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE
> LIABLE FOR
> ++ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY
> DAMAGES
> ++ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
> WHETHER IN AN
> ++ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
> ARISING OUT
> ++ * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
> SOFTWARE.
> ++ */
> ++
> ++#ifndef _IPQESS_H_
> ++#define _IPQESS_H_
> ++
> ++#define IPQESS_CPU_CORES_SUPPORTED 4
> ++#define IPQESS_MAX_PORTID_SUPPORTED 5
> ++#define IPQESS_MAX_VLAN_SUPPORTED
> IPQESS_MAX_PORTID_SUPPORTED
> ++#define IPQESS_MAX_PORTID_BITMAP_INDEX
> (IPQESS_MAX_PORTID_SUPPORTED + 1)
> ++#define IPQESS_MAX_PORTID_BITMAP_SUPPORTED 0x1f	/* 0001_1111 =
> 0x1f */
> ++#define IPQESS_MAX_NETDEV_PER_QUEUE 4 /* 3 Netdev per queue, 1
> space for indexing */
> ++
> ++#define IPQESS_NETDEV_QUEUES	4
> ++
> ++#define IPQESS_TPD_EOP_SHIFT 31
> ++
> ++#define IPQESS_PORT_ID_SHIFT 12
> ++#define IPQESS_PORT_ID_MASK 0x7
> ++
> ++/* tpd word 3 bit 18-28 */
> ++#define IPQESS_TPD_PORT_BITMAP_SHIFT 18
> ++
> ++#define IPQESS_TPD_FROM_CPU_SHIFT 25
> ++
> ++#define IPQESS_RX_RING_SIZE 128
> ++#define IPQESS_RX_HEAD_BUFF_SIZE 1540
> ++#define IPQESS_TX_RING_SIZE 128
> ++#define IPQESS_MAX_RX_QUEUE 8
> ++#define IPQESS_MAX_TX_QUEUE 16
> ++
> ++
> ++/* Configurations */
> ++#define IPQESS_INTR_CLEAR_TYPE 0
> ++#define IPQESS_INTR_SW_IDX_W_TYPE 0
> ++#define IPQESS_FIFO_THRESH_TYPE 0
> ++#define IPQESS_RSS_TYPE 0
> ++#define IPQESS_RX_IMT 0x0020
> ++#define IPQESS_TX_IMT 0x0050
> ++#define IPQESS_TPD_BURST 5
> ++#define IPQESS_TXF_BURST 0x100
> ++#define IPQESS_RFD_BURST 8
> ++#define IPQESS_RFD_THR 16
> ++#define IPQESS_RFD_LTHR 0
> ++
> ++/* RX/TX per CPU based mask/shift */
> ++#define IPQESS_TX_PER_CPU_MASK 0xF
> ++#define IPQESS_RX_PER_CPU_MASK 0x3
> ++#define IPQESS_TX_PER_CPU_MASK_SHIFT 0x2
> ++#define IPQESS_RX_PER_CPU_MASK_SHIFT 0x1
> ++#define IPQESS_TX_CPU_START_SHIFT 0x2
> ++#define IPQESS_RX_CPU_START_SHIFT 0x1
> ++
> ++/* Flags used in transmit direction */
> ++#define IPQESS_HW_CHECKSUM 0x00000001
> ++#define IPQESS_VLAN_TX_TAG_INSERT_FLAG 0x00000002
> ++#define IPQESS_VLAN_TX_TAG_INSERT_DEFAULT_FLAG 0x00000004
> ++
> ++#define IPQESS_DESC_LAST 0x1
> ++#define IPQESS_DESC_SINGLE 0x2
> ++#define IPQESS_DESC_PAGE 0x4
> ++#define IPQESS_DESC_PAGELIST 0x8
> ++#define IPQESS_DESC_SKB_NONE 0x10
> ++#define IPQESS_DESC_SKB_REUSE 0x20
> ++
> ++
> ++#define IPQESS_MAX_SKB_FRAGS (MAX_SKB_FRAGS + 1)
> ++
> ++/* Ethtool specific list of IPQESS supported features */
> ++#define IPQESS_SUPPORTED_FEATURES (SUPPORTED_10baseT_Half \
> ++					| SUPPORTED_10baseT_Full \
> ++					| SUPPORTED_100baseT_Half \
> ++					| SUPPORTED_100baseT_Full \
> ++					| SUPPORTED_1000baseT_Full)
> ++
> ++/* Receive side Atheros Header */
> ++#define IPQESS_RX_ATH_HDR_VERSION 0x2
> ++#define IPQESS_RX_ATH_HDR_VERSION_SHIFT 14
> ++#define IPQESS_RX_ATH_HDR_PRIORITY_SHIFT 11
> ++#define IPQESS_RX_ATH_PORT_TYPE_SHIFT 6
> ++#define IPQESS_RX_ATH_HDR_RSTP_PORT_TYPE 0x4
> ++
> ++/* Transmit side Atheros Header */
> ++#define IPQESS_TX_ATH_HDR_PORT_BITMAP_MASK 0x7F
> ++#define IPQESS_TX_ATH_HDR_FROM_CPU_MASK 0x80
> ++#define IPQESS_TX_ATH_HDR_FROM_CPU_SHIFT 7
> ++
> ++#define IPQESS_TXQ_START_CORE0 8
> ++#define IPQESS_TXQ_START_CORE1 12
> ++#define IPQESS_TXQ_START_CORE2 0
> ++#define IPQESS_TXQ_START_CORE3 4
> ++
> ++#define IPQESS_TXQ_IRQ_MASK_CORE0 0x0F00
> ++#define IPQESS_TXQ_IRQ_MASK_CORE1 0xF000
> ++#define IPQESS_TXQ_IRQ_MASK_CORE2 0x000F
> ++#define IPQESS_TXQ_IRQ_MASK_CORE3 0x00F0
> ++
> ++#define IPQESS_ETH_HDR_LEN 12
> ++#define IPQESS_ETH_TYPE_MASK 0xFFFF
> ++
> ++#define IPQESS_RX_BUFFER_WRITE 16
> ++#define IPQESS_RFD_AVAIL_THR 80
> ++
> ++#define IPQESS_GMAC_NO_MDIO_PHY	PHY_MAX_ADDR
> ++
> ++extern int ssdk_rfs_ipct_rule_set(__be32 ip_src, __be32 ip_dst,
> ++				  __be16 sport, __be16 dport,
> ++				  uint8_t proto, u16 loadbalance, bool action);
> ++struct ipqesstool_statistics {
> ++	u32 tx_q0_pkt;
> ++	u32 tx_q1_pkt;
> ++	u32 tx_q2_pkt;
> ++	u32 tx_q3_pkt;
> ++	u32 tx_q4_pkt;
> ++	u32 tx_q5_pkt;
> ++	u32 tx_q6_pkt;
> ++	u32 tx_q7_pkt;
> ++	u32 tx_q8_pkt;
> ++	u32 tx_q9_pkt;
> ++	u32 tx_q10_pkt;
> ++	u32 tx_q11_pkt;
> ++	u32 tx_q12_pkt;
> ++	u32 tx_q13_pkt;
> ++	u32 tx_q14_pkt;
> ++	u32 tx_q15_pkt;
> ++	u32 tx_q0_byte;
> ++	u32 tx_q1_byte;
> ++	u32 tx_q2_byte;
> ++	u32 tx_q3_byte;
> ++	u32 tx_q4_byte;
> ++	u32 tx_q5_byte;
> ++	u32 tx_q6_byte;
> ++	u32 tx_q7_byte;
> ++	u32 tx_q8_byte;
> ++	u32 tx_q9_byte;
> ++	u32 tx_q10_byte;
> ++	u32 tx_q11_byte;
> ++	u32 tx_q12_byte;
> ++	u32 tx_q13_byte;
> ++	u32 tx_q14_byte;
> ++	u32 tx_q15_byte;
> ++	u32 rx_q0_pkt;
> ++	u32 rx_q1_pkt;
> ++	u32 rx_q2_pkt;
> ++	u32 rx_q3_pkt;
> ++	u32 rx_q4_pkt;
> ++	u32 rx_q5_pkt;
> ++	u32 rx_q6_pkt;
> ++	u32 rx_q7_pkt;
> ++	u32 rx_q0_byte;
> ++	u32 rx_q1_byte;
> ++	u32 rx_q2_byte;
> ++	u32 rx_q3_byte;
> ++	u32 rx_q4_byte;
> ++	u32 rx_q5_byte;
> ++	u32 rx_q6_byte;
> ++	u32 rx_q7_byte;
> ++	u32 tx_desc_error;
> ++};
> ++
> ++struct ipqess_tx_desc {
> ++	__le16  len;
> ++	__le16  svlan_tag;
> ++	__le32  word1;
> ++	__le32  addr;
> ++	__le32  word3;
> ++};
> ++
> ++struct ipqess_rx_desc {
> ++	u16 rrd0;
> ++	u16 rrd1;
> ++	u16 rrd2;
> ++	u16 rrd3;
> ++	u16 rrd4;
> ++	u16 rrd5;
> ++	u16 rrd6;
> ++	u16 rrd7;
> ++};
> ++
> ++struct ipqess_buf {
> ++	struct sk_buff *skb;
> ++	dma_addr_t dma;
> ++	u16 length;
> ++	u32 flags;
> ++};
> ++
> ++struct queue {
> ++	u32 idx;
> ++	u32 tx_mask;
> ++	u32 tx_status;
> ++	u32 tx_start;
> ++	struct ipqess *ess;
> ++};
> ++
> ++struct ipqess_tx_ring {
> ++	u32 idx;
> ++	struct ipqess *ess;
> ++	struct napi_struct napi_tx;
> ++	struct netdev_queue *nq;
> ++	void *hw_desc;
> ++	struct ipqess_buf *buf;
> ++	int netdev_bmp;
> ++	u16 count;
> ++	dma_addr_t dma;
> ++	u16 head;
> ++	u16 tail;
> ++};
> ++
> ++struct ipqess_rx_ring {
> ++	u32 idx;
> ++	struct ipqess *ess;
> ++	struct napi_struct napi_rx;
> ++	void **hw_desc;
> ++	struct ipqess_buf *buf;
> ++	struct work_struct refill_work;
> ++	atomic_t refill_count;
> ++	dma_addr_t dma;
> ++	u16 head;
> ++	u16 tail;
> ++};
> ++
> ++struct ipqess {
> ++	struct net_device *netdev;
> ++	void __iomem *hw_addr;
> ++
> ++	struct device_node *of_node;
> ++
> ++	struct ipqess_tx_ring tx_ring[IPQESS_NETDEV_QUEUES];
> ++	struct ipqess_rx_ring rx_ring[IPQESS_NETDEV_QUEUES];
> ++	struct platform_device *pdev;
> ++	struct ipqesstool_statistics ipqessstats;
> ++	u32 tx_irq[16];
> ++	u32 rx_irq[8];
> ++	u32 from_cpu;
> ++	struct queue queue[CONFIG_NR_CPUS];
> ++	spinlock_t stats_lock;
> ++
> ++	struct net_device_stats stats;
> ++	u32 flags;
> ++	unsigned long state_flags;
> ++};
> ++
> ++void ipqess_set_ethtool_ops(struct net_device *netdev);
> ++
> ++/* register definition */
> ++#define IPQESS_REG_MAS_CTRL 0x0
> ++#define IPQESS_REG_TIMEOUT_CTRL 0x004
> ++#define IPQESS_REG_DBG0 0x008
> ++#define IPQESS_REG_DBG1 0x00C
> ++#define IPQESS_REG_SW_CTRL0 0x100
> ++#define IPQESS_REG_SW_CTRL1 0x104
> ++
> ++/* Interrupt Status Register */
> ++#define IPQESS_REG_RX_ISR 0x200
> ++#define IPQESS_REG_TX_ISR 0x208
> ++#define IPQESS_REG_MISC_ISR 0x210
> ++#define IPQESS_REG_WOL_ISR 0x218
> ++
> ++#define IPQESS_MISC_ISR_RX_URG_Q(x) (1 << x)
> ++
> ++#define IPQESS_MISC_ISR_AXIR_TIMEOUT 0x00000100
> ++#define IPQESS_MISC_ISR_AXIR_ERR 0x00000200
> ++#define IPQESS_MISC_ISR_TXF_DEAD 0x00000400
> ++#define IPQESS_MISC_ISR_AXIW_ERR 0x00000800
> ++#define IPQESS_MISC_ISR_AXIW_TIMEOUT 0x00001000
> ++
> ++#define IPQESS_WOL_ISR 0x00000001
> ++
> ++/* Interrupt Mask Register */
> ++#define IPQESS_REG_MISC_IMR 0x214
> ++#define IPQESS_REG_WOL_IMR 0x218
> ++
> ++#define IPQESS_RX_IMR_NORMAL_MASK 0x1
> ++#define IPQESS_TX_IMR_NORMAL_MASK 0x1
> ++#define IPQESS_MISC_IMR_NORMAL_MASK 0x80001FFF
> ++#define IPQESS_WOL_IMR_NORMAL_MASK 0x1
> ++
> ++/* Edma receive consumer index */
> ++#define IPQESS_REG_RX_SW_CONS_IDX_Q(x) (0x220 + ((x) << 3)) /* x is the
> queue id */
> ++/* Edma transmit consumer index */
> ++#define IPQESS_REG_TX_SW_CONS_IDX_Q(x) (0x240 + ((x) << 2)) /* x is the
> queue id */
> ++
> ++/* IRQ Moderator Initial Timer Register */
> ++#define IPQESS_REG_IRQ_MODRT_TIMER_INIT 0x280
> ++#define IPQESS_IRQ_MODRT_TIMER_MASK 0xFFFF
> ++#define IPQESS_IRQ_MODRT_RX_TIMER_SHIFT 0
> ++#define IPQESS_IRQ_MODRT_TX_TIMER_SHIFT 16
> ++
> ++/* Interrupt Control Register */
> ++#define IPQESS_REG_INTR_CTRL 0x284
> ++#define IPQESS_INTR_CLR_TYP_SHIFT 0
> ++#define IPQESS_INTR_SW_IDX_W_TYP_SHIFT 1
> ++#define IPQESS_INTR_CLEAR_TYPE_W1 0
> ++#define IPQESS_INTR_CLEAR_TYPE_R 1
> ++
> ++/* RX Interrupt Mask Register */
> ++#define IPQESS_REG_RX_INT_MASK_Q(x) (0x300 + ((x) << 3)) /* x = queue id
> */
> ++
> ++/* TX Interrupt mask register */
> ++#define IPQESS_REG_TX_INT_MASK_Q(x) (0x340 + ((x) << 2)) /* x = queue id
> */
> ++
> ++/* Load Ptr Register
> ++ * Software sets this bit after the initialization of the head and tail
> ++ */
> ++#define IPQESS_REG_TX_SRAM_PART 0x400
> ++#define IPQESS_LOAD_PTR_SHIFT 16
> ++
> ++/* TXQ Control Register */
> ++#define IPQESS_REG_TXQ_CTRL 0x404
> ++#define IPQESS_TXQ_CTRL_IP_OPTION_EN 0x10
> ++#define IPQESS_TXQ_CTRL_TXQ_EN 0x20
> ++#define IPQESS_TXQ_CTRL_ENH_MODE 0x40
> ++#define IPQESS_TXQ_CTRL_LS_8023_EN 0x80
> ++#define IPQESS_TXQ_CTRL_TPD_BURST_EN 0x100
> ++#define IPQESS_TXQ_CTRL_LSO_BREAK_EN 0x200
> ++#define IPQESS_TXQ_NUM_TPD_BURST_MASK 0xF
> ++#define IPQESS_TXQ_TXF_BURST_NUM_MASK 0xFFFF
> ++#define IPQESS_TXQ_NUM_TPD_BURST_SHIFT 0
> ++#define IPQESS_TXQ_TXF_BURST_NUM_SHIFT 16
> ++
> ++#define	IPQESS_REG_TXF_WATER_MARK 0x408 /* In 8-bytes */
> ++#define IPQESS_TXF_WATER_MARK_MASK 0x0FFF
> ++#define IPQESS_TXF_LOW_WATER_MARK_SHIFT 0
> ++#define IPQESS_TXF_HIGH_WATER_MARK_SHIFT 16
> ++#define IPQESS_TXQ_CTRL_BURST_MODE_EN 0x80000000
> ++
> ++/* WRR Control Register */
> ++#define IPQESS_REG_WRR_CTRL_Q0_Q3 0x40c
> ++#define IPQESS_REG_WRR_CTRL_Q4_Q7 0x410
> ++#define IPQESS_REG_WRR_CTRL_Q8_Q11 0x414
> ++#define IPQESS_REG_WRR_CTRL_Q12_Q15 0x418
> ++
> ++/* Weight round robin(WRR), it takes queue as input, and computes
> ++ * starting bits where we need to write the weight for a particular
> ++ * queue
> ++ */
> ++#define IPQESS_WRR_SHIFT(x) (((x) * 5) % 20)
> ++
> ++/* Tx Descriptor Control Register */
> ++#define IPQESS_REG_TPD_RING_SIZE 0x41C
> ++#define IPQESS_TPD_RING_SIZE_SHIFT 0
> ++#define IPQESS_TPD_RING_SIZE_MASK 0xFFFF
> ++
> ++/* Transmit descriptor base address */
> ++#define IPQESS_REG_TPD_BASE_ADDR_Q(x) (0x420 + ((x) << 2)) /* x =
> queue id */
> ++
> ++/* TPD Index Register */
> ++#define IPQESS_REG_TPD_IDX_Q(x) (0x460 + ((x) << 2)) /* x = queue id */
> ++
> ++#define IPQESS_TPD_PROD_IDX_BITS 0x0000FFFF
> ++#define IPQESS_TPD_CONS_IDX_BITS 0xFFFF0000
> ++#define IPQESS_TPD_PROD_IDX_MASK 0xFFFF
> ++#define IPQESS_TPD_CONS_IDX_MASK 0xFFFF
> ++#define IPQESS_TPD_PROD_IDX_SHIFT 0
> ++#define IPQESS_TPD_CONS_IDX_SHIFT 16
> ++
> ++/* TX Virtual Queue Mapping Control Register */
> ++#define IPQESS_REG_VQ_CTRL0 0x4A0
> ++#define IPQESS_REG_VQ_CTRL1 0x4A4
> ++
> ++/* Virtual QID shift, it takes queue as input, and computes
> ++ * Virtual QID position in virtual qid control register
> ++ */
> ++#define IPQESS_VQ_ID_SHIFT(i) (((i) * 3) % 24)
> ++
> ++/* Virtual Queue Default Value */
> ++#define IPQESS_VQ_REG_VALUE 0x240240
> ++
> ++/* Tx side Port Interface Control Register */
> ++#define IPQESS_REG_PORT_CTRL 0x4A8
> ++#define IPQESS_PAD_EN_SHIFT 15
> ++
> ++/* Tx side VLAN Configuration Register */
> ++#define IPQESS_REG_VLAN_CFG 0x4AC
> ++
> ++#define IPQESS_TX_CVLAN 16
> ++#define IPQESS_TX_INS_CVLAN 17
> ++#define IPQESS_TX_CVLAN_TAG_SHIFT 0
> ++
> ++#define IPQESS_TX_SVLAN 14
> ++#define IPQESS_TX_INS_SVLAN 15
> ++#define IPQESS_TX_SVLAN_TAG_SHIFT 16
> ++
> ++/* Tx Queue Packet Statistic Register */
> ++#define IPQESS_REG_TX_STAT_PKT_Q(x) (0x700 + ((x) << 3)) /* x = queue id
> */
> ++
> ++#define IPQESS_TX_STAT_PKT_MASK 0xFFFFFF
> ++
> ++/* Tx Queue Byte Statistic Register */
> ++#define IPQESS_REG_TX_STAT_BYTE_Q(x) (0x704 + ((x) << 3)) /* x = queue id
> */
> ++
> ++/* Load Balance Based Ring Offset Register */
> ++#define IPQESS_REG_LB_RING 0x800
> ++#define IPQESS_LB_RING_ENTRY_MASK 0xff
> ++#define IPQESS_LB_RING_ID_MASK 0x7
> ++#define IPQESS_LB_RING_PROFILE_ID_MASK 0x3
> ++#define IPQESS_LB_RING_ENTRY_BIT_OFFSET 8
> ++#define IPQESS_LB_RING_ID_OFFSET 0
> ++#define IPQESS_LB_RING_PROFILE_ID_OFFSET 3
> ++#define IPQESS_LB_REG_VALUE 0x6040200
> ++
> ++/* Load Balance Priority Mapping Register */
> ++#define IPQESS_REG_LB_PRI_START 0x804
> ++#define IPQESS_REG_LB_PRI_END 0x810
> ++#define IPQESS_LB_PRI_REG_INC 4
> ++#define IPQESS_LB_PRI_ENTRY_BIT_OFFSET 4
> ++#define IPQESS_LB_PRI_ENTRY_MASK 0xf
> ++
> ++/* RSS Priority Mapping Register */
> ++#define IPQESS_REG_RSS_PRI 0x820
> ++#define IPQESS_RSS_PRI_ENTRY_MASK 0xf
> ++#define IPQESS_RSS_RING_ID_MASK 0x7
> ++#define IPQESS_RSS_PRI_ENTRY_BIT_OFFSET 4
> ++
> ++/* RSS Indirection Register */
> ++#define IPQESS_REG_RSS_IDT(x) (0x840 + ((x) << 2)) /* x = No. of indirection
> table */
> ++#define IPQESS_NUM_IDT 16
> ++#define IPQESS_RSS_IDT_VALUE 0x64206420
> ++
> ++/* Default RSS Ring Register */
> ++#define IPQESS_REG_DEF_RSS 0x890
> ++#define IPQESS_DEF_RSS_MASK 0x7
> ++
> ++/* RSS Hash Function Type Register */
> ++#define IPQESS_REG_RSS_TYPE 0x894
> ++#define IPQESS_RSS_TYPE_NONE 0x01
> ++#define IPQESS_RSS_TYPE_IPV4TCP 0x02
> ++#define IPQESS_RSS_TYPE_IPV6_TCP 0x04
> ++#define IPQESS_RSS_TYPE_IPV4_UDP 0x08
> ++#define IPQESS_RSS_TYPE_IPV6UDP 0x10
> ++#define IPQESS_RSS_TYPE_IPV4 0x20
> ++#define IPQESS_RSS_TYPE_IPV6 0x40
> ++#define IPQESS_RSS_HASH_MODE_MASK 0x7f
> ++
> ++#define IPQESS_REG_RSS_HASH_VALUE 0x8C0
> ++
> ++#define IPQESS_REG_RSS_TYPE_RESULT 0x8C4
> ++
> ++#define IPQESS_HASH_TYPE_START 0
> ++#define IPQESS_HASH_TYPE_END 5
> ++#define IPQESS_HASH_TYPE_SHIFT 12
> ++
> ++#define IPQESS_RFS_FLOW_ENTRIES 1024
> ++#define IPQESS_RFS_FLOW_ENTRIES_MASK (IPQESS_RFS_FLOW_ENTRIES -
> 1)
> ++#define IPQESS_RFS_EXPIRE_COUNT_PER_CALL 128
> ++
> ++/* RFD Base Address Register */
> ++#define IPQESS_REG_RFD_BASE_ADDR_Q(x) (0x950 + ((x) << 3)) /* x =
> queue id */
> ++
> ++/* RFD Index Register */
> ++#define IPQESS_REG_RFD_IDX_Q(x) (0x9B0 + ((x) << 3))
> ++
> ++#define IPQESS_RFD_PROD_IDX_BITS 0x00000FFF
> ++#define IPQESS_RFD_CONS_IDX_BITS 0x0FFF0000
> ++#define IPQESS_RFD_PROD_IDX_MASK 0xFFF
> ++#define IPQESS_RFD_CONS_IDX_MASK 0xFFF
> ++#define IPQESS_RFD_PROD_IDX_SHIFT 0
> ++#define IPQESS_RFD_CONS_IDX_SHIFT 16
> ++
> ++/* Rx Descriptor Control Register */
> ++#define IPQESS_REG_RX_DESC0 0xA10
> ++#define IPQESS_RFD_RING_SIZE_MASK 0xFFF
> ++#define IPQESS_RX_BUF_SIZE_MASK 0xFFFF
> ++#define IPQESS_RFD_RING_SIZE_SHIFT 0
> ++#define IPQESS_RX_BUF_SIZE_SHIFT 16
> ++
> ++#define IPQESS_REG_RX_DESC1 0xA14
> ++#define IPQESS_RXQ_RFD_BURST_NUM_MASK 0x3F
> ++#define IPQESS_RXQ_RFD_PF_THRESH_MASK 0x1F
> ++#define IPQESS_RXQ_RFD_LOW_THRESH_MASK 0xFFF
> ++#define IPQESS_RXQ_RFD_BURST_NUM_SHIFT 0
> ++#define IPQESS_RXQ_RFD_PF_THRESH_SHIFT 8
> ++#define IPQESS_RXQ_RFD_LOW_THRESH_SHIFT 16
> ++
> ++/* RXQ Control Register */
> ++#define IPQESS_REG_RXQ_CTRL 0xA18
> ++#define IPQESS_FIFO_THRESH_TYPE_SHIF 0
> ++#define IPQESS_FIFO_THRESH_128_BYTE 0x0
> ++#define IPQESS_FIFO_THRESH_64_BYTE 0x1
> ++#define IPQESS_RXQ_CTRL_RMV_VLAN 0x00000002
> ++#define IPQESS_RXQ_CTRL_EN 0x0000FF00
> ++
> ++/* AXI Burst Size Config */
> ++#define IPQESS_REG_AXIW_CTRL_MAXWRSIZE 0xA1C
> ++#define IPQESS_AXIW_MAXWRSIZE_VALUE 0x0
> ++
> ++/* Rx Statistics Register */
> ++#define IPQESS_REG_RX_STAT_BYTE_Q(x) (0xA30 + ((x) << 2)) /* x = queue
> id */
> ++#define IPQESS_REG_RX_STAT_PKT_Q(x) (0xA50 + ((x) << 2)) /* x = queue id
> */
> ++
> ++/* WoL Pattern Length Register */
> ++#define IPQESS_REG_WOL_PATTERN_LEN0 0xC00
> ++#define IPQESS_WOL_PT_LEN_MASK 0xFF
> ++#define IPQESS_WOL_PT0_LEN_SHIFT 0
> ++#define IPQESS_WOL_PT1_LEN_SHIFT 8
> ++#define IPQESS_WOL_PT2_LEN_SHIFT 16
> ++#define IPQESS_WOL_PT3_LEN_SHIFT 24
> ++
> ++#define IPQESS_REG_WOL_PATTERN_LEN1 0xC04
> ++#define IPQESS_WOL_PT4_LEN_SHIFT 0
> ++#define IPQESS_WOL_PT5_LEN_SHIFT 8
> ++#define IPQESS_WOL_PT6_LEN_SHIFT 16
> ++
> ++/* WoL Control Register */
> ++#define IPQESS_REG_WOL_CTRL 0xC08
> ++#define IPQESS_WOL_WK_EN 0x00000001
> ++#define IPQESS_WOL_MG_EN 0x00000002
> ++#define IPQESS_WOL_PT0_EN 0x00000004
> ++#define IPQESS_WOL_PT1_EN 0x00000008
> ++#define IPQESS_WOL_PT2_EN 0x00000010
> ++#define IPQESS_WOL_PT3_EN 0x00000020
> ++#define IPQESS_WOL_PT4_EN 0x00000040
> ++#define IPQESS_WOL_PT5_EN 0x00000080
> ++#define IPQESS_WOL_PT6_EN 0x00000100
> ++
> ++/* MAC Control Register */
> ++#define IPQESS_REG_MAC_CTRL0 0xC20
> ++#define IPQESS_REG_MAC_CTRL1 0xC24
> ++
> ++/* WoL Pattern Register */
> ++#define IPQESS_REG_WOL_PATTERN_START 0x5000
> ++#define IPQESS_PATTERN_PART_REG_OFFSET 0x40
> ++
> ++
> ++/* TX descriptor fields */
> ++#define IPQESS_TPD_HDR_SHIFT 0
> ++#define IPQESS_TPD_PPPOE_EN 0x00000100
> ++#define IPQESS_TPD_IP_CSUM_EN 0x00000200
> ++#define IPQESS_TPD_TCP_CSUM_EN 0x0000400
> ++#define IPQESS_TPD_UDP_CSUM_EN 0x00000800
> ++#define IPQESS_TPD_CUSTOM_CSUM_EN 0x00000C00
> ++#define IPQESS_TPD_LSO_EN 0x00001000
> ++#define IPQESS_TPD_LSO_V2_EN 0x00002000
> ++#define IPQESS_TPD_IPV4_EN 0x00010000
> ++#define IPQESS_TPD_MSS_MASK 0x1FFF
> ++#define IPQESS_TPD_MSS_SHIFT 18
> ++#define IPQESS_TPD_CUSTOM_CSUM_SHIFT 18
> ++
> ++/* RRD descriptor fields */
> ++#define IPQESS_RRD_NUM_RFD_MASK 0x000F
> ++#define IPQESS_RRD_PKT_SIZE_MASK 0x3FFF
> ++#define IPQESS_RRD_SRC_PORT_NUM_MASK 0x4000
> ++#define IPQESS_RRD_SVLAN 0x8000
> ++#define IPQESS_RRD_FLOW_COOKIE_MASK 0x07FF;
> ++
> ++#define IPQESS_RRD_PKT_SIZE_MASK 0x3FFF
> ++#define IPQESS_RRD_CSUM_FAIL_MASK 0xC000
> ++#define IPQESS_RRD_CVLAN 0x0001
> ++#define IPQESS_RRD_DESC_VALID 0x8000
> ++
> ++#define IPQESS_RRD_PRIORITY_SHIFT 4
> ++#define IPQESS_RRD_PRIORITY_MASK 0x7
> ++#define IPQESS_RRD_PORT_TYPE_SHIFT 7
> ++#define IPQESS_RRD_PORT_TYPE_MASK 0x1F
> ++
> ++#endif
> +diff --git a/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess_ethtool.c
> b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess_ethtool.c
> +new file mode 100644
> +index 0000000000..0b75904d48
> +--- /dev/null
> ++++ b/target/linux/ipq40xx/files-
> 4.14/drivers/net/ethernet/qualcomm/ipqess_ethtool.c
> +@@ -0,0 +1,191 @@
> ++/*
> ++ * Copyright (c) 2015 - 2016, The Linux Foundation. All rights reserved.
> ++ *
> ++ * Permission to use, copy, modify, and/or distribute this software for
> ++ * any purpose with or without fee is hereby granted, provided that the
> ++ * above copyright notice and this permission notice appear in all copies.
> ++ * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
> WARRANTIES
> ++ * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES
> OF
> ++ * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE
> LIABLE FOR
> ++ * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY
> DAMAGES
> ++ * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
> WHETHER IN AN
> ++ * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
> ARISING OUT
> ++ * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
> SOFTWARE.
> ++ */
> ++
> ++#include <linux/ethtool.h>
> ++#include <linux/netdevice.h>
> ++#include <linux/string.h>
> ++#include <linux/phy.h>
> ++#include "ipqess.h"
> ++
> ++struct ipqesstool_stats {
> ++	uint8_t string[ETH_GSTRING_LEN];
> ++	uint32_t offset;
> ++};
> ++
> ++#define IPQESS_STAT(m)    offsetof(struct ipqesstool_statistics, m)
> ++#define DRVINFO_LEN	32
> ++
> ++static const struct ipqesstool_stats ipqess_stats[] = {
> ++	{"tx_q0_pkt", IPQESS_STAT(tx_q0_pkt)},
> ++	{"tx_q1_pkt", IPQESS_STAT(tx_q1_pkt)},
> ++	{"tx_q2_pkt", IPQESS_STAT(tx_q2_pkt)},
> ++	{"tx_q3_pkt", IPQESS_STAT(tx_q3_pkt)},
> ++	{"tx_q4_pkt", IPQESS_STAT(tx_q4_pkt)},
> ++	{"tx_q5_pkt", IPQESS_STAT(tx_q5_pkt)},
> ++	{"tx_q6_pkt", IPQESS_STAT(tx_q6_pkt)},
> ++	{"tx_q7_pkt", IPQESS_STAT(tx_q7_pkt)},
> ++	{"tx_q8_pkt", IPQESS_STAT(tx_q8_pkt)},
> ++	{"tx_q9_pkt", IPQESS_STAT(tx_q9_pkt)},
> ++	{"tx_q10_pkt", IPQESS_STAT(tx_q10_pkt)},
> ++	{"tx_q11_pkt", IPQESS_STAT(tx_q11_pkt)},
> ++	{"tx_q12_pkt", IPQESS_STAT(tx_q12_pkt)},
> ++	{"tx_q13_pkt", IPQESS_STAT(tx_q13_pkt)},
> ++	{"tx_q14_pkt", IPQESS_STAT(tx_q14_pkt)},
> ++	{"tx_q15_pkt", IPQESS_STAT(tx_q15_pkt)},
> ++	{"tx_q0_byte", IPQESS_STAT(tx_q0_byte)},
> ++	{"tx_q1_byte", IPQESS_STAT(tx_q1_byte)},
> ++	{"tx_q2_byte", IPQESS_STAT(tx_q2_byte)},
> ++	{"tx_q3_byte", IPQESS_STAT(tx_q3_byte)},
> ++	{"tx_q4_byte", IPQESS_STAT(tx_q4_byte)},
> ++	{"tx_q5_byte", IPQESS_STAT(tx_q5_byte)},
> ++	{"tx_q6_byte", IPQESS_STAT(tx_q6_byte)},
> ++	{"tx_q7_byte", IPQESS_STAT(tx_q7_byte)},
> ++	{"tx_q8_byte", IPQESS_STAT(tx_q8_byte)},
> ++	{"tx_q9_byte", IPQESS_STAT(tx_q9_byte)},
> ++	{"tx_q10_byte", IPQESS_STAT(tx_q10_byte)},
> ++	{"tx_q11_byte", IPQESS_STAT(tx_q11_byte)},
> ++	{"tx_q12_byte", IPQESS_STAT(tx_q12_byte)},
> ++	{"tx_q13_byte", IPQESS_STAT(tx_q13_byte)},
> ++	{"tx_q14_byte", IPQESS_STAT(tx_q14_byte)},
> ++	{"tx_q15_byte", IPQESS_STAT(tx_q15_byte)},
> ++	{"rx_q0_pkt", IPQESS_STAT(rx_q0_pkt)},
> ++	{"rx_q1_pkt", IPQESS_STAT(rx_q1_pkt)},
> ++	{"rx_q2_pkt", IPQESS_STAT(rx_q2_pkt)},
> ++	{"rx_q3_pkt", IPQESS_STAT(rx_q3_pkt)},
> ++	{"rx_q4_pkt", IPQESS_STAT(rx_q4_pkt)},
> ++	{"rx_q5_pkt", IPQESS_STAT(rx_q5_pkt)},
> ++	{"rx_q6_pkt", IPQESS_STAT(rx_q6_pkt)},
> ++	{"rx_q7_pkt", IPQESS_STAT(rx_q7_pkt)},
> ++	{"rx_q0_byte", IPQESS_STAT(rx_q0_byte)},
> ++	{"rx_q1_byte", IPQESS_STAT(rx_q1_byte)},
> ++	{"rx_q2_byte", IPQESS_STAT(rx_q2_byte)},
> ++	{"rx_q3_byte", IPQESS_STAT(rx_q3_byte)},
> ++	{"rx_q4_byte", IPQESS_STAT(rx_q4_byte)},
> ++	{"rx_q5_byte", IPQESS_STAT(rx_q5_byte)},
> ++	{"rx_q6_byte", IPQESS_STAT(rx_q6_byte)},
> ++	{"rx_q7_byte", IPQESS_STAT(rx_q7_byte)},
> ++	{"tx_desc_error", IPQESS_STAT(tx_desc_error)},
> ++};
> ++
> ++static int ipqess_get_strset_count(struct net_device *netdev, int sset)
> ++{
> ++	switch (sset) {
> ++	case ETH_SS_STATS:
> ++		return ARRAY_SIZE(ipqess_stats);
> ++	default:
> ++		netdev_dbg(netdev, "%s: Invalid string set", __func__);
> ++		return -EOPNOTSUPP;
> ++	}
> ++}
> ++
> ++static void ipqess_get_strings(struct net_device *netdev, uint32_t stringset,
> ++			       uint8_t *data)
> ++{
> ++	uint8_t *p = data;
> ++	uint32_t i;
> ++
> ++	switch (stringset) {
> ++	case ETH_SS_STATS:
> ++		for (i = 0; i < ARRAY_SIZE(ipqess_stats); i++) {
> ++			memcpy(p, ipqess_stats[i].string,
> ++			       min((size_t)ETH_GSTRING_LEN,
> ++			       strlen(ipqess_stats[i].string) + 1));
> ++			p += ETH_GSTRING_LEN;
> ++		}
> ++		break;
> ++	}
> ++}
> ++
> ++static void ipqess_get_drvinfo(struct net_device *dev,
> ++			       struct ethtool_drvinfo *info)
> ++{
> ++	strlcpy(info->driver, "qca_ipqess", DRVINFO_LEN);
> ++	strlcpy(info->bus_info, "axi", ETHTOOL_BUSINFO_LEN);
> ++}
> ++
> ++static int ipqess_get_settings(struct net_device *netdev,
> ++			       struct ethtool_cmd *ecmd)
> ++{
> ++	struct phy_device *phydev = NULL;
> ++	uint16_t phyreg;
> ++
> ++	phydev = netdev->phydev;
> ++
> ++	ecmd->advertising = phydev->advertising;
> ++	ecmd->autoneg = phydev->autoneg;
> ++	ecmd->speed = phydev->speed;
> ++	ecmd->duplex = phydev->duplex;
> ++	ecmd->phy_address = phydev->mdio.addr;
> ++
> ++	phyreg = (uint16_t)phy_read(netdev->phydev, MII_LPA);
> ++	if (phyreg & LPA_10HALF)
> ++		ecmd->lp_advertising |= ADVERTISED_10baseT_Half;
> ++
> ++	if (phyreg & LPA_10FULL)
> ++		ecmd->lp_advertising |= ADVERTISED_10baseT_Full;
> ++
> ++	if (phyreg & LPA_100HALF)
> ++		ecmd->lp_advertising |= ADVERTISED_100baseT_Half;
> ++
> ++	if (phyreg & LPA_100FULL)
> ++		ecmd->lp_advertising |= ADVERTISED_100baseT_Full;
> ++
> ++	phyreg = (uint16_t)phy_read(netdev->phydev, MII_STAT1000);
> ++	if (phyreg & LPA_1000HALF)
> ++		ecmd->lp_advertising |= ADVERTISED_1000baseT_Half;
> ++
> ++	if (phyreg & LPA_1000FULL)
> ++		ecmd->lp_advertising |= ADVERTISED_1000baseT_Full;
> ++
> ++	return 0;
> ++}
> ++
> ++static int ipqess_set_settings(struct net_device *netdev,
> ++			     struct ethtool_cmd *ecmd)
> ++{
> ++	struct phy_device *phydev = NULL;
> ++
> ++	phydev = netdev->phydev;
> ++	phydev->advertising = ecmd->advertising;
> ++	phydev->autoneg = ecmd->autoneg;
> ++	phydev->speed = ethtool_cmd_speed(ecmd);
> ++	phydev->duplex = ecmd->duplex;
> ++
> ++	genphy_config_aneg(phydev);
> ++
> ++	return 0;
> ++}
> ++
> ++static void ipqess_get_ringparam(struct net_device *netdev,
> ++			       struct ethtool_ringparam *ring)
> ++{
> ++	ring->tx_max_pending = IPQESS_TX_RING_SIZE;
> ++	ring->rx_max_pending = IPQESS_RX_RING_SIZE;
> ++}
> ++
> ++static const struct ethtool_ops ipqesstool_ops = {
> ++	.get_drvinfo = &ipqess_get_drvinfo,
> ++	.get_link = &ethtool_op_get_link,
> ++	.get_settings = &ipqess_get_settings,
> ++	.set_settings = &ipqess_set_settings,
> ++	.get_strings = &ipqess_get_strings,
> ++	.get_sset_count = &ipqess_get_strset_count,
> ++	.get_ringparam = ipqess_get_ringparam,
> ++};
> ++
> ++void ipqess_set_ethtool_ops(struct net_device *netdev)
> ++{
> ++	netdev->ethtool_ops = &ipqesstool_ops;
> ++}
> +diff --git a/target/linux/ipq40xx/patches-4.14/999-0-ipqess.patch
> b/target/linux/ipq40xx/patches-4.14/999-0-ipqess.patch
> +new file mode 100644
> +index 0000000000..c0ffe6f052
> +--- /dev/null
> ++++ b/target/linux/ipq40xx/patches-4.14/999-0-ipqess.patch
> +@@ -0,0 +1,167 @@
> ++Index: linux-4.9.34/drivers/net/ethernet/qualcomm/Kconfig
> ++===========================================================
> ========
> ++--- linux-4.9.34.orig/drivers/net/ethernet/qualcomm/Kconfig
> +++++ linux-4.9.34/drivers/net/ethernet/qualcomm/Kconfig
> ++@@ -15,6 +15,15 @@ config NET_VENDOR_QUALCOMM
> ++
> ++ if NET_VENDOR_QUALCOMM
> ++
> +++config IPQ_ESS
> +++	tristate "Qualcomm Atheros IPQ ESS support"
> +++	depends on OF
> +++	---help---
> +++	  This SPI protocol driver supports the Qualcomm Atheros QCA7000.
> +++
> +++	  To compile this driver as a module, choose M here. The module
> +++	  will be called qcaspi.
> +++
> ++ config QCA7000
> ++ 	tristate "Qualcomm Atheros QCA7000 support"
> ++ 	depends on SPI_MASTER && OF
> ++Index: linux-4.9.34/drivers/net/ethernet/qualcomm/Makefile
> ++===========================================================
> ========
> ++--- linux-4.9.34.orig/drivers/net/ethernet/qualcomm/Makefile
> +++++ linux-4.9.34/drivers/net/ethernet/qualcomm/Makefile
> ++@@ -2,6 +2,9 @@
> ++ # Makefile for the Qualcomm network device drivers.
> ++ #
> ++
> +++obj-$(CONFIG_IPQ_ESS) += ipq_ess.o
> +++ipq_ess-objs := ipqess.o ipqess_ethtool.o
> +++
> ++ obj-$(CONFIG_QCA7000) += qca_7k_common.o
> ++ obj-$(CONFIG_QCA7000_SPI) += qcaspi.o
> ++ qcaspi-objs := qca_7k.o qca_debug.o qca_spi.o
> ++Index: linux-4.9.34/arch/arm/boot/dts/qcom-ipq4019.dtsi
> ++===========================================================
> ========
> ++--- linux-4.9.34.orig/arch/arm/boot/dts/qcom-ipq4019.dtsi
> +++++ linux-4.9.34/arch/arm/boot/dts/qcom-ipq4019.dtsi
> ++@@ -44,8 +44,7 @@
> ++ 		spi1 = &spi_1;
> ++ 		i2c0 = &i2c_0;
> ++ 		i2c1 = &i2c_1;
> ++-		ethernet0 = &gmac0;
> ++-		ethernet1 = &gmac1;
> +++		ethernet0 = &gmac;
> ++ 	};
> ++
> ++ 	cpus {
> ++@@ -389,6 +388,53 @@
> ++ 			status = "disabled";
> ++ 		};
> ++
> +++		gmac: edma at c080000 {
> +++			compatible = "qcom,ess-edma";
> +++			reg = <0xc080000 0x8000>;
> +++			interrupts = <GIC_SPI  65 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  66 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  67 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  68 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  69 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  79 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  71 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  72 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  73 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  74 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  75 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  76 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  77 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  78 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  79 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI  80 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 240 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 241 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 242 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 243 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 244 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 245 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 246 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 247 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 248 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 249 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 250 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 251 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 252 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 253 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 254 IRQ_TYPE_EDGE_RISING>,
> +++				     <GIC_SPI 255 IRQ_TYPE_EDGE_RISING>;
> +++
> +++			status = "disabled";
> +++
> +++			phy-mode = "sgmii";
> +++
> +++			fixed-link {
> +++				speed = <1000>;
> +++				full-duplex;
> +++				pause;
> +++			};
> +++		};
> +++
> ++ 		pcie0: pci at 40000000 {
> ++ 			compatible = "qcom,pcie-ipq4019";
> ++ 			reg =  <0x40000000 0xf1d
> ++@@ -462,64 +508,6 @@
> ++ 			status = "disabled";
> ++ 		};
> ++
> ++-		edma at c080000 {
> ++-			compatible = "qcom,ess-edma";
> ++-			reg = <0xc080000 0x8000>;
> ++-			qcom,page-mode = <0>;
> ++-			qcom,rx_head_buf_size = <1540>;
> ++-			qcom,mdio_supported;
> ++-			qcom,poll_required = <1>;
> ++-			qcom,num_gmac = <2>;
> ++-			interrupts = <0  65 IRQ_TYPE_EDGE_RISING
> ++-				      0  66 IRQ_TYPE_EDGE_RISING
> ++-				      0  67 IRQ_TYPE_EDGE_RISING
> ++-				      0  68 IRQ_TYPE_EDGE_RISING
> ++-				      0  69 IRQ_TYPE_EDGE_RISING
> ++-				      0  70 IRQ_TYPE_EDGE_RISING
> ++-				      0  71 IRQ_TYPE_EDGE_RISING
> ++-				      0  72 IRQ_TYPE_EDGE_RISING
> ++-				      0  73 IRQ_TYPE_EDGE_RISING
> ++-				      0  74 IRQ_TYPE_EDGE_RISING
> ++-				      0  75 IRQ_TYPE_EDGE_RISING
> ++-				      0  76 IRQ_TYPE_EDGE_RISING
> ++-				      0  77 IRQ_TYPE_EDGE_RISING
> ++-				      0  78 IRQ_TYPE_EDGE_RISING
> ++-				      0  79 IRQ_TYPE_EDGE_RISING
> ++-				      0  80 IRQ_TYPE_EDGE_RISING
> ++-				      0 240 IRQ_TYPE_EDGE_RISING
> ++-				      0 241 IRQ_TYPE_EDGE_RISING
> ++-				      0 242 IRQ_TYPE_EDGE_RISING
> ++-				      0 243 IRQ_TYPE_EDGE_RISING
> ++-				      0 244 IRQ_TYPE_EDGE_RISING
> ++-				      0 245 IRQ_TYPE_EDGE_RISING
> ++-				      0 246 IRQ_TYPE_EDGE_RISING
> ++-				      0 247 IRQ_TYPE_EDGE_RISING
> ++-				      0 248 IRQ_TYPE_EDGE_RISING
> ++-				      0 249 IRQ_TYPE_EDGE_RISING
> ++-				      0 250 IRQ_TYPE_EDGE_RISING
> ++-				      0 251 IRQ_TYPE_EDGE_RISING
> ++-				      0 252 IRQ_TYPE_EDGE_RISING
> ++-				      0 253 IRQ_TYPE_EDGE_RISING
> ++-				      0 254 IRQ_TYPE_EDGE_RISING
> ++-				      0 255 IRQ_TYPE_EDGE_RISING>;
> ++-
> ++-			status = "disabled";
> ++-
> ++-			gmac0: gmac0 {
> ++-				local-mac-address = [00 00 00 00 00 00];
> ++-				vlan_tag = <1 0x1f>;
> ++-			};
> ++-
> ++-			gmac1: gmac1 {
> ++-				local-mac-address = [00 00 00 00 00 00];
> ++-				qcom,phy_mdio_addr = <4>;
> ++-				qcom,poll_required = <1>;
> ++-				qcom,forced_speed = <1000>;
> ++-				qcom,forced_duplex = <1>;
> ++-				vlan_tag = <2 0x20>;
> ++-			};
> ++-		};
> ++-
> ++ 		usb3_ss_phy: ssphy at 9a000 {
> ++ 			compatible = "qca,uni-ssphy";
> ++ 			reg = <0x9a000 0x800>;
> +--
> +2.11.0
> +
> diff --git a/build_patches/openwrt/0007-ipqess-integration.patch
> b/build_patches/openwrt/0007-ipqess-integration.patch
> new file mode 100644
> index 0000000..3a6b91f
> --- /dev/null
> +++ b/build_patches/openwrt/0007-ipqess-integration.patch
> @@ -0,0 +1,207 @@
> +From 63fcc7024ae7cb989d1817dc1eeb128f702548bf Mon Sep 17 00:00:00 2001
> +From: Christian Dresel <fff at chrisi01.de>
> +Date: Sat, 24 Nov 2018 17:22:23 +0100
> +Subject: [PATCH] ipqess integration
> +
> +Signed-off-by: Christian Dresel <fff at chrisi01.de>
> +
> +rebase
> +
> +Signed-off-by: Christian Dresel <fff at chrisi01.de>
> +---
> + target/linux/ipq40xx/base-files/etc/board.d/01_leds   |  2 +-
> + .../linux/ipq40xx/base-files/etc/board.d/02_network   | 13 +++++++++++--
> + .../files-4.14/arch/arm/boot/dts/qcom-ipq4018-a42.dts | 16 ----------------
> + .../arch/arm/boot/dts/qcom-ipq4018-jalapeno.dts       | 14 --------------
> + .../files-4.14/arch/arm/boot/dts/qcom-ipq4019-a62.dts |  8 --------
> + .../arch/arm/boot/dts/qcom-ipq4028-wpj428.dts         | 16 ----------------
> + .../arch/arm/boot/dts/qcom-ipq4029-mr33.dts           | 19 +++++++++----------
> + 7 files changed, 21 insertions(+), 67 deletions(-)
> +
> +diff --git a/target/linux/ipq40xx/base-files/etc/board.d/01_leds
> b/target/linux/ipq40xx/base-files/etc/board.d/01_leds
> +index fcba2aea54..ceec20d99a 100755
> +--- a/target/linux/ipq40xx/base-files/etc/board.d/01_leds
> ++++ b/target/linux/ipq40xx/base-files/etc/board.d/01_leds
> +@@ -20,7 +20,7 @@ asus,rt-ac58u)
> + 	;;
> + avm,fritzbox-4040)
> + 	ucidef_set_led_wlan "wlan" "WLAN" "fritz4040:green:wlan" "phy0tpt"
> "phy1tpt"
> +-	ucidef_set_led_netdev "wan" "WAN" "fritz4040:green:wan" "eth1"
> ++	ucidef_set_led_switch "wan" "WAN" "fritz4040:green:wan" "switch0"
> "0x20"
> + 	ucidef_set_led_switch "lan" "LAN" "fritz4040:green:lan" "switch0"
> "0x1e"
> + 	;;
> + glinet,gl-b1300)
> +diff --git a/target/linux/ipq40xx/base-files/etc/board.d/02_network
> b/target/linux/ipq40xx/base-files/etc/board.d/02_network
> +index 03e0c0e16c..7c84139d11 100755
> +--- a/target/linux/ipq40xx/base-files/etc/board.d/02_network
> ++++ b/target/linux/ipq40xx/base-files/etc/board.d/02_network
> +@@ -26,9 +26,8 @@ asus,rt-ac58u)
> + 	ucidef_set_interface_macaddr "wan" "$wan_mac_addr"
> + 	;;
> + avm,fritzbox-4040)
> +-	ucidef_set_interfaces_lan_wan "eth0" "eth1"
> + 	ucidef_add_switch "switch0" \
> +-		"0u at eth0" "1:lan" "2:lan" "3:lan" "4:lan"
> ++		"0 at eth0" "1:lan:4" "2:lan:3" "3:lan:2" "4:lan:1" "5:wan"
> + 	;;
> + compex,wpj428)
> + 	ucidef_set_interface_lan "eth0 eth1"
> +@@ -53,6 +52,16 @@ zyxel,wre6606)
> + 	;;
> + esac
> +
> ++case "$board" in
> ++avm,fritzbox-4040)
> ++	lan_mac=$(fritz_tffs -b -n maca -i $(find_mtd_part "tffs1"))
> ++	wan_mac=$(fritz_tffs -b -n macb -i $(find_mtd_part "tffs1"))
> ++	;;
> ++esac
> ++[ -n "$lan_mac" ] && ucidef_set_interface_macaddr "lan" "$lan_mac"
> ++[ -n "$wan_mac" ] && ucidef_set_interface_macaddr "wan" "$wan_mac"
> ++
> ++
> + board_config_flush
> +
> + exit 0
> +diff --git a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> a42.dts b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> a42.dts
> +index 20330afc66..f0fb05a801 100644
> +--- a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-a42.dts
> ++++ b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> a42.dts
> +@@ -185,22 +185,6 @@
> + 	status = "okay";
> + };
> +
> +-&gmac0 {
> +-	qcom,phy_mdio_addr = <4>;
> +-	qcom,poll_required = <1>;
> +-	qcom,forced_speed = <1000>;
> +-	qcom,forced_duplex = <1>;
> +-	vlan_tag = <2 0x20>;
> +-};
> +-
> +-&gmac1 {
> +-	qcom,phy_mdio_addr = <3>;
> +-	qcom,poll_required = <1>;
> +-	qcom,forced_speed = <1000>;
> +-	qcom,forced_duplex = <1>;
> +-	vlan_tag = <1 0x10>;
> +-};
> +-
> + &usb2_hs_phy {
> + 	status = "okay";
> + };
> +diff --git a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> jalapeno.dts b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-
> ipq4018-jalapeno.dts
> +index ee203b0f1e..fd871f2985 100644
> +--- a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> jalapeno.dts
> ++++ b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4018-
> jalapeno.dts
> +@@ -230,20 +230,6 @@
> + 	status = "okay";
> + };
> +
> +-&gmac0 {
> +-	qcom,poll_required = <1>;
> +-	qcom,poll_required_dynamic = <1>;
> +-	qcom,phy_mdio_addr = <3>;
> +-	vlan_tag = <1 0x10>;
> +-};
> +-
> +-&gmac1 {
> +-	qcom,poll_required = <1>;
> +-	qcom,poll_required_dynamic = <1>;
> +-	qcom,phy_mdio_addr = <4>;
> +-	vlan_tag = <2 0x20>;
> +-};
> +-
> + &wifi0 {
> + 	status = "okay";
> + 	qcom,ath10k-calibration-variant = "8devices-Jalapeno";
> +diff --git a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4019-
> a62.dts b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4019-
> a62.dts
> +index dd6d6bd775..8e0f6f4e0c 100644
> +--- a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4019-a62.dts
> ++++ b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4019-
> a62.dts
> +@@ -195,14 +195,6 @@
> + 	status = "okay";
> + };
> +
> +-&gmac0 {
> +-	qcom,phy_mdio_addr = <3>;
> +-	qcom,poll_required = <1>;
> +-	qcom,forced_speed = <1000>;
> +-	qcom,forced_duplex = <1>;
> +-	vlan_tag = <1 0x10>;
> +-};
> +-
> + &usb2_hs_phy {
> + 	status = "okay";
> + };
> +diff --git a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4028-
> wpj428.dts b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4028-
> wpj428.dts
> +index f9f0f96ae9..79d0d22ab7 100644
> +--- a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4028-
> wpj428.dts
> ++++ b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4028-
> wpj428.dts
> +@@ -233,22 +233,6 @@
> + 	status = "okay";
> + };
> +
> +-&gmac0 {
> +-	qcom,phy_mdio_addr = <4>;
> +-	qcom,poll_required = <1>;
> +-	qcom,forced_speed = <1000>;
> +-	qcom,forced_duplex = <1>;
> +-	vlan_tag = <2 0x20>;
> +-};
> +-
> +-&gmac1 {
> +-	qcom,phy_mdio_addr = <3>;
> +-	qcom,poll_required = <1>;
> +-	qcom,forced_speed = <1000>;
> +-	qcom,forced_duplex = <1>;
> +-	vlan_tag = <1 0x10>;
> +-};
> +-
> + &usb3_ss_phy {
> + 	status = "okay";
> + };
> +diff --git a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4029-
> mr33.dts b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4029-
> mr33.dts
> +index 0be1960655..c913b3c354 100644
> +--- a/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4029-
> mr33.dts
> ++++ b/target/linux/ipq40xx/files-4.14/arch/arm/boot/dts/qcom-ipq4029-
> mr33.dts
> +@@ -97,14 +97,19 @@
> +
> + 		ess-switch at c000000 {
> + 			switch_mac_mode = <0x3>; /* mac mode for RGMII
> RMII */
> +-			switch_lan_bmp = <0x0>; /* lan port bitmap */
> +-			switch_wan_bmp = <0x10>; /* wan port bitmap */
> ++			mdio {
> ++				#address-cells = <1>;
> ++				#size-cells = <0>;
> ++
> ++				phy: phy4 at 4 {
> ++					reg = <4>;
> ++				};
> ++			};
> + 		};
> +
> + 		edma at c080000 {
> +-			qcom,single-phy;
> +-			qcom,num_gmac = <1>;
> + 			phy-mode = "rgmii-rxid";
> ++			phy-handle = <&phy>;
> + 			status = "okay";
> + 		};
> + 	};
> +@@ -138,12 +143,6 @@
> + 	status = "okay";
> + };
> +
> +-&gmac0 {
> +-	qcom,phy_mdio_addr = <1>;
> +-	qcom,poll_required = <1>;
> +-	vlan_tag = <0 0x20>;
> +-};
> +-
> + &i2c_0 {
> + 	pinctrl-0 = <&i2c_0_pins>;
> + 	pinctrl-names = "default";
> +--
> +2.11.0
> +
> diff --git a/build_patches/openwrt/0008-Enable-QinQ-on-the-switch.patch
> b/build_patches/openwrt/0008-Enable-QinQ-on-the-switch.patch
> new file mode 100644
> index 0000000..8e9ce77
> --- /dev/null
> +++ b/build_patches/openwrt/0008-Enable-QinQ-on-the-switch.patch
> @@ -0,0 +1,106 @@
> +From 57f78fa2d1d5175ce3e55c3e786b7a768eb0e56f Mon Sep 17 00:00:00 2001
> +From: Christian Dresel <fff at chrisi01.de>
> +Date: Sat, 24 Nov 2018 00:28:58 +0100
> +Subject: [PATCH] Enable-QinQ-on-the-switch
> +
> +Signed-off-by: Christian Dresel <fff at chrisi01.de>
> +---
> + .../715-ar40xx-Enable-QinQ-on-the-switch.patch     | 86
> ++++++++++++++++++++++
> + 1 file changed, 86 insertions(+)
> + create mode 100644 target/linux/ipq40xx/patches-4.14/715-ar40xx-Enable-
> QinQ-on-the-switch.patch
> +
> +diff --git a/target/linux/ipq40xx/patches-4.14/715-ar40xx-Enable-QinQ-on-the-
> switch.patch b/target/linux/ipq40xx/patches-4.14/715-ar40xx-Enable-QinQ-on-
> the-switch.patch
> +new file mode 100644
> +index 0000000000..c8edcbf5bc
> +--- /dev/null
> ++++ b/target/linux/ipq40xx/patches-4.14/715-ar40xx-Enable-QinQ-on-the-
> switch.patch
> +@@ -0,0 +1,86 @@
> ++From: Sven Eckelmann <sven at narfation.org>
> ++Date: Fri, 17 Mar 2017 11:04:50 +0100
> ++Subject: [PATCH] ar40xx: Enable QinQ on the switch
> ++
> ++The switch used in by IPQ40xx is using VLANs by default to select the
> ++outgoing port. It was therefore not possible to sent or receive 802.1q
> ++tagged frames over the CPU port. This can be allowed by changing the port
> ++configuration and lookup configuration.
> ++
> ++The resulting VLAN-tagged frames send or received by the CPU will therefore
> ++look like QinQ frames. The outer VLAN tag is the port-VLAN of the port from
> ++which the data was received or towards which the data has to be sent. The
> ++inner VLAN tag (when it exists) is the VLAN which was configrued on top of
> ++the ethernet device.
> ++
> ++---
> ++
> ++diff --git a/drivers/net/phy/ar40xx.c b/drivers/net/phy/ar40xx.c
> ++--- a/drivers/net/phy/ar40xx.c
> +++++ b/drivers/net/phy/ar40xx.c
> ++@@ -1246,6 +1246,10 @@ ar40xx_init_globals(struct ar40xx_priv *priv)
> ++ 	t = (AR40XX_PORT0_FC_THRESH_ON_DFLT << 16) |
> ++ 	      AR40XX_PORT0_FC_THRESH_OFF_DFLT;
> ++ 	ar40xx_write(priv, AR40XX_REG_PORT_FLOWCTRL_THRESH(0), t);
> +++
> +++	/* set service tag to 802.1q */
> +++	//t = ETH_P_8021Q | AR40XX_ESS_SERVICE_TAG_STAG;
> +++	//ar40xx_write(priv, AR40XX_ESS_SERVICE_TAG, t);
> ++ }
> ++
> ++ static void
> ++@@ -1571,7 +1575,11 @@ ar40xx_setup_port(struct ar40xx_priv *priv, int
> port, u32 members)
> ++ 	u32 pvid = priv->vlan_id[priv->pvid[port]];
> ++
> ++ 	if (priv->vlan) {
> ++-		egress = AR40XX_PORT_VLAN1_OUT_MODE_UNMOD;
> +++		//if (priv->vlan_tagged & BIT(port))
> +++		//	egress = AR40XX_PORT_VLAN1_OUT_MODE_TAG;
> +++		//else
> +++			egress = AR40XX_PORT_VLAN1_OUT_MODE_UNMOD;
> +++
> ++ 		ingress = AR40XX_IN_SECURE;
> ++ 	} else {
> ++ 		egress = AR40XX_PORT_VLAN1_OUT_MODE_UNTOUCH;
> ++@@ -1582,8 +1590,17 @@ ar40xx_setup_port(struct ar40xx_priv *priv, int
> port, u32 members)
> ++ 	t |= pvid << AR40XX_PORT_VLAN0_DEF_CVID_S;
> ++ 	ar40xx_write(priv, AR40XX_REG_PORT_VLAN0(port), t);
> ++
> ++-	t = AR40XX_PORT_VLAN1_PORT_VLAN_PROP;
> ++-	t |= egress << AR40XX_PORT_VLAN1_OUT_MODE_S;
> +++	t = egress << AR40XX_PORT_VLAN1_OUT_MODE_S;
> +++
> +++	///* set CPU port to core port */
> +++	//if (port == 0)
> +++	//	t |= AR40XX_PORT_VLAN1_CORE_PORT;
> +++
> +++	//if (priv->vlan_tagged & BIT(port))
> +++		t |= AR40XX_PORT_VLAN1_PORT_VLAN_PROP;
> +++	//else
> +++	//	t |= AR40XX_PORT_VLAN1_PORT_TLS_MODE;
> +++
> ++ 	ar40xx_write(priv, AR40XX_REG_PORT_VLAN1(port), t);
> ++
> ++ 	t = members;
> ++diff --git a/drivers/net/phy/ar40xx.h b/drivers/net/phy/ar40xx.h
> ++--- a/drivers/net/phy/ar40xx.h
> +++++ b/drivers/net/phy/ar40xx.h
> ++@@ -151,6 +151,9 @@ struct ar40xx_mib_desc {
> ++ #define   AR40XX_MIB_FUNC_NO_OP		0x0
> ++ #define   AR40XX_MIB_FUNC_FLUSH		0x1
> ++
> +++#define AR40XX_ESS_SERVICE_TAG		0x48
> +++#define AR40XX_ESS_SERVICE_TAG_STAG	BIT(17)
> +++
> ++ #define AR40XX_REG_PORT_STATUS(_i)		(0x07c + (_i) * 4)
> ++ #define   AR40XX_PORT_SPEED			BITS(0, 2)
> ++ #define   AR40XX_PORT_STATUS_SPEED_S	0
> ++@@ -179,6 +182,8 @@ struct ar40xx_mib_desc {
> ++ #define   AR40XX_PORT_VLAN0_DEF_CVID_S		16
> ++
> ++ #define AR40XX_REG_PORT_VLAN1(_i)		(0x424 + (_i) * 0x8)
> +++#define   AR40XX_PORT_VLAN1_CORE_PORT		BIT(9)
> +++#define   AR40XX_PORT_VLAN1_PORT_TLS_MODE	BIT(7)
> ++ #define   AR40XX_PORT_VLAN1_PORT_VLAN_PROP	BIT(6)
> ++ #define   AR40XX_PORT_VLAN1_OUT_MODE		BITS(12, 2)
> ++ #define   AR40XX_PORT_VLAN1_OUT_MODE_S		12
> +--
> +2.11.0
> +
> diff --git a/build_patches/openwrt/0009-enable-IPQ_ESS.patch
> b/build_patches/openwrt/0009-enable-IPQ_ESS.patch
> new file mode 100644
> index 0000000..8471930
> --- /dev/null
> +++ b/build_patches/openwrt/0009-enable-IPQ_ESS.patch
> @@ -0,0 +1,23 @@
> +From a9373adae4503b84c73d1f095b26683ac5e7063b Mon Sep 17 00:00:00 2001
> +From: Christian Dresel <fff at chrisi01.de>
> +Date: Sat, 24 Nov 2018 23:34:58 +0100
> +Subject: [PATCH] enable IPQ_ESS
> +
> +Signed-off-by: Christian Dresel <fff at chrisi01.de>
> +---
> + target/linux/ipq40xx/config-4.14 | 1 +
> + 1 file changed, 1 insertion(+)
> +
> +diff --git a/target/linux/ipq40xx/config-4.14 b/target/linux/ipq40xx/config-4.14
> +index fa52d58bed..184e8fd5fb 100644
> +--- a/target/linux/ipq40xx/config-4.14
> ++++ b/target/linux/ipq40xx/config-4.14
> +@@ -482,3 +482,4 @@ CONFIG_ZBOOT_ROM_BSS=0
> + CONFIG_ZBOOT_ROM_TEXT=0
> + CONFIG_ZLIB_DEFLATE=y
> + CONFIG_ZLIB_INFLATE=y
> ++CONFIG_IPQ_ESS=y
> +\ No newline at end of file
> +--
> +2.11.0
> +
> diff --git a/buildscript b/buildscript
> index f8d435c..df82a36 100755
> --- a/buildscript
> +++ b/buildscript
> @@ -279,10 +279,10 @@ cp_firmware() {
>          filename_build=${filename_build//tiny/t}
>          cp "$target/bin/targets/${chipset}/${subtarget}/$image"
> "./bin/$filename_build"
> 
> -        for region in "" "-eu" "-us"; do
> -            image_factory=${image/sysupgrade/factory$region}
> +        for factory in "factory" "factory-eu" "factory-us" "eva"; do
> +            image_factory=${image/sysupgrade/$factory}
>              if [[ -f "$target/bin/targets/${chipset}/${subtarget}/$image_factory" ]];
> then
> -              filename_build_factory=${filename_build/sysupgrade/factory$region}
> +              filename_build_factory=${filename_build/sysupgrade/$factory}
>                if [ ${#image_factory} -lt ${#filename_build_factory} ]; then
>                  echo "Warning: The factory image file name
> (${filename_build_factory}) is longer than the OpenWrt one, which might lead
> to incompatibility with the stock firmware."
>                fi
> diff --git a/src/packages/fff/fff-boardname/files/etc/uci-defaults/50-fff-
> boardname b/src/packages/fff/fff-boardname/files/etc/uci-defaults/50-fff-
> boardname
> index ee9c3d3..75f8ee8 100644
> --- a/src/packages/fff/fff-boardname/files/etc/uci-defaults/50-fff-boardname
> +++ b/src/packages/fff/fff-boardname/files/etc/uci-defaults/50-fff-boardname
> @@ -70,6 +70,9 @@ case "$BOARD" in
>      archer-c7)
>          BOARD=archer-c7-v2
>          ;;
> +    avm,fritzbox-4040)
> +        BOARD=avm_fritzbox-4040
> +        ;;
>  esac
> 
>  uci set board.model.name=$BOARD
> diff --git a/src/packages/fff/fff-network/ipq40xx/network.avm_fritzbox-4040
> b/src/packages/fff/fff-network/ipq40xx/network.avm_fritzbox-4040
> new file mode 100644
> index 0000000..80ec186
> --- /dev/null
> +++ b/src/packages/fff/fff-network/ipq40xx/network.avm_fritzbox-4040
> @@ -0,0 +1,8 @@
> +WANDEV=eth0
> +SWITCHDEV=eth0
> +CLIENT_PORTS="3 4 0t"
> +WAN_PORTS="5 0t"
> +BATMAN_PORTS="1 2 0t"
> +
> +ROUTERMAC=$(cat /sys/class/net/eth0/address)
> +
> diff --git a/src/packages/fff/fff-sysupgrade/files/etc/sysupgrade.sh
> b/src/packages/fff/fff-sysupgrade/files/etc/sysupgrade.sh
> index 87ac48a..6241514 100755
> --- a/src/packages/fff/fff-sysupgrade/files/etc/sysupgrade.sh
> +++ b/src/packages/fff/fff-sysupgrade/files/etc/sysupgrade.sh
> @@ -19,6 +19,9 @@ case $BOARD in
>      tl-wdr4900-v1 )
>          SOC="mpc85xx-g"
>          ;;
> +    avm_fritzbox-4040 )
> +        SOC="ipq40xx"
> +        ;;
>      * )
>          SOC="ar71xx-t"
>          ;;
> --
> 2.11.0
-------------- nächster Teil --------------
Ein Dateianhang mit Binärdaten wurde abgetrennt...
Dateiname   : nicht verfügbar
Dateityp    : application/pgp-signature
Dateigröße  : 834 bytes
Beschreibung: nicht verfügbar
URL         : <http://lists.freifunk.net/pipermail/franken-dev-freifunk.net/attachments/20181128/98b1b0da/attachment.sig>


Mehr Informationen über die Mailingliste franken-dev