/* * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along with * this program; If not, see . */ #ifndef __ARCH_X86_IOMMU_H__ #define __ARCH_X86_IOMMU_H__ #include #include #include #include #include #include #include #define DEFAULT_DOMAIN_ADDRESS_WIDTH 48 struct g2m_ioport { struct list_head list; unsigned int gport; unsigned int mport; unsigned int np; }; #define IOMMU_PAGE_SHIFT 12 #define IOMMU_PAGE_SIZE (1 << IOMMU_PAGE_SHIFT) #define IOMMU_PAGE_MASK (~(IOMMU_PAGE_SIZE - 1)) typedef uint64_t daddr_t; #define __dfn_to_daddr(dfn) ((daddr_t)(dfn) << IOMMU_PAGE_SHIFT) #define __daddr_to_dfn(daddr) ((daddr) >> IOMMU_PAGE_SHIFT) #define dfn_to_daddr(dfn) __dfn_to_daddr(dfn_x(dfn)) #define daddr_to_dfn(daddr) _dfn(__daddr_to_dfn(daddr)) struct arch_iommu { spinlock_t mapping_lock; /* io page table lock */ struct { struct page_list_head list; spinlock_t lock; } pgtables; struct list_head identity_maps; union { /* Intel VT-d */ struct { uint64_t pgd_maddr; /* io page directory machine address */ unsigned int agaw; /* adjusted guest address width, 0 is level 2 30-bit */ unsigned long *iommu_bitmap; /* bitmap of iommu(s) that the domain uses */ } vtd; /* AMD IOMMU */ struct { unsigned int paging_mode; struct page_info *root_table; struct guest_iommu *g_iommu; } amd; }; }; extern struct iommu_ops iommu_ops; #ifdef NDEBUG # include # define iommu_call(ops, fn, args...) ({ \ (void)(ops); \ alternative_call(iommu_ops.fn, ## args); \ }) # define iommu_vcall(ops, fn, args...) ({ \ (void)(ops); \ alternative_vcall(iommu_ops.fn, ## args); \ }) #endif static inline const struct iommu_ops *iommu_get_ops(void) { BUG_ON(!iommu_ops.init); return &iommu_ops; } struct iommu_init_ops { const struct iommu_ops *ops; int (*setup)(void); bool (*supports_x2apic)(void); }; extern const struct iommu_init_ops *iommu_init_ops; void iommu_update_ire_from_apic(unsigned int apic, unsigned int reg, unsigned int value); unsigned int iommu_read_apic_from_ire(unsigned int apic, unsigned int reg); int iommu_setup_hpet_msi(struct msi_desc *); static inline int iommu_adjust_irq_affinities(void) { return iommu_ops.adjust_irq_affinities ? iommu_ops.adjust_irq_affinities() : 0; } static inline bool iommu_supports_x2apic(void) { return iommu_init_ops && iommu_init_ops->supports_x2apic ? iommu_init_ops->supports_x2apic() : false; } int iommu_enable_x2apic(void); static inline void iommu_disable_x2apic(void) { if ( x2apic_enabled && iommu_ops.disable_x2apic ) iommu_ops.disable_x2apic(); } int iommu_identity_mapping(struct domain *d, p2m_access_t p2ma, paddr_t base, paddr_t end, unsigned int flag); void iommu_identity_map_teardown(struct domain *d); extern bool untrusted_msi; int pi_update_irte(const struct pi_desc *pi_desc, const struct pirq *pirq, const uint8_t gvec); #define iommu_sync_cache(addr, size) ({ \ const struct iommu_ops *ops = iommu_get_ops(); \ \ if ( ops->sync_cache ) \ iommu_vcall(ops, sync_cache, addr, size); \ }) int __must_check iommu_free_pgtables(struct domain *d); struct page_info *__must_check iommu_alloc_pgtable(struct domain *d); #endif /* !__ARCH_X86_IOMMU_H__ */ /* * Local variables: * mode: C * c-file-style: "BSD" * c-basic-offset: 4 * indent-tabs-mode: nil * End: */