mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-17 02:15:57 +00:00
Merge ra.kernel.org:/pub/scm/linux/kernel/git/davem/net
Version bump conflict in batman-adv, take what's in net-next. iavf conflict, adjustment of netdev_ops in net-next conflicting with poll controller method removal in net. Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
commit
a06ee256e5
@ -33,4 +33,3 @@ Video Function Calls
|
|||||||
video-clear-buffer
|
video-clear-buffer
|
||||||
video-set-streamtype
|
video-set-streamtype
|
||||||
video-set-format
|
video-set-format
|
||||||
video-set-attributes
|
|
||||||
|
@ -4510,7 +4510,8 @@ Do not enable KVM_FEATURE_PV_UNHALT if you disable HLT exits.
|
|||||||
Architectures: s390
|
Architectures: s390
|
||||||
Parameters: none
|
Parameters: none
|
||||||
Returns: 0 on success, -EINVAL if hpage module parameter was not set
|
Returns: 0 on success, -EINVAL if hpage module parameter was not set
|
||||||
or cmma is enabled
|
or cmma is enabled, or the VM has the KVM_VM_S390_UCONTROL
|
||||||
|
flag set
|
||||||
|
|
||||||
With this capability the KVM support for memory backing with 1m pages
|
With this capability the KVM support for memory backing with 1m pages
|
||||||
through hugetlbfs can be enabled for a VM. After the capability is
|
through hugetlbfs can be enabled for a VM. After the capability is
|
||||||
@ -4521,6 +4522,15 @@ hpage module parameter is not set to 1, -EINVAL is returned.
|
|||||||
While it is generally possible to create a huge page backed VM without
|
While it is generally possible to create a huge page backed VM without
|
||||||
this capability, the VM will not be able to run.
|
this capability, the VM will not be able to run.
|
||||||
|
|
||||||
|
7.14 KVM_CAP_MSR_PLATFORM_INFO
|
||||||
|
|
||||||
|
Architectures: x86
|
||||||
|
Parameters: args[0] whether feature should be enabled or not
|
||||||
|
|
||||||
|
With this capability, a guest may read the MSR_PLATFORM_INFO MSR. Otherwise,
|
||||||
|
a #GP would be raised when the guest tries to access. Currently, this
|
||||||
|
capability does not enable write permissions of this MSR for the guest.
|
||||||
|
|
||||||
8. Other capabilities.
|
8. Other capabilities.
|
||||||
----------------------
|
----------------------
|
||||||
|
|
||||||
|
14
MAINTAINERS
14
MAINTAINERS
@ -12269,6 +12269,7 @@ F: Documentation/networking/rds.txt
|
|||||||
|
|
||||||
RDT - RESOURCE ALLOCATION
|
RDT - RESOURCE ALLOCATION
|
||||||
M: Fenghua Yu <fenghua.yu@intel.com>
|
M: Fenghua Yu <fenghua.yu@intel.com>
|
||||||
|
M: Reinette Chatre <reinette.chatre@intel.com>
|
||||||
L: linux-kernel@vger.kernel.org
|
L: linux-kernel@vger.kernel.org
|
||||||
S: Supported
|
S: Supported
|
||||||
F: arch/x86/kernel/cpu/intel_rdt*
|
F: arch/x86/kernel/cpu/intel_rdt*
|
||||||
@ -13458,9 +13459,8 @@ F: drivers/i2c/busses/i2c-synquacer.c
|
|||||||
F: Documentation/devicetree/bindings/i2c/i2c-synquacer.txt
|
F: Documentation/devicetree/bindings/i2c/i2c-synquacer.txt
|
||||||
|
|
||||||
SOCIONEXT UNIPHIER SOUND DRIVER
|
SOCIONEXT UNIPHIER SOUND DRIVER
|
||||||
M: Katsuhiro Suzuki <suzuki.katsuhiro@socionext.com>
|
|
||||||
L: alsa-devel@alsa-project.org (moderated for non-subscribers)
|
L: alsa-devel@alsa-project.org (moderated for non-subscribers)
|
||||||
S: Maintained
|
S: Orphan
|
||||||
F: sound/soc/uniphier/
|
F: sound/soc/uniphier/
|
||||||
|
|
||||||
SOEKRIS NET48XX LED SUPPORT
|
SOEKRIS NET48XX LED SUPPORT
|
||||||
@ -15922,6 +15922,7 @@ F: net/x25/
|
|||||||
X86 ARCHITECTURE (32-BIT AND 64-BIT)
|
X86 ARCHITECTURE (32-BIT AND 64-BIT)
|
||||||
M: Thomas Gleixner <tglx@linutronix.de>
|
M: Thomas Gleixner <tglx@linutronix.de>
|
||||||
M: Ingo Molnar <mingo@redhat.com>
|
M: Ingo Molnar <mingo@redhat.com>
|
||||||
|
M: Borislav Petkov <bp@alien8.de>
|
||||||
R: "H. Peter Anvin" <hpa@zytor.com>
|
R: "H. Peter Anvin" <hpa@zytor.com>
|
||||||
M: x86@kernel.org
|
M: x86@kernel.org
|
||||||
L: linux-kernel@vger.kernel.org
|
L: linux-kernel@vger.kernel.org
|
||||||
@ -15950,6 +15951,15 @@ M: Borislav Petkov <bp@alien8.de>
|
|||||||
S: Maintained
|
S: Maintained
|
||||||
F: arch/x86/kernel/cpu/microcode/*
|
F: arch/x86/kernel/cpu/microcode/*
|
||||||
|
|
||||||
|
X86 MM
|
||||||
|
M: Dave Hansen <dave.hansen@linux.intel.com>
|
||||||
|
M: Andy Lutomirski <luto@kernel.org>
|
||||||
|
M: Peter Zijlstra <peterz@infradead.org>
|
||||||
|
L: linux-kernel@vger.kernel.org
|
||||||
|
T: git git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip.git x86/mm
|
||||||
|
S: Maintained
|
||||||
|
F: arch/x86/mm/
|
||||||
|
|
||||||
X86 PLATFORM DRIVERS
|
X86 PLATFORM DRIVERS
|
||||||
M: Darren Hart <dvhart@infradead.org>
|
M: Darren Hart <dvhart@infradead.org>
|
||||||
M: Andy Shevchenko <andy@infradead.org>
|
M: Andy Shevchenko <andy@infradead.org>
|
||||||
|
2
Makefile
2
Makefile
@ -2,7 +2,7 @@
|
|||||||
VERSION = 4
|
VERSION = 4
|
||||||
PATCHLEVEL = 19
|
PATCHLEVEL = 19
|
||||||
SUBLEVEL = 0
|
SUBLEVEL = 0
|
||||||
EXTRAVERSION = -rc4
|
EXTRAVERSION = -rc5
|
||||||
NAME = Merciless Moray
|
NAME = Merciless Moray
|
||||||
|
|
||||||
# *DOCUMENTATION*
|
# *DOCUMENTATION*
|
||||||
|
@ -1051,7 +1051,6 @@ static inline void vmemmap_remove_mapping(unsigned long start,
|
|||||||
return hash__vmemmap_remove_mapping(start, page_size);
|
return hash__vmemmap_remove_mapping(start, page_size);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
struct page *realmode_pfn_to_page(unsigned long pfn);
|
|
||||||
|
|
||||||
static inline pte_t pmd_pte(pmd_t pmd)
|
static inline pte_t pmd_pte(pmd_t pmd)
|
||||||
{
|
{
|
||||||
|
@ -220,8 +220,6 @@ extern void iommu_del_device(struct device *dev);
|
|||||||
extern int __init tce_iommu_bus_notifier_init(void);
|
extern int __init tce_iommu_bus_notifier_init(void);
|
||||||
extern long iommu_tce_xchg(struct iommu_table *tbl, unsigned long entry,
|
extern long iommu_tce_xchg(struct iommu_table *tbl, unsigned long entry,
|
||||||
unsigned long *hpa, enum dma_data_direction *direction);
|
unsigned long *hpa, enum dma_data_direction *direction);
|
||||||
extern long iommu_tce_xchg_rm(struct iommu_table *tbl, unsigned long entry,
|
|
||||||
unsigned long *hpa, enum dma_data_direction *direction);
|
|
||||||
#else
|
#else
|
||||||
static inline void iommu_register_group(struct iommu_table_group *table_group,
|
static inline void iommu_register_group(struct iommu_table_group *table_group,
|
||||||
int pci_domain_number,
|
int pci_domain_number,
|
||||||
|
@ -38,6 +38,7 @@ extern long mm_iommu_ua_to_hpa(struct mm_iommu_table_group_mem_t *mem,
|
|||||||
unsigned long ua, unsigned int pageshift, unsigned long *hpa);
|
unsigned long ua, unsigned int pageshift, unsigned long *hpa);
|
||||||
extern long mm_iommu_ua_to_hpa_rm(struct mm_iommu_table_group_mem_t *mem,
|
extern long mm_iommu_ua_to_hpa_rm(struct mm_iommu_table_group_mem_t *mem,
|
||||||
unsigned long ua, unsigned int pageshift, unsigned long *hpa);
|
unsigned long ua, unsigned int pageshift, unsigned long *hpa);
|
||||||
|
extern void mm_iommu_ua_mark_dirty_rm(struct mm_struct *mm, unsigned long ua);
|
||||||
extern long mm_iommu_mapped_inc(struct mm_iommu_table_group_mem_t *mem);
|
extern long mm_iommu_mapped_inc(struct mm_iommu_table_group_mem_t *mem);
|
||||||
extern void mm_iommu_mapped_dec(struct mm_iommu_table_group_mem_t *mem);
|
extern void mm_iommu_mapped_dec(struct mm_iommu_table_group_mem_t *mem);
|
||||||
#endif
|
#endif
|
||||||
|
@ -1013,31 +1013,6 @@ long iommu_tce_xchg(struct iommu_table *tbl, unsigned long entry,
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(iommu_tce_xchg);
|
EXPORT_SYMBOL_GPL(iommu_tce_xchg);
|
||||||
|
|
||||||
#ifdef CONFIG_PPC_BOOK3S_64
|
|
||||||
long iommu_tce_xchg_rm(struct iommu_table *tbl, unsigned long entry,
|
|
||||||
unsigned long *hpa, enum dma_data_direction *direction)
|
|
||||||
{
|
|
||||||
long ret;
|
|
||||||
|
|
||||||
ret = tbl->it_ops->exchange_rm(tbl, entry, hpa, direction);
|
|
||||||
|
|
||||||
if (!ret && ((*direction == DMA_FROM_DEVICE) ||
|
|
||||||
(*direction == DMA_BIDIRECTIONAL))) {
|
|
||||||
struct page *pg = realmode_pfn_to_page(*hpa >> PAGE_SHIFT);
|
|
||||||
|
|
||||||
if (likely(pg)) {
|
|
||||||
SetPageDirty(pg);
|
|
||||||
} else {
|
|
||||||
tbl->it_ops->exchange_rm(tbl, entry, hpa, direction);
|
|
||||||
ret = -EFAULT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(iommu_tce_xchg_rm);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
int iommu_take_ownership(struct iommu_table *tbl)
|
int iommu_take_ownership(struct iommu_table *tbl)
|
||||||
{
|
{
|
||||||
unsigned long flags, i, sz = (tbl->it_size + 7) >> 3;
|
unsigned long flags, i, sz = (tbl->it_size + 7) >> 3;
|
||||||
|
@ -525,8 +525,8 @@ int kvmppc_book3s_radix_page_fault(struct kvm_run *run, struct kvm_vcpu *vcpu,
|
|||||||
unsigned long ea, unsigned long dsisr)
|
unsigned long ea, unsigned long dsisr)
|
||||||
{
|
{
|
||||||
struct kvm *kvm = vcpu->kvm;
|
struct kvm *kvm = vcpu->kvm;
|
||||||
unsigned long mmu_seq, pte_size;
|
unsigned long mmu_seq;
|
||||||
unsigned long gpa, gfn, hva, pfn;
|
unsigned long gpa, gfn, hva;
|
||||||
struct kvm_memory_slot *memslot;
|
struct kvm_memory_slot *memslot;
|
||||||
struct page *page = NULL;
|
struct page *page = NULL;
|
||||||
long ret;
|
long ret;
|
||||||
@ -623,9 +623,10 @@ int kvmppc_book3s_radix_page_fault(struct kvm_run *run, struct kvm_vcpu *vcpu,
|
|||||||
*/
|
*/
|
||||||
hva = gfn_to_hva_memslot(memslot, gfn);
|
hva = gfn_to_hva_memslot(memslot, gfn);
|
||||||
if (upgrade_p && __get_user_pages_fast(hva, 1, 1, &page) == 1) {
|
if (upgrade_p && __get_user_pages_fast(hva, 1, 1, &page) == 1) {
|
||||||
pfn = page_to_pfn(page);
|
|
||||||
upgrade_write = true;
|
upgrade_write = true;
|
||||||
} else {
|
} else {
|
||||||
|
unsigned long pfn;
|
||||||
|
|
||||||
/* Call KVM generic code to do the slow-path check */
|
/* Call KVM generic code to do the slow-path check */
|
||||||
pfn = __gfn_to_pfn_memslot(memslot, gfn, false, NULL,
|
pfn = __gfn_to_pfn_memslot(memslot, gfn, false, NULL,
|
||||||
writing, upgrade_p);
|
writing, upgrade_p);
|
||||||
@ -639,61 +640,43 @@ int kvmppc_book3s_radix_page_fault(struct kvm_run *run, struct kvm_vcpu *vcpu,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* See if we can insert a 1GB or 2MB large PTE here */
|
/*
|
||||||
level = 0;
|
* Read the PTE from the process' radix tree and use that
|
||||||
if (page && PageCompound(page)) {
|
* so we get the shift and attribute bits.
|
||||||
pte_size = PAGE_SIZE << compound_order(compound_head(page));
|
*/
|
||||||
if (pte_size >= PUD_SIZE &&
|
local_irq_disable();
|
||||||
(gpa & (PUD_SIZE - PAGE_SIZE)) ==
|
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
||||||
(hva & (PUD_SIZE - PAGE_SIZE))) {
|
pte = *ptep;
|
||||||
level = 2;
|
local_irq_enable();
|
||||||
pfn &= ~((PUD_SIZE >> PAGE_SHIFT) - 1);
|
|
||||||
} else if (pte_size >= PMD_SIZE &&
|
/* Get pte level from shift/size */
|
||||||
(gpa & (PMD_SIZE - PAGE_SIZE)) ==
|
if (shift == PUD_SHIFT &&
|
||||||
(hva & (PMD_SIZE - PAGE_SIZE))) {
|
(gpa & (PUD_SIZE - PAGE_SIZE)) ==
|
||||||
level = 1;
|
(hva & (PUD_SIZE - PAGE_SIZE))) {
|
||||||
pfn &= ~((PMD_SIZE >> PAGE_SHIFT) - 1);
|
level = 2;
|
||||||
|
} else if (shift == PMD_SHIFT &&
|
||||||
|
(gpa & (PMD_SIZE - PAGE_SIZE)) ==
|
||||||
|
(hva & (PMD_SIZE - PAGE_SIZE))) {
|
||||||
|
level = 1;
|
||||||
|
} else {
|
||||||
|
level = 0;
|
||||||
|
if (shift > PAGE_SHIFT) {
|
||||||
|
/*
|
||||||
|
* If the pte maps more than one page, bring over
|
||||||
|
* bits from the virtual address to get the real
|
||||||
|
* address of the specific single page we want.
|
||||||
|
*/
|
||||||
|
unsigned long rpnmask = (1ul << shift) - PAGE_SIZE;
|
||||||
|
pte = __pte(pte_val(pte) | (hva & rpnmask));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
pte = __pte(pte_val(pte) | _PAGE_EXEC | _PAGE_ACCESSED);
|
||||||
* Compute the PTE value that we need to insert.
|
if (writing || upgrade_write) {
|
||||||
*/
|
if (pte_val(pte) & _PAGE_WRITE)
|
||||||
if (page) {
|
pte = __pte(pte_val(pte) | _PAGE_DIRTY);
|
||||||
pgflags = _PAGE_READ | _PAGE_EXEC | _PAGE_PRESENT | _PAGE_PTE |
|
|
||||||
_PAGE_ACCESSED;
|
|
||||||
if (writing || upgrade_write)
|
|
||||||
pgflags |= _PAGE_WRITE | _PAGE_DIRTY;
|
|
||||||
pte = pfn_pte(pfn, __pgprot(pgflags));
|
|
||||||
} else {
|
} else {
|
||||||
/*
|
pte = __pte(pte_val(pte) & ~(_PAGE_WRITE | _PAGE_DIRTY));
|
||||||
* Read the PTE from the process' radix tree and use that
|
|
||||||
* so we get the attribute bits.
|
|
||||||
*/
|
|
||||||
local_irq_disable();
|
|
||||||
ptep = __find_linux_pte(vcpu->arch.pgdir, hva, NULL, &shift);
|
|
||||||
pte = *ptep;
|
|
||||||
local_irq_enable();
|
|
||||||
if (shift == PUD_SHIFT &&
|
|
||||||
(gpa & (PUD_SIZE - PAGE_SIZE)) ==
|
|
||||||
(hva & (PUD_SIZE - PAGE_SIZE))) {
|
|
||||||
level = 2;
|
|
||||||
} else if (shift == PMD_SHIFT &&
|
|
||||||
(gpa & (PMD_SIZE - PAGE_SIZE)) ==
|
|
||||||
(hva & (PMD_SIZE - PAGE_SIZE))) {
|
|
||||||
level = 1;
|
|
||||||
} else if (shift && shift != PAGE_SHIFT) {
|
|
||||||
/* Adjust PFN */
|
|
||||||
unsigned long mask = (1ul << shift) - PAGE_SIZE;
|
|
||||||
pte = __pte(pte_val(pte) | (hva & mask));
|
|
||||||
}
|
|
||||||
pte = __pte(pte_val(pte) | _PAGE_EXEC | _PAGE_ACCESSED);
|
|
||||||
if (writing || upgrade_write) {
|
|
||||||
if (pte_val(pte) & _PAGE_WRITE)
|
|
||||||
pte = __pte(pte_val(pte) | _PAGE_DIRTY);
|
|
||||||
} else {
|
|
||||||
pte = __pte(pte_val(pte) & ~(_PAGE_WRITE | _PAGE_DIRTY));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Allocate space in the tree and write the PTE */
|
/* Allocate space in the tree and write the PTE */
|
||||||
|
@ -187,12 +187,35 @@ long kvmppc_gpa_to_ua(struct kvm *kvm, unsigned long gpa,
|
|||||||
EXPORT_SYMBOL_GPL(kvmppc_gpa_to_ua);
|
EXPORT_SYMBOL_GPL(kvmppc_gpa_to_ua);
|
||||||
|
|
||||||
#ifdef CONFIG_KVM_BOOK3S_HV_POSSIBLE
|
#ifdef CONFIG_KVM_BOOK3S_HV_POSSIBLE
|
||||||
static void kvmppc_rm_clear_tce(struct iommu_table *tbl, unsigned long entry)
|
static long iommu_tce_xchg_rm(struct mm_struct *mm, struct iommu_table *tbl,
|
||||||
|
unsigned long entry, unsigned long *hpa,
|
||||||
|
enum dma_data_direction *direction)
|
||||||
|
{
|
||||||
|
long ret;
|
||||||
|
|
||||||
|
ret = tbl->it_ops->exchange_rm(tbl, entry, hpa, direction);
|
||||||
|
|
||||||
|
if (!ret && ((*direction == DMA_FROM_DEVICE) ||
|
||||||
|
(*direction == DMA_BIDIRECTIONAL))) {
|
||||||
|
__be64 *pua = IOMMU_TABLE_USERSPACE_ENTRY_RM(tbl, entry);
|
||||||
|
/*
|
||||||
|
* kvmppc_rm_tce_iommu_do_map() updates the UA cache after
|
||||||
|
* calling this so we still get here a valid UA.
|
||||||
|
*/
|
||||||
|
if (pua && *pua)
|
||||||
|
mm_iommu_ua_mark_dirty_rm(mm, be64_to_cpu(*pua));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void kvmppc_rm_clear_tce(struct kvm *kvm, struct iommu_table *tbl,
|
||||||
|
unsigned long entry)
|
||||||
{
|
{
|
||||||
unsigned long hpa = 0;
|
unsigned long hpa = 0;
|
||||||
enum dma_data_direction dir = DMA_NONE;
|
enum dma_data_direction dir = DMA_NONE;
|
||||||
|
|
||||||
iommu_tce_xchg_rm(tbl, entry, &hpa, &dir);
|
iommu_tce_xchg_rm(kvm->mm, tbl, entry, &hpa, &dir);
|
||||||
}
|
}
|
||||||
|
|
||||||
static long kvmppc_rm_tce_iommu_mapped_dec(struct kvm *kvm,
|
static long kvmppc_rm_tce_iommu_mapped_dec(struct kvm *kvm,
|
||||||
@ -224,7 +247,7 @@ static long kvmppc_rm_tce_iommu_do_unmap(struct kvm *kvm,
|
|||||||
unsigned long hpa = 0;
|
unsigned long hpa = 0;
|
||||||
long ret;
|
long ret;
|
||||||
|
|
||||||
if (iommu_tce_xchg_rm(tbl, entry, &hpa, &dir))
|
if (iommu_tce_xchg_rm(kvm->mm, tbl, entry, &hpa, &dir))
|
||||||
/*
|
/*
|
||||||
* real mode xchg can fail if struct page crosses
|
* real mode xchg can fail if struct page crosses
|
||||||
* a page boundary
|
* a page boundary
|
||||||
@ -236,7 +259,7 @@ static long kvmppc_rm_tce_iommu_do_unmap(struct kvm *kvm,
|
|||||||
|
|
||||||
ret = kvmppc_rm_tce_iommu_mapped_dec(kvm, tbl, entry);
|
ret = kvmppc_rm_tce_iommu_mapped_dec(kvm, tbl, entry);
|
||||||
if (ret)
|
if (ret)
|
||||||
iommu_tce_xchg_rm(tbl, entry, &hpa, &dir);
|
iommu_tce_xchg_rm(kvm->mm, tbl, entry, &hpa, &dir);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -282,7 +305,7 @@ static long kvmppc_rm_tce_iommu_do_map(struct kvm *kvm, struct iommu_table *tbl,
|
|||||||
if (WARN_ON_ONCE_RM(mm_iommu_mapped_inc(mem)))
|
if (WARN_ON_ONCE_RM(mm_iommu_mapped_inc(mem)))
|
||||||
return H_CLOSED;
|
return H_CLOSED;
|
||||||
|
|
||||||
ret = iommu_tce_xchg_rm(tbl, entry, &hpa, &dir);
|
ret = iommu_tce_xchg_rm(kvm->mm, tbl, entry, &hpa, &dir);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
mm_iommu_mapped_dec(mem);
|
mm_iommu_mapped_dec(mem);
|
||||||
/*
|
/*
|
||||||
@ -371,7 +394,7 @@ long kvmppc_rm_h_put_tce(struct kvm_vcpu *vcpu, unsigned long liobn,
|
|||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
WARN_ON_ONCE_RM(1);
|
WARN_ON_ONCE_RM(1);
|
||||||
kvmppc_rm_clear_tce(stit->tbl, entry);
|
kvmppc_rm_clear_tce(vcpu->kvm, stit->tbl, entry);
|
||||||
}
|
}
|
||||||
|
|
||||||
kvmppc_tce_put(stt, entry, tce);
|
kvmppc_tce_put(stt, entry, tce);
|
||||||
@ -520,7 +543,7 @@ long kvmppc_rm_h_put_tce_indirect(struct kvm_vcpu *vcpu,
|
|||||||
goto unlock_exit;
|
goto unlock_exit;
|
||||||
|
|
||||||
WARN_ON_ONCE_RM(1);
|
WARN_ON_ONCE_RM(1);
|
||||||
kvmppc_rm_clear_tce(stit->tbl, entry);
|
kvmppc_rm_clear_tce(vcpu->kvm, stit->tbl, entry);
|
||||||
}
|
}
|
||||||
|
|
||||||
kvmppc_tce_put(stt, entry + i, tce);
|
kvmppc_tce_put(stt, entry + i, tce);
|
||||||
@ -571,7 +594,7 @@ long kvmppc_rm_h_stuff_tce(struct kvm_vcpu *vcpu,
|
|||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
WARN_ON_ONCE_RM(1);
|
WARN_ON_ONCE_RM(1);
|
||||||
kvmppc_rm_clear_tce(stit->tbl, entry);
|
kvmppc_rm_clear_tce(vcpu->kvm, stit->tbl, entry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -308,55 +308,6 @@ void register_page_bootmem_memmap(unsigned long section_nr,
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* We do not have access to the sparsemem vmemmap, so we fallback to
|
|
||||||
* walking the list of sparsemem blocks which we already maintain for
|
|
||||||
* the sake of crashdump. In the long run, we might want to maintain
|
|
||||||
* a tree if performance of that linear walk becomes a problem.
|
|
||||||
*
|
|
||||||
* realmode_pfn_to_page functions can fail due to:
|
|
||||||
* 1) As real sparsemem blocks do not lay in RAM continously (they
|
|
||||||
* are in virtual address space which is not available in the real mode),
|
|
||||||
* the requested page struct can be split between blocks so get_page/put_page
|
|
||||||
* may fail.
|
|
||||||
* 2) When huge pages are used, the get_page/put_page API will fail
|
|
||||||
* in real mode as the linked addresses in the page struct are virtual
|
|
||||||
* too.
|
|
||||||
*/
|
|
||||||
struct page *realmode_pfn_to_page(unsigned long pfn)
|
|
||||||
{
|
|
||||||
struct vmemmap_backing *vmem_back;
|
|
||||||
struct page *page;
|
|
||||||
unsigned long page_size = 1 << mmu_psize_defs[mmu_vmemmap_psize].shift;
|
|
||||||
unsigned long pg_va = (unsigned long) pfn_to_page(pfn);
|
|
||||||
|
|
||||||
for (vmem_back = vmemmap_list; vmem_back; vmem_back = vmem_back->list) {
|
|
||||||
if (pg_va < vmem_back->virt_addr)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
/* After vmemmap_list entry free is possible, need check all */
|
|
||||||
if ((pg_va + sizeof(struct page)) <=
|
|
||||||
(vmem_back->virt_addr + page_size)) {
|
|
||||||
page = (struct page *) (vmem_back->phys + pg_va -
|
|
||||||
vmem_back->virt_addr);
|
|
||||||
return page;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Probably that page struct is split between real pages */
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(realmode_pfn_to_page);
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
struct page *realmode_pfn_to_page(unsigned long pfn)
|
|
||||||
{
|
|
||||||
struct page *page = pfn_to_page(pfn);
|
|
||||||
return page;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(realmode_pfn_to_page);
|
|
||||||
|
|
||||||
#endif /* CONFIG_SPARSEMEM_VMEMMAP */
|
#endif /* CONFIG_SPARSEMEM_VMEMMAP */
|
||||||
|
|
||||||
#ifdef CONFIG_PPC_BOOK3S_64
|
#ifdef CONFIG_PPC_BOOK3S_64
|
||||||
|
@ -18,11 +18,15 @@
|
|||||||
#include <linux/migrate.h>
|
#include <linux/migrate.h>
|
||||||
#include <linux/hugetlb.h>
|
#include <linux/hugetlb.h>
|
||||||
#include <linux/swap.h>
|
#include <linux/swap.h>
|
||||||
|
#include <linux/sizes.h>
|
||||||
#include <asm/mmu_context.h>
|
#include <asm/mmu_context.h>
|
||||||
#include <asm/pte-walk.h>
|
#include <asm/pte-walk.h>
|
||||||
|
|
||||||
static DEFINE_MUTEX(mem_list_mutex);
|
static DEFINE_MUTEX(mem_list_mutex);
|
||||||
|
|
||||||
|
#define MM_IOMMU_TABLE_GROUP_PAGE_DIRTY 0x1
|
||||||
|
#define MM_IOMMU_TABLE_GROUP_PAGE_MASK ~(SZ_4K - 1)
|
||||||
|
|
||||||
struct mm_iommu_table_group_mem_t {
|
struct mm_iommu_table_group_mem_t {
|
||||||
struct list_head next;
|
struct list_head next;
|
||||||
struct rcu_head rcu;
|
struct rcu_head rcu;
|
||||||
@ -263,6 +267,9 @@ static void mm_iommu_unpin(struct mm_iommu_table_group_mem_t *mem)
|
|||||||
if (!page)
|
if (!page)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (mem->hpas[i] & MM_IOMMU_TABLE_GROUP_PAGE_DIRTY)
|
||||||
|
SetPageDirty(page);
|
||||||
|
|
||||||
put_page(page);
|
put_page(page);
|
||||||
mem->hpas[i] = 0;
|
mem->hpas[i] = 0;
|
||||||
}
|
}
|
||||||
@ -360,7 +367,6 @@ struct mm_iommu_table_group_mem_t *mm_iommu_lookup_rm(struct mm_struct *mm,
|
|||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(mm_iommu_lookup_rm);
|
|
||||||
|
|
||||||
struct mm_iommu_table_group_mem_t *mm_iommu_find(struct mm_struct *mm,
|
struct mm_iommu_table_group_mem_t *mm_iommu_find(struct mm_struct *mm,
|
||||||
unsigned long ua, unsigned long entries)
|
unsigned long ua, unsigned long entries)
|
||||||
@ -390,7 +396,7 @@ long mm_iommu_ua_to_hpa(struct mm_iommu_table_group_mem_t *mem,
|
|||||||
if (pageshift > mem->pageshift)
|
if (pageshift > mem->pageshift)
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
*hpa = *va | (ua & ~PAGE_MASK);
|
*hpa = (*va & MM_IOMMU_TABLE_GROUP_PAGE_MASK) | (ua & ~PAGE_MASK);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -413,11 +419,31 @@ long mm_iommu_ua_to_hpa_rm(struct mm_iommu_table_group_mem_t *mem,
|
|||||||
if (!pa)
|
if (!pa)
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
||||||
*hpa = *pa | (ua & ~PAGE_MASK);
|
*hpa = (*pa & MM_IOMMU_TABLE_GROUP_PAGE_MASK) | (ua & ~PAGE_MASK);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(mm_iommu_ua_to_hpa_rm);
|
|
||||||
|
extern void mm_iommu_ua_mark_dirty_rm(struct mm_struct *mm, unsigned long ua)
|
||||||
|
{
|
||||||
|
struct mm_iommu_table_group_mem_t *mem;
|
||||||
|
long entry;
|
||||||
|
void *va;
|
||||||
|
unsigned long *pa;
|
||||||
|
|
||||||
|
mem = mm_iommu_lookup_rm(mm, ua, PAGE_SIZE);
|
||||||
|
if (!mem)
|
||||||
|
return;
|
||||||
|
|
||||||
|
entry = (ua - mem->ua) >> PAGE_SHIFT;
|
||||||
|
va = &mem->hpas[entry];
|
||||||
|
|
||||||
|
pa = (void *) vmalloc_to_phys(va);
|
||||||
|
if (!pa)
|
||||||
|
return;
|
||||||
|
|
||||||
|
*pa |= MM_IOMMU_TABLE_GROUP_PAGE_DIRTY;
|
||||||
|
}
|
||||||
|
|
||||||
long mm_iommu_mapped_inc(struct mm_iommu_table_group_mem_t *mem)
|
long mm_iommu_mapped_inc(struct mm_iommu_table_group_mem_t *mem)
|
||||||
{
|
{
|
||||||
|
@ -481,7 +481,7 @@ int kvm_vm_ioctl_check_extension(struct kvm *kvm, long ext)
|
|||||||
break;
|
break;
|
||||||
case KVM_CAP_S390_HPAGE_1M:
|
case KVM_CAP_S390_HPAGE_1M:
|
||||||
r = 0;
|
r = 0;
|
||||||
if (hpage)
|
if (hpage && !kvm_is_ucontrol(kvm))
|
||||||
r = 1;
|
r = 1;
|
||||||
break;
|
break;
|
||||||
case KVM_CAP_S390_MEM_OP:
|
case KVM_CAP_S390_MEM_OP:
|
||||||
@ -691,7 +691,7 @@ static int kvm_vm_ioctl_enable_cap(struct kvm *kvm, struct kvm_enable_cap *cap)
|
|||||||
mutex_lock(&kvm->lock);
|
mutex_lock(&kvm->lock);
|
||||||
if (kvm->created_vcpus)
|
if (kvm->created_vcpus)
|
||||||
r = -EBUSY;
|
r = -EBUSY;
|
||||||
else if (!hpage || kvm->arch.use_cmma)
|
else if (!hpage || kvm->arch.use_cmma || kvm_is_ucontrol(kvm))
|
||||||
r = -EINVAL;
|
r = -EINVAL;
|
||||||
else {
|
else {
|
||||||
r = 0;
|
r = 0;
|
||||||
|
@ -708,11 +708,13 @@ void gmap_discard(struct gmap *gmap, unsigned long from, unsigned long to)
|
|||||||
vmaddr |= gaddr & ~PMD_MASK;
|
vmaddr |= gaddr & ~PMD_MASK;
|
||||||
/* Find vma in the parent mm */
|
/* Find vma in the parent mm */
|
||||||
vma = find_vma(gmap->mm, vmaddr);
|
vma = find_vma(gmap->mm, vmaddr);
|
||||||
|
if (!vma)
|
||||||
|
continue;
|
||||||
/*
|
/*
|
||||||
* We do not discard pages that are backed by
|
* We do not discard pages that are backed by
|
||||||
* hugetlbfs, so we don't have to refault them.
|
* hugetlbfs, so we don't have to refault them.
|
||||||
*/
|
*/
|
||||||
if (vma && is_vm_hugetlb_page(vma))
|
if (is_vm_hugetlb_page(vma))
|
||||||
continue;
|
continue;
|
||||||
size = min(to - gaddr, PMD_SIZE - (gaddr & ~PMD_MASK));
|
size = min(to - gaddr, PMD_SIZE - (gaddr & ~PMD_MASK));
|
||||||
zap_page_range(vma, vmaddr, size);
|
zap_page_range(vma, vmaddr, size);
|
||||||
|
@ -379,7 +379,6 @@ static int __init crypto_aegis128_aesni_module_init(void)
|
|||||||
{
|
{
|
||||||
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
||||||
!boot_cpu_has(X86_FEATURE_AES) ||
|
!boot_cpu_has(X86_FEATURE_AES) ||
|
||||||
!boot_cpu_has(X86_FEATURE_OSXSAVE) ||
|
|
||||||
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
@ -379,7 +379,6 @@ static int __init crypto_aegis128l_aesni_module_init(void)
|
|||||||
{
|
{
|
||||||
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
||||||
!boot_cpu_has(X86_FEATURE_AES) ||
|
!boot_cpu_has(X86_FEATURE_AES) ||
|
||||||
!boot_cpu_has(X86_FEATURE_OSXSAVE) ||
|
|
||||||
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
@ -379,7 +379,6 @@ static int __init crypto_aegis256_aesni_module_init(void)
|
|||||||
{
|
{
|
||||||
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
||||||
!boot_cpu_has(X86_FEATURE_AES) ||
|
!boot_cpu_has(X86_FEATURE_AES) ||
|
||||||
!boot_cpu_has(X86_FEATURE_OSXSAVE) ||
|
|
||||||
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
@ -40,7 +40,6 @@ MORUS1280_DECLARE_ALGS(sse2, "morus1280-sse2", 350);
|
|||||||
static int __init crypto_morus1280_sse2_module_init(void)
|
static int __init crypto_morus1280_sse2_module_init(void)
|
||||||
{
|
{
|
||||||
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
||||||
!boot_cpu_has(X86_FEATURE_OSXSAVE) ||
|
|
||||||
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
@ -40,7 +40,6 @@ MORUS640_DECLARE_ALGS(sse2, "morus640-sse2", 400);
|
|||||||
static int __init crypto_morus640_sse2_module_init(void)
|
static int __init crypto_morus640_sse2_module_init(void)
|
||||||
{
|
{
|
||||||
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
if (!boot_cpu_has(X86_FEATURE_XMM2) ||
|
||||||
!boot_cpu_has(X86_FEATURE_OSXSAVE) ||
|
|
||||||
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
!cpu_has_xfeatures(XFEATURE_MASK_SSE, NULL))
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
@ -95,8 +95,8 @@ static void hv_apic_eoi_write(u32 reg, u32 val)
|
|||||||
*/
|
*/
|
||||||
static bool __send_ipi_mask_ex(const struct cpumask *mask, int vector)
|
static bool __send_ipi_mask_ex(const struct cpumask *mask, int vector)
|
||||||
{
|
{
|
||||||
struct ipi_arg_ex **arg;
|
struct hv_send_ipi_ex **arg;
|
||||||
struct ipi_arg_ex *ipi_arg;
|
struct hv_send_ipi_ex *ipi_arg;
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int nr_bank = 0;
|
int nr_bank = 0;
|
||||||
int ret = 1;
|
int ret = 1;
|
||||||
@ -105,7 +105,7 @@ static bool __send_ipi_mask_ex(const struct cpumask *mask, int vector)
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
local_irq_save(flags);
|
local_irq_save(flags);
|
||||||
arg = (struct ipi_arg_ex **)this_cpu_ptr(hyperv_pcpu_input_arg);
|
arg = (struct hv_send_ipi_ex **)this_cpu_ptr(hyperv_pcpu_input_arg);
|
||||||
|
|
||||||
ipi_arg = *arg;
|
ipi_arg = *arg;
|
||||||
if (unlikely(!ipi_arg))
|
if (unlikely(!ipi_arg))
|
||||||
@ -135,7 +135,7 @@ ipi_mask_ex_done:
|
|||||||
static bool __send_ipi_mask(const struct cpumask *mask, int vector)
|
static bool __send_ipi_mask(const struct cpumask *mask, int vector)
|
||||||
{
|
{
|
||||||
int cur_cpu, vcpu;
|
int cur_cpu, vcpu;
|
||||||
struct ipi_arg_non_ex ipi_arg;
|
struct hv_send_ipi ipi_arg;
|
||||||
int ret = 1;
|
int ret = 1;
|
||||||
|
|
||||||
trace_hyperv_send_ipi_mask(mask, vector);
|
trace_hyperv_send_ipi_mask(mask, vector);
|
||||||
|
@ -14,6 +14,16 @@
|
|||||||
#ifndef _ASM_X86_FIXMAP_H
|
#ifndef _ASM_X86_FIXMAP_H
|
||||||
#define _ASM_X86_FIXMAP_H
|
#define _ASM_X86_FIXMAP_H
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Exposed to assembly code for setting up initial page tables. Cannot be
|
||||||
|
* calculated in assembly code (fixmap entries are an enum), but is sanity
|
||||||
|
* checked in the actual fixmap C code to make sure that the fixmap is
|
||||||
|
* covered fully.
|
||||||
|
*/
|
||||||
|
#define FIXMAP_PMD_NUM 2
|
||||||
|
/* fixmap starts downwards from the 507th entry in level2_fixmap_pgt */
|
||||||
|
#define FIXMAP_PMD_TOP 507
|
||||||
|
|
||||||
#ifndef __ASSEMBLY__
|
#ifndef __ASSEMBLY__
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <asm/acpi.h>
|
#include <asm/acpi.h>
|
||||||
|
@ -726,19 +726,21 @@ struct hv_enlightened_vmcs {
|
|||||||
#define HV_STIMER_AUTOENABLE (1ULL << 3)
|
#define HV_STIMER_AUTOENABLE (1ULL << 3)
|
||||||
#define HV_STIMER_SINT(config) (__u8)(((config) >> 16) & 0x0F)
|
#define HV_STIMER_SINT(config) (__u8)(((config) >> 16) & 0x0F)
|
||||||
|
|
||||||
struct ipi_arg_non_ex {
|
|
||||||
u32 vector;
|
|
||||||
u32 reserved;
|
|
||||||
u64 cpu_mask;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct hv_vpset {
|
struct hv_vpset {
|
||||||
u64 format;
|
u64 format;
|
||||||
u64 valid_bank_mask;
|
u64 valid_bank_mask;
|
||||||
u64 bank_contents[];
|
u64 bank_contents[];
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ipi_arg_ex {
|
/* HvCallSendSyntheticClusterIpi hypercall */
|
||||||
|
struct hv_send_ipi {
|
||||||
|
u32 vector;
|
||||||
|
u32 reserved;
|
||||||
|
u64 cpu_mask;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* HvCallSendSyntheticClusterIpiEx hypercall */
|
||||||
|
struct hv_send_ipi_ex {
|
||||||
u32 vector;
|
u32 vector;
|
||||||
u32 reserved;
|
u32 reserved;
|
||||||
struct hv_vpset vp_set;
|
struct hv_vpset vp_set;
|
||||||
|
@ -869,6 +869,8 @@ struct kvm_arch {
|
|||||||
|
|
||||||
bool x2apic_format;
|
bool x2apic_format;
|
||||||
bool x2apic_broadcast_quirk_disabled;
|
bool x2apic_broadcast_quirk_disabled;
|
||||||
|
|
||||||
|
bool guest_can_read_msr_platform_info;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct kvm_vm_stat {
|
struct kvm_vm_stat {
|
||||||
@ -1022,6 +1024,7 @@ struct kvm_x86_ops {
|
|||||||
void (*refresh_apicv_exec_ctrl)(struct kvm_vcpu *vcpu);
|
void (*refresh_apicv_exec_ctrl)(struct kvm_vcpu *vcpu);
|
||||||
void (*hwapic_irr_update)(struct kvm_vcpu *vcpu, int max_irr);
|
void (*hwapic_irr_update)(struct kvm_vcpu *vcpu, int max_irr);
|
||||||
void (*hwapic_isr_update)(struct kvm_vcpu *vcpu, int isr);
|
void (*hwapic_isr_update)(struct kvm_vcpu *vcpu, int isr);
|
||||||
|
bool (*guest_apic_has_interrupt)(struct kvm_vcpu *vcpu);
|
||||||
void (*load_eoi_exitmap)(struct kvm_vcpu *vcpu, u64 *eoi_exit_bitmap);
|
void (*load_eoi_exitmap)(struct kvm_vcpu *vcpu, u64 *eoi_exit_bitmap);
|
||||||
void (*set_virtual_apic_mode)(struct kvm_vcpu *vcpu);
|
void (*set_virtual_apic_mode)(struct kvm_vcpu *vcpu);
|
||||||
void (*set_apic_access_page_addr)(struct kvm_vcpu *vcpu, hpa_t hpa);
|
void (*set_apic_access_page_addr)(struct kvm_vcpu *vcpu, hpa_t hpa);
|
||||||
@ -1055,6 +1058,7 @@ struct kvm_x86_ops {
|
|||||||
bool (*umip_emulated)(void);
|
bool (*umip_emulated)(void);
|
||||||
|
|
||||||
int (*check_nested_events)(struct kvm_vcpu *vcpu, bool external_intr);
|
int (*check_nested_events)(struct kvm_vcpu *vcpu, bool external_intr);
|
||||||
|
void (*request_immediate_exit)(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
void (*sched_in)(struct kvm_vcpu *kvm, int cpu);
|
void (*sched_in)(struct kvm_vcpu *kvm, int cpu);
|
||||||
|
|
||||||
@ -1482,6 +1486,7 @@ extern bool kvm_find_async_pf_gfn(struct kvm_vcpu *vcpu, gfn_t gfn);
|
|||||||
|
|
||||||
int kvm_skip_emulated_instruction(struct kvm_vcpu *vcpu);
|
int kvm_skip_emulated_instruction(struct kvm_vcpu *vcpu);
|
||||||
int kvm_complete_insn_gp(struct kvm_vcpu *vcpu, int err);
|
int kvm_complete_insn_gp(struct kvm_vcpu *vcpu, int err);
|
||||||
|
void __kvm_request_immediate_exit(struct kvm_vcpu *vcpu);
|
||||||
|
|
||||||
int kvm_is_in_guest(void);
|
int kvm_is_in_guest(void);
|
||||||
|
|
||||||
|
@ -48,10 +48,13 @@ int __init early_set_memory_encrypted(unsigned long vaddr, unsigned long size);
|
|||||||
|
|
||||||
/* Architecture __weak replacement functions */
|
/* Architecture __weak replacement functions */
|
||||||
void __init mem_encrypt_init(void);
|
void __init mem_encrypt_init(void);
|
||||||
|
void __init mem_encrypt_free_decrypted_mem(void);
|
||||||
|
|
||||||
bool sme_active(void);
|
bool sme_active(void);
|
||||||
bool sev_active(void);
|
bool sev_active(void);
|
||||||
|
|
||||||
|
#define __bss_decrypted __attribute__((__section__(".bss..decrypted")))
|
||||||
|
|
||||||
#else /* !CONFIG_AMD_MEM_ENCRYPT */
|
#else /* !CONFIG_AMD_MEM_ENCRYPT */
|
||||||
|
|
||||||
#define sme_me_mask 0ULL
|
#define sme_me_mask 0ULL
|
||||||
@ -77,6 +80,8 @@ early_set_memory_decrypted(unsigned long vaddr, unsigned long size) { return 0;
|
|||||||
static inline int __init
|
static inline int __init
|
||||||
early_set_memory_encrypted(unsigned long vaddr, unsigned long size) { return 0; }
|
early_set_memory_encrypted(unsigned long vaddr, unsigned long size) { return 0; }
|
||||||
|
|
||||||
|
#define __bss_decrypted
|
||||||
|
|
||||||
#endif /* CONFIG_AMD_MEM_ENCRYPT */
|
#endif /* CONFIG_AMD_MEM_ENCRYPT */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -88,6 +93,8 @@ early_set_memory_encrypted(unsigned long vaddr, unsigned long size) { return 0;
|
|||||||
#define __sme_pa(x) (__pa(x) | sme_me_mask)
|
#define __sme_pa(x) (__pa(x) | sme_me_mask)
|
||||||
#define __sme_pa_nodebug(x) (__pa_nodebug(x) | sme_me_mask)
|
#define __sme_pa_nodebug(x) (__pa_nodebug(x) | sme_me_mask)
|
||||||
|
|
||||||
|
extern char __start_bss_decrypted[], __end_bss_decrypted[], __start_bss_decrypted_unused[];
|
||||||
|
|
||||||
#endif /* __ASSEMBLY__ */
|
#endif /* __ASSEMBLY__ */
|
||||||
|
|
||||||
#endif /* __X86_MEM_ENCRYPT_H__ */
|
#endif /* __X86_MEM_ENCRYPT_H__ */
|
||||||
|
@ -14,6 +14,7 @@
|
|||||||
#include <asm/processor.h>
|
#include <asm/processor.h>
|
||||||
#include <linux/bitops.h>
|
#include <linux/bitops.h>
|
||||||
#include <linux/threads.h>
|
#include <linux/threads.h>
|
||||||
|
#include <asm/fixmap.h>
|
||||||
|
|
||||||
extern p4d_t level4_kernel_pgt[512];
|
extern p4d_t level4_kernel_pgt[512];
|
||||||
extern p4d_t level4_ident_pgt[512];
|
extern p4d_t level4_ident_pgt[512];
|
||||||
@ -22,7 +23,7 @@ extern pud_t level3_ident_pgt[512];
|
|||||||
extern pmd_t level2_kernel_pgt[512];
|
extern pmd_t level2_kernel_pgt[512];
|
||||||
extern pmd_t level2_fixmap_pgt[512];
|
extern pmd_t level2_fixmap_pgt[512];
|
||||||
extern pmd_t level2_ident_pgt[512];
|
extern pmd_t level2_ident_pgt[512];
|
||||||
extern pte_t level1_fixmap_pgt[512];
|
extern pte_t level1_fixmap_pgt[512 * FIXMAP_PMD_NUM];
|
||||||
extern pgd_t init_top_pgt[];
|
extern pgd_t init_top_pgt[];
|
||||||
|
|
||||||
#define swapper_pg_dir init_top_pgt
|
#define swapper_pg_dir init_top_pgt
|
||||||
|
@ -377,6 +377,7 @@ struct kvm_sync_regs {
|
|||||||
|
|
||||||
#define KVM_X86_QUIRK_LINT0_REENABLED (1 << 0)
|
#define KVM_X86_QUIRK_LINT0_REENABLED (1 << 0)
|
||||||
#define KVM_X86_QUIRK_CD_NW_CLEARED (1 << 1)
|
#define KVM_X86_QUIRK_CD_NW_CLEARED (1 << 1)
|
||||||
|
#define KVM_X86_QUIRK_LAPIC_MMIO_HOLE (1 << 2)
|
||||||
|
|
||||||
#define KVM_STATE_NESTED_GUEST_MODE 0x00000001
|
#define KVM_STATE_NESTED_GUEST_MODE 0x00000001
|
||||||
#define KVM_STATE_NESTED_RUN_PENDING 0x00000002
|
#define KVM_STATE_NESTED_RUN_PENDING 0x00000002
|
||||||
|
@ -382,6 +382,11 @@ static inline bool is_mbm_event(int e)
|
|||||||
e <= QOS_L3_MBM_LOCAL_EVENT_ID);
|
e <= QOS_L3_MBM_LOCAL_EVENT_ID);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct rdt_parse_data {
|
||||||
|
struct rdtgroup *rdtgrp;
|
||||||
|
char *buf;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct rdt_resource - attributes of an RDT resource
|
* struct rdt_resource - attributes of an RDT resource
|
||||||
* @rid: The index of the resource
|
* @rid: The index of the resource
|
||||||
@ -423,16 +428,19 @@ struct rdt_resource {
|
|||||||
struct rdt_cache cache;
|
struct rdt_cache cache;
|
||||||
struct rdt_membw membw;
|
struct rdt_membw membw;
|
||||||
const char *format_str;
|
const char *format_str;
|
||||||
int (*parse_ctrlval) (void *data, struct rdt_resource *r,
|
int (*parse_ctrlval)(struct rdt_parse_data *data,
|
||||||
struct rdt_domain *d);
|
struct rdt_resource *r,
|
||||||
|
struct rdt_domain *d);
|
||||||
struct list_head evt_list;
|
struct list_head evt_list;
|
||||||
int num_rmid;
|
int num_rmid;
|
||||||
unsigned int mon_scale;
|
unsigned int mon_scale;
|
||||||
unsigned long fflags;
|
unsigned long fflags;
|
||||||
};
|
};
|
||||||
|
|
||||||
int parse_cbm(void *_data, struct rdt_resource *r, struct rdt_domain *d);
|
int parse_cbm(struct rdt_parse_data *data, struct rdt_resource *r,
|
||||||
int parse_bw(void *_buf, struct rdt_resource *r, struct rdt_domain *d);
|
struct rdt_domain *d);
|
||||||
|
int parse_bw(struct rdt_parse_data *data, struct rdt_resource *r,
|
||||||
|
struct rdt_domain *d);
|
||||||
|
|
||||||
extern struct mutex rdtgroup_mutex;
|
extern struct mutex rdtgroup_mutex;
|
||||||
|
|
||||||
@ -536,6 +544,7 @@ int rdtgroup_pseudo_lock_create(struct rdtgroup *rdtgrp);
|
|||||||
void rdtgroup_pseudo_lock_remove(struct rdtgroup *rdtgrp);
|
void rdtgroup_pseudo_lock_remove(struct rdtgroup *rdtgrp);
|
||||||
struct rdt_domain *get_domain_from_cpu(int cpu, struct rdt_resource *r);
|
struct rdt_domain *get_domain_from_cpu(int cpu, struct rdt_resource *r);
|
||||||
int update_domains(struct rdt_resource *r, int closid);
|
int update_domains(struct rdt_resource *r, int closid);
|
||||||
|
int closids_supported(void);
|
||||||
void closid_free(int closid);
|
void closid_free(int closid);
|
||||||
int alloc_rmid(void);
|
int alloc_rmid(void);
|
||||||
void free_rmid(u32 rmid);
|
void free_rmid(u32 rmid);
|
||||||
|
@ -64,19 +64,19 @@ static bool bw_validate(char *buf, unsigned long *data, struct rdt_resource *r)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
int parse_bw(void *_buf, struct rdt_resource *r, struct rdt_domain *d)
|
int parse_bw(struct rdt_parse_data *data, struct rdt_resource *r,
|
||||||
|
struct rdt_domain *d)
|
||||||
{
|
{
|
||||||
unsigned long data;
|
unsigned long bw_val;
|
||||||
char *buf = _buf;
|
|
||||||
|
|
||||||
if (d->have_new_ctrl) {
|
if (d->have_new_ctrl) {
|
||||||
rdt_last_cmd_printf("duplicate domain %d\n", d->id);
|
rdt_last_cmd_printf("duplicate domain %d\n", d->id);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!bw_validate(buf, &data, r))
|
if (!bw_validate(data->buf, &bw_val, r))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
d->new_ctrl = data;
|
d->new_ctrl = bw_val;
|
||||||
d->have_new_ctrl = true;
|
d->have_new_ctrl = true;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -123,18 +123,13 @@ static bool cbm_validate(char *buf, u32 *data, struct rdt_resource *r)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct rdt_cbm_parse_data {
|
|
||||||
struct rdtgroup *rdtgrp;
|
|
||||||
char *buf;
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read one cache bit mask (hex). Check that it is valid for the current
|
* Read one cache bit mask (hex). Check that it is valid for the current
|
||||||
* resource type.
|
* resource type.
|
||||||
*/
|
*/
|
||||||
int parse_cbm(void *_data, struct rdt_resource *r, struct rdt_domain *d)
|
int parse_cbm(struct rdt_parse_data *data, struct rdt_resource *r,
|
||||||
|
struct rdt_domain *d)
|
||||||
{
|
{
|
||||||
struct rdt_cbm_parse_data *data = _data;
|
|
||||||
struct rdtgroup *rdtgrp = data->rdtgrp;
|
struct rdtgroup *rdtgrp = data->rdtgrp;
|
||||||
u32 cbm_val;
|
u32 cbm_val;
|
||||||
|
|
||||||
@ -195,11 +190,17 @@ int parse_cbm(void *_data, struct rdt_resource *r, struct rdt_domain *d)
|
|||||||
static int parse_line(char *line, struct rdt_resource *r,
|
static int parse_line(char *line, struct rdt_resource *r,
|
||||||
struct rdtgroup *rdtgrp)
|
struct rdtgroup *rdtgrp)
|
||||||
{
|
{
|
||||||
struct rdt_cbm_parse_data data;
|
struct rdt_parse_data data;
|
||||||
char *dom = NULL, *id;
|
char *dom = NULL, *id;
|
||||||
struct rdt_domain *d;
|
struct rdt_domain *d;
|
||||||
unsigned long dom_id;
|
unsigned long dom_id;
|
||||||
|
|
||||||
|
if (rdtgrp->mode == RDT_MODE_PSEUDO_LOCKSETUP &&
|
||||||
|
r->rid == RDT_RESOURCE_MBA) {
|
||||||
|
rdt_last_cmd_puts("Cannot pseudo-lock MBA resource\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
next:
|
next:
|
||||||
if (!line || line[0] == '\0')
|
if (!line || line[0] == '\0')
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -97,6 +97,12 @@ void rdt_last_cmd_printf(const char *fmt, ...)
|
|||||||
* limited as the number of resources grows.
|
* limited as the number of resources grows.
|
||||||
*/
|
*/
|
||||||
static int closid_free_map;
|
static int closid_free_map;
|
||||||
|
static int closid_free_map_len;
|
||||||
|
|
||||||
|
int closids_supported(void)
|
||||||
|
{
|
||||||
|
return closid_free_map_len;
|
||||||
|
}
|
||||||
|
|
||||||
static void closid_init(void)
|
static void closid_init(void)
|
||||||
{
|
{
|
||||||
@ -111,6 +117,7 @@ static void closid_init(void)
|
|||||||
|
|
||||||
/* CLOSID 0 is always reserved for the default group */
|
/* CLOSID 0 is always reserved for the default group */
|
||||||
closid_free_map &= ~1;
|
closid_free_map &= ~1;
|
||||||
|
closid_free_map_len = rdt_min_closid;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int closid_alloc(void)
|
static int closid_alloc(void)
|
||||||
@ -802,7 +809,7 @@ static int rdt_bit_usage_show(struct kernfs_open_file *of,
|
|||||||
sw_shareable = 0;
|
sw_shareable = 0;
|
||||||
exclusive = 0;
|
exclusive = 0;
|
||||||
seq_printf(seq, "%d=", dom->id);
|
seq_printf(seq, "%d=", dom->id);
|
||||||
for (i = 0; i < r->num_closid; i++, ctrl++) {
|
for (i = 0; i < closids_supported(); i++, ctrl++) {
|
||||||
if (!closid_allocated(i))
|
if (!closid_allocated(i))
|
||||||
continue;
|
continue;
|
||||||
mode = rdtgroup_mode_by_closid(i);
|
mode = rdtgroup_mode_by_closid(i);
|
||||||
@ -989,7 +996,7 @@ bool rdtgroup_cbm_overlaps(struct rdt_resource *r, struct rdt_domain *d,
|
|||||||
|
|
||||||
/* Check for overlap with other resource groups */
|
/* Check for overlap with other resource groups */
|
||||||
ctrl = d->ctrl_val;
|
ctrl = d->ctrl_val;
|
||||||
for (i = 0; i < r->num_closid; i++, ctrl++) {
|
for (i = 0; i < closids_supported(); i++, ctrl++) {
|
||||||
ctrl_b = (unsigned long *)ctrl;
|
ctrl_b = (unsigned long *)ctrl;
|
||||||
mode = rdtgroup_mode_by_closid(i);
|
mode = rdtgroup_mode_by_closid(i);
|
||||||
if (closid_allocated(i) && i != closid &&
|
if (closid_allocated(i) && i != closid &&
|
||||||
@ -1024,16 +1031,27 @@ static bool rdtgroup_mode_test_exclusive(struct rdtgroup *rdtgrp)
|
|||||||
{
|
{
|
||||||
int closid = rdtgrp->closid;
|
int closid = rdtgrp->closid;
|
||||||
struct rdt_resource *r;
|
struct rdt_resource *r;
|
||||||
|
bool has_cache = false;
|
||||||
struct rdt_domain *d;
|
struct rdt_domain *d;
|
||||||
|
|
||||||
for_each_alloc_enabled_rdt_resource(r) {
|
for_each_alloc_enabled_rdt_resource(r) {
|
||||||
|
if (r->rid == RDT_RESOURCE_MBA)
|
||||||
|
continue;
|
||||||
|
has_cache = true;
|
||||||
list_for_each_entry(d, &r->domains, list) {
|
list_for_each_entry(d, &r->domains, list) {
|
||||||
if (rdtgroup_cbm_overlaps(r, d, d->ctrl_val[closid],
|
if (rdtgroup_cbm_overlaps(r, d, d->ctrl_val[closid],
|
||||||
rdtgrp->closid, false))
|
rdtgrp->closid, false)) {
|
||||||
|
rdt_last_cmd_puts("schemata overlaps\n");
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!has_cache) {
|
||||||
|
rdt_last_cmd_puts("cannot be exclusive without CAT/CDP\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1085,7 +1103,6 @@ static ssize_t rdtgroup_mode_write(struct kernfs_open_file *of,
|
|||||||
rdtgrp->mode = RDT_MODE_SHAREABLE;
|
rdtgrp->mode = RDT_MODE_SHAREABLE;
|
||||||
} else if (!strcmp(buf, "exclusive")) {
|
} else if (!strcmp(buf, "exclusive")) {
|
||||||
if (!rdtgroup_mode_test_exclusive(rdtgrp)) {
|
if (!rdtgroup_mode_test_exclusive(rdtgrp)) {
|
||||||
rdt_last_cmd_printf("schemata overlaps\n");
|
|
||||||
ret = -EINVAL;
|
ret = -EINVAL;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
@ -1155,8 +1172,8 @@ static int rdtgroup_size_show(struct kernfs_open_file *of,
|
|||||||
struct rdt_resource *r;
|
struct rdt_resource *r;
|
||||||
struct rdt_domain *d;
|
struct rdt_domain *d;
|
||||||
unsigned int size;
|
unsigned int size;
|
||||||
bool sep = false;
|
bool sep;
|
||||||
u32 cbm;
|
u32 ctrl;
|
||||||
|
|
||||||
rdtgrp = rdtgroup_kn_lock_live(of->kn);
|
rdtgrp = rdtgroup_kn_lock_live(of->kn);
|
||||||
if (!rdtgrp) {
|
if (!rdtgrp) {
|
||||||
@ -1174,6 +1191,7 @@ static int rdtgroup_size_show(struct kernfs_open_file *of,
|
|||||||
}
|
}
|
||||||
|
|
||||||
for_each_alloc_enabled_rdt_resource(r) {
|
for_each_alloc_enabled_rdt_resource(r) {
|
||||||
|
sep = false;
|
||||||
seq_printf(s, "%*s:", max_name_width, r->name);
|
seq_printf(s, "%*s:", max_name_width, r->name);
|
||||||
list_for_each_entry(d, &r->domains, list) {
|
list_for_each_entry(d, &r->domains, list) {
|
||||||
if (sep)
|
if (sep)
|
||||||
@ -1181,8 +1199,13 @@ static int rdtgroup_size_show(struct kernfs_open_file *of,
|
|||||||
if (rdtgrp->mode == RDT_MODE_PSEUDO_LOCKSETUP) {
|
if (rdtgrp->mode == RDT_MODE_PSEUDO_LOCKSETUP) {
|
||||||
size = 0;
|
size = 0;
|
||||||
} else {
|
} else {
|
||||||
cbm = d->ctrl_val[rdtgrp->closid];
|
ctrl = (!is_mba_sc(r) ?
|
||||||
size = rdtgroup_cbm_to_size(r, d, cbm);
|
d->ctrl_val[rdtgrp->closid] :
|
||||||
|
d->mbps_val[rdtgrp->closid]);
|
||||||
|
if (r->rid == RDT_RESOURCE_MBA)
|
||||||
|
size = ctrl;
|
||||||
|
else
|
||||||
|
size = rdtgroup_cbm_to_size(r, d, ctrl);
|
||||||
}
|
}
|
||||||
seq_printf(s, "%d=%u", d->id, size);
|
seq_printf(s, "%d=%u", d->id, size);
|
||||||
sep = true;
|
sep = true;
|
||||||
@ -2336,12 +2359,18 @@ static int rdtgroup_init_alloc(struct rdtgroup *rdtgrp)
|
|||||||
u32 *ctrl;
|
u32 *ctrl;
|
||||||
|
|
||||||
for_each_alloc_enabled_rdt_resource(r) {
|
for_each_alloc_enabled_rdt_resource(r) {
|
||||||
|
/*
|
||||||
|
* Only initialize default allocations for CBM cache
|
||||||
|
* resources
|
||||||
|
*/
|
||||||
|
if (r->rid == RDT_RESOURCE_MBA)
|
||||||
|
continue;
|
||||||
list_for_each_entry(d, &r->domains, list) {
|
list_for_each_entry(d, &r->domains, list) {
|
||||||
d->have_new_ctrl = false;
|
d->have_new_ctrl = false;
|
||||||
d->new_ctrl = r->cache.shareable_bits;
|
d->new_ctrl = r->cache.shareable_bits;
|
||||||
used_b = r->cache.shareable_bits;
|
used_b = r->cache.shareable_bits;
|
||||||
ctrl = d->ctrl_val;
|
ctrl = d->ctrl_val;
|
||||||
for (i = 0; i < r->num_closid; i++, ctrl++) {
|
for (i = 0; i < closids_supported(); i++, ctrl++) {
|
||||||
if (closid_allocated(i) && i != closid) {
|
if (closid_allocated(i) && i != closid) {
|
||||||
mode = rdtgroup_mode_by_closid(i);
|
mode = rdtgroup_mode_by_closid(i);
|
||||||
if (mode == RDT_MODE_PSEUDO_LOCKSETUP)
|
if (mode == RDT_MODE_PSEUDO_LOCKSETUP)
|
||||||
@ -2373,6 +2402,12 @@ static int rdtgroup_init_alloc(struct rdtgroup *rdtgrp)
|
|||||||
}
|
}
|
||||||
|
|
||||||
for_each_alloc_enabled_rdt_resource(r) {
|
for_each_alloc_enabled_rdt_resource(r) {
|
||||||
|
/*
|
||||||
|
* Only initialize default allocations for CBM cache
|
||||||
|
* resources
|
||||||
|
*/
|
||||||
|
if (r->rid == RDT_RESOURCE_MBA)
|
||||||
|
continue;
|
||||||
ret = update_domains(r, rdtgrp->closid);
|
ret = update_domains(r, rdtgrp->closid);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
rdt_last_cmd_puts("failed to initialize allocations\n");
|
rdt_last_cmd_puts("failed to initialize allocations\n");
|
||||||
|
@ -35,6 +35,7 @@
|
|||||||
#include <asm/bootparam_utils.h>
|
#include <asm/bootparam_utils.h>
|
||||||
#include <asm/microcode.h>
|
#include <asm/microcode.h>
|
||||||
#include <asm/kasan.h>
|
#include <asm/kasan.h>
|
||||||
|
#include <asm/fixmap.h>
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Manage page tables very early on.
|
* Manage page tables very early on.
|
||||||
@ -112,6 +113,7 @@ static bool __head check_la57_support(unsigned long physaddr)
|
|||||||
unsigned long __head __startup_64(unsigned long physaddr,
|
unsigned long __head __startup_64(unsigned long physaddr,
|
||||||
struct boot_params *bp)
|
struct boot_params *bp)
|
||||||
{
|
{
|
||||||
|
unsigned long vaddr, vaddr_end;
|
||||||
unsigned long load_delta, *p;
|
unsigned long load_delta, *p;
|
||||||
unsigned long pgtable_flags;
|
unsigned long pgtable_flags;
|
||||||
pgdval_t *pgd;
|
pgdval_t *pgd;
|
||||||
@ -165,7 +167,8 @@ unsigned long __head __startup_64(unsigned long physaddr,
|
|||||||
pud[511] += load_delta;
|
pud[511] += load_delta;
|
||||||
|
|
||||||
pmd = fixup_pointer(level2_fixmap_pgt, physaddr);
|
pmd = fixup_pointer(level2_fixmap_pgt, physaddr);
|
||||||
pmd[506] += load_delta;
|
for (i = FIXMAP_PMD_TOP; i > FIXMAP_PMD_TOP - FIXMAP_PMD_NUM; i--)
|
||||||
|
pmd[i] += load_delta;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Set up the identity mapping for the switchover. These
|
* Set up the identity mapping for the switchover. These
|
||||||
@ -234,6 +237,21 @@ unsigned long __head __startup_64(unsigned long physaddr,
|
|||||||
/* Encrypt the kernel and related (if SME is active) */
|
/* Encrypt the kernel and related (if SME is active) */
|
||||||
sme_encrypt_kernel(bp);
|
sme_encrypt_kernel(bp);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Clear the memory encryption mask from the .bss..decrypted section.
|
||||||
|
* The bss section will be memset to zero later in the initialization so
|
||||||
|
* there is no need to zero it after changing the memory encryption
|
||||||
|
* attribute.
|
||||||
|
*/
|
||||||
|
if (mem_encrypt_active()) {
|
||||||
|
vaddr = (unsigned long)__start_bss_decrypted;
|
||||||
|
vaddr_end = (unsigned long)__end_bss_decrypted;
|
||||||
|
for (; vaddr < vaddr_end; vaddr += PMD_SIZE) {
|
||||||
|
i = pmd_index(vaddr);
|
||||||
|
pmd[i] -= sme_get_me_mask();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return the SME encryption mask (if SME is active) to be used as a
|
* Return the SME encryption mask (if SME is active) to be used as a
|
||||||
* modifier for the initial pgdir entry programmed into CR3.
|
* modifier for the initial pgdir entry programmed into CR3.
|
||||||
|
@ -24,6 +24,7 @@
|
|||||||
#include "../entry/calling.h"
|
#include "../entry/calling.h"
|
||||||
#include <asm/export.h>
|
#include <asm/export.h>
|
||||||
#include <asm/nospec-branch.h>
|
#include <asm/nospec-branch.h>
|
||||||
|
#include <asm/fixmap.h>
|
||||||
|
|
||||||
#ifdef CONFIG_PARAVIRT
|
#ifdef CONFIG_PARAVIRT
|
||||||
#include <asm/asm-offsets.h>
|
#include <asm/asm-offsets.h>
|
||||||
@ -445,13 +446,20 @@ NEXT_PAGE(level2_kernel_pgt)
|
|||||||
KERNEL_IMAGE_SIZE/PMD_SIZE)
|
KERNEL_IMAGE_SIZE/PMD_SIZE)
|
||||||
|
|
||||||
NEXT_PAGE(level2_fixmap_pgt)
|
NEXT_PAGE(level2_fixmap_pgt)
|
||||||
.fill 506,8,0
|
.fill (512 - 4 - FIXMAP_PMD_NUM),8,0
|
||||||
.quad level1_fixmap_pgt - __START_KERNEL_map + _PAGE_TABLE_NOENC
|
pgtno = 0
|
||||||
/* 8MB reserved for vsyscalls + a 2MB hole = 4 + 1 entries */
|
.rept (FIXMAP_PMD_NUM)
|
||||||
.fill 5,8,0
|
.quad level1_fixmap_pgt + (pgtno << PAGE_SHIFT) - __START_KERNEL_map \
|
||||||
|
+ _PAGE_TABLE_NOENC;
|
||||||
|
pgtno = pgtno + 1
|
||||||
|
.endr
|
||||||
|
/* 6 MB reserved space + a 2MB hole */
|
||||||
|
.fill 4,8,0
|
||||||
|
|
||||||
NEXT_PAGE(level1_fixmap_pgt)
|
NEXT_PAGE(level1_fixmap_pgt)
|
||||||
|
.rept (FIXMAP_PMD_NUM)
|
||||||
.fill 512,8,0
|
.fill 512,8,0
|
||||||
|
.endr
|
||||||
|
|
||||||
#undef PMDS
|
#undef PMDS
|
||||||
|
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
#include <linux/sched/clock.h>
|
#include <linux/sched/clock.h>
|
||||||
#include <linux/mm.h>
|
#include <linux/mm.h>
|
||||||
#include <linux/slab.h>
|
#include <linux/slab.h>
|
||||||
|
#include <linux/set_memory.h>
|
||||||
|
|
||||||
#include <asm/hypervisor.h>
|
#include <asm/hypervisor.h>
|
||||||
#include <asm/mem_encrypt.h>
|
#include <asm/mem_encrypt.h>
|
||||||
@ -61,9 +62,10 @@ early_param("no-kvmclock-vsyscall", parse_no_kvmclock_vsyscall);
|
|||||||
(PAGE_SIZE / sizeof(struct pvclock_vsyscall_time_info))
|
(PAGE_SIZE / sizeof(struct pvclock_vsyscall_time_info))
|
||||||
|
|
||||||
static struct pvclock_vsyscall_time_info
|
static struct pvclock_vsyscall_time_info
|
||||||
hv_clock_boot[HVC_BOOT_ARRAY_SIZE] __aligned(PAGE_SIZE);
|
hv_clock_boot[HVC_BOOT_ARRAY_SIZE] __bss_decrypted __aligned(PAGE_SIZE);
|
||||||
static struct pvclock_wall_clock wall_clock;
|
static struct pvclock_wall_clock wall_clock __bss_decrypted;
|
||||||
static DEFINE_PER_CPU(struct pvclock_vsyscall_time_info *, hv_clock_per_cpu);
|
static DEFINE_PER_CPU(struct pvclock_vsyscall_time_info *, hv_clock_per_cpu);
|
||||||
|
static struct pvclock_vsyscall_time_info *hvclock_mem;
|
||||||
|
|
||||||
static inline struct pvclock_vcpu_time_info *this_cpu_pvti(void)
|
static inline struct pvclock_vcpu_time_info *this_cpu_pvti(void)
|
||||||
{
|
{
|
||||||
@ -236,6 +238,45 @@ static void kvm_shutdown(void)
|
|||||||
native_machine_shutdown();
|
native_machine_shutdown();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void __init kvmclock_init_mem(void)
|
||||||
|
{
|
||||||
|
unsigned long ncpus;
|
||||||
|
unsigned int order;
|
||||||
|
struct page *p;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
if (HVC_BOOT_ARRAY_SIZE >= num_possible_cpus())
|
||||||
|
return;
|
||||||
|
|
||||||
|
ncpus = num_possible_cpus() - HVC_BOOT_ARRAY_SIZE;
|
||||||
|
order = get_order(ncpus * sizeof(*hvclock_mem));
|
||||||
|
|
||||||
|
p = alloc_pages(GFP_KERNEL, order);
|
||||||
|
if (!p) {
|
||||||
|
pr_warn("%s: failed to alloc %d pages", __func__, (1U << order));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
hvclock_mem = page_address(p);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* hvclock is shared between the guest and the hypervisor, must
|
||||||
|
* be mapped decrypted.
|
||||||
|
*/
|
||||||
|
if (sev_active()) {
|
||||||
|
r = set_memory_decrypted((unsigned long) hvclock_mem,
|
||||||
|
1UL << order);
|
||||||
|
if (r) {
|
||||||
|
__free_pages(p, order);
|
||||||
|
hvclock_mem = NULL;
|
||||||
|
pr_warn("kvmclock: set_memory_decrypted() failed. Disabling\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
memset(hvclock_mem, 0, PAGE_SIZE << order);
|
||||||
|
}
|
||||||
|
|
||||||
static int __init kvm_setup_vsyscall_timeinfo(void)
|
static int __init kvm_setup_vsyscall_timeinfo(void)
|
||||||
{
|
{
|
||||||
#ifdef CONFIG_X86_64
|
#ifdef CONFIG_X86_64
|
||||||
@ -250,6 +291,9 @@ static int __init kvm_setup_vsyscall_timeinfo(void)
|
|||||||
|
|
||||||
kvm_clock.archdata.vclock_mode = VCLOCK_PVCLOCK;
|
kvm_clock.archdata.vclock_mode = VCLOCK_PVCLOCK;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
kvmclock_init_mem();
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
early_initcall(kvm_setup_vsyscall_timeinfo);
|
early_initcall(kvm_setup_vsyscall_timeinfo);
|
||||||
@ -269,8 +313,10 @@ static int kvmclock_setup_percpu(unsigned int cpu)
|
|||||||
/* Use the static page for the first CPUs, allocate otherwise */
|
/* Use the static page for the first CPUs, allocate otherwise */
|
||||||
if (cpu < HVC_BOOT_ARRAY_SIZE)
|
if (cpu < HVC_BOOT_ARRAY_SIZE)
|
||||||
p = &hv_clock_boot[cpu];
|
p = &hv_clock_boot[cpu];
|
||||||
|
else if (hvclock_mem)
|
||||||
|
p = hvclock_mem + cpu - HVC_BOOT_ARRAY_SIZE;
|
||||||
else
|
else
|
||||||
p = kzalloc(sizeof(*p), GFP_KERNEL);
|
return -ENOMEM;
|
||||||
|
|
||||||
per_cpu(hv_clock_per_cpu, cpu) = p;
|
per_cpu(hv_clock_per_cpu, cpu) = p;
|
||||||
return p ? 0 : -ENOMEM;
|
return p ? 0 : -ENOMEM;
|
||||||
|
@ -91,7 +91,7 @@ unsigned paravirt_patch_call(void *insnbuf,
|
|||||||
|
|
||||||
if (len < 5) {
|
if (len < 5) {
|
||||||
#ifdef CONFIG_RETPOLINE
|
#ifdef CONFIG_RETPOLINE
|
||||||
WARN_ONCE("Failing to patch indirect CALL in %ps\n", (void *)addr);
|
WARN_ONCE(1, "Failing to patch indirect CALL in %ps\n", (void *)addr);
|
||||||
#endif
|
#endif
|
||||||
return len; /* call too long for patch site */
|
return len; /* call too long for patch site */
|
||||||
}
|
}
|
||||||
@ -111,7 +111,7 @@ unsigned paravirt_patch_jmp(void *insnbuf, const void *target,
|
|||||||
|
|
||||||
if (len < 5) {
|
if (len < 5) {
|
||||||
#ifdef CONFIG_RETPOLINE
|
#ifdef CONFIG_RETPOLINE
|
||||||
WARN_ONCE("Failing to patch indirect JMP in %ps\n", (void *)addr);
|
WARN_ONCE(1, "Failing to patch indirect JMP in %ps\n", (void *)addr);
|
||||||
#endif
|
#endif
|
||||||
return len; /* call too long for patch site */
|
return len; /* call too long for patch site */
|
||||||
}
|
}
|
||||||
|
@ -65,6 +65,23 @@ jiffies_64 = jiffies;
|
|||||||
#define ALIGN_ENTRY_TEXT_BEGIN . = ALIGN(PMD_SIZE);
|
#define ALIGN_ENTRY_TEXT_BEGIN . = ALIGN(PMD_SIZE);
|
||||||
#define ALIGN_ENTRY_TEXT_END . = ALIGN(PMD_SIZE);
|
#define ALIGN_ENTRY_TEXT_END . = ALIGN(PMD_SIZE);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This section contains data which will be mapped as decrypted. Memory
|
||||||
|
* encryption operates on a page basis. Make this section PMD-aligned
|
||||||
|
* to avoid splitting the pages while mapping the section early.
|
||||||
|
*
|
||||||
|
* Note: We use a separate section so that only this section gets
|
||||||
|
* decrypted to avoid exposing more than we wish.
|
||||||
|
*/
|
||||||
|
#define BSS_DECRYPTED \
|
||||||
|
. = ALIGN(PMD_SIZE); \
|
||||||
|
__start_bss_decrypted = .; \
|
||||||
|
*(.bss..decrypted); \
|
||||||
|
. = ALIGN(PAGE_SIZE); \
|
||||||
|
__start_bss_decrypted_unused = .; \
|
||||||
|
. = ALIGN(PMD_SIZE); \
|
||||||
|
__end_bss_decrypted = .; \
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
#define X86_ALIGN_RODATA_BEGIN
|
#define X86_ALIGN_RODATA_BEGIN
|
||||||
@ -74,6 +91,7 @@ jiffies_64 = jiffies;
|
|||||||
|
|
||||||
#define ALIGN_ENTRY_TEXT_BEGIN
|
#define ALIGN_ENTRY_TEXT_BEGIN
|
||||||
#define ALIGN_ENTRY_TEXT_END
|
#define ALIGN_ENTRY_TEXT_END
|
||||||
|
#define BSS_DECRYPTED
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -355,6 +373,7 @@ SECTIONS
|
|||||||
__bss_start = .;
|
__bss_start = .;
|
||||||
*(.bss..page_aligned)
|
*(.bss..page_aligned)
|
||||||
*(.bss)
|
*(.bss)
|
||||||
|
BSS_DECRYPTED
|
||||||
. = ALIGN(PAGE_SIZE);
|
. = ALIGN(PAGE_SIZE);
|
||||||
__bss_stop = .;
|
__bss_stop = .;
|
||||||
}
|
}
|
||||||
|
@ -1344,9 +1344,8 @@ EXPORT_SYMBOL_GPL(kvm_lapic_reg_read);
|
|||||||
|
|
||||||
static int apic_mmio_in_range(struct kvm_lapic *apic, gpa_t addr)
|
static int apic_mmio_in_range(struct kvm_lapic *apic, gpa_t addr)
|
||||||
{
|
{
|
||||||
return kvm_apic_hw_enabled(apic) &&
|
return addr >= apic->base_address &&
|
||||||
addr >= apic->base_address &&
|
addr < apic->base_address + LAPIC_MMIO_LENGTH;
|
||||||
addr < apic->base_address + LAPIC_MMIO_LENGTH;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int apic_mmio_read(struct kvm_vcpu *vcpu, struct kvm_io_device *this,
|
static int apic_mmio_read(struct kvm_vcpu *vcpu, struct kvm_io_device *this,
|
||||||
@ -1358,6 +1357,15 @@ static int apic_mmio_read(struct kvm_vcpu *vcpu, struct kvm_io_device *this,
|
|||||||
if (!apic_mmio_in_range(apic, address))
|
if (!apic_mmio_in_range(apic, address))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
if (!kvm_apic_hw_enabled(apic) || apic_x2apic_mode(apic)) {
|
||||||
|
if (!kvm_check_has_quirk(vcpu->kvm,
|
||||||
|
KVM_X86_QUIRK_LAPIC_MMIO_HOLE))
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
memset(data, 0xff, len);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
kvm_lapic_reg_read(apic, offset, len, data);
|
kvm_lapic_reg_read(apic, offset, len, data);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -1917,6 +1925,14 @@ static int apic_mmio_write(struct kvm_vcpu *vcpu, struct kvm_io_device *this,
|
|||||||
if (!apic_mmio_in_range(apic, address))
|
if (!apic_mmio_in_range(apic, address))
|
||||||
return -EOPNOTSUPP;
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
if (!kvm_apic_hw_enabled(apic) || apic_x2apic_mode(apic)) {
|
||||||
|
if (!kvm_check_has_quirk(vcpu->kvm,
|
||||||
|
KVM_X86_QUIRK_LAPIC_MMIO_HOLE))
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* APIC register must be aligned on 128-bits boundary.
|
* APIC register must be aligned on 128-bits boundary.
|
||||||
* 32/64/128 bits registers must be accessed thru 32 bits.
|
* 32/64/128 bits registers must be accessed thru 32 bits.
|
||||||
|
@ -899,7 +899,7 @@ static void walk_shadow_page_lockless_end(struct kvm_vcpu *vcpu)
|
|||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Make sure the write to vcpu->mode is not reordered in front of
|
* Make sure the write to vcpu->mode is not reordered in front of
|
||||||
* reads to sptes. If it does, kvm_commit_zap_page() can see us
|
* reads to sptes. If it does, kvm_mmu_commit_zap_page() can see us
|
||||||
* OUTSIDE_GUEST_MODE and proceed to free the shadow page table.
|
* OUTSIDE_GUEST_MODE and proceed to free the shadow page table.
|
||||||
*/
|
*/
|
||||||
smp_store_release(&vcpu->mode, OUTSIDE_GUEST_MODE);
|
smp_store_release(&vcpu->mode, OUTSIDE_GUEST_MODE);
|
||||||
@ -5417,7 +5417,12 @@ void kvm_mmu_setup(struct kvm_vcpu *vcpu)
|
|||||||
{
|
{
|
||||||
MMU_WARN_ON(VALID_PAGE(vcpu->arch.mmu.root_hpa));
|
MMU_WARN_ON(VALID_PAGE(vcpu->arch.mmu.root_hpa));
|
||||||
|
|
||||||
kvm_init_mmu(vcpu, true);
|
/*
|
||||||
|
* kvm_mmu_setup() is called only on vCPU initialization.
|
||||||
|
* Therefore, no need to reset mmu roots as they are not yet
|
||||||
|
* initialized.
|
||||||
|
*/
|
||||||
|
kvm_init_mmu(vcpu, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void kvm_mmu_invalidate_zap_pages_in_memslot(struct kvm *kvm,
|
static void kvm_mmu_invalidate_zap_pages_in_memslot(struct kvm *kvm,
|
||||||
|
@ -1226,8 +1226,7 @@ static __init int sev_hardware_setup(void)
|
|||||||
min_sev_asid = cpuid_edx(0x8000001F);
|
min_sev_asid = cpuid_edx(0x8000001F);
|
||||||
|
|
||||||
/* Initialize SEV ASID bitmap */
|
/* Initialize SEV ASID bitmap */
|
||||||
sev_asid_bitmap = kcalloc(BITS_TO_LONGS(max_sev_asid),
|
sev_asid_bitmap = bitmap_zalloc(max_sev_asid, GFP_KERNEL);
|
||||||
sizeof(unsigned long), GFP_KERNEL);
|
|
||||||
if (!sev_asid_bitmap)
|
if (!sev_asid_bitmap)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
@ -1405,7 +1404,7 @@ static __exit void svm_hardware_unsetup(void)
|
|||||||
int cpu;
|
int cpu;
|
||||||
|
|
||||||
if (svm_sev_enabled())
|
if (svm_sev_enabled())
|
||||||
kfree(sev_asid_bitmap);
|
bitmap_free(sev_asid_bitmap);
|
||||||
|
|
||||||
for_each_possible_cpu(cpu)
|
for_each_possible_cpu(cpu)
|
||||||
svm_cpu_uninit(cpu);
|
svm_cpu_uninit(cpu);
|
||||||
@ -7149,6 +7148,8 @@ static struct kvm_x86_ops svm_x86_ops __ro_after_init = {
|
|||||||
.check_intercept = svm_check_intercept,
|
.check_intercept = svm_check_intercept,
|
||||||
.handle_external_intr = svm_handle_external_intr,
|
.handle_external_intr = svm_handle_external_intr,
|
||||||
|
|
||||||
|
.request_immediate_exit = __kvm_request_immediate_exit,
|
||||||
|
|
||||||
.sched_in = svm_sched_in,
|
.sched_in = svm_sched_in,
|
||||||
|
|
||||||
.pmu_ops = &amd_pmu_ops,
|
.pmu_ops = &amd_pmu_ops,
|
||||||
|
@ -397,6 +397,7 @@ struct loaded_vmcs {
|
|||||||
int cpu;
|
int cpu;
|
||||||
bool launched;
|
bool launched;
|
||||||
bool nmi_known_unmasked;
|
bool nmi_known_unmasked;
|
||||||
|
bool hv_timer_armed;
|
||||||
/* Support for vnmi-less CPUs */
|
/* Support for vnmi-less CPUs */
|
||||||
int soft_vnmi_blocked;
|
int soft_vnmi_blocked;
|
||||||
ktime_t entry_time;
|
ktime_t entry_time;
|
||||||
@ -1019,6 +1020,8 @@ struct vcpu_vmx {
|
|||||||
int ple_window;
|
int ple_window;
|
||||||
bool ple_window_dirty;
|
bool ple_window_dirty;
|
||||||
|
|
||||||
|
bool req_immediate_exit;
|
||||||
|
|
||||||
/* Support for PML */
|
/* Support for PML */
|
||||||
#define PML_ENTITY_NUM 512
|
#define PML_ENTITY_NUM 512
|
||||||
struct page *pml_pg;
|
struct page *pml_pg;
|
||||||
@ -2864,6 +2867,8 @@ static void vmx_prepare_switch_to_guest(struct kvm_vcpu *vcpu)
|
|||||||
u16 fs_sel, gs_sel;
|
u16 fs_sel, gs_sel;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
vmx->req_immediate_exit = false;
|
||||||
|
|
||||||
if (vmx->loaded_cpu_state)
|
if (vmx->loaded_cpu_state)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -5393,9 +5398,10 @@ static int vmx_set_cr4(struct kvm_vcpu *vcpu, unsigned long cr4)
|
|||||||
* To use VMXON (and later other VMX instructions), a guest
|
* To use VMXON (and later other VMX instructions), a guest
|
||||||
* must first be able to turn on cr4.VMXE (see handle_vmon()).
|
* must first be able to turn on cr4.VMXE (see handle_vmon()).
|
||||||
* So basically the check on whether to allow nested VMX
|
* So basically the check on whether to allow nested VMX
|
||||||
* is here.
|
* is here. We operate under the default treatment of SMM,
|
||||||
|
* so VMX cannot be enabled under SMM.
|
||||||
*/
|
*/
|
||||||
if (!nested_vmx_allowed(vcpu))
|
if (!nested_vmx_allowed(vcpu) || is_smm(vcpu))
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6183,6 +6189,27 @@ static void vmx_complete_nested_posted_interrupt(struct kvm_vcpu *vcpu)
|
|||||||
nested_mark_vmcs12_pages_dirty(vcpu);
|
nested_mark_vmcs12_pages_dirty(vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool vmx_guest_apic_has_interrupt(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
||||||
|
void *vapic_page;
|
||||||
|
u32 vppr;
|
||||||
|
int rvi;
|
||||||
|
|
||||||
|
if (WARN_ON_ONCE(!is_guest_mode(vcpu)) ||
|
||||||
|
!nested_cpu_has_vid(get_vmcs12(vcpu)) ||
|
||||||
|
WARN_ON_ONCE(!vmx->nested.virtual_apic_page))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
rvi = vmcs_read16(GUEST_INTR_STATUS) & 0xff;
|
||||||
|
|
||||||
|
vapic_page = kmap(vmx->nested.virtual_apic_page);
|
||||||
|
vppr = *((u32 *)(vapic_page + APIC_PROCPRI));
|
||||||
|
kunmap(vmx->nested.virtual_apic_page);
|
||||||
|
|
||||||
|
return ((rvi & 0xf0) > (vppr & 0xf0));
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool kvm_vcpu_trigger_posted_interrupt(struct kvm_vcpu *vcpu,
|
static inline bool kvm_vcpu_trigger_posted_interrupt(struct kvm_vcpu *vcpu,
|
||||||
bool nested)
|
bool nested)
|
||||||
{
|
{
|
||||||
@ -7966,6 +7993,9 @@ static __init int hardware_setup(void)
|
|||||||
kvm_x86_ops->enable_log_dirty_pt_masked = NULL;
|
kvm_x86_ops->enable_log_dirty_pt_masked = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!cpu_has_vmx_preemption_timer())
|
||||||
|
kvm_x86_ops->request_immediate_exit = __kvm_request_immediate_exit;
|
||||||
|
|
||||||
if (cpu_has_vmx_preemption_timer() && enable_preemption_timer) {
|
if (cpu_has_vmx_preemption_timer() && enable_preemption_timer) {
|
||||||
u64 vmx_msr;
|
u64 vmx_msr;
|
||||||
|
|
||||||
@ -9208,7 +9238,8 @@ static int handle_pml_full(struct kvm_vcpu *vcpu)
|
|||||||
|
|
||||||
static int handle_preemption_timer(struct kvm_vcpu *vcpu)
|
static int handle_preemption_timer(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
kvm_lapic_expired_hv_timer(vcpu);
|
if (!to_vmx(vcpu)->req_immediate_exit)
|
||||||
|
kvm_lapic_expired_hv_timer(vcpu);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -10595,24 +10626,43 @@ static void atomic_switch_perf_msrs(struct vcpu_vmx *vmx)
|
|||||||
msrs[i].host, false);
|
msrs[i].host, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vmx_arm_hv_timer(struct kvm_vcpu *vcpu)
|
static void vmx_arm_hv_timer(struct vcpu_vmx *vmx, u32 val)
|
||||||
|
{
|
||||||
|
vmcs_write32(VMX_PREEMPTION_TIMER_VALUE, val);
|
||||||
|
if (!vmx->loaded_vmcs->hv_timer_armed)
|
||||||
|
vmcs_set_bits(PIN_BASED_VM_EXEC_CONTROL,
|
||||||
|
PIN_BASED_VMX_PREEMPTION_TIMER);
|
||||||
|
vmx->loaded_vmcs->hv_timer_armed = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void vmx_update_hv_timer(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
||||||
u64 tscl;
|
u64 tscl;
|
||||||
u32 delta_tsc;
|
u32 delta_tsc;
|
||||||
|
|
||||||
if (vmx->hv_deadline_tsc == -1)
|
if (vmx->req_immediate_exit) {
|
||||||
|
vmx_arm_hv_timer(vmx, 0);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
tscl = rdtsc();
|
if (vmx->hv_deadline_tsc != -1) {
|
||||||
if (vmx->hv_deadline_tsc > tscl)
|
tscl = rdtsc();
|
||||||
/* sure to be 32 bit only because checked on set_hv_timer */
|
if (vmx->hv_deadline_tsc > tscl)
|
||||||
delta_tsc = (u32)((vmx->hv_deadline_tsc - tscl) >>
|
/* set_hv_timer ensures the delta fits in 32-bits */
|
||||||
cpu_preemption_timer_multi);
|
delta_tsc = (u32)((vmx->hv_deadline_tsc - tscl) >>
|
||||||
else
|
cpu_preemption_timer_multi);
|
||||||
delta_tsc = 0;
|
else
|
||||||
|
delta_tsc = 0;
|
||||||
|
|
||||||
vmcs_write32(VMX_PREEMPTION_TIMER_VALUE, delta_tsc);
|
vmx_arm_hv_timer(vmx, delta_tsc);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (vmx->loaded_vmcs->hv_timer_armed)
|
||||||
|
vmcs_clear_bits(PIN_BASED_VM_EXEC_CONTROL,
|
||||||
|
PIN_BASED_VMX_PREEMPTION_TIMER);
|
||||||
|
vmx->loaded_vmcs->hv_timer_armed = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void __noclone vmx_vcpu_run(struct kvm_vcpu *vcpu)
|
static void __noclone vmx_vcpu_run(struct kvm_vcpu *vcpu)
|
||||||
@ -10672,7 +10722,7 @@ static void __noclone vmx_vcpu_run(struct kvm_vcpu *vcpu)
|
|||||||
|
|
||||||
atomic_switch_perf_msrs(vmx);
|
atomic_switch_perf_msrs(vmx);
|
||||||
|
|
||||||
vmx_arm_hv_timer(vcpu);
|
vmx_update_hv_timer(vcpu);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If this vCPU has touched SPEC_CTRL, restore the guest's value if
|
* If this vCPU has touched SPEC_CTRL, restore the guest's value if
|
||||||
@ -11427,16 +11477,18 @@ static void vmx_start_preemption_timer(struct kvm_vcpu *vcpu)
|
|||||||
u64 preemption_timeout = get_vmcs12(vcpu)->vmx_preemption_timer_value;
|
u64 preemption_timeout = get_vmcs12(vcpu)->vmx_preemption_timer_value;
|
||||||
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
||||||
|
|
||||||
if (vcpu->arch.virtual_tsc_khz == 0)
|
/*
|
||||||
return;
|
* A timer value of zero is architecturally guaranteed to cause
|
||||||
|
* a VMExit prior to executing any instructions in the guest.
|
||||||
/* Make sure short timeouts reliably trigger an immediate vmexit.
|
*/
|
||||||
* hrtimer_start does not guarantee this. */
|
if (preemption_timeout == 0) {
|
||||||
if (preemption_timeout <= 1) {
|
|
||||||
vmx_preemption_timer_fn(&vmx->nested.preemption_timer);
|
vmx_preemption_timer_fn(&vmx->nested.preemption_timer);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (vcpu->arch.virtual_tsc_khz == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
preemption_timeout <<= VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE;
|
preemption_timeout <<= VMX_MISC_EMULATED_PREEMPTION_TIMER_RATE;
|
||||||
preemption_timeout *= 1000000;
|
preemption_timeout *= 1000000;
|
||||||
do_div(preemption_timeout, vcpu->arch.virtual_tsc_khz);
|
do_div(preemption_timeout, vcpu->arch.virtual_tsc_khz);
|
||||||
@ -11646,11 +11698,15 @@ static int nested_vmx_check_apicv_controls(struct kvm_vcpu *vcpu,
|
|||||||
* bits 15:8 should be zero in posted_intr_nv,
|
* bits 15:8 should be zero in posted_intr_nv,
|
||||||
* the descriptor address has been already checked
|
* the descriptor address has been already checked
|
||||||
* in nested_get_vmcs12_pages.
|
* in nested_get_vmcs12_pages.
|
||||||
|
*
|
||||||
|
* bits 5:0 of posted_intr_desc_addr should be zero.
|
||||||
*/
|
*/
|
||||||
if (nested_cpu_has_posted_intr(vmcs12) &&
|
if (nested_cpu_has_posted_intr(vmcs12) &&
|
||||||
(!nested_cpu_has_vid(vmcs12) ||
|
(!nested_cpu_has_vid(vmcs12) ||
|
||||||
!nested_exit_intr_ack_set(vcpu) ||
|
!nested_exit_intr_ack_set(vcpu) ||
|
||||||
vmcs12->posted_intr_nv & 0xff00))
|
(vmcs12->posted_intr_nv & 0xff00) ||
|
||||||
|
(vmcs12->posted_intr_desc_addr & 0x3f) ||
|
||||||
|
(!page_address_valid(vcpu, vmcs12->posted_intr_desc_addr))))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
/* tpr shadow is needed by all apicv features. */
|
/* tpr shadow is needed by all apicv features. */
|
||||||
@ -12076,11 +12132,10 @@ static int prepare_vmcs02(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12,
|
|||||||
|
|
||||||
exec_control = vmcs12->pin_based_vm_exec_control;
|
exec_control = vmcs12->pin_based_vm_exec_control;
|
||||||
|
|
||||||
/* Preemption timer setting is only taken from vmcs01. */
|
/* Preemption timer setting is computed directly in vmx_vcpu_run. */
|
||||||
exec_control &= ~PIN_BASED_VMX_PREEMPTION_TIMER;
|
|
||||||
exec_control |= vmcs_config.pin_based_exec_ctrl;
|
exec_control |= vmcs_config.pin_based_exec_ctrl;
|
||||||
if (vmx->hv_deadline_tsc == -1)
|
exec_control &= ~PIN_BASED_VMX_PREEMPTION_TIMER;
|
||||||
exec_control &= ~PIN_BASED_VMX_PREEMPTION_TIMER;
|
vmx->loaded_vmcs->hv_timer_armed = false;
|
||||||
|
|
||||||
/* Posted interrupts setting is only taken from vmcs12. */
|
/* Posted interrupts setting is only taken from vmcs12. */
|
||||||
if (nested_cpu_has_posted_intr(vmcs12)) {
|
if (nested_cpu_has_posted_intr(vmcs12)) {
|
||||||
@ -12318,6 +12373,9 @@ static int check_vmentry_prereqs(struct kvm_vcpu *vcpu, struct vmcs12 *vmcs12)
|
|||||||
vmcs12->guest_activity_state != GUEST_ACTIVITY_HLT)
|
vmcs12->guest_activity_state != GUEST_ACTIVITY_HLT)
|
||||||
return VMXERR_ENTRY_INVALID_CONTROL_FIELD;
|
return VMXERR_ENTRY_INVALID_CONTROL_FIELD;
|
||||||
|
|
||||||
|
if (nested_cpu_has_vpid(vmcs12) && !vmcs12->virtual_processor_id)
|
||||||
|
return VMXERR_ENTRY_INVALID_CONTROL_FIELD;
|
||||||
|
|
||||||
if (nested_vmx_check_io_bitmap_controls(vcpu, vmcs12))
|
if (nested_vmx_check_io_bitmap_controls(vcpu, vmcs12))
|
||||||
return VMXERR_ENTRY_INVALID_CONTROL_FIELD;
|
return VMXERR_ENTRY_INVALID_CONTROL_FIELD;
|
||||||
|
|
||||||
@ -12863,6 +12921,11 @@ static int vmx_check_nested_events(struct kvm_vcpu *vcpu, bool external_intr)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void vmx_request_immediate_exit(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
to_vmx(vcpu)->req_immediate_exit = true;
|
||||||
|
}
|
||||||
|
|
||||||
static u32 vmx_get_preemption_timer_value(struct kvm_vcpu *vcpu)
|
static u32 vmx_get_preemption_timer_value(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
ktime_t remaining =
|
ktime_t remaining =
|
||||||
@ -13253,12 +13316,7 @@ static void nested_vmx_vmexit(struct kvm_vcpu *vcpu, u32 exit_reason,
|
|||||||
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, vmx->msr_autoload.host.nr);
|
vmcs_write32(VM_EXIT_MSR_LOAD_COUNT, vmx->msr_autoload.host.nr);
|
||||||
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, vmx->msr_autoload.guest.nr);
|
vmcs_write32(VM_ENTRY_MSR_LOAD_COUNT, vmx->msr_autoload.guest.nr);
|
||||||
vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset);
|
vmcs_write64(TSC_OFFSET, vcpu->arch.tsc_offset);
|
||||||
if (vmx->hv_deadline_tsc == -1)
|
|
||||||
vmcs_clear_bits(PIN_BASED_VM_EXEC_CONTROL,
|
|
||||||
PIN_BASED_VMX_PREEMPTION_TIMER);
|
|
||||||
else
|
|
||||||
vmcs_set_bits(PIN_BASED_VM_EXEC_CONTROL,
|
|
||||||
PIN_BASED_VMX_PREEMPTION_TIMER);
|
|
||||||
if (kvm_has_tsc_control)
|
if (kvm_has_tsc_control)
|
||||||
decache_tsc_multiplier(vmx);
|
decache_tsc_multiplier(vmx);
|
||||||
|
|
||||||
@ -13462,18 +13520,12 @@ static int vmx_set_hv_timer(struct kvm_vcpu *vcpu, u64 guest_deadline_tsc)
|
|||||||
return -ERANGE;
|
return -ERANGE;
|
||||||
|
|
||||||
vmx->hv_deadline_tsc = tscl + delta_tsc;
|
vmx->hv_deadline_tsc = tscl + delta_tsc;
|
||||||
vmcs_set_bits(PIN_BASED_VM_EXEC_CONTROL,
|
|
||||||
PIN_BASED_VMX_PREEMPTION_TIMER);
|
|
||||||
|
|
||||||
return delta_tsc == 0;
|
return delta_tsc == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vmx_cancel_hv_timer(struct kvm_vcpu *vcpu)
|
static void vmx_cancel_hv_timer(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
struct vcpu_vmx *vmx = to_vmx(vcpu);
|
to_vmx(vcpu)->hv_deadline_tsc = -1;
|
||||||
vmx->hv_deadline_tsc = -1;
|
|
||||||
vmcs_clear_bits(PIN_BASED_VM_EXEC_CONTROL,
|
|
||||||
PIN_BASED_VMX_PREEMPTION_TIMER);
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -13954,6 +14006,14 @@ static int vmx_set_nested_state(struct kvm_vcpu *vcpu,
|
|||||||
~(KVM_STATE_NESTED_SMM_GUEST_MODE | KVM_STATE_NESTED_SMM_VMXON))
|
~(KVM_STATE_NESTED_SMM_GUEST_MODE | KVM_STATE_NESTED_SMM_VMXON))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SMM temporarily disables VMX, so we cannot be in guest mode,
|
||||||
|
* nor can VMLAUNCH/VMRESUME be pending. Outside SMM, SMM flags
|
||||||
|
* must be zero.
|
||||||
|
*/
|
||||||
|
if (is_smm(vcpu) ? kvm_state->flags : kvm_state->vmx.smm.flags)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
if ((kvm_state->vmx.smm.flags & KVM_STATE_NESTED_SMM_GUEST_MODE) &&
|
if ((kvm_state->vmx.smm.flags & KVM_STATE_NESTED_SMM_GUEST_MODE) &&
|
||||||
!(kvm_state->vmx.smm.flags & KVM_STATE_NESTED_SMM_VMXON))
|
!(kvm_state->vmx.smm.flags & KVM_STATE_NESTED_SMM_VMXON))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@ -14097,6 +14157,7 @@ static struct kvm_x86_ops vmx_x86_ops __ro_after_init = {
|
|||||||
.apicv_post_state_restore = vmx_apicv_post_state_restore,
|
.apicv_post_state_restore = vmx_apicv_post_state_restore,
|
||||||
.hwapic_irr_update = vmx_hwapic_irr_update,
|
.hwapic_irr_update = vmx_hwapic_irr_update,
|
||||||
.hwapic_isr_update = vmx_hwapic_isr_update,
|
.hwapic_isr_update = vmx_hwapic_isr_update,
|
||||||
|
.guest_apic_has_interrupt = vmx_guest_apic_has_interrupt,
|
||||||
.sync_pir_to_irr = vmx_sync_pir_to_irr,
|
.sync_pir_to_irr = vmx_sync_pir_to_irr,
|
||||||
.deliver_posted_interrupt = vmx_deliver_posted_interrupt,
|
.deliver_posted_interrupt = vmx_deliver_posted_interrupt,
|
||||||
|
|
||||||
@ -14130,6 +14191,7 @@ static struct kvm_x86_ops vmx_x86_ops __ro_after_init = {
|
|||||||
.umip_emulated = vmx_umip_emulated,
|
.umip_emulated = vmx_umip_emulated,
|
||||||
|
|
||||||
.check_nested_events = vmx_check_nested_events,
|
.check_nested_events = vmx_check_nested_events,
|
||||||
|
.request_immediate_exit = vmx_request_immediate_exit,
|
||||||
|
|
||||||
.sched_in = vmx_sched_in,
|
.sched_in = vmx_sched_in,
|
||||||
|
|
||||||
|
@ -628,7 +628,7 @@ bool pdptrs_changed(struct kvm_vcpu *vcpu)
|
|||||||
gfn_t gfn;
|
gfn_t gfn;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
if (is_long_mode(vcpu) || !is_pae(vcpu))
|
if (is_long_mode(vcpu) || !is_pae(vcpu) || !is_paging(vcpu))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!test_bit(VCPU_EXREG_PDPTR,
|
if (!test_bit(VCPU_EXREG_PDPTR,
|
||||||
@ -2537,7 +2537,6 @@ int kvm_set_msr_common(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
|
|||||||
break;
|
break;
|
||||||
case MSR_PLATFORM_INFO:
|
case MSR_PLATFORM_INFO:
|
||||||
if (!msr_info->host_initiated ||
|
if (!msr_info->host_initiated ||
|
||||||
data & ~MSR_PLATFORM_INFO_CPUID_FAULT ||
|
|
||||||
(!(data & MSR_PLATFORM_INFO_CPUID_FAULT) &&
|
(!(data & MSR_PLATFORM_INFO_CPUID_FAULT) &&
|
||||||
cpuid_fault_enabled(vcpu)))
|
cpuid_fault_enabled(vcpu)))
|
||||||
return 1;
|
return 1;
|
||||||
@ -2780,6 +2779,9 @@ int kvm_get_msr_common(struct kvm_vcpu *vcpu, struct msr_data *msr_info)
|
|||||||
msr_info->data = vcpu->arch.osvw.status;
|
msr_info->data = vcpu->arch.osvw.status;
|
||||||
break;
|
break;
|
||||||
case MSR_PLATFORM_INFO:
|
case MSR_PLATFORM_INFO:
|
||||||
|
if (!msr_info->host_initiated &&
|
||||||
|
!vcpu->kvm->arch.guest_can_read_msr_platform_info)
|
||||||
|
return 1;
|
||||||
msr_info->data = vcpu->arch.msr_platform_info;
|
msr_info->data = vcpu->arch.msr_platform_info;
|
||||||
break;
|
break;
|
||||||
case MSR_MISC_FEATURES_ENABLES:
|
case MSR_MISC_FEATURES_ENABLES:
|
||||||
@ -2927,6 +2929,7 @@ int kvm_vm_ioctl_check_extension(struct kvm *kvm, long ext)
|
|||||||
case KVM_CAP_SPLIT_IRQCHIP:
|
case KVM_CAP_SPLIT_IRQCHIP:
|
||||||
case KVM_CAP_IMMEDIATE_EXIT:
|
case KVM_CAP_IMMEDIATE_EXIT:
|
||||||
case KVM_CAP_GET_MSR_FEATURES:
|
case KVM_CAP_GET_MSR_FEATURES:
|
||||||
|
case KVM_CAP_MSR_PLATFORM_INFO:
|
||||||
r = 1;
|
r = 1;
|
||||||
break;
|
break;
|
||||||
case KVM_CAP_SYNC_REGS:
|
case KVM_CAP_SYNC_REGS:
|
||||||
@ -4007,19 +4010,23 @@ long kvm_arch_vcpu_ioctl(struct file *filp,
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
BUILD_BUG_ON(sizeof(user_data_size) != sizeof(user_kvm_nested_state->size));
|
BUILD_BUG_ON(sizeof(user_data_size) != sizeof(user_kvm_nested_state->size));
|
||||||
|
r = -EFAULT;
|
||||||
if (get_user(user_data_size, &user_kvm_nested_state->size))
|
if (get_user(user_data_size, &user_kvm_nested_state->size))
|
||||||
return -EFAULT;
|
break;
|
||||||
|
|
||||||
r = kvm_x86_ops->get_nested_state(vcpu, user_kvm_nested_state,
|
r = kvm_x86_ops->get_nested_state(vcpu, user_kvm_nested_state,
|
||||||
user_data_size);
|
user_data_size);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
break;
|
||||||
|
|
||||||
if (r > user_data_size) {
|
if (r > user_data_size) {
|
||||||
if (put_user(r, &user_kvm_nested_state->size))
|
if (put_user(r, &user_kvm_nested_state->size))
|
||||||
return -EFAULT;
|
r = -EFAULT;
|
||||||
return -E2BIG;
|
else
|
||||||
|
r = -E2BIG;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = 0;
|
r = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -4031,19 +4038,21 @@ long kvm_arch_vcpu_ioctl(struct file *filp,
|
|||||||
if (!kvm_x86_ops->set_nested_state)
|
if (!kvm_x86_ops->set_nested_state)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
r = -EFAULT;
|
||||||
if (copy_from_user(&kvm_state, user_kvm_nested_state, sizeof(kvm_state)))
|
if (copy_from_user(&kvm_state, user_kvm_nested_state, sizeof(kvm_state)))
|
||||||
return -EFAULT;
|
break;
|
||||||
|
|
||||||
|
r = -EINVAL;
|
||||||
if (kvm_state.size < sizeof(kvm_state))
|
if (kvm_state.size < sizeof(kvm_state))
|
||||||
return -EINVAL;
|
break;
|
||||||
|
|
||||||
if (kvm_state.flags &
|
if (kvm_state.flags &
|
||||||
~(KVM_STATE_NESTED_RUN_PENDING | KVM_STATE_NESTED_GUEST_MODE))
|
~(KVM_STATE_NESTED_RUN_PENDING | KVM_STATE_NESTED_GUEST_MODE))
|
||||||
return -EINVAL;
|
break;
|
||||||
|
|
||||||
/* nested_run_pending implies guest_mode. */
|
/* nested_run_pending implies guest_mode. */
|
||||||
if (kvm_state.flags == KVM_STATE_NESTED_RUN_PENDING)
|
if (kvm_state.flags == KVM_STATE_NESTED_RUN_PENDING)
|
||||||
return -EINVAL;
|
break;
|
||||||
|
|
||||||
r = kvm_x86_ops->set_nested_state(vcpu, user_kvm_nested_state, &kvm_state);
|
r = kvm_x86_ops->set_nested_state(vcpu, user_kvm_nested_state, &kvm_state);
|
||||||
break;
|
break;
|
||||||
@ -4350,6 +4359,10 @@ split_irqchip_unlock:
|
|||||||
kvm->arch.pause_in_guest = true;
|
kvm->arch.pause_in_guest = true;
|
||||||
r = 0;
|
r = 0;
|
||||||
break;
|
break;
|
||||||
|
case KVM_CAP_MSR_PLATFORM_INFO:
|
||||||
|
kvm->arch.guest_can_read_msr_platform_info = cap->args[0];
|
||||||
|
r = 0;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
r = -EINVAL;
|
r = -EINVAL;
|
||||||
break;
|
break;
|
||||||
@ -7361,6 +7374,12 @@ void kvm_vcpu_reload_apic_access_page(struct kvm_vcpu *vcpu)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(kvm_vcpu_reload_apic_access_page);
|
EXPORT_SYMBOL_GPL(kvm_vcpu_reload_apic_access_page);
|
||||||
|
|
||||||
|
void __kvm_request_immediate_exit(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
smp_send_reschedule(vcpu->cpu);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(__kvm_request_immediate_exit);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns 1 to let vcpu_run() continue the guest execution loop without
|
* Returns 1 to let vcpu_run() continue the guest execution loop without
|
||||||
* exiting to the userspace. Otherwise, the value will be returned to the
|
* exiting to the userspace. Otherwise, the value will be returned to the
|
||||||
@ -7565,7 +7584,7 @@ static int vcpu_enter_guest(struct kvm_vcpu *vcpu)
|
|||||||
|
|
||||||
if (req_immediate_exit) {
|
if (req_immediate_exit) {
|
||||||
kvm_make_request(KVM_REQ_EVENT, vcpu);
|
kvm_make_request(KVM_REQ_EVENT, vcpu);
|
||||||
smp_send_reschedule(vcpu->cpu);
|
kvm_x86_ops->request_immediate_exit(vcpu);
|
||||||
}
|
}
|
||||||
|
|
||||||
trace_kvm_entry(vcpu->vcpu_id);
|
trace_kvm_entry(vcpu->vcpu_id);
|
||||||
@ -7829,6 +7848,29 @@ static int complete_emulated_mmio(struct kvm_vcpu *vcpu)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Swap (qemu) user FPU context for the guest FPU context. */
|
||||||
|
static void kvm_load_guest_fpu(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
preempt_disable();
|
||||||
|
copy_fpregs_to_fpstate(&vcpu->arch.user_fpu);
|
||||||
|
/* PKRU is separately restored in kvm_x86_ops->run. */
|
||||||
|
__copy_kernel_to_fpregs(&vcpu->arch.guest_fpu.state,
|
||||||
|
~XFEATURE_MASK_PKRU);
|
||||||
|
preempt_enable();
|
||||||
|
trace_kvm_fpu(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When vcpu_run ends, restore user space FPU context. */
|
||||||
|
static void kvm_put_guest_fpu(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
preempt_disable();
|
||||||
|
copy_fpregs_to_fpstate(&vcpu->arch.guest_fpu);
|
||||||
|
copy_kernel_to_fpregs(&vcpu->arch.user_fpu.state);
|
||||||
|
preempt_enable();
|
||||||
|
++vcpu->stat.fpu_reload;
|
||||||
|
trace_kvm_fpu(0);
|
||||||
|
}
|
||||||
|
|
||||||
int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
int kvm_arch_vcpu_ioctl_run(struct kvm_vcpu *vcpu, struct kvm_run *kvm_run)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
@ -8177,7 +8219,7 @@ static int __set_sregs(struct kvm_vcpu *vcpu, struct kvm_sregs *sregs)
|
|||||||
kvm_update_cpuid(vcpu);
|
kvm_update_cpuid(vcpu);
|
||||||
|
|
||||||
idx = srcu_read_lock(&vcpu->kvm->srcu);
|
idx = srcu_read_lock(&vcpu->kvm->srcu);
|
||||||
if (!is_long_mode(vcpu) && is_pae(vcpu)) {
|
if (!is_long_mode(vcpu) && is_pae(vcpu) && is_paging(vcpu)) {
|
||||||
load_pdptrs(vcpu, vcpu->arch.walk_mmu, kvm_read_cr3(vcpu));
|
load_pdptrs(vcpu, vcpu->arch.walk_mmu, kvm_read_cr3(vcpu));
|
||||||
mmu_reset_needed = 1;
|
mmu_reset_needed = 1;
|
||||||
}
|
}
|
||||||
@ -8406,29 +8448,6 @@ static void fx_init(struct kvm_vcpu *vcpu)
|
|||||||
vcpu->arch.cr0 |= X86_CR0_ET;
|
vcpu->arch.cr0 |= X86_CR0_ET;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Swap (qemu) user FPU context for the guest FPU context. */
|
|
||||||
void kvm_load_guest_fpu(struct kvm_vcpu *vcpu)
|
|
||||||
{
|
|
||||||
preempt_disable();
|
|
||||||
copy_fpregs_to_fpstate(&vcpu->arch.user_fpu);
|
|
||||||
/* PKRU is separately restored in kvm_x86_ops->run. */
|
|
||||||
__copy_kernel_to_fpregs(&vcpu->arch.guest_fpu.state,
|
|
||||||
~XFEATURE_MASK_PKRU);
|
|
||||||
preempt_enable();
|
|
||||||
trace_kvm_fpu(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* When vcpu_run ends, restore user space FPU context. */
|
|
||||||
void kvm_put_guest_fpu(struct kvm_vcpu *vcpu)
|
|
||||||
{
|
|
||||||
preempt_disable();
|
|
||||||
copy_fpregs_to_fpstate(&vcpu->arch.guest_fpu);
|
|
||||||
copy_kernel_to_fpregs(&vcpu->arch.user_fpu.state);
|
|
||||||
preempt_enable();
|
|
||||||
++vcpu->stat.fpu_reload;
|
|
||||||
trace_kvm_fpu(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
void kvm_arch_vcpu_free(struct kvm_vcpu *vcpu)
|
void kvm_arch_vcpu_free(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
void *wbinvd_dirty_mask = vcpu->arch.wbinvd_dirty_mask;
|
void *wbinvd_dirty_mask = vcpu->arch.wbinvd_dirty_mask;
|
||||||
@ -8852,6 +8871,8 @@ int kvm_arch_init_vm(struct kvm *kvm, unsigned long type)
|
|||||||
kvm->arch.kvmclock_offset = -ktime_get_boot_ns();
|
kvm->arch.kvmclock_offset = -ktime_get_boot_ns();
|
||||||
pvclock_update_vm_gtod_copy(kvm);
|
pvclock_update_vm_gtod_copy(kvm);
|
||||||
|
|
||||||
|
kvm->arch.guest_can_read_msr_platform_info = true;
|
||||||
|
|
||||||
INIT_DELAYED_WORK(&kvm->arch.kvmclock_update_work, kvmclock_update_fn);
|
INIT_DELAYED_WORK(&kvm->arch.kvmclock_update_work, kvmclock_update_fn);
|
||||||
INIT_DELAYED_WORK(&kvm->arch.kvmclock_sync_work, kvmclock_sync_fn);
|
INIT_DELAYED_WORK(&kvm->arch.kvmclock_sync_work, kvmclock_sync_fn);
|
||||||
|
|
||||||
@ -9200,6 +9221,13 @@ void kvm_arch_flush_shadow_memslot(struct kvm *kvm,
|
|||||||
kvm_page_track_flush_slot(kvm, slot);
|
kvm_page_track_flush_slot(kvm, slot);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool kvm_guest_apic_has_interrupt(struct kvm_vcpu *vcpu)
|
||||||
|
{
|
||||||
|
return (is_guest_mode(vcpu) &&
|
||||||
|
kvm_x86_ops->guest_apic_has_interrupt &&
|
||||||
|
kvm_x86_ops->guest_apic_has_interrupt(vcpu));
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool kvm_vcpu_has_events(struct kvm_vcpu *vcpu)
|
static inline bool kvm_vcpu_has_events(struct kvm_vcpu *vcpu)
|
||||||
{
|
{
|
||||||
if (!list_empty_careful(&vcpu->async_pf.done))
|
if (!list_empty_careful(&vcpu->async_pf.done))
|
||||||
@ -9224,7 +9252,8 @@ static inline bool kvm_vcpu_has_events(struct kvm_vcpu *vcpu)
|
|||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (kvm_arch_interrupt_allowed(vcpu) &&
|
if (kvm_arch_interrupt_allowed(vcpu) &&
|
||||||
kvm_cpu_has_interrupt(vcpu))
|
(kvm_cpu_has_interrupt(vcpu) ||
|
||||||
|
kvm_guest_apic_has_interrupt(vcpu)))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (kvm_hv_has_stimer_pending(vcpu))
|
if (kvm_hv_has_stimer_pending(vcpu))
|
||||||
|
@ -815,10 +815,14 @@ void free_kernel_image_pages(void *begin, void *end)
|
|||||||
set_memory_np_noalias(begin_ul, len_pages);
|
set_memory_np_noalias(begin_ul, len_pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void __weak mem_encrypt_free_decrypted_mem(void) { }
|
||||||
|
|
||||||
void __ref free_initmem(void)
|
void __ref free_initmem(void)
|
||||||
{
|
{
|
||||||
e820__reallocate_tables();
|
e820__reallocate_tables();
|
||||||
|
|
||||||
|
mem_encrypt_free_decrypted_mem();
|
||||||
|
|
||||||
free_kernel_image_pages(&__init_begin, &__init_end);
|
free_kernel_image_pages(&__init_begin, &__init_end);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -348,6 +348,30 @@ bool sev_active(void)
|
|||||||
EXPORT_SYMBOL(sev_active);
|
EXPORT_SYMBOL(sev_active);
|
||||||
|
|
||||||
/* Architecture __weak replacement functions */
|
/* Architecture __weak replacement functions */
|
||||||
|
void __init mem_encrypt_free_decrypted_mem(void)
|
||||||
|
{
|
||||||
|
unsigned long vaddr, vaddr_end, npages;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
vaddr = (unsigned long)__start_bss_decrypted_unused;
|
||||||
|
vaddr_end = (unsigned long)__end_bss_decrypted;
|
||||||
|
npages = (vaddr_end - vaddr) >> PAGE_SHIFT;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The unused memory range was mapped decrypted, change the encryption
|
||||||
|
* attribute from decrypted to encrypted before freeing it.
|
||||||
|
*/
|
||||||
|
if (mem_encrypt_active()) {
|
||||||
|
r = set_memory_encrypted(vaddr, npages);
|
||||||
|
if (r) {
|
||||||
|
pr_warn("failed to free unused decrypted pages\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
free_init_pages("unused decrypted", vaddr, vaddr_end);
|
||||||
|
}
|
||||||
|
|
||||||
void __init mem_encrypt_init(void)
|
void __init mem_encrypt_init(void)
|
||||||
{
|
{
|
||||||
if (!sme_me_mask)
|
if (!sme_me_mask)
|
||||||
|
@ -637,6 +637,15 @@ void __native_set_fixmap(enum fixed_addresses idx, pte_t pte)
|
|||||||
{
|
{
|
||||||
unsigned long address = __fix_to_virt(idx);
|
unsigned long address = __fix_to_virt(idx);
|
||||||
|
|
||||||
|
#ifdef CONFIG_X86_64
|
||||||
|
/*
|
||||||
|
* Ensure that the static initial page tables are covering the
|
||||||
|
* fixmap completely.
|
||||||
|
*/
|
||||||
|
BUILD_BUG_ON(__end_of_permanent_fixed_addresses >
|
||||||
|
(FIXMAP_PMD_NUM * PTRS_PER_PTE));
|
||||||
|
#endif
|
||||||
|
|
||||||
if (idx >= __end_of_fixed_addresses) {
|
if (idx >= __end_of_fixed_addresses) {
|
||||||
BUG();
|
BUG();
|
||||||
return;
|
return;
|
||||||
|
@ -1907,7 +1907,7 @@ void __init xen_setup_kernel_pagetable(pgd_t *pgd, unsigned long max_pfn)
|
|||||||
/* L3_k[511] -> level2_fixmap_pgt */
|
/* L3_k[511] -> level2_fixmap_pgt */
|
||||||
convert_pfn_mfn(level3_kernel_pgt);
|
convert_pfn_mfn(level3_kernel_pgt);
|
||||||
|
|
||||||
/* L3_k[511][506] -> level1_fixmap_pgt */
|
/* L3_k[511][508-FIXMAP_PMD_NUM ... 507] -> level1_fixmap_pgt */
|
||||||
convert_pfn_mfn(level2_fixmap_pgt);
|
convert_pfn_mfn(level2_fixmap_pgt);
|
||||||
|
|
||||||
/* We get [511][511] and have Xen's version of level2_kernel_pgt */
|
/* We get [511][511] and have Xen's version of level2_kernel_pgt */
|
||||||
@ -1952,7 +1952,11 @@ void __init xen_setup_kernel_pagetable(pgd_t *pgd, unsigned long max_pfn)
|
|||||||
set_page_prot(level2_ident_pgt, PAGE_KERNEL_RO);
|
set_page_prot(level2_ident_pgt, PAGE_KERNEL_RO);
|
||||||
set_page_prot(level2_kernel_pgt, PAGE_KERNEL_RO);
|
set_page_prot(level2_kernel_pgt, PAGE_KERNEL_RO);
|
||||||
set_page_prot(level2_fixmap_pgt, PAGE_KERNEL_RO);
|
set_page_prot(level2_fixmap_pgt, PAGE_KERNEL_RO);
|
||||||
set_page_prot(level1_fixmap_pgt, PAGE_KERNEL_RO);
|
|
||||||
|
for (i = 0; i < FIXMAP_PMD_NUM; i++) {
|
||||||
|
set_page_prot(level1_fixmap_pgt + i * PTRS_PER_PTE,
|
||||||
|
PAGE_KERNEL_RO);
|
||||||
|
}
|
||||||
|
|
||||||
/* Pin down new L4 */
|
/* Pin down new L4 */
|
||||||
pin_pagetable_pfn(MMUEXT_PIN_L4_TABLE,
|
pin_pagetable_pfn(MMUEXT_PIN_L4_TABLE,
|
||||||
|
@ -478,7 +478,7 @@ static void xen_convert_regs(const struct xen_pmu_regs *xen_regs,
|
|||||||
irqreturn_t xen_pmu_irq_handler(int irq, void *dev_id)
|
irqreturn_t xen_pmu_irq_handler(int irq, void *dev_id)
|
||||||
{
|
{
|
||||||
int err, ret = IRQ_NONE;
|
int err, ret = IRQ_NONE;
|
||||||
struct pt_regs regs;
|
struct pt_regs regs = {0};
|
||||||
const struct xen_pmu_data *xenpmu_data = get_xenpmu_data();
|
const struct xen_pmu_data *xenpmu_data = get_xenpmu_data();
|
||||||
uint8_t xenpmu_flags = get_xenpmu_flags();
|
uint8_t xenpmu_flags = get_xenpmu_flags();
|
||||||
|
|
||||||
|
@ -1684,7 +1684,7 @@ void generic_end_io_acct(struct request_queue *q, int req_op,
|
|||||||
const int sgrp = op_stat_group(req_op);
|
const int sgrp = op_stat_group(req_op);
|
||||||
int cpu = part_stat_lock();
|
int cpu = part_stat_lock();
|
||||||
|
|
||||||
part_stat_add(cpu, part, ticks[sgrp], duration);
|
part_stat_add(cpu, part, nsecs[sgrp], jiffies_to_nsecs(duration));
|
||||||
part_round_stats(q, cpu, part);
|
part_round_stats(q, cpu, part);
|
||||||
part_dec_in_flight(q, part, op_is_write(req_op));
|
part_dec_in_flight(q, part, op_is_write(req_op));
|
||||||
|
|
||||||
|
@ -2733,17 +2733,15 @@ void blk_account_io_done(struct request *req, u64 now)
|
|||||||
* containing request is enough.
|
* containing request is enough.
|
||||||
*/
|
*/
|
||||||
if (blk_do_io_stat(req) && !(req->rq_flags & RQF_FLUSH_SEQ)) {
|
if (blk_do_io_stat(req) && !(req->rq_flags & RQF_FLUSH_SEQ)) {
|
||||||
unsigned long duration;
|
|
||||||
const int sgrp = op_stat_group(req_op(req));
|
const int sgrp = op_stat_group(req_op(req));
|
||||||
struct hd_struct *part;
|
struct hd_struct *part;
|
||||||
int cpu;
|
int cpu;
|
||||||
|
|
||||||
duration = nsecs_to_jiffies(now - req->start_time_ns);
|
|
||||||
cpu = part_stat_lock();
|
cpu = part_stat_lock();
|
||||||
part = req->part;
|
part = req->part;
|
||||||
|
|
||||||
part_stat_inc(cpu, part, ios[sgrp]);
|
part_stat_inc(cpu, part, ios[sgrp]);
|
||||||
part_stat_add(cpu, part, ticks[sgrp], duration);
|
part_stat_add(cpu, part, nsecs[sgrp], now - req->start_time_ns);
|
||||||
part_round_stats(req->q, cpu, part);
|
part_round_stats(req->q, cpu, part);
|
||||||
part_dec_in_flight(req->q, part, rq_data_dir(req));
|
part_dec_in_flight(req->q, part, rq_data_dir(req));
|
||||||
|
|
||||||
|
@ -1343,18 +1343,18 @@ static int diskstats_show(struct seq_file *seqf, void *v)
|
|||||||
part_stat_read(hd, ios[STAT_READ]),
|
part_stat_read(hd, ios[STAT_READ]),
|
||||||
part_stat_read(hd, merges[STAT_READ]),
|
part_stat_read(hd, merges[STAT_READ]),
|
||||||
part_stat_read(hd, sectors[STAT_READ]),
|
part_stat_read(hd, sectors[STAT_READ]),
|
||||||
jiffies_to_msecs(part_stat_read(hd, ticks[STAT_READ])),
|
(unsigned int)part_stat_read_msecs(hd, STAT_READ),
|
||||||
part_stat_read(hd, ios[STAT_WRITE]),
|
part_stat_read(hd, ios[STAT_WRITE]),
|
||||||
part_stat_read(hd, merges[STAT_WRITE]),
|
part_stat_read(hd, merges[STAT_WRITE]),
|
||||||
part_stat_read(hd, sectors[STAT_WRITE]),
|
part_stat_read(hd, sectors[STAT_WRITE]),
|
||||||
jiffies_to_msecs(part_stat_read(hd, ticks[STAT_WRITE])),
|
(unsigned int)part_stat_read_msecs(hd, STAT_WRITE),
|
||||||
inflight[0],
|
inflight[0],
|
||||||
jiffies_to_msecs(part_stat_read(hd, io_ticks)),
|
jiffies_to_msecs(part_stat_read(hd, io_ticks)),
|
||||||
jiffies_to_msecs(part_stat_read(hd, time_in_queue)),
|
jiffies_to_msecs(part_stat_read(hd, time_in_queue)),
|
||||||
part_stat_read(hd, ios[STAT_DISCARD]),
|
part_stat_read(hd, ios[STAT_DISCARD]),
|
||||||
part_stat_read(hd, merges[STAT_DISCARD]),
|
part_stat_read(hd, merges[STAT_DISCARD]),
|
||||||
part_stat_read(hd, sectors[STAT_DISCARD]),
|
part_stat_read(hd, sectors[STAT_DISCARD]),
|
||||||
jiffies_to_msecs(part_stat_read(hd, ticks[STAT_DISCARD]))
|
(unsigned int)part_stat_read_msecs(hd, STAT_DISCARD)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
disk_part_iter_exit(&piter);
|
disk_part_iter_exit(&piter);
|
||||||
|
@ -136,18 +136,18 @@ ssize_t part_stat_show(struct device *dev,
|
|||||||
part_stat_read(p, ios[STAT_READ]),
|
part_stat_read(p, ios[STAT_READ]),
|
||||||
part_stat_read(p, merges[STAT_READ]),
|
part_stat_read(p, merges[STAT_READ]),
|
||||||
(unsigned long long)part_stat_read(p, sectors[STAT_READ]),
|
(unsigned long long)part_stat_read(p, sectors[STAT_READ]),
|
||||||
jiffies_to_msecs(part_stat_read(p, ticks[STAT_READ])),
|
(unsigned int)part_stat_read_msecs(p, STAT_READ),
|
||||||
part_stat_read(p, ios[STAT_WRITE]),
|
part_stat_read(p, ios[STAT_WRITE]),
|
||||||
part_stat_read(p, merges[STAT_WRITE]),
|
part_stat_read(p, merges[STAT_WRITE]),
|
||||||
(unsigned long long)part_stat_read(p, sectors[STAT_WRITE]),
|
(unsigned long long)part_stat_read(p, sectors[STAT_WRITE]),
|
||||||
jiffies_to_msecs(part_stat_read(p, ticks[STAT_WRITE])),
|
(unsigned int)part_stat_read_msecs(p, STAT_WRITE),
|
||||||
inflight[0],
|
inflight[0],
|
||||||
jiffies_to_msecs(part_stat_read(p, io_ticks)),
|
jiffies_to_msecs(part_stat_read(p, io_ticks)),
|
||||||
jiffies_to_msecs(part_stat_read(p, time_in_queue)),
|
jiffies_to_msecs(part_stat_read(p, time_in_queue)),
|
||||||
part_stat_read(p, ios[STAT_DISCARD]),
|
part_stat_read(p, ios[STAT_DISCARD]),
|
||||||
part_stat_read(p, merges[STAT_DISCARD]),
|
part_stat_read(p, merges[STAT_DISCARD]),
|
||||||
(unsigned long long)part_stat_read(p, sectors[STAT_DISCARD]),
|
(unsigned long long)part_stat_read(p, sectors[STAT_DISCARD]),
|
||||||
jiffies_to_msecs(part_stat_read(p, ticks[STAT_DISCARD])));
|
(unsigned int)part_stat_read_msecs(p, STAT_DISCARD));
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t part_inflight_show(struct device *dev, struct device_attribute *attr,
|
ssize_t part_inflight_show(struct device *dev, struct device_attribute *attr,
|
||||||
|
@ -5359,10 +5359,20 @@ void ata_qc_complete(struct ata_queued_cmd *qc)
|
|||||||
*/
|
*/
|
||||||
int ata_qc_complete_multiple(struct ata_port *ap, u64 qc_active)
|
int ata_qc_complete_multiple(struct ata_port *ap, u64 qc_active)
|
||||||
{
|
{
|
||||||
|
u64 done_mask, ap_qc_active = ap->qc_active;
|
||||||
int nr_done = 0;
|
int nr_done = 0;
|
||||||
u64 done_mask;
|
|
||||||
|
|
||||||
done_mask = ap->qc_active ^ qc_active;
|
/*
|
||||||
|
* If the internal tag is set on ap->qc_active, then we care about
|
||||||
|
* bit0 on the passed in qc_active mask. Move that bit up to match
|
||||||
|
* the internal tag.
|
||||||
|
*/
|
||||||
|
if (ap_qc_active & (1ULL << ATA_TAG_INTERNAL)) {
|
||||||
|
qc_active |= (qc_active & 0x01) << ATA_TAG_INTERNAL;
|
||||||
|
qc_active ^= qc_active & 0x01;
|
||||||
|
}
|
||||||
|
|
||||||
|
done_mask = ap_qc_active ^ qc_active;
|
||||||
|
|
||||||
if (unlikely(done_mask & qc_active)) {
|
if (unlikely(done_mask & qc_active)) {
|
||||||
ata_port_err(ap, "illegal qc_active transition (%08llx->%08llx)\n",
|
ata_port_err(ap, "illegal qc_active transition (%08llx->%08llx)\n",
|
||||||
|
@ -3467,6 +3467,9 @@ static int fd_locked_ioctl(struct block_device *bdev, fmode_t mode, unsigned int
|
|||||||
(struct floppy_struct **)&outparam);
|
(struct floppy_struct **)&outparam);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
memcpy(&inparam.g, outparam,
|
||||||
|
offsetof(struct floppy_struct, name));
|
||||||
|
outparam = &inparam.g;
|
||||||
break;
|
break;
|
||||||
case FDMSGON:
|
case FDMSGON:
|
||||||
UDP->flags |= FTD_MSG;
|
UDP->flags |= FTD_MSG;
|
||||||
|
@ -38,6 +38,17 @@ static DEFINE_MUTEX(sev_cmd_mutex);
|
|||||||
static struct sev_misc_dev *misc_dev;
|
static struct sev_misc_dev *misc_dev;
|
||||||
static struct psp_device *psp_master;
|
static struct psp_device *psp_master;
|
||||||
|
|
||||||
|
static int psp_cmd_timeout = 100;
|
||||||
|
module_param(psp_cmd_timeout, int, 0644);
|
||||||
|
MODULE_PARM_DESC(psp_cmd_timeout, " default timeout value, in seconds, for PSP commands");
|
||||||
|
|
||||||
|
static int psp_probe_timeout = 5;
|
||||||
|
module_param(psp_probe_timeout, int, 0644);
|
||||||
|
MODULE_PARM_DESC(psp_probe_timeout, " default timeout value, in seconds, during PSP device probe");
|
||||||
|
|
||||||
|
static bool psp_dead;
|
||||||
|
static int psp_timeout;
|
||||||
|
|
||||||
static struct psp_device *psp_alloc_struct(struct sp_device *sp)
|
static struct psp_device *psp_alloc_struct(struct sp_device *sp)
|
||||||
{
|
{
|
||||||
struct device *dev = sp->dev;
|
struct device *dev = sp->dev;
|
||||||
@ -82,10 +93,19 @@ done:
|
|||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sev_wait_cmd_ioc(struct psp_device *psp, unsigned int *reg)
|
static int sev_wait_cmd_ioc(struct psp_device *psp,
|
||||||
|
unsigned int *reg, unsigned int timeout)
|
||||||
{
|
{
|
||||||
wait_event(psp->sev_int_queue, psp->sev_int_rcvd);
|
int ret;
|
||||||
|
|
||||||
|
ret = wait_event_timeout(psp->sev_int_queue,
|
||||||
|
psp->sev_int_rcvd, timeout * HZ);
|
||||||
|
if (!ret)
|
||||||
|
return -ETIMEDOUT;
|
||||||
|
|
||||||
*reg = ioread32(psp->io_regs + psp->vdata->cmdresp_reg);
|
*reg = ioread32(psp->io_regs + psp->vdata->cmdresp_reg);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int sev_cmd_buffer_len(int cmd)
|
static int sev_cmd_buffer_len(int cmd)
|
||||||
@ -133,12 +153,15 @@ static int __sev_do_cmd_locked(int cmd, void *data, int *psp_ret)
|
|||||||
if (!psp)
|
if (!psp)
|
||||||
return -ENODEV;
|
return -ENODEV;
|
||||||
|
|
||||||
|
if (psp_dead)
|
||||||
|
return -EBUSY;
|
||||||
|
|
||||||
/* Get the physical address of the command buffer */
|
/* Get the physical address of the command buffer */
|
||||||
phys_lsb = data ? lower_32_bits(__psp_pa(data)) : 0;
|
phys_lsb = data ? lower_32_bits(__psp_pa(data)) : 0;
|
||||||
phys_msb = data ? upper_32_bits(__psp_pa(data)) : 0;
|
phys_msb = data ? upper_32_bits(__psp_pa(data)) : 0;
|
||||||
|
|
||||||
dev_dbg(psp->dev, "sev command id %#x buffer 0x%08x%08x\n",
|
dev_dbg(psp->dev, "sev command id %#x buffer 0x%08x%08x timeout %us\n",
|
||||||
cmd, phys_msb, phys_lsb);
|
cmd, phys_msb, phys_lsb, psp_timeout);
|
||||||
|
|
||||||
print_hex_dump_debug("(in): ", DUMP_PREFIX_OFFSET, 16, 2, data,
|
print_hex_dump_debug("(in): ", DUMP_PREFIX_OFFSET, 16, 2, data,
|
||||||
sev_cmd_buffer_len(cmd), false);
|
sev_cmd_buffer_len(cmd), false);
|
||||||
@ -154,7 +177,18 @@ static int __sev_do_cmd_locked(int cmd, void *data, int *psp_ret)
|
|||||||
iowrite32(reg, psp->io_regs + psp->vdata->cmdresp_reg);
|
iowrite32(reg, psp->io_regs + psp->vdata->cmdresp_reg);
|
||||||
|
|
||||||
/* wait for command completion */
|
/* wait for command completion */
|
||||||
sev_wait_cmd_ioc(psp, ®);
|
ret = sev_wait_cmd_ioc(psp, ®, psp_timeout);
|
||||||
|
if (ret) {
|
||||||
|
if (psp_ret)
|
||||||
|
*psp_ret = 0;
|
||||||
|
|
||||||
|
dev_err(psp->dev, "sev command %#x timed out, disabling PSP \n", cmd);
|
||||||
|
psp_dead = true;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
psp_timeout = psp_cmd_timeout;
|
||||||
|
|
||||||
if (psp_ret)
|
if (psp_ret)
|
||||||
*psp_ret = reg & PSP_CMDRESP_ERR_MASK;
|
*psp_ret = reg & PSP_CMDRESP_ERR_MASK;
|
||||||
@ -888,6 +922,8 @@ void psp_pci_init(void)
|
|||||||
|
|
||||||
psp_master = sp->psp_data;
|
psp_master = sp->psp_data;
|
||||||
|
|
||||||
|
psp_timeout = psp_probe_timeout;
|
||||||
|
|
||||||
if (sev_get_api_version())
|
if (sev_get_api_version())
|
||||||
goto err;
|
goto err;
|
||||||
|
|
||||||
|
@ -90,14 +90,17 @@ config EFI_ARMSTUB
|
|||||||
config EFI_ARMSTUB_DTB_LOADER
|
config EFI_ARMSTUB_DTB_LOADER
|
||||||
bool "Enable the DTB loader"
|
bool "Enable the DTB loader"
|
||||||
depends on EFI_ARMSTUB
|
depends on EFI_ARMSTUB
|
||||||
|
default y
|
||||||
help
|
help
|
||||||
Select this config option to add support for the dtb= command
|
Select this config option to add support for the dtb= command
|
||||||
line parameter, allowing a device tree blob to be loaded into
|
line parameter, allowing a device tree blob to be loaded into
|
||||||
memory from the EFI System Partition by the stub.
|
memory from the EFI System Partition by the stub.
|
||||||
|
|
||||||
The device tree is typically provided by the platform or by
|
If the device tree is provided by the platform or by
|
||||||
the bootloader, so this option is mostly for development
|
the bootloader this option may not be needed.
|
||||||
purposes only.
|
But, for various development reasons and to maintain existing
|
||||||
|
functionality for bootloaders that do not have such support
|
||||||
|
this option is necessary.
|
||||||
|
|
||||||
config EFI_BOOTLOADER_CONTROL
|
config EFI_BOOTLOADER_CONTROL
|
||||||
tristate "EFI Bootloader Control"
|
tristate "EFI Bootloader Control"
|
||||||
|
@ -272,7 +272,7 @@ void amdgpu_amdkfd_gpu_reset(struct kgd_dev *kgd)
|
|||||||
|
|
||||||
int alloc_gtt_mem(struct kgd_dev *kgd, size_t size,
|
int alloc_gtt_mem(struct kgd_dev *kgd, size_t size,
|
||||||
void **mem_obj, uint64_t *gpu_addr,
|
void **mem_obj, uint64_t *gpu_addr,
|
||||||
void **cpu_ptr)
|
void **cpu_ptr, bool mqd_gfx9)
|
||||||
{
|
{
|
||||||
struct amdgpu_device *adev = (struct amdgpu_device *)kgd;
|
struct amdgpu_device *adev = (struct amdgpu_device *)kgd;
|
||||||
struct amdgpu_bo *bo = NULL;
|
struct amdgpu_bo *bo = NULL;
|
||||||
@ -287,6 +287,10 @@ int alloc_gtt_mem(struct kgd_dev *kgd, size_t size,
|
|||||||
bp.flags = AMDGPU_GEM_CREATE_CPU_GTT_USWC;
|
bp.flags = AMDGPU_GEM_CREATE_CPU_GTT_USWC;
|
||||||
bp.type = ttm_bo_type_kernel;
|
bp.type = ttm_bo_type_kernel;
|
||||||
bp.resv = NULL;
|
bp.resv = NULL;
|
||||||
|
|
||||||
|
if (mqd_gfx9)
|
||||||
|
bp.flags |= AMDGPU_GEM_CREATE_MQD_GFX9;
|
||||||
|
|
||||||
r = amdgpu_bo_create(adev, &bp, &bo);
|
r = amdgpu_bo_create(adev, &bp, &bo);
|
||||||
if (r) {
|
if (r) {
|
||||||
dev_err(adev->dev,
|
dev_err(adev->dev,
|
||||||
|
@ -136,7 +136,7 @@ void amdgpu_amdkfd_gpu_reset(struct kgd_dev *kgd);
|
|||||||
/* Shared API */
|
/* Shared API */
|
||||||
int alloc_gtt_mem(struct kgd_dev *kgd, size_t size,
|
int alloc_gtt_mem(struct kgd_dev *kgd, size_t size,
|
||||||
void **mem_obj, uint64_t *gpu_addr,
|
void **mem_obj, uint64_t *gpu_addr,
|
||||||
void **cpu_ptr);
|
void **cpu_ptr, bool mqd_gfx9);
|
||||||
void free_gtt_mem(struct kgd_dev *kgd, void *mem_obj);
|
void free_gtt_mem(struct kgd_dev *kgd, void *mem_obj);
|
||||||
void get_local_mem_info(struct kgd_dev *kgd,
|
void get_local_mem_info(struct kgd_dev *kgd,
|
||||||
struct kfd_local_mem_info *mem_info);
|
struct kfd_local_mem_info *mem_info);
|
||||||
|
@ -685,7 +685,7 @@ static int kgd_hqd_sdma_destroy(struct kgd_dev *kgd, void *mqd,
|
|||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
temp = RREG32(sdma_base_addr + mmSDMA0_RLC0_CONTEXT_STATUS);
|
temp = RREG32(sdma_base_addr + mmSDMA0_RLC0_CONTEXT_STATUS);
|
||||||
if (temp & SDMA0_STATUS_REG__RB_CMD_IDLE__SHIFT)
|
if (temp & SDMA0_RLC0_CONTEXT_STATUS__IDLE_MASK)
|
||||||
break;
|
break;
|
||||||
if (time_after(jiffies, end_jiffies))
|
if (time_after(jiffies, end_jiffies))
|
||||||
return -ETIME;
|
return -ETIME;
|
||||||
|
@ -367,12 +367,14 @@ static int amdgpu_cgs_get_firmware_info(struct cgs_device *cgs_device,
|
|||||||
break;
|
break;
|
||||||
case CHIP_POLARIS10:
|
case CHIP_POLARIS10:
|
||||||
if (type == CGS_UCODE_ID_SMU) {
|
if (type == CGS_UCODE_ID_SMU) {
|
||||||
if ((adev->pdev->device == 0x67df) &&
|
if (((adev->pdev->device == 0x67df) &&
|
||||||
((adev->pdev->revision == 0xe0) ||
|
((adev->pdev->revision == 0xe0) ||
|
||||||
(adev->pdev->revision == 0xe3) ||
|
(adev->pdev->revision == 0xe3) ||
|
||||||
(adev->pdev->revision == 0xe4) ||
|
(adev->pdev->revision == 0xe4) ||
|
||||||
(adev->pdev->revision == 0xe5) ||
|
(adev->pdev->revision == 0xe5) ||
|
||||||
(adev->pdev->revision == 0xe7) ||
|
(adev->pdev->revision == 0xe7) ||
|
||||||
|
(adev->pdev->revision == 0xef))) ||
|
||||||
|
((adev->pdev->device == 0x6fdf) &&
|
||||||
(adev->pdev->revision == 0xef))) {
|
(adev->pdev->revision == 0xef))) {
|
||||||
info->is_kicker = true;
|
info->is_kicker = true;
|
||||||
strcpy(fw_name, "amdgpu/polaris10_k_smc.bin");
|
strcpy(fw_name, "amdgpu/polaris10_k_smc.bin");
|
||||||
|
@ -740,6 +740,7 @@ static const struct pci_device_id pciidlist[] = {
|
|||||||
{0x1002, 0x67CA, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
{0x1002, 0x67CA, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
||||||
{0x1002, 0x67CC, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
{0x1002, 0x67CC, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
||||||
{0x1002, 0x67CF, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
{0x1002, 0x67CF, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
||||||
|
{0x1002, 0x6FDF, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS10},
|
||||||
/* Polaris12 */
|
/* Polaris12 */
|
||||||
{0x1002, 0x6980, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS12},
|
{0x1002, 0x6980, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS12},
|
||||||
{0x1002, 0x6981, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS12},
|
{0x1002, 0x6981, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_POLARIS12},
|
||||||
|
@ -457,7 +457,8 @@ bool kgd2kfd_device_init(struct kfd_dev *kfd,
|
|||||||
|
|
||||||
if (kfd->kfd2kgd->init_gtt_mem_allocation(
|
if (kfd->kfd2kgd->init_gtt_mem_allocation(
|
||||||
kfd->kgd, size, &kfd->gtt_mem,
|
kfd->kgd, size, &kfd->gtt_mem,
|
||||||
&kfd->gtt_start_gpu_addr, &kfd->gtt_start_cpu_ptr)){
|
&kfd->gtt_start_gpu_addr, &kfd->gtt_start_cpu_ptr,
|
||||||
|
false)) {
|
||||||
dev_err(kfd_device, "Could not allocate %d bytes\n", size);
|
dev_err(kfd_device, "Could not allocate %d bytes\n", size);
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
@ -62,9 +62,20 @@ int kfd_iommu_device_init(struct kfd_dev *kfd)
|
|||||||
struct amd_iommu_device_info iommu_info;
|
struct amd_iommu_device_info iommu_info;
|
||||||
unsigned int pasid_limit;
|
unsigned int pasid_limit;
|
||||||
int err;
|
int err;
|
||||||
|
struct kfd_topology_device *top_dev;
|
||||||
|
|
||||||
if (!kfd->device_info->needs_iommu_device)
|
top_dev = kfd_topology_device_by_id(kfd->id);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Overwrite ATS capability according to needs_iommu_device to fix
|
||||||
|
* potential missing corresponding bit in CRAT of BIOS.
|
||||||
|
*/
|
||||||
|
if (!kfd->device_info->needs_iommu_device) {
|
||||||
|
top_dev->node_props.capability &= ~HSA_CAP_ATS_PRESENT;
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
top_dev->node_props.capability |= HSA_CAP_ATS_PRESENT;
|
||||||
|
|
||||||
iommu_info.flags = 0;
|
iommu_info.flags = 0;
|
||||||
err = amd_iommu_device_info(kfd->pdev, &iommu_info);
|
err = amd_iommu_device_info(kfd->pdev, &iommu_info);
|
||||||
|
@ -88,7 +88,7 @@ static int init_mqd(struct mqd_manager *mm, void **mqd,
|
|||||||
ALIGN(sizeof(struct v9_mqd), PAGE_SIZE),
|
ALIGN(sizeof(struct v9_mqd), PAGE_SIZE),
|
||||||
&((*mqd_mem_obj)->gtt_mem),
|
&((*mqd_mem_obj)->gtt_mem),
|
||||||
&((*mqd_mem_obj)->gpu_addr),
|
&((*mqd_mem_obj)->gpu_addr),
|
||||||
(void *)&((*mqd_mem_obj)->cpu_ptr));
|
(void *)&((*mqd_mem_obj)->cpu_ptr), true);
|
||||||
} else
|
} else
|
||||||
retval = kfd_gtt_sa_allocate(mm->dev, sizeof(struct v9_mqd),
|
retval = kfd_gtt_sa_allocate(mm->dev, sizeof(struct v9_mqd),
|
||||||
mqd_mem_obj);
|
mqd_mem_obj);
|
||||||
|
@ -806,6 +806,7 @@ int kfd_topology_add_device(struct kfd_dev *gpu);
|
|||||||
int kfd_topology_remove_device(struct kfd_dev *gpu);
|
int kfd_topology_remove_device(struct kfd_dev *gpu);
|
||||||
struct kfd_topology_device *kfd_topology_device_by_proximity_domain(
|
struct kfd_topology_device *kfd_topology_device_by_proximity_domain(
|
||||||
uint32_t proximity_domain);
|
uint32_t proximity_domain);
|
||||||
|
struct kfd_topology_device *kfd_topology_device_by_id(uint32_t gpu_id);
|
||||||
struct kfd_dev *kfd_device_by_id(uint32_t gpu_id);
|
struct kfd_dev *kfd_device_by_id(uint32_t gpu_id);
|
||||||
struct kfd_dev *kfd_device_by_pci_dev(const struct pci_dev *pdev);
|
struct kfd_dev *kfd_device_by_pci_dev(const struct pci_dev *pdev);
|
||||||
int kfd_topology_enum_kfd_devices(uint8_t idx, struct kfd_dev **kdev);
|
int kfd_topology_enum_kfd_devices(uint8_t idx, struct kfd_dev **kdev);
|
||||||
|
@ -63,22 +63,33 @@ struct kfd_topology_device *kfd_topology_device_by_proximity_domain(
|
|||||||
return device;
|
return device;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct kfd_dev *kfd_device_by_id(uint32_t gpu_id)
|
struct kfd_topology_device *kfd_topology_device_by_id(uint32_t gpu_id)
|
||||||
{
|
{
|
||||||
struct kfd_topology_device *top_dev;
|
struct kfd_topology_device *top_dev = NULL;
|
||||||
struct kfd_dev *device = NULL;
|
struct kfd_topology_device *ret = NULL;
|
||||||
|
|
||||||
down_read(&topology_lock);
|
down_read(&topology_lock);
|
||||||
|
|
||||||
list_for_each_entry(top_dev, &topology_device_list, list)
|
list_for_each_entry(top_dev, &topology_device_list, list)
|
||||||
if (top_dev->gpu_id == gpu_id) {
|
if (top_dev->gpu_id == gpu_id) {
|
||||||
device = top_dev->gpu;
|
ret = top_dev;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
up_read(&topology_lock);
|
up_read(&topology_lock);
|
||||||
|
|
||||||
return device;
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct kfd_dev *kfd_device_by_id(uint32_t gpu_id)
|
||||||
|
{
|
||||||
|
struct kfd_topology_device *top_dev;
|
||||||
|
|
||||||
|
top_dev = kfd_topology_device_by_id(gpu_id);
|
||||||
|
if (!top_dev)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
return top_dev->gpu;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct kfd_dev *kfd_device_by_pci_dev(const struct pci_dev *pdev)
|
struct kfd_dev *kfd_device_by_pci_dev(const struct pci_dev *pdev)
|
||||||
|
@ -292,7 +292,7 @@ struct tile_config {
|
|||||||
struct kfd2kgd_calls {
|
struct kfd2kgd_calls {
|
||||||
int (*init_gtt_mem_allocation)(struct kgd_dev *kgd, size_t size,
|
int (*init_gtt_mem_allocation)(struct kgd_dev *kgd, size_t size,
|
||||||
void **mem_obj, uint64_t *gpu_addr,
|
void **mem_obj, uint64_t *gpu_addr,
|
||||||
void **cpu_ptr);
|
void **cpu_ptr, bool mqd_gfx9);
|
||||||
|
|
||||||
void (*free_gtt_mem)(struct kgd_dev *kgd, void *mem_obj);
|
void (*free_gtt_mem)(struct kgd_dev *kgd, void *mem_obj);
|
||||||
|
|
||||||
|
@ -2067,7 +2067,7 @@ static void __drm_state_dump(struct drm_device *dev, struct drm_printer *p,
|
|||||||
struct drm_connector *connector;
|
struct drm_connector *connector;
|
||||||
struct drm_connector_list_iter conn_iter;
|
struct drm_connector_list_iter conn_iter;
|
||||||
|
|
||||||
if (!drm_core_check_feature(dev, DRIVER_ATOMIC))
|
if (!drm_drv_uses_atomic_modeset(dev))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
list_for_each_entry(plane, &config->plane_list, head) {
|
list_for_each_entry(plane, &config->plane_list, head) {
|
||||||
|
@ -151,7 +151,7 @@ int drm_debugfs_init(struct drm_minor *minor, int minor_id,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (drm_core_check_feature(dev, DRIVER_ATOMIC)) {
|
if (drm_drv_uses_atomic_modeset(dev)) {
|
||||||
ret = drm_atomic_debugfs_init(minor);
|
ret = drm_atomic_debugfs_init(minor);
|
||||||
if (ret) {
|
if (ret) {
|
||||||
DRM_ERROR("Failed to create atomic debugfs files\n");
|
DRM_ERROR("Failed to create atomic debugfs files\n");
|
||||||
|
@ -2370,7 +2370,6 @@ static int drm_pick_crtcs(struct drm_fb_helper *fb_helper,
|
|||||||
{
|
{
|
||||||
int c, o;
|
int c, o;
|
||||||
struct drm_connector *connector;
|
struct drm_connector *connector;
|
||||||
const struct drm_connector_helper_funcs *connector_funcs;
|
|
||||||
int my_score, best_score, score;
|
int my_score, best_score, score;
|
||||||
struct drm_fb_helper_crtc **crtcs, *crtc;
|
struct drm_fb_helper_crtc **crtcs, *crtc;
|
||||||
struct drm_fb_helper_connector *fb_helper_conn;
|
struct drm_fb_helper_connector *fb_helper_conn;
|
||||||
@ -2399,8 +2398,6 @@ static int drm_pick_crtcs(struct drm_fb_helper *fb_helper,
|
|||||||
if (drm_has_preferred_mode(fb_helper_conn, width, height))
|
if (drm_has_preferred_mode(fb_helper_conn, width, height))
|
||||||
my_score++;
|
my_score++;
|
||||||
|
|
||||||
connector_funcs = connector->helper_private;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* select a crtc for this connector and then attempt to configure
|
* select a crtc for this connector and then attempt to configure
|
||||||
* remaining connectors
|
* remaining connectors
|
||||||
|
@ -3210,6 +3210,7 @@ static int init_bxt_mmio_info(struct intel_gvt *gvt)
|
|||||||
MMIO_D(BXT_DSI_PLL_ENABLE, D_BXT);
|
MMIO_D(BXT_DSI_PLL_ENABLE, D_BXT);
|
||||||
|
|
||||||
MMIO_D(GEN9_CLKGATE_DIS_0, D_BXT);
|
MMIO_D(GEN9_CLKGATE_DIS_0, D_BXT);
|
||||||
|
MMIO_D(GEN9_CLKGATE_DIS_4, D_BXT);
|
||||||
|
|
||||||
MMIO_D(HSW_TVIDEO_DIP_GCP(TRANSCODER_A), D_BXT);
|
MMIO_D(HSW_TVIDEO_DIP_GCP(TRANSCODER_A), D_BXT);
|
||||||
MMIO_D(HSW_TVIDEO_DIP_GCP(TRANSCODER_B), D_BXT);
|
MMIO_D(HSW_TVIDEO_DIP_GCP(TRANSCODER_B), D_BXT);
|
||||||
|
@ -1833,6 +1833,8 @@ static bool kvmgt_is_valid_gfn(unsigned long handle, unsigned long gfn)
|
|||||||
{
|
{
|
||||||
struct kvmgt_guest_info *info;
|
struct kvmgt_guest_info *info;
|
||||||
struct kvm *kvm;
|
struct kvm *kvm;
|
||||||
|
int idx;
|
||||||
|
bool ret;
|
||||||
|
|
||||||
if (!handle_valid(handle))
|
if (!handle_valid(handle))
|
||||||
return false;
|
return false;
|
||||||
@ -1840,8 +1842,11 @@ static bool kvmgt_is_valid_gfn(unsigned long handle, unsigned long gfn)
|
|||||||
info = (struct kvmgt_guest_info *)handle;
|
info = (struct kvmgt_guest_info *)handle;
|
||||||
kvm = info->kvm;
|
kvm = info->kvm;
|
||||||
|
|
||||||
return kvm_is_visible_gfn(kvm, gfn);
|
idx = srcu_read_lock(&kvm->srcu);
|
||||||
|
ret = kvm_is_visible_gfn(kvm, gfn);
|
||||||
|
srcu_read_unlock(&kvm->srcu, idx);
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct intel_gvt_mpt kvmgt_mpt = {
|
struct intel_gvt_mpt kvmgt_mpt = {
|
||||||
|
@ -244,6 +244,34 @@ void intel_vgpu_reset_mmio(struct intel_vgpu *vgpu, bool dmlr)
|
|||||||
|
|
||||||
/* set the bit 0:2(Core C-State ) to C0 */
|
/* set the bit 0:2(Core C-State ) to C0 */
|
||||||
vgpu_vreg_t(vgpu, GEN6_GT_CORE_STATUS) = 0;
|
vgpu_vreg_t(vgpu, GEN6_GT_CORE_STATUS) = 0;
|
||||||
|
|
||||||
|
if (IS_BROXTON(vgpu->gvt->dev_priv)) {
|
||||||
|
vgpu_vreg_t(vgpu, BXT_P_CR_GT_DISP_PWRON) &=
|
||||||
|
~(BIT(0) | BIT(1));
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PORT_CL1CM_DW0(DPIO_PHY0)) &=
|
||||||
|
~PHY_POWER_GOOD;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PORT_CL1CM_DW0(DPIO_PHY1)) &=
|
||||||
|
~PHY_POWER_GOOD;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL_FAMILY(DPIO_PHY0)) &=
|
||||||
|
~BIT(30);
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL_FAMILY(DPIO_PHY1)) &=
|
||||||
|
~BIT(30);
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_A)) &=
|
||||||
|
~BXT_PHY_LANE_ENABLED;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_A)) |=
|
||||||
|
BXT_PHY_CMNLANE_POWERDOWN_ACK |
|
||||||
|
BXT_PHY_LANE_POWERDOWN_ACK;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_B)) &=
|
||||||
|
~BXT_PHY_LANE_ENABLED;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_B)) |=
|
||||||
|
BXT_PHY_CMNLANE_POWERDOWN_ACK |
|
||||||
|
BXT_PHY_LANE_POWERDOWN_ACK;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_C)) &=
|
||||||
|
~BXT_PHY_LANE_ENABLED;
|
||||||
|
vgpu_vreg_t(vgpu, BXT_PHY_CTL(PORT_C)) |=
|
||||||
|
BXT_PHY_CMNLANE_POWERDOWN_ACK |
|
||||||
|
BXT_PHY_LANE_POWERDOWN_ACK;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
#define GVT_GEN8_MMIO_RESET_OFFSET (0x44200)
|
#define GVT_GEN8_MMIO_RESET_OFFSET (0x44200)
|
||||||
/* only reset the engine related, so starting with 0x44200
|
/* only reset the engine related, so starting with 0x44200
|
||||||
|
@ -281,6 +281,7 @@ void intel_gvt_destroy_vgpu(struct intel_vgpu *vgpu)
|
|||||||
intel_vgpu_clean_submission(vgpu);
|
intel_vgpu_clean_submission(vgpu);
|
||||||
intel_vgpu_clean_display(vgpu);
|
intel_vgpu_clean_display(vgpu);
|
||||||
intel_vgpu_clean_opregion(vgpu);
|
intel_vgpu_clean_opregion(vgpu);
|
||||||
|
intel_vgpu_reset_ggtt(vgpu, true);
|
||||||
intel_vgpu_clean_gtt(vgpu);
|
intel_vgpu_clean_gtt(vgpu);
|
||||||
intel_gvt_hypervisor_detach_vgpu(vgpu);
|
intel_gvt_hypervisor_detach_vgpu(vgpu);
|
||||||
intel_vgpu_free_resource(vgpu);
|
intel_vgpu_free_resource(vgpu);
|
||||||
|
@ -111,7 +111,8 @@ static int vexpress_muxfpga_probe(struct platform_device *pdev)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static const struct of_device_id vexpress_muxfpga_match[] = {
|
static const struct of_device_id vexpress_muxfpga_match[] = {
|
||||||
{ .compatible = "arm,vexpress-muxfpga", }
|
{ .compatible = "arm,vexpress-muxfpga", },
|
||||||
|
{}
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct platform_driver vexpress_muxfpga_driver = {
|
static struct platform_driver vexpress_muxfpga_driver = {
|
||||||
|
@ -418,7 +418,6 @@ static const struct of_device_id sun4i_drv_of_table[] = {
|
|||||||
{ .compatible = "allwinner,sun8i-a33-display-engine" },
|
{ .compatible = "allwinner,sun8i-a33-display-engine" },
|
||||||
{ .compatible = "allwinner,sun8i-a83t-display-engine" },
|
{ .compatible = "allwinner,sun8i-a83t-display-engine" },
|
||||||
{ .compatible = "allwinner,sun8i-h3-display-engine" },
|
{ .compatible = "allwinner,sun8i-h3-display-engine" },
|
||||||
{ .compatible = "allwinner,sun8i-r40-display-engine" },
|
|
||||||
{ .compatible = "allwinner,sun8i-v3s-display-engine" },
|
{ .compatible = "allwinner,sun8i-v3s-display-engine" },
|
||||||
{ .compatible = "allwinner,sun9i-a80-display-engine" },
|
{ .compatible = "allwinner,sun9i-a80-display-engine" },
|
||||||
{ }
|
{ }
|
||||||
|
@ -398,7 +398,6 @@ static struct regmap_config sun8i_hdmi_phy_regmap_config = {
|
|||||||
|
|
||||||
static const struct sun8i_hdmi_phy_variant sun50i_a64_hdmi_phy = {
|
static const struct sun8i_hdmi_phy_variant sun50i_a64_hdmi_phy = {
|
||||||
.has_phy_clk = true,
|
.has_phy_clk = true,
|
||||||
.has_second_pll = true,
|
|
||||||
.phy_init = &sun8i_hdmi_phy_init_h3,
|
.phy_init = &sun8i_hdmi_phy_init_h3,
|
||||||
.phy_disable = &sun8i_hdmi_phy_disable_h3,
|
.phy_disable = &sun8i_hdmi_phy_disable_h3,
|
||||||
.phy_config = &sun8i_hdmi_phy_config_h3,
|
.phy_config = &sun8i_hdmi_phy_config_h3,
|
||||||
|
@ -545,22 +545,6 @@ static const struct sun8i_mixer_cfg sun8i_h3_mixer0_cfg = {
|
|||||||
.vi_num = 1,
|
.vi_num = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
static const struct sun8i_mixer_cfg sun8i_r40_mixer0_cfg = {
|
|
||||||
.ccsc = 0,
|
|
||||||
.mod_rate = 297000000,
|
|
||||||
.scaler_mask = 0xf,
|
|
||||||
.ui_num = 3,
|
|
||||||
.vi_num = 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct sun8i_mixer_cfg sun8i_r40_mixer1_cfg = {
|
|
||||||
.ccsc = 1,
|
|
||||||
.mod_rate = 297000000,
|
|
||||||
.scaler_mask = 0x3,
|
|
||||||
.ui_num = 1,
|
|
||||||
.vi_num = 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct sun8i_mixer_cfg sun8i_v3s_mixer_cfg = {
|
static const struct sun8i_mixer_cfg sun8i_v3s_mixer_cfg = {
|
||||||
.vi_num = 2,
|
.vi_num = 2,
|
||||||
.ui_num = 1,
|
.ui_num = 1,
|
||||||
@ -582,14 +566,6 @@ static const struct of_device_id sun8i_mixer_of_table[] = {
|
|||||||
.compatible = "allwinner,sun8i-h3-de2-mixer-0",
|
.compatible = "allwinner,sun8i-h3-de2-mixer-0",
|
||||||
.data = &sun8i_h3_mixer0_cfg,
|
.data = &sun8i_h3_mixer0_cfg,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
.compatible = "allwinner,sun8i-r40-de2-mixer-0",
|
|
||||||
.data = &sun8i_r40_mixer0_cfg,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
.compatible = "allwinner,sun8i-r40-de2-mixer-1",
|
|
||||||
.data = &sun8i_r40_mixer1_cfg,
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
.compatible = "allwinner,sun8i-v3s-de2-mixer",
|
.compatible = "allwinner,sun8i-v3s-de2-mixer",
|
||||||
.data = &sun8i_v3s_mixer_cfg,
|
.data = &sun8i_v3s_mixer_cfg,
|
||||||
|
@ -253,7 +253,6 @@ static int sun8i_tcon_top_remove(struct platform_device *pdev)
|
|||||||
|
|
||||||
/* sun4i_drv uses this list to check if a device node is a TCON TOP */
|
/* sun4i_drv uses this list to check if a device node is a TCON TOP */
|
||||||
const struct of_device_id sun8i_tcon_top_of_table[] = {
|
const struct of_device_id sun8i_tcon_top_of_table[] = {
|
||||||
{ .compatible = "allwinner,sun8i-r40-tcon-top" },
|
|
||||||
{ /* sentinel */ }
|
{ /* sentinel */ }
|
||||||
};
|
};
|
||||||
MODULE_DEVICE_TABLE(of, sun8i_tcon_top_of_table);
|
MODULE_DEVICE_TABLE(of, sun8i_tcon_top_of_table);
|
||||||
|
@ -432,9 +432,11 @@ static void udl_fbdev_destroy(struct drm_device *dev,
|
|||||||
{
|
{
|
||||||
drm_fb_helper_unregister_fbi(&ufbdev->helper);
|
drm_fb_helper_unregister_fbi(&ufbdev->helper);
|
||||||
drm_fb_helper_fini(&ufbdev->helper);
|
drm_fb_helper_fini(&ufbdev->helper);
|
||||||
drm_framebuffer_unregister_private(&ufbdev->ufb.base);
|
if (ufbdev->ufb.obj) {
|
||||||
drm_framebuffer_cleanup(&ufbdev->ufb.base);
|
drm_framebuffer_unregister_private(&ufbdev->ufb.base);
|
||||||
drm_gem_object_put_unlocked(&ufbdev->ufb.obj->base);
|
drm_framebuffer_cleanup(&ufbdev->ufb.base);
|
||||||
|
drm_gem_object_put_unlocked(&ufbdev->ufb.obj->base);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int udl_fbdev_init(struct drm_device *dev)
|
int udl_fbdev_init(struct drm_device *dev)
|
||||||
|
@ -297,6 +297,9 @@ static int vc4_plane_setup_clipping_and_scaling(struct drm_plane_state *state)
|
|||||||
vc4_state->y_scaling[0] = vc4_get_scaling_mode(vc4_state->src_h[0],
|
vc4_state->y_scaling[0] = vc4_get_scaling_mode(vc4_state->src_h[0],
|
||||||
vc4_state->crtc_h);
|
vc4_state->crtc_h);
|
||||||
|
|
||||||
|
vc4_state->is_unity = (vc4_state->x_scaling[0] == VC4_SCALING_NONE &&
|
||||||
|
vc4_state->y_scaling[0] == VC4_SCALING_NONE);
|
||||||
|
|
||||||
if (num_planes > 1) {
|
if (num_planes > 1) {
|
||||||
vc4_state->is_yuv = true;
|
vc4_state->is_yuv = true;
|
||||||
|
|
||||||
@ -312,24 +315,17 @@ static int vc4_plane_setup_clipping_and_scaling(struct drm_plane_state *state)
|
|||||||
vc4_get_scaling_mode(vc4_state->src_h[1],
|
vc4_get_scaling_mode(vc4_state->src_h[1],
|
||||||
vc4_state->crtc_h);
|
vc4_state->crtc_h);
|
||||||
|
|
||||||
/* YUV conversion requires that scaling be enabled,
|
/* YUV conversion requires that horizontal scaling be enabled,
|
||||||
* even on a plane that's otherwise 1:1. Choose TPZ
|
* even on a plane that's otherwise 1:1. Looks like only PPF
|
||||||
* for simplicity.
|
* works in that case, so let's pick that one.
|
||||||
*/
|
*/
|
||||||
if (vc4_state->x_scaling[0] == VC4_SCALING_NONE)
|
if (vc4_state->is_unity)
|
||||||
vc4_state->x_scaling[0] = VC4_SCALING_TPZ;
|
vc4_state->x_scaling[0] = VC4_SCALING_PPF;
|
||||||
if (vc4_state->y_scaling[0] == VC4_SCALING_NONE)
|
|
||||||
vc4_state->y_scaling[0] = VC4_SCALING_TPZ;
|
|
||||||
} else {
|
} else {
|
||||||
vc4_state->x_scaling[1] = VC4_SCALING_NONE;
|
vc4_state->x_scaling[1] = VC4_SCALING_NONE;
|
||||||
vc4_state->y_scaling[1] = VC4_SCALING_NONE;
|
vc4_state->y_scaling[1] = VC4_SCALING_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
vc4_state->is_unity = (vc4_state->x_scaling[0] == VC4_SCALING_NONE &&
|
|
||||||
vc4_state->y_scaling[0] == VC4_SCALING_NONE &&
|
|
||||||
vc4_state->x_scaling[1] == VC4_SCALING_NONE &&
|
|
||||||
vc4_state->y_scaling[1] == VC4_SCALING_NONE);
|
|
||||||
|
|
||||||
/* No configuring scaling on the cursor plane, since it gets
|
/* No configuring scaling on the cursor plane, since it gets
|
||||||
non-vblank-synced updates, and scaling requires requires
|
non-vblank-synced updates, and scaling requires requires
|
||||||
LBM changes which have to be vblank-synced.
|
LBM changes which have to be vblank-synced.
|
||||||
@ -672,7 +668,10 @@ static int vc4_plane_mode_set(struct drm_plane *plane,
|
|||||||
vc4_dlist_write(vc4_state, SCALER_CSC2_ITR_R_601_5);
|
vc4_dlist_write(vc4_state, SCALER_CSC2_ITR_R_601_5);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!vc4_state->is_unity) {
|
if (vc4_state->x_scaling[0] != VC4_SCALING_NONE ||
|
||||||
|
vc4_state->x_scaling[1] != VC4_SCALING_NONE ||
|
||||||
|
vc4_state->y_scaling[0] != VC4_SCALING_NONE ||
|
||||||
|
vc4_state->y_scaling[1] != VC4_SCALING_NONE) {
|
||||||
/* LBM Base Address. */
|
/* LBM Base Address. */
|
||||||
if (vc4_state->y_scaling[0] != VC4_SCALING_NONE ||
|
if (vc4_state->y_scaling[0] != VC4_SCALING_NONE ||
|
||||||
vc4_state->y_scaling[1] != VC4_SCALING_NONE) {
|
vc4_state->y_scaling[1] != VC4_SCALING_NONE) {
|
||||||
|
@ -3729,7 +3729,7 @@ int vmw_validate_single_buffer(struct vmw_private *dev_priv,
|
|||||||
{
|
{
|
||||||
struct vmw_buffer_object *vbo =
|
struct vmw_buffer_object *vbo =
|
||||||
container_of(bo, struct vmw_buffer_object, base);
|
container_of(bo, struct vmw_buffer_object, base);
|
||||||
struct ttm_operation_ctx ctx = { interruptible, true };
|
struct ttm_operation_ctx ctx = { interruptible, false };
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
if (vbo->pin_count > 0)
|
if (vbo->pin_count > 0)
|
||||||
|
@ -1512,21 +1512,19 @@ static int vmw_kms_check_display_memory(struct drm_device *dev,
|
|||||||
struct drm_rect *rects)
|
struct drm_rect *rects)
|
||||||
{
|
{
|
||||||
struct vmw_private *dev_priv = vmw_priv(dev);
|
struct vmw_private *dev_priv = vmw_priv(dev);
|
||||||
struct drm_mode_config *mode_config = &dev->mode_config;
|
|
||||||
struct drm_rect bounding_box = {0};
|
struct drm_rect bounding_box = {0};
|
||||||
u64 total_pixels = 0, pixel_mem, bb_mem;
|
u64 total_pixels = 0, pixel_mem, bb_mem;
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
for (i = 0; i < num_rects; i++) {
|
for (i = 0; i < num_rects; i++) {
|
||||||
/*
|
/*
|
||||||
* Currently this check is limiting the topology within max
|
* For STDU only individual screen (screen target) is limited by
|
||||||
* texture/screentarget size. This should change in future when
|
* SCREENTARGET_MAX_WIDTH/HEIGHT registers.
|
||||||
* user-space support multiple fb with topology.
|
|
||||||
*/
|
*/
|
||||||
if (rects[i].x1 < 0 || rects[i].y1 < 0 ||
|
if (dev_priv->active_display_unit == vmw_du_screen_target &&
|
||||||
rects[i].x2 > mode_config->max_width ||
|
(drm_rect_width(&rects[i]) > dev_priv->stdu_max_width ||
|
||||||
rects[i].y2 > mode_config->max_height) {
|
drm_rect_height(&rects[i]) > dev_priv->stdu_max_height)) {
|
||||||
DRM_ERROR("Invalid GUI layout.\n");
|
DRM_ERROR("Screen size not supported.\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1615,7 +1613,7 @@ static int vmw_kms_check_topology(struct drm_device *dev,
|
|||||||
struct drm_connector_state *conn_state;
|
struct drm_connector_state *conn_state;
|
||||||
struct vmw_connector_state *vmw_conn_state;
|
struct vmw_connector_state *vmw_conn_state;
|
||||||
|
|
||||||
if (!new_crtc_state->enable && old_crtc_state->enable) {
|
if (!new_crtc_state->enable) {
|
||||||
rects[i].x1 = 0;
|
rects[i].x1 = 0;
|
||||||
rects[i].y1 = 0;
|
rects[i].y1 = 0;
|
||||||
rects[i].x2 = 0;
|
rects[i].x2 = 0;
|
||||||
@ -2216,12 +2214,16 @@ int vmw_du_connector_fill_modes(struct drm_connector *connector,
|
|||||||
if (dev_priv->assume_16bpp)
|
if (dev_priv->assume_16bpp)
|
||||||
assumed_bpp = 2;
|
assumed_bpp = 2;
|
||||||
|
|
||||||
|
max_width = min(max_width, dev_priv->texture_max_width);
|
||||||
|
max_height = min(max_height, dev_priv->texture_max_height);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* For STDU extra limit for a mode on SVGA_REG_SCREENTARGET_MAX_WIDTH/
|
||||||
|
* HEIGHT registers.
|
||||||
|
*/
|
||||||
if (dev_priv->active_display_unit == vmw_du_screen_target) {
|
if (dev_priv->active_display_unit == vmw_du_screen_target) {
|
||||||
max_width = min(max_width, dev_priv->stdu_max_width);
|
max_width = min(max_width, dev_priv->stdu_max_width);
|
||||||
max_width = min(max_width, dev_priv->texture_max_width);
|
|
||||||
|
|
||||||
max_height = min(max_height, dev_priv->stdu_max_height);
|
max_height = min(max_height, dev_priv->stdu_max_height);
|
||||||
max_height = min(max_height, dev_priv->texture_max_height);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Add preferred mode */
|
/* Add preferred mode */
|
||||||
@ -2376,6 +2378,7 @@ int vmw_kms_update_layout_ioctl(struct drm_device *dev, void *data,
|
|||||||
struct drm_file *file_priv)
|
struct drm_file *file_priv)
|
||||||
{
|
{
|
||||||
struct vmw_private *dev_priv = vmw_priv(dev);
|
struct vmw_private *dev_priv = vmw_priv(dev);
|
||||||
|
struct drm_mode_config *mode_config = &dev->mode_config;
|
||||||
struct drm_vmw_update_layout_arg *arg =
|
struct drm_vmw_update_layout_arg *arg =
|
||||||
(struct drm_vmw_update_layout_arg *)data;
|
(struct drm_vmw_update_layout_arg *)data;
|
||||||
void __user *user_rects;
|
void __user *user_rects;
|
||||||
@ -2421,6 +2424,21 @@ int vmw_kms_update_layout_ioctl(struct drm_device *dev, void *data,
|
|||||||
drm_rects[i].y1 = curr_rect.y;
|
drm_rects[i].y1 = curr_rect.y;
|
||||||
drm_rects[i].x2 = curr_rect.x + curr_rect.w;
|
drm_rects[i].x2 = curr_rect.x + curr_rect.w;
|
||||||
drm_rects[i].y2 = curr_rect.y + curr_rect.h;
|
drm_rects[i].y2 = curr_rect.y + curr_rect.h;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Currently this check is limiting the topology within
|
||||||
|
* mode_config->max (which actually is max texture size
|
||||||
|
* supported by virtual device). This limit is here to address
|
||||||
|
* window managers that create a big framebuffer for whole
|
||||||
|
* topology.
|
||||||
|
*/
|
||||||
|
if (drm_rects[i].x1 < 0 || drm_rects[i].y1 < 0 ||
|
||||||
|
drm_rects[i].x2 > mode_config->max_width ||
|
||||||
|
drm_rects[i].y2 > mode_config->max_height) {
|
||||||
|
DRM_ERROR("Invalid GUI layout.\n");
|
||||||
|
ret = -EINVAL;
|
||||||
|
goto out_free;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = vmw_kms_check_display_memory(dev, arg->num_outputs, drm_rects);
|
ret = vmw_kms_check_display_memory(dev, arg->num_outputs, drm_rects);
|
||||||
|
@ -1600,31 +1600,6 @@ int vmw_kms_stdu_init_display(struct vmw_private *dev_priv)
|
|||||||
|
|
||||||
dev_priv->active_display_unit = vmw_du_screen_target;
|
dev_priv->active_display_unit = vmw_du_screen_target;
|
||||||
|
|
||||||
if (dev_priv->capabilities & SVGA_CAP_3D) {
|
|
||||||
/*
|
|
||||||
* For 3D VMs, display (scanout) buffer size is the smaller of
|
|
||||||
* max texture and max STDU
|
|
||||||
*/
|
|
||||||
uint32_t max_width, max_height;
|
|
||||||
|
|
||||||
max_width = min(dev_priv->texture_max_width,
|
|
||||||
dev_priv->stdu_max_width);
|
|
||||||
max_height = min(dev_priv->texture_max_height,
|
|
||||||
dev_priv->stdu_max_height);
|
|
||||||
|
|
||||||
dev->mode_config.max_width = max_width;
|
|
||||||
dev->mode_config.max_height = max_height;
|
|
||||||
} else {
|
|
||||||
/*
|
|
||||||
* Given various display aspect ratios, there's no way to
|
|
||||||
* estimate these using prim_bb_mem. So just set these to
|
|
||||||
* something arbitrarily large and we will reject any layout
|
|
||||||
* that doesn't fit prim_bb_mem later
|
|
||||||
*/
|
|
||||||
dev->mode_config.max_width = 8192;
|
|
||||||
dev->mode_config.max_height = 8192;
|
|
||||||
}
|
|
||||||
|
|
||||||
vmw_kms_create_implicit_placement_property(dev_priv, false);
|
vmw_kms_create_implicit_placement_property(dev_priv, false);
|
||||||
|
|
||||||
for (i = 0; i < VMWGFX_NUM_DISPLAY_UNITS; ++i) {
|
for (i = 0; i < VMWGFX_NUM_DISPLAY_UNITS; ++i) {
|
||||||
|
@ -1404,22 +1404,17 @@ int vmw_surface_gb_priv_define(struct drm_device *dev,
|
|||||||
*srf_out = NULL;
|
*srf_out = NULL;
|
||||||
|
|
||||||
if (for_scanout) {
|
if (for_scanout) {
|
||||||
uint32_t max_width, max_height;
|
|
||||||
|
|
||||||
if (!svga3dsurface_is_screen_target_format(format)) {
|
if (!svga3dsurface_is_screen_target_format(format)) {
|
||||||
DRM_ERROR("Invalid Screen Target surface format.");
|
DRM_ERROR("Invalid Screen Target surface format.");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
max_width = min(dev_priv->texture_max_width,
|
if (size.width > dev_priv->texture_max_width ||
|
||||||
dev_priv->stdu_max_width);
|
size.height > dev_priv->texture_max_height) {
|
||||||
max_height = min(dev_priv->texture_max_height,
|
|
||||||
dev_priv->stdu_max_height);
|
|
||||||
|
|
||||||
if (size.width > max_width || size.height > max_height) {
|
|
||||||
DRM_ERROR("%ux%u\n, exceeds max surface size %ux%u",
|
DRM_ERROR("%ux%u\n, exceeds max surface size %ux%u",
|
||||||
size.width, size.height,
|
size.width, size.height,
|
||||||
max_width, max_height);
|
dev_priv->texture_max_width,
|
||||||
|
dev_priv->texture_max_height);
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@ -1495,8 +1490,17 @@ int vmw_surface_gb_priv_define(struct drm_device *dev,
|
|||||||
if (srf->flags & SVGA3D_SURFACE_BIND_STREAM_OUTPUT)
|
if (srf->flags & SVGA3D_SURFACE_BIND_STREAM_OUTPUT)
|
||||||
srf->res.backup_size += sizeof(SVGA3dDXSOState);
|
srf->res.backup_size += sizeof(SVGA3dDXSOState);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Don't set SVGA3D_SURFACE_SCREENTARGET flag for a scanout surface with
|
||||||
|
* size greater than STDU max width/height. This is really a workaround
|
||||||
|
* to support creation of big framebuffer requested by some user-space
|
||||||
|
* for whole topology. That big framebuffer won't really be used for
|
||||||
|
* binding with screen target as during prepare_fb a separate surface is
|
||||||
|
* created so it's safe to ignore SVGA3D_SURFACE_SCREENTARGET flag.
|
||||||
|
*/
|
||||||
if (dev_priv->active_display_unit == vmw_du_screen_target &&
|
if (dev_priv->active_display_unit == vmw_du_screen_target &&
|
||||||
for_scanout)
|
for_scanout && size.width <= dev_priv->stdu_max_width &&
|
||||||
|
size.height <= dev_priv->stdu_max_height)
|
||||||
srf->flags |= SVGA3D_SURFACE_SCREENTARGET;
|
srf->flags |= SVGA3D_SURFACE_SCREENTARGET;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -215,6 +215,8 @@ static void vga_switcheroo_enable(void)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
client->id = ret | ID_BIT_AUDIO;
|
client->id = ret | ID_BIT_AUDIO;
|
||||||
|
if (client->ops->gpu_bound)
|
||||||
|
client->ops->gpu_bound(client->pdev, ret);
|
||||||
}
|
}
|
||||||
|
|
||||||
vga_switcheroo_debugfs_init(&vgasr_priv);
|
vga_switcheroo_debugfs_init(&vgasr_priv);
|
||||||
|
@ -207,8 +207,6 @@ superio_exit(int ioreg)
|
|||||||
|
|
||||||
#define NUM_FAN 7
|
#define NUM_FAN 7
|
||||||
|
|
||||||
#define TEMP_SOURCE_VIRTUAL 0x1f
|
|
||||||
|
|
||||||
/* Common and NCT6775 specific data */
|
/* Common and NCT6775 specific data */
|
||||||
|
|
||||||
/* Voltage min/max registers for nr=7..14 are in bank 5 */
|
/* Voltage min/max registers for nr=7..14 are in bank 5 */
|
||||||
@ -299,8 +297,9 @@ static const u16 NCT6775_REG_PWM_READ[] = {
|
|||||||
|
|
||||||
static const u16 NCT6775_REG_FAN[] = { 0x630, 0x632, 0x634, 0x636, 0x638 };
|
static const u16 NCT6775_REG_FAN[] = { 0x630, 0x632, 0x634, 0x636, 0x638 };
|
||||||
static const u16 NCT6775_REG_FAN_MIN[] = { 0x3b, 0x3c, 0x3d };
|
static const u16 NCT6775_REG_FAN_MIN[] = { 0x3b, 0x3c, 0x3d };
|
||||||
static const u16 NCT6775_REG_FAN_PULSES[] = { 0x641, 0x642, 0x643, 0x644, 0 };
|
static const u16 NCT6775_REG_FAN_PULSES[NUM_FAN] = {
|
||||||
static const u16 NCT6775_FAN_PULSE_SHIFT[] = { 0, 0, 0, 0, 0, 0 };
|
0x641, 0x642, 0x643, 0x644 };
|
||||||
|
static const u16 NCT6775_FAN_PULSE_SHIFT[NUM_FAN] = { };
|
||||||
|
|
||||||
static const u16 NCT6775_REG_TEMP[] = {
|
static const u16 NCT6775_REG_TEMP[] = {
|
||||||
0x27, 0x150, 0x250, 0x62b, 0x62c, 0x62d };
|
0x27, 0x150, 0x250, 0x62b, 0x62c, 0x62d };
|
||||||
@ -373,6 +372,7 @@ static const char *const nct6775_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6775_TEMP_MASK 0x001ffffe
|
#define NCT6775_TEMP_MASK 0x001ffffe
|
||||||
|
#define NCT6775_VIRT_TEMP_MASK 0x00000000
|
||||||
|
|
||||||
static const u16 NCT6775_REG_TEMP_ALTERNATE[32] = {
|
static const u16 NCT6775_REG_TEMP_ALTERNATE[32] = {
|
||||||
[13] = 0x661,
|
[13] = 0x661,
|
||||||
@ -425,8 +425,8 @@ static const u8 NCT6776_PWM_MODE_MASK[] = { 0x01, 0, 0, 0, 0, 0 };
|
|||||||
|
|
||||||
static const u16 NCT6776_REG_FAN_MIN[] = {
|
static const u16 NCT6776_REG_FAN_MIN[] = {
|
||||||
0x63a, 0x63c, 0x63e, 0x640, 0x642, 0x64a, 0x64c };
|
0x63a, 0x63c, 0x63e, 0x640, 0x642, 0x64a, 0x64c };
|
||||||
static const u16 NCT6776_REG_FAN_PULSES[] = {
|
static const u16 NCT6776_REG_FAN_PULSES[NUM_FAN] = {
|
||||||
0x644, 0x645, 0x646, 0x647, 0x648, 0x649, 0 };
|
0x644, 0x645, 0x646, 0x647, 0x648, 0x649 };
|
||||||
|
|
||||||
static const u16 NCT6776_REG_WEIGHT_DUTY_BASE[] = {
|
static const u16 NCT6776_REG_WEIGHT_DUTY_BASE[] = {
|
||||||
0x13e, 0x23e, 0x33e, 0x83e, 0x93e, 0xa3e };
|
0x13e, 0x23e, 0x33e, 0x83e, 0x93e, 0xa3e };
|
||||||
@ -461,6 +461,7 @@ static const char *const nct6776_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6776_TEMP_MASK 0x007ffffe
|
#define NCT6776_TEMP_MASK 0x007ffffe
|
||||||
|
#define NCT6776_VIRT_TEMP_MASK 0x00000000
|
||||||
|
|
||||||
static const u16 NCT6776_REG_TEMP_ALTERNATE[32] = {
|
static const u16 NCT6776_REG_TEMP_ALTERNATE[32] = {
|
||||||
[14] = 0x401,
|
[14] = 0x401,
|
||||||
@ -501,9 +502,9 @@ static const s8 NCT6779_BEEP_BITS[] = {
|
|||||||
30, 31 }; /* intrusion0, intrusion1 */
|
30, 31 }; /* intrusion0, intrusion1 */
|
||||||
|
|
||||||
static const u16 NCT6779_REG_FAN[] = {
|
static const u16 NCT6779_REG_FAN[] = {
|
||||||
0x4b0, 0x4b2, 0x4b4, 0x4b6, 0x4b8, 0x4ba, 0x660 };
|
0x4c0, 0x4c2, 0x4c4, 0x4c6, 0x4c8, 0x4ca, 0x4ce };
|
||||||
static const u16 NCT6779_REG_FAN_PULSES[] = {
|
static const u16 NCT6779_REG_FAN_PULSES[NUM_FAN] = {
|
||||||
0x644, 0x645, 0x646, 0x647, 0x648, 0x649, 0 };
|
0x644, 0x645, 0x646, 0x647, 0x648, 0x649 };
|
||||||
|
|
||||||
static const u16 NCT6779_REG_CRITICAL_PWM_ENABLE[] = {
|
static const u16 NCT6779_REG_CRITICAL_PWM_ENABLE[] = {
|
||||||
0x136, 0x236, 0x336, 0x836, 0x936, 0xa36, 0xb36 };
|
0x136, 0x236, 0x336, 0x836, 0x936, 0xa36, 0xb36 };
|
||||||
@ -559,7 +560,9 @@ static const char *const nct6779_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6779_TEMP_MASK 0x07ffff7e
|
#define NCT6779_TEMP_MASK 0x07ffff7e
|
||||||
|
#define NCT6779_VIRT_TEMP_MASK 0x00000000
|
||||||
#define NCT6791_TEMP_MASK 0x87ffff7e
|
#define NCT6791_TEMP_MASK 0x87ffff7e
|
||||||
|
#define NCT6791_VIRT_TEMP_MASK 0x80000000
|
||||||
|
|
||||||
static const u16 NCT6779_REG_TEMP_ALTERNATE[32]
|
static const u16 NCT6779_REG_TEMP_ALTERNATE[32]
|
||||||
= { 0x490, 0x491, 0x492, 0x493, 0x494, 0x495, 0, 0,
|
= { 0x490, 0x491, 0x492, 0x493, 0x494, 0x495, 0, 0,
|
||||||
@ -638,6 +641,7 @@ static const char *const nct6792_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6792_TEMP_MASK 0x9fffff7e
|
#define NCT6792_TEMP_MASK 0x9fffff7e
|
||||||
|
#define NCT6792_VIRT_TEMP_MASK 0x80000000
|
||||||
|
|
||||||
static const char *const nct6793_temp_label[] = {
|
static const char *const nct6793_temp_label[] = {
|
||||||
"",
|
"",
|
||||||
@ -675,6 +679,7 @@ static const char *const nct6793_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6793_TEMP_MASK 0xbfff037e
|
#define NCT6793_TEMP_MASK 0xbfff037e
|
||||||
|
#define NCT6793_VIRT_TEMP_MASK 0x80000000
|
||||||
|
|
||||||
static const char *const nct6795_temp_label[] = {
|
static const char *const nct6795_temp_label[] = {
|
||||||
"",
|
"",
|
||||||
@ -712,6 +717,7 @@ static const char *const nct6795_temp_label[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6795_TEMP_MASK 0xbfffff7e
|
#define NCT6795_TEMP_MASK 0xbfffff7e
|
||||||
|
#define NCT6795_VIRT_TEMP_MASK 0x80000000
|
||||||
|
|
||||||
static const char *const nct6796_temp_label[] = {
|
static const char *const nct6796_temp_label[] = {
|
||||||
"",
|
"",
|
||||||
@ -724,8 +730,8 @@ static const char *const nct6796_temp_label[] = {
|
|||||||
"AUXTIN4",
|
"AUXTIN4",
|
||||||
"SMBUSMASTER 0",
|
"SMBUSMASTER 0",
|
||||||
"SMBUSMASTER 1",
|
"SMBUSMASTER 1",
|
||||||
"",
|
"Virtual_TEMP",
|
||||||
"",
|
"Virtual_TEMP",
|
||||||
"",
|
"",
|
||||||
"",
|
"",
|
||||||
"",
|
"",
|
||||||
@ -748,7 +754,8 @@ static const char *const nct6796_temp_label[] = {
|
|||||||
"Virtual_TEMP"
|
"Virtual_TEMP"
|
||||||
};
|
};
|
||||||
|
|
||||||
#define NCT6796_TEMP_MASK 0xbfff03fe
|
#define NCT6796_TEMP_MASK 0xbfff0ffe
|
||||||
|
#define NCT6796_VIRT_TEMP_MASK 0x80000c00
|
||||||
|
|
||||||
/* NCT6102D/NCT6106D specific data */
|
/* NCT6102D/NCT6106D specific data */
|
||||||
|
|
||||||
@ -779,8 +786,8 @@ static const u16 NCT6106_REG_TEMP_CONFIG[] = {
|
|||||||
|
|
||||||
static const u16 NCT6106_REG_FAN[] = { 0x20, 0x22, 0x24 };
|
static const u16 NCT6106_REG_FAN[] = { 0x20, 0x22, 0x24 };
|
||||||
static const u16 NCT6106_REG_FAN_MIN[] = { 0xe0, 0xe2, 0xe4 };
|
static const u16 NCT6106_REG_FAN_MIN[] = { 0xe0, 0xe2, 0xe4 };
|
||||||
static const u16 NCT6106_REG_FAN_PULSES[] = { 0xf6, 0xf6, 0xf6, 0, 0 };
|
static const u16 NCT6106_REG_FAN_PULSES[] = { 0xf6, 0xf6, 0xf6 };
|
||||||
static const u16 NCT6106_FAN_PULSE_SHIFT[] = { 0, 2, 4, 0, 0 };
|
static const u16 NCT6106_FAN_PULSE_SHIFT[] = { 0, 2, 4 };
|
||||||
|
|
||||||
static const u8 NCT6106_REG_PWM_MODE[] = { 0xf3, 0xf3, 0xf3 };
|
static const u8 NCT6106_REG_PWM_MODE[] = { 0xf3, 0xf3, 0xf3 };
|
||||||
static const u8 NCT6106_PWM_MODE_MASK[] = { 0x01, 0x02, 0x04 };
|
static const u8 NCT6106_PWM_MODE_MASK[] = { 0x01, 0x02, 0x04 };
|
||||||
@ -917,6 +924,11 @@ static unsigned int fan_from_reg16(u16 reg, unsigned int divreg)
|
|||||||
return 1350000U / (reg << divreg);
|
return 1350000U / (reg << divreg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int fan_from_reg_rpm(u16 reg, unsigned int divreg)
|
||||||
|
{
|
||||||
|
return reg;
|
||||||
|
}
|
||||||
|
|
||||||
static u16 fan_to_reg(u32 fan, unsigned int divreg)
|
static u16 fan_to_reg(u32 fan, unsigned int divreg)
|
||||||
{
|
{
|
||||||
if (!fan)
|
if (!fan)
|
||||||
@ -969,6 +981,7 @@ struct nct6775_data {
|
|||||||
u16 reg_temp_config[NUM_TEMP];
|
u16 reg_temp_config[NUM_TEMP];
|
||||||
const char * const *temp_label;
|
const char * const *temp_label;
|
||||||
u32 temp_mask;
|
u32 temp_mask;
|
||||||
|
u32 virt_temp_mask;
|
||||||
|
|
||||||
u16 REG_CONFIG;
|
u16 REG_CONFIG;
|
||||||
u16 REG_VBAT;
|
u16 REG_VBAT;
|
||||||
@ -1276,11 +1289,11 @@ static bool is_word_sized(struct nct6775_data *data, u16 reg)
|
|||||||
case nct6795:
|
case nct6795:
|
||||||
case nct6796:
|
case nct6796:
|
||||||
return reg == 0x150 || reg == 0x153 || reg == 0x155 ||
|
return reg == 0x150 || reg == 0x153 || reg == 0x155 ||
|
||||||
((reg & 0xfff0) == 0x4b0 && (reg & 0x000f) < 0x0b) ||
|
(reg & 0xfff0) == 0x4c0 ||
|
||||||
reg == 0x402 ||
|
reg == 0x402 ||
|
||||||
reg == 0x63a || reg == 0x63c || reg == 0x63e ||
|
reg == 0x63a || reg == 0x63c || reg == 0x63e ||
|
||||||
reg == 0x640 || reg == 0x642 || reg == 0x64a ||
|
reg == 0x640 || reg == 0x642 || reg == 0x64a ||
|
||||||
reg == 0x64c || reg == 0x660 ||
|
reg == 0x64c ||
|
||||||
reg == 0x73 || reg == 0x75 || reg == 0x77 || reg == 0x79 ||
|
reg == 0x73 || reg == 0x75 || reg == 0x77 || reg == 0x79 ||
|
||||||
reg == 0x7b || reg == 0x7d;
|
reg == 0x7b || reg == 0x7d;
|
||||||
}
|
}
|
||||||
@ -1558,7 +1571,7 @@ static void nct6775_update_pwm(struct device *dev)
|
|||||||
reg = nct6775_read_value(data, data->REG_WEIGHT_TEMP_SEL[i]);
|
reg = nct6775_read_value(data, data->REG_WEIGHT_TEMP_SEL[i]);
|
||||||
data->pwm_weight_temp_sel[i] = reg & 0x1f;
|
data->pwm_weight_temp_sel[i] = reg & 0x1f;
|
||||||
/* If weight is disabled, report weight source as 0 */
|
/* If weight is disabled, report weight source as 0 */
|
||||||
if (j == 1 && !(reg & 0x80))
|
if (!(reg & 0x80))
|
||||||
data->pwm_weight_temp_sel[i] = 0;
|
data->pwm_weight_temp_sel[i] = 0;
|
||||||
|
|
||||||
/* Weight temp data */
|
/* Weight temp data */
|
||||||
@ -1682,9 +1695,13 @@ static struct nct6775_data *nct6775_update_device(struct device *dev)
|
|||||||
if (data->has_fan_min & BIT(i))
|
if (data->has_fan_min & BIT(i))
|
||||||
data->fan_min[i] = nct6775_read_value(data,
|
data->fan_min[i] = nct6775_read_value(data,
|
||||||
data->REG_FAN_MIN[i]);
|
data->REG_FAN_MIN[i]);
|
||||||
data->fan_pulses[i] =
|
|
||||||
(nct6775_read_value(data, data->REG_FAN_PULSES[i])
|
if (data->REG_FAN_PULSES[i]) {
|
||||||
>> data->FAN_PULSE_SHIFT[i]) & 0x03;
|
data->fan_pulses[i] =
|
||||||
|
(nct6775_read_value(data,
|
||||||
|
data->REG_FAN_PULSES[i])
|
||||||
|
>> data->FAN_PULSE_SHIFT[i]) & 0x03;
|
||||||
|
}
|
||||||
|
|
||||||
nct6775_select_fan_div(dev, data, i, reg);
|
nct6775_select_fan_div(dev, data, i, reg);
|
||||||
}
|
}
|
||||||
@ -3639,6 +3656,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
data->temp_label = nct6776_temp_label;
|
data->temp_label = nct6776_temp_label;
|
||||||
data->temp_mask = NCT6776_TEMP_MASK;
|
data->temp_mask = NCT6776_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6776_VIRT_TEMP_MASK;
|
||||||
|
|
||||||
data->REG_VBAT = NCT6106_REG_VBAT;
|
data->REG_VBAT = NCT6106_REG_VBAT;
|
||||||
data->REG_DIODE = NCT6106_REG_DIODE;
|
data->REG_DIODE = NCT6106_REG_DIODE;
|
||||||
@ -3717,6 +3735,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
data->temp_label = nct6775_temp_label;
|
data->temp_label = nct6775_temp_label;
|
||||||
data->temp_mask = NCT6775_TEMP_MASK;
|
data->temp_mask = NCT6775_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6775_VIRT_TEMP_MASK;
|
||||||
|
|
||||||
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
||||||
data->REG_VBAT = NCT6775_REG_VBAT;
|
data->REG_VBAT = NCT6775_REG_VBAT;
|
||||||
@ -3789,6 +3808,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
data->temp_label = nct6776_temp_label;
|
data->temp_label = nct6776_temp_label;
|
||||||
data->temp_mask = NCT6776_TEMP_MASK;
|
data->temp_mask = NCT6776_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6776_VIRT_TEMP_MASK;
|
||||||
|
|
||||||
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
||||||
data->REG_VBAT = NCT6775_REG_VBAT;
|
data->REG_VBAT = NCT6775_REG_VBAT;
|
||||||
@ -3853,7 +3873,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
data->ALARM_BITS = NCT6779_ALARM_BITS;
|
data->ALARM_BITS = NCT6779_ALARM_BITS;
|
||||||
data->BEEP_BITS = NCT6779_BEEP_BITS;
|
data->BEEP_BITS = NCT6779_BEEP_BITS;
|
||||||
|
|
||||||
data->fan_from_reg = fan_from_reg13;
|
data->fan_from_reg = fan_from_reg_rpm;
|
||||||
data->fan_from_reg_min = fan_from_reg13;
|
data->fan_from_reg_min = fan_from_reg13;
|
||||||
data->target_temp_mask = 0xff;
|
data->target_temp_mask = 0xff;
|
||||||
data->tolerance_mask = 0x07;
|
data->tolerance_mask = 0x07;
|
||||||
@ -3861,6 +3881,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
|
|
||||||
data->temp_label = nct6779_temp_label;
|
data->temp_label = nct6779_temp_label;
|
||||||
data->temp_mask = NCT6779_TEMP_MASK;
|
data->temp_mask = NCT6779_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6779_VIRT_TEMP_MASK;
|
||||||
|
|
||||||
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
data->REG_CONFIG = NCT6775_REG_CONFIG;
|
||||||
data->REG_VBAT = NCT6775_REG_VBAT;
|
data->REG_VBAT = NCT6775_REG_VBAT;
|
||||||
@ -3933,7 +3954,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
data->ALARM_BITS = NCT6791_ALARM_BITS;
|
data->ALARM_BITS = NCT6791_ALARM_BITS;
|
||||||
data->BEEP_BITS = NCT6779_BEEP_BITS;
|
data->BEEP_BITS = NCT6779_BEEP_BITS;
|
||||||
|
|
||||||
data->fan_from_reg = fan_from_reg13;
|
data->fan_from_reg = fan_from_reg_rpm;
|
||||||
data->fan_from_reg_min = fan_from_reg13;
|
data->fan_from_reg_min = fan_from_reg13;
|
||||||
data->target_temp_mask = 0xff;
|
data->target_temp_mask = 0xff;
|
||||||
data->tolerance_mask = 0x07;
|
data->tolerance_mask = 0x07;
|
||||||
@ -3944,22 +3965,27 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
case nct6791:
|
case nct6791:
|
||||||
data->temp_label = nct6779_temp_label;
|
data->temp_label = nct6779_temp_label;
|
||||||
data->temp_mask = NCT6791_TEMP_MASK;
|
data->temp_mask = NCT6791_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6791_VIRT_TEMP_MASK;
|
||||||
break;
|
break;
|
||||||
case nct6792:
|
case nct6792:
|
||||||
data->temp_label = nct6792_temp_label;
|
data->temp_label = nct6792_temp_label;
|
||||||
data->temp_mask = NCT6792_TEMP_MASK;
|
data->temp_mask = NCT6792_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6792_VIRT_TEMP_MASK;
|
||||||
break;
|
break;
|
||||||
case nct6793:
|
case nct6793:
|
||||||
data->temp_label = nct6793_temp_label;
|
data->temp_label = nct6793_temp_label;
|
||||||
data->temp_mask = NCT6793_TEMP_MASK;
|
data->temp_mask = NCT6793_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6793_VIRT_TEMP_MASK;
|
||||||
break;
|
break;
|
||||||
case nct6795:
|
case nct6795:
|
||||||
data->temp_label = nct6795_temp_label;
|
data->temp_label = nct6795_temp_label;
|
||||||
data->temp_mask = NCT6795_TEMP_MASK;
|
data->temp_mask = NCT6795_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6795_VIRT_TEMP_MASK;
|
||||||
break;
|
break;
|
||||||
case nct6796:
|
case nct6796:
|
||||||
data->temp_label = nct6796_temp_label;
|
data->temp_label = nct6796_temp_label;
|
||||||
data->temp_mask = NCT6796_TEMP_MASK;
|
data->temp_mask = NCT6796_TEMP_MASK;
|
||||||
|
data->virt_temp_mask = NCT6796_VIRT_TEMP_MASK;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -4143,7 +4169,7 @@ static int nct6775_probe(struct platform_device *pdev)
|
|||||||
* for each fan reflects a different temperature, and there
|
* for each fan reflects a different temperature, and there
|
||||||
* are no duplicates.
|
* are no duplicates.
|
||||||
*/
|
*/
|
||||||
if (src != TEMP_SOURCE_VIRTUAL) {
|
if (!(data->virt_temp_mask & BIT(src))) {
|
||||||
if (mask & BIT(src))
|
if (mask & BIT(src))
|
||||||
continue;
|
continue;
|
||||||
mask |= BIT(src);
|
mask |= BIT(src);
|
||||||
|
@ -139,7 +139,8 @@ static int intel_th_remove(struct device *dev)
|
|||||||
th->thdev[i] = NULL;
|
th->thdev[i] = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
th->num_thdevs = lowest;
|
if (lowest >= 0)
|
||||||
|
th->num_thdevs = lowest;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (thdrv->attr_group)
|
if (thdrv->attr_group)
|
||||||
@ -487,7 +488,7 @@ static const struct intel_th_subdevice {
|
|||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
.start = TH_MMIO_SW,
|
.start = 1, /* use resource[1] */
|
||||||
.end = 0,
|
.end = 0,
|
||||||
.flags = IORESOURCE_MEM,
|
.flags = IORESOURCE_MEM,
|
||||||
},
|
},
|
||||||
@ -580,6 +581,7 @@ intel_th_subdevice_alloc(struct intel_th *th,
|
|||||||
struct intel_th_device *thdev;
|
struct intel_th_device *thdev;
|
||||||
struct resource res[3];
|
struct resource res[3];
|
||||||
unsigned int req = 0;
|
unsigned int req = 0;
|
||||||
|
bool is64bit = false;
|
||||||
int r, err;
|
int r, err;
|
||||||
|
|
||||||
thdev = intel_th_device_alloc(th, subdev->type, subdev->name,
|
thdev = intel_th_device_alloc(th, subdev->type, subdev->name,
|
||||||
@ -589,12 +591,18 @@ intel_th_subdevice_alloc(struct intel_th *th,
|
|||||||
|
|
||||||
thdev->drvdata = th->drvdata;
|
thdev->drvdata = th->drvdata;
|
||||||
|
|
||||||
|
for (r = 0; r < th->num_resources; r++)
|
||||||
|
if (th->resource[r].flags & IORESOURCE_MEM_64) {
|
||||||
|
is64bit = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
memcpy(res, subdev->res,
|
memcpy(res, subdev->res,
|
||||||
sizeof(struct resource) * subdev->nres);
|
sizeof(struct resource) * subdev->nres);
|
||||||
|
|
||||||
for (r = 0; r < subdev->nres; r++) {
|
for (r = 0; r < subdev->nres; r++) {
|
||||||
struct resource *devres = th->resource;
|
struct resource *devres = th->resource;
|
||||||
int bar = TH_MMIO_CONFIG;
|
int bar = 0; /* cut subdevices' MMIO from resource[0] */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Take .end == 0 to mean 'take the whole bar',
|
* Take .end == 0 to mean 'take the whole bar',
|
||||||
@ -603,6 +611,8 @@ intel_th_subdevice_alloc(struct intel_th *th,
|
|||||||
*/
|
*/
|
||||||
if (!res[r].end && res[r].flags == IORESOURCE_MEM) {
|
if (!res[r].end && res[r].flags == IORESOURCE_MEM) {
|
||||||
bar = res[r].start;
|
bar = res[r].start;
|
||||||
|
if (is64bit)
|
||||||
|
bar *= 2;
|
||||||
res[r].start = 0;
|
res[r].start = 0;
|
||||||
res[r].end = resource_size(&devres[bar]) - 1;
|
res[r].end = resource_size(&devres[bar]) - 1;
|
||||||
}
|
}
|
||||||
|
@ -160,6 +160,11 @@ static const struct pci_device_id intel_th_pci_id_table[] = {
|
|||||||
PCI_DEVICE(PCI_VENDOR_ID_INTEL, 0x18e1),
|
PCI_DEVICE(PCI_VENDOR_ID_INTEL, 0x18e1),
|
||||||
.driver_data = (kernel_ulong_t)&intel_th_2x,
|
.driver_data = (kernel_ulong_t)&intel_th_2x,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
/* Ice Lake PCH */
|
||||||
|
PCI_DEVICE(PCI_VENDOR_ID_INTEL, 0x34a6),
|
||||||
|
.driver_data = (kernel_ulong_t)&intel_th_2x,
|
||||||
|
},
|
||||||
{ 0 },
|
{ 0 },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1159,41 +1159,21 @@ static int mt9v111_probe(struct i2c_client *client)
|
|||||||
V4L2_CID_AUTO_WHITE_BALANCE,
|
V4L2_CID_AUTO_WHITE_BALANCE,
|
||||||
0, 1, 1,
|
0, 1, 1,
|
||||||
V4L2_WHITE_BALANCE_AUTO);
|
V4L2_WHITE_BALANCE_AUTO);
|
||||||
if (IS_ERR_OR_NULL(mt9v111->auto_awb)) {
|
|
||||||
ret = PTR_ERR(mt9v111->auto_awb);
|
|
||||||
goto error_free_ctrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
mt9v111->auto_exp = v4l2_ctrl_new_std_menu(&mt9v111->ctrls,
|
mt9v111->auto_exp = v4l2_ctrl_new_std_menu(&mt9v111->ctrls,
|
||||||
&mt9v111_ctrl_ops,
|
&mt9v111_ctrl_ops,
|
||||||
V4L2_CID_EXPOSURE_AUTO,
|
V4L2_CID_EXPOSURE_AUTO,
|
||||||
V4L2_EXPOSURE_MANUAL,
|
V4L2_EXPOSURE_MANUAL,
|
||||||
0, V4L2_EXPOSURE_AUTO);
|
0, V4L2_EXPOSURE_AUTO);
|
||||||
if (IS_ERR_OR_NULL(mt9v111->auto_exp)) {
|
|
||||||
ret = PTR_ERR(mt9v111->auto_exp);
|
|
||||||
goto error_free_ctrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Initialize timings */
|
|
||||||
mt9v111->hblank = v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
mt9v111->hblank = v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
||||||
V4L2_CID_HBLANK,
|
V4L2_CID_HBLANK,
|
||||||
MT9V111_CORE_R05_MIN_HBLANK,
|
MT9V111_CORE_R05_MIN_HBLANK,
|
||||||
MT9V111_CORE_R05_MAX_HBLANK, 1,
|
MT9V111_CORE_R05_MAX_HBLANK, 1,
|
||||||
MT9V111_CORE_R05_DEF_HBLANK);
|
MT9V111_CORE_R05_DEF_HBLANK);
|
||||||
if (IS_ERR_OR_NULL(mt9v111->hblank)) {
|
|
||||||
ret = PTR_ERR(mt9v111->hblank);
|
|
||||||
goto error_free_ctrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
mt9v111->vblank = v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
mt9v111->vblank = v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
||||||
V4L2_CID_VBLANK,
|
V4L2_CID_VBLANK,
|
||||||
MT9V111_CORE_R06_MIN_VBLANK,
|
MT9V111_CORE_R06_MIN_VBLANK,
|
||||||
MT9V111_CORE_R06_MAX_VBLANK, 1,
|
MT9V111_CORE_R06_MAX_VBLANK, 1,
|
||||||
MT9V111_CORE_R06_DEF_VBLANK);
|
MT9V111_CORE_R06_DEF_VBLANK);
|
||||||
if (IS_ERR_OR_NULL(mt9v111->vblank)) {
|
|
||||||
ret = PTR_ERR(mt9v111->vblank);
|
|
||||||
goto error_free_ctrls;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* PIXEL_RATE is fixed: just expose it to user space. */
|
/* PIXEL_RATE is fixed: just expose it to user space. */
|
||||||
v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
v4l2_ctrl_new_std(&mt9v111->ctrls, &mt9v111_ctrl_ops,
|
||||||
@ -1201,6 +1181,10 @@ static int mt9v111_probe(struct i2c_client *client)
|
|||||||
DIV_ROUND_CLOSEST(mt9v111->sysclk, 2), 1,
|
DIV_ROUND_CLOSEST(mt9v111->sysclk, 2), 1,
|
||||||
DIV_ROUND_CLOSEST(mt9v111->sysclk, 2));
|
DIV_ROUND_CLOSEST(mt9v111->sysclk, 2));
|
||||||
|
|
||||||
|
if (mt9v111->ctrls.error) {
|
||||||
|
ret = mt9v111->ctrls.error;
|
||||||
|
goto error_free_ctrls;
|
||||||
|
}
|
||||||
mt9v111->sd.ctrl_handler = &mt9v111->ctrls;
|
mt9v111->sd.ctrl_handler = &mt9v111->ctrls;
|
||||||
|
|
||||||
/* Start with default configuration: 640x480 UYVY. */
|
/* Start with default configuration: 640x480 UYVY. */
|
||||||
@ -1226,26 +1210,27 @@ static int mt9v111_probe(struct i2c_client *client)
|
|||||||
mt9v111->pad.flags = MEDIA_PAD_FL_SOURCE;
|
mt9v111->pad.flags = MEDIA_PAD_FL_SOURCE;
|
||||||
ret = media_entity_pads_init(&mt9v111->sd.entity, 1, &mt9v111->pad);
|
ret = media_entity_pads_init(&mt9v111->sd.entity, 1, &mt9v111->pad);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_free_ctrls;
|
goto error_free_entity;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
ret = mt9v111_chip_probe(mt9v111);
|
ret = mt9v111_chip_probe(mt9v111);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_free_ctrls;
|
goto error_free_entity;
|
||||||
|
|
||||||
ret = v4l2_async_register_subdev(&mt9v111->sd);
|
ret = v4l2_async_register_subdev(&mt9v111->sd);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto error_free_ctrls;
|
goto error_free_entity;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
error_free_ctrls:
|
error_free_entity:
|
||||||
v4l2_ctrl_handler_free(&mt9v111->ctrls);
|
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_MEDIA_CONTROLLER)
|
#if IS_ENABLED(CONFIG_MEDIA_CONTROLLER)
|
||||||
media_entity_cleanup(&mt9v111->sd.entity);
|
media_entity_cleanup(&mt9v111->sd.entity);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
error_free_ctrls:
|
||||||
|
v4l2_ctrl_handler_free(&mt9v111->ctrls);
|
||||||
|
|
||||||
mutex_destroy(&mt9v111->pwr_mutex);
|
mutex_destroy(&mt9v111->pwr_mutex);
|
||||||
mutex_destroy(&mt9v111->stream_mutex);
|
mutex_destroy(&mt9v111->stream_mutex);
|
||||||
|
|
||||||
@ -1259,12 +1244,12 @@ static int mt9v111_remove(struct i2c_client *client)
|
|||||||
|
|
||||||
v4l2_async_unregister_subdev(sd);
|
v4l2_async_unregister_subdev(sd);
|
||||||
|
|
||||||
v4l2_ctrl_handler_free(&mt9v111->ctrls);
|
|
||||||
|
|
||||||
#if IS_ENABLED(CONFIG_MEDIA_CONTROLLER)
|
#if IS_ENABLED(CONFIG_MEDIA_CONTROLLER)
|
||||||
media_entity_cleanup(&sd->entity);
|
media_entity_cleanup(&sd->entity);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
v4l2_ctrl_handler_free(&mt9v111->ctrls);
|
||||||
|
|
||||||
mutex_destroy(&mt9v111->pwr_mutex);
|
mutex_destroy(&mt9v111->pwr_mutex);
|
||||||
mutex_destroy(&mt9v111->stream_mutex);
|
mutex_destroy(&mt9v111->stream_mutex);
|
||||||
|
|
||||||
|
@ -541,6 +541,8 @@ config VIDEO_CROS_EC_CEC
|
|||||||
depends on MFD_CROS_EC
|
depends on MFD_CROS_EC
|
||||||
select CEC_CORE
|
select CEC_CORE
|
||||||
select CEC_NOTIFIER
|
select CEC_NOTIFIER
|
||||||
|
select CHROME_PLATFORMS
|
||||||
|
select CROS_EC_PROTO
|
||||||
---help---
|
---help---
|
||||||
If you say yes here you will get support for the
|
If you say yes here you will get support for the
|
||||||
ChromeOS Embedded Controller's CEC.
|
ChromeOS Embedded Controller's CEC.
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#include <linux/clk.h>
|
#include <linux/clk.h>
|
||||||
#include <linux/completion.h>
|
#include <linux/completion.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
|
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
|
|
||||||
#define CAMSS_CSI_PHY_LNn_CFG2(n) (0x004 + 0x40 * (n))
|
#define CAMSS_CSI_PHY_LNn_CFG2(n) (0x004 + 0x40 * (n))
|
||||||
#define CAMSS_CSI_PHY_LNn_CFG3(n) (0x008 + 0x40 * (n))
|
#define CAMSS_CSI_PHY_LNn_CFG3(n) (0x008 + 0x40 * (n))
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
|
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
|
|
||||||
#define CSIPHY_3PH_LNn_CFG1(n) (0x000 + 0x100 * (n))
|
#define CSIPHY_3PH_LNn_CFG1(n) (0x000 + 0x100 * (n))
|
||||||
#define CSIPHY_3PH_LNn_CFG1_SWI_REC_DLY_PRG (BIT(7) | BIT(6))
|
#define CSIPHY_3PH_LNn_CFG1_SWI_REC_DLY_PRG (BIT(7) | BIT(6))
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#include <linux/clk.h>
|
#include <linux/clk.h>
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
#include <linux/platform_device.h>
|
#include <linux/platform_device.h>
|
||||||
|
@ -10,6 +10,7 @@
|
|||||||
#include <linux/clk.h>
|
#include <linux/clk.h>
|
||||||
#include <linux/completion.h>
|
#include <linux/completion.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
#include <linux/iopoll.h>
|
#include <linux/iopoll.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/mutex.h>
|
#include <linux/mutex.h>
|
||||||
@ -1076,8 +1077,8 @@ int msm_ispif_subdev_init(struct ispif_device *ispif,
|
|||||||
else
|
else
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
ispif->line = kcalloc(ispif->line_num, sizeof(*ispif->line),
|
ispif->line = devm_kcalloc(dev, ispif->line_num, sizeof(*ispif->line),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!ispif->line)
|
if (!ispif->line)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
#include <linux/iopoll.h>
|
#include <linux/iopoll.h>
|
||||||
|
|
||||||
#include "camss-vfe.h"
|
#include "camss-vfe.h"
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
|
#include <linux/io.h>
|
||||||
#include <linux/iopoll.h>
|
#include <linux/iopoll.h>
|
||||||
|
|
||||||
#include "camss-vfe.h"
|
#include "camss-vfe.h"
|
||||||
|
@ -848,17 +848,18 @@ static int camss_probe(struct platform_device *pdev)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
camss->csiphy = kcalloc(camss->csiphy_num, sizeof(*camss->csiphy),
|
camss->csiphy = devm_kcalloc(dev, camss->csiphy_num,
|
||||||
GFP_KERNEL);
|
sizeof(*camss->csiphy), GFP_KERNEL);
|
||||||
if (!camss->csiphy)
|
if (!camss->csiphy)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
camss->csid = kcalloc(camss->csid_num, sizeof(*camss->csid),
|
camss->csid = devm_kcalloc(dev, camss->csid_num, sizeof(*camss->csid),
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
if (!camss->csid)
|
if (!camss->csid)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
camss->vfe = kcalloc(camss->vfe_num, sizeof(*camss->vfe), GFP_KERNEL);
|
camss->vfe = devm_kcalloc(dev, camss->vfe_num, sizeof(*camss->vfe),
|
||||||
|
GFP_KERNEL);
|
||||||
if (!camss->vfe)
|
if (!camss->vfe)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
@ -993,12 +994,12 @@ static const struct of_device_id camss_dt_match[] = {
|
|||||||
|
|
||||||
MODULE_DEVICE_TABLE(of, camss_dt_match);
|
MODULE_DEVICE_TABLE(of, camss_dt_match);
|
||||||
|
|
||||||
static int camss_runtime_suspend(struct device *dev)
|
static int __maybe_unused camss_runtime_suspend(struct device *dev)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int camss_runtime_resume(struct device *dev)
|
static int __maybe_unused camss_runtime_resume(struct device *dev)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -402,8 +402,10 @@ static int af9035_i2c_master_xfer(struct i2c_adapter *adap,
|
|||||||
if (msg[0].addr == state->af9033_i2c_addr[1])
|
if (msg[0].addr == state->af9033_i2c_addr[1])
|
||||||
reg |= 0x100000;
|
reg |= 0x100000;
|
||||||
|
|
||||||
ret = af9035_wr_regs(d, reg, &msg[0].buf[3],
|
ret = (msg[0].len >= 3) ? af9035_wr_regs(d, reg,
|
||||||
msg[0].len - 3);
|
&msg[0].buf[3],
|
||||||
|
msg[0].len - 3)
|
||||||
|
: -EOPNOTSUPP;
|
||||||
} else {
|
} else {
|
||||||
/* I2C write */
|
/* I2C write */
|
||||||
u8 buf[MAX_XFER_SIZE];
|
u8 buf[MAX_XFER_SIZE];
|
||||||
|
@ -528,8 +528,8 @@ static int usbhs_omap_get_dt_pdata(struct device *dev,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static const struct of_device_id usbhs_child_match_table[] = {
|
static const struct of_device_id usbhs_child_match_table[] = {
|
||||||
{ .compatible = "ti,omap-ehci", },
|
{ .compatible = "ti,ehci-omap", },
|
||||||
{ .compatible = "ti,omap-ohci", },
|
{ .compatible = "ti,ohci-omap3", },
|
||||||
{ }
|
{ }
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -855,6 +855,7 @@ static struct platform_driver usbhs_omap_driver = {
|
|||||||
.pm = &usbhsomap_dev_pm_ops,
|
.pm = &usbhsomap_dev_pm_ops,
|
||||||
.of_match_table = usbhs_omap_dt_ids,
|
.of_match_table = usbhs_omap_dt_ids,
|
||||||
},
|
},
|
||||||
|
.probe = usbhs_omap_probe,
|
||||||
.remove = usbhs_omap_remove,
|
.remove = usbhs_omap_remove,
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -864,9 +865,9 @@ MODULE_ALIAS("platform:" USBHS_DRIVER_NAME);
|
|||||||
MODULE_LICENSE("GPL v2");
|
MODULE_LICENSE("GPL v2");
|
||||||
MODULE_DESCRIPTION("usb host common core driver for omap EHCI and OHCI");
|
MODULE_DESCRIPTION("usb host common core driver for omap EHCI and OHCI");
|
||||||
|
|
||||||
static int __init omap_usbhs_drvinit(void)
|
static int omap_usbhs_drvinit(void)
|
||||||
{
|
{
|
||||||
return platform_driver_probe(&usbhs_omap_driver, usbhs_omap_probe);
|
return platform_driver_register(&usbhs_omap_driver);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -878,7 +879,7 @@ static int __init omap_usbhs_drvinit(void)
|
|||||||
*/
|
*/
|
||||||
fs_initcall_sync(omap_usbhs_drvinit);
|
fs_initcall_sync(omap_usbhs_drvinit);
|
||||||
|
|
||||||
static void __exit omap_usbhs_drvexit(void)
|
static void omap_usbhs_drvexit(void)
|
||||||
{
|
{
|
||||||
platform_driver_unregister(&usbhs_omap_driver);
|
platform_driver_unregister(&usbhs_omap_driver);
|
||||||
}
|
}
|
||||||
|
@ -39,13 +39,23 @@ static int m25p80_read_reg(struct spi_nor *nor, u8 code, u8 *val, int len)
|
|||||||
struct spi_mem_op op = SPI_MEM_OP(SPI_MEM_OP_CMD(code, 1),
|
struct spi_mem_op op = SPI_MEM_OP(SPI_MEM_OP_CMD(code, 1),
|
||||||
SPI_MEM_OP_NO_ADDR,
|
SPI_MEM_OP_NO_ADDR,
|
||||||
SPI_MEM_OP_NO_DUMMY,
|
SPI_MEM_OP_NO_DUMMY,
|
||||||
SPI_MEM_OP_DATA_IN(len, val, 1));
|
SPI_MEM_OP_DATA_IN(len, NULL, 1));
|
||||||
|
void *scratchbuf;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
scratchbuf = kmalloc(len, GFP_KERNEL);
|
||||||
|
if (!scratchbuf)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
op.data.buf.in = scratchbuf;
|
||||||
ret = spi_mem_exec_op(flash->spimem, &op);
|
ret = spi_mem_exec_op(flash->spimem, &op);
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
dev_err(&flash->spimem->spi->dev, "error %d reading %x\n", ret,
|
dev_err(&flash->spimem->spi->dev, "error %d reading %x\n", ret,
|
||||||
code);
|
code);
|
||||||
|
else
|
||||||
|
memcpy(val, scratchbuf, len);
|
||||||
|
|
||||||
|
kfree(scratchbuf);
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@ -56,9 +66,19 @@ static int m25p80_write_reg(struct spi_nor *nor, u8 opcode, u8 *buf, int len)
|
|||||||
struct spi_mem_op op = SPI_MEM_OP(SPI_MEM_OP_CMD(opcode, 1),
|
struct spi_mem_op op = SPI_MEM_OP(SPI_MEM_OP_CMD(opcode, 1),
|
||||||
SPI_MEM_OP_NO_ADDR,
|
SPI_MEM_OP_NO_ADDR,
|
||||||
SPI_MEM_OP_NO_DUMMY,
|
SPI_MEM_OP_NO_DUMMY,
|
||||||
SPI_MEM_OP_DATA_OUT(len, buf, 1));
|
SPI_MEM_OP_DATA_OUT(len, NULL, 1));
|
||||||
|
void *scratchbuf;
|
||||||
|
int ret;
|
||||||
|
|
||||||
return spi_mem_exec_op(flash->spimem, &op);
|
scratchbuf = kmemdup(buf, len, GFP_KERNEL);
|
||||||
|
if (!scratchbuf)
|
||||||
|
return -ENOMEM;
|
||||||
|
|
||||||
|
op.data.buf.out = scratchbuf;
|
||||||
|
ret = spi_mem_exec_op(flash->spimem, &op);
|
||||||
|
kfree(scratchbuf);
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static ssize_t m25p80_write(struct spi_nor *nor, loff_t to, size_t len,
|
static ssize_t m25p80_write(struct spi_nor *nor, loff_t to, size_t len,
|
||||||
|
@ -873,8 +873,11 @@ static int mtd_part_of_parse(struct mtd_info *master,
|
|||||||
int ret, err = 0;
|
int ret, err = 0;
|
||||||
|
|
||||||
np = mtd_get_of_node(master);
|
np = mtd_get_of_node(master);
|
||||||
if (!mtd_is_partition(master))
|
if (mtd_is_partition(master))
|
||||||
|
of_node_get(np);
|
||||||
|
else
|
||||||
np = of_get_child_by_name(np, "partitions");
|
np = of_get_child_by_name(np, "partitions");
|
||||||
|
|
||||||
of_property_for_each_string(np, "compatible", prop, compat) {
|
of_property_for_each_string(np, "compatible", prop, compat) {
|
||||||
parser = mtd_part_get_compatible_parser(compat);
|
parser = mtd_part_get_compatible_parser(compat);
|
||||||
if (!parser)
|
if (!parser)
|
||||||
|
@ -596,6 +596,12 @@ static int denali_dma_xfer(struct denali_nand_info *denali, void *buf,
|
|||||||
}
|
}
|
||||||
|
|
||||||
iowrite32(DMA_ENABLE__FLAG, denali->reg + DMA_ENABLE);
|
iowrite32(DMA_ENABLE__FLAG, denali->reg + DMA_ENABLE);
|
||||||
|
/*
|
||||||
|
* The ->setup_dma() hook kicks DMA by using the data/command
|
||||||
|
* interface, which belongs to a different AXI port from the
|
||||||
|
* register interface. Read back the register to avoid a race.
|
||||||
|
*/
|
||||||
|
ioread32(denali->reg + DMA_ENABLE);
|
||||||
|
|
||||||
denali_reset_irq(denali);
|
denali_reset_irq(denali);
|
||||||
denali->setup_dma(denali, dma_addr, page, write);
|
denali->setup_dma(denali, dma_addr, page, write);
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user