mirror of
https://github.com/torvalds/linux.git
synced 2024-12-31 23:31:29 +00:00
Merge branches 'dma-debug/next', 'amd-iommu/command-cleanups', 'amd-iommu/ats' and 'amd-iommu/extended-features' into iommu/2.6.40
Conflicts: arch/x86/include/asm/amd_iommu_types.h arch/x86/kernel/amd_iommu.c arch/x86/kernel/amd_iommu_init.c
This commit is contained in:
commit
604c307bf4
@ -690,6 +690,7 @@ config AMD_IOMMU
|
|||||||
bool "AMD IOMMU support"
|
bool "AMD IOMMU support"
|
||||||
select SWIOTLB
|
select SWIOTLB
|
||||||
select PCI_MSI
|
select PCI_MSI
|
||||||
|
select PCI_IOV
|
||||||
depends on X86_64 && PCI && ACPI
|
depends on X86_64 && PCI && ACPI
|
||||||
---help---
|
---help---
|
||||||
With this option you can enable support for AMD IOMMU hardware in
|
With this option you can enable support for AMD IOMMU hardware in
|
||||||
|
@ -19,13 +19,11 @@
|
|||||||
#ifndef _ASM_X86_AMD_IOMMU_PROTO_H
|
#ifndef _ASM_X86_AMD_IOMMU_PROTO_H
|
||||||
#define _ASM_X86_AMD_IOMMU_PROTO_H
|
#define _ASM_X86_AMD_IOMMU_PROTO_H
|
||||||
|
|
||||||
struct amd_iommu;
|
#include <asm/amd_iommu_types.h>
|
||||||
|
|
||||||
extern int amd_iommu_init_dma_ops(void);
|
extern int amd_iommu_init_dma_ops(void);
|
||||||
extern int amd_iommu_init_passthrough(void);
|
extern int amd_iommu_init_passthrough(void);
|
||||||
extern irqreturn_t amd_iommu_int_handler(int irq, void *data);
|
extern irqreturn_t amd_iommu_int_handler(int irq, void *data);
|
||||||
extern void amd_iommu_flush_all_domains(void);
|
|
||||||
extern void amd_iommu_flush_all_devices(void);
|
|
||||||
extern void amd_iommu_apply_erratum_63(u16 devid);
|
extern void amd_iommu_apply_erratum_63(u16 devid);
|
||||||
extern void amd_iommu_reset_cmd_buffer(struct amd_iommu *iommu);
|
extern void amd_iommu_reset_cmd_buffer(struct amd_iommu *iommu);
|
||||||
extern int amd_iommu_init_devices(void);
|
extern int amd_iommu_init_devices(void);
|
||||||
@ -44,4 +42,12 @@ static inline bool is_rd890_iommu(struct pci_dev *pdev)
|
|||||||
(pdev->device == PCI_DEVICE_ID_RD890_IOMMU);
|
(pdev->device == PCI_DEVICE_ID_RD890_IOMMU);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool iommu_feature(struct amd_iommu *iommu, u64 f)
|
||||||
|
{
|
||||||
|
if (!(iommu->cap & (1 << IOMMU_CAP_EFR)))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return !!(iommu->features & f);
|
||||||
|
}
|
||||||
|
|
||||||
#endif /* _ASM_X86_AMD_IOMMU_PROTO_H */
|
#endif /* _ASM_X86_AMD_IOMMU_PROTO_H */
|
||||||
|
@ -68,12 +68,25 @@
|
|||||||
#define MMIO_CONTROL_OFFSET 0x0018
|
#define MMIO_CONTROL_OFFSET 0x0018
|
||||||
#define MMIO_EXCL_BASE_OFFSET 0x0020
|
#define MMIO_EXCL_BASE_OFFSET 0x0020
|
||||||
#define MMIO_EXCL_LIMIT_OFFSET 0x0028
|
#define MMIO_EXCL_LIMIT_OFFSET 0x0028
|
||||||
|
#define MMIO_EXT_FEATURES 0x0030
|
||||||
#define MMIO_CMD_HEAD_OFFSET 0x2000
|
#define MMIO_CMD_HEAD_OFFSET 0x2000
|
||||||
#define MMIO_CMD_TAIL_OFFSET 0x2008
|
#define MMIO_CMD_TAIL_OFFSET 0x2008
|
||||||
#define MMIO_EVT_HEAD_OFFSET 0x2010
|
#define MMIO_EVT_HEAD_OFFSET 0x2010
|
||||||
#define MMIO_EVT_TAIL_OFFSET 0x2018
|
#define MMIO_EVT_TAIL_OFFSET 0x2018
|
||||||
#define MMIO_STATUS_OFFSET 0x2020
|
#define MMIO_STATUS_OFFSET 0x2020
|
||||||
|
|
||||||
|
|
||||||
|
/* Extended Feature Bits */
|
||||||
|
#define FEATURE_PREFETCH (1ULL<<0)
|
||||||
|
#define FEATURE_PPR (1ULL<<1)
|
||||||
|
#define FEATURE_X2APIC (1ULL<<2)
|
||||||
|
#define FEATURE_NX (1ULL<<3)
|
||||||
|
#define FEATURE_GT (1ULL<<4)
|
||||||
|
#define FEATURE_IA (1ULL<<6)
|
||||||
|
#define FEATURE_GA (1ULL<<7)
|
||||||
|
#define FEATURE_HE (1ULL<<8)
|
||||||
|
#define FEATURE_PC (1ULL<<9)
|
||||||
|
|
||||||
/* MMIO status bits */
|
/* MMIO status bits */
|
||||||
#define MMIO_STATUS_COM_WAIT_INT_MASK 0x04
|
#define MMIO_STATUS_COM_WAIT_INT_MASK 0x04
|
||||||
|
|
||||||
@ -114,6 +127,8 @@
|
|||||||
#define CMD_COMPL_WAIT 0x01
|
#define CMD_COMPL_WAIT 0x01
|
||||||
#define CMD_INV_DEV_ENTRY 0x02
|
#define CMD_INV_DEV_ENTRY 0x02
|
||||||
#define CMD_INV_IOMMU_PAGES 0x03
|
#define CMD_INV_IOMMU_PAGES 0x03
|
||||||
|
#define CMD_INV_IOTLB_PAGES 0x04
|
||||||
|
#define CMD_INV_ALL 0x08
|
||||||
|
|
||||||
#define CMD_COMPL_WAIT_STORE_MASK 0x01
|
#define CMD_COMPL_WAIT_STORE_MASK 0x01
|
||||||
#define CMD_COMPL_WAIT_INT_MASK 0x02
|
#define CMD_COMPL_WAIT_INT_MASK 0x02
|
||||||
@ -215,6 +230,8 @@
|
|||||||
#define IOMMU_PTE_IR (1ULL << 61)
|
#define IOMMU_PTE_IR (1ULL << 61)
|
||||||
#define IOMMU_PTE_IW (1ULL << 62)
|
#define IOMMU_PTE_IW (1ULL << 62)
|
||||||
|
|
||||||
|
#define DTE_FLAG_IOTLB 0x01
|
||||||
|
|
||||||
#define IOMMU_PAGE_MASK (((1ULL << 52) - 1) & ~0xfffULL)
|
#define IOMMU_PAGE_MASK (((1ULL << 52) - 1) & ~0xfffULL)
|
||||||
#define IOMMU_PTE_PRESENT(pte) ((pte) & IOMMU_PTE_P)
|
#define IOMMU_PTE_PRESENT(pte) ((pte) & IOMMU_PTE_P)
|
||||||
#define IOMMU_PTE_PAGE(pte) (phys_to_virt((pte) & IOMMU_PAGE_MASK))
|
#define IOMMU_PTE_PAGE(pte) (phys_to_virt((pte) & IOMMU_PAGE_MASK))
|
||||||
@ -227,6 +244,7 @@
|
|||||||
/* IOMMU capabilities */
|
/* IOMMU capabilities */
|
||||||
#define IOMMU_CAP_IOTLB 24
|
#define IOMMU_CAP_IOTLB 24
|
||||||
#define IOMMU_CAP_NPCACHE 26
|
#define IOMMU_CAP_NPCACHE 26
|
||||||
|
#define IOMMU_CAP_EFR 27
|
||||||
|
|
||||||
#define MAX_DOMAIN_ID 65536
|
#define MAX_DOMAIN_ID 65536
|
||||||
|
|
||||||
@ -249,6 +267,8 @@ extern bool amd_iommu_dump;
|
|||||||
|
|
||||||
/* global flag if IOMMUs cache non-present entries */
|
/* global flag if IOMMUs cache non-present entries */
|
||||||
extern bool amd_iommu_np_cache;
|
extern bool amd_iommu_np_cache;
|
||||||
|
/* Only true if all IOMMUs support device IOTLBs */
|
||||||
|
extern bool amd_iommu_iotlb_sup;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Make iterating over all IOMMUs easier
|
* Make iterating over all IOMMUs easier
|
||||||
@ -371,6 +391,9 @@ struct amd_iommu {
|
|||||||
/* flags read from acpi table */
|
/* flags read from acpi table */
|
||||||
u8 acpi_flags;
|
u8 acpi_flags;
|
||||||
|
|
||||||
|
/* Extended features */
|
||||||
|
u64 features;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Capability pointer. There could be more than one IOMMU per PCI
|
* Capability pointer. There could be more than one IOMMU per PCI
|
||||||
* device function if there are more than one AMD IOMMU capability
|
* device function if there are more than one AMD IOMMU capability
|
||||||
@ -409,9 +432,6 @@ struct amd_iommu {
|
|||||||
/* if one, we need to send a completion wait command */
|
/* if one, we need to send a completion wait command */
|
||||||
bool need_sync;
|
bool need_sync;
|
||||||
|
|
||||||
/* becomes true if a command buffer reset is running */
|
|
||||||
bool reset_in_progress;
|
|
||||||
|
|
||||||
/* default dma_ops domain for that IOMMU */
|
/* default dma_ops domain for that IOMMU */
|
||||||
struct dma_ops_domain *default_dom;
|
struct dma_ops_domain *default_dom;
|
||||||
|
|
||||||
|
@ -18,6 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
|
#include <linux/pci-ats.h>
|
||||||
#include <linux/bitmap.h>
|
#include <linux/bitmap.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
#include <linux/debugfs.h>
|
#include <linux/debugfs.h>
|
||||||
@ -25,6 +26,7 @@
|
|||||||
#include <linux/dma-mapping.h>
|
#include <linux/dma-mapping.h>
|
||||||
#include <linux/iommu-helper.h>
|
#include <linux/iommu-helper.h>
|
||||||
#include <linux/iommu.h>
|
#include <linux/iommu.h>
|
||||||
|
#include <linux/delay.h>
|
||||||
#include <asm/proto.h>
|
#include <asm/proto.h>
|
||||||
#include <asm/iommu.h>
|
#include <asm/iommu.h>
|
||||||
#include <asm/gart.h>
|
#include <asm/gart.h>
|
||||||
@ -34,7 +36,7 @@
|
|||||||
|
|
||||||
#define CMD_SET_TYPE(cmd, t) ((cmd)->data[1] |= ((t) << 28))
|
#define CMD_SET_TYPE(cmd, t) ((cmd)->data[1] |= ((t) << 28))
|
||||||
|
|
||||||
#define EXIT_LOOP_COUNT 10000000
|
#define LOOP_TIMEOUT 100000
|
||||||
|
|
||||||
static DEFINE_RWLOCK(amd_iommu_devtable_lock);
|
static DEFINE_RWLOCK(amd_iommu_devtable_lock);
|
||||||
|
|
||||||
@ -57,7 +59,6 @@ struct iommu_cmd {
|
|||||||
u32 data[4];
|
u32 data[4];
|
||||||
};
|
};
|
||||||
|
|
||||||
static void reset_iommu_command_buffer(struct amd_iommu *iommu);
|
|
||||||
static void update_domain(struct protection_domain *domain);
|
static void update_domain(struct protection_domain *domain);
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -322,8 +323,6 @@ static void iommu_print_event(struct amd_iommu *iommu, void *__evt)
|
|||||||
break;
|
break;
|
||||||
case EVENT_TYPE_ILL_CMD:
|
case EVENT_TYPE_ILL_CMD:
|
||||||
printk("ILLEGAL_COMMAND_ERROR address=0x%016llx]\n", address);
|
printk("ILLEGAL_COMMAND_ERROR address=0x%016llx]\n", address);
|
||||||
iommu->reset_in_progress = true;
|
|
||||||
reset_iommu_command_buffer(iommu);
|
|
||||||
dump_command(address);
|
dump_command(address);
|
||||||
break;
|
break;
|
||||||
case EVENT_TYPE_CMD_HARD_ERR:
|
case EVENT_TYPE_CMD_HARD_ERR:
|
||||||
@ -383,124 +382,349 @@ irqreturn_t amd_iommu_int_handler(int irq, void *data)
|
|||||||
*
|
*
|
||||||
****************************************************************************/
|
****************************************************************************/
|
||||||
|
|
||||||
/*
|
static int wait_on_sem(volatile u64 *sem)
|
||||||
* Writes the command to the IOMMUs command buffer and informs the
|
|
||||||
* hardware about the new command. Must be called with iommu->lock held.
|
|
||||||
*/
|
|
||||||
static int __iommu_queue_command(struct amd_iommu *iommu, struct iommu_cmd *cmd)
|
|
||||||
{
|
{
|
||||||
u32 tail, head;
|
int i = 0;
|
||||||
u8 *target;
|
|
||||||
|
|
||||||
WARN_ON(iommu->cmd_buf_size & CMD_BUFFER_UNINITIALIZED);
|
while (*sem == 0 && i < LOOP_TIMEOUT) {
|
||||||
tail = readl(iommu->mmio_base + MMIO_CMD_TAIL_OFFSET);
|
udelay(1);
|
||||||
target = iommu->cmd_buf + tail;
|
i += 1;
|
||||||
memcpy_toio(target, cmd, sizeof(*cmd));
|
}
|
||||||
tail = (tail + sizeof(*cmd)) % iommu->cmd_buf_size;
|
|
||||||
head = readl(iommu->mmio_base + MMIO_CMD_HEAD_OFFSET);
|
if (i == LOOP_TIMEOUT) {
|
||||||
if (tail == head)
|
pr_alert("AMD-Vi: Completion-Wait loop timed out\n");
|
||||||
return -ENOMEM;
|
return -EIO;
|
||||||
writel(tail, iommu->mmio_base + MMIO_CMD_TAIL_OFFSET);
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void copy_cmd_to_buffer(struct amd_iommu *iommu,
|
||||||
|
struct iommu_cmd *cmd,
|
||||||
|
u32 tail)
|
||||||
|
{
|
||||||
|
u8 *target;
|
||||||
|
|
||||||
|
target = iommu->cmd_buf + tail;
|
||||||
|
tail = (tail + sizeof(*cmd)) % iommu->cmd_buf_size;
|
||||||
|
|
||||||
|
/* Copy command to buffer */
|
||||||
|
memcpy(target, cmd, sizeof(*cmd));
|
||||||
|
|
||||||
|
/* Tell the IOMMU about it */
|
||||||
|
writel(tail, iommu->mmio_base + MMIO_CMD_TAIL_OFFSET);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void build_completion_wait(struct iommu_cmd *cmd, u64 address)
|
||||||
|
{
|
||||||
|
WARN_ON(address & 0x7ULL);
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(*cmd));
|
||||||
|
cmd->data[0] = lower_32_bits(__pa(address)) | CMD_COMPL_WAIT_STORE_MASK;
|
||||||
|
cmd->data[1] = upper_32_bits(__pa(address));
|
||||||
|
cmd->data[2] = 1;
|
||||||
|
CMD_SET_TYPE(cmd, CMD_COMPL_WAIT);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void build_inv_dte(struct iommu_cmd *cmd, u16 devid)
|
||||||
|
{
|
||||||
|
memset(cmd, 0, sizeof(*cmd));
|
||||||
|
cmd->data[0] = devid;
|
||||||
|
CMD_SET_TYPE(cmd, CMD_INV_DEV_ENTRY);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void build_inv_iommu_pages(struct iommu_cmd *cmd, u64 address,
|
||||||
|
size_t size, u16 domid, int pde)
|
||||||
|
{
|
||||||
|
u64 pages;
|
||||||
|
int s;
|
||||||
|
|
||||||
|
pages = iommu_num_pages(address, size, PAGE_SIZE);
|
||||||
|
s = 0;
|
||||||
|
|
||||||
|
if (pages > 1) {
|
||||||
/*
|
/*
|
||||||
* General queuing function for commands. Takes iommu->lock and calls
|
* If we have to flush more than one page, flush all
|
||||||
* __iommu_queue_command().
|
* TLB entries for this domain
|
||||||
|
*/
|
||||||
|
address = CMD_INV_IOMMU_ALL_PAGES_ADDRESS;
|
||||||
|
s = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
address &= PAGE_MASK;
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(*cmd));
|
||||||
|
cmd->data[1] |= domid;
|
||||||
|
cmd->data[2] = lower_32_bits(address);
|
||||||
|
cmd->data[3] = upper_32_bits(address);
|
||||||
|
CMD_SET_TYPE(cmd, CMD_INV_IOMMU_PAGES);
|
||||||
|
if (s) /* size bit - we flush more than one 4kb page */
|
||||||
|
cmd->data[2] |= CMD_INV_IOMMU_PAGES_SIZE_MASK;
|
||||||
|
if (pde) /* PDE bit - we wan't flush everything not only the PTEs */
|
||||||
|
cmd->data[2] |= CMD_INV_IOMMU_PAGES_PDE_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void build_inv_iotlb_pages(struct iommu_cmd *cmd, u16 devid, int qdep,
|
||||||
|
u64 address, size_t size)
|
||||||
|
{
|
||||||
|
u64 pages;
|
||||||
|
int s;
|
||||||
|
|
||||||
|
pages = iommu_num_pages(address, size, PAGE_SIZE);
|
||||||
|
s = 0;
|
||||||
|
|
||||||
|
if (pages > 1) {
|
||||||
|
/*
|
||||||
|
* If we have to flush more than one page, flush all
|
||||||
|
* TLB entries for this domain
|
||||||
|
*/
|
||||||
|
address = CMD_INV_IOMMU_ALL_PAGES_ADDRESS;
|
||||||
|
s = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
address &= PAGE_MASK;
|
||||||
|
|
||||||
|
memset(cmd, 0, sizeof(*cmd));
|
||||||
|
cmd->data[0] = devid;
|
||||||
|
cmd->data[0] |= (qdep & 0xff) << 24;
|
||||||
|
cmd->data[1] = devid;
|
||||||
|
cmd->data[2] = lower_32_bits(address);
|
||||||
|
cmd->data[3] = upper_32_bits(address);
|
||||||
|
CMD_SET_TYPE(cmd, CMD_INV_IOTLB_PAGES);
|
||||||
|
if (s)
|
||||||
|
cmd->data[2] |= CMD_INV_IOMMU_PAGES_SIZE_MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void build_inv_all(struct iommu_cmd *cmd)
|
||||||
|
{
|
||||||
|
memset(cmd, 0, sizeof(*cmd));
|
||||||
|
CMD_SET_TYPE(cmd, CMD_INV_ALL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Writes the command to the IOMMUs command buffer and informs the
|
||||||
|
* hardware about the new command.
|
||||||
*/
|
*/
|
||||||
static int iommu_queue_command(struct amd_iommu *iommu, struct iommu_cmd *cmd)
|
static int iommu_queue_command(struct amd_iommu *iommu, struct iommu_cmd *cmd)
|
||||||
{
|
{
|
||||||
|
u32 left, tail, head, next_tail;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
|
WARN_ON(iommu->cmd_buf_size & CMD_BUFFER_UNINITIALIZED);
|
||||||
|
|
||||||
|
again:
|
||||||
|
spin_lock_irqsave(&iommu->lock, flags);
|
||||||
|
|
||||||
|
head = readl(iommu->mmio_base + MMIO_CMD_HEAD_OFFSET);
|
||||||
|
tail = readl(iommu->mmio_base + MMIO_CMD_TAIL_OFFSET);
|
||||||
|
next_tail = (tail + sizeof(*cmd)) % iommu->cmd_buf_size;
|
||||||
|
left = (head - next_tail) % iommu->cmd_buf_size;
|
||||||
|
|
||||||
|
if (left <= 2) {
|
||||||
|
struct iommu_cmd sync_cmd;
|
||||||
|
volatile u64 sem = 0;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
spin_lock_irqsave(&iommu->lock, flags);
|
build_completion_wait(&sync_cmd, (u64)&sem);
|
||||||
ret = __iommu_queue_command(iommu, cmd);
|
copy_cmd_to_buffer(iommu, &sync_cmd, tail);
|
||||||
if (!ret)
|
|
||||||
iommu->need_sync = true;
|
|
||||||
spin_unlock_irqrestore(&iommu->lock, flags);
|
spin_unlock_irqrestore(&iommu->lock, flags);
|
||||||
|
|
||||||
|
if ((ret = wait_on_sem(&sem)) != 0)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
goto again;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
copy_cmd_to_buffer(iommu, cmd, tail);
|
||||||
* This function waits until an IOMMU has completed a completion
|
|
||||||
* wait command
|
|
||||||
*/
|
|
||||||
static void __iommu_wait_for_completion(struct amd_iommu *iommu)
|
|
||||||
{
|
|
||||||
int ready = 0;
|
|
||||||
unsigned status = 0;
|
|
||||||
unsigned long i = 0;
|
|
||||||
|
|
||||||
INC_STATS_COUNTER(compl_wait);
|
/* We need to sync now to make sure all commands are processed */
|
||||||
|
iommu->need_sync = true;
|
||||||
|
|
||||||
while (!ready && (i < EXIT_LOOP_COUNT)) {
|
spin_unlock_irqrestore(&iommu->lock, flags);
|
||||||
++i;
|
|
||||||
/* wait for the bit to become one */
|
|
||||||
status = readl(iommu->mmio_base + MMIO_STATUS_OFFSET);
|
|
||||||
ready = status & MMIO_STATUS_COM_WAIT_INT_MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* set bit back to zero */
|
return 0;
|
||||||
status &= ~MMIO_STATUS_COM_WAIT_INT_MASK;
|
|
||||||
writel(status, iommu->mmio_base + MMIO_STATUS_OFFSET);
|
|
||||||
|
|
||||||
if (unlikely(i == EXIT_LOOP_COUNT))
|
|
||||||
iommu->reset_in_progress = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function queues a completion wait command into the command
|
* This function queues a completion wait command into the command
|
||||||
* buffer of an IOMMU
|
* buffer of an IOMMU
|
||||||
*/
|
*/
|
||||||
static int __iommu_completion_wait(struct amd_iommu *iommu)
|
static int iommu_completion_wait(struct amd_iommu *iommu)
|
||||||
|
{
|
||||||
|
struct iommu_cmd cmd;
|
||||||
|
volatile u64 sem = 0;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (!iommu->need_sync)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
build_completion_wait(&cmd, (u64)&sem);
|
||||||
|
|
||||||
|
ret = iommu_queue_command(iommu, &cmd);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
return wait_on_sem(&sem);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int iommu_flush_dte(struct amd_iommu *iommu, u16 devid)
|
||||||
{
|
{
|
||||||
struct iommu_cmd cmd;
|
struct iommu_cmd cmd;
|
||||||
|
|
||||||
memset(&cmd, 0, sizeof(cmd));
|
build_inv_dte(&cmd, devid);
|
||||||
cmd.data[0] = CMD_COMPL_WAIT_INT_MASK;
|
|
||||||
CMD_SET_TYPE(&cmd, CMD_COMPL_WAIT);
|
|
||||||
|
|
||||||
return __iommu_queue_command(iommu, &cmd);
|
return iommu_queue_command(iommu, &cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void iommu_flush_dte_all(struct amd_iommu *iommu)
|
||||||
|
{
|
||||||
|
u32 devid;
|
||||||
|
|
||||||
|
for (devid = 0; devid <= 0xffff; ++devid)
|
||||||
|
iommu_flush_dte(iommu, devid);
|
||||||
|
|
||||||
|
iommu_completion_wait(iommu);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function is called whenever we need to ensure that the IOMMU has
|
* This function uses heavy locking and may disable irqs for some time. But
|
||||||
* completed execution of all commands we sent. It sends a
|
* this is no issue because it is only called during resume.
|
||||||
* COMPLETION_WAIT command and waits for it to finish. The IOMMU informs
|
|
||||||
* us about that by writing a value to a physical address we pass with
|
|
||||||
* the command.
|
|
||||||
*/
|
*/
|
||||||
static int iommu_completion_wait(struct amd_iommu *iommu)
|
static void iommu_flush_tlb_all(struct amd_iommu *iommu)
|
||||||
{
|
{
|
||||||
int ret = 0;
|
u32 dom_id;
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&iommu->lock, flags);
|
for (dom_id = 0; dom_id <= 0xffff; ++dom_id) {
|
||||||
|
struct iommu_cmd cmd;
|
||||||
if (!iommu->need_sync)
|
build_inv_iommu_pages(&cmd, 0, CMD_INV_IOMMU_ALL_PAGES_ADDRESS,
|
||||||
goto out;
|
dom_id, 1);
|
||||||
|
iommu_queue_command(iommu, &cmd);
|
||||||
ret = __iommu_completion_wait(iommu);
|
|
||||||
|
|
||||||
iommu->need_sync = false;
|
|
||||||
|
|
||||||
if (ret)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
__iommu_wait_for_completion(iommu);
|
|
||||||
|
|
||||||
out:
|
|
||||||
spin_unlock_irqrestore(&iommu->lock, flags);
|
|
||||||
|
|
||||||
if (iommu->reset_in_progress)
|
|
||||||
reset_iommu_command_buffer(iommu);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void iommu_flush_complete(struct protection_domain *domain)
|
iommu_completion_wait(iommu);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void iommu_flush_all(struct amd_iommu *iommu)
|
||||||
|
{
|
||||||
|
struct iommu_cmd cmd;
|
||||||
|
|
||||||
|
build_inv_all(&cmd);
|
||||||
|
|
||||||
|
iommu_queue_command(iommu, &cmd);
|
||||||
|
iommu_completion_wait(iommu);
|
||||||
|
}
|
||||||
|
|
||||||
|
void iommu_flush_all_caches(struct amd_iommu *iommu)
|
||||||
|
{
|
||||||
|
if (iommu_feature(iommu, FEATURE_IA)) {
|
||||||
|
iommu_flush_all(iommu);
|
||||||
|
} else {
|
||||||
|
iommu_flush_dte_all(iommu);
|
||||||
|
iommu_flush_tlb_all(iommu);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Command send function for flushing on-device TLB
|
||||||
|
*/
|
||||||
|
static int device_flush_iotlb(struct device *dev, u64 address, size_t size)
|
||||||
|
{
|
||||||
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||||||
|
struct amd_iommu *iommu;
|
||||||
|
struct iommu_cmd cmd;
|
||||||
|
u16 devid;
|
||||||
|
int qdep;
|
||||||
|
|
||||||
|
qdep = pci_ats_queue_depth(pdev);
|
||||||
|
devid = get_device_id(dev);
|
||||||
|
iommu = amd_iommu_rlookup_table[devid];
|
||||||
|
|
||||||
|
build_inv_iotlb_pages(&cmd, devid, qdep, address, size);
|
||||||
|
|
||||||
|
return iommu_queue_command(iommu, &cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Command send function for invalidating a device table entry
|
||||||
|
*/
|
||||||
|
static int device_flush_dte(struct device *dev)
|
||||||
|
{
|
||||||
|
struct amd_iommu *iommu;
|
||||||
|
struct pci_dev *pdev;
|
||||||
|
u16 devid;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
pdev = to_pci_dev(dev);
|
||||||
|
devid = get_device_id(dev);
|
||||||
|
iommu = amd_iommu_rlookup_table[devid];
|
||||||
|
|
||||||
|
ret = iommu_flush_dte(iommu, devid);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
if (pci_ats_enabled(pdev))
|
||||||
|
ret = device_flush_iotlb(dev, 0, ~0UL);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* TLB invalidation function which is called from the mapping functions.
|
||||||
|
* It invalidates a single PTE if the range to flush is within a single
|
||||||
|
* page. Otherwise it flushes the whole TLB of the IOMMU.
|
||||||
|
*/
|
||||||
|
static void __domain_flush_pages(struct protection_domain *domain,
|
||||||
|
u64 address, size_t size, int pde)
|
||||||
|
{
|
||||||
|
struct iommu_dev_data *dev_data;
|
||||||
|
struct iommu_cmd cmd;
|
||||||
|
int ret = 0, i;
|
||||||
|
|
||||||
|
build_inv_iommu_pages(&cmd, address, size, domain->id, pde);
|
||||||
|
|
||||||
|
for (i = 0; i < amd_iommus_present; ++i) {
|
||||||
|
if (!domain->dev_iommu[i])
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Devices of this domain are behind this IOMMU
|
||||||
|
* We need a TLB flush
|
||||||
|
*/
|
||||||
|
ret |= iommu_queue_command(amd_iommus[i], &cmd);
|
||||||
|
}
|
||||||
|
|
||||||
|
list_for_each_entry(dev_data, &domain->dev_list, list) {
|
||||||
|
struct pci_dev *pdev = to_pci_dev(dev_data->dev);
|
||||||
|
|
||||||
|
if (!pci_ats_enabled(pdev))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
ret |= device_flush_iotlb(dev_data->dev, address, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
WARN_ON(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void domain_flush_pages(struct protection_domain *domain,
|
||||||
|
u64 address, size_t size)
|
||||||
|
{
|
||||||
|
__domain_flush_pages(domain, address, size, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Flush the whole IO/TLB for a given protection domain */
|
||||||
|
static void domain_flush_tlb(struct protection_domain *domain)
|
||||||
|
{
|
||||||
|
__domain_flush_pages(domain, 0, CMD_INV_IOMMU_ALL_PAGES_ADDRESS, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Flush the whole IO/TLB for a given protection domain - including PDE */
|
||||||
|
static void domain_flush_tlb_pde(struct protection_domain *domain)
|
||||||
|
{
|
||||||
|
__domain_flush_pages(domain, 0, CMD_INV_IOMMU_ALL_PAGES_ADDRESS, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void domain_flush_complete(struct protection_domain *domain)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
@ -516,118 +740,11 @@ static void iommu_flush_complete(struct protection_domain *domain)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* Command send function for invalidating a device table entry
|
|
||||||
*/
|
|
||||||
static int iommu_flush_device(struct device *dev)
|
|
||||||
{
|
|
||||||
struct amd_iommu *iommu;
|
|
||||||
struct iommu_cmd cmd;
|
|
||||||
u16 devid;
|
|
||||||
|
|
||||||
devid = get_device_id(dev);
|
|
||||||
iommu = amd_iommu_rlookup_table[devid];
|
|
||||||
|
|
||||||
/* Build command */
|
|
||||||
memset(&cmd, 0, sizeof(cmd));
|
|
||||||
CMD_SET_TYPE(&cmd, CMD_INV_DEV_ENTRY);
|
|
||||||
cmd.data[0] = devid;
|
|
||||||
|
|
||||||
return iommu_queue_command(iommu, &cmd);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void __iommu_build_inv_iommu_pages(struct iommu_cmd *cmd, u64 address,
|
|
||||||
u16 domid, int pde, int s)
|
|
||||||
{
|
|
||||||
memset(cmd, 0, sizeof(*cmd));
|
|
||||||
address &= PAGE_MASK;
|
|
||||||
CMD_SET_TYPE(cmd, CMD_INV_IOMMU_PAGES);
|
|
||||||
cmd->data[1] |= domid;
|
|
||||||
cmd->data[2] = lower_32_bits(address);
|
|
||||||
cmd->data[3] = upper_32_bits(address);
|
|
||||||
if (s) /* size bit - we flush more than one 4kb page */
|
|
||||||
cmd->data[2] |= CMD_INV_IOMMU_PAGES_SIZE_MASK;
|
|
||||||
if (pde) /* PDE bit - we wan't flush everything not only the PTEs */
|
|
||||||
cmd->data[2] |= CMD_INV_IOMMU_PAGES_PDE_MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Generic command send function for invalidaing TLB entries
|
|
||||||
*/
|
|
||||||
static int iommu_queue_inv_iommu_pages(struct amd_iommu *iommu,
|
|
||||||
u64 address, u16 domid, int pde, int s)
|
|
||||||
{
|
|
||||||
struct iommu_cmd cmd;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
__iommu_build_inv_iommu_pages(&cmd, address, domid, pde, s);
|
|
||||||
|
|
||||||
ret = iommu_queue_command(iommu, &cmd);
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* TLB invalidation function which is called from the mapping functions.
|
|
||||||
* It invalidates a single PTE if the range to flush is within a single
|
|
||||||
* page. Otherwise it flushes the whole TLB of the IOMMU.
|
|
||||||
*/
|
|
||||||
static void __iommu_flush_pages(struct protection_domain *domain,
|
|
||||||
u64 address, size_t size, int pde)
|
|
||||||
{
|
|
||||||
int s = 0, i;
|
|
||||||
unsigned long pages = iommu_num_pages(address, size, PAGE_SIZE);
|
|
||||||
|
|
||||||
address &= PAGE_MASK;
|
|
||||||
|
|
||||||
if (pages > 1) {
|
|
||||||
/*
|
|
||||||
* If we have to flush more than one page, flush all
|
|
||||||
* TLB entries for this domain
|
|
||||||
*/
|
|
||||||
address = CMD_INV_IOMMU_ALL_PAGES_ADDRESS;
|
|
||||||
s = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
for (i = 0; i < amd_iommus_present; ++i) {
|
|
||||||
if (!domain->dev_iommu[i])
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Devices of this domain are behind this IOMMU
|
|
||||||
* We need a TLB flush
|
|
||||||
*/
|
|
||||||
iommu_queue_inv_iommu_pages(amd_iommus[i], address,
|
|
||||||
domain->id, pde, s);
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void iommu_flush_pages(struct protection_domain *domain,
|
|
||||||
u64 address, size_t size)
|
|
||||||
{
|
|
||||||
__iommu_flush_pages(domain, address, size, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Flush the whole IO/TLB for a given protection domain */
|
|
||||||
static void iommu_flush_tlb(struct protection_domain *domain)
|
|
||||||
{
|
|
||||||
__iommu_flush_pages(domain, 0, CMD_INV_IOMMU_ALL_PAGES_ADDRESS, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Flush the whole IO/TLB for a given protection domain - including PDE */
|
|
||||||
static void iommu_flush_tlb_pde(struct protection_domain *domain)
|
|
||||||
{
|
|
||||||
__iommu_flush_pages(domain, 0, CMD_INV_IOMMU_ALL_PAGES_ADDRESS, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function flushes the DTEs for all devices in domain
|
* This function flushes the DTEs for all devices in domain
|
||||||
*/
|
*/
|
||||||
static void iommu_flush_domain_devices(struct protection_domain *domain)
|
static void domain_flush_devices(struct protection_domain *domain)
|
||||||
{
|
{
|
||||||
struct iommu_dev_data *dev_data;
|
struct iommu_dev_data *dev_data;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
@ -635,66 +752,11 @@ static void iommu_flush_domain_devices(struct protection_domain *domain)
|
|||||||
spin_lock_irqsave(&domain->lock, flags);
|
spin_lock_irqsave(&domain->lock, flags);
|
||||||
|
|
||||||
list_for_each_entry(dev_data, &domain->dev_list, list)
|
list_for_each_entry(dev_data, &domain->dev_list, list)
|
||||||
iommu_flush_device(dev_data->dev);
|
device_flush_dte(dev_data->dev);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void iommu_flush_all_domain_devices(void)
|
|
||||||
{
|
|
||||||
struct protection_domain *domain;
|
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&amd_iommu_pd_lock, flags);
|
|
||||||
|
|
||||||
list_for_each_entry(domain, &amd_iommu_pd_list, list) {
|
|
||||||
iommu_flush_domain_devices(domain);
|
|
||||||
iommu_flush_complete(domain);
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock_irqrestore(&amd_iommu_pd_lock, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
void amd_iommu_flush_all_devices(void)
|
|
||||||
{
|
|
||||||
iommu_flush_all_domain_devices();
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This function uses heavy locking and may disable irqs for some time. But
|
|
||||||
* this is no issue because it is only called during resume.
|
|
||||||
*/
|
|
||||||
void amd_iommu_flush_all_domains(void)
|
|
||||||
{
|
|
||||||
struct protection_domain *domain;
|
|
||||||
unsigned long flags;
|
|
||||||
|
|
||||||
spin_lock_irqsave(&amd_iommu_pd_lock, flags);
|
|
||||||
|
|
||||||
list_for_each_entry(domain, &amd_iommu_pd_list, list) {
|
|
||||||
spin_lock(&domain->lock);
|
|
||||||
iommu_flush_tlb_pde(domain);
|
|
||||||
iommu_flush_complete(domain);
|
|
||||||
spin_unlock(&domain->lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
spin_unlock_irqrestore(&amd_iommu_pd_lock, flags);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void reset_iommu_command_buffer(struct amd_iommu *iommu)
|
|
||||||
{
|
|
||||||
pr_err("AMD-Vi: Resetting IOMMU command buffer\n");
|
|
||||||
|
|
||||||
if (iommu->reset_in_progress)
|
|
||||||
panic("AMD-Vi: ILLEGAL_COMMAND_ERROR while resetting command buffer\n");
|
|
||||||
|
|
||||||
amd_iommu_reset_cmd_buffer(iommu);
|
|
||||||
amd_iommu_flush_all_devices();
|
|
||||||
amd_iommu_flush_all_domains();
|
|
||||||
|
|
||||||
iommu->reset_in_progress = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
*
|
*
|
||||||
* The functions below are used the create the page table mappings for
|
* The functions below are used the create the page table mappings for
|
||||||
@ -1410,14 +1472,19 @@ static bool dma_ops_domain(struct protection_domain *domain)
|
|||||||
return domain->flags & PD_DMA_OPS_MASK;
|
return domain->flags & PD_DMA_OPS_MASK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void set_dte_entry(u16 devid, struct protection_domain *domain)
|
static void set_dte_entry(u16 devid, struct protection_domain *domain, bool ats)
|
||||||
{
|
{
|
||||||
u64 pte_root = virt_to_phys(domain->pt_root);
|
u64 pte_root = virt_to_phys(domain->pt_root);
|
||||||
|
u32 flags = 0;
|
||||||
|
|
||||||
pte_root |= (domain->mode & DEV_ENTRY_MODE_MASK)
|
pte_root |= (domain->mode & DEV_ENTRY_MODE_MASK)
|
||||||
<< DEV_ENTRY_MODE_SHIFT;
|
<< DEV_ENTRY_MODE_SHIFT;
|
||||||
pte_root |= IOMMU_PTE_IR | IOMMU_PTE_IW | IOMMU_PTE_P | IOMMU_PTE_TV;
|
pte_root |= IOMMU_PTE_IR | IOMMU_PTE_IW | IOMMU_PTE_P | IOMMU_PTE_TV;
|
||||||
|
|
||||||
|
if (ats)
|
||||||
|
flags |= DTE_FLAG_IOTLB;
|
||||||
|
|
||||||
|
amd_iommu_dev_table[devid].data[3] |= flags;
|
||||||
amd_iommu_dev_table[devid].data[2] = domain->id;
|
amd_iommu_dev_table[devid].data[2] = domain->id;
|
||||||
amd_iommu_dev_table[devid].data[1] = upper_32_bits(pte_root);
|
amd_iommu_dev_table[devid].data[1] = upper_32_bits(pte_root);
|
||||||
amd_iommu_dev_table[devid].data[0] = lower_32_bits(pte_root);
|
amd_iommu_dev_table[devid].data[0] = lower_32_bits(pte_root);
|
||||||
@ -1437,34 +1504,42 @@ static void do_attach(struct device *dev, struct protection_domain *domain)
|
|||||||
{
|
{
|
||||||
struct iommu_dev_data *dev_data;
|
struct iommu_dev_data *dev_data;
|
||||||
struct amd_iommu *iommu;
|
struct amd_iommu *iommu;
|
||||||
|
struct pci_dev *pdev;
|
||||||
|
bool ats = false;
|
||||||
u16 devid;
|
u16 devid;
|
||||||
|
|
||||||
devid = get_device_id(dev);
|
devid = get_device_id(dev);
|
||||||
iommu = amd_iommu_rlookup_table[devid];
|
iommu = amd_iommu_rlookup_table[devid];
|
||||||
dev_data = get_dev_data(dev);
|
dev_data = get_dev_data(dev);
|
||||||
|
pdev = to_pci_dev(dev);
|
||||||
|
|
||||||
|
if (amd_iommu_iotlb_sup)
|
||||||
|
ats = pci_ats_enabled(pdev);
|
||||||
|
|
||||||
/* Update data structures */
|
/* Update data structures */
|
||||||
dev_data->domain = domain;
|
dev_data->domain = domain;
|
||||||
list_add(&dev_data->list, &domain->dev_list);
|
list_add(&dev_data->list, &domain->dev_list);
|
||||||
set_dte_entry(devid, domain);
|
set_dte_entry(devid, domain, ats);
|
||||||
|
|
||||||
/* Do reference counting */
|
/* Do reference counting */
|
||||||
domain->dev_iommu[iommu->index] += 1;
|
domain->dev_iommu[iommu->index] += 1;
|
||||||
domain->dev_cnt += 1;
|
domain->dev_cnt += 1;
|
||||||
|
|
||||||
/* Flush the DTE entry */
|
/* Flush the DTE entry */
|
||||||
iommu_flush_device(dev);
|
device_flush_dte(dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void do_detach(struct device *dev)
|
static void do_detach(struct device *dev)
|
||||||
{
|
{
|
||||||
struct iommu_dev_data *dev_data;
|
struct iommu_dev_data *dev_data;
|
||||||
struct amd_iommu *iommu;
|
struct amd_iommu *iommu;
|
||||||
|
struct pci_dev *pdev;
|
||||||
u16 devid;
|
u16 devid;
|
||||||
|
|
||||||
devid = get_device_id(dev);
|
devid = get_device_id(dev);
|
||||||
iommu = amd_iommu_rlookup_table[devid];
|
iommu = amd_iommu_rlookup_table[devid];
|
||||||
dev_data = get_dev_data(dev);
|
dev_data = get_dev_data(dev);
|
||||||
|
pdev = to_pci_dev(dev);
|
||||||
|
|
||||||
/* decrease reference counters */
|
/* decrease reference counters */
|
||||||
dev_data->domain->dev_iommu[iommu->index] -= 1;
|
dev_data->domain->dev_iommu[iommu->index] -= 1;
|
||||||
@ -1476,7 +1551,7 @@ static void do_detach(struct device *dev)
|
|||||||
clear_dte_entry(devid);
|
clear_dte_entry(devid);
|
||||||
|
|
||||||
/* Flush the DTE entry */
|
/* Flush the DTE entry */
|
||||||
iommu_flush_device(dev);
|
device_flush_dte(dev);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1539,9 +1614,13 @@ out_unlock:
|
|||||||
static int attach_device(struct device *dev,
|
static int attach_device(struct device *dev,
|
||||||
struct protection_domain *domain)
|
struct protection_domain *domain)
|
||||||
{
|
{
|
||||||
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
if (amd_iommu_iotlb_sup)
|
||||||
|
pci_enable_ats(pdev, PAGE_SHIFT);
|
||||||
|
|
||||||
write_lock_irqsave(&amd_iommu_devtable_lock, flags);
|
write_lock_irqsave(&amd_iommu_devtable_lock, flags);
|
||||||
ret = __attach_device(dev, domain);
|
ret = __attach_device(dev, domain);
|
||||||
write_unlock_irqrestore(&amd_iommu_devtable_lock, flags);
|
write_unlock_irqrestore(&amd_iommu_devtable_lock, flags);
|
||||||
@ -1551,7 +1630,7 @@ static int attach_device(struct device *dev,
|
|||||||
* left the caches in the IOMMU dirty. So we have to flush
|
* left the caches in the IOMMU dirty. So we have to flush
|
||||||
* here to evict all dirty stuff.
|
* here to evict all dirty stuff.
|
||||||
*/
|
*/
|
||||||
iommu_flush_tlb_pde(domain);
|
domain_flush_tlb_pde(domain);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -1598,12 +1677,16 @@ static void __detach_device(struct device *dev)
|
|||||||
*/
|
*/
|
||||||
static void detach_device(struct device *dev)
|
static void detach_device(struct device *dev)
|
||||||
{
|
{
|
||||||
|
struct pci_dev *pdev = to_pci_dev(dev);
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
|
|
||||||
/* lock device table */
|
/* lock device table */
|
||||||
write_lock_irqsave(&amd_iommu_devtable_lock, flags);
|
write_lock_irqsave(&amd_iommu_devtable_lock, flags);
|
||||||
__detach_device(dev);
|
__detach_device(dev);
|
||||||
write_unlock_irqrestore(&amd_iommu_devtable_lock, flags);
|
write_unlock_irqrestore(&amd_iommu_devtable_lock, flags);
|
||||||
|
|
||||||
|
if (amd_iommu_iotlb_sup && pci_ats_enabled(pdev))
|
||||||
|
pci_disable_ats(pdev);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1692,7 +1775,7 @@ static int device_change_notifier(struct notifier_block *nb,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
iommu_flush_device(dev);
|
device_flush_dte(dev);
|
||||||
iommu_completion_wait(iommu);
|
iommu_completion_wait(iommu);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
@ -1753,8 +1836,9 @@ static void update_device_table(struct protection_domain *domain)
|
|||||||
struct iommu_dev_data *dev_data;
|
struct iommu_dev_data *dev_data;
|
||||||
|
|
||||||
list_for_each_entry(dev_data, &domain->dev_list, list) {
|
list_for_each_entry(dev_data, &domain->dev_list, list) {
|
||||||
|
struct pci_dev *pdev = to_pci_dev(dev_data->dev);
|
||||||
u16 devid = get_device_id(dev_data->dev);
|
u16 devid = get_device_id(dev_data->dev);
|
||||||
set_dte_entry(devid, domain);
|
set_dte_entry(devid, domain, pci_ats_enabled(pdev));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1764,8 +1848,9 @@ static void update_domain(struct protection_domain *domain)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
update_device_table(domain);
|
update_device_table(domain);
|
||||||
iommu_flush_domain_devices(domain);
|
|
||||||
iommu_flush_tlb_pde(domain);
|
domain_flush_devices(domain);
|
||||||
|
domain_flush_tlb_pde(domain);
|
||||||
|
|
||||||
domain->updated = false;
|
domain->updated = false;
|
||||||
}
|
}
|
||||||
@ -1924,10 +2009,10 @@ retry:
|
|||||||
ADD_STATS_COUNTER(alloced_io_mem, size);
|
ADD_STATS_COUNTER(alloced_io_mem, size);
|
||||||
|
|
||||||
if (unlikely(dma_dom->need_flush && !amd_iommu_unmap_flush)) {
|
if (unlikely(dma_dom->need_flush && !amd_iommu_unmap_flush)) {
|
||||||
iommu_flush_tlb(&dma_dom->domain);
|
domain_flush_tlb(&dma_dom->domain);
|
||||||
dma_dom->need_flush = false;
|
dma_dom->need_flush = false;
|
||||||
} else if (unlikely(amd_iommu_np_cache))
|
} else if (unlikely(amd_iommu_np_cache))
|
||||||
iommu_flush_pages(&dma_dom->domain, address, size);
|
domain_flush_pages(&dma_dom->domain, address, size);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return address;
|
return address;
|
||||||
@ -1976,7 +2061,7 @@ static void __unmap_single(struct dma_ops_domain *dma_dom,
|
|||||||
dma_ops_free_addresses(dma_dom, dma_addr, pages);
|
dma_ops_free_addresses(dma_dom, dma_addr, pages);
|
||||||
|
|
||||||
if (amd_iommu_unmap_flush || dma_dom->need_flush) {
|
if (amd_iommu_unmap_flush || dma_dom->need_flush) {
|
||||||
iommu_flush_pages(&dma_dom->domain, flush_addr, size);
|
domain_flush_pages(&dma_dom->domain, flush_addr, size);
|
||||||
dma_dom->need_flush = false;
|
dma_dom->need_flush = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -2012,7 +2097,7 @@ static dma_addr_t map_page(struct device *dev, struct page *page,
|
|||||||
if (addr == DMA_ERROR_CODE)
|
if (addr == DMA_ERROR_CODE)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
@ -2039,7 +2124,7 @@ static void unmap_page(struct device *dev, dma_addr_t dma_addr, size_t size,
|
|||||||
|
|
||||||
__unmap_single(domain->priv, dma_addr, size, dir);
|
__unmap_single(domain->priv, dma_addr, size, dir);
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
}
|
}
|
||||||
@ -2104,7 +2189,7 @@ static int map_sg(struct device *dev, struct scatterlist *sglist,
|
|||||||
goto unmap;
|
goto unmap;
|
||||||
}
|
}
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
out:
|
out:
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
@ -2150,7 +2235,7 @@ static void unmap_sg(struct device *dev, struct scatterlist *sglist,
|
|||||||
s->dma_address = s->dma_length = 0;
|
s->dma_address = s->dma_length = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
}
|
}
|
||||||
@ -2200,7 +2285,7 @@ static void *alloc_coherent(struct device *dev, size_t size,
|
|||||||
goto out_free;
|
goto out_free;
|
||||||
}
|
}
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
|
|
||||||
@ -2232,7 +2317,7 @@ static void free_coherent(struct device *dev, size_t size,
|
|||||||
|
|
||||||
__unmap_single(domain->priv, dma_addr, size, DMA_BIDIRECTIONAL);
|
__unmap_single(domain->priv, dma_addr, size, DMA_BIDIRECTIONAL);
|
||||||
|
|
||||||
iommu_flush_complete(domain);
|
domain_flush_complete(domain);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&domain->lock, flags);
|
spin_unlock_irqrestore(&domain->lock, flags);
|
||||||
|
|
||||||
@ -2476,7 +2561,7 @@ static void amd_iommu_detach_device(struct iommu_domain *dom,
|
|||||||
if (!iommu)
|
if (!iommu)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
iommu_flush_device(dev);
|
device_flush_dte(dev);
|
||||||
iommu_completion_wait(iommu);
|
iommu_completion_wait(iommu);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2542,7 +2627,7 @@ static int amd_iommu_unmap(struct iommu_domain *dom, unsigned long iova,
|
|||||||
unmap_size = iommu_unmap_page(domain, iova, page_size);
|
unmap_size = iommu_unmap_page(domain, iova, page_size);
|
||||||
mutex_unlock(&domain->api_lock);
|
mutex_unlock(&domain->api_lock);
|
||||||
|
|
||||||
iommu_flush_tlb_pde(domain);
|
domain_flush_tlb_pde(domain);
|
||||||
|
|
||||||
return get_order(unmap_size);
|
return get_order(unmap_size);
|
||||||
}
|
}
|
||||||
|
@ -137,6 +137,7 @@ int amd_iommus_present;
|
|||||||
|
|
||||||
/* IOMMUs have a non-present cache? */
|
/* IOMMUs have a non-present cache? */
|
||||||
bool amd_iommu_np_cache __read_mostly;
|
bool amd_iommu_np_cache __read_mostly;
|
||||||
|
bool amd_iommu_iotlb_sup __read_mostly = true;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* The ACPI table parsing functions set this variable on an error
|
* The ACPI table parsing functions set this variable on an error
|
||||||
@ -180,6 +181,12 @@ static u32 dev_table_size; /* size of the device table */
|
|||||||
static u32 alias_table_size; /* size of the alias table */
|
static u32 alias_table_size; /* size of the alias table */
|
||||||
static u32 rlookup_table_size; /* size if the rlookup table */
|
static u32 rlookup_table_size; /* size if the rlookup table */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This function flushes all internal caches of
|
||||||
|
* the IOMMU used by this driver.
|
||||||
|
*/
|
||||||
|
extern void iommu_flush_all_caches(struct amd_iommu *iommu);
|
||||||
|
|
||||||
static inline void update_last_devid(u16 devid)
|
static inline void update_last_devid(u16 devid)
|
||||||
{
|
{
|
||||||
if (devid > amd_iommu_last_bdf)
|
if (devid > amd_iommu_last_bdf)
|
||||||
@ -293,9 +300,23 @@ static void iommu_feature_disable(struct amd_iommu *iommu, u8 bit)
|
|||||||
/* Function to enable the hardware */
|
/* Function to enable the hardware */
|
||||||
static void iommu_enable(struct amd_iommu *iommu)
|
static void iommu_enable(struct amd_iommu *iommu)
|
||||||
{
|
{
|
||||||
printk(KERN_INFO "AMD-Vi: Enabling IOMMU at %s cap 0x%hx\n",
|
static const char * const feat_str[] = {
|
||||||
|
"PreF", "PPR", "X2APIC", "NX", "GT", "[5]",
|
||||||
|
"IA", "GA", "HE", "PC", NULL
|
||||||
|
};
|
||||||
|
int i;
|
||||||
|
|
||||||
|
printk(KERN_INFO "AMD-Vi: Enabling IOMMU at %s cap 0x%hx",
|
||||||
dev_name(&iommu->dev->dev), iommu->cap_ptr);
|
dev_name(&iommu->dev->dev), iommu->cap_ptr);
|
||||||
|
|
||||||
|
if (iommu->cap & (1 << IOMMU_CAP_EFR)) {
|
||||||
|
printk(KERN_CONT " extended features: ");
|
||||||
|
for (i = 0; feat_str[i]; ++i)
|
||||||
|
if (iommu_feature(iommu, (1ULL << i)))
|
||||||
|
printk(KERN_CONT " %s", feat_str[i]);
|
||||||
|
}
|
||||||
|
printk(KERN_CONT "\n");
|
||||||
|
|
||||||
iommu_feature_enable(iommu, CONTROL_IOMMU_EN);
|
iommu_feature_enable(iommu, CONTROL_IOMMU_EN);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -651,7 +672,7 @@ static void __init set_device_exclusion_range(u16 devid, struct ivmd_header *m)
|
|||||||
static void __init init_iommu_from_pci(struct amd_iommu *iommu)
|
static void __init init_iommu_from_pci(struct amd_iommu *iommu)
|
||||||
{
|
{
|
||||||
int cap_ptr = iommu->cap_ptr;
|
int cap_ptr = iommu->cap_ptr;
|
||||||
u32 range, misc;
|
u32 range, misc, low, high;
|
||||||
int i, j;
|
int i, j;
|
||||||
|
|
||||||
pci_read_config_dword(iommu->dev, cap_ptr + MMIO_CAP_HDR_OFFSET,
|
pci_read_config_dword(iommu->dev, cap_ptr + MMIO_CAP_HDR_OFFSET,
|
||||||
@ -667,6 +688,15 @@ static void __init init_iommu_from_pci(struct amd_iommu *iommu)
|
|||||||
MMIO_GET_LD(range));
|
MMIO_GET_LD(range));
|
||||||
iommu->evt_msi_num = MMIO_MSI_NUM(misc);
|
iommu->evt_msi_num = MMIO_MSI_NUM(misc);
|
||||||
|
|
||||||
|
if (!(iommu->cap & (1 << IOMMU_CAP_IOTLB)))
|
||||||
|
amd_iommu_iotlb_sup = false;
|
||||||
|
|
||||||
|
/* read extended feature bits */
|
||||||
|
low = readl(iommu->mmio_base + MMIO_EXT_FEATURES);
|
||||||
|
high = readl(iommu->mmio_base + MMIO_EXT_FEATURES + 4);
|
||||||
|
|
||||||
|
iommu->features = ((u64)high << 32) | low;
|
||||||
|
|
||||||
if (!is_rd890_iommu(iommu->dev))
|
if (!is_rd890_iommu(iommu->dev))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -1244,6 +1274,7 @@ static void enable_iommus(void)
|
|||||||
iommu_set_exclusion_range(iommu);
|
iommu_set_exclusion_range(iommu);
|
||||||
iommu_init_msi(iommu);
|
iommu_init_msi(iommu);
|
||||||
iommu_enable(iommu);
|
iommu_enable(iommu);
|
||||||
|
iommu_flush_all_caches(iommu);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1274,8 +1305,8 @@ static void amd_iommu_resume(void)
|
|||||||
* we have to flush after the IOMMUs are enabled because a
|
* we have to flush after the IOMMUs are enabled because a
|
||||||
* disabled IOMMU will never execute the commands we send
|
* disabled IOMMU will never execute the commands we send
|
||||||
*/
|
*/
|
||||||
amd_iommu_flush_all_devices();
|
for_each_iommu(iommu)
|
||||||
amd_iommu_flush_all_domains();
|
iommu_flush_all_caches(iommu);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int amd_iommu_suspend(void)
|
static int amd_iommu_suspend(void)
|
||||||
|
@ -39,6 +39,7 @@
|
|||||||
#include <linux/syscore_ops.h>
|
#include <linux/syscore_ops.h>
|
||||||
#include <linux/tboot.h>
|
#include <linux/tboot.h>
|
||||||
#include <linux/dmi.h>
|
#include <linux/dmi.h>
|
||||||
|
#include <linux/pci-ats.h>
|
||||||
#include <asm/cacheflush.h>
|
#include <asm/cacheflush.h>
|
||||||
#include <asm/iommu.h>
|
#include <asm/iommu.h>
|
||||||
#include "pci.h"
|
#include "pci.h"
|
||||||
|
@ -13,6 +13,7 @@
|
|||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
|
#include <linux/pci-ats.h>
|
||||||
#include "pci.h"
|
#include "pci.h"
|
||||||
|
|
||||||
#define VIRTFN_ID_LEN 16
|
#define VIRTFN_ID_LEN 16
|
||||||
|
@ -250,15 +250,6 @@ struct pci_sriov {
|
|||||||
u8 __iomem *mstate; /* VF Migration State Array */
|
u8 __iomem *mstate; /* VF Migration State Array */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Address Translation Service */
|
|
||||||
struct pci_ats {
|
|
||||||
int pos; /* capability position */
|
|
||||||
int stu; /* Smallest Translation Unit */
|
|
||||||
int qdep; /* Invalidate Queue Depth */
|
|
||||||
int ref_cnt; /* Physical Function reference count */
|
|
||||||
unsigned int is_enabled:1; /* Enable bit is set */
|
|
||||||
};
|
|
||||||
|
|
||||||
#ifdef CONFIG_PCI_IOV
|
#ifdef CONFIG_PCI_IOV
|
||||||
extern int pci_iov_init(struct pci_dev *dev);
|
extern int pci_iov_init(struct pci_dev *dev);
|
||||||
extern void pci_iov_release(struct pci_dev *dev);
|
extern void pci_iov_release(struct pci_dev *dev);
|
||||||
@ -269,19 +260,6 @@ extern resource_size_t pci_sriov_resource_alignment(struct pci_dev *dev,
|
|||||||
extern void pci_restore_iov_state(struct pci_dev *dev);
|
extern void pci_restore_iov_state(struct pci_dev *dev);
|
||||||
extern int pci_iov_bus_range(struct pci_bus *bus);
|
extern int pci_iov_bus_range(struct pci_bus *bus);
|
||||||
|
|
||||||
extern int pci_enable_ats(struct pci_dev *dev, int ps);
|
|
||||||
extern void pci_disable_ats(struct pci_dev *dev);
|
|
||||||
extern int pci_ats_queue_depth(struct pci_dev *dev);
|
|
||||||
/**
|
|
||||||
* pci_ats_enabled - query the ATS status
|
|
||||||
* @dev: the PCI device
|
|
||||||
*
|
|
||||||
* Returns 1 if ATS capability is enabled, or 0 if not.
|
|
||||||
*/
|
|
||||||
static inline int pci_ats_enabled(struct pci_dev *dev)
|
|
||||||
{
|
|
||||||
return dev->ats && dev->ats->is_enabled;
|
|
||||||
}
|
|
||||||
#else
|
#else
|
||||||
static inline int pci_iov_init(struct pci_dev *dev)
|
static inline int pci_iov_init(struct pci_dev *dev)
|
||||||
{
|
{
|
||||||
@ -304,21 +282,6 @@ static inline int pci_iov_bus_range(struct pci_bus *bus)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline int pci_enable_ats(struct pci_dev *dev, int ps)
|
|
||||||
{
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
|
||||||
static inline void pci_disable_ats(struct pci_dev *dev)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
static inline int pci_ats_queue_depth(struct pci_dev *dev)
|
|
||||||
{
|
|
||||||
return -ENODEV;
|
|
||||||
}
|
|
||||||
static inline int pci_ats_enabled(struct pci_dev *dev)
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif /* CONFIG_PCI_IOV */
|
#endif /* CONFIG_PCI_IOV */
|
||||||
|
|
||||||
static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
static inline resource_size_t pci_resource_alignment(struct pci_dev *dev,
|
||||||
|
52
include/linux/pci-ats.h
Normal file
52
include/linux/pci-ats.h
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
#ifndef LINUX_PCI_ATS_H
|
||||||
|
#define LINUX_PCI_ATS_H
|
||||||
|
|
||||||
|
/* Address Translation Service */
|
||||||
|
struct pci_ats {
|
||||||
|
int pos; /* capability position */
|
||||||
|
int stu; /* Smallest Translation Unit */
|
||||||
|
int qdep; /* Invalidate Queue Depth */
|
||||||
|
int ref_cnt; /* Physical Function reference count */
|
||||||
|
unsigned int is_enabled:1; /* Enable bit is set */
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef CONFIG_PCI_IOV
|
||||||
|
|
||||||
|
extern int pci_enable_ats(struct pci_dev *dev, int ps);
|
||||||
|
extern void pci_disable_ats(struct pci_dev *dev);
|
||||||
|
extern int pci_ats_queue_depth(struct pci_dev *dev);
|
||||||
|
/**
|
||||||
|
* pci_ats_enabled - query the ATS status
|
||||||
|
* @dev: the PCI device
|
||||||
|
*
|
||||||
|
* Returns 1 if ATS capability is enabled, or 0 if not.
|
||||||
|
*/
|
||||||
|
static inline int pci_ats_enabled(struct pci_dev *dev)
|
||||||
|
{
|
||||||
|
return dev->ats && dev->ats->is_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
|
#else /* CONFIG_PCI_IOV */
|
||||||
|
|
||||||
|
static inline int pci_enable_ats(struct pci_dev *dev, int ps)
|
||||||
|
{
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void pci_disable_ats(struct pci_dev *dev)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int pci_ats_queue_depth(struct pci_dev *dev)
|
||||||
|
{
|
||||||
|
return -ENODEV;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline int pci_ats_enabled(struct pci_dev *dev)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* CONFIG_PCI_IOV */
|
||||||
|
|
||||||
|
#endif /* LINUX_PCI_ATS_H*/
|
@ -649,7 +649,7 @@ out_err:
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int device_dma_allocations(struct device *dev)
|
static int device_dma_allocations(struct device *dev, struct dma_debug_entry **out_entry)
|
||||||
{
|
{
|
||||||
struct dma_debug_entry *entry;
|
struct dma_debug_entry *entry;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
@ -660,8 +660,10 @@ static int device_dma_allocations(struct device *dev)
|
|||||||
for (i = 0; i < HASH_SIZE; ++i) {
|
for (i = 0; i < HASH_SIZE; ++i) {
|
||||||
spin_lock(&dma_entry_hash[i].lock);
|
spin_lock(&dma_entry_hash[i].lock);
|
||||||
list_for_each_entry(entry, &dma_entry_hash[i].list, list) {
|
list_for_each_entry(entry, &dma_entry_hash[i].list, list) {
|
||||||
if (entry->dev == dev)
|
if (entry->dev == dev) {
|
||||||
count += 1;
|
count += 1;
|
||||||
|
*out_entry = entry;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
spin_unlock(&dma_entry_hash[i].lock);
|
spin_unlock(&dma_entry_hash[i].lock);
|
||||||
}
|
}
|
||||||
@ -674,6 +676,7 @@ static int device_dma_allocations(struct device *dev)
|
|||||||
static int dma_debug_device_change(struct notifier_block *nb, unsigned long action, void *data)
|
static int dma_debug_device_change(struct notifier_block *nb, unsigned long action, void *data)
|
||||||
{
|
{
|
||||||
struct device *dev = data;
|
struct device *dev = data;
|
||||||
|
struct dma_debug_entry *uninitialized_var(entry);
|
||||||
int count;
|
int count;
|
||||||
|
|
||||||
if (global_disable)
|
if (global_disable)
|
||||||
@ -681,12 +684,17 @@ static int dma_debug_device_change(struct notifier_block *nb, unsigned long acti
|
|||||||
|
|
||||||
switch (action) {
|
switch (action) {
|
||||||
case BUS_NOTIFY_UNBOUND_DRIVER:
|
case BUS_NOTIFY_UNBOUND_DRIVER:
|
||||||
count = device_dma_allocations(dev);
|
count = device_dma_allocations(dev, &entry);
|
||||||
if (count == 0)
|
if (count == 0)
|
||||||
break;
|
break;
|
||||||
err_printk(dev, NULL, "DMA-API: device driver has pending "
|
err_printk(dev, entry, "DMA-API: device driver has pending "
|
||||||
"DMA allocations while released from device "
|
"DMA allocations while released from device "
|
||||||
"[count=%d]\n", count);
|
"[count=%d]\n"
|
||||||
|
"One of leaked entries details: "
|
||||||
|
"[device address=0x%016llx] [size=%llu bytes] "
|
||||||
|
"[mapped with %s] [mapped as %s]\n",
|
||||||
|
count, entry->dev_addr, entry->size,
|
||||||
|
dir2name[entry->direction], type2name[entry->type]);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
|
Loading…
Reference in New Issue
Block a user