Commit 66427778 authored by Vladimir Oltean's avatar Vladimir Oltean Committed by David S. Miller

net: dsa: sja1105: Change the PTP command access pattern

The PTP command register contains enable bits for:
- Putting the 64-bit PTPCLKVAL register in add/subtract or write mode
- Taking timestamps off of the corrected vs free-running clock
- Starting/stopping the TTEthernet scheduling
- Starting/stopping PPS output
- Resetting the switch

When a command needs to be issued (e.g. "change the PTPCLKVAL from write
mode to add/subtract mode"), one cannot simply write to the command
register setting the PTPCLKADD bit to 1, because that would zeroize the
other settings. One also cannot do a read-modify-write (that would be
too easy for this hardware) because not all bits of the command register
are readable over SPI.

So this leaves us with the only option of keeping the value of the PTP
command register in the driver, and operating on that.

Actually there are 2 types of PTP operations now:
- Operations that modify the cached PTP command. These operate on
  ptp_data->cmd as a pointer.
- Operations that apply all previously cached PTP settings, but don't
  otherwise cache what they did themselves. The sja1105_ptp_reset
  function is such an example. It copies the ptp_data->cmd on stack
  before modifying and writing it to SPI.

This practically means that struct sja1105_ptp_cmd is no longer an
implementation detail, since it needs to be stored in full into struct
sja1105_ptp_data, and hence in struct sja1105_private. So the (*ptp_cmd)
function prototype can change and take struct sja1105_ptp_cmd as second
argument now.
Signed-off-by: default avatarVladimir Oltean <olteanv@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent a9d6ed7a
...@@ -72,7 +72,8 @@ struct sja1105_info { ...@@ -72,7 +72,8 @@ struct sja1105_info {
const struct sja1105_dynamic_table_ops *dyn_ops; const struct sja1105_dynamic_table_ops *dyn_ops;
const struct sja1105_table_ops *static_ops; const struct sja1105_table_ops *static_ops;
const struct sja1105_regs *regs; const struct sja1105_regs *regs;
int (*ptp_cmd)(const struct dsa_switch *ds, const void *data); int (*ptp_cmd)(const struct dsa_switch *ds,
const struct sja1105_ptp_cmd *cmd);
int (*reset_cmd)(const void *ctx, const void *data); int (*reset_cmd)(const void *ctx, const void *data);
int (*setup_rgmii_delay)(const void *ctx, int port); int (*setup_rgmii_delay)(const void *ctx, int port);
/* Prototypes from include/net/dsa.h */ /* Prototypes from include/net/dsa.h */
......
...@@ -59,10 +59,6 @@ ...@@ -59,10 +59,6 @@
#define ptp_data_to_sja1105(d) \ #define ptp_data_to_sja1105(d) \
container_of((d), struct sja1105_private, ptp_data) container_of((d), struct sja1105_private, ptp_data)
struct sja1105_ptp_cmd {
u64 resptp; /* reset */
};
static int sja1105_init_avb_params(struct sja1105_private *priv, static int sja1105_init_avb_params(struct sja1105_private *priv,
bool on) bool on)
{ {
...@@ -212,10 +208,10 @@ int sja1105_get_ts_info(struct dsa_switch *ds, int port, ...@@ -212,10 +208,10 @@ int sja1105_get_ts_info(struct dsa_switch *ds, int port,
return 0; return 0;
} }
int sja1105et_ptp_cmd(const struct dsa_switch *ds, const void *data) int sja1105et_ptp_cmd(const struct dsa_switch *ds,
const struct sja1105_ptp_cmd *cmd)
{ {
const struct sja1105_private *priv = ds->priv; const struct sja1105_private *priv = ds->priv;
const struct sja1105_ptp_cmd *cmd = data;
const struct sja1105_regs *regs = priv->info->regs; const struct sja1105_regs *regs = priv->info->regs;
const int size = SJA1105_SIZE_PTP_CMD; const int size = SJA1105_SIZE_PTP_CMD;
u8 buf[SJA1105_SIZE_PTP_CMD] = {0}; u8 buf[SJA1105_SIZE_PTP_CMD] = {0};
...@@ -229,10 +225,10 @@ int sja1105et_ptp_cmd(const struct dsa_switch *ds, const void *data) ...@@ -229,10 +225,10 @@ int sja1105et_ptp_cmd(const struct dsa_switch *ds, const void *data)
SJA1105_SIZE_PTP_CMD); SJA1105_SIZE_PTP_CMD);
} }
int sja1105pqrs_ptp_cmd(const struct dsa_switch *ds, const void *data) int sja1105pqrs_ptp_cmd(const struct dsa_switch *ds,
const struct sja1105_ptp_cmd *cmd)
{ {
const struct sja1105_private *priv = ds->priv; const struct sja1105_private *priv = ds->priv;
const struct sja1105_ptp_cmd *cmd = data;
const struct sja1105_regs *regs = priv->info->regs; const struct sja1105_regs *regs = priv->info->regs;
const int size = SJA1105_SIZE_PTP_CMD; const int size = SJA1105_SIZE_PTP_CMD;
u8 buf[SJA1105_SIZE_PTP_CMD] = {0}; u8 buf[SJA1105_SIZE_PTP_CMD] = {0};
...@@ -422,7 +418,7 @@ int sja1105_ptp_reset(struct dsa_switch *ds) ...@@ -422,7 +418,7 @@ int sja1105_ptp_reset(struct dsa_switch *ds)
{ {
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
struct sja1105_ptp_data *ptp_data = &priv->ptp_data; struct sja1105_ptp_data *ptp_data = &priv->ptp_data;
struct sja1105_ptp_cmd cmd = {0}; struct sja1105_ptp_cmd cmd = ptp_data->cmd;
int rc; int rc;
mutex_lock(&ptp_data->lock); mutex_lock(&ptp_data->lock);
......
...@@ -6,9 +6,14 @@ ...@@ -6,9 +6,14 @@
#if IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP) #if IS_ENABLED(CONFIG_NET_DSA_SJA1105_PTP)
struct sja1105_ptp_cmd {
u64 resptp; /* reset */
};
struct sja1105_ptp_data { struct sja1105_ptp_data {
struct ptp_clock_info caps; struct ptp_clock_info caps;
struct ptp_clock *clock; struct ptp_clock *clock;
struct sja1105_ptp_cmd cmd;
/* The cycle counter translates the PTP timestamps (based on /* The cycle counter translates the PTP timestamps (based on
* a free-running counter) into a software time domain. * a free-running counter) into a software time domain.
*/ */
...@@ -23,9 +28,11 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds); ...@@ -23,9 +28,11 @@ int sja1105_ptp_clock_register(struct dsa_switch *ds);
void sja1105_ptp_clock_unregister(struct dsa_switch *ds); void sja1105_ptp_clock_unregister(struct dsa_switch *ds);
int sja1105et_ptp_cmd(const struct dsa_switch *ds, const void *data); int sja1105et_ptp_cmd(const struct dsa_switch *ds,
const struct sja1105_ptp_cmd *cmd);
int sja1105pqrs_ptp_cmd(const struct dsa_switch *ds, const void *data); int sja1105pqrs_ptp_cmd(const struct dsa_switch *ds,
const struct sja1105_ptp_cmd *cmd);
int sja1105_get_ts_info(struct dsa_switch *ds, int port, int sja1105_get_ts_info(struct dsa_switch *ds, int port,
struct ethtool_ts_info *ts); struct ethtool_ts_info *ts);
...@@ -47,6 +54,8 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr); ...@@ -47,6 +54,8 @@ int sja1105_hwtstamp_set(struct dsa_switch *ds, int port, struct ifreq *ifr);
#else #else
struct sja1105_ptp_cmd;
/* Structures cannot be empty in C. Bah! /* Structures cannot be empty in C. Bah!
* Keep the mutex as the only element, which is a bit more difficult to * Keep the mutex as the only element, which is a bit more difficult to
* refactor out of sja1105_main.c anyway. * refactor out of sja1105_main.c anyway.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment