Commit 6a244b75 authored by Alex Elder's avatar Alex Elder Committed by Jakub Kicinski

net: ipa: introduce ipa_reg()

Create a new function that returns a register descriptor given its
ID.  Change ipa_reg_offset() and ipa_reg_n_offset() so they take a
register descriptor argument rather than an IPA pointer and register
ID.  Have them accept null pointers (and return an invalid 0 offset),
to avoid the need for excessive error checking.  (A warning is issued
whenever ipa_reg() returns 0).

Call ipa_reg() or ipa_reg_n() to look up information about the
register before calls to ipa_reg_offset() and ipa_reg_n_offset().
Delay looking up offsets until they're needed to read or write
registers.
Signed-off-by: default avatarAlex Elder <elder@linaro.org>
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent 82a06807
...@@ -305,6 +305,7 @@ static bool ipa_cmd_register_write_offset_valid(struct ipa *ipa, ...@@ -305,6 +305,7 @@ static bool ipa_cmd_register_write_offset_valid(struct ipa *ipa,
/* Check whether offsets passed to register_write are valid */ /* Check whether offsets passed to register_write are valid */
static bool ipa_cmd_register_write_valid(struct ipa *ipa) static bool ipa_cmd_register_write_valid(struct ipa *ipa)
{ {
const struct ipa_reg *reg;
const char *name; const char *name;
u32 offset; u32 offset;
...@@ -312,7 +313,8 @@ static bool ipa_cmd_register_write_valid(struct ipa *ipa) ...@@ -312,7 +313,8 @@ static bool ipa_cmd_register_write_valid(struct ipa *ipa)
* offset will fit in a register write IPA immediate command. * offset will fit in a register write IPA immediate command.
*/ */
if (ipa_table_hash_support(ipa)) { if (ipa_table_hash_support(ipa)) {
offset = ipa_reg_offset(ipa, FILT_ROUT_HASH_FLUSH); reg = ipa_reg(ipa, FILT_ROUT_HASH_FLUSH);
offset = ipa_reg_offset(reg);
name = "filter/route hash flush"; name = "filter/route hash flush";
if (!ipa_cmd_register_write_offset_valid(ipa, name, offset)) if (!ipa_cmd_register_write_offset_valid(ipa, name, offset))
return false; return false;
...@@ -325,7 +327,8 @@ static bool ipa_cmd_register_write_valid(struct ipa *ipa) ...@@ -325,7 +327,8 @@ static bool ipa_cmd_register_write_valid(struct ipa *ipa)
* worst case (highest endpoint number) offset of that endpoint * worst case (highest endpoint number) offset of that endpoint
* fits in the register write command field(s) that must hold it. * fits in the register write command field(s) that must hold it.
*/ */
offset = ipa_reg_n_offset(ipa, ENDP_STATUS, IPA_ENDPOINT_COUNT - 1); reg = ipa_reg(ipa, ENDP_STATUS);
offset = ipa_reg_n_offset(reg, IPA_ENDPOINT_COUNT - 1);
name = "maximal endpoint status"; name = "maximal endpoint status";
if (!ipa_cmd_register_write_offset_valid(ipa, name, offset)) if (!ipa_cmd_register_write_offset_valid(ipa, name, offset))
return false; return false;
......
This diff is collapsed.
...@@ -53,13 +53,15 @@ static void ipa_interrupt_process(struct ipa_interrupt *interrupt, u32 irq_id) ...@@ -53,13 +53,15 @@ static void ipa_interrupt_process(struct ipa_interrupt *interrupt, u32 irq_id)
{ {
bool uc_irq = ipa_interrupt_uc(interrupt, irq_id); bool uc_irq = ipa_interrupt_uc(interrupt, irq_id);
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
const struct ipa_reg *reg;
u32 mask = BIT(irq_id); u32 mask = BIT(irq_id);
u32 offset; u32 offset;
/* For microcontroller interrupts, clear the interrupt right away, /* For microcontroller interrupts, clear the interrupt right away,
* "to avoid clearing unhandled interrupts." * "to avoid clearing unhandled interrupts."
*/ */
offset = ipa_reg_offset(ipa, IPA_IRQ_CLR); reg = ipa_reg(ipa, IPA_IRQ_CLR);
offset = ipa_reg_offset(reg);
if (uc_irq) if (uc_irq)
iowrite32(mask, ipa->reg_virt + offset); iowrite32(mask, ipa->reg_virt + offset);
...@@ -80,6 +82,7 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id) ...@@ -80,6 +82,7 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id)
struct ipa_interrupt *interrupt = dev_id; struct ipa_interrupt *interrupt = dev_id;
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
u32 enabled = interrupt->enabled; u32 enabled = interrupt->enabled;
const struct ipa_reg *reg;
struct device *dev; struct device *dev;
u32 pending; u32 pending;
u32 offset; u32 offset;
...@@ -95,7 +98,8 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id) ...@@ -95,7 +98,8 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id)
* including conditions whose interrupt is not enabled. Handle * including conditions whose interrupt is not enabled. Handle
* only the enabled ones. * only the enabled ones.
*/ */
offset = ipa_reg_offset(ipa, IPA_IRQ_STTS); reg = ipa_reg(ipa, IPA_IRQ_STTS);
offset = ipa_reg_offset(reg);
pending = ioread32(ipa->reg_virt + offset); pending = ioread32(ipa->reg_virt + offset);
while ((mask = pending & enabled)) { while ((mask = pending & enabled)) {
do { do {
...@@ -112,7 +116,8 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id) ...@@ -112,7 +116,8 @@ static irqreturn_t ipa_isr_thread(int irq, void *dev_id)
if (pending) { if (pending) {
dev_dbg(dev, "clearing disabled IPA interrupts 0x%08x\n", dev_dbg(dev, "clearing disabled IPA interrupts 0x%08x\n",
pending); pending);
offset = ipa_reg_offset(ipa, IPA_IRQ_CLR); reg = ipa_reg(ipa, IPA_IRQ_CLR);
offset = ipa_reg_offset(reg);
iowrite32(pending, ipa->reg_virt + offset); iowrite32(pending, ipa->reg_virt + offset);
} }
out_power_put: out_power_put:
...@@ -128,6 +133,7 @@ static void ipa_interrupt_suspend_control(struct ipa_interrupt *interrupt, ...@@ -128,6 +133,7 @@ static void ipa_interrupt_suspend_control(struct ipa_interrupt *interrupt,
{ {
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
u32 mask = BIT(endpoint_id); u32 mask = BIT(endpoint_id);
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
...@@ -137,7 +143,8 @@ static void ipa_interrupt_suspend_control(struct ipa_interrupt *interrupt, ...@@ -137,7 +143,8 @@ static void ipa_interrupt_suspend_control(struct ipa_interrupt *interrupt,
if (ipa->version == IPA_VERSION_3_0) if (ipa->version == IPA_VERSION_3_0)
return; return;
offset = ipa_reg_offset(ipa, IRQ_SUSPEND_EN); reg = ipa_reg(ipa, IRQ_SUSPEND_EN);
offset = ipa_reg_offset(reg);
val = ioread32(ipa->reg_virt + offset); val = ioread32(ipa->reg_virt + offset);
if (enable) if (enable)
val |= mask; val |= mask;
...@@ -164,15 +171,18 @@ ipa_interrupt_suspend_disable(struct ipa_interrupt *interrupt, u32 endpoint_id) ...@@ -164,15 +171,18 @@ ipa_interrupt_suspend_disable(struct ipa_interrupt *interrupt, u32 endpoint_id)
void ipa_interrupt_suspend_clear_all(struct ipa_interrupt *interrupt) void ipa_interrupt_suspend_clear_all(struct ipa_interrupt *interrupt)
{ {
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
const struct ipa_reg *reg;
u32 val; u32 val;
val = ioread32(ipa->reg_virt + ipa_reg_offset(ipa, IRQ_SUSPEND_INFO)); reg = ipa_reg(ipa, IRQ_SUSPEND_INFO);
val = ioread32(ipa->reg_virt + ipa_reg_offset(reg));
/* SUSPEND interrupt status isn't cleared on IPA version 3.0 */ /* SUSPEND interrupt status isn't cleared on IPA version 3.0 */
if (ipa->version == IPA_VERSION_3_0) if (ipa->version == IPA_VERSION_3_0)
return; return;
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, IRQ_SUSPEND_CLR)); reg = ipa_reg(ipa, IRQ_SUSPEND_CLR);
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
} }
/* Simulate arrival of an IPA TX_SUSPEND interrupt */ /* Simulate arrival of an IPA TX_SUSPEND interrupt */
...@@ -186,7 +196,7 @@ void ipa_interrupt_add(struct ipa_interrupt *interrupt, ...@@ -186,7 +196,7 @@ void ipa_interrupt_add(struct ipa_interrupt *interrupt,
enum ipa_irq_id ipa_irq, ipa_irq_handler_t handler) enum ipa_irq_id ipa_irq, ipa_irq_handler_t handler)
{ {
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
u32 offset; const struct ipa_reg *reg;
if (WARN_ON(ipa_irq >= IPA_IRQ_COUNT)) if (WARN_ON(ipa_irq >= IPA_IRQ_COUNT))
return; return;
...@@ -195,8 +205,9 @@ void ipa_interrupt_add(struct ipa_interrupt *interrupt, ...@@ -195,8 +205,9 @@ void ipa_interrupt_add(struct ipa_interrupt *interrupt,
/* Update the IPA interrupt mask to enable it */ /* Update the IPA interrupt mask to enable it */
interrupt->enabled |= BIT(ipa_irq); interrupt->enabled |= BIT(ipa_irq);
offset = ipa_reg_offset(ipa, IPA_IRQ_EN);
iowrite32(interrupt->enabled, ipa->reg_virt + offset); reg = ipa_reg(ipa, IPA_IRQ_EN);
iowrite32(interrupt->enabled, ipa->reg_virt + ipa_reg_offset(reg));
} }
/* Remove the handler for an IPA interrupt type */ /* Remove the handler for an IPA interrupt type */
...@@ -204,15 +215,16 @@ void ...@@ -204,15 +215,16 @@ void
ipa_interrupt_remove(struct ipa_interrupt *interrupt, enum ipa_irq_id ipa_irq) ipa_interrupt_remove(struct ipa_interrupt *interrupt, enum ipa_irq_id ipa_irq)
{ {
struct ipa *ipa = interrupt->ipa; struct ipa *ipa = interrupt->ipa;
u32 offset; const struct ipa_reg *reg;
if (WARN_ON(ipa_irq >= IPA_IRQ_COUNT)) if (WARN_ON(ipa_irq >= IPA_IRQ_COUNT))
return; return;
/* Update the IPA interrupt mask to disable it */ /* Update the IPA interrupt mask to disable it */
interrupt->enabled &= ~BIT(ipa_irq); interrupt->enabled &= ~BIT(ipa_irq);
offset = ipa_reg_offset(ipa, IPA_IRQ_EN);
iowrite32(interrupt->enabled, ipa->reg_virt + offset); reg = ipa_reg(ipa, IPA_IRQ_EN);
iowrite32(interrupt->enabled, ipa->reg_virt + ipa_reg_offset(reg));
interrupt->handler[ipa_irq] = NULL; interrupt->handler[ipa_irq] = NULL;
} }
...@@ -222,6 +234,7 @@ struct ipa_interrupt *ipa_interrupt_config(struct ipa *ipa) ...@@ -222,6 +234,7 @@ struct ipa_interrupt *ipa_interrupt_config(struct ipa *ipa)
{ {
struct device *dev = &ipa->pdev->dev; struct device *dev = &ipa->pdev->dev;
struct ipa_interrupt *interrupt; struct ipa_interrupt *interrupt;
const struct ipa_reg *reg;
unsigned int irq; unsigned int irq;
int ret; int ret;
...@@ -240,7 +253,8 @@ struct ipa_interrupt *ipa_interrupt_config(struct ipa *ipa) ...@@ -240,7 +253,8 @@ struct ipa_interrupt *ipa_interrupt_config(struct ipa *ipa)
interrupt->irq = irq; interrupt->irq = irq;
/* Start with all IPA interrupts disabled */ /* Start with all IPA interrupts disabled */
iowrite32(0, ipa->reg_virt + ipa_reg_offset(ipa, IPA_IRQ_EN)); reg = ipa_reg(ipa, IPA_IRQ_EN);
iowrite32(0, ipa->reg_virt + ipa_reg_offset(reg));
ret = request_threaded_irq(irq, NULL, ipa_isr_thread, IRQF_ONESHOT, ret = request_threaded_irq(irq, NULL, ipa_isr_thread, IRQF_ONESHOT,
"ipa", interrupt); "ipa", interrupt);
......
...@@ -186,20 +186,22 @@ static void ipa_teardown(struct ipa *ipa) ...@@ -186,20 +186,22 @@ static void ipa_teardown(struct ipa *ipa)
static void static void
ipa_hardware_config_bcr(struct ipa *ipa, const struct ipa_data *data) ipa_hardware_config_bcr(struct ipa *ipa, const struct ipa_data *data)
{ {
const struct ipa_reg *reg;
u32 val; u32 val;
/* IPA v4.5+ has no backward compatibility register */ /* IPA v4.5+ has no backward compatibility register */
if (ipa->version >= IPA_VERSION_4_5) if (ipa->version >= IPA_VERSION_4_5)
return; return;
reg = ipa_reg(ipa, IPA_BCR);
val = data->backward_compat; val = data->backward_compat;
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, IPA_BCR));
} }
static void ipa_hardware_config_tx(struct ipa *ipa) static void ipa_hardware_config_tx(struct ipa *ipa)
{ {
enum ipa_version version = ipa->version; enum ipa_version version = ipa->version;
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
...@@ -207,7 +209,9 @@ static void ipa_hardware_config_tx(struct ipa *ipa) ...@@ -207,7 +209,9 @@ static void ipa_hardware_config_tx(struct ipa *ipa)
return; return;
/* Disable PA mask to allow HOLB drop */ /* Disable PA mask to allow HOLB drop */
offset = ipa_reg_offset(ipa, IPA_TX_CFG); reg = ipa_reg(ipa, IPA_TX_CFG);
offset = ipa_reg_offset(reg);
val = ioread32(ipa->reg_virt + offset); val = ioread32(ipa->reg_virt + offset);
val &= ~PA_MASK_EN_FMASK; val &= ~PA_MASK_EN_FMASK;
...@@ -218,28 +222,29 @@ static void ipa_hardware_config_tx(struct ipa *ipa) ...@@ -218,28 +222,29 @@ static void ipa_hardware_config_tx(struct ipa *ipa)
static void ipa_hardware_config_clkon(struct ipa *ipa) static void ipa_hardware_config_clkon(struct ipa *ipa)
{ {
enum ipa_version version = ipa->version; enum ipa_version version = ipa->version;
const struct ipa_reg *reg;
u32 val; u32 val;
if (version < IPA_VERSION_3_1 || version >= IPA_VERSION_4_5) if (version >= IPA_VERSION_4_5)
return; return;
/* Implement some hardware workarounds */ if (version < IPA_VERSION_4_0 && version != IPA_VERSION_3_1)
return;
if (version >= IPA_VERSION_4_0) { /* Implement some hardware workarounds */
/* Enable open global clocks in the CLKON configuration */ reg = ipa_reg(ipa, CLKON_CFG);
val = GLOBAL_FMASK | GLOBAL_2X_CLK_FMASK; if (version == IPA_VERSION_3_1)
} else if (version == IPA_VERSION_3_1) {
val = MISC_FMASK; /* Disable MISC clock gating */ val = MISC_FMASK; /* Disable MISC clock gating */
} else { else /* Enable open global clocks in the CLKON configuration */
return; val = GLOBAL_FMASK | GLOBAL_2X_CLK_FMASK; /* IPA v4.0+ */
}
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, CLKON_CFG)); iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
} }
/* Configure bus access behavior for IPA components */ /* Configure bus access behavior for IPA components */
static void ipa_hardware_config_comp(struct ipa *ipa) static void ipa_hardware_config_comp(struct ipa *ipa)
{ {
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
...@@ -247,7 +252,8 @@ static void ipa_hardware_config_comp(struct ipa *ipa) ...@@ -247,7 +252,8 @@ static void ipa_hardware_config_comp(struct ipa *ipa)
if (ipa->version < IPA_VERSION_4_0) if (ipa->version < IPA_VERSION_4_0)
return; return;
offset = ipa_reg_offset(ipa, COMP_CFG); reg = ipa_reg(ipa, COMP_CFG);
offset = ipa_reg_offset(reg);
val = ioread32(ipa->reg_virt + offset); val = ioread32(ipa->reg_virt + offset);
if (ipa->version == IPA_VERSION_4_0) { if (ipa->version == IPA_VERSION_4_0) {
...@@ -272,6 +278,7 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data) ...@@ -272,6 +278,7 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data)
{ {
const struct ipa_qsb_data *data0; const struct ipa_qsb_data *data0;
const struct ipa_qsb_data *data1; const struct ipa_qsb_data *data1;
const struct ipa_reg *reg;
u32 val; u32 val;
/* QMB 0 represents DDR; QMB 1 (if present) represents PCIe */ /* QMB 0 represents DDR; QMB 1 (if present) represents PCIe */
...@@ -280,13 +287,18 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data) ...@@ -280,13 +287,18 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data)
data1 = &data->qsb_data[IPA_QSB_MASTER_PCIE]; data1 = &data->qsb_data[IPA_QSB_MASTER_PCIE];
/* Max outstanding write accesses for QSB masters */ /* Max outstanding write accesses for QSB masters */
reg = ipa_reg(ipa, QSB_MAX_WRITES);
val = u32_encode_bits(data0->max_writes, GEN_QMB_0_MAX_WRITES_FMASK); val = u32_encode_bits(data0->max_writes, GEN_QMB_0_MAX_WRITES_FMASK);
if (data->qsb_count > 1) if (data->qsb_count > 1)
val |= u32_encode_bits(data1->max_writes, val |= u32_encode_bits(data1->max_writes,
GEN_QMB_1_MAX_WRITES_FMASK); GEN_QMB_1_MAX_WRITES_FMASK);
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, QSB_MAX_WRITES));
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
/* Max outstanding read accesses for QSB masters */ /* Max outstanding read accesses for QSB masters */
reg = ipa_reg(ipa, QSB_MAX_READS);
val = u32_encode_bits(data0->max_reads, GEN_QMB_0_MAX_READS_FMASK); val = u32_encode_bits(data0->max_reads, GEN_QMB_0_MAX_READS_FMASK);
if (ipa->version >= IPA_VERSION_4_0) if (ipa->version >= IPA_VERSION_4_0)
val |= u32_encode_bits(data0->max_reads_beats, val |= u32_encode_bits(data0->max_reads_beats,
...@@ -298,7 +310,8 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data) ...@@ -298,7 +310,8 @@ ipa_hardware_config_qsb(struct ipa *ipa, const struct ipa_data *data)
val |= u32_encode_bits(data1->max_reads_beats, val |= u32_encode_bits(data1->max_reads_beats,
GEN_QMB_1_MAX_READS_BEATS_FMASK); GEN_QMB_1_MAX_READS_BEATS_FMASK);
} }
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, QSB_MAX_READS));
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
} }
/* The internal inactivity timer clock is used for the aggregation timer */ /* The internal inactivity timer clock is used for the aggregation timer */
...@@ -334,13 +347,15 @@ static __always_inline u32 ipa_aggr_granularity_val(u32 usec) ...@@ -334,13 +347,15 @@ static __always_inline u32 ipa_aggr_granularity_val(u32 usec)
*/ */
static void ipa_qtime_config(struct ipa *ipa) static void ipa_qtime_config(struct ipa *ipa)
{ {
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
/* Timer clock divider must be disabled when we change the rate */ /* Timer clock divider must be disabled when we change the rate */
offset = ipa_reg_offset(ipa, TIMERS_XO_CLK_DIV_CFG); reg = ipa_reg(ipa, TIMERS_XO_CLK_DIV_CFG);
iowrite32(0, ipa->reg_virt + offset); iowrite32(0, ipa->reg_virt + ipa_reg_offset(reg));
reg = ipa_reg(ipa, QTIME_TIMESTAMP_CFG);
/* Set DPL time stamp resolution to use Qtime (instead of 1 msec) */ /* Set DPL time stamp resolution to use Qtime (instead of 1 msec) */
val = u32_encode_bits(DPL_TIMESTAMP_SHIFT, DPL_TIMESTAMP_LSB_FMASK); val = u32_encode_bits(DPL_TIMESTAMP_SHIFT, DPL_TIMESTAMP_LSB_FMASK);
val |= u32_encode_bits(1, DPL_TIMESTAMP_SEL_FMASK); val |= u32_encode_bits(1, DPL_TIMESTAMP_SEL_FMASK);
...@@ -348,21 +363,21 @@ static void ipa_qtime_config(struct ipa *ipa) ...@@ -348,21 +363,21 @@ static void ipa_qtime_config(struct ipa *ipa)
val |= u32_encode_bits(TAG_TIMESTAMP_SHIFT, TAG_TIMESTAMP_LSB_FMASK); val |= u32_encode_bits(TAG_TIMESTAMP_SHIFT, TAG_TIMESTAMP_LSB_FMASK);
val |= u32_encode_bits(NAT_TIMESTAMP_SHIFT, NAT_TIMESTAMP_LSB_FMASK); val |= u32_encode_bits(NAT_TIMESTAMP_SHIFT, NAT_TIMESTAMP_LSB_FMASK);
offset = ipa_reg_offset(ipa, QTIME_TIMESTAMP_CFG); iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
iowrite32(val, ipa->reg_virt + offset);
/* Set granularity of pulse generators used for other timers */ /* Set granularity of pulse generators used for other timers */
reg = ipa_reg(ipa, TIMERS_PULSE_GRAN_CFG);
val = u32_encode_bits(IPA_GRAN_100_US, GRAN_0_FMASK); val = u32_encode_bits(IPA_GRAN_100_US, GRAN_0_FMASK);
val |= u32_encode_bits(IPA_GRAN_1_MS, GRAN_1_FMASK); val |= u32_encode_bits(IPA_GRAN_1_MS, GRAN_1_FMASK);
val |= u32_encode_bits(IPA_GRAN_1_MS, GRAN_2_FMASK); val |= u32_encode_bits(IPA_GRAN_1_MS, GRAN_2_FMASK);
offset = ipa_reg_offset(ipa, TIMERS_PULSE_GRAN_CFG); iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
iowrite32(val, ipa->reg_virt + offset);
/* Actual divider is 1 more than value supplied here */ /* Actual divider is 1 more than value supplied here */
reg = ipa_reg(ipa, TIMERS_XO_CLK_DIV_CFG);
offset = ipa_reg_offset(reg);
val = u32_encode_bits(IPA_XO_CLOCK_DIVIDER - 1, DIV_VALUE_FMASK); val = u32_encode_bits(IPA_XO_CLOCK_DIVIDER - 1, DIV_VALUE_FMASK);
offset = ipa_reg_offset(ipa, TIMERS_XO_CLK_DIV_CFG);
iowrite32(val, ipa->reg_virt + offset); iowrite32(val, ipa->reg_virt + offset);
/* Divider value is set; re-enable the common timer clock divider */ /* Divider value is set; re-enable the common timer clock divider */
...@@ -374,16 +389,13 @@ static void ipa_qtime_config(struct ipa *ipa) ...@@ -374,16 +389,13 @@ static void ipa_qtime_config(struct ipa *ipa)
/* Before IPA v4.5 timing is controlled by a counter register */ /* Before IPA v4.5 timing is controlled by a counter register */
static void ipa_hardware_config_counter(struct ipa *ipa) static void ipa_hardware_config_counter(struct ipa *ipa)
{ {
u32 granularity; u32 granularity = ipa_aggr_granularity_val(IPA_AGGR_GRANULARITY);
u32 offset; const struct ipa_reg *reg;
u32 val; u32 val;
granularity = ipa_aggr_granularity_val(IPA_AGGR_GRANULARITY); reg = ipa_reg(ipa, COUNTER_CFG);
val = u32_encode_bits(granularity, AGGR_GRANULARITY_FMASK); val = u32_encode_bits(granularity, AGGR_GRANULARITY_FMASK);
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
offset = ipa_reg_offset(ipa, COUNTER_CFG);
iowrite32(val, ipa->reg_virt + offset);
} }
static void ipa_hardware_config_timing(struct ipa *ipa) static void ipa_hardware_config_timing(struct ipa *ipa)
...@@ -396,30 +408,30 @@ static void ipa_hardware_config_timing(struct ipa *ipa) ...@@ -396,30 +408,30 @@ static void ipa_hardware_config_timing(struct ipa *ipa)
static void ipa_hardware_config_hashing(struct ipa *ipa) static void ipa_hardware_config_hashing(struct ipa *ipa)
{ {
u32 offset; const struct ipa_reg *reg;
if (ipa->version != IPA_VERSION_4_2) if (ipa->version != IPA_VERSION_4_2)
return; return;
/* IPA v4.2 does not support hashed tables, so disable them */ /* IPA v4.2 does not support hashed tables, so disable them */
offset = ipa_reg_offset(ipa, FILT_ROUT_HASH_EN); reg = ipa_reg(ipa, FILT_ROUT_HASH_EN);
iowrite32(0, ipa->reg_virt + offset); iowrite32(0, ipa->reg_virt + ipa_reg_offset(reg));
} }
static void ipa_idle_indication_cfg(struct ipa *ipa, static void ipa_idle_indication_cfg(struct ipa *ipa,
u32 enter_idle_debounce_thresh, u32 enter_idle_debounce_thresh,
bool const_non_idle_enable) bool const_non_idle_enable)
{ {
u32 offset; const struct ipa_reg *reg;
u32 val; u32 val;
reg = ipa_reg(ipa, IDLE_INDICATION_CFG);
val = u32_encode_bits(enter_idle_debounce_thresh, val = u32_encode_bits(enter_idle_debounce_thresh,
ENTER_IDLE_DEBOUNCE_THRESH_FMASK); ENTER_IDLE_DEBOUNCE_THRESH_FMASK);
if (const_non_idle_enable) if (const_non_idle_enable)
val |= CONST_NON_IDLE_ENABLE_FMASK; val |= CONST_NON_IDLE_ENABLE_FMASK;
offset = ipa_reg_offset(ipa, IDLE_INDICATION_CFG); iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
iowrite32(val, ipa->reg_virt + offset);
} }
/** /**
......
...@@ -75,6 +75,7 @@ ipa_mem_zero_region_add(struct gsi_trans *trans, enum ipa_mem_id mem_id) ...@@ -75,6 +75,7 @@ ipa_mem_zero_region_add(struct gsi_trans *trans, enum ipa_mem_id mem_id)
int ipa_mem_setup(struct ipa *ipa) int ipa_mem_setup(struct ipa *ipa)
{ {
dma_addr_t addr = ipa->zero_addr; dma_addr_t addr = ipa->zero_addr;
const struct ipa_reg *reg;
const struct ipa_mem *mem; const struct ipa_mem *mem;
struct gsi_trans *trans; struct gsi_trans *trans;
u32 offset; u32 offset;
...@@ -112,10 +113,10 @@ int ipa_mem_setup(struct ipa *ipa) ...@@ -112,10 +113,10 @@ int ipa_mem_setup(struct ipa *ipa)
/* Tell the hardware where the processing context area is located */ /* Tell the hardware where the processing context area is located */
mem = ipa_mem_find(ipa, IPA_MEM_MODEM_PROC_CTX); mem = ipa_mem_find(ipa, IPA_MEM_MODEM_PROC_CTX);
offset = ipa->mem_offset + mem->offset; offset = ipa->mem_offset + mem->offset;
val = proc_cntxt_base_addr_encoded(ipa->version, offset);
offset = ipa_reg_offset(ipa, LOCAL_PKT_PROC_CNTXT); reg = ipa_reg(ipa, LOCAL_PKT_PROC_CNTXT);
iowrite32(val, ipa->reg_virt + offset); val = proc_cntxt_base_addr_encoded(ipa->version, offset);
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
return 0; return 0;
} }
...@@ -308,6 +309,7 @@ static bool ipa_mem_size_valid(struct ipa *ipa) ...@@ -308,6 +309,7 @@ static bool ipa_mem_size_valid(struct ipa *ipa)
int ipa_mem_config(struct ipa *ipa) int ipa_mem_config(struct ipa *ipa)
{ {
struct device *dev = &ipa->pdev->dev; struct device *dev = &ipa->pdev->dev;
const struct ipa_reg *reg;
const struct ipa_mem *mem; const struct ipa_mem *mem;
dma_addr_t addr; dma_addr_t addr;
u32 mem_size; u32 mem_size;
...@@ -316,7 +318,8 @@ int ipa_mem_config(struct ipa *ipa) ...@@ -316,7 +318,8 @@ int ipa_mem_config(struct ipa *ipa)
u32 i; u32 i;
/* Check the advertised location and size of the shared memory area */ /* Check the advertised location and size of the shared memory area */
val = ioread32(ipa->reg_virt + ipa_reg_offset(ipa, SHARED_MEM_SIZE)); reg = ipa_reg(ipa, SHARED_MEM_SIZE);
val = ioread32(ipa->reg_virt + ipa_reg_offset(reg));
/* The fields in the register are in 8 byte units */ /* The fields in the register are in 8 byte units */
ipa->mem_offset = 8 * u32_get_bits(val, SHARED_MEM_BADDR_FMASK); ipa->mem_offset = 8 * u32_get_bits(val, SHARED_MEM_BADDR_FMASK);
......
...@@ -67,17 +67,12 @@ static bool ipa_reg_valid(struct ipa *ipa, enum ipa_reg_id reg_id) ...@@ -67,17 +67,12 @@ static bool ipa_reg_valid(struct ipa *ipa, enum ipa_reg_id reg_id)
return valid && ipa->regs->reg[reg_id]; return valid && ipa->regs->reg[reg_id];
} }
/* Assumes the endpoint transfer direction is valid; 0 is a bad offset */ const struct ipa_reg *ipa_reg(struct ipa *ipa, enum ipa_reg_id reg_id)
u32 __ipa_reg_offset(struct ipa *ipa, enum ipa_reg_id reg_id, u32 n)
{ {
const struct ipa_reg *reg;
if (WARN_ON(!ipa_reg_valid(ipa, reg_id))) if (WARN_ON(!ipa_reg_valid(ipa, reg_id)))
return 0; return NULL;
reg = ipa->regs->reg[reg_id];
return reg->offset + n * reg->stride; return ipa->regs->reg[reg_id];
} }
static const struct ipa_regs *ipa_regs(enum ipa_version version) static const struct ipa_regs *ipa_regs(enum ipa_version version)
......
...@@ -746,19 +746,18 @@ extern const struct ipa_regs ipa_regs_v4_5; ...@@ -746,19 +746,18 @@ extern const struct ipa_regs ipa_regs_v4_5;
extern const struct ipa_regs ipa_regs_v4_9; extern const struct ipa_regs ipa_regs_v4_9;
extern const struct ipa_regs ipa_regs_v4_11; extern const struct ipa_regs ipa_regs_v4_11;
u32 __ipa_reg_offset(struct ipa *ipa, enum ipa_reg_id reg_id, u32 n);
const struct ipa_reg *ipa_reg(struct ipa *ipa, enum ipa_reg_id reg_id); const struct ipa_reg *ipa_reg(struct ipa *ipa, enum ipa_reg_id reg_id);
static inline u32 ipa_reg_offset(struct ipa *ipa, enum ipa_reg_id reg_id) /* Returns 0 for NULL reg; warning will have already been issued */
static inline u32 ipa_reg_offset(const struct ipa_reg *reg)
{ {
return __ipa_reg_offset(ipa, reg_id, 0); return reg ? reg->offset : 0;
} }
static inline u32 /* Returns 0 for NULL reg; warning will have already been issued */
ipa_reg_n_offset(struct ipa *ipa, enum ipa_reg_id reg_id, u32 n) static inline u32 ipa_reg_n_offset(const struct ipa_reg *reg, u32 n)
{ {
return __ipa_reg_offset(ipa, reg_id, n); return reg ? reg->offset + n * reg->stride : 0;
} }
int ipa_reg_init(struct ipa *ipa); int ipa_reg_init(struct ipa *ipa);
......
...@@ -91,36 +91,37 @@ static void ipa_resource_config_src(struct ipa *ipa, u32 resource_type, ...@@ -91,36 +91,37 @@ static void ipa_resource_config_src(struct ipa *ipa, u32 resource_type,
u32 group_count = data->rsrc_group_src_count; u32 group_count = data->rsrc_group_src_count;
const struct ipa_resource_limits *ylimits; const struct ipa_resource_limits *ylimits;
const struct ipa_resource *resource; const struct ipa_resource *resource;
const struct ipa_reg *reg;
u32 offset; u32 offset;
resource = &data->resource_src[resource_type]; resource = &data->resource_src[resource_type];
offset = ipa_reg_n_offset(ipa, SRC_RSRC_GRP_01_RSRC_TYPE, reg = ipa_reg(ipa, SRC_RSRC_GRP_01_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 1 ? NULL : &resource->limits[1]; ylimits = group_count == 1 ? NULL : &resource->limits[1];
ipa_resource_config_common(ipa, offset, &resource->limits[0], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[0], ylimits);
if (group_count < 3) if (group_count < 3)
return; return;
offset = ipa_reg_n_offset(ipa, SRC_RSRC_GRP_23_RSRC_TYPE, reg = ipa_reg(ipa, SRC_RSRC_GRP_23_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 3 ? NULL : &resource->limits[3]; ylimits = group_count == 3 ? NULL : &resource->limits[3];
ipa_resource_config_common(ipa, offset, &resource->limits[2], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[2], ylimits);
if (group_count < 5) if (group_count < 5)
return; return;
offset = ipa_reg_n_offset(ipa, SRC_RSRC_GRP_45_RSRC_TYPE, reg = ipa_reg(ipa, SRC_RSRC_GRP_45_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 5 ? NULL : &resource->limits[5]; ylimits = group_count == 5 ? NULL : &resource->limits[5];
ipa_resource_config_common(ipa, offset, &resource->limits[4], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[4], ylimits);
if (group_count < 7) if (group_count < 7)
return; return;
offset = ipa_reg_n_offset(ipa, SRC_RSRC_GRP_67_RSRC_TYPE, reg = ipa_reg(ipa, SRC_RSRC_GRP_67_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 7 ? NULL : &resource->limits[7]; ylimits = group_count == 7 ? NULL : &resource->limits[7];
ipa_resource_config_common(ipa, offset, &resource->limits[6], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[6], ylimits);
} }
...@@ -131,36 +132,37 @@ static void ipa_resource_config_dst(struct ipa *ipa, u32 resource_type, ...@@ -131,36 +132,37 @@ static void ipa_resource_config_dst(struct ipa *ipa, u32 resource_type,
u32 group_count = data->rsrc_group_dst_count; u32 group_count = data->rsrc_group_dst_count;
const struct ipa_resource_limits *ylimits; const struct ipa_resource_limits *ylimits;
const struct ipa_resource *resource; const struct ipa_resource *resource;
const struct ipa_reg *reg;
u32 offset; u32 offset;
resource = &data->resource_dst[resource_type]; resource = &data->resource_dst[resource_type];
offset = ipa_reg_n_offset(ipa, DST_RSRC_GRP_01_RSRC_TYPE, reg = ipa_reg(ipa, DST_RSRC_GRP_01_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 1 ? NULL : &resource->limits[1]; ylimits = group_count == 1 ? NULL : &resource->limits[1];
ipa_resource_config_common(ipa, offset, &resource->limits[0], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[0], ylimits);
if (group_count < 3) if (group_count < 3)
return; return;
offset = ipa_reg_n_offset(ipa, DST_RSRC_GRP_23_RSRC_TYPE, reg = ipa_reg(ipa, DST_RSRC_GRP_23_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 3 ? NULL : &resource->limits[3]; ylimits = group_count == 3 ? NULL : &resource->limits[3];
ipa_resource_config_common(ipa, offset, &resource->limits[2], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[2], ylimits);
if (group_count < 5) if (group_count < 5)
return; return;
offset = ipa_reg_n_offset(ipa, DST_RSRC_GRP_45_RSRC_TYPE, reg = ipa_reg(ipa, DST_RSRC_GRP_45_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 5 ? NULL : &resource->limits[5]; ylimits = group_count == 5 ? NULL : &resource->limits[5];
ipa_resource_config_common(ipa, offset, &resource->limits[4], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[4], ylimits);
if (group_count < 7) if (group_count < 7)
return; return;
offset = ipa_reg_n_offset(ipa, DST_RSRC_GRP_67_RSRC_TYPE, reg = ipa_reg(ipa, DST_RSRC_GRP_67_RSRC_TYPE);
resource_type); offset = ipa_reg_n_offset(reg, resource_type);
ylimits = group_count == 7 ? NULL : &resource->limits[7]; ylimits = group_count == 7 ? NULL : &resource->limits[7];
ipa_resource_config_common(ipa, offset, &resource->limits[6], ylimits); ipa_resource_config_common(ipa, offset, &resource->limits[6], ylimits);
} }
......
...@@ -384,8 +384,9 @@ void ipa_table_reset(struct ipa *ipa, bool modem) ...@@ -384,8 +384,9 @@ void ipa_table_reset(struct ipa *ipa, bool modem)
int ipa_table_hash_flush(struct ipa *ipa) int ipa_table_hash_flush(struct ipa *ipa)
{ {
u32 offset = ipa_reg_offset(ipa, FILT_ROUT_HASH_FLUSH); const struct ipa_reg *reg;
struct gsi_trans *trans; struct gsi_trans *trans;
u32 offset;
u32 val; u32 val;
if (!ipa_table_hash_support(ipa)) if (!ipa_table_hash_support(ipa))
...@@ -397,6 +398,9 @@ int ipa_table_hash_flush(struct ipa *ipa) ...@@ -397,6 +398,9 @@ int ipa_table_hash_flush(struct ipa *ipa)
return -EBUSY; return -EBUSY;
} }
reg = ipa_reg(ipa, FILT_ROUT_HASH_FLUSH);
offset = ipa_reg_offset(reg);
val = IPV4_FILTER_HASH_FMASK | IPV6_FILTER_HASH_FMASK; val = IPV4_FILTER_HASH_FMASK | IPV6_FILTER_HASH_FMASK;
val |= IPV6_ROUTER_HASH_FMASK | IPV4_ROUTER_HASH_FMASK; val |= IPV6_ROUTER_HASH_FMASK | IPV4_ROUTER_HASH_FMASK;
...@@ -517,10 +521,12 @@ static void ipa_filter_tuple_zero(struct ipa_endpoint *endpoint) ...@@ -517,10 +521,12 @@ static void ipa_filter_tuple_zero(struct ipa_endpoint *endpoint)
{ {
u32 endpoint_id = endpoint->endpoint_id; u32 endpoint_id = endpoint->endpoint_id;
struct ipa *ipa = endpoint->ipa; struct ipa *ipa = endpoint->ipa;
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
offset = ipa_reg_n_offset(ipa, ENDP_FILTER_ROUTER_HSH_CFG, endpoint_id); reg = ipa_reg(ipa, ENDP_FILTER_ROUTER_HSH_CFG);
offset = ipa_reg_n_offset(reg, endpoint_id);
val = ioread32(endpoint->ipa->reg_virt + offset); val = ioread32(endpoint->ipa->reg_virt + offset);
...@@ -566,10 +572,12 @@ static bool ipa_route_id_modem(u32 route_id) ...@@ -566,10 +572,12 @@ static bool ipa_route_id_modem(u32 route_id)
*/ */
static void ipa_route_tuple_zero(struct ipa *ipa, u32 route_id) static void ipa_route_tuple_zero(struct ipa *ipa, u32 route_id)
{ {
const struct ipa_reg *reg;
u32 offset; u32 offset;
u32 val; u32 val;
offset = ipa_reg_n_offset(ipa, ENDP_FILTER_ROUTER_HSH_CFG, route_id); reg = ipa_reg(ipa, ENDP_FILTER_ROUTER_HSH_CFG);
offset = ipa_reg_n_offset(reg, route_id);
val = ioread32(ipa->reg_virt + offset); val = ioread32(ipa->reg_virt + offset);
......
...@@ -222,6 +222,7 @@ void ipa_uc_power(struct ipa *ipa) ...@@ -222,6 +222,7 @@ void ipa_uc_power(struct ipa *ipa)
static void send_uc_command(struct ipa *ipa, u32 command, u32 command_param) static void send_uc_command(struct ipa *ipa, u32 command, u32 command_param)
{ {
struct ipa_uc_mem_area *shared = ipa_uc_shared(ipa); struct ipa_uc_mem_area *shared = ipa_uc_shared(ipa);
const struct ipa_reg *reg;
u32 val; u32 val;
/* Fill in the command data */ /* Fill in the command data */
...@@ -232,8 +233,10 @@ static void send_uc_command(struct ipa *ipa, u32 command, u32 command_param) ...@@ -232,8 +233,10 @@ static void send_uc_command(struct ipa *ipa, u32 command, u32 command_param)
shared->response_param = 0; shared->response_param = 0;
/* Use an interrupt to tell the microcontroller the command is ready */ /* Use an interrupt to tell the microcontroller the command is ready */
reg = ipa_reg(ipa, IPA_IRQ_UC);
val = u32_encode_bits(1, UC_INTR_FMASK); val = u32_encode_bits(1, UC_INTR_FMASK);
iowrite32(val, ipa->reg_virt + ipa_reg_offset(ipa, IPA_IRQ_UC));
iowrite32(val, ipa->reg_virt + ipa_reg_offset(reg));
} }
/* Tell the microcontroller the AP is shutting down */ /* Tell the microcontroller the AP is shutting down */
......
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