18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (C) 2015, 2016 ARM Ltd. 48c2ecf20Sopenharmony_ci */ 58c2ecf20Sopenharmony_ci 68c2ecf20Sopenharmony_ci#include <linux/kvm.h> 78c2ecf20Sopenharmony_ci#include <linux/kvm_host.h> 88c2ecf20Sopenharmony_ci#include <trace/events/kvm.h> 98c2ecf20Sopenharmony_ci#include <kvm/arm_vgic.h> 108c2ecf20Sopenharmony_ci#include "vgic.h" 118c2ecf20Sopenharmony_ci 128c2ecf20Sopenharmony_ci/** 138c2ecf20Sopenharmony_ci * vgic_irqfd_set_irq: inject the IRQ corresponding to the 148c2ecf20Sopenharmony_ci * irqchip routing entry 158c2ecf20Sopenharmony_ci * 168c2ecf20Sopenharmony_ci * This is the entry point for irqfd IRQ injection 178c2ecf20Sopenharmony_ci */ 188c2ecf20Sopenharmony_cistatic int vgic_irqfd_set_irq(struct kvm_kernel_irq_routing_entry *e, 198c2ecf20Sopenharmony_ci struct kvm *kvm, int irq_source_id, 208c2ecf20Sopenharmony_ci int level, bool line_status) 218c2ecf20Sopenharmony_ci{ 228c2ecf20Sopenharmony_ci unsigned int spi_id = e->irqchip.pin + VGIC_NR_PRIVATE_IRQS; 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_ci if (!vgic_valid_spi(kvm, spi_id)) 258c2ecf20Sopenharmony_ci return -EINVAL; 268c2ecf20Sopenharmony_ci return kvm_vgic_inject_irq(kvm, 0, spi_id, level, NULL); 278c2ecf20Sopenharmony_ci} 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_ci/** 308c2ecf20Sopenharmony_ci * kvm_set_routing_entry: populate a kvm routing entry 318c2ecf20Sopenharmony_ci * from a user routing entry 328c2ecf20Sopenharmony_ci * 338c2ecf20Sopenharmony_ci * @kvm: the VM this entry is applied to 348c2ecf20Sopenharmony_ci * @e: kvm kernel routing entry handle 358c2ecf20Sopenharmony_ci * @ue: user api routing entry handle 368c2ecf20Sopenharmony_ci * return 0 on success, -EINVAL on errors. 378c2ecf20Sopenharmony_ci */ 388c2ecf20Sopenharmony_ciint kvm_set_routing_entry(struct kvm *kvm, 398c2ecf20Sopenharmony_ci struct kvm_kernel_irq_routing_entry *e, 408c2ecf20Sopenharmony_ci const struct kvm_irq_routing_entry *ue) 418c2ecf20Sopenharmony_ci{ 428c2ecf20Sopenharmony_ci int r = -EINVAL; 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_ci switch (ue->type) { 458c2ecf20Sopenharmony_ci case KVM_IRQ_ROUTING_IRQCHIP: 468c2ecf20Sopenharmony_ci e->set = vgic_irqfd_set_irq; 478c2ecf20Sopenharmony_ci e->irqchip.irqchip = ue->u.irqchip.irqchip; 488c2ecf20Sopenharmony_ci e->irqchip.pin = ue->u.irqchip.pin; 498c2ecf20Sopenharmony_ci if ((e->irqchip.pin >= KVM_IRQCHIP_NUM_PINS) || 508c2ecf20Sopenharmony_ci (e->irqchip.irqchip >= KVM_NR_IRQCHIPS)) 518c2ecf20Sopenharmony_ci goto out; 528c2ecf20Sopenharmony_ci break; 538c2ecf20Sopenharmony_ci case KVM_IRQ_ROUTING_MSI: 548c2ecf20Sopenharmony_ci e->set = kvm_set_msi; 558c2ecf20Sopenharmony_ci e->msi.address_lo = ue->u.msi.address_lo; 568c2ecf20Sopenharmony_ci e->msi.address_hi = ue->u.msi.address_hi; 578c2ecf20Sopenharmony_ci e->msi.data = ue->u.msi.data; 588c2ecf20Sopenharmony_ci e->msi.flags = ue->flags; 598c2ecf20Sopenharmony_ci e->msi.devid = ue->u.msi.devid; 608c2ecf20Sopenharmony_ci break; 618c2ecf20Sopenharmony_ci default: 628c2ecf20Sopenharmony_ci goto out; 638c2ecf20Sopenharmony_ci } 648c2ecf20Sopenharmony_ci r = 0; 658c2ecf20Sopenharmony_ciout: 668c2ecf20Sopenharmony_ci return r; 678c2ecf20Sopenharmony_ci} 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_cistatic void kvm_populate_msi(struct kvm_kernel_irq_routing_entry *e, 708c2ecf20Sopenharmony_ci struct kvm_msi *msi) 718c2ecf20Sopenharmony_ci{ 728c2ecf20Sopenharmony_ci msi->address_lo = e->msi.address_lo; 738c2ecf20Sopenharmony_ci msi->address_hi = e->msi.address_hi; 748c2ecf20Sopenharmony_ci msi->data = e->msi.data; 758c2ecf20Sopenharmony_ci msi->flags = e->msi.flags; 768c2ecf20Sopenharmony_ci msi->devid = e->msi.devid; 778c2ecf20Sopenharmony_ci} 788c2ecf20Sopenharmony_ci/** 798c2ecf20Sopenharmony_ci * kvm_set_msi: inject the MSI corresponding to the 808c2ecf20Sopenharmony_ci * MSI routing entry 818c2ecf20Sopenharmony_ci * 828c2ecf20Sopenharmony_ci * This is the entry point for irqfd MSI injection 838c2ecf20Sopenharmony_ci * and userspace MSI injection. 848c2ecf20Sopenharmony_ci */ 858c2ecf20Sopenharmony_ciint kvm_set_msi(struct kvm_kernel_irq_routing_entry *e, 868c2ecf20Sopenharmony_ci struct kvm *kvm, int irq_source_id, 878c2ecf20Sopenharmony_ci int level, bool line_status) 888c2ecf20Sopenharmony_ci{ 898c2ecf20Sopenharmony_ci struct kvm_msi msi; 908c2ecf20Sopenharmony_ci 918c2ecf20Sopenharmony_ci if (!vgic_has_its(kvm)) 928c2ecf20Sopenharmony_ci return -ENODEV; 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci if (!level) 958c2ecf20Sopenharmony_ci return -1; 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci kvm_populate_msi(e, &msi); 988c2ecf20Sopenharmony_ci return vgic_its_inject_msi(kvm, &msi); 998c2ecf20Sopenharmony_ci} 1008c2ecf20Sopenharmony_ci 1018c2ecf20Sopenharmony_ci/** 1028c2ecf20Sopenharmony_ci * kvm_arch_set_irq_inatomic: fast-path for irqfd injection 1038c2ecf20Sopenharmony_ci */ 1048c2ecf20Sopenharmony_ciint kvm_arch_set_irq_inatomic(struct kvm_kernel_irq_routing_entry *e, 1058c2ecf20Sopenharmony_ci struct kvm *kvm, int irq_source_id, int level, 1068c2ecf20Sopenharmony_ci bool line_status) 1078c2ecf20Sopenharmony_ci{ 1088c2ecf20Sopenharmony_ci if (!level) 1098c2ecf20Sopenharmony_ci return -EWOULDBLOCK; 1108c2ecf20Sopenharmony_ci 1118c2ecf20Sopenharmony_ci switch (e->type) { 1128c2ecf20Sopenharmony_ci case KVM_IRQ_ROUTING_MSI: { 1138c2ecf20Sopenharmony_ci struct kvm_msi msi; 1148c2ecf20Sopenharmony_ci 1158c2ecf20Sopenharmony_ci if (!vgic_has_its(kvm)) 1168c2ecf20Sopenharmony_ci break; 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_ci kvm_populate_msi(e, &msi); 1198c2ecf20Sopenharmony_ci return vgic_its_inject_cached_translation(kvm, &msi); 1208c2ecf20Sopenharmony_ci } 1218c2ecf20Sopenharmony_ci 1228c2ecf20Sopenharmony_ci case KVM_IRQ_ROUTING_IRQCHIP: 1238c2ecf20Sopenharmony_ci /* 1248c2ecf20Sopenharmony_ci * Injecting SPIs is always possible in atomic context 1258c2ecf20Sopenharmony_ci * as long as the damn vgic is initialized. 1268c2ecf20Sopenharmony_ci */ 1278c2ecf20Sopenharmony_ci if (unlikely(!vgic_initialized(kvm))) 1288c2ecf20Sopenharmony_ci break; 1298c2ecf20Sopenharmony_ci return vgic_irqfd_set_irq(e, kvm, irq_source_id, 1, line_status); 1308c2ecf20Sopenharmony_ci } 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_ci return -EWOULDBLOCK; 1338c2ecf20Sopenharmony_ci} 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_ciint kvm_vgic_setup_default_irq_routing(struct kvm *kvm) 1368c2ecf20Sopenharmony_ci{ 1378c2ecf20Sopenharmony_ci struct kvm_irq_routing_entry *entries; 1388c2ecf20Sopenharmony_ci struct vgic_dist *dist = &kvm->arch.vgic; 1398c2ecf20Sopenharmony_ci u32 nr = dist->nr_spis; 1408c2ecf20Sopenharmony_ci int i, ret; 1418c2ecf20Sopenharmony_ci 1428c2ecf20Sopenharmony_ci entries = kcalloc(nr, sizeof(*entries), GFP_KERNEL); 1438c2ecf20Sopenharmony_ci if (!entries) 1448c2ecf20Sopenharmony_ci return -ENOMEM; 1458c2ecf20Sopenharmony_ci 1468c2ecf20Sopenharmony_ci for (i = 0; i < nr; i++) { 1478c2ecf20Sopenharmony_ci entries[i].gsi = i; 1488c2ecf20Sopenharmony_ci entries[i].type = KVM_IRQ_ROUTING_IRQCHIP; 1498c2ecf20Sopenharmony_ci entries[i].u.irqchip.irqchip = 0; 1508c2ecf20Sopenharmony_ci entries[i].u.irqchip.pin = i; 1518c2ecf20Sopenharmony_ci } 1528c2ecf20Sopenharmony_ci ret = kvm_set_irq_routing(kvm, entries, nr, 0); 1538c2ecf20Sopenharmony_ci kfree(entries); 1548c2ecf20Sopenharmony_ci return ret; 1558c2ecf20Sopenharmony_ci} 156