diff options
author | Jens Freimann <jfrei@linux.vnet.ibm.com> | 2013-10-07 10:13:45 -0400 |
---|---|---|
committer | Christian Borntraeger <borntraeger@de.ibm.com> | 2014-01-30 04:25:20 -0500 |
commit | c05c4186bbe4e99d64e8a36f7ca7f480da5d109f (patch) | |
tree | 746d5b7f59b3c74668ab2b0abcfb48b64b4512e9 /arch/s390/kvm | |
parent | 81aa8efe0190cf5bf7eaafb57341cd7d0aea96cd (diff) |
KVM: s390: add floating irq controller
This patch adds a floating irq controller as a kvm_device.
It will be necessary for migration of floating interrupts as well
as for hardening the reset code by allowing user space to explicitly
remove all pending floating interrupts.
Signed-off-by: Jens Freimann <jfrei@linux.vnet.ibm.com>
Reviewed-by: Cornelia Huck <cornelia.huck@de.ibm.com>
Signed-off-by: Christian Borntraeger <borntraeger@de.ibm.com>
Diffstat (limited to 'arch/s390/kvm')
-rw-r--r-- | arch/s390/kvm/interrupt.c | 304 | ||||
-rw-r--r-- | arch/s390/kvm/kvm-s390.c | 1 |
2 files changed, 254 insertions, 51 deletions
diff --git a/arch/s390/kvm/interrupt.c b/arch/s390/kvm/interrupt.c index 5f79d2d79ca7..a5f18babed4c 100644 --- a/arch/s390/kvm/interrupt.c +++ b/arch/s390/kvm/interrupt.c | |||
@@ -659,53 +659,86 @@ struct kvm_s390_interrupt_info *kvm_s390_get_io_int(struct kvm *kvm, | |||
659 | return inti; | 659 | return inti; |
660 | } | 660 | } |
661 | 661 | ||
662 | int kvm_s390_inject_vm(struct kvm *kvm, | 662 | static void __inject_vm(struct kvm *kvm, struct kvm_s390_interrupt_info *inti) |
663 | struct kvm_s390_interrupt *s390int) | ||
664 | { | 663 | { |
665 | struct kvm_s390_local_interrupt *li; | 664 | struct kvm_s390_local_interrupt *li; |
666 | struct kvm_s390_float_interrupt *fi; | 665 | struct kvm_s390_float_interrupt *fi; |
667 | struct kvm_s390_interrupt_info *inti, *iter; | 666 | struct kvm_s390_interrupt_info *iter; |
668 | int sigcpu; | 667 | int sigcpu; |
669 | 668 | ||
669 | mutex_lock(&kvm->lock); | ||
670 | fi = &kvm->arch.float_int; | ||
671 | spin_lock(&fi->lock); | ||
672 | if (!is_ioint(inti->type)) { | ||
673 | list_add_tail(&inti->list, &fi->list); | ||
674 | } else { | ||
675 | u64 isc_bits = int_word_to_isc_bits(inti->io.io_int_word); | ||
676 | |||
677 | /* Keep I/O interrupts sorted in isc order. */ | ||
678 | list_for_each_entry(iter, &fi->list, list) { | ||
679 | if (!is_ioint(iter->type)) | ||
680 | continue; | ||
681 | if (int_word_to_isc_bits(iter->io.io_int_word) | ||
682 | <= isc_bits) | ||
683 | continue; | ||
684 | break; | ||
685 | } | ||
686 | list_add_tail(&inti->list, &iter->list); | ||
687 | } | ||
688 | atomic_set(&fi->active, 1); | ||
689 | sigcpu = find_first_bit(fi->idle_mask, KVM_MAX_VCPUS); | ||
690 | if (sigcpu == KVM_MAX_VCPUS) { | ||
691 | do { | ||
692 | sigcpu = fi->next_rr_cpu++; | ||
693 | if (sigcpu == KVM_MAX_VCPUS) | ||
694 | sigcpu = fi->next_rr_cpu = 0; | ||
695 | } while (fi->local_int[sigcpu] == NULL); | ||
696 | } | ||
697 | li = fi->local_int[sigcpu]; | ||
698 | spin_lock_bh(&li->lock); | ||
699 | atomic_set_mask(CPUSTAT_EXT_INT, li->cpuflags); | ||
700 | if (waitqueue_active(li->wq)) | ||
701 | wake_up_interruptible(li->wq); | ||
702 | spin_unlock_bh(&li->lock); | ||
703 | spin_unlock(&fi->lock); | ||
704 | mutex_unlock(&kvm->lock); | ||
705 | } | ||
706 | |||
707 | int kvm_s390_inject_vm(struct kvm *kvm, | ||
708 | struct kvm_s390_interrupt *s390int) | ||
709 | { | ||
710 | struct kvm_s390_interrupt_info *inti; | ||
711 | |||
670 | inti = kzalloc(sizeof(*inti), GFP_KERNEL); | 712 | inti = kzalloc(sizeof(*inti), GFP_KERNEL); |
671 | if (!inti) | 713 | if (!inti) |
672 | return -ENOMEM; | 714 | return -ENOMEM; |
673 | 715 | ||
674 | switch (s390int->type) { | 716 | inti->type = s390int->type; |
717 | switch (inti->type) { | ||
675 | case KVM_S390_INT_VIRTIO: | 718 | case KVM_S390_INT_VIRTIO: |
676 | VM_EVENT(kvm, 5, "inject: virtio parm:%x,parm64:%llx", | 719 | VM_EVENT(kvm, 5, "inject: virtio parm:%x,parm64:%llx", |
677 | s390int->parm, s390int->parm64); | 720 | s390int->parm, s390int->parm64); |
678 | inti->type = s390int->type; | ||
679 | inti->ext.ext_params = s390int->parm; | 721 | inti->ext.ext_params = s390int->parm; |
680 | inti->ext.ext_params2 = s390int->parm64; | 722 | inti->ext.ext_params2 = s390int->parm64; |
681 | break; | 723 | break; |
682 | case KVM_S390_INT_SERVICE: | 724 | case KVM_S390_INT_SERVICE: |
683 | VM_EVENT(kvm, 5, "inject: sclp parm:%x", s390int->parm); | 725 | VM_EVENT(kvm, 5, "inject: sclp parm:%x", s390int->parm); |
684 | inti->type = s390int->type; | ||
685 | inti->ext.ext_params = s390int->parm; | 726 | inti->ext.ext_params = s390int->parm; |
686 | break; | 727 | break; |
687 | case KVM_S390_PROGRAM_INT: | ||
688 | case KVM_S390_SIGP_STOP: | ||
689 | case KVM_S390_INT_EXTERNAL_CALL: | ||
690 | case KVM_S390_INT_EMERGENCY: | ||
691 | kfree(inti); | ||
692 | return -EINVAL; | ||
693 | case KVM_S390_MCHK: | 728 | case KVM_S390_MCHK: |
694 | VM_EVENT(kvm, 5, "inject: machine check parm64:%llx", | 729 | VM_EVENT(kvm, 5, "inject: machine check parm64:%llx", |
695 | s390int->parm64); | 730 | s390int->parm64); |
696 | inti->type = s390int->type; | ||
697 | inti->mchk.cr14 = s390int->parm; /* upper bits are not used */ | 731 | inti->mchk.cr14 = s390int->parm; /* upper bits are not used */ |
698 | inti->mchk.mcic = s390int->parm64; | 732 | inti->mchk.mcic = s390int->parm64; |
699 | break; | 733 | break; |
700 | case KVM_S390_INT_IO_MIN...KVM_S390_INT_IO_MAX: | 734 | case KVM_S390_INT_IO_MIN...KVM_S390_INT_IO_MAX: |
701 | if (s390int->type & IOINT_AI_MASK) | 735 | if (inti->type & IOINT_AI_MASK) |
702 | VM_EVENT(kvm, 5, "%s", "inject: I/O (AI)"); | 736 | VM_EVENT(kvm, 5, "%s", "inject: I/O (AI)"); |
703 | else | 737 | else |
704 | VM_EVENT(kvm, 5, "inject: I/O css %x ss %x schid %04x", | 738 | VM_EVENT(kvm, 5, "inject: I/O css %x ss %x schid %04x", |
705 | s390int->type & IOINT_CSSID_MASK, | 739 | s390int->type & IOINT_CSSID_MASK, |
706 | s390int->type & IOINT_SSID_MASK, | 740 | s390int->type & IOINT_SSID_MASK, |
707 | s390int->type & IOINT_SCHID_MASK); | 741 | s390int->type & IOINT_SCHID_MASK); |
708 | inti->type = s390int->type; | ||
709 | inti->io.subchannel_id = s390int->parm >> 16; | 742 | inti->io.subchannel_id = s390int->parm >> 16; |
710 | inti->io.subchannel_nr = s390int->parm & 0x0000ffffu; | 743 | inti->io.subchannel_nr = s390int->parm & 0x0000ffffu; |
711 | inti->io.io_int_parm = s390int->parm64 >> 32; | 744 | inti->io.io_int_parm = s390int->parm64 >> 32; |
@@ -718,42 +751,7 @@ int kvm_s390_inject_vm(struct kvm *kvm, | |||
718 | trace_kvm_s390_inject_vm(s390int->type, s390int->parm, s390int->parm64, | 751 | trace_kvm_s390_inject_vm(s390int->type, s390int->parm, s390int->parm64, |
719 | 2); | 752 | 2); |
720 | 753 | ||
721 | mutex_lock(&kvm->lock); | 754 | __inject_vm(kvm, inti); |
722 | fi = &kvm->arch.float_int; | ||
723 | spin_lock(&fi->lock); | ||
724 | if (!is_ioint(inti->type)) | ||
725 | list_add_tail(&inti->list, &fi->list); | ||
726 | else { | ||
727 | u64 isc_bits = int_word_to_isc_bits(inti->io.io_int_word); | ||
728 | |||
729 | /* Keep I/O interrupts sorted in isc order. */ | ||
730 | list_for_each_entry(iter, &fi->list, list) { | ||
731 | if (!is_ioint(iter->type)) | ||
732 | continue; | ||
733 | if (int_word_to_isc_bits(iter->io.io_int_word) | ||
734 | <= isc_bits) | ||
735 | continue; | ||
736 | break; | ||
737 | } | ||
738 | list_add_tail(&inti->list, &iter->list); | ||
739 | } | ||
740 | atomic_set(&fi->active, 1); | ||
741 | sigcpu = find_first_bit(fi->idle_mask, KVM_MAX_VCPUS); | ||
742 | if (sigcpu == KVM_MAX_VCPUS) { | ||
743 | do { | ||
744 | sigcpu = fi->next_rr_cpu++; | ||
745 | if (sigcpu == KVM_MAX_VCPUS) | ||
746 | sigcpu = fi->next_rr_cpu = 0; | ||
747 | } while (fi->local_int[sigcpu] == NULL); | ||
748 | } | ||
749 | li = fi->local_int[sigcpu]; | ||
750 | spin_lock_bh(&li->lock); | ||
751 | atomic_set_mask(CPUSTAT_EXT_INT, li->cpuflags); | ||
752 | if (waitqueue_active(li->wq)) | ||
753 | wake_up_interruptible(li->wq); | ||
754 | spin_unlock_bh(&li->lock); | ||
755 | spin_unlock(&fi->lock); | ||
756 | mutex_unlock(&kvm->lock); | ||
757 | return 0; | 755 | return 0; |
758 | } | 756 | } |
759 | 757 | ||
@@ -841,3 +839,207 @@ int kvm_s390_inject_vcpu(struct kvm_vcpu *vcpu, | |||
841 | mutex_unlock(&vcpu->kvm->lock); | 839 | mutex_unlock(&vcpu->kvm->lock); |
842 | return 0; | 840 | return 0; |
843 | } | 841 | } |
842 | |||
843 | static void clear_floating_interrupts(struct kvm *kvm) | ||
844 | { | ||
845 | struct kvm_s390_float_interrupt *fi; | ||
846 | struct kvm_s390_interrupt_info *n, *inti = NULL; | ||
847 | |||
848 | mutex_lock(&kvm->lock); | ||
849 | fi = &kvm->arch.float_int; | ||
850 | spin_lock(&fi->lock); | ||
851 | list_for_each_entry_safe(inti, n, &fi->list, list) { | ||
852 | list_del(&inti->list); | ||
853 | kfree(inti); | ||
854 | } | ||
855 | atomic_set(&fi->active, 0); | ||
856 | spin_unlock(&fi->lock); | ||
857 | mutex_unlock(&kvm->lock); | ||
858 | } | ||
859 | |||
860 | static inline int copy_irq_to_user(struct kvm_s390_interrupt_info *inti, | ||
861 | u8 *addr) | ||
862 | { | ||
863 | struct kvm_s390_irq __user *uptr = (struct kvm_s390_irq __user *) addr; | ||
864 | struct kvm_s390_irq irq = {0}; | ||
865 | |||
866 | irq.type = inti->type; | ||
867 | switch (inti->type) { | ||
868 | case KVM_S390_INT_VIRTIO: | ||
869 | case KVM_S390_INT_SERVICE: | ||
870 | irq.u.ext = inti->ext; | ||
871 | break; | ||
872 | case KVM_S390_INT_IO_MIN...KVM_S390_INT_IO_MAX: | ||
873 | irq.u.io = inti->io; | ||
874 | break; | ||
875 | case KVM_S390_MCHK: | ||
876 | irq.u.mchk = inti->mchk; | ||
877 | break; | ||
878 | default: | ||
879 | return -EINVAL; | ||
880 | } | ||
881 | |||
882 | if (copy_to_user(uptr, &irq, sizeof(irq))) | ||
883 | return -EFAULT; | ||
884 | |||
885 | return 0; | ||
886 | } | ||
887 | |||
888 | static int get_all_floating_irqs(struct kvm *kvm, __u8 *buf, __u64 len) | ||
889 | { | ||
890 | struct kvm_s390_interrupt_info *inti; | ||
891 | struct kvm_s390_float_interrupt *fi; | ||
892 | int ret = 0; | ||
893 | int n = 0; | ||
894 | |||
895 | mutex_lock(&kvm->lock); | ||
896 | fi = &kvm->arch.float_int; | ||
897 | spin_lock(&fi->lock); | ||
898 | |||
899 | list_for_each_entry(inti, &fi->list, list) { | ||
900 | if (len < sizeof(struct kvm_s390_irq)) { | ||
901 | /* signal userspace to try again */ | ||
902 | ret = -ENOMEM; | ||
903 | break; | ||
904 | } | ||
905 | ret = copy_irq_to_user(inti, buf); | ||
906 | if (ret) | ||
907 | break; | ||
908 | buf += sizeof(struct kvm_s390_irq); | ||
909 | len -= sizeof(struct kvm_s390_irq); | ||
910 | n++; | ||
911 | } | ||
912 | |||
913 | spin_unlock(&fi->lock); | ||
914 | mutex_unlock(&kvm->lock); | ||
915 | |||
916 | return ret < 0 ? ret : n; | ||
917 | } | ||
918 | |||
919 | static int flic_get_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | ||
920 | { | ||
921 | int r; | ||
922 | |||
923 | switch (attr->group) { | ||
924 | case KVM_DEV_FLIC_GET_ALL_IRQS: | ||
925 | r = get_all_floating_irqs(dev->kvm, (u8 *) attr->addr, | ||
926 | attr->attr); | ||
927 | break; | ||
928 | default: | ||
929 | r = -EINVAL; | ||
930 | } | ||
931 | |||
932 | return r; | ||
933 | } | ||
934 | |||
935 | static inline int copy_irq_from_user(struct kvm_s390_interrupt_info *inti, | ||
936 | u64 addr) | ||
937 | { | ||
938 | struct kvm_s390_irq __user *uptr = (struct kvm_s390_irq __user *) addr; | ||
939 | void *target = NULL; | ||
940 | void __user *source; | ||
941 | u64 size; | ||
942 | |||
943 | if (get_user(inti->type, (u64 __user *)addr)) | ||
944 | return -EFAULT; | ||
945 | |||
946 | switch (inti->type) { | ||
947 | case KVM_S390_INT_VIRTIO: | ||
948 | case KVM_S390_INT_SERVICE: | ||
949 | target = (void *) &inti->ext; | ||
950 | source = &uptr->u.ext; | ||
951 | size = sizeof(inti->ext); | ||
952 | break; | ||
953 | case KVM_S390_INT_IO_MIN...KVM_S390_INT_IO_MAX: | ||
954 | target = (void *) &inti->io; | ||
955 | source = &uptr->u.io; | ||
956 | size = sizeof(inti->io); | ||
957 | break; | ||
958 | case KVM_S390_MCHK: | ||
959 | target = (void *) &inti->mchk; | ||
960 | source = &uptr->u.mchk; | ||
961 | size = sizeof(inti->mchk); | ||
962 | break; | ||
963 | default: | ||
964 | return -EINVAL; | ||
965 | } | ||
966 | |||
967 | if (copy_from_user(target, source, size)) | ||
968 | return -EFAULT; | ||
969 | |||
970 | return 0; | ||
971 | } | ||
972 | |||
973 | static int enqueue_floating_irq(struct kvm_device *dev, | ||
974 | struct kvm_device_attr *attr) | ||
975 | { | ||
976 | struct kvm_s390_interrupt_info *inti = NULL; | ||
977 | int r = 0; | ||
978 | int len = attr->attr; | ||
979 | |||
980 | if (len % sizeof(struct kvm_s390_irq) != 0) | ||
981 | return -EINVAL; | ||
982 | else if (len > KVM_S390_FLIC_MAX_BUFFER) | ||
983 | return -EINVAL; | ||
984 | |||
985 | while (len >= sizeof(struct kvm_s390_irq)) { | ||
986 | inti = kzalloc(sizeof(*inti), GFP_KERNEL); | ||
987 | if (!inti) | ||
988 | return -ENOMEM; | ||
989 | |||
990 | r = copy_irq_from_user(inti, attr->addr); | ||
991 | if (r) { | ||
992 | kfree(inti); | ||
993 | return r; | ||
994 | } | ||
995 | __inject_vm(dev->kvm, inti); | ||
996 | len -= sizeof(struct kvm_s390_irq); | ||
997 | attr->addr += sizeof(struct kvm_s390_irq); | ||
998 | } | ||
999 | |||
1000 | return r; | ||
1001 | } | ||
1002 | |||
1003 | static int flic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | ||
1004 | { | ||
1005 | int r = 0; | ||
1006 | |||
1007 | switch (attr->group) { | ||
1008 | case KVM_DEV_FLIC_ENQUEUE: | ||
1009 | r = enqueue_floating_irq(dev, attr); | ||
1010 | break; | ||
1011 | case KVM_DEV_FLIC_CLEAR_IRQS: | ||
1012 | r = 0; | ||
1013 | clear_floating_interrupts(dev->kvm); | ||
1014 | break; | ||
1015 | default: | ||
1016 | r = -EINVAL; | ||
1017 | } | ||
1018 | |||
1019 | return r; | ||
1020 | } | ||
1021 | |||
1022 | static int flic_create(struct kvm_device *dev, u32 type) | ||
1023 | { | ||
1024 | if (!dev) | ||
1025 | return -EINVAL; | ||
1026 | if (dev->kvm->arch.flic) | ||
1027 | return -EINVAL; | ||
1028 | dev->kvm->arch.flic = dev; | ||
1029 | return 0; | ||
1030 | } | ||
1031 | |||
1032 | static void flic_destroy(struct kvm_device *dev) | ||
1033 | { | ||
1034 | dev->kvm->arch.flic = NULL; | ||
1035 | kfree(dev); | ||
1036 | } | ||
1037 | |||
1038 | /* s390 floating irq controller (flic) */ | ||
1039 | struct kvm_device_ops kvm_flic_ops = { | ||
1040 | .name = "kvm-flic", | ||
1041 | .get_attr = flic_get_attr, | ||
1042 | .set_attr = flic_set_attr, | ||
1043 | .create = flic_create, | ||
1044 | .destroy = flic_destroy, | ||
1045 | }; | ||
diff --git a/arch/s390/kvm/kvm-s390.c b/arch/s390/kvm/kvm-s390.c index e0676f390d57..782420f3c4d5 100644 --- a/arch/s390/kvm/kvm-s390.c +++ b/arch/s390/kvm/kvm-s390.c | |||
@@ -157,6 +157,7 @@ int kvm_dev_ioctl_check_extension(long ext) | |||
157 | case KVM_CAP_ENABLE_CAP: | 157 | case KVM_CAP_ENABLE_CAP: |
158 | case KVM_CAP_S390_CSS_SUPPORT: | 158 | case KVM_CAP_S390_CSS_SUPPORT: |
159 | case KVM_CAP_IOEVENTFD: | 159 | case KVM_CAP_IOEVENTFD: |
160 | case KVM_CAP_DEVICE_CTRL: | ||
160 | r = 1; | 161 | r = 1; |
161 | break; | 162 | break; |
162 | case KVM_CAP_NR_VCPUS: | 163 | case KVM_CAP_NR_VCPUS: |