Skip to content

Commit

Permalink
---
Browse files Browse the repository at this point in the history
yaml
---
r: 282763
b: refs/heads/master
c: b16137b
h: refs/heads/master
i:
  282761: 98aa5a3
  282759: 932da2a
v: v3
  • Loading branch information
Joerg Roedel committed Dec 12, 2011
1 parent b0c3c22 commit 63cfc9c
Show file tree
Hide file tree
Showing 4 changed files with 136 additions and 1 deletion.
2 changes: 1 addition & 1 deletion [refs]
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
---
refs/heads/master: 22e266c79b5bd5441243863c89ea237e6e845295
refs/heads/master: b16137b11b4b4d4bb27b61bba7e05f5fda5968f4
130 changes: 130 additions & 0 deletions trunk/drivers/iommu/amd_iommu.c
Original file line number Diff line number Diff line change
Expand Up @@ -1637,8 +1637,45 @@ static void free_pagetable(struct protection_domain *domain)
domain->pt_root = NULL;
}

static void free_gcr3_tbl_level1(u64 *tbl)
{
u64 *ptr;
int i;

for (i = 0; i < 512; ++i) {
if (!(tbl[i] & GCR3_VALID))
continue;

ptr = __va(tbl[i] & PAGE_MASK);

free_page((unsigned long)ptr);
}
}

static void free_gcr3_tbl_level2(u64 *tbl)
{
u64 *ptr;
int i;

for (i = 0; i < 512; ++i) {
if (!(tbl[i] & GCR3_VALID))
continue;

ptr = __va(tbl[i] & PAGE_MASK);

free_gcr3_tbl_level1(ptr);
}
}

static void free_gcr3_table(struct protection_domain *domain)
{
if (domain->glx == 2)
free_gcr3_tbl_level2(domain->gcr3_tbl);
else if (domain->glx == 1)
free_gcr3_tbl_level1(domain->gcr3_tbl);
else if (domain->glx != 0)
BUG();

free_page((unsigned long)domain->gcr3_tbl);
}

Expand Down Expand Up @@ -3282,3 +3319,96 @@ int amd_iommu_flush_tlb(struct iommu_domain *dom, int pasid)
}
EXPORT_SYMBOL(amd_iommu_flush_tlb);

static u64 *__get_gcr3_pte(u64 *root, int level, int pasid, bool alloc)
{
int index;
u64 *pte;

while (true) {

index = (pasid >> (9 * level)) & 0x1ff;
pte = &root[index];

if (level == 0)
break;

if (!(*pte & GCR3_VALID)) {
if (!alloc)
return NULL;

root = (void *)get_zeroed_page(GFP_ATOMIC);
if (root == NULL)
return NULL;

*pte = __pa(root) | GCR3_VALID;
}

root = __va(*pte & PAGE_MASK);

level -= 1;
}

return pte;
}

static int __set_gcr3(struct protection_domain *domain, int pasid,
unsigned long cr3)
{
u64 *pte;

if (domain->mode != PAGE_MODE_NONE)
return -EINVAL;

pte = __get_gcr3_pte(domain->gcr3_tbl, domain->glx, pasid, true);
if (pte == NULL)
return -ENOMEM;

*pte = (cr3 & PAGE_MASK) | GCR3_VALID;

return __amd_iommu_flush_tlb(domain, pasid);
}

static int __clear_gcr3(struct protection_domain *domain, int pasid)
{
u64 *pte;

if (domain->mode != PAGE_MODE_NONE)
return -EINVAL;

pte = __get_gcr3_pte(domain->gcr3_tbl, domain->glx, pasid, false);
if (pte == NULL)
return 0;

*pte = 0;

return __amd_iommu_flush_tlb(domain, pasid);
}

int amd_iommu_domain_set_gcr3(struct iommu_domain *dom, int pasid,
unsigned long cr3)
{
struct protection_domain *domain = dom->priv;
unsigned long flags;
int ret;

spin_lock_irqsave(&domain->lock, flags);
ret = __set_gcr3(domain, pasid, cr3);
spin_unlock_irqrestore(&domain->lock, flags);

return ret;
}
EXPORT_SYMBOL(amd_iommu_domain_set_gcr3);

int amd_iommu_domain_clear_gcr3(struct iommu_domain *dom, int pasid)
{
struct protection_domain *domain = dom->priv;
unsigned long flags;
int ret;

spin_lock_irqsave(&domain->lock, flags);
ret = __clear_gcr3(domain, pasid);
spin_unlock_irqrestore(&domain->lock, flags);

return ret;
}
EXPORT_SYMBOL(amd_iommu_domain_clear_gcr3);
4 changes: 4 additions & 0 deletions trunk/drivers/iommu/amd_iommu_proto.h
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,10 @@ extern int amd_iommu_domain_enable_v2(struct iommu_domain *dom, int pasids);
extern int amd_iommu_flush_page(struct iommu_domain *dom, int pasid,
u64 address);
extern int amd_iommu_flush_tlb(struct iommu_domain *dom, int pasid);
extern int amd_iommu_domain_set_gcr3(struct iommu_domain *dom, int pasid,
unsigned long cr3);
extern int amd_iommu_domain_clear_gcr3(struct iommu_domain *dom, int pasid);


#ifndef CONFIG_AMD_IOMMU_STATS

Expand Down
1 change: 1 addition & 0 deletions trunk/drivers/iommu/amd_iommu_types.h
Original file line number Diff line number Diff line change
Expand Up @@ -279,6 +279,7 @@
#define DTE_GCR3_SHIFT_B 16
#define DTE_GCR3_SHIFT_C 43

#define GCR3_VALID 0x01ULL

#define IOMMU_PAGE_MASK (((1ULL << 52) - 1) & ~0xfffULL)
#define IOMMU_PTE_PRESENT(pte) ((pte) & IOMMU_PTE_P)
Expand Down

0 comments on commit 63cfc9c

Please sign in to comment.