Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 62418
b: refs/heads/master
c: 820a149
h: refs/heads/master
v: v3
  • Loading branch information
Muli Ben-Yehuda authored and Linus Torvalds committed Jul 22, 2007
1 parent e4d2fdb commit ba3fb81
Show file tree
Hide file tree
Showing 2 changed files with 18 additions and 24 deletions.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 7354b07595b2e43b75fe353fcf18e73eb0427c9b
refs/heads/master: 820a149705c2c2a37989554a4f4a34e3d0b0df1f
40 changes: 17 additions & 23 deletions trunk/arch/x86_64/kernel/pci-calgary.c
Original file line number Diff line number Diff line change
Expand Up @@ -233,6 +233,7 @@ static void iommu_range_reserve(struct iommu_table *tbl,
unsigned long index;
unsigned long end;
unsigned long badbit;
unsigned long flags;

index = start_addr >> PAGE_SHIFT;

Expand All @@ -244,6 +245,8 @@ static void iommu_range_reserve(struct iommu_table *tbl,
if (end > tbl->it_size) /* don't go off the table */
end = tbl->it_size;

spin_lock_irqsave(&tbl->it_lock, flags);

badbit = verify_bit_range(tbl->it_map, 0, index, end);
if (badbit != ~0UL) {
if (printk_ratelimit())
Expand All @@ -253,15 +256,20 @@ static void iommu_range_reserve(struct iommu_table *tbl,
}

set_bit_string(tbl->it_map, index, npages);

spin_unlock_irqrestore(&tbl->it_lock, flags);
}

static unsigned long iommu_range_alloc(struct iommu_table *tbl,
unsigned int npages)
{
unsigned long flags;
unsigned long offset;

BUG_ON(npages == 0);

spin_lock_irqsave(&tbl->it_lock, flags);

offset = find_next_zero_string(tbl->it_map, tbl->it_hint,
tbl->it_size, npages);
if (offset == ~0UL) {
Expand All @@ -270,6 +278,7 @@ static unsigned long iommu_range_alloc(struct iommu_table *tbl,
tbl->it_size, npages);
if (offset == ~0UL) {
printk(KERN_WARNING "Calgary: IOMMU full.\n");
spin_unlock_irqrestore(&tbl->it_lock, flags);
if (panic_on_overflow)
panic("Calgary: fix the allocator.\n");
else
Expand All @@ -281,17 +290,17 @@ static unsigned long iommu_range_alloc(struct iommu_table *tbl,
tbl->it_hint = offset + npages;
BUG_ON(tbl->it_hint > tbl->it_size);

spin_unlock_irqrestore(&tbl->it_lock, flags);

return offset;
}

static dma_addr_t iommu_alloc(struct iommu_table *tbl, void *vaddr,
unsigned int npages, int direction)
{
unsigned long entry, flags;
unsigned long entry;
dma_addr_t ret = bad_dma_address;

spin_lock_irqsave(&tbl->it_lock, flags);

entry = iommu_range_alloc(tbl, npages);

if (unlikely(entry == bad_dma_address))
Expand All @@ -304,12 +313,9 @@ static dma_addr_t iommu_alloc(struct iommu_table *tbl, void *vaddr,
tce_build(tbl, entry, npages, (unsigned long)vaddr & PAGE_MASK,
direction);

spin_unlock_irqrestore(&tbl->it_lock, flags);

return ret;

error:
spin_unlock_irqrestore(&tbl->it_lock, flags);
printk(KERN_WARNING "Calgary: failed to allocate %u pages in "
"iommu %p\n", npages, tbl);
return bad_dma_address;
Expand All @@ -321,6 +327,7 @@ static void __iommu_free(struct iommu_table *tbl, dma_addr_t dma_addr,
unsigned long entry;
unsigned long badbit;
unsigned long badend;
unsigned long flags;

/* were we called with bad_dma_address? */
badend = bad_dma_address + (EMERGENCY_PAGES * PAGE_SIZE);
Expand All @@ -337,6 +344,8 @@ static void __iommu_free(struct iommu_table *tbl, dma_addr_t dma_addr,

tce_free(tbl, entry, npages);

spin_lock_irqsave(&tbl->it_lock, flags);

badbit = verify_bit_range(tbl->it_map, 1, entry, entry + npages);
if (badbit != ~0UL) {
if (printk_ratelimit())
Expand All @@ -346,18 +355,14 @@ static void __iommu_free(struct iommu_table *tbl, dma_addr_t dma_addr,
}

__clear_bit_string(tbl->it_map, entry, npages);

spin_unlock_irqrestore(&tbl->it_lock, flags);
}

static void iommu_free(struct iommu_table *tbl, dma_addr_t dma_addr,
unsigned int npages)
{
unsigned long flags;

spin_lock_irqsave(&tbl->it_lock, flags);

__iommu_free(tbl, dma_addr, npages);

spin_unlock_irqrestore(&tbl->it_lock, flags);
}

static inline struct iommu_table *find_iommu_table(struct device *dev)
Expand Down Expand Up @@ -402,17 +407,12 @@ static void __calgary_unmap_sg(struct iommu_table *tbl,
void calgary_unmap_sg(struct device *dev, struct scatterlist *sglist,
int nelems, int direction)
{
unsigned long flags;
struct iommu_table *tbl = find_iommu_table(dev);

if (!translate_phb(to_pci_dev(dev)))
return;

spin_lock_irqsave(&tbl->it_lock, flags);

__calgary_unmap_sg(tbl, sglist, nelems, direction);

spin_unlock_irqrestore(&tbl->it_lock, flags);
}

static int calgary_nontranslate_map_sg(struct device* dev,
Expand All @@ -433,7 +433,6 @@ int calgary_map_sg(struct device *dev, struct scatterlist *sg,
int nelems, int direction)
{
struct iommu_table *tbl = find_iommu_table(dev);
unsigned long flags;
unsigned long vaddr;
unsigned int npages;
unsigned long entry;
Expand All @@ -442,8 +441,6 @@ int calgary_map_sg(struct device *dev, struct scatterlist *sg,
if (!translate_phb(to_pci_dev(dev)))
return calgary_nontranslate_map_sg(dev, sg, nelems, direction);

spin_lock_irqsave(&tbl->it_lock, flags);

for (i = 0; i < nelems; i++ ) {
struct scatterlist *s = &sg[i];
BUG_ON(!s->page);
Expand All @@ -467,16 +464,13 @@ int calgary_map_sg(struct device *dev, struct scatterlist *sg,
s->dma_length = s->length;
}

spin_unlock_irqrestore(&tbl->it_lock, flags);

return nelems;
error:
__calgary_unmap_sg(tbl, sg, nelems, direction);
for (i = 0; i < nelems; i++) {
sg[i].dma_address = bad_dma_address;
sg[i].dma_length = 0;
}
spin_unlock_irqrestore(&tbl->it_lock, flags);
return 0;
}

Expand Down

0 comments on commit ba3fb81

Please sign in to comment.