aboutsummaryrefslogtreecommitdiffstats
path: root/arch/arm/kvm/coproc.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/arm/kvm/coproc.c')
-rw-r--r--arch/arm/kvm/coproc.c164
1 files changed, 161 insertions, 3 deletions
diff --git a/arch/arm/kvm/coproc.c b/arch/arm/kvm/coproc.c
index 95a0f5e5c1fc..1827b643af15 100644
--- a/arch/arm/kvm/coproc.c
+++ b/arch/arm/kvm/coproc.c
@@ -35,6 +35,12 @@
35 * Co-processor emulation 35 * Co-processor emulation
36 *****************************************************************************/ 36 *****************************************************************************/
37 37
38/* 3 bits per cache level, as per CLIDR, but non-existent caches always 0 */
39static u32 cache_levels;
40
41/* CSSELR values; used to index KVM_REG_ARM_DEMUX_ID_CCSIDR */
42#define CSSELR_MAX 12
43
38int kvm_handle_cp10_id(struct kvm_vcpu *vcpu, struct kvm_run *run) 44int kvm_handle_cp10_id(struct kvm_vcpu *vcpu, struct kvm_run *run)
39{ 45{
40 kvm_inject_undefined(vcpu); 46 kvm_inject_undefined(vcpu);
@@ -548,11 +554,113 @@ static int set_invariant_cp15(u64 id, void __user *uaddr)
548 return 0; 554 return 0;
549} 555}
550 556
557static bool is_valid_cache(u32 val)
558{
559 u32 level, ctype;
560
561 if (val >= CSSELR_MAX)
562 return -ENOENT;
563
564 /* Bottom bit is Instruction or Data bit. Next 3 bits are level. */
565 level = (val >> 1);
566 ctype = (cache_levels >> (level * 3)) & 7;
567
568 switch (ctype) {
569 case 0: /* No cache */
570 return false;
571 case 1: /* Instruction cache only */
572 return (val & 1);
573 case 2: /* Data cache only */
574 case 4: /* Unified cache */
575 return !(val & 1);
576 case 3: /* Separate instruction and data caches */
577 return true;
578 default: /* Reserved: we can't know instruction or data. */
579 return false;
580 }
581}
582
583/* Which cache CCSIDR represents depends on CSSELR value. */
584static u32 get_ccsidr(u32 csselr)
585{
586 u32 ccsidr;
587
588 /* Make sure noone else changes CSSELR during this! */
589 local_irq_disable();
590 /* Put value into CSSELR */
591 asm volatile("mcr p15, 2, %0, c0, c0, 0" : : "r" (csselr));
592 isb();
593 /* Read result out of CCSIDR */
594 asm volatile("mrc p15, 1, %0, c0, c0, 0" : "=r" (ccsidr));
595 local_irq_enable();
596
597 return ccsidr;
598}
599
600static int demux_c15_get(u64 id, void __user *uaddr)
601{
602 u32 val;
603 u32 __user *uval = uaddr;
604
605 /* Fail if we have unknown bits set. */
606 if (id & ~(KVM_REG_ARCH_MASK|KVM_REG_SIZE_MASK|KVM_REG_ARM_COPROC_MASK
607 | ((1 << KVM_REG_ARM_COPROC_SHIFT)-1)))
608 return -ENOENT;
609
610 switch (id & KVM_REG_ARM_DEMUX_ID_MASK) {
611 case KVM_REG_ARM_DEMUX_ID_CCSIDR:
612 if (KVM_REG_SIZE(id) != 4)
613 return -ENOENT;
614 val = (id & KVM_REG_ARM_DEMUX_VAL_MASK)
615 >> KVM_REG_ARM_DEMUX_VAL_SHIFT;
616 if (!is_valid_cache(val))
617 return -ENOENT;
618
619 return put_user(get_ccsidr(val), uval);
620 default:
621 return -ENOENT;
622 }
623}
624
625static int demux_c15_set(u64 id, void __user *uaddr)
626{
627 u32 val, newval;
628 u32 __user *uval = uaddr;
629
630 /* Fail if we have unknown bits set. */
631 if (id & ~(KVM_REG_ARCH_MASK|KVM_REG_SIZE_MASK|KVM_REG_ARM_COPROC_MASK
632 | ((1 << KVM_REG_ARM_COPROC_SHIFT)-1)))
633 return -ENOENT;
634
635 switch (id & KVM_REG_ARM_DEMUX_ID_MASK) {
636 case KVM_REG_ARM_DEMUX_ID_CCSIDR:
637 if (KVM_REG_SIZE(id) != 4)
638 return -ENOENT;
639 val = (id & KVM_REG_ARM_DEMUX_VAL_MASK)
640 >> KVM_REG_ARM_DEMUX_VAL_SHIFT;
641 if (!is_valid_cache(val))
642 return -ENOENT;
643
644 if (get_user(newval, uval))
645 return -EFAULT;
646
647 /* This is also invariant: you can't change it. */
648 if (newval != get_ccsidr(val))
649 return -EINVAL;
650 return 0;
651 default:
652 return -ENOENT;
653 }
654}
655
551int kvm_arm_coproc_get_reg(struct kvm_vcpu *vcpu, const struct kvm_one_reg *reg) 656int kvm_arm_coproc_get_reg(struct kvm_vcpu *vcpu, const struct kvm_one_reg *reg)
552{ 657{
553 const struct coproc_reg *r; 658 const struct coproc_reg *r;
554 void __user *uaddr = (void __user *)(long)reg->addr; 659 void __user *uaddr = (void __user *)(long)reg->addr;
555 660
661 if ((reg->id & KVM_REG_ARM_COPROC_MASK) == KVM_REG_ARM_DEMUX)
662 return demux_c15_get(reg->id, uaddr);
663
556 r = index_to_coproc_reg(vcpu, reg->id); 664 r = index_to_coproc_reg(vcpu, reg->id);
557 if (!r) 665 if (!r)
558 return get_invariant_cp15(reg->id, uaddr); 666 return get_invariant_cp15(reg->id, uaddr);
@@ -566,6 +674,9 @@ int kvm_arm_coproc_set_reg(struct kvm_vcpu *vcpu, const struct kvm_one_reg *reg)
566 const struct coproc_reg *r; 674 const struct coproc_reg *r;
567 void __user *uaddr = (void __user *)(long)reg->addr; 675 void __user *uaddr = (void __user *)(long)reg->addr;
568 676
677 if ((reg->id & KVM_REG_ARM_COPROC_MASK) == KVM_REG_ARM_DEMUX)
678 return demux_c15_set(reg->id, uaddr);
679
569 r = index_to_coproc_reg(vcpu, reg->id); 680 r = index_to_coproc_reg(vcpu, reg->id);
570 if (!r) 681 if (!r)
571 return set_invariant_cp15(reg->id, uaddr); 682 return set_invariant_cp15(reg->id, uaddr);
@@ -574,6 +685,33 @@ int kvm_arm_coproc_set_reg(struct kvm_vcpu *vcpu, const struct kvm_one_reg *reg)
574 return reg_from_user(&vcpu->arch.cp15[r->reg], uaddr, reg->id); 685 return reg_from_user(&vcpu->arch.cp15[r->reg], uaddr, reg->id);
575} 686}
576 687
688static unsigned int num_demux_regs(void)
689{
690 unsigned int i, count = 0;
691
692 for (i = 0; i < CSSELR_MAX; i++)
693 if (is_valid_cache(i))
694 count++;
695
696 return count;
697}
698
699static int write_demux_regids(u64 __user *uindices)
700{
701 u64 val = KVM_REG_ARM | KVM_REG_SIZE_U32 | KVM_REG_ARM_DEMUX;
702 unsigned int i;
703
704 val |= KVM_REG_ARM_DEMUX_ID_CCSIDR;
705 for (i = 0; i < CSSELR_MAX; i++) {
706 if (!is_valid_cache(i))
707 continue;
708 if (put_user(val | i, uindices))
709 return -EFAULT;
710 uindices++;
711 }
712 return 0;
713}
714
577static u64 cp15_to_index(const struct coproc_reg *reg) 715static u64 cp15_to_index(const struct coproc_reg *reg)
578{ 716{
579 u64 val = KVM_REG_ARM | (15 << KVM_REG_ARM_COPROC_SHIFT); 717 u64 val = KVM_REG_ARM | (15 << KVM_REG_ARM_COPROC_SHIFT);
@@ -649,6 +787,7 @@ static int walk_cp15(struct kvm_vcpu *vcpu, u64 __user *uind)
649unsigned long kvm_arm_num_coproc_regs(struct kvm_vcpu *vcpu) 787unsigned long kvm_arm_num_coproc_regs(struct kvm_vcpu *vcpu)
650{ 788{
651 return ARRAY_SIZE(invariant_cp15) 789 return ARRAY_SIZE(invariant_cp15)
790 + num_demux_regs()
652 + walk_cp15(vcpu, (u64 __user *)NULL); 791 + walk_cp15(vcpu, (u64 __user *)NULL);
653} 792}
654 793
@@ -665,9 +804,11 @@ int kvm_arm_copy_coproc_indices(struct kvm_vcpu *vcpu, u64 __user *uindices)
665 } 804 }
666 805
667 err = walk_cp15(vcpu, uindices); 806 err = walk_cp15(vcpu, uindices);
668 if (err > 0) 807 if (err < 0)
669 err = 0; 808 return err;
670 return err; 809 uindices += err;
810
811 return write_demux_regids(uindices);
671} 812}
672 813
673void kvm_coproc_table_init(void) 814void kvm_coproc_table_init(void)
@@ -681,6 +822,23 @@ void kvm_coproc_table_init(void)
681 /* We abuse the reset function to overwrite the table itself. */ 822 /* We abuse the reset function to overwrite the table itself. */
682 for (i = 0; i < ARRAY_SIZE(invariant_cp15); i++) 823 for (i = 0; i < ARRAY_SIZE(invariant_cp15); i++)
683 invariant_cp15[i].reset(NULL, &invariant_cp15[i]); 824 invariant_cp15[i].reset(NULL, &invariant_cp15[i]);
825
826 /*
827 * CLIDR format is awkward, so clean it up. See ARM B4.1.20:
828 *
829 * If software reads the Cache Type fields from Ctype1
830 * upwards, once it has seen a value of 0b000, no caches
831 * exist at further-out levels of the hierarchy. So, for
832 * example, if Ctype3 is the first Cache Type field with a
833 * value of 0b000, the values of Ctype4 to Ctype7 must be
834 * ignored.
835 */
836 asm volatile("mrc p15, 1, %0, c0, c0, 1" : "=r" (cache_levels));
837 for (i = 0; i < 7; i++)
838 if (((cache_levels >> (i*3)) & 7) == 0)
839 break;
840 /* Clear all higher bits. */
841 cache_levels &= (1 << (i*3))-1;
684} 842}
685 843
686/** 844/**