diff options
Diffstat (limited to 'virt/kvm/arm/vgic.c')
-rw-r--r-- | virt/kvm/arm/vgic.c | 78 |
1 files changed, 54 insertions, 24 deletions
diff --git a/virt/kvm/arm/vgic.c b/virt/kvm/arm/vgic.c index 7589e2c82db2..06073fac03a9 100644 --- a/virt/kvm/arm/vgic.c +++ b/virt/kvm/arm/vgic.c | |||
@@ -2440,7 +2440,8 @@ out: | |||
2440 | return ret; | 2440 | return ret; |
2441 | } | 2441 | } |
2442 | 2442 | ||
2443 | static int vgic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | 2443 | static int vgic_set_common_attr(struct kvm_device *dev, |
2444 | struct kvm_device_attr *attr) | ||
2444 | { | 2445 | { |
2445 | int r; | 2446 | int r; |
2446 | 2447 | ||
@@ -2456,17 +2457,6 @@ static int vgic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | |||
2456 | r = kvm_vgic_addr(dev->kvm, type, &addr, true); | 2457 | r = kvm_vgic_addr(dev->kvm, type, &addr, true); |
2457 | return (r == -ENODEV) ? -ENXIO : r; | 2458 | return (r == -ENODEV) ? -ENXIO : r; |
2458 | } | 2459 | } |
2459 | |||
2460 | case KVM_DEV_ARM_VGIC_GRP_DIST_REGS: | ||
2461 | case KVM_DEV_ARM_VGIC_GRP_CPU_REGS: { | ||
2462 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | ||
2463 | u32 reg; | ||
2464 | |||
2465 | if (get_user(reg, uaddr)) | ||
2466 | return -EFAULT; | ||
2467 | |||
2468 | return vgic_attr_regs_access(dev, attr, ®, true); | ||
2469 | } | ||
2470 | case KVM_DEV_ARM_VGIC_GRP_NR_IRQS: { | 2460 | case KVM_DEV_ARM_VGIC_GRP_NR_IRQS: { |
2471 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | 2461 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; |
2472 | u32 val; | 2462 | u32 val; |
@@ -2510,7 +2500,33 @@ static int vgic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | |||
2510 | return -ENXIO; | 2500 | return -ENXIO; |
2511 | } | 2501 | } |
2512 | 2502 | ||
2513 | static int vgic_get_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | 2503 | static int vgic_set_attr(struct kvm_device *dev, struct kvm_device_attr *attr) |
2504 | { | ||
2505 | int ret; | ||
2506 | |||
2507 | ret = vgic_set_common_attr(dev, attr); | ||
2508 | if (ret != -ENXIO) | ||
2509 | return ret; | ||
2510 | |||
2511 | switch (attr->group) { | ||
2512 | case KVM_DEV_ARM_VGIC_GRP_DIST_REGS: | ||
2513 | case KVM_DEV_ARM_VGIC_GRP_CPU_REGS: { | ||
2514 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | ||
2515 | u32 reg; | ||
2516 | |||
2517 | if (get_user(reg, uaddr)) | ||
2518 | return -EFAULT; | ||
2519 | |||
2520 | return vgic_attr_regs_access(dev, attr, ®, true); | ||
2521 | } | ||
2522 | |||
2523 | } | ||
2524 | |||
2525 | return -ENXIO; | ||
2526 | } | ||
2527 | |||
2528 | static int vgic_get_common_attr(struct kvm_device *dev, | ||
2529 | struct kvm_device_attr *attr) | ||
2514 | { | 2530 | { |
2515 | int r = -ENXIO; | 2531 | int r = -ENXIO; |
2516 | 2532 | ||
@@ -2528,27 +2544,41 @@ static int vgic_get_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | |||
2528 | return -EFAULT; | 2544 | return -EFAULT; |
2529 | break; | 2545 | break; |
2530 | } | 2546 | } |
2547 | case KVM_DEV_ARM_VGIC_GRP_NR_IRQS: { | ||
2548 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | ||
2549 | |||
2550 | r = put_user(dev->kvm->arch.vgic.nr_irqs, uaddr); | ||
2551 | break; | ||
2552 | } | ||
2553 | |||
2554 | } | ||
2555 | |||
2556 | return r; | ||
2557 | } | ||
2558 | |||
2559 | static int vgic_get_attr(struct kvm_device *dev, struct kvm_device_attr *attr) | ||
2560 | { | ||
2561 | int ret; | ||
2562 | |||
2563 | ret = vgic_get_common_attr(dev, attr); | ||
2564 | if (ret != -ENXIO) | ||
2565 | return ret; | ||
2531 | 2566 | ||
2567 | switch (attr->group) { | ||
2532 | case KVM_DEV_ARM_VGIC_GRP_DIST_REGS: | 2568 | case KVM_DEV_ARM_VGIC_GRP_DIST_REGS: |
2533 | case KVM_DEV_ARM_VGIC_GRP_CPU_REGS: { | 2569 | case KVM_DEV_ARM_VGIC_GRP_CPU_REGS: { |
2534 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | 2570 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; |
2535 | u32 reg = 0; | 2571 | u32 reg = 0; |
2536 | 2572 | ||
2537 | r = vgic_attr_regs_access(dev, attr, ®, false); | 2573 | ret = vgic_attr_regs_access(dev, attr, ®, false); |
2538 | if (r) | 2574 | if (ret) |
2539 | return r; | 2575 | return ret; |
2540 | r = put_user(reg, uaddr); | 2576 | return put_user(reg, uaddr); |
2541 | break; | ||
2542 | } | ||
2543 | case KVM_DEV_ARM_VGIC_GRP_NR_IRQS: { | ||
2544 | u32 __user *uaddr = (u32 __user *)(long)attr->addr; | ||
2545 | r = put_user(dev->kvm->arch.vgic.nr_irqs, uaddr); | ||
2546 | break; | ||
2547 | } | 2577 | } |
2548 | 2578 | ||
2549 | } | 2579 | } |
2550 | 2580 | ||
2551 | return r; | 2581 | return -ENXIO; |
2552 | } | 2582 | } |
2553 | 2583 | ||
2554 | static int vgic_has_attr_regs(const struct mmio_range *ranges, | 2584 | static int vgic_has_attr_regs(const struct mmio_range *ranges, |