215 lines
7.6 KiB
Diff
215 lines
7.6 KiB
Diff
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
From: Muhammed Efe Cetin <efectn@protonmail.com>
|
|
Date: Fri, 11 Aug 2023 17:56:00 +0300
|
|
Subject: irqchip/irq-gic-v3-its: fix its timeout issue for rk35xx boards
|
|
|
|
---
|
|
drivers/irqchip/irq-gic-v3-its.c | 79 +++++++++-
|
|
1 file changed, 72 insertions(+), 7 deletions(-)
|
|
|
|
diff --git a/drivers/irqchip/irq-gic-v3-its.c b/drivers/irqchip/irq-gic-v3-its.c
|
|
index 111111111111..222222222222 100644
|
|
--- a/drivers/irqchip/irq-gic-v3-its.c
|
|
+++ b/drivers/irqchip/irq-gic-v3-its.c
|
|
@@ -163,6 +163,7 @@ struct its_device {
|
|
struct its_node *its;
|
|
struct event_lpi_map event_map;
|
|
void *itt;
|
|
+ u32 itt_sz;
|
|
u32 nr_ites;
|
|
u32 device_id;
|
|
bool shared;
|
|
@@ -2191,6 +2192,9 @@ static void gic_reset_prop_table(void *va)
|
|
static struct page *its_allocate_prop_table(gfp_t gfp_flags)
|
|
{
|
|
struct page *prop_page;
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ gfp_flags |= GFP_DMA32;
|
|
|
|
prop_page = alloc_pages(gfp_flags, get_order(LPI_PROPBASE_SZ));
|
|
if (!prop_page)
|
|
@@ -2315,6 +2319,7 @@ static int its_setup_baser(struct its_node *its, struct its_baser *baser,
|
|
u32 alloc_pages, psz;
|
|
struct page *page;
|
|
void *base;
|
|
+ gfp_t gfp_flags;
|
|
|
|
psz = baser->psz;
|
|
alloc_pages = (PAGE_ORDER_TO_SIZE(order) / psz);
|
|
@@ -2326,7 +2331,11 @@ static int its_setup_baser(struct its_node *its, struct its_baser *baser,
|
|
order = get_order(GITS_BASER_PAGES_MAX * psz);
|
|
}
|
|
|
|
- page = alloc_pages_node(its->numa_node, GFP_KERNEL | __GFP_ZERO, order);
|
|
+ gfp_flags = GFP_KERNEL | __GFP_ZERO;
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ gfp_flags |= GFP_DMA32;
|
|
+
|
|
+ page = alloc_pages_node(its->numa_node, gfp_flags, order);
|
|
if (!page)
|
|
return -ENOMEM;
|
|
|
|
@@ -2376,6 +2385,15 @@ static int its_setup_baser(struct its_node *its, struct its_baser *baser,
|
|
its_write_baser(its, baser, val);
|
|
tmp = baser->val;
|
|
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") ||
|
|
+ of_machine_is_compatible("rockchip,rk3588")) {
|
|
+ if (tmp & GITS_BASER_SHAREABILITY_MASK)
|
|
+ tmp &= ~GITS_BASER_SHAREABILITY_MASK;
|
|
+ else
|
|
+ gic_flush_dcache_to_poc(base, PAGE_ORDER_TO_SIZE(order));
|
|
+ }
|
|
+
|
|
if ((val ^ tmp) & GITS_BASER_SHAREABILITY_MASK) {
|
|
/*
|
|
* Shareability didn't stick. Just use
|
|
@@ -2965,7 +2983,9 @@ static int its_alloc_collections(struct its_node *its)
|
|
static struct page *its_allocate_pending_table(gfp_t gfp_flags)
|
|
{
|
|
struct page *pend_page;
|
|
-
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ gfp_flags |= GFP_DMA32;
|
|
pend_page = alloc_pages(gfp_flags | __GFP_ZERO,
|
|
get_order(LPI_PENDBASE_SZ));
|
|
if (!pend_page)
|
|
@@ -3124,6 +3144,11 @@ static void its_cpu_init_lpis(void)
|
|
if (!rdists_support_shareable())
|
|
tmp &= ~GICR_PROPBASER_SHAREABILITY_MASK;
|
|
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") ||
|
|
+ of_machine_is_compatible("rockchip,rk3588"))
|
|
+ tmp &= ~GICR_PROPBASER_SHAREABILITY_MASK;
|
|
+
|
|
if ((tmp ^ val) & GICR_PROPBASER_SHAREABILITY_MASK) {
|
|
if (!(tmp & GICR_PROPBASER_SHAREABILITY_MASK)) {
|
|
/*
|
|
@@ -3151,6 +3176,11 @@ static void its_cpu_init_lpis(void)
|
|
if (!rdists_support_shareable())
|
|
tmp &= ~GICR_PENDBASER_SHAREABILITY_MASK;
|
|
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") ||
|
|
+ of_machine_is_compatible("rockchip,rk3588"))
|
|
+ tmp &= ~GICR_PENDBASER_SHAREABILITY_MASK;
|
|
+
|
|
if (!(tmp & GICR_PENDBASER_SHAREABILITY_MASK)) {
|
|
/*
|
|
* The HW reports non-shareable, we must remove the
|
|
@@ -3314,7 +3344,11 @@ static bool its_alloc_table_entry(struct its_node *its,
|
|
|
|
/* Allocate memory for 2nd level table */
|
|
if (!table[idx]) {
|
|
- page = alloc_pages_node(its->numa_node, GFP_KERNEL | __GFP_ZERO,
|
|
+ gfp_t gfp_flags = GFP_KERNEL | __GFP_ZERO;
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ gfp_flags |= GFP_DMA32;
|
|
+ page = alloc_pages_node(its->numa_node, gfp_flags,
|
|
get_order(baser->psz));
|
|
if (!page)
|
|
return false;
|
|
@@ -3403,6 +3437,7 @@ static struct its_device *its_create_device(struct its_node *its, u32 dev_id,
|
|
int nr_lpis;
|
|
int nr_ites;
|
|
int sz;
|
|
+ gfp_t gfp_flags;
|
|
|
|
if (!its_alloc_device_table(its, dev_id))
|
|
return NULL;
|
|
@@ -3418,7 +3453,15 @@ static struct its_device *its_create_device(struct its_node *its, u32 dev_id,
|
|
nr_ites = max(2, nvecs);
|
|
sz = nr_ites * (FIELD_GET(GITS_TYPER_ITT_ENTRY_SIZE, its->typer) + 1);
|
|
sz = max(sz, ITS_ITT_ALIGN) + ITS_ITT_ALIGN - 1;
|
|
- itt = kzalloc_node(sz, GFP_KERNEL, its->numa_node);
|
|
+ gfp_flags = GFP_KERNEL;
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588")) {
|
|
+ gfp_flags |= GFP_DMA32;
|
|
+ itt = (void *)__get_free_pages(gfp_flags, get_order(sz));
|
|
+ } else {
|
|
+ itt = kzalloc_node(sz, gfp_flags, its->numa_node);
|
|
+ }
|
|
+
|
|
if (alloc_lpis) {
|
|
lpi_map = its_lpi_alloc(nvecs, &lpi_base, &nr_lpis);
|
|
if (lpi_map)
|
|
@@ -3432,7 +3475,13 @@ static struct its_device *its_create_device(struct its_node *its, u32 dev_id,
|
|
|
|
if (!dev || !itt || !col_map || (!lpi_map && alloc_lpis)) {
|
|
kfree(dev);
|
|
- kfree(itt);
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ free_pages((unsigned long)itt, get_order(sz));
|
|
+ else
|
|
+ kfree(itt);
|
|
+
|
|
bitmap_free(lpi_map);
|
|
kfree(col_map);
|
|
return NULL;
|
|
@@ -3442,6 +3491,7 @@ static struct its_device *its_create_device(struct its_node *its, u32 dev_id,
|
|
|
|
dev->its = its;
|
|
dev->itt = itt;
|
|
+ dev->itt_sz = sz;
|
|
dev->nr_ites = nr_ites;
|
|
dev->event_map.lpi_map = lpi_map;
|
|
dev->event_map.col_map = col_map;
|
|
@@ -3469,7 +3519,13 @@ static void its_free_device(struct its_device *its_dev)
|
|
list_del(&its_dev->entry);
|
|
raw_spin_unlock_irqrestore(&its_dev->its->lock, flags);
|
|
kfree(its_dev->event_map.col_map);
|
|
- kfree(its_dev->itt);
|
|
+
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ free_pages((unsigned long)its_dev->itt, get_order(its_dev->itt_sz));
|
|
+ else
|
|
+ kfree(its_dev->itt);
|
|
+
|
|
kfree(its_dev);
|
|
}
|
|
|
|
@@ -5078,6 +5134,7 @@ static int __init its_probe_one(struct its_node *its)
|
|
struct page *page;
|
|
u32 ctlr;
|
|
int err;
|
|
+ gfp_t gfp_flags;
|
|
|
|
its_enable_quirks(its);
|
|
|
|
@@ -5111,7 +5168,10 @@ static int __init its_probe_one(struct its_node *its)
|
|
}
|
|
}
|
|
|
|
- page = alloc_pages_node(its->numa_node, GFP_KERNEL | __GFP_ZERO,
|
|
+ gfp_flags = GFP_KERNEL | __GFP_ZERO;
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") || of_machine_is_compatible("rockchip,rk3566") || of_machine_is_compatible("rockchip,rk3588"))
|
|
+ gfp_flags |= GFP_DMA32;
|
|
+ page = alloc_pages_node(its->numa_node, gfp_flags,
|
|
get_order(ITS_CMD_QUEUE_SZ));
|
|
if (!page) {
|
|
err = -ENOMEM;
|
|
@@ -5140,6 +5200,11 @@ static int __init its_probe_one(struct its_node *its)
|
|
if (its->flags & ITS_FLAGS_FORCE_NON_SHAREABLE)
|
|
tmp &= ~GITS_CBASER_SHAREABILITY_MASK;
|
|
|
|
+ if (of_machine_is_compatible("rockchip,rk3568") ||
|
|
+ of_machine_is_compatible("rockchip,rk3566") ||
|
|
+ of_machine_is_compatible("rockchip,rk3588"))
|
|
+ tmp &= ~GITS_CBASER_SHAREABILITY_MASK;
|
|
+
|
|
if ((tmp ^ baser) & GITS_CBASER_SHAREABILITY_MASK) {
|
|
if (!(tmp & GITS_CBASER_SHAREABILITY_MASK)) {
|
|
/*
|
|
--
|
|
Armbian
|
|
|