Skip to content

Commit

Permalink
powerpc/pseries/iommu: Use a locallock instead local_irq_save()
Browse files Browse the repository at this point in the history
The locallock protects the per-CPU variable tce_page. The function
attempts to allocate memory while tce_page is protected (by disabling
interrupts).

Use local_irq_save() instead of local_irq_disable().

Cc: stable-rt@vger.kernel.org
Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
  • Loading branch information
Sebastian Andrzej Siewior committed Sep 13, 2021
1 parent c1da300 commit 46cc89c
Showing 1 changed file with 20 additions and 11 deletions.
31 changes: 20 additions & 11 deletions arch/powerpc/platforms/pseries/iommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
#include <linux/of.h>
#include <linux/iommu.h>
#include <linux/rculist.h>
#include <linux/local_lock.h>
#include <asm/io.h>
#include <asm/prom.h>
#include <asm/rtas.h>
Expand Down Expand Up @@ -195,7 +196,13 @@ static int tce_build_pSeriesLP(unsigned long liobn, long tcenum, long tceshift,
return ret;
}

static DEFINE_PER_CPU(__be64 *, tce_page);
struct tce_page {
__be64 * page;
local_lock_t lock;
};
static DEFINE_PER_CPU(struct tce_page, tce_page) = {
.lock = INIT_LOCAL_LOCK(lock),
};

static int tce_buildmulti_pSeriesLP(struct iommu_table *tbl, long tcenum,
long npages, unsigned long uaddr,
Expand All @@ -218,9 +225,10 @@ static int tce_buildmulti_pSeriesLP(struct iommu_table *tbl, long tcenum,
direction, attrs);
}

local_irq_save(flags); /* to protect tcep and the page behind it */
/* to protect tcep and the page behind it */
local_lock_irqsave(&tce_page.lock, flags);

tcep = __this_cpu_read(tce_page);
tcep = __this_cpu_read(tce_page.page);

/* This is safe to do since interrupts are off when we're called
* from iommu_alloc{,_sg}()
Expand All @@ -229,12 +237,12 @@ static int tce_buildmulti_pSeriesLP(struct iommu_table *tbl, long tcenum,
tcep = (__be64 *)__get_free_page(GFP_ATOMIC);
/* If allocation fails, fall back to the loop implementation */
if (!tcep) {
local_irq_restore(flags);
local_unlock_irqrestore(&tce_page.lock, flags);
return tce_build_pSeriesLP(tbl->it_index, tcenum,
tceshift,
npages, uaddr, direction, attrs);
}
__this_cpu_write(tce_page, tcep);
__this_cpu_write(tce_page.page, tcep);
}

rpn = __pa(uaddr) >> tceshift;
Expand Down Expand Up @@ -264,7 +272,7 @@ static int tce_buildmulti_pSeriesLP(struct iommu_table *tbl, long tcenum,
tcenum += limit;
} while (npages > 0 && !rc);

local_irq_restore(flags);
local_unlock_irqrestore(&tce_page.lock, flags);

if (unlikely(rc == H_NOT_ENOUGH_RESOURCES)) {
ret = (int)rc;
Expand Down Expand Up @@ -440,16 +448,17 @@ static int tce_setrange_multi_pSeriesLP(unsigned long start_pfn,
DMA_BIDIRECTIONAL, 0);
}

local_irq_disable(); /* to protect tcep and the page behind it */
tcep = __this_cpu_read(tce_page);
/* to protect tcep and the page behind it */
local_lock_irq(&tce_page.lock);
tcep = __this_cpu_read(tce_page.page);

if (!tcep) {
tcep = (__be64 *)__get_free_page(GFP_ATOMIC);
if (!tcep) {
local_irq_enable();
local_unlock_irq(&tce_page.lock);
return -ENOMEM;
}
__this_cpu_write(tce_page, tcep);
__this_cpu_write(tce_page.page, tcep);
}

proto_tce = TCE_PCI_READ | TCE_PCI_WRITE;
Expand Down Expand Up @@ -492,7 +501,7 @@ static int tce_setrange_multi_pSeriesLP(unsigned long start_pfn,

/* error cleanup: caller will clear whole range */

local_irq_enable();
local_unlock_irq(&tce_page.lock);
return rc;
}

Expand Down

0 comments on commit 46cc89c

Please sign in to comment.