|
@@ -51,17 +51,30 @@ static int mv88e6xxx_ptp_write(struct mv88e6xxx_chip *chip, int addr,
|
|
return chip->info->ops->avb_ops->ptp_write(chip, addr, data);
|
|
return chip->info->ops->avb_ops->ptp_write(chip, addr, data);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static int mv88e6xxx_ptp_read(struct mv88e6xxx_chip *chip, int addr,
|
|
|
|
+ u16 *data)
|
|
|
|
+{
|
|
|
|
+ if (!chip->info->ops->avb_ops->ptp_read)
|
|
|
|
+ return -EOPNOTSUPP;
|
|
|
|
+
|
|
|
|
+ return chip->info->ops->avb_ops->ptp_read(chip, addr, data, 1);
|
|
|
|
+}
|
|
|
|
+
|
|
/* TX_TSTAMP_TIMEOUT: This limits the time spent polling for a TX
|
|
/* TX_TSTAMP_TIMEOUT: This limits the time spent polling for a TX
|
|
* timestamp. When working properly, hardware will produce a timestamp
|
|
* timestamp. When working properly, hardware will produce a timestamp
|
|
* within 1ms. Software may enounter delays due to MDIO contention, so
|
|
* within 1ms. Software may enounter delays due to MDIO contention, so
|
|
* the timeout is set accordingly.
|
|
* the timeout is set accordingly.
|
|
*/
|
|
*/
|
|
-#define TX_TSTAMP_TIMEOUT msecs_to_jiffies(20)
|
|
|
|
|
|
+#define TX_TSTAMP_TIMEOUT msecs_to_jiffies(40)
|
|
|
|
|
|
int mv88e6xxx_get_ts_info(struct dsa_switch *ds, int port,
|
|
int mv88e6xxx_get_ts_info(struct dsa_switch *ds, int port,
|
|
struct ethtool_ts_info *info)
|
|
struct ethtool_ts_info *info)
|
|
{
|
|
{
|
|
- struct mv88e6xxx_chip *chip = ds->priv;
|
|
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops;
|
|
|
|
+ struct mv88e6xxx_chip *chip;
|
|
|
|
+
|
|
|
|
+ chip = ds->priv;
|
|
|
|
+ ptp_ops = chip->info->ops->ptp_ops;
|
|
|
|
|
|
if (!chip->info->ptp_support)
|
|
if (!chip->info->ptp_support)
|
|
return -EOPNOTSUPP;
|
|
return -EOPNOTSUPP;
|
|
@@ -74,17 +87,7 @@ int mv88e6xxx_get_ts_info(struct dsa_switch *ds, int port,
|
|
info->tx_types =
|
|
info->tx_types =
|
|
(1 << HWTSTAMP_TX_OFF) |
|
|
(1 << HWTSTAMP_TX_OFF) |
|
|
(1 << HWTSTAMP_TX_ON);
|
|
(1 << HWTSTAMP_TX_ON);
|
|
- info->rx_filters =
|
|
|
|
- (1 << HWTSTAMP_FILTER_NONE) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L4_EVENT) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L4_SYNC) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L2_EVENT) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L2_SYNC) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_EVENT) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_SYNC) |
|
|
|
|
- (1 << HWTSTAMP_FILTER_PTP_V2_DELAY_REQ);
|
|
|
|
|
|
+ info->rx_filters = ptp_ops->rx_filters;
|
|
|
|
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
@@ -92,10 +95,9 @@ int mv88e6xxx_get_ts_info(struct dsa_switch *ds, int port,
|
|
static int mv88e6xxx_set_hwtstamp_config(struct mv88e6xxx_chip *chip, int port,
|
|
static int mv88e6xxx_set_hwtstamp_config(struct mv88e6xxx_chip *chip, int port,
|
|
struct hwtstamp_config *config)
|
|
struct hwtstamp_config *config)
|
|
{
|
|
{
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops = chip->info->ops->ptp_ops;
|
|
struct mv88e6xxx_port_hwtstamp *ps = &chip->port_hwtstamp[port];
|
|
struct mv88e6xxx_port_hwtstamp *ps = &chip->port_hwtstamp[port];
|
|
bool tstamp_enable = false;
|
|
bool tstamp_enable = false;
|
|
- u16 port_config0;
|
|
|
|
- int err;
|
|
|
|
|
|
|
|
/* Prevent the TX/RX paths from trying to interact with the
|
|
/* Prevent the TX/RX paths from trying to interact with the
|
|
* timestamp hardware while we reconfigure it.
|
|
* timestamp hardware while we reconfigure it.
|
|
@@ -120,6 +122,14 @@ static int mv88e6xxx_set_hwtstamp_config(struct mv88e6xxx_chip *chip, int port,
|
|
/* The switch supports timestamping both L2 and L4; one cannot be
|
|
/* The switch supports timestamping both L2 and L4; one cannot be
|
|
* disabled independently of the other.
|
|
* disabled independently of the other.
|
|
*/
|
|
*/
|
|
|
|
+
|
|
|
|
+ if (!(BIT(config->rx_filter) & ptp_ops->rx_filters)) {
|
|
|
|
+ config->rx_filter = HWTSTAMP_FILTER_NONE;
|
|
|
|
+ dev_dbg(chip->dev, "Unsupported rx_filter %d\n",
|
|
|
|
+ config->rx_filter);
|
|
|
|
+ return -ERANGE;
|
|
|
|
+ }
|
|
|
|
+
|
|
switch (config->rx_filter) {
|
|
switch (config->rx_filter) {
|
|
case HWTSTAMP_FILTER_NONE:
|
|
case HWTSTAMP_FILTER_NONE:
|
|
tstamp_enable = false;
|
|
tstamp_enable = false;
|
|
@@ -141,24 +151,22 @@ static int mv88e6xxx_set_hwtstamp_config(struct mv88e6xxx_chip *chip, int port,
|
|
return -ERANGE;
|
|
return -ERANGE;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ mutex_lock(&chip->reg_lock);
|
|
if (tstamp_enable) {
|
|
if (tstamp_enable) {
|
|
- /* Disable transportSpecific value matching, so that packets
|
|
|
|
- * with either 1588 (0) and 802.1AS (1) will be timestamped.
|
|
|
|
- */
|
|
|
|
- port_config0 = MV88E6XXX_PORT_PTP_CFG0_DISABLE_TSPEC_MATCH;
|
|
|
|
|
|
+ chip->enable_count += 1;
|
|
|
|
+ if (chip->enable_count == 1 && ptp_ops->global_enable)
|
|
|
|
+ ptp_ops->global_enable(chip);
|
|
|
|
+ if (ptp_ops->port_enable)
|
|
|
|
+ ptp_ops->port_enable(chip, port);
|
|
} else {
|
|
} else {
|
|
- /* Disable PTP. This disables both RX and TX timestamping. */
|
|
|
|
- port_config0 = MV88E6XXX_PORT_PTP_CFG0_DISABLE_PTP;
|
|
|
|
|
|
+ if (ptp_ops->port_disable)
|
|
|
|
+ ptp_ops->port_disable(chip, port);
|
|
|
|
+ chip->enable_count -= 1;
|
|
|
|
+ if (chip->enable_count == 0 && ptp_ops->global_disable)
|
|
|
|
+ ptp_ops->global_disable(chip);
|
|
}
|
|
}
|
|
-
|
|
|
|
- mutex_lock(&chip->reg_lock);
|
|
|
|
- err = mv88e6xxx_port_ptp_write(chip, port, MV88E6XXX_PORT_PTP_CFG0,
|
|
|
|
- port_config0);
|
|
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
|
|
|
|
- if (err < 0)
|
|
|
|
- return err;
|
|
|
|
-
|
|
|
|
/* Once hardware has been configured, enable timestamp checks
|
|
/* Once hardware has been configured, enable timestamp checks
|
|
* in the RX/TX paths.
|
|
* in the RX/TX paths.
|
|
*/
|
|
*/
|
|
@@ -338,17 +346,18 @@ static void mv88e6xxx_get_rxts(struct mv88e6xxx_chip *chip,
|
|
static void mv88e6xxx_rxtstamp_work(struct mv88e6xxx_chip *chip,
|
|
static void mv88e6xxx_rxtstamp_work(struct mv88e6xxx_chip *chip,
|
|
struct mv88e6xxx_port_hwtstamp *ps)
|
|
struct mv88e6xxx_port_hwtstamp *ps)
|
|
{
|
|
{
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops = chip->info->ops->ptp_ops;
|
|
struct sk_buff *skb;
|
|
struct sk_buff *skb;
|
|
|
|
|
|
skb = skb_dequeue(&ps->rx_queue);
|
|
skb = skb_dequeue(&ps->rx_queue);
|
|
|
|
|
|
if (skb)
|
|
if (skb)
|
|
- mv88e6xxx_get_rxts(chip, ps, skb, MV88E6XXX_PORT_PTP_ARR0_STS,
|
|
|
|
|
|
+ mv88e6xxx_get_rxts(chip, ps, skb, ptp_ops->arr0_sts_reg,
|
|
&ps->rx_queue);
|
|
&ps->rx_queue);
|
|
|
|
|
|
skb = skb_dequeue(&ps->rx_queue2);
|
|
skb = skb_dequeue(&ps->rx_queue2);
|
|
if (skb)
|
|
if (skb)
|
|
- mv88e6xxx_get_rxts(chip, ps, skb, MV88E6XXX_PORT_PTP_ARR1_STS,
|
|
|
|
|
|
+ mv88e6xxx_get_rxts(chip, ps, skb, ptp_ops->arr1_sts_reg,
|
|
&ps->rx_queue2);
|
|
&ps->rx_queue2);
|
|
}
|
|
}
|
|
|
|
|
|
@@ -389,6 +398,7 @@ bool mv88e6xxx_port_rxtstamp(struct dsa_switch *ds, int port,
|
|
static int mv88e6xxx_txtstamp_work(struct mv88e6xxx_chip *chip,
|
|
static int mv88e6xxx_txtstamp_work(struct mv88e6xxx_chip *chip,
|
|
struct mv88e6xxx_port_hwtstamp *ps)
|
|
struct mv88e6xxx_port_hwtstamp *ps)
|
|
{
|
|
{
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops = chip->info->ops->ptp_ops;
|
|
struct skb_shared_hwtstamps shhwtstamps;
|
|
struct skb_shared_hwtstamps shhwtstamps;
|
|
u16 departure_block[4], status;
|
|
u16 departure_block[4], status;
|
|
struct sk_buff *tmp_skb;
|
|
struct sk_buff *tmp_skb;
|
|
@@ -401,7 +411,7 @@ static int mv88e6xxx_txtstamp_work(struct mv88e6xxx_chip *chip,
|
|
|
|
|
|
mutex_lock(&chip->reg_lock);
|
|
mutex_lock(&chip->reg_lock);
|
|
err = mv88e6xxx_port_ptp_read(chip, ps->port_id,
|
|
err = mv88e6xxx_port_ptp_read(chip, ps->port_id,
|
|
- MV88E6XXX_PORT_PTP_DEP_STS,
|
|
|
|
|
|
+ ptp_ops->dep_sts_reg,
|
|
departure_block,
|
|
departure_block,
|
|
ARRAY_SIZE(departure_block));
|
|
ARRAY_SIZE(departure_block));
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
@@ -425,8 +435,7 @@ static int mv88e6xxx_txtstamp_work(struct mv88e6xxx_chip *chip,
|
|
|
|
|
|
/* We have the timestamp; go ahead and clear valid now */
|
|
/* We have the timestamp; go ahead and clear valid now */
|
|
mutex_lock(&chip->reg_lock);
|
|
mutex_lock(&chip->reg_lock);
|
|
- mv88e6xxx_port_ptp_write(chip, ps->port_id,
|
|
|
|
- MV88E6XXX_PORT_PTP_DEP_STS, 0);
|
|
|
|
|
|
+ mv88e6xxx_port_ptp_write(chip, ps->port_id, ptp_ops->dep_sts_reg, 0);
|
|
mutex_unlock(&chip->reg_lock);
|
|
mutex_unlock(&chip->reg_lock);
|
|
|
|
|
|
status = departure_block[0] & MV88E6XXX_PTP_TS_STATUS_MASK;
|
|
status = departure_block[0] & MV88E6XXX_PTP_TS_STATUS_MASK;
|
|
@@ -522,8 +531,48 @@ bool mv88e6xxx_port_txtstamp(struct dsa_switch *ds, int port,
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+int mv88e6165_global_disable(struct mv88e6xxx_chip *chip)
|
|
|
|
+{
|
|
|
|
+ u16 val;
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ err = mv88e6xxx_ptp_read(chip, MV88E6165_PTP_CFG, &val);
|
|
|
|
+ if (err)
|
|
|
|
+ return err;
|
|
|
|
+ val |= MV88E6165_PTP_CFG_DISABLE_PTP;
|
|
|
|
+
|
|
|
|
+ return mv88e6xxx_ptp_write(chip, MV88E6165_PTP_CFG, val);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+int mv88e6165_global_enable(struct mv88e6xxx_chip *chip)
|
|
|
|
+{
|
|
|
|
+ u16 val;
|
|
|
|
+ int err;
|
|
|
|
+
|
|
|
|
+ err = mv88e6xxx_ptp_read(chip, MV88E6165_PTP_CFG, &val);
|
|
|
|
+ if (err)
|
|
|
|
+ return err;
|
|
|
|
+
|
|
|
|
+ val &= ~(MV88E6165_PTP_CFG_DISABLE_PTP | MV88E6165_PTP_CFG_TSPEC_MASK);
|
|
|
|
+
|
|
|
|
+ return mv88e6xxx_ptp_write(chip, MV88E6165_PTP_CFG, val);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+int mv88e6352_hwtstamp_port_disable(struct mv88e6xxx_chip *chip, int port)
|
|
|
|
+{
|
|
|
|
+ return mv88e6xxx_port_ptp_write(chip, port, MV88E6XXX_PORT_PTP_CFG0,
|
|
|
|
+ MV88E6XXX_PORT_PTP_CFG0_DISABLE_PTP);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+int mv88e6352_hwtstamp_port_enable(struct mv88e6xxx_chip *chip, int port)
|
|
|
|
+{
|
|
|
|
+ return mv88e6xxx_port_ptp_write(chip, port, MV88E6XXX_PORT_PTP_CFG0,
|
|
|
|
+ MV88E6XXX_PORT_PTP_CFG0_DISABLE_TSPEC_MATCH);
|
|
|
|
+}
|
|
|
|
+
|
|
static int mv88e6xxx_hwtstamp_port_setup(struct mv88e6xxx_chip *chip, int port)
|
|
static int mv88e6xxx_hwtstamp_port_setup(struct mv88e6xxx_chip *chip, int port)
|
|
{
|
|
{
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops = chip->info->ops->ptp_ops;
|
|
struct mv88e6xxx_port_hwtstamp *ps = &chip->port_hwtstamp[port];
|
|
struct mv88e6xxx_port_hwtstamp *ps = &chip->port_hwtstamp[port];
|
|
|
|
|
|
ps->port_id = port;
|
|
ps->port_id = port;
|
|
@@ -531,12 +580,15 @@ static int mv88e6xxx_hwtstamp_port_setup(struct mv88e6xxx_chip *chip, int port)
|
|
skb_queue_head_init(&ps->rx_queue);
|
|
skb_queue_head_init(&ps->rx_queue);
|
|
skb_queue_head_init(&ps->rx_queue2);
|
|
skb_queue_head_init(&ps->rx_queue2);
|
|
|
|
|
|
- return mv88e6xxx_port_ptp_write(chip, port, MV88E6XXX_PORT_PTP_CFG0,
|
|
|
|
- MV88E6XXX_PORT_PTP_CFG0_DISABLE_PTP);
|
|
|
|
|
|
+ if (ptp_ops->port_disable)
|
|
|
|
+ return ptp_ops->port_disable(chip, port);
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
}
|
|
}
|
|
|
|
|
|
int mv88e6xxx_hwtstamp_setup(struct mv88e6xxx_chip *chip)
|
|
int mv88e6xxx_hwtstamp_setup(struct mv88e6xxx_chip *chip)
|
|
{
|
|
{
|
|
|
|
+ const struct mv88e6xxx_ptp_ops *ptp_ops = chip->info->ops->ptp_ops;
|
|
int err;
|
|
int err;
|
|
int i;
|
|
int i;
|
|
|
|
|
|
@@ -547,6 +599,18 @@ int mv88e6xxx_hwtstamp_setup(struct mv88e6xxx_chip *chip)
|
|
return err;
|
|
return err;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ /* Disable PTP globally */
|
|
|
|
+ if (ptp_ops->global_disable) {
|
|
|
|
+ err = ptp_ops->global_disable(chip);
|
|
|
|
+ if (err)
|
|
|
|
+ return err;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /* Set the ethertype of L2 PTP messages */
|
|
|
|
+ err = mv88e6xxx_ptp_write(chip, MV88E6XXX_PTP_GC_ETYPE, ETH_P_1588);
|
|
|
|
+ if (err)
|
|
|
|
+ return err;
|
|
|
|
+
|
|
/* MV88E6XXX_PTP_MSG_TYPE is a mask of PTP message types to
|
|
/* MV88E6XXX_PTP_MSG_TYPE is a mask of PTP message types to
|
|
* timestamp. This affects all ports that have timestamping enabled,
|
|
* timestamp. This affects all ports that have timestamping enabled,
|
|
* but the timestamp config is per-port; thus we configure all events
|
|
* but the timestamp config is per-port; thus we configure all events
|