blob: 44ad21541faca4288a6c02bb52ef0e724d0bceda [file] [log] [blame]
/* SPDX-License-Identifier: GPL-2.0-only */
/*
* Guest Ultravisor Call tests
*
* Copyright (c) 2020 IBM Corp
*
* Authors:
* Janosch Frank <frankja@linux.ibm.com>
*/
#include <libcflat.h>
#include <alloc_page.h>
#include <asm/page.h>
#include <asm/asm-offsets.h>
#include <asm/interrupt.h>
#include <asm/facility.h>
#include <asm/uv.h>
#include <sclp.h>
#include <uv.h>
static unsigned long page;
static void test_priv(void)
{
struct uv_cb_header uvcb = {};
report_prefix_push("privileged");
report_prefix_push("query");
uvcb.cmd = UVC_CMD_QUI;
uvcb.len = sizeof(struct uv_cb_qui);
expect_pgm_int();
enter_pstate();
uv_call_once(0, (u64)&uvcb);
check_pgm_int_code(PGM_INT_CODE_PRIVILEGED_OPERATION);
report_prefix_pop();
report_prefix_push("share");
uvcb.cmd = UVC_CMD_SET_SHARED_ACCESS;
uvcb.len = sizeof(struct uv_cb_share);
expect_pgm_int();
enter_pstate();
uv_call_once(0, (u64)&uvcb);
check_pgm_int_code(PGM_INT_CODE_PRIVILEGED_OPERATION);
report_prefix_pop();
report_prefix_push("unshare");
uvcb.cmd = UVC_CMD_REMOVE_SHARED_ACCESS;
uvcb.len = sizeof(struct uv_cb_share);
expect_pgm_int();
enter_pstate();
uv_call_once(0, (u64)&uvcb);
check_pgm_int_code(PGM_INT_CODE_PRIVILEGED_OPERATION);
report_prefix_pop();
report_prefix_pop();
}
static void test_query(void)
{
struct uv_cb_qui uvcb = {
.header.cmd = UVC_CMD_QUI,
/* A dword below the minimum length */
.header.len = 0xa0,
};
int cc;
report_prefix_push("query");
cc = uv_call(0, (u64)&uvcb);
report(cc == 1 && uvcb.header.rc == UVC_RC_INV_LEN, "length");
uvcb.header.len = sizeof(uvcb);
cc = uv_call(0, (u64)&uvcb);
report((!cc && uvcb.header.rc == UVC_RC_EXECUTED) ||
(cc == 1 && uvcb.header.rc == 0x100),
"successful query");
/*
* These bits have been introduced with the very first
* Ultravisor version and are expected to always be available
* because they are basic building blocks.
*/
report(test_bit_inv(BIT_UVC_CMD_QUI, &uvcb.inst_calls_list[0]),
"query indicated");
report(test_bit_inv(BIT_UVC_CMD_SET_SHARED_ACCESS, &uvcb.inst_calls_list[0]),
"share indicated");
report(test_bit_inv(BIT_UVC_CMD_REMOVE_SHARED_ACCESS, &uvcb.inst_calls_list[0]),
"unshare indicated");
report_prefix_pop();
}
static void test_sharing(void)
{
struct uv_cb_share uvcb = {
.header.cmd = UVC_CMD_SET_SHARED_ACCESS,
.header.len = sizeof(uvcb) - 8,
.paddr = page,
};
int cc;
report_prefix_push("share");
cc = uv_call(0, (u64)&uvcb);
report(cc == 1 && uvcb.header.rc == UVC_RC_INV_LEN, "length");
uvcb.header.len = sizeof(uvcb);
cc = uv_call(0, (u64)&uvcb);
report(cc == 0 && uvcb.header.rc == UVC_RC_EXECUTED, "share");
uvcb.paddr = get_ram_size() + PAGE_SIZE;
cc = uv_call(0, (u64)&uvcb);
report(cc == 1 && uvcb.header.rc == 0x101, "invalid memory");
uvcb.paddr = page;
report_prefix_pop();
report_prefix_push("unshare");
uvcb.header.cmd = UVC_CMD_REMOVE_SHARED_ACCESS;
uvcb.header.len -= 8;
cc = uv_call(0, (u64)&uvcb);
report(cc == 1 && uvcb.header.rc == UVC_RC_INV_LEN, "length");
uvcb.header.len = sizeof(uvcb);
cc = uv_call(0, (u64)&uvcb);
report(cc == 0 && uvcb.header.rc == UVC_RC_EXECUTED, "unshare");
report_prefix_pop();
report_prefix_pop();
}
static struct {
const char *name;
uint16_t cmd;
uint16_t len;
int call_bit;
} invalid_cmds[] = {
{ "bogus", 0x4242, sizeof(struct uv_cb_header), -1 },
{ "init", UVC_CMD_INIT_UV, sizeof(struct uv_cb_init), BIT_UVC_CMD_INIT_UV },
{ "create conf", UVC_CMD_CREATE_SEC_CONF, sizeof(struct uv_cb_cgc), BIT_UVC_CMD_CREATE_SEC_CONF },
{ "destroy conf", UVC_CMD_DESTROY_SEC_CONF, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_DESTROY_SEC_CONF },
{ "create cpu", UVC_CMD_CREATE_SEC_CPU, sizeof(struct uv_cb_csc), BIT_UVC_CMD_CREATE_SEC_CPU },
{ "destroy cpu", UVC_CMD_DESTROY_SEC_CPU, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_DESTROY_SEC_CPU },
{ "conv to", UVC_CMD_CONV_TO_SEC_STOR, sizeof(struct uv_cb_cts), BIT_UVC_CMD_CONV_TO_SEC_STOR },
{ "conv from", UVC_CMD_CONV_FROM_SEC_STOR, sizeof(struct uv_cb_cfs), BIT_UVC_CMD_CONV_FROM_SEC_STOR },
{ "set sec conf", UVC_CMD_SET_SEC_CONF_PARAMS, sizeof(struct uv_cb_ssc), BIT_UVC_CMD_SET_SEC_PARMS },
{ "unpack", UVC_CMD_UNPACK_IMG, sizeof(struct uv_cb_unp), BIT_UVC_CMD_UNPACK_IMG },
{ "verify", UVC_CMD_VERIFY_IMG, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_VERIFY_IMG },
{ "cpu reset", UVC_CMD_CPU_RESET, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_CPU_RESET },
{ "cpu initial reset", UVC_CMD_CPU_RESET_INITIAL, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_CPU_RESET_INITIAL },
{ "conf clear reset", UVC_CMD_PERF_CONF_CLEAR_RESET, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_PREPARE_CLEAR_RESET },
{ "cpu clear reset", UVC_CMD_CPU_RESET_CLEAR, sizeof(struct uv_cb_nodata), BIT_UVC_CMD_CPU_PERFORM_CLEAR_RESET },
{ "cpu set state", UVC_CMD_CPU_SET_STATE, sizeof(struct uv_cb_cpu_set_state), BIT_UVC_CMD_CPU_SET_STATE },
{ "pin shared", UVC_CMD_PIN_PAGE_SHARED, sizeof(struct uv_cb_cfs), BIT_UVC_CMD_PIN_PAGE_SHARED },
{ "unpin shared", UVC_CMD_UNPIN_PAGE_SHARED, sizeof(struct uv_cb_cts), BIT_UVC_CMD_UNPIN_PAGE_SHARED },
{ NULL, 0, 0 },
};
static void test_invalid(void)
{
struct uv_cb_header *hdr = (void *)page;
int cc, i;
report_prefix_push("invalid");
for (i = 0; invalid_cmds[i].name; i++) {
hdr->cmd = invalid_cmds[i].cmd;
hdr->len = invalid_cmds[i].len;
cc = uv_call(0, (u64)hdr);
report(cc == 1 && hdr->rc == UVC_RC_INV_CMD &&
(invalid_cmds[i].call_bit == -1 || !uv_query_test_call(invalid_cmds[i].call_bit)),
"%s", invalid_cmds[i].name);
}
report_prefix_pop();
}
int main(void)
{
bool has_uvc = test_facility(158);
report_prefix_push("uvc");
if (!has_uvc) {
report_skip("Ultravisor call facility is not available");
goto done;
}
if (!uv_os_is_guest()) {
report_skip("Not a protected guest");
goto done;
}
page = (unsigned long)alloc_page();
test_priv();
test_invalid();
test_query();
test_sharing();
free_page((void *)page);
done:
report_prefix_pop();
return report_summary();
}