Commit a50623fb authored by Paul Mackerras's avatar Paul Mackerras

KVM: PPC: Book3S PR: Fix failure status setting in treclaim. emulation

The treclaim. emulation needs to record failure status in the TEXASR
register if the transaction had not previously failed.  However, the
current code first does kvmppc_save_tm_pr() (which does a treclaim.
itself) and then checks the failure summary bit in TEXASR after that.
Since treclaim. itself causes transaction failure, the FS bit is
always set, so we were never updating TEXASR with the failure cause
supplied by the guest as the RA parameter to the treclaim. instruction.
This caused the tm-unavailable test in tools/testing/selftests/powerpc/tm
to fail.

To fix this, we need to read TEXASR before calling kvmppc_save_tm_pr(),
and base the final value of TEXASR on that value.

Fixes: 03c81682 ("KVM: PPC: Book3S PR: Add emulation for treclaim.")
Reviewed-by: default avatarSimon Guo <wei.guo.simon@gmail.com>
Signed-off-by: default avatarPaul Mackerras <paulus@ozlabs.org>
parent 916ccadc
...@@ -138,6 +138,7 @@ static void kvmppc_emulate_treclaim(struct kvm_vcpu *vcpu, int ra_val) ...@@ -138,6 +138,7 @@ static void kvmppc_emulate_treclaim(struct kvm_vcpu *vcpu, int ra_val)
{ {
unsigned long guest_msr = kvmppc_get_msr(vcpu); unsigned long guest_msr = kvmppc_get_msr(vcpu);
int fc_val = ra_val ? ra_val : 1; int fc_val = ra_val ? ra_val : 1;
uint64_t texasr;
/* CR0 = 0 | MSR[TS] | 0 */ /* CR0 = 0 | MSR[TS] | 0 */
vcpu->arch.cr = (vcpu->arch.cr & ~(CR0_MASK << CR0_SHIFT)) | vcpu->arch.cr = (vcpu->arch.cr & ~(CR0_MASK << CR0_SHIFT)) |
...@@ -145,25 +146,26 @@ static void kvmppc_emulate_treclaim(struct kvm_vcpu *vcpu, int ra_val) ...@@ -145,25 +146,26 @@ static void kvmppc_emulate_treclaim(struct kvm_vcpu *vcpu, int ra_val)
<< CR0_SHIFT); << CR0_SHIFT);
preempt_disable(); preempt_disable();
tm_enable();
texasr = mfspr(SPRN_TEXASR);
kvmppc_save_tm_pr(vcpu); kvmppc_save_tm_pr(vcpu);
kvmppc_copyfrom_vcpu_tm(vcpu); kvmppc_copyfrom_vcpu_tm(vcpu);
tm_enable();
vcpu->arch.texasr = mfspr(SPRN_TEXASR);
/* failure recording depends on Failure Summary bit */ /* failure recording depends on Failure Summary bit */
if (!(vcpu->arch.texasr & TEXASR_FS)) { if (!(texasr & TEXASR_FS)) {
vcpu->arch.texasr &= ~TEXASR_FC; texasr &= ~TEXASR_FC;
vcpu->arch.texasr |= ((u64)fc_val << TEXASR_FC_LG); texasr |= ((u64)fc_val << TEXASR_FC_LG) | TEXASR_FS;
vcpu->arch.texasr &= ~(TEXASR_PR | TEXASR_HV); texasr &= ~(TEXASR_PR | TEXASR_HV);
if (kvmppc_get_msr(vcpu) & MSR_PR) if (kvmppc_get_msr(vcpu) & MSR_PR)
vcpu->arch.texasr |= TEXASR_PR; texasr |= TEXASR_PR;
if (kvmppc_get_msr(vcpu) & MSR_HV) if (kvmppc_get_msr(vcpu) & MSR_HV)
vcpu->arch.texasr |= TEXASR_HV; texasr |= TEXASR_HV;
vcpu->arch.texasr = texasr;
vcpu->arch.tfiar = kvmppc_get_pc(vcpu); vcpu->arch.tfiar = kvmppc_get_pc(vcpu);
mtspr(SPRN_TEXASR, vcpu->arch.texasr); mtspr(SPRN_TEXASR, texasr);
mtspr(SPRN_TFIAR, vcpu->arch.tfiar); mtspr(SPRN_TFIAR, vcpu->arch.tfiar);
} }
tm_disable(); tm_disable();
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment