Commit 589ce5f4 authored by Marc Zyngier's avatar Marc Zyngier

irqchip/gic-v3-its: Fix 64bit GIC{R,ITS}_TYPER accesses

The GICv3 architecture specification mentions that a 64bit
register can be accessed using two 32bit accesses. What it
doesn't mention is that this is only guaranteed on a system
that implements AArch32, and a pure AArch64 system is allowed
not to support this. This causes issues with the GICR_TYPER
and GITS_TYPER registers, which are both RO 64bit registers.

In order to solve this, this patch switches the TYPER accesses
to the gic_read_typer macro already used in other parts of the
driver. This makes sure that we always use a 64bit access on
64bit systems, and two 32bit accesses on 32bit system.
Signed-off-by: default avatarMarc Zyngier <marc.zyngier@arm.com>
parent 1001354c
...@@ -1023,7 +1023,7 @@ static void its_free_tables(struct its_node *its) ...@@ -1023,7 +1023,7 @@ static void its_free_tables(struct its_node *its)
static int its_alloc_tables(struct its_node *its) static int its_alloc_tables(struct its_node *its)
{ {
u64 typer = readq_relaxed(its->base + GITS_TYPER); u64 typer = gic_read_typer(its->base + GITS_TYPER);
u32 ids = GITS_TYPER_DEVBITS(typer); u32 ids = GITS_TYPER_DEVBITS(typer);
u64 shr = GITS_BASER_InnerShareable; u64 shr = GITS_BASER_InnerShareable;
u64 cache = GITS_BASER_WaWb; u64 cache = GITS_BASER_WaWb;
...@@ -1198,7 +1198,7 @@ static void its_cpu_init_collection(void) ...@@ -1198,7 +1198,7 @@ static void its_cpu_init_collection(void)
* We now have to bind each collection to its target * We now have to bind each collection to its target
* redistributor. * redistributor.
*/ */
if (readq_relaxed(its->base + GITS_TYPER) & GITS_TYPER_PTA) { if (gic_read_typer(its->base + GITS_TYPER) & GITS_TYPER_PTA) {
/* /*
* This ITS wants the physical address of the * This ITS wants the physical address of the
* redistributor. * redistributor.
...@@ -1208,7 +1208,7 @@ static void its_cpu_init_collection(void) ...@@ -1208,7 +1208,7 @@ static void its_cpu_init_collection(void)
/* /*
* This ITS wants a linear CPU number. * This ITS wants a linear CPU number.
*/ */
target = readq_relaxed(gic_data_rdist_rd_base() + GICR_TYPER); target = gic_read_typer(gic_data_rdist_rd_base() + GICR_TYPER);
target = GICR_TYPER_CPU_NUMBER(target) << 16; target = GICR_TYPER_CPU_NUMBER(target) << 16;
} }
...@@ -1691,7 +1691,7 @@ static int __init its_probe_one(struct resource *res, ...@@ -1691,7 +1691,7 @@ static int __init its_probe_one(struct resource *res,
INIT_LIST_HEAD(&its->its_device_list); INIT_LIST_HEAD(&its->its_device_list);
its->base = its_base; its->base = its_base;
its->phys_base = res->start; its->phys_base = res->start;
its->ite_size = ((readl_relaxed(its_base + GITS_TYPER) >> 4) & 0xf) + 1; its->ite_size = ((gic_read_typer(its_base + GITS_TYPER) >> 4) & 0xf) + 1;
its->numa_node = numa_node; its->numa_node = numa_node;
its->cmd_base = kzalloc(ITS_CMD_QUEUE_SZ, GFP_KERNEL); its->cmd_base = kzalloc(ITS_CMD_QUEUE_SZ, GFP_KERNEL);
...@@ -1763,7 +1763,7 @@ static int __init its_probe_one(struct resource *res, ...@@ -1763,7 +1763,7 @@ static int __init its_probe_one(struct resource *res,
static bool gic_rdists_supports_plpis(void) static bool gic_rdists_supports_plpis(void)
{ {
return !!(readl_relaxed(gic_data_rdist_rd_base() + GICR_TYPER) & GICR_TYPER_PLPIS); return !!(gic_read_typer(gic_data_rdist_rd_base() + GICR_TYPER) & GICR_TYPER_PLPIS);
} }
int its_cpu_init(void) int its_cpu_init(void)
......
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