Re: [PATCH] kvm: arm/arm64 : fix vm's hanging at startup time

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



>Hi,
>
>On Wed, Nov 21, 2018 at 04:56:54PM +0800, peng.hao2@xxxxxxxxxx wrote:
>> >On 19/11/2018 09:10, Mark Rutland wrote:
>> >> On Sat, Nov 17, 2018 at 10:58:37AM +0800, peng.hao2@xxxxxxxxxx wrote:
>> >>>> On 16/11/18 00:23, peng.hao2@xxxxxxxxxx wrote:
>> >>>>>> Hi,
>> >>>>>>> When virtual machine starts, hang up.
>> >>>>>>
>> >>>>>> I take it you mean the *guest* hangs? Because it doesn't get a timer
>> >>>>>> interrupt?
>> >>>>>>
>> >>>>>>> The kernel version of guest
>> >>>>>>> is 4.16. Host support vgic_v3.
>> >>>>>>
>> >>>>>> Your host kernel is something recent, I guess?
>> >>>>>>
>> >>>>>>> It was mainly due to the incorrect vgic_irq's(intid=27) group value
>> >>>>>>> during injection interruption. when kvm_vgic_vcpu_init is called,
>> >>>>>>> dist is not initialized at this time. Unable to get vgic V3 or V2
>> >>>>>>> correctly, so group is not set.
>> >>>>>>
>> >>>>>> Mmh, that shouldn't happen with (v)GICv3. Do you use QEMU (which
>> >>>>>> version?) or some other userland tool?
>> >>>>>>
>> >>>>>
>> >>>>> QEMU emulator version 3.0.50 .
>> >>>>>
>> >>>>>>> group is setted to 1 when vgic_mmio_write_group is invoked at some
>> >>>>>>> time.
>> >>>>>>> when irq->group=0 (intid=27), No ICH_LR_GROUP flag was set and
>> >>>>>>> interrupt injection failed.
>> >>>>>>>
>> >>>>>>> Signed-off-by: Peng Hao <peng.hao2@xxxxxxxxxx>
>> >>>>>>> ---
>> >>>>>>>   virt/kvm/arm/vgic/vgic-v3.c | 2 +-
>> >>>>>>>   1 file changed, 1 insertion(+), 1 deletion(-)
>> >>>>>>>
>> >>>>>>> diff --git a/virt/kvm/arm/vgic/vgic-v3.c b/virt/kvm/arm/vgic/vgic-v3.c
>> >>>>>>> index 9c0dd23..d101000 100644
>> >>>>>>> --- a/virt/kvm/arm/vgic/vgic-v3.c
>> >>>>>>> +++ b/virt/kvm/arm/vgic/vgic-v3.c
>> >>>>>>> @@ -198,7 +198,7 @@ void vgic_v3_populate_lr(struct kvm_vcpu *vcpu,
>> >>>>>>> struct vgic_irq *irq, int lr) if (vgic_irq_is_mapped_level(irq) &&
>> >>>>>>> (val & ICH_LR_PENDING_BIT)) irq->line_level = false;
>> >>>>>>>
>> >>>>>>> -    if (irq->group)
>> >>>>>>> +    if (model == KVM_DEV_TYPE_ARM_VGIC_V3)
>> >>>>>>
>> >>>>>> This is not the right fix, not only because it basically reverts the
>> >>>>>> GICv3 part of 87322099052 (KVM: arm/arm64: vgic: Signal IRQs using
>> >>>>>> their configured group).
>> >>>>>>
>> >>>>>> Can you try to work out why kvm_vgic_vcpu_init() is apparently called
>> >>>>>> before dist->vgic_model is set, also what value it has?
>> >>>>>> If I understand the code correctly, that shouldn't happen for a GICv3.
>> >>>>>>
>> >>>>> Even if the value of  group is correctly assigned in kvm_vgic_vcpu_init, the group is then written 0 through >vgic_mmio_write_group.
>> >>>>>   If the interrupt comes at this time, the interrupt injection fails.
>> >>>>
>> >>>> Does that mean that the guest is configuring its interrupts as Group0?
>> >>>> That sounds wrong, Linux should configure all it's interrupts as
>> >>>> non-secure group1.
>> >>>
>> >>> no, I think that uefi dose this, not linux.
>> >>> 1. kvm_vgic_vcpu_init
>> >>> 2. vgic_create
>> >>> 3. kvm_vgic_dist_init
>> >>> 4.vgic_mmio_write_group: uefi as guest, write group=0
>> >>> 5.vgic_mmio_write_group: linux as guest, write group=1
>> >>
>> >> Is this the same issue fixed by EDK2 commit:
>> >>
>> >> 66127011a544b90e ("ArmPkg/ArmGicDxe ARM: fix encoding for GICv3 interrupt acknowledge")
>> >>
>> >> ... where EDK2 would try to use IAR0 rather than IAR1?
>> >>
>> >> The commit messages notes this lead to a boot-time hang.
>> >
>> >I managed to trigger an issue with a really old EFI implementation that
>> >doesn't configure its interrupts as Group1, and yet tries to ACK its
>> >interrupts using the Group1 accessor. Guess what? It is not going to work.
>> >
>> >Commit c7fefb690661f2e38afcb8200bd318ecf38ab961 in the edk2 tree seems
>> >to be the fix (I only assume it does, I haven't actually checked). A
>> >recent build, as found in Debian Buster, works perfectly (tested with
>> >both QEMU v2.12 and tip of tree).
>> >
>> >Now, I really don't get what you're saying about Linux not getting
>> >interrupts. How do you get to booting Linux if EFI is not making any
>> >forward progress? Are you trying them independently?
>> >
>> I start linux with bypassing uefi, the print info is the same.
>> [507107.748908]  vgic_mmio_write_group:## intid/27 group=0
>> [507107.752185]  vgic_mmio_write_group:## intid/27 group=0
>> [507107.899566]  vgic_mmio_write_group:## intid/27 group=1
>> [507107.907370]  vgic_mmio_write_group:## intid/27 group=1
>> the command line is like this:
>> /home/qemu-patch/linshi/qemu/aarch64-softmmu/qemu-system-aarch64  -machine virt-3.1,accel=kvm,usb=off,dump-guest-core=off,gic-version=3  -kernel /home/kernelboot/vmlinuz-4.16.0+ -initrd /home/kernelboot/initramfs-4.16.0+.img -append root=/dev/mapper/cla-root ro crashkernel=auto rd.lvm.lv=cla/root rd.lvm.lv=cla/swap.UTF-8  -drive file=/home/centos74-ph/boot.qcow2,format=qcow2,if=none,id=drive-scsi0-0-0-0 -device scsi-hd,bus=scsi0.0,channel=0,scsi-id=0,lun=0,drive=drive-scsi0-0-0-0,id=scsi0-0-0-0,bootindex=1  -vnc 0.0.0.0:0 -k en-us -device virtio-gpu-pci,id=video0,max_outputs=1,bus=pci.3,addr=0x0 -device pvpanic-mmio -msg >timestamp=on
>>
>> This is strange. That's probably the Linux 4.16  kernel setting group to 0, and I'll continue to track in guest.
>
>Could you try the following patch:
>
>diff --git a/virt/kvm/arm/vgic/vgic-init.c b/virt/kvm/arm/vgic/vgic-init.c
>index c0c0b88af1d5..6fa858c7a5a6 100644
>--- a/virt/kvm/arm/vgic/vgic-init.c
>+++ b/virt/kvm/arm/vgic/vgic-init.c
>@@ -231,13 +231,6 @@ int kvm_vgic_vcpu_init(struct kvm_vcpu *vcpu)
>irq->config = VGIC_CONFIG_LEVEL;
>}
>-        /*
>-         * GICv3 can only be created via the KVM_DEVICE_CREATE API and
>-         * so we always know the emulation type at this point as it's
>-         * either explicitly configured as GICv3, or explicitly
>-         * configured as GICv2, or not configured yet which also
>-         * implies GICv2.
>-         */
>if (dist->vgic_model == KVM_DEV_TYPE_ARM_VGIC_V3)
>irq->group = 1;
>else
>@@ -298,6 +291,16 @@ int vgic_init(struct kvm *kvm)
>if (ret)
>goto out;
>+    /* Initialize groups on CPUs created before the VGIC type was known */
>+    kvm_for_each_vcpu(i, vcpu, kvm) {
>+        struct vgic_cpu *vgic_cpu = &vcpu->arch.vgic_cpu;
>+
>+        for (i = 0; i < VGIC_NR_PRIVATE_IRQS; i++) {
>+            struct vgic_irq *irq = &vgic_cpu->private_irqs[i];
>+            irq->group = 1;
>+        }
>+    }
>+
>if (vgic_has_its(kvm)) {
>ret = vgic_v4_init(kvm);
>if (ret)
>
Sorry for the late reply. 
I test the patch and it works. But I think there is randomness here.
1.vgic_init:  set intid/27 group=1
2.vgic_mmio_write_group:## intid/27 group=0
3.!!! receive a  vtimer irq   ----- it may exist.
4.vgic_mmio_write_group:## intid/27 group=1

I made an attempt before:
1. kvm_vgic_vcpu_init
     according to if  kvm_vgic_global_state.type == VGIC_V3, set group=1
2.vgic_mmio_write_group:## intid/27 group=0
3.!!! receive a  vtimer irq   -----I really receive a vtimer irq, then the guest hangs.

thanks.
>
>Thanks,
>
>Christoffer
_______________________________________________
kvmarm mailing list
kvmarm@xxxxxxxxxxxxxxxxxxxxx
https://lists.cs.columbia.edu/mailman/listinfo/kvmarm

[Index of Archives]     [Linux KVM]     [Spice Development]     [Libvirt]     [Libvirt Users]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux