#define pr_fmt(fmt) …
#define dev_fmt(fmt) …
#include <linux/ratelimit.h>
#include <linux/pci.h>
#include <linux/acpi.h>
#include <linux/pci-ats.h>
#include <linux/bitmap.h>
#include <linux/slab.h>
#include <linux/debugfs.h>
#include <linux/scatterlist.h>
#include <linux/dma-map-ops.h>
#include <linux/dma-direct.h>
#include <linux/iommu-helper.h>
#include <linux/delay.h>
#include <linux/amd-iommu.h>
#include <linux/notifier.h>
#include <linux/export.h>
#include <linux/irq.h>
#include <linux/msi.h>
#include <linux/irqdomain.h>
#include <linux/percpu.h>
#include <linux/io-pgtable.h>
#include <linux/cc_platform.h>
#include <asm/irq_remapping.h>
#include <asm/io_apic.h>
#include <asm/apic.h>
#include <asm/hw_irq.h>
#include <asm/proto.h>
#include <asm/iommu.h>
#include <asm/gart.h>
#include <asm/dma.h>
#include <uapi/linux/iommufd.h>
#include "amd_iommu.h"
#include "../dma-iommu.h"
#include "../irq_remapping.h"
#include "../iommu-pages.h"
#define CMD_SET_TYPE(cmd, t) …
#define MSI_RANGE_START …
#define MSI_RANGE_END …
#define HT_RANGE_START …
#define HT_RANGE_END …
static DEFINE_SPINLOCK(pd_bitmap_lock);
LIST_HEAD(…);
LIST_HEAD(…);
LIST_HEAD(…);
const struct iommu_ops amd_iommu_ops;
static const struct iommu_dirty_ops amd_dirty_ops;
int amd_iommu_max_glx_val = …;
struct iommu_cmd { … };
struct kmem_cache *amd_iommu_irq_cache;
static void detach_device(struct device *dev);
static void set_dte_entry(struct amd_iommu *iommu,
struct iommu_dev_data *dev_data);
static inline bool pdom_is_v2_pgtbl_mode(struct protection_domain *pdom)
{ … }
static inline bool pdom_is_in_pt_mode(struct protection_domain *pdom)
{ … }
static inline bool pdom_is_sva_capable(struct protection_domain *pdom)
{ … }
static inline int get_acpihid_device_id(struct device *dev,
struct acpihid_map_entry **entry)
{ … }
static inline int get_device_sbdf_id(struct device *dev)
{ … }
struct dev_table_entry *get_dev_table(struct amd_iommu *iommu)
{ … }
static inline u16 get_device_segment(struct device *dev)
{ … }
void amd_iommu_set_rlookup_table(struct amd_iommu *iommu, u16 devid)
{ … }
static struct amd_iommu *__rlookup_amd_iommu(u16 seg, u16 devid)
{ … }
static struct amd_iommu *rlookup_amd_iommu(struct device *dev)
{ … }
static struct iommu_dev_data *alloc_dev_data(struct amd_iommu *iommu, u16 devid)
{ … }
static struct iommu_dev_data *search_dev_data(struct amd_iommu *iommu, u16 devid)
{ … }
static int clone_alias(struct pci_dev *pdev, u16 alias, void *data)
{ … }
static void clone_aliases(struct amd_iommu *iommu, struct device *dev)
{ … }
static void setup_aliases(struct amd_iommu *iommu, struct device *dev)
{ … }
static struct iommu_dev_data *find_dev_data(struct amd_iommu *iommu, u16 devid)
{ … }
static struct iommu_group *acpihid_device_group(struct device *dev)
{ … }
static inline bool pdev_pasid_supported(struct iommu_dev_data *dev_data)
{ … }
static u32 pdev_get_caps(struct pci_dev *pdev)
{ … }
static inline int pdev_enable_cap_ats(struct pci_dev *pdev)
{ … }
static inline void pdev_disable_cap_ats(struct pci_dev *pdev)
{ … }
static inline int pdev_enable_cap_pri(struct pci_dev *pdev)
{ … }
static inline void pdev_disable_cap_pri(struct pci_dev *pdev)
{ … }
static inline int pdev_enable_cap_pasid(struct pci_dev *pdev)
{ … }
static inline void pdev_disable_cap_pasid(struct pci_dev *pdev)
{ … }
static void pdev_enable_caps(struct pci_dev *pdev)
{ … }
static void pdev_disable_caps(struct pci_dev *pdev)
{ … }
static bool check_device(struct device *dev)
{ … }
static int iommu_init_device(struct amd_iommu *iommu, struct device *dev)
{ … }
static void iommu_ignore_device(struct amd_iommu *iommu, struct device *dev)
{ … }
static void amd_iommu_uninit_device(struct device *dev)
{ … }
static void dump_dte_entry(struct amd_iommu *iommu, u16 devid)
{ … }
static void dump_command(unsigned long phys_addr)
{ … }
static void amd_iommu_report_rmp_hw_error(struct amd_iommu *iommu, volatile u32 *event)
{ … }
static void amd_iommu_report_rmp_fault(struct amd_iommu *iommu, volatile u32 *event)
{ … }
#define IS_IOMMU_MEM_TRANSACTION(flags) …
#define IS_WRITE_REQUEST(flags) …
static void amd_iommu_report_page_fault(struct amd_iommu *iommu,
u16 devid, u16 domain_id,
u64 address, int flags)
{ … }
static void iommu_print_event(struct amd_iommu *iommu, void *__evt)
{ … }
static void iommu_poll_events(struct amd_iommu *iommu)
{ … }
#ifdef CONFIG_IRQ_REMAP
static int (*iommu_ga_log_notifier)(u32);
int amd_iommu_register_ga_log_notifier(int (*notifier)(u32))
{ … }
EXPORT_SYMBOL(…);
static void iommu_poll_ga_log(struct amd_iommu *iommu)
{ … }
static void
amd_iommu_set_pci_msi_domain(struct device *dev, struct amd_iommu *iommu)
{ … }
#else
static inline void
amd_iommu_set_pci_msi_domain(struct device *dev, struct amd_iommu *iommu) { }
#endif
static void amd_iommu_handle_irq(void *data, const char *evt_type,
u32 int_mask, u32 overflow_mask,
void (*int_handler)(struct amd_iommu *),
void (*overflow_handler)(struct amd_iommu *))
{ … }
irqreturn_t amd_iommu_int_thread_evtlog(int irq, void *data)
{ … }
irqreturn_t amd_iommu_int_thread_pprlog(int irq, void *data)
{ … }
irqreturn_t amd_iommu_int_thread_galog(int irq, void *data)
{ … }
irqreturn_t amd_iommu_int_thread(int irq, void *data)
{ … }
irqreturn_t amd_iommu_int_handler(int irq, void *data)
{ … }
static int wait_on_sem(struct amd_iommu *iommu, u64 data)
{ … }
static void copy_cmd_to_buffer(struct amd_iommu *iommu,
struct iommu_cmd *cmd)
{ … }
static void build_completion_wait(struct iommu_cmd *cmd,
struct amd_iommu *iommu,
u64 data)
{ … }
static void build_inv_dte(struct iommu_cmd *cmd, u16 devid)
{ … }
static inline u64 build_inv_address(u64 address, size_t size)
{ … }
static void build_inv_iommu_pages(struct iommu_cmd *cmd, u64 address,
size_t size, u16 domid,
ioasid_t pasid, bool gn)
{ … }
static void build_inv_iotlb_pages(struct iommu_cmd *cmd, u16 devid, int qdep,
u64 address, size_t size,
ioasid_t pasid, bool gn)
{ … }
static void build_complete_ppr(struct iommu_cmd *cmd, u16 devid, u32 pasid,
int status, int tag, u8 gn)
{ … }
static void build_inv_all(struct iommu_cmd *cmd)
{ … }
static void build_inv_irt(struct iommu_cmd *cmd, u16 devid)
{ … }
static int __iommu_queue_command_sync(struct amd_iommu *iommu,
struct iommu_cmd *cmd,
bool sync)
{ … }
static int iommu_queue_command_sync(struct amd_iommu *iommu,
struct iommu_cmd *cmd,
bool sync)
{ … }
static int iommu_queue_command(struct amd_iommu *iommu, struct iommu_cmd *cmd)
{ … }
static int iommu_completion_wait(struct amd_iommu *iommu)
{ … }
static void domain_flush_complete(struct protection_domain *domain)
{ … }
static int iommu_flush_dte(struct amd_iommu *iommu, u16 devid)
{ … }
static void amd_iommu_flush_dte_all(struct amd_iommu *iommu)
{ … }
static void amd_iommu_flush_tlb_all(struct amd_iommu *iommu)
{ … }
static void amd_iommu_flush_tlb_domid(struct amd_iommu *iommu, u32 dom_id)
{ … }
static void amd_iommu_flush_all(struct amd_iommu *iommu)
{ … }
static void iommu_flush_irt(struct amd_iommu *iommu, u16 devid)
{ … }
static void amd_iommu_flush_irt_all(struct amd_iommu *iommu)
{ … }
void amd_iommu_flush_all_caches(struct amd_iommu *iommu)
{ … }
static int device_flush_iotlb(struct iommu_dev_data *dev_data, u64 address,
size_t size, ioasid_t pasid, bool gn)
{ … }
static int device_flush_dte_alias(struct pci_dev *pdev, u16 alias, void *data)
{ … }
static int device_flush_dte(struct iommu_dev_data *dev_data)
{ … }
static int domain_flush_pages_v2(struct protection_domain *pdom,
u64 address, size_t size)
{ … }
static int domain_flush_pages_v1(struct protection_domain *pdom,
u64 address, size_t size)
{ … }
static void __domain_flush_pages(struct protection_domain *domain,
u64 address, size_t size)
{ … }
void amd_iommu_domain_flush_pages(struct protection_domain *domain,
u64 address, size_t size)
{ … }
static void amd_iommu_domain_flush_all(struct protection_domain *domain)
{ … }
void amd_iommu_dev_flush_pasid_pages(struct iommu_dev_data *dev_data,
ioasid_t pasid, u64 address, size_t size)
{ … }
static void dev_flush_pasid_all(struct iommu_dev_data *dev_data,
ioasid_t pasid)
{ … }
static void domain_flush_np_cache(struct protection_domain *domain,
dma_addr_t iova, size_t size)
{ … }
void amd_iommu_update_and_flush_device_table(struct protection_domain *domain)
{ … }
void amd_iommu_domain_update(struct protection_domain *domain)
{ … }
int amd_iommu_complete_ppr(struct device *dev, u32 pasid, int status, int tag)
{ … }
static u16 domain_id_alloc(void)
{ … }
static void domain_id_free(int id)
{ … }
static void free_gcr3_tbl_level1(u64 *tbl)
{ … }
static void free_gcr3_tbl_level2(u64 *tbl)
{ … }
static void free_gcr3_table(struct gcr3_tbl_info *gcr3_info)
{ … }
static int get_gcr3_levels(int pasids)
{ … }
static int setup_gcr3_table(struct gcr3_tbl_info *gcr3_info,
struct amd_iommu *iommu, int pasids)
{ … }
static u64 *__get_gcr3_pte(struct gcr3_tbl_info *gcr3_info,
ioasid_t pasid, bool alloc)
{ … }
static int update_gcr3(struct iommu_dev_data *dev_data,
ioasid_t pasid, unsigned long gcr3, bool set)
{ … }
int amd_iommu_set_gcr3(struct iommu_dev_data *dev_data, ioasid_t pasid,
unsigned long gcr3)
{ … }
int amd_iommu_clear_gcr3(struct iommu_dev_data *dev_data, ioasid_t pasid)
{ … }
static void set_dte_entry(struct amd_iommu *iommu,
struct iommu_dev_data *dev_data)
{ … }
static void clear_dte_entry(struct amd_iommu *iommu, u16 devid)
{ … }
static void dev_update_dte(struct iommu_dev_data *dev_data, bool set)
{ … }
static int init_gcr3_table(struct iommu_dev_data *dev_data,
struct protection_domain *pdom)
{ … }
static void destroy_gcr3_table(struct iommu_dev_data *dev_data,
struct protection_domain *pdom)
{ … }
static int do_attach(struct iommu_dev_data *dev_data,
struct protection_domain *domain)
{ … }
static void do_detach(struct iommu_dev_data *dev_data)
{ … }
static int attach_device(struct device *dev,
struct protection_domain *domain)
{ … }
static void detach_device(struct device *dev)
{ … }
static struct iommu_device *amd_iommu_probe_device(struct device *dev)
{ … }
static void amd_iommu_release_device(struct device *dev)
{ … }
static struct iommu_group *amd_iommu_device_group(struct device *dev)
{ … }
static void cleanup_domain(struct protection_domain *domain)
{ … }
void protection_domain_free(struct protection_domain *domain)
{ … }
struct protection_domain *protection_domain_alloc(unsigned int type, int nid)
{ … }
static inline u64 dma_max_address(void)
{ … }
static bool amd_iommu_hd_support(struct amd_iommu *iommu)
{ … }
static struct iommu_domain *do_iommu_domain_alloc(unsigned int type,
struct device *dev, u32 flags)
{ … }
static struct iommu_domain *amd_iommu_domain_alloc(unsigned int type)
{ … }
static struct iommu_domain *
amd_iommu_domain_alloc_user(struct device *dev, u32 flags,
struct iommu_domain *parent,
const struct iommu_user_data *user_data)
{ … }
void amd_iommu_domain_free(struct iommu_domain *dom)
{ … }
static int blocked_domain_attach_device(struct iommu_domain *domain,
struct device *dev)
{ … }
static struct iommu_domain blocked_domain = …;
static int amd_iommu_attach_device(struct iommu_domain *dom,
struct device *dev)
{ … }
static int amd_iommu_iotlb_sync_map(struct iommu_domain *dom,
unsigned long iova, size_t size)
{ … }
static int amd_iommu_map_pages(struct iommu_domain *dom, unsigned long iova,
phys_addr_t paddr, size_t pgsize, size_t pgcount,
int iommu_prot, gfp_t gfp, size_t *mapped)
{ … }
static void amd_iommu_iotlb_gather_add_page(struct iommu_domain *domain,
struct iommu_iotlb_gather *gather,
unsigned long iova, size_t size)
{ … }
static size_t amd_iommu_unmap_pages(struct iommu_domain *dom, unsigned long iova,
size_t pgsize, size_t pgcount,
struct iommu_iotlb_gather *gather)
{ … }
static phys_addr_t amd_iommu_iova_to_phys(struct iommu_domain *dom,
dma_addr_t iova)
{ … }
static bool amd_iommu_capable(struct device *dev, enum iommu_cap cap)
{ … }
static int amd_iommu_set_dirty_tracking(struct iommu_domain *domain,
bool enable)
{ … }
static int amd_iommu_read_and_clear_dirty(struct iommu_domain *domain,
unsigned long iova, size_t size,
unsigned long flags,
struct iommu_dirty_bitmap *dirty)
{ … }
static void amd_iommu_get_resv_regions(struct device *dev,
struct list_head *head)
{ … }
static bool amd_iommu_is_attach_deferred(struct device *dev)
{ … }
static void amd_iommu_flush_iotlb_all(struct iommu_domain *domain)
{ … }
static void amd_iommu_iotlb_sync(struct iommu_domain *domain,
struct iommu_iotlb_gather *gather)
{ … }
static int amd_iommu_def_domain_type(struct device *dev)
{ … }
static bool amd_iommu_enforce_cache_coherency(struct iommu_domain *domain)
{ … }
static const struct iommu_dirty_ops amd_dirty_ops = …;
static int amd_iommu_dev_enable_feature(struct device *dev,
enum iommu_dev_features feat)
{ … }
static int amd_iommu_dev_disable_feature(struct device *dev,
enum iommu_dev_features feat)
{ … }
const struct iommu_ops amd_iommu_ops = …;
#ifdef CONFIG_IRQ_REMAP
static struct irq_chip amd_ir_chip;
static DEFINE_SPINLOCK(iommu_table_lock);
static void iommu_flush_irt_and_complete(struct amd_iommu *iommu, u16 devid)
{ … }
static void set_dte_irq_entry(struct amd_iommu *iommu, u16 devid,
struct irq_remap_table *table)
{ … }
static struct irq_remap_table *get_irq_table(struct amd_iommu *iommu, u16 devid)
{ … }
static struct irq_remap_table *__alloc_irq_table(void)
{ … }
static void set_remap_table_entry(struct amd_iommu *iommu, u16 devid,
struct irq_remap_table *table)
{ … }
static int set_remap_table_entry_alias(struct pci_dev *pdev, u16 alias,
void *data)
{ … }
static struct irq_remap_table *alloc_irq_table(struct amd_iommu *iommu,
u16 devid, struct pci_dev *pdev)
{ … }
static int alloc_irq_index(struct amd_iommu *iommu, u16 devid, int count,
bool align, struct pci_dev *pdev)
{ … }
static int __modify_irte_ga(struct amd_iommu *iommu, u16 devid, int index,
struct irte_ga *irte)
{ … }
static int modify_irte_ga(struct amd_iommu *iommu, u16 devid, int index,
struct irte_ga *irte)
{ … }
static int modify_irte(struct amd_iommu *iommu,
u16 devid, int index, union irte *irte)
{ … }
static void free_irte(struct amd_iommu *iommu, u16 devid, int index)
{ … }
static void irte_prepare(void *entry,
u32 delivery_mode, bool dest_mode,
u8 vector, u32 dest_apicid, int devid)
{ … }
static void irte_ga_prepare(void *entry,
u32 delivery_mode, bool dest_mode,
u8 vector, u32 dest_apicid, int devid)
{ … }
static void irte_activate(struct amd_iommu *iommu, void *entry, u16 devid, u16 index)
{ … }
static void irte_ga_activate(struct amd_iommu *iommu, void *entry, u16 devid, u16 index)
{ … }
static void irte_deactivate(struct amd_iommu *iommu, void *entry, u16 devid, u16 index)
{ … }
static void irte_ga_deactivate(struct amd_iommu *iommu, void *entry, u16 devid, u16 index)
{ … }
static void irte_set_affinity(struct amd_iommu *iommu, void *entry, u16 devid, u16 index,
u8 vector, u32 dest_apicid)
{ … }
static void irte_ga_set_affinity(struct amd_iommu *iommu, void *entry, u16 devid, u16 index,
u8 vector, u32 dest_apicid)
{ … }
#define IRTE_ALLOCATED …
static void irte_set_allocated(struct irq_remap_table *table, int index)
{ … }
static void irte_ga_set_allocated(struct irq_remap_table *table, int index)
{ … }
static bool irte_is_allocated(struct irq_remap_table *table, int index)
{ … }
static bool irte_ga_is_allocated(struct irq_remap_table *table, int index)
{ … }
static void irte_clear_allocated(struct irq_remap_table *table, int index)
{ … }
static void irte_ga_clear_allocated(struct irq_remap_table *table, int index)
{ … }
static int get_devid(struct irq_alloc_info *info)
{ … }
struct irq_remap_ops amd_iommu_irq_ops = …;
static void fill_msi_msg(struct msi_msg *msg, u32 index)
{ … }
static void irq_remapping_prepare_irte(struct amd_ir_data *data,
struct irq_cfg *irq_cfg,
struct irq_alloc_info *info,
int devid, int index, int sub_handle)
{ … }
struct amd_irte_ops irte_32_ops = …;
struct amd_irte_ops irte_128_ops = …;
static int irq_remapping_alloc(struct irq_domain *domain, unsigned int virq,
unsigned int nr_irqs, void *arg)
{ … }
static void irq_remapping_free(struct irq_domain *domain, unsigned int virq,
unsigned int nr_irqs)
{ … }
static void amd_ir_update_irte(struct irq_data *irqd, struct amd_iommu *iommu,
struct amd_ir_data *ir_data,
struct irq_2_irte *irte_info,
struct irq_cfg *cfg);
static int irq_remapping_activate(struct irq_domain *domain,
struct irq_data *irq_data, bool reserve)
{ … }
static void irq_remapping_deactivate(struct irq_domain *domain,
struct irq_data *irq_data)
{ … }
static int irq_remapping_select(struct irq_domain *d, struct irq_fwspec *fwspec,
enum irq_domain_bus_token bus_token)
{ … }
static const struct irq_domain_ops amd_ir_domain_ops = …;
int amd_iommu_activate_guest_mode(void *data)
{ … }
EXPORT_SYMBOL(…);
int amd_iommu_deactivate_guest_mode(void *data)
{ … }
EXPORT_SYMBOL(…);
static int amd_ir_set_vcpu_affinity(struct irq_data *data, void *vcpu_info)
{ … }
static void amd_ir_update_irte(struct irq_data *irqd, struct amd_iommu *iommu,
struct amd_ir_data *ir_data,
struct irq_2_irte *irte_info,
struct irq_cfg *cfg)
{ … }
static int amd_ir_set_affinity(struct irq_data *data,
const struct cpumask *mask, bool force)
{ … }
static void ir_compose_msi_msg(struct irq_data *irq_data, struct msi_msg *msg)
{ … }
static struct irq_chip amd_ir_chip = …;
static const struct msi_parent_ops amdvi_msi_parent_ops = …;
int amd_iommu_create_irq_domain(struct amd_iommu *iommu)
{ … }
int amd_iommu_update_ga(int cpu, bool is_run, void *data)
{ … }
EXPORT_SYMBOL(…);
#endif