diff options
author | Alexander Graf <agraf@suse.de> | 2010-08-05 09:44:41 -0400 |
---|---|---|
committer | Avi Kivity <avi@redhat.com> | 2010-10-24 04:52:14 -0400 |
commit | df08bd10266ce6132278f6b4ddc4bb0a12330b56 (patch) | |
tree | c0edb224d7b9fa46d18b66e6fcb622005f6e91c3 /arch/powerpc/kernel | |
parent | 9ee18b1e08e6a5648aeaaf998eabc72b5304cc17 (diff) |
KVM: PPC: Make PV mtmsrd L=1 work with r30 and r31
We had an arbitrary limitation in mtmsrd L=1 that kept us from using r30 and
r31 as input registers. Let's get rid of that and get more potential speedups!
Signed-off-by: Alexander Graf <agraf@suse.de>
Diffstat (limited to 'arch/powerpc/kernel')
-rw-r--r-- | arch/powerpc/kernel/kvm.c | 21 | ||||
-rw-r--r-- | arch/powerpc/kernel/kvm_emul.S | 8 |
2 files changed, 24 insertions, 5 deletions
diff --git a/arch/powerpc/kernel/kvm.c b/arch/powerpc/kernel/kvm.c index 10b681c092ed..48a033865410 100644 --- a/arch/powerpc/kernel/kvm.c +++ b/arch/powerpc/kernel/kvm.c | |||
@@ -158,6 +158,7 @@ static u32 *kvm_alloc(int len) | |||
158 | 158 | ||
159 | extern u32 kvm_emulate_mtmsrd_branch_offs; | 159 | extern u32 kvm_emulate_mtmsrd_branch_offs; |
160 | extern u32 kvm_emulate_mtmsrd_reg_offs; | 160 | extern u32 kvm_emulate_mtmsrd_reg_offs; |
161 | extern u32 kvm_emulate_mtmsrd_orig_ins_offs; | ||
161 | extern u32 kvm_emulate_mtmsrd_len; | 162 | extern u32 kvm_emulate_mtmsrd_len; |
162 | extern u32 kvm_emulate_mtmsrd[]; | 163 | extern u32 kvm_emulate_mtmsrd[]; |
163 | 164 | ||
@@ -186,7 +187,21 @@ static void kvm_patch_ins_mtmsrd(u32 *inst, u32 rt) | |||
186 | /* Modify the chunk to fit the invocation */ | 187 | /* Modify the chunk to fit the invocation */ |
187 | memcpy(p, kvm_emulate_mtmsrd, kvm_emulate_mtmsrd_len * 4); | 188 | memcpy(p, kvm_emulate_mtmsrd, kvm_emulate_mtmsrd_len * 4); |
188 | p[kvm_emulate_mtmsrd_branch_offs] |= distance_end & KVM_INST_B_MASK; | 189 | p[kvm_emulate_mtmsrd_branch_offs] |= distance_end & KVM_INST_B_MASK; |
189 | p[kvm_emulate_mtmsrd_reg_offs] |= rt; | 190 | switch (get_rt(rt)) { |
191 | case 30: | ||
192 | kvm_patch_ins_ll(&p[kvm_emulate_mtmsrd_reg_offs], | ||
193 | magic_var(scratch2), KVM_RT_30); | ||
194 | break; | ||
195 | case 31: | ||
196 | kvm_patch_ins_ll(&p[kvm_emulate_mtmsrd_reg_offs], | ||
197 | magic_var(scratch1), KVM_RT_30); | ||
198 | break; | ||
199 | default: | ||
200 | p[kvm_emulate_mtmsrd_reg_offs] |= rt; | ||
201 | break; | ||
202 | } | ||
203 | |||
204 | p[kvm_emulate_mtmsrd_orig_ins_offs] = *inst; | ||
190 | flush_icache_range((ulong)p, (ulong)p + kvm_emulate_mtmsrd_len * 4); | 205 | flush_icache_range((ulong)p, (ulong)p + kvm_emulate_mtmsrd_len * 4); |
191 | 206 | ||
192 | /* Patch the invocation */ | 207 | /* Patch the invocation */ |
@@ -423,9 +438,7 @@ static void kvm_check_ins(u32 *inst, u32 features) | |||
423 | 438 | ||
424 | /* Rewrites */ | 439 | /* Rewrites */ |
425 | case KVM_INST_MTMSRD_L1: | 440 | case KVM_INST_MTMSRD_L1: |
426 | /* We use r30 and r31 during the hook */ | 441 | kvm_patch_ins_mtmsrd(inst, inst_rt); |
427 | if (get_rt(inst_rt) < 30) | ||
428 | kvm_patch_ins_mtmsrd(inst, inst_rt); | ||
429 | break; | 442 | break; |
430 | case KVM_INST_MTMSR: | 443 | case KVM_INST_MTMSR: |
431 | case KVM_INST_MTMSRD_L0: | 444 | case KVM_INST_MTMSRD_L0: |
diff --git a/arch/powerpc/kernel/kvm_emul.S b/arch/powerpc/kernel/kvm_emul.S index 65305325250b..f2b1b2523e61 100644 --- a/arch/powerpc/kernel/kvm_emul.S +++ b/arch/powerpc/kernel/kvm_emul.S | |||
@@ -78,7 +78,8 @@ kvm_emulate_mtmsrd: | |||
78 | 78 | ||
79 | /* OR the register's (MSR_EE|MSR_RI) on MSR */ | 79 | /* OR the register's (MSR_EE|MSR_RI) on MSR */ |
80 | kvm_emulate_mtmsrd_reg: | 80 | kvm_emulate_mtmsrd_reg: |
81 | andi. r30, r0, (MSR_EE|MSR_RI) | 81 | ori r30, r0, 0 |
82 | andi. r30, r30, (MSR_EE|MSR_RI) | ||
82 | or r31, r31, r30 | 83 | or r31, r31, r30 |
83 | 84 | ||
84 | /* Put MSR back into magic page */ | 85 | /* Put MSR back into magic page */ |
@@ -96,6 +97,7 @@ kvm_emulate_mtmsrd_reg: | |||
96 | SCRATCH_RESTORE | 97 | SCRATCH_RESTORE |
97 | 98 | ||
98 | /* Nag hypervisor */ | 99 | /* Nag hypervisor */ |
100 | kvm_emulate_mtmsrd_orig_ins: | ||
99 | tlbsync | 101 | tlbsync |
100 | 102 | ||
101 | b kvm_emulate_mtmsrd_branch | 103 | b kvm_emulate_mtmsrd_branch |
@@ -117,6 +119,10 @@ kvm_emulate_mtmsrd_branch_offs: | |||
117 | kvm_emulate_mtmsrd_reg_offs: | 119 | kvm_emulate_mtmsrd_reg_offs: |
118 | .long (kvm_emulate_mtmsrd_reg - kvm_emulate_mtmsrd) / 4 | 120 | .long (kvm_emulate_mtmsrd_reg - kvm_emulate_mtmsrd) / 4 |
119 | 121 | ||
122 | .global kvm_emulate_mtmsrd_orig_ins_offs | ||
123 | kvm_emulate_mtmsrd_orig_ins_offs: | ||
124 | .long (kvm_emulate_mtmsrd_orig_ins - kvm_emulate_mtmsrd) / 4 | ||
125 | |||
120 | .global kvm_emulate_mtmsrd_len | 126 | .global kvm_emulate_mtmsrd_len |
121 | kvm_emulate_mtmsrd_len: | 127 | kvm_emulate_mtmsrd_len: |
122 | .long (kvm_emulate_mtmsrd_end - kvm_emulate_mtmsrd) / 4 | 128 | .long (kvm_emulate_mtmsrd_end - kvm_emulate_mtmsrd) / 4 |