Skip to content

Commit fb443ce

Browse files
committed
Merge tag 'kvmarm-fixes-6.10-2' of git://git.kernel.org/pub/scm/linux/kernel/git/kvmarm/kvmarm into HEAD
KVM/arm64 fixes for 6.10, take #2 - Fix dangling references to a redistributor region if the vgic was prematurely destroyed. - Properly mark FFA buffers as released, ensuring that both parties can make forward progress.
2 parents 676f819 + d66e50b commit fb443ce

File tree

4 files changed

+27
-4
lines changed

4 files changed

+27
-4
lines changed

arch/arm64/kvm/hyp/nvhe/ffa.c

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -177,6 +177,14 @@ static void ffa_retrieve_req(struct arm_smccc_res *res, u32 len)
177177
res);
178178
}
179179

180+
static void ffa_rx_release(struct arm_smccc_res *res)
181+
{
182+
arm_smccc_1_1_smc(FFA_RX_RELEASE,
183+
0, 0,
184+
0, 0, 0, 0, 0,
185+
res);
186+
}
187+
180188
static void do_ffa_rxtx_map(struct arm_smccc_res *res,
181189
struct kvm_cpu_context *ctxt)
182190
{
@@ -543,16 +551,19 @@ static void do_ffa_mem_reclaim(struct arm_smccc_res *res,
543551
if (WARN_ON(offset > len ||
544552
fraglen > KVM_FFA_MBOX_NR_PAGES * PAGE_SIZE)) {
545553
ret = FFA_RET_ABORTED;
554+
ffa_rx_release(res);
546555
goto out_unlock;
547556
}
548557

549558
if (len > ffa_desc_buf.len) {
550559
ret = FFA_RET_NO_MEMORY;
560+
ffa_rx_release(res);
551561
goto out_unlock;
552562
}
553563

554564
buf = ffa_desc_buf.buf;
555565
memcpy(buf, hyp_buffers.rx, fraglen);
566+
ffa_rx_release(res);
556567

557568
for (fragoff = fraglen; fragoff < len; fragoff += fraglen) {
558569
ffa_mem_frag_rx(res, handle_lo, handle_hi, fragoff);
@@ -563,6 +574,7 @@ static void do_ffa_mem_reclaim(struct arm_smccc_res *res,
563574

564575
fraglen = res->a3;
565576
memcpy((void *)buf + fragoff, hyp_buffers.rx, fraglen);
577+
ffa_rx_release(res);
566578
}
567579

568580
ffa_mem_reclaim(res, handle_lo, handle_hi, flags);

arch/arm64/kvm/vgic/vgic-init.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -391,7 +391,7 @@ static void kvm_vgic_dist_destroy(struct kvm *kvm)
391391

392392
if (dist->vgic_model == KVM_DEV_TYPE_ARM_VGIC_V3) {
393393
list_for_each_entry_safe(rdreg, next, &dist->rd_regions, list)
394-
vgic_v3_free_redist_region(rdreg);
394+
vgic_v3_free_redist_region(kvm, rdreg);
395395
INIT_LIST_HEAD(&dist->rd_regions);
396396
} else {
397397
dist->vgic_cpu_base = VGIC_ADDR_UNDEF;

arch/arm64/kvm/vgic/vgic-mmio-v3.c

Lines changed: 13 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -919,8 +919,19 @@ static int vgic_v3_alloc_redist_region(struct kvm *kvm, uint32_t index,
919919
return ret;
920920
}
921921

922-
void vgic_v3_free_redist_region(struct vgic_redist_region *rdreg)
922+
void vgic_v3_free_redist_region(struct kvm *kvm, struct vgic_redist_region *rdreg)
923923
{
924+
struct kvm_vcpu *vcpu;
925+
unsigned long c;
926+
927+
lockdep_assert_held(&kvm->arch.config_lock);
928+
929+
/* Garbage collect the region */
930+
kvm_for_each_vcpu(c, vcpu, kvm) {
931+
if (vcpu->arch.vgic_cpu.rdreg == rdreg)
932+
vcpu->arch.vgic_cpu.rdreg = NULL;
933+
}
934+
924935
list_del(&rdreg->list);
925936
kfree(rdreg);
926937
}
@@ -945,7 +956,7 @@ int vgic_v3_set_redist_base(struct kvm *kvm, u32 index, u64 addr, u32 count)
945956

946957
mutex_lock(&kvm->arch.config_lock);
947958
rdreg = vgic_v3_rdist_region_from_index(kvm, index);
948-
vgic_v3_free_redist_region(rdreg);
959+
vgic_v3_free_redist_region(kvm, rdreg);
949960
mutex_unlock(&kvm->arch.config_lock);
950961
return ret;
951962
}

arch/arm64/kvm/vgic/vgic.h

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -316,7 +316,7 @@ vgic_v3_rd_region_size(struct kvm *kvm, struct vgic_redist_region *rdreg)
316316

317317
struct vgic_redist_region *vgic_v3_rdist_region_from_index(struct kvm *kvm,
318318
u32 index);
319-
void vgic_v3_free_redist_region(struct vgic_redist_region *rdreg);
319+
void vgic_v3_free_redist_region(struct kvm *kvm, struct vgic_redist_region *rdreg);
320320

321321
bool vgic_v3_rdist_overlap(struct kvm *kvm, gpa_t base, size_t size);
322322

0 commit comments

Comments
 (0)