blob: 88d52b74a586e302fefaf8c696a193e420dcb3c2 [file] [log] [blame]
/* SPDX-License-Identifier: GPL-2.0-only */
/*
* Perform Set Clock tests
*
* Copyright IBM Corp. 2022
*
* Authors:
* Nico Boehr <nrb@linux.ibm.com>
*/
#include <libcflat.h>
#include <uv.h>
#include <asm/interrupt.h>
#include <asm/time.h>
static inline int sck(uint64_t *time)
{
int cc;
asm volatile(
" sck %[time]\n"
" ipm %[cc]\n"
" srl %[cc],28\n"
: [cc] "=d"(cc)
: [time] "Q"(*time)
: "cc"
);
return cc;
}
static inline int stck(uint64_t *time)
{
int cc;
asm volatile(
" stck %[time]\n"
" ipm %[cc]\n"
" srl %[cc],28\n"
: [cc] "=d" (cc), [time] "=Q" (*time)
:
: "cc", "memory"
);
return cc;
}
static void test_priv(void)
{
uint64_t time_to_set_privileged = 0xfacef00dcafe0000,
time_to_set_nonprivileged = 0xcafe0000,
time_verify;
int cc;
report_prefix_push("privileged");
cc = sck(&time_to_set_privileged);
report(!cc, "set clock cc=%d", cc);
cc = stck(&time_verify);
report(!cc, "store clock cc=%d", cc);
report(time_verify > time_to_set_privileged,
"privileged set affected the clock");
report_prefix_pop();
report_prefix_push("unprivileged");
expect_pgm_int();
enter_pstate();
sck(&time_to_set_nonprivileged);
leave_pstate();
check_pgm_int_code(PGM_INT_CODE_PRIVILEGED_OPERATION);
cc = stck(&time_verify);
report(!cc, "store clock cc=%d", cc);
report(time_verify > time_to_set_privileged,
"unprivileged set did not affect the clock");
report_prefix_pop();
}
static void test_align(void)
{
const int align_to = 8;
char unalign[sizeof(uint64_t) + align_to] __attribute__((aligned(8)));
report_prefix_push("Unaligned operand");
for (int i = 1; i < align_to; i *= 2) {
report_prefix_pushf("%d", i);
expect_pgm_int();
sck((uint64_t *)(unalign + i));
check_pgm_int_code(PGM_INT_CODE_SPECIFICATION);
report_prefix_pop();
}
report_prefix_pop();
}
static void test_set(void)
{
uint64_t start = 0, end = 0, time = 0xcafef00dbeef;
const uint64_t ticks_per_ms = 1000 << 12, ms_to_wait = 5;
int cc;
report_prefix_push("set");
cc = sck(&time);
report(!cc, "set clock cc=%d", cc);
cc = stck(&start);
report(!cc, "store start clock cc=%d", cc);
report(start >= time, "start >= set value");
mdelay(ms_to_wait);
cc = stck(&end);
report(!cc, "store end clock cc=%d", cc);
report(end > time, "end > set value");
report(end - start > (ticks_per_ms * ms_to_wait), "Advances");
report_prefix_pop();
}
int main(void)
{
report_prefix_push("sck");
if (uv_os_is_guest()) {
report_skip("Test unsupported under PV");
goto out;
}
test_align();
test_set();
test_priv();
out:
report_prefix_pop();
return report_summary();
}