blob: 5cc1cd16cfdadbe4b80f675349572b856ce152b1 [file] [log] [blame]
/*
* Test Special Purpose Registers
*
* Copyright 2017 Thomas Huth, Red Hat Inc.
*
* This work is licensed under the terms of the GNU LGPL, version 2.
*
* The basic idea of this test is to check whether the contents of the Special
* Purpose Registers (SPRs) are preserved correctly during migration. So we
* fill in the SPRs with a well-known value, read the values back (since not
* all bits might be retained in the SPRs), then wait for migration to complete
* (if the '-w' option has been specified) so that the user has a chance to
* migrate the VM. Alternatively, the test can also simply sleep a little bit
* with the H_CEDE hypercall, in the hope that we'll get scheduled to another
* host CPU and thus register contents might have changed, too (in case of
* bugs). Finally, we read back the values from the SPRs and compare them with
* the values before the migration. Mismatches are reported as test failures.
* Note that we do not test all SPRs since some of the registers change their
* content automatically, and some are only accessible with hypervisor privi-
* leges or have bad side effects, so we have to omit those registers.
*/
#include <libcflat.h>
#include <util.h>
#include <migrate.h>
#include <alloc.h>
#include <asm/handlers.h>
#include <asm/hcall.h>
#include <asm/processor.h>
#include <asm/barrier.h>
#define mfspr(nr) ({ \
uint64_t ret; \
asm volatile("mfspr %0,%1" : "=r"(ret) : "i"(nr)); \
ret; \
})
#define mtspr(nr, val) \
asm volatile("mtspr %0,%1" : : "i"(nr), "r"(val))
uint64_t before[1024], after[1024];
/* Common SPRs for all PowerPC CPUs */
static void set_sprs_common(uint64_t val)
{
mtspr(9, val); /* CTR */
// mtspr(273, val); /* SPRG1 */ /* Used by our exception handler */
mtspr(274, val); /* SPRG2 */
mtspr(275, val); /* SPRG3 */
}
/* SPRs from PowerPC Operating Environment Architecture, Book III, Vers. 2.01 */
static void set_sprs_book3s_201(uint64_t val)
{
mtspr(18, val); /* DSISR */
mtspr(19, val); /* DAR */
mtspr(152, val); /* CTRL */
mtspr(256, val); /* VRSAVE */
mtspr(786, val); /* MMCRA */
mtspr(795, val); /* MMCR0 */
mtspr(798, val); /* MMCR1 */
}
/* SPRs from PowerISA 2.07 Book III-S */
static void set_sprs_book3s_207(uint64_t val)
{
mtspr(3, val); /* DSCR */
mtspr(13, val); /* AMR */
mtspr(17, val); /* DSCR */
mtspr(18, val); /* DSISR */
mtspr(19, val); /* DAR */
mtspr(29, val); /* AMR */
mtspr(61, val); /* IAMR */
// mtspr(152, val); /* CTRL */ /* TODO: Needs a fix in KVM */
mtspr(153, val); /* FSCR */
mtspr(157, val); /* UAMOR */
mtspr(159, val); /* PSPB */
mtspr(256, val); /* VRSAVE */
// mtspr(272, val); /* SPRG0 */ /* Used by our exception handler */
mtspr(769, val); /* MMCR2 */
mtspr(770, val); /* MMCRA */
mtspr(771, val); /* PMC1 */
mtspr(772, val); /* PMC2 */
mtspr(773, val); /* PMC3 */
mtspr(774, val); /* PMC4 */
mtspr(775, val); /* PMC5 */
mtspr(776, val); /* PMC6 */
mtspr(779, (val & 0xfffffffffbab3fffULL) | 0xfa0b2070); /* MMCR0 */
mtspr(784, val); /* SIER */
mtspr(785, val); /* MMCR2 */
mtspr(786, val); /* MMCRA */
mtspr(787, val); /* PMC1 */
mtspr(788, val); /* PMC2 */
mtspr(789, val); /* PMC3 */
mtspr(790, val); /* PMC4 */
mtspr(791, val); /* PMC5 */
mtspr(792, val); /* PMC6 */
mtspr(795, (val & 0xfffffffffbab3fffULL) | 0xfa0b2070); /* MMCR0 */
mtspr(796, val); /* SIAR */
mtspr(797, val); /* SDAR */
mtspr(798, val); /* MMCR1 */
mtspr(800, val); /* BESCRS */
mtspr(801, val); /* BESCCRSU */
mtspr(802, val); /* BESCRR */
mtspr(803, val); /* BESCRRU */
mtspr(804, val); /* EBBHR */
mtspr(805, val); /* EBBRR */
mtspr(806, val); /* BESCR */
mtspr(815, val); /* TAR */
}
/* SPRs from PowerISA 3.00 Book III */
static void set_sprs_book3s_300(uint64_t val)
{
set_sprs_book3s_207(val);
mtspr(48, val); /* PIDR */
mtspr(144, val); /* TIDR */
mtspr(823, val); /* PSSCR */
}
static void set_sprs(uint64_t val)
{
uint32_t pvr = mfspr(287); /* Processor Version Register */
set_sprs_common(val);
switch (pvr >> 16) {
case 0x39: /* PPC970 */
case 0x3C: /* PPC970FX */
case 0x44: /* PPC970MP */
set_sprs_book3s_201(val);
break;
case 0x4b: /* POWER8E */
case 0x4c: /* POWER8NVL */
case 0x4d: /* POWER8 */
set_sprs_book3s_207(val);
break;
case 0x4e: /* POWER9 */
set_sprs_book3s_300(val);
break;
default:
puts("Warning: Unknown processor version!\n");
}
}
static void get_sprs_common(uint64_t *v)
{
v[9] = mfspr(9); /* CTR */
// v[273] = mfspr(273); /* SPRG1 */ /* Used by our exception handler */
v[274] = mfspr(274); /* SPRG2 */
v[275] = mfspr(275); /* SPRG3 */
}
static void get_sprs_book3s_201(uint64_t *v)
{
v[18] = mfspr(18); /* DSISR */
v[19] = mfspr(19); /* DAR */
v[136] = mfspr(136); /* CTRL */
v[256] = mfspr(256); /* VRSAVE */
v[786] = mfspr(786); /* MMCRA */
v[795] = mfspr(795); /* MMCR0 */
v[798] = mfspr(798); /* MMCR1 */
}
static void get_sprs_book3s_207(uint64_t *v)
{
v[3] = mfspr(3); /* DSCR */
v[13] = mfspr(13); /* AMR */
v[17] = mfspr(17); /* DSCR */
v[18] = mfspr(18); /* DSISR */
v[19] = mfspr(19); /* DAR */
v[29] = mfspr(29); /* AMR */
v[61] = mfspr(61); /* IAMR */
// v[136] = mfspr(136); /* CTRL */ /* TODO: Needs a fix in KVM */
v[153] = mfspr(153); /* FSCR */
v[157] = mfspr(157); /* UAMOR */
v[159] = mfspr(159); /* PSPB */
v[256] = mfspr(256); /* VRSAVE */
v[259] = mfspr(259); /* SPRG3 (read only) */
// v[272] = mfspr(272); /* SPRG0 */ /* Used by our exception handler */
v[769] = mfspr(769); /* MMCR2 */
v[770] = mfspr(770); /* MMCRA */
v[771] = mfspr(771); /* PMC1 */
v[772] = mfspr(772); /* PMC2 */
v[773] = mfspr(773); /* PMC3 */
v[774] = mfspr(774); /* PMC4 */
v[775] = mfspr(775); /* PMC5 */
v[776] = mfspr(776); /* PMC6 */
v[779] = mfspr(779); /* MMCR0 */
v[780] = mfspr(780); /* SIAR (read only) */
v[781] = mfspr(781); /* SDAR (read only) */
v[782] = mfspr(782); /* MMCR1 (read only) */
v[784] = mfspr(784); /* SIER */
v[785] = mfspr(785); /* MMCR2 */
v[786] = mfspr(786); /* MMCRA */
v[787] = mfspr(787); /* PMC1 */
v[788] = mfspr(788); /* PMC2 */
v[789] = mfspr(789); /* PMC3 */
v[790] = mfspr(790); /* PMC4 */
v[791] = mfspr(791); /* PMC5 */
v[792] = mfspr(792); /* PMC6 */
v[795] = mfspr(795); /* MMCR0 */
v[796] = mfspr(796); /* SIAR */
v[797] = mfspr(797); /* SDAR */
v[798] = mfspr(798); /* MMCR1 */
v[800] = mfspr(800); /* BESCRS */
v[801] = mfspr(801); /* BESCCRSU */
v[802] = mfspr(802); /* BESCRR */
v[803] = mfspr(803); /* BESCRRU */
v[804] = mfspr(804); /* EBBHR */
v[805] = mfspr(805); /* EBBRR */
v[806] = mfspr(806); /* BESCR */
v[815] = mfspr(815); /* TAR */
}
static void get_sprs_book3s_300(uint64_t *v)
{
get_sprs_book3s_207(v);
v[48] = mfspr(48); /* PIDR */
v[144] = mfspr(144); /* TIDR */
v[823] = mfspr(823); /* PSSCR */
}
static void get_sprs(uint64_t *v)
{
uint32_t pvr = mfspr(287); /* Processor Version Register */
get_sprs_common(v);
switch (pvr >> 16) {
case 0x39: /* PPC970 */
case 0x3C: /* PPC970FX */
case 0x44: /* PPC970MP */
get_sprs_book3s_201(v);
break;
case 0x4b: /* POWER8E */
case 0x4c: /* POWER8NVL */
case 0x4d: /* POWER8 */
get_sprs_book3s_207(v);
break;
case 0x4e: /* POWER9 */
get_sprs_book3s_300(v);
break;
}
}
int main(int argc, char **argv)
{
int i;
bool pause = false;
uint64_t pat = 0xcafefacec0debabeULL;
const uint64_t patterns[] = {
0xcafefacec0debabeULL, ~0xcafefacec0debabeULL,
0xAAAA5555AAAA5555ULL, 0x5555AAAA5555AAAAULL,
0x1234567890ABCDEFULL, 0xFEDCBA0987654321ULL,
-1ULL,
};
static uint64_t decr = 0x7FFFFFFF; /* Max value */
for (i = 1; i < argc; i++) {
if (!strcmp(argv[i], "-w")) {
pause = true;
} else if (!strcmp(argv[i], "-p")) {
i += 1;
if (i >= argc || *argv[i] < '0'
|| *argv[i] >= '0' + ARRAY_SIZE(patterns))
report_abort("Error: bad value for -p");
pat ^= patterns[*argv[i] - '0'];
} else if (!strcmp(argv[i], "-t")) {
/* Randomize with timebase register */
asm volatile("mftb %0" : "=r"(i));
pat ^= i;
asm volatile("mftb %0" : "=r"(i));
pat ^= ~(uint64_t)i << 32;
} else {
report_abort("Warning: Unsupported argument: %s",
argv[i]);
}
}
printf("Settings SPRs to %#lx...\n", pat);
set_sprs(pat);
memset(before, 0, sizeof(before));
memset(after, 0, sizeof(after));
get_sprs(before);
if (pause) {
migrate_once();
} else {
puts("Sleeping...\n");
handle_exception(0x900, &dec_except_handler, &decr);
asm volatile ("mtdec %0" : : "r" (0x3FFFFFFF));
hcall(H_CEDE);
}
get_sprs(after);
puts("Checking SPRs...\n");
for (i = 0; i < 1024; i++) {
if (before[i] != 0 || after[i] != 0)
report(before[i] == after[i],
"SPR %d:\t%#018lx <==> %#018lx", i, before[i],
after[i]);
}
return report_summary();
}