Add the rt linux 4.1.3-rt3 as base
[kvmfornfv.git] / kernel / drivers / net / dsa / bcm_sf2.h
diff --git a/kernel/drivers/net/dsa/bcm_sf2.h b/kernel/drivers/net/dsa/bcm_sf2.h
new file mode 100644 (file)
index 0000000..22e2ebf
--- /dev/null
@@ -0,0 +1,149 @@
+/*
+ * Broadcom Starfighter2 private context
+ *
+ * Copyright (C) 2014, Broadcom Corporation
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ */
+
+#ifndef __BCM_SF2_H
+#define __BCM_SF2_H
+
+#include <linux/platform_device.h>
+#include <linux/kernel.h>
+#include <linux/io.h>
+#include <linux/spinlock.h>
+#include <linux/mutex.h>
+#include <linux/mii.h>
+#include <linux/ethtool.h>
+
+#include <net/dsa.h>
+
+#include "bcm_sf2_regs.h"
+
+struct bcm_sf2_hw_params {
+       u16     top_rev;
+       u16     core_rev;
+       u16     gphy_rev;
+       u32     num_gphy;
+       u8      num_acb_queue;
+       u8      num_rgmii;
+       u8      num_ports;
+       u8      fcb_pause_override:1;
+       u8      acb_packets_inflight:1;
+};
+
+#define BCM_SF2_REGS_NAME {\
+       "core", "reg", "intrl2_0", "intrl2_1", "fcb", "acb" \
+}
+
+#define BCM_SF2_REGS_NUM       6
+
+struct bcm_sf2_port_status {
+       unsigned int link;
+
+       struct ethtool_eee eee;
+
+       u32 vlan_ctl_mask;
+};
+
+struct bcm_sf2_priv {
+       /* Base registers, keep those in order with BCM_SF2_REGS_NAME */
+       void __iomem                    *core;
+       void __iomem                    *reg;
+       void __iomem                    *intrl2_0;
+       void __iomem                    *intrl2_1;
+       void __iomem                    *fcb;
+       void __iomem                    *acb;
+
+       /* spinlock protecting access to the indirect registers */
+       spinlock_t                      indir_lock;
+
+       int                             irq0;
+       int                             irq1;
+       u32                             irq0_stat;
+       u32                             irq0_mask;
+       u32                             irq1_stat;
+       u32                             irq1_mask;
+
+       /* Mutex protecting access to the MIB counters */
+       struct mutex                    stats_mutex;
+
+       struct bcm_sf2_hw_params        hw_params;
+
+       struct bcm_sf2_port_status      port_sts[DSA_MAX_PORTS];
+
+       /* Mask of ports enabled for Wake-on-LAN */
+       u32                             wol_ports_mask;
+};
+
+struct bcm_sf2_hw_stats {
+       const char      *string;
+       u16             reg;
+       u8              sizeof_stat;
+};
+
+#define SF2_IO_MACRO(name) \
+static inline u32 name##_readl(struct bcm_sf2_priv *priv, u32 off)     \
+{                                                                      \
+       return __raw_readl(priv->name + off);                           \
+}                                                                      \
+static inline void name##_writel(struct bcm_sf2_priv *priv,            \
+                                 u32 val, u32 off)                     \
+{                                                                      \
+       __raw_writel(val, priv->name + off);                            \
+}                                                                      \
+
+/* Accesses to 64-bits register requires us to latch the hi/lo pairs
+ * using the REG_DIR_DATA_{READ,WRITE} ancillary registers. The 'indir_lock'
+ * spinlock is automatically grabbed and released to provide relative
+ * atomiticy with latched reads/writes.
+ */
+#define SF2_IO64_MACRO(name) \
+static inline u64 name##_readq(struct bcm_sf2_priv *priv, u32 off)     \
+{                                                                      \
+       u32 indir, dir;                                                 \
+       spin_lock(&priv->indir_lock);                                   \
+       dir = __raw_readl(priv->name + off);                            \
+       indir = reg_readl(priv, REG_DIR_DATA_READ);                     \
+       spin_unlock(&priv->indir_lock);                                 \
+       return (u64)indir << 32 | dir;                                  \
+}                                                                      \
+static inline void name##_writeq(struct bcm_sf2_priv *priv, u32 off,   \
+                                                       u64 val)        \
+{                                                                      \
+       spin_lock(&priv->indir_lock);                                   \
+       reg_writel(priv, upper_32_bits(val), REG_DIR_DATA_WRITE);       \
+       __raw_writel(lower_32_bits(val), priv->name + off);             \
+       spin_unlock(&priv->indir_lock);                                 \
+}
+
+#define SWITCH_INTR_L2(which)                                          \
+static inline void intrl2_##which##_mask_clear(struct bcm_sf2_priv *priv, \
+                                               u32 mask)               \
+{                                                                      \
+       intrl2_##which##_writel(priv, mask, INTRL2_CPU_MASK_CLEAR);     \
+       priv->irq##which##_mask &= ~(mask);                             \
+}                                                                      \
+static inline void intrl2_##which##_mask_set(struct bcm_sf2_priv *priv, \
+                                               u32 mask)               \
+{                                                                      \
+       intrl2_## which##_writel(priv, mask, INTRL2_CPU_MASK_SET);      \
+       priv->irq##which##_mask |= (mask);                              \
+}                                                                      \
+
+SF2_IO_MACRO(core);
+SF2_IO_MACRO(reg);
+SF2_IO64_MACRO(core);
+SF2_IO_MACRO(intrl2_0);
+SF2_IO_MACRO(intrl2_1);
+SF2_IO_MACRO(fcb);
+SF2_IO_MACRO(acb);
+
+SWITCH_INTR_L2(0);
+SWITCH_INTR_L2(1);
+
+#endif /* __BCM_SF2_H */