Commit 220d9122 authored by Jayachandran C's avatar Jayachandran C Committed by John Crispin

MIPS: Netlogic: Optimize EIMR/EIRR accesses in 32-bit

Provide functions ack_c0_eirr(), set_c0_eimr(), clear_c0_eimr()
and read_c0_eirr_and_eimr() that do the EIMR and EIRR operations
and update the interrupt handling code to use these functions.
Also, use the EIMR register functions to mask interrupts in the
irq code.

The 64-bit interrupt request and mask registers (EIRR and EIMR) are
accessed when the interrupts are off, and the common operations are
to set or clear a bit in these registers. Using the 64-bit c0 access
functions for these operations is not optimal in 32-bit, because it
will disable/restore interrupts and split/join the 64-bit value during
each register access.
Signed-off-by: default avatarJayachandran C <jchandra@broadcom.com>
Patchwork: http://patchwork.linux-mips.org/patch/4790/Signed-off-by: default avatarJohn Crispin <blogic@openwrt.org>
parent f0cb40e5
...@@ -68,6 +68,85 @@ do { \ ...@@ -68,6 +68,85 @@ do { \
__write_64bit_c0_register($9, 7, (val)); \ __write_64bit_c0_register($9, 7, (val)); \
} while (0) } while (0)
/*
* Handling the 64 bit EIMR and EIRR registers in 32-bit mode with
* standard functions will be very inefficient. This provides
* optimized functions for the normal operations on the registers.
*
* Call with interrupts disabled.
*/
static inline void ack_c0_eirr(int irq)
{
__asm__ __volatile__(
".set push\n\t"
".set mips64\n\t"
".set noat\n\t"
"li $1, 1\n\t"
"dsllv $1, $1, %0\n\t"
"dmtc0 $1, $9, 6\n\t"
".set pop"
: : "r" (irq));
}
static inline void set_c0_eimr(int irq)
{
__asm__ __volatile__(
".set push\n\t"
".set mips64\n\t"
".set noat\n\t"
"li $1, 1\n\t"
"dsllv %0, $1, %0\n\t"
"dmfc0 $1, $9, 7\n\t"
"or $1, %0\n\t"
"dmtc0 $1, $9, 7\n\t"
".set pop"
: "+r" (irq));
}
static inline void clear_c0_eimr(int irq)
{
__asm__ __volatile__(
".set push\n\t"
".set mips64\n\t"
".set noat\n\t"
"li $1, 1\n\t"
"dsllv %0, $1, %0\n\t"
"dmfc0 $1, $9, 7\n\t"
"or $1, %0\n\t"
"xor $1, %0\n\t"
"dmtc0 $1, $9, 7\n\t"
".set pop"
: "+r" (irq));
}
/*
* Read c0 eimr and c0 eirr, do AND of the two values, the result is
* the interrupts which are raised and are not masked.
*/
static inline uint64_t read_c0_eirr_and_eimr(void)
{
uint64_t val;
#ifdef CONFIG_64BIT
val = read_c0_eimr() & read_c0_eirr();
#else
__asm__ __volatile__(
".set push\n\t"
".set mips64\n\t"
".set noat\n\t"
"dmfc0 %M0, $9, 6\n\t"
"dmfc0 %L0, $9, 7\n\t"
"and %M0, %L0\n\t"
"dsll %L0, %M0, 32\n\t"
"dsra %M0, %M0, 32\n\t"
"dsra %L0, %L0, 32\n\t"
".set pop"
: "=r" (val));
#endif
return val;
}
static inline int hard_smp_processor_id(void) static inline int hard_smp_processor_id(void)
{ {
return __read_32bit_c0_register($15, 1) & 0x3ff; return __read_32bit_c0_register($15, 1) & 0x3ff;
......
...@@ -105,21 +105,23 @@ static void xlp_pic_disable(struct irq_data *d) ...@@ -105,21 +105,23 @@ static void xlp_pic_disable(struct irq_data *d)
static void xlp_pic_mask_ack(struct irq_data *d) static void xlp_pic_mask_ack(struct irq_data *d)
{ {
struct nlm_pic_irq *pd = irq_data_get_irq_handler_data(d); struct nlm_pic_irq *pd = irq_data_get_irq_handler_data(d);
uint64_t mask = 1ull << pd->picirq;
write_c0_eirr(mask); /* ack by writing EIRR */ clear_c0_eimr(pd->picirq);
ack_c0_eirr(pd->picirq);
} }
static void xlp_pic_unmask(struct irq_data *d) static void xlp_pic_unmask(struct irq_data *d)
{ {
struct nlm_pic_irq *pd = irq_data_get_irq_handler_data(d); struct nlm_pic_irq *pd = irq_data_get_irq_handler_data(d);
if (!pd) BUG_ON(!pd);
return;
if (pd->extra_ack) if (pd->extra_ack)
pd->extra_ack(d); pd->extra_ack(d);
/* re-enable the intr on this cpu */
set_c0_eimr(pd->picirq);
/* Ack is a single write, no need to lock */ /* Ack is a single write, no need to lock */
nlm_pic_ack(pd->node->picbase, pd->irt); nlm_pic_ack(pd->node->picbase, pd->irt);
} }
...@@ -134,32 +136,17 @@ static struct irq_chip xlp_pic = { ...@@ -134,32 +136,17 @@ static struct irq_chip xlp_pic = {
static void cpuintr_disable(struct irq_data *d) static void cpuintr_disable(struct irq_data *d)
{ {
uint64_t eimr; clear_c0_eimr(d->irq);
uint64_t mask = 1ull << d->irq;
eimr = read_c0_eimr();
write_c0_eimr(eimr & ~mask);
} }
static void cpuintr_enable(struct irq_data *d) static void cpuintr_enable(struct irq_data *d)
{ {
uint64_t eimr; set_c0_eimr(d->irq);
uint64_t mask = 1ull << d->irq;
eimr = read_c0_eimr();
write_c0_eimr(eimr | mask);
} }
static void cpuintr_ack(struct irq_data *d) static void cpuintr_ack(struct irq_data *d)
{ {
uint64_t mask = 1ull << d->irq; ack_c0_eirr(d->irq);
write_c0_eirr(mask);
}
static void cpuintr_nop(struct irq_data *d)
{
WARN(d->irq >= PIC_IRQ_BASE, "Bad irq %d", d->irq);
} }
/* /*
...@@ -170,9 +157,9 @@ struct irq_chip nlm_cpu_intr = { ...@@ -170,9 +157,9 @@ struct irq_chip nlm_cpu_intr = {
.name = "XLP-CPU-INTR", .name = "XLP-CPU-INTR",
.irq_enable = cpuintr_enable, .irq_enable = cpuintr_enable,
.irq_disable = cpuintr_disable, .irq_disable = cpuintr_disable,
.irq_mask = cpuintr_nop, .irq_mask = cpuintr_disable,
.irq_ack = cpuintr_nop, .irq_ack = cpuintr_ack,
.irq_eoi = cpuintr_ack, .irq_eoi = cpuintr_enable,
}; };
static void __init nlm_init_percpu_irqs(void) static void __init nlm_init_percpu_irqs(void)
...@@ -265,7 +252,7 @@ asmlinkage void plat_irq_dispatch(void) ...@@ -265,7 +252,7 @@ asmlinkage void plat_irq_dispatch(void)
int i, node; int i, node;
node = nlm_nodeid(); node = nlm_nodeid();
eirr = read_c0_eirr() & read_c0_eimr(); eirr = read_c0_eirr_and_eimr();
i = __ilog2_u64(eirr); i = __ilog2_u64(eirr);
if (i == -1) if (i == -1)
......
...@@ -84,15 +84,19 @@ void nlm_send_ipi_mask(const struct cpumask *mask, unsigned int action) ...@@ -84,15 +84,19 @@ void nlm_send_ipi_mask(const struct cpumask *mask, unsigned int action)
/* IRQ_IPI_SMP_FUNCTION Handler */ /* IRQ_IPI_SMP_FUNCTION Handler */
void nlm_smp_function_ipi_handler(unsigned int irq, struct irq_desc *desc) void nlm_smp_function_ipi_handler(unsigned int irq, struct irq_desc *desc)
{ {
write_c0_eirr(1ull << irq); clear_c0_eimr(irq);
ack_c0_eirr(irq);
smp_call_function_interrupt(); smp_call_function_interrupt();
set_c0_eimr(irq);
} }
/* IRQ_IPI_SMP_RESCHEDULE handler */ /* IRQ_IPI_SMP_RESCHEDULE handler */
void nlm_smp_resched_ipi_handler(unsigned int irq, struct irq_desc *desc) void nlm_smp_resched_ipi_handler(unsigned int irq, struct irq_desc *desc)
{ {
write_c0_eirr(1ull << irq); clear_c0_eimr(irq);
ack_c0_eirr(irq);
scheduler_ipi(); scheduler_ipi();
set_c0_eimr(irq);
} }
/* /*
......
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