Commit 9970de8b authored by David S. Miller's avatar David S. Miller

Merge branch 'PTP_CLK-pin-configuration-for-SJA1105-DSA-driver'

Vladimir Oltean says:

====================
PTP_CLK pin configuration for SJA1105 DSA driver

This series adds support for the PTP_CLK pin on SJA1105 to be configured
via the PTP subsystem, in the "periodic output" and "external timestamp
input" modes.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents b0f83057 747e5eb3
...@@ -38,10 +38,13 @@ struct sja1105_regs { ...@@ -38,10 +38,13 @@ struct sja1105_regs {
u64 config; u64 config;
u64 sgmii; u64 sgmii;
u64 rmii_pll1; u64 rmii_pll1;
u64 ptppinst;
u64 ptppindur;
u64 ptp_control; u64 ptp_control;
u64 ptpclkval; u64 ptpclkval;
u64 ptpclkrate; u64 ptpclkrate;
u64 ptpclkcorp; u64 ptpclkcorp;
u64 ptpsyncts;
u64 ptpschtm; u64 ptpschtm;
u64 ptpegr_ts[SJA1105_NUM_PORTS]; u64 ptpegr_ts[SJA1105_NUM_PORTS];
u64 pad_mii_tx[SJA1105_NUM_PORTS]; u64 pad_mii_tx[SJA1105_NUM_PORTS];
...@@ -214,5 +217,7 @@ size_t sja1105_vlan_lookup_entry_packing(void *buf, void *entry_ptr, ...@@ -214,5 +217,7 @@ size_t sja1105_vlan_lookup_entry_packing(void *buf, void *entry_ptr,
enum packing_op op); enum packing_op op);
size_t sja1105pqrs_mac_config_entry_packing(void *buf, void *entry_ptr, size_t sja1105pqrs_mac_config_entry_packing(void *buf, void *entry_ptr,
enum packing_op op); enum packing_op op);
size_t sja1105pqrs_avb_params_entry_packing(void *buf, void *entry_ptr,
enum packing_op op);
#endif #endif
...@@ -124,6 +124,9 @@ ...@@ -124,6 +124,9 @@
#define SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD \ #define SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD \
SJA1105_SIZE_DYN_CMD SJA1105_SIZE_DYN_CMD
#define SJA1105PQRS_SIZE_AVB_PARAMS_DYN_CMD \
(SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY)
#define SJA1105_MAX_DYN_CMD_SIZE \ #define SJA1105_MAX_DYN_CMD_SIZE \
SJA1105PQRS_SIZE_MAC_CONFIG_DYN_CMD SJA1105PQRS_SIZE_MAC_CONFIG_DYN_CMD
...@@ -481,6 +484,18 @@ sja1105et_general_params_entry_packing(void *buf, void *entry_ptr, ...@@ -481,6 +484,18 @@ sja1105et_general_params_entry_packing(void *buf, void *entry_ptr,
return 0; return 0;
} }
static void
sja1105pqrs_avb_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd,
enum packing_op op)
{
u8 *p = buf + SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY;
const int size = SJA1105_SIZE_DYN_CMD;
sja1105_packing(p, &cmd->valid, 31, 31, size, op);
sja1105_packing(p, &cmd->errors, 30, 30, size, op);
sja1105_packing(p, &cmd->rdwrset, 29, 29, size, op);
}
#define OP_READ BIT(0) #define OP_READ BIT(0)
#define OP_WRITE BIT(1) #define OP_WRITE BIT(1)
#define OP_DEL BIT(2) #define OP_DEL BIT(2)
...@@ -610,7 +625,14 @@ struct sja1105_dynamic_table_ops sja1105pqrs_dyn_ops[BLK_IDX_MAX_DYN] = { ...@@ -610,7 +625,14 @@ struct sja1105_dynamic_table_ops sja1105pqrs_dyn_ops[BLK_IDX_MAX_DYN] = {
.addr = 0x38, .addr = 0x38,
}, },
[BLK_IDX_L2_FORWARDING_PARAMS] = {0}, [BLK_IDX_L2_FORWARDING_PARAMS] = {0},
[BLK_IDX_AVB_PARAMS] = {0}, [BLK_IDX_AVB_PARAMS] = {
.entry_packing = sja1105pqrs_avb_params_entry_packing,
.cmd_packing = sja1105pqrs_avb_params_cmd_packing,
.max_entry_count = SJA1105_MAX_AVB_PARAMS_COUNT,
.access = (OP_READ | OP_WRITE),
.packed_size = SJA1105PQRS_SIZE_AVB_PARAMS_DYN_CMD,
.addr = 0x8003,
},
[BLK_IDX_GENERAL_PARAMS] = { [BLK_IDX_GENERAL_PARAMS] = {
.entry_packing = sja1105et_general_params_entry_packing, .entry_packing = sja1105et_general_params_entry_packing,
.cmd_packing = sja1105et_general_params_cmd_packing, .cmd_packing = sja1105et_general_params_cmd_packing,
......
...@@ -479,6 +479,43 @@ static int sja1105_init_general_params(struct sja1105_private *priv) ...@@ -479,6 +479,43 @@ static int sja1105_init_general_params(struct sja1105_private *priv)
return 0; return 0;
} }
static int sja1105_init_avb_params(struct sja1105_private *priv)
{
struct sja1105_avb_params_entry *avb;
struct sja1105_table *table;
table = &priv->static_config.tables[BLK_IDX_AVB_PARAMS];
/* Discard previous AVB Parameters Table */
if (table->entry_count) {
kfree(table->entries);
table->entry_count = 0;
}
table->entries = kcalloc(SJA1105_MAX_AVB_PARAMS_COUNT,
table->ops->unpacked_entry_size, GFP_KERNEL);
if (!table->entries)
return -ENOMEM;
table->entry_count = SJA1105_MAX_AVB_PARAMS_COUNT;
avb = table->entries;
/* Configure the MAC addresses for meta frames */
avb->destmeta = SJA1105_META_DMAC;
avb->srcmeta = SJA1105_META_SMAC;
/* On P/Q/R/S, configure the direction of the PTP_CLK pin as input by
* default. This is because there might be boards with a hardware
* layout where enabling the pin as output might cause an electrical
* clash. On E/T the pin is always an output, which the board designers
* probably already knew, so even if there are going to be electrical
* issues, there's nothing we can do.
*/
avb->cas_master = false;
return 0;
}
#define SJA1105_RATE_MBPS(speed) (((speed) * 64000) / 1000) #define SJA1105_RATE_MBPS(speed) (((speed) * 64000) / 1000)
static void sja1105_setup_policer(struct sja1105_l2_policing_entry *policing, static void sja1105_setup_policer(struct sja1105_l2_policing_entry *policing,
...@@ -567,6 +604,9 @@ static int sja1105_static_config_load(struct sja1105_private *priv, ...@@ -567,6 +604,9 @@ static int sja1105_static_config_load(struct sja1105_private *priv,
if (rc < 0) if (rc < 0)
return rc; return rc;
rc = sja1105_init_general_params(priv); rc = sja1105_init_general_params(priv);
if (rc < 0)
return rc;
rc = sja1105_init_avb_params(priv);
if (rc < 0) if (rc < 0)
return rc; return rc;
......
This diff is collapsed.
...@@ -21,7 +21,36 @@ static inline s64 sja1105_ticks_to_ns(s64 ticks) ...@@ -21,7 +21,36 @@ static inline s64 sja1105_ticks_to_ns(s64 ticks)
return ticks * SJA1105_TICK_NS; return ticks * SJA1105_TICK_NS;
} }
/* Calculate the first base_time in the future that satisfies this
* relationship:
*
* future_base_time = base_time + N x cycle_time >= now, or
*
* now - base_time
* N >= ---------------
* cycle_time
*
* Because N is an integer, the ceiling value of the above "a / b" ratio
* is in fact precisely the floor value of "(a + b - 1) / b", which is
* easier to calculate only having integer division tools.
*/
static inline s64 future_base_time(s64 base_time, s64 cycle_time, s64 now)
{
s64 a, b, n;
if (base_time >= now)
return base_time;
a = now - base_time;
b = cycle_time;
n = div_s64(a + b - 1, b);
return base_time + n * cycle_time;
}
struct sja1105_ptp_cmd { struct sja1105_ptp_cmd {
u64 startptpcp; /* start toggling PTP_CLK pin */
u64 stopptpcp; /* stop toggling PTP_CLK pin */
u64 ptpstrtsch; /* start schedule */ u64 ptpstrtsch; /* start schedule */
u64 ptpstopsch; /* stop schedule */ u64 ptpstopsch; /* stop schedule */
u64 resptp; /* reset */ u64 resptp; /* reset */
...@@ -30,12 +59,14 @@ struct sja1105_ptp_cmd { ...@@ -30,12 +59,14 @@ struct sja1105_ptp_cmd {
}; };
struct sja1105_ptp_data { struct sja1105_ptp_data {
struct delayed_work extts_work;
struct sk_buff_head skb_rxtstamp_queue; struct sk_buff_head skb_rxtstamp_queue;
struct ptp_clock_info caps; struct ptp_clock_info caps;
struct ptp_clock *clock; struct ptp_clock *clock;
struct sja1105_ptp_cmd cmd; struct sja1105_ptp_cmd cmd;
/* Serializes all operations on the PTP hardware clock */ /* Serializes all operations on the PTP hardware clock */
struct mutex lock; struct mutex lock;
u64 ptpsyncts;
}; };
int sja1105_ptp_clock_register(struct dsa_switch *ds); int sja1105_ptp_clock_register(struct dsa_switch *ds);
......
...@@ -458,6 +458,8 @@ static struct sja1105_regs sja1105et_regs = { ...@@ -458,6 +458,8 @@ static struct sja1105_regs sja1105et_regs = {
.rmii_ext_tx_clk = {0x100018, 0x10001F, 0x100026, 0x10002D, 0x100034}, .rmii_ext_tx_clk = {0x100018, 0x10001F, 0x100026, 0x10002D, 0x100034},
.ptpegr_ts = {0xC0, 0xC2, 0xC4, 0xC6, 0xC8}, .ptpegr_ts = {0xC0, 0xC2, 0xC4, 0xC6, 0xC8},
.ptpschtm = 0x12, /* Spans 0x12 to 0x13 */ .ptpschtm = 0x12, /* Spans 0x12 to 0x13 */
.ptppinst = 0x14,
.ptppindur = 0x16,
.ptp_control = 0x17, .ptp_control = 0x17,
.ptpclkval = 0x18, /* Spans 0x18 to 0x19 */ .ptpclkval = 0x18, /* Spans 0x18 to 0x19 */
.ptpclkrate = 0x1A, .ptpclkrate = 0x1A,
...@@ -491,10 +493,13 @@ static struct sja1105_regs sja1105pqrs_regs = { ...@@ -491,10 +493,13 @@ static struct sja1105_regs sja1105pqrs_regs = {
.qlevel = {0x604, 0x614, 0x624, 0x634, 0x644}, .qlevel = {0x604, 0x614, 0x624, 0x634, 0x644},
.ptpegr_ts = {0xC0, 0xC4, 0xC8, 0xCC, 0xD0}, .ptpegr_ts = {0xC0, 0xC4, 0xC8, 0xCC, 0xD0},
.ptpschtm = 0x13, /* Spans 0x13 to 0x14 */ .ptpschtm = 0x13, /* Spans 0x13 to 0x14 */
.ptppinst = 0x15,
.ptppindur = 0x17,
.ptp_control = 0x18, .ptp_control = 0x18,
.ptpclkval = 0x19, .ptpclkval = 0x19,
.ptpclkrate = 0x1B, .ptpclkrate = 0x1B,
.ptpclkcorp = 0x1E, .ptpclkcorp = 0x1E,
.ptpsyncts = 0x1F,
}; };
struct sja1105_info sja1105e_info = { struct sja1105_info sja1105e_info = {
......
...@@ -102,12 +102,13 @@ static size_t sja1105et_avb_params_entry_packing(void *buf, void *entry_ptr, ...@@ -102,12 +102,13 @@ static size_t sja1105et_avb_params_entry_packing(void *buf, void *entry_ptr,
return size; return size;
} }
static size_t sja1105pqrs_avb_params_entry_packing(void *buf, void *entry_ptr, size_t sja1105pqrs_avb_params_entry_packing(void *buf, void *entry_ptr,
enum packing_op op) enum packing_op op)
{ {
const size_t size = SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY; const size_t size = SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY;
struct sja1105_avb_params_entry *entry = entry_ptr; struct sja1105_avb_params_entry *entry = entry_ptr;
sja1105_packing(buf, &entry->cas_master, 126, 126, size, op);
sja1105_packing(buf, &entry->destmeta, 125, 78, size, op); sja1105_packing(buf, &entry->destmeta, 125, 78, size, op);
sja1105_packing(buf, &entry->srcmeta, 77, 30, size, op); sja1105_packing(buf, &entry->srcmeta, 77, 30, size, op);
return size; return size;
......
...@@ -230,6 +230,7 @@ struct sja1105_l2_policing_entry { ...@@ -230,6 +230,7 @@ struct sja1105_l2_policing_entry {
}; };
struct sja1105_avb_params_entry { struct sja1105_avb_params_entry {
u64 cas_master;
u64 destmeta; u64 destmeta;
u64 srcmeta; u64 srcmeta;
}; };
......
...@@ -28,33 +28,6 @@ static s64 sja1105_delta_to_ns(s64 delta) ...@@ -28,33 +28,6 @@ static s64 sja1105_delta_to_ns(s64 delta)
return delta * 200; return delta * 200;
} }
/* Calculate the first base_time in the future that satisfies this
* relationship:
*
* future_base_time = base_time + N x cycle_time >= now, or
*
* now - base_time
* N >= ---------------
* cycle_time
*
* Because N is an integer, the ceiling value of the above "a / b" ratio
* is in fact precisely the floor value of "(a + b - 1) / b", which is
* easier to calculate only having integer division tools.
*/
static s64 future_base_time(s64 base_time, s64 cycle_time, s64 now)
{
s64 a, b, n;
if (base_time >= now)
return base_time;
a = now - base_time;
b = cycle_time;
n = div_s64(a + b - 1, b);
return base_time + n * cycle_time;
}
static int sja1105_tas_set_runtime_params(struct sja1105_private *priv) static int sja1105_tas_set_runtime_params(struct sja1105_private *priv)
{ {
struct sja1105_tas_data *tas_data = &priv->tas_data; struct sja1105_tas_data *tas_data = &priv->tas_data;
......
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