622 lines
14 KiB
C
622 lines
14 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
|
||
|
#include "asm/hvcall.h"
|
||
|
#include <linux/log2.h>
|
||
|
#include <asm/pgalloc.h>
|
||
|
#include <asm/guest-state-buffer.h>
|
||
|
|
||
|
static const u16 kvmppc_gse_iden_len[__KVMPPC_GSE_TYPE_MAX] = {
|
||
|
[KVMPPC_GSE_BE32] = sizeof(__be32),
|
||
|
[KVMPPC_GSE_BE64] = sizeof(__be64),
|
||
|
[KVMPPC_GSE_VEC128] = sizeof(vector128),
|
||
|
[KVMPPC_GSE_PARTITION_TABLE] = sizeof(struct kvmppc_gs_part_table),
|
||
|
[KVMPPC_GSE_PROCESS_TABLE] = sizeof(struct kvmppc_gs_proc_table),
|
||
|
[KVMPPC_GSE_BUFFER] = sizeof(struct kvmppc_gs_buff_info),
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsb_new() - create a new guest state buffer
|
||
|
* @size: total size of the guest state buffer (includes header)
|
||
|
* @guest_id: guest_id
|
||
|
* @vcpu_id: vcpu_id
|
||
|
* @flags: GFP flags
|
||
|
*
|
||
|
* Returns a guest state buffer.
|
||
|
*/
|
||
|
struct kvmppc_gs_buff *kvmppc_gsb_new(size_t size, unsigned long guest_id,
|
||
|
unsigned long vcpu_id, gfp_t flags)
|
||
|
{
|
||
|
struct kvmppc_gs_buff *gsb;
|
||
|
|
||
|
gsb = kzalloc(sizeof(*gsb), flags);
|
||
|
if (!gsb)
|
||
|
return NULL;
|
||
|
|
||
|
size = roundup_pow_of_two(size);
|
||
|
gsb->hdr = kzalloc(size, GFP_KERNEL);
|
||
|
if (!gsb->hdr)
|
||
|
goto free;
|
||
|
|
||
|
gsb->capacity = size;
|
||
|
gsb->len = sizeof(struct kvmppc_gs_header);
|
||
|
gsb->vcpu_id = vcpu_id;
|
||
|
gsb->guest_id = guest_id;
|
||
|
|
||
|
gsb->hdr->nelems = cpu_to_be32(0);
|
||
|
|
||
|
return gsb;
|
||
|
|
||
|
free:
|
||
|
kfree(gsb);
|
||
|
return NULL;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsb_new);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsb_free() - free a guest state buffer
|
||
|
* @gsb: guest state buffer
|
||
|
*/
|
||
|
void kvmppc_gsb_free(struct kvmppc_gs_buff *gsb)
|
||
|
{
|
||
|
kfree(gsb->hdr);
|
||
|
kfree(gsb);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsb_free);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsb_put() - allocate space in a guest state buffer
|
||
|
* @gsb: buffer to allocate in
|
||
|
* @size: amount of space to allocate
|
||
|
*
|
||
|
* Returns a pointer to the amount of space requested within the buffer and
|
||
|
* increments the count of elements in the buffer.
|
||
|
*
|
||
|
* Does not check if there is enough space in the buffer.
|
||
|
*/
|
||
|
void *kvmppc_gsb_put(struct kvmppc_gs_buff *gsb, size_t size)
|
||
|
{
|
||
|
u32 nelems = kvmppc_gsb_nelems(gsb);
|
||
|
void *p;
|
||
|
|
||
|
p = (void *)kvmppc_gsb_header(gsb) + kvmppc_gsb_len(gsb);
|
||
|
gsb->len += size;
|
||
|
|
||
|
kvmppc_gsb_header(gsb)->nelems = cpu_to_be32(nelems + 1);
|
||
|
return p;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsb_put);
|
||
|
|
||
|
static int kvmppc_gsid_class(u16 iden)
|
||
|
{
|
||
|
if ((iden >= KVMPPC_GSE_GUESTWIDE_START) &&
|
||
|
(iden <= KVMPPC_GSE_GUESTWIDE_END))
|
||
|
return KVMPPC_GS_CLASS_GUESTWIDE;
|
||
|
|
||
|
if ((iden >= KVMPPC_GSE_META_START) && (iden <= KVMPPC_GSE_META_END))
|
||
|
return KVMPPC_GS_CLASS_META;
|
||
|
|
||
|
if ((iden >= KVMPPC_GSE_DW_REGS_START) &&
|
||
|
(iden <= KVMPPC_GSE_DW_REGS_END))
|
||
|
return KVMPPC_GS_CLASS_DWORD_REG;
|
||
|
|
||
|
if ((iden >= KVMPPC_GSE_W_REGS_START) &&
|
||
|
(iden <= KVMPPC_GSE_W_REGS_END))
|
||
|
return KVMPPC_GS_CLASS_WORD_REG;
|
||
|
|
||
|
if ((iden >= KVMPPC_GSE_VSRS_START) && (iden <= KVMPPC_GSE_VSRS_END))
|
||
|
return KVMPPC_GS_CLASS_VECTOR;
|
||
|
|
||
|
if ((iden >= KVMPPC_GSE_INTR_REGS_START) &&
|
||
|
(iden <= KVMPPC_GSE_INTR_REGS_END))
|
||
|
return KVMPPC_GS_CLASS_INTR;
|
||
|
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
static int kvmppc_gsid_type(u16 iden)
|
||
|
{
|
||
|
int type = -1;
|
||
|
|
||
|
switch (kvmppc_gsid_class(iden)) {
|
||
|
case KVMPPC_GS_CLASS_GUESTWIDE:
|
||
|
switch (iden) {
|
||
|
case KVMPPC_GSID_HOST_STATE_SIZE:
|
||
|
case KVMPPC_GSID_RUN_OUTPUT_MIN_SIZE:
|
||
|
case KVMPPC_GSID_TB_OFFSET:
|
||
|
type = KVMPPC_GSE_BE64;
|
||
|
break;
|
||
|
case KVMPPC_GSID_PARTITION_TABLE:
|
||
|
type = KVMPPC_GSE_PARTITION_TABLE;
|
||
|
break;
|
||
|
case KVMPPC_GSID_PROCESS_TABLE:
|
||
|
type = KVMPPC_GSE_PROCESS_TABLE;
|
||
|
break;
|
||
|
case KVMPPC_GSID_LOGICAL_PVR:
|
||
|
type = KVMPPC_GSE_BE32;
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_META:
|
||
|
switch (iden) {
|
||
|
case KVMPPC_GSID_RUN_INPUT:
|
||
|
case KVMPPC_GSID_RUN_OUTPUT:
|
||
|
type = KVMPPC_GSE_BUFFER;
|
||
|
break;
|
||
|
case KVMPPC_GSID_VPA:
|
||
|
type = KVMPPC_GSE_BE64;
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_DWORD_REG:
|
||
|
type = KVMPPC_GSE_BE64;
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_WORD_REG:
|
||
|
type = KVMPPC_GSE_BE32;
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_VECTOR:
|
||
|
type = KVMPPC_GSE_VEC128;
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_INTR:
|
||
|
switch (iden) {
|
||
|
case KVMPPC_GSID_HDAR:
|
||
|
case KVMPPC_GSID_ASDR:
|
||
|
case KVMPPC_GSID_HEIR:
|
||
|
type = KVMPPC_GSE_BE64;
|
||
|
break;
|
||
|
case KVMPPC_GSID_HDSISR:
|
||
|
type = KVMPPC_GSE_BE32;
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return type;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsid_flags() - the flags for a guest state ID
|
||
|
* @iden: guest state ID
|
||
|
*
|
||
|
* Returns any flags for the guest state ID.
|
||
|
*/
|
||
|
unsigned long kvmppc_gsid_flags(u16 iden)
|
||
|
{
|
||
|
unsigned long flags = 0;
|
||
|
|
||
|
switch (kvmppc_gsid_class(iden)) {
|
||
|
case KVMPPC_GS_CLASS_GUESTWIDE:
|
||
|
flags = KVMPPC_GS_FLAGS_WIDE;
|
||
|
break;
|
||
|
case KVMPPC_GS_CLASS_META:
|
||
|
case KVMPPC_GS_CLASS_DWORD_REG:
|
||
|
case KVMPPC_GS_CLASS_WORD_REG:
|
||
|
case KVMPPC_GS_CLASS_VECTOR:
|
||
|
case KVMPPC_GS_CLASS_INTR:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return flags;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsid_flags);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsid_size() - the size of a guest state ID
|
||
|
* @iden: guest state ID
|
||
|
*
|
||
|
* Returns the size of guest state ID.
|
||
|
*/
|
||
|
u16 kvmppc_gsid_size(u16 iden)
|
||
|
{
|
||
|
int type;
|
||
|
|
||
|
type = kvmppc_gsid_type(iden);
|
||
|
if (type == -1)
|
||
|
return 0;
|
||
|
|
||
|
if (type >= __KVMPPC_GSE_TYPE_MAX)
|
||
|
return 0;
|
||
|
|
||
|
return kvmppc_gse_iden_len[type];
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsid_size);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsid_mask() - the settable bits of a guest state ID
|
||
|
* @iden: guest state ID
|
||
|
*
|
||
|
* Returns a mask of settable bits for a guest state ID.
|
||
|
*/
|
||
|
u64 kvmppc_gsid_mask(u16 iden)
|
||
|
{
|
||
|
u64 mask = ~0ull;
|
||
|
|
||
|
switch (iden) {
|
||
|
case KVMPPC_GSID_LPCR:
|
||
|
mask = LPCR_DPFD | LPCR_ILE | LPCR_AIL | LPCR_LD | LPCR_MER |
|
||
|
LPCR_GTSE;
|
||
|
break;
|
||
|
case KVMPPC_GSID_MSR:
|
||
|
mask = ~(MSR_HV | MSR_S | MSR_ME);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return mask;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsid_mask);
|
||
|
|
||
|
/**
|
||
|
* __kvmppc_gse_put() - add a guest state element to a buffer
|
||
|
* @gsb: buffer to the element to
|
||
|
* @iden: guest state ID
|
||
|
* @size: length of data
|
||
|
* @data: pointer to data
|
||
|
*/
|
||
|
int __kvmppc_gse_put(struct kvmppc_gs_buff *gsb, u16 iden, u16 size,
|
||
|
const void *data)
|
||
|
{
|
||
|
struct kvmppc_gs_elem *gse;
|
||
|
u16 total_size;
|
||
|
|
||
|
total_size = sizeof(*gse) + size;
|
||
|
if (total_size + kvmppc_gsb_len(gsb) > kvmppc_gsb_capacity(gsb))
|
||
|
return -ENOMEM;
|
||
|
|
||
|
if (kvmppc_gsid_size(iden) != size)
|
||
|
return -EINVAL;
|
||
|
|
||
|
gse = kvmppc_gsb_put(gsb, total_size);
|
||
|
gse->iden = cpu_to_be16(iden);
|
||
|
gse->len = cpu_to_be16(size);
|
||
|
memcpy(gse->data, data, size);
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(__kvmppc_gse_put);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gse_parse() - create a parse map from a guest state buffer
|
||
|
* @gsp: guest state parser
|
||
|
* @gsb: guest state buffer
|
||
|
*/
|
||
|
int kvmppc_gse_parse(struct kvmppc_gs_parser *gsp, struct kvmppc_gs_buff *gsb)
|
||
|
{
|
||
|
struct kvmppc_gs_elem *curr;
|
||
|
int rem, i;
|
||
|
|
||
|
kvmppc_gsb_for_each_elem(i, curr, gsb, rem) {
|
||
|
if (kvmppc_gse_len(curr) !=
|
||
|
kvmppc_gsid_size(kvmppc_gse_iden(curr)))
|
||
|
return -EINVAL;
|
||
|
kvmppc_gsp_insert(gsp, kvmppc_gse_iden(curr), curr);
|
||
|
}
|
||
|
|
||
|
if (kvmppc_gsb_nelems(gsb) != i)
|
||
|
return -EINVAL;
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gse_parse);
|
||
|
|
||
|
static inline int kvmppc_gse_flatten_iden(u16 iden)
|
||
|
{
|
||
|
int bit = 0;
|
||
|
int class;
|
||
|
|
||
|
class = kvmppc_gsid_class(iden);
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_GUESTWIDE) {
|
||
|
bit += iden - KVMPPC_GSE_GUESTWIDE_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
bit += KVMPPC_GSE_GUESTWIDE_COUNT;
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_META) {
|
||
|
bit += iden - KVMPPC_GSE_META_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
bit += KVMPPC_GSE_META_COUNT;
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_DWORD_REG) {
|
||
|
bit += iden - KVMPPC_GSE_DW_REGS_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
bit += KVMPPC_GSE_DW_REGS_COUNT;
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_WORD_REG) {
|
||
|
bit += iden - KVMPPC_GSE_W_REGS_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
bit += KVMPPC_GSE_W_REGS_COUNT;
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_VECTOR) {
|
||
|
bit += iden - KVMPPC_GSE_VSRS_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
bit += KVMPPC_GSE_VSRS_COUNT;
|
||
|
|
||
|
if (class == KVMPPC_GS_CLASS_INTR) {
|
||
|
bit += iden - KVMPPC_GSE_INTR_REGS_START;
|
||
|
return bit;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static inline u16 kvmppc_gse_unflatten_iden(int bit)
|
||
|
{
|
||
|
u16 iden;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_GUESTWIDE_COUNT) {
|
||
|
iden = KVMPPC_GSE_GUESTWIDE_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
bit -= KVMPPC_GSE_GUESTWIDE_COUNT;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_META_COUNT) {
|
||
|
iden = KVMPPC_GSE_META_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
bit -= KVMPPC_GSE_META_COUNT;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_DW_REGS_COUNT) {
|
||
|
iden = KVMPPC_GSE_DW_REGS_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
bit -= KVMPPC_GSE_DW_REGS_COUNT;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_W_REGS_COUNT) {
|
||
|
iden = KVMPPC_GSE_W_REGS_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
bit -= KVMPPC_GSE_W_REGS_COUNT;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_VSRS_COUNT) {
|
||
|
iden = KVMPPC_GSE_VSRS_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
bit -= KVMPPC_GSE_VSRS_COUNT;
|
||
|
|
||
|
if (bit < KVMPPC_GSE_IDEN_COUNT) {
|
||
|
iden = KVMPPC_GSE_INTR_REGS_START + bit;
|
||
|
return iden;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsp_insert() - add a mapping from an guest state ID to an element
|
||
|
* @gsp: guest state parser
|
||
|
* @iden: guest state id (key)
|
||
|
* @gse: guest state element (value)
|
||
|
*/
|
||
|
void kvmppc_gsp_insert(struct kvmppc_gs_parser *gsp, u16 iden,
|
||
|
struct kvmppc_gs_elem *gse)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
i = kvmppc_gse_flatten_iden(iden);
|
||
|
kvmppc_gsbm_set(&gsp->iterator, iden);
|
||
|
gsp->gses[i] = gse;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsp_insert);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsp_lookup() - lookup an element from a guest state ID
|
||
|
* @gsp: guest state parser
|
||
|
* @iden: guest state ID (key)
|
||
|
*
|
||
|
* Returns the guest state element if present.
|
||
|
*/
|
||
|
struct kvmppc_gs_elem *kvmppc_gsp_lookup(struct kvmppc_gs_parser *gsp, u16 iden)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
i = kvmppc_gse_flatten_iden(iden);
|
||
|
return gsp->gses[i];
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsp_lookup);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsbm_set() - set the guest state ID
|
||
|
* @gsbm: guest state bitmap
|
||
|
* @iden: guest state ID
|
||
|
*/
|
||
|
void kvmppc_gsbm_set(struct kvmppc_gs_bitmap *gsbm, u16 iden)
|
||
|
{
|
||
|
set_bit(kvmppc_gse_flatten_iden(iden), gsbm->bitmap);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsbm_set);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsbm_clear() - clear the guest state ID
|
||
|
* @gsbm: guest state bitmap
|
||
|
* @iden: guest state ID
|
||
|
*/
|
||
|
void kvmppc_gsbm_clear(struct kvmppc_gs_bitmap *gsbm, u16 iden)
|
||
|
{
|
||
|
clear_bit(kvmppc_gse_flatten_iden(iden), gsbm->bitmap);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsbm_clear);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsbm_test() - test the guest state ID
|
||
|
* @gsbm: guest state bitmap
|
||
|
* @iden: guest state ID
|
||
|
*/
|
||
|
bool kvmppc_gsbm_test(struct kvmppc_gs_bitmap *gsbm, u16 iden)
|
||
|
{
|
||
|
return test_bit(kvmppc_gse_flatten_iden(iden), gsbm->bitmap);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsbm_test);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsbm_next() - return the next set guest state ID
|
||
|
* @gsbm: guest state bitmap
|
||
|
* @prev: last guest state ID
|
||
|
*/
|
||
|
u16 kvmppc_gsbm_next(struct kvmppc_gs_bitmap *gsbm, u16 prev)
|
||
|
{
|
||
|
int bit, pbit;
|
||
|
|
||
|
pbit = prev ? kvmppc_gse_flatten_iden(prev) + 1 : 0;
|
||
|
bit = find_next_bit(gsbm->bitmap, KVMPPC_GSE_IDEN_COUNT, pbit);
|
||
|
|
||
|
if (bit < KVMPPC_GSE_IDEN_COUNT)
|
||
|
return kvmppc_gse_unflatten_iden(bit);
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsbm_next);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_init() - initialize a guest state message
|
||
|
* @gsm: guest state message
|
||
|
* @ops: callbacks
|
||
|
* @data: private data
|
||
|
* @flags: guest wide or thread wide
|
||
|
*/
|
||
|
int kvmppc_gsm_init(struct kvmppc_gs_msg *gsm, struct kvmppc_gs_msg_ops *ops,
|
||
|
void *data, unsigned long flags)
|
||
|
{
|
||
|
memset(gsm, 0, sizeof(*gsm));
|
||
|
gsm->ops = ops;
|
||
|
gsm->data = data;
|
||
|
gsm->flags = flags;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_init);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_new() - creates a new guest state message
|
||
|
* @ops: callbacks
|
||
|
* @data: private data
|
||
|
* @flags: guest wide or thread wide
|
||
|
* @gfp_flags: GFP allocation flags
|
||
|
*
|
||
|
* Returns an initialized guest state message.
|
||
|
*/
|
||
|
struct kvmppc_gs_msg *kvmppc_gsm_new(struct kvmppc_gs_msg_ops *ops, void *data,
|
||
|
unsigned long flags, gfp_t gfp_flags)
|
||
|
{
|
||
|
struct kvmppc_gs_msg *gsm;
|
||
|
|
||
|
gsm = kzalloc(sizeof(*gsm), gfp_flags);
|
||
|
if (!gsm)
|
||
|
return NULL;
|
||
|
|
||
|
kvmppc_gsm_init(gsm, ops, data, flags);
|
||
|
|
||
|
return gsm;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_new);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_size() - creates a new guest state message
|
||
|
* @gsm: self
|
||
|
*
|
||
|
* Returns the size required for the message.
|
||
|
*/
|
||
|
size_t kvmppc_gsm_size(struct kvmppc_gs_msg *gsm)
|
||
|
{
|
||
|
if (gsm->ops->get_size)
|
||
|
return gsm->ops->get_size(gsm);
|
||
|
return 0;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_size);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_free() - free guest state message
|
||
|
* @gsm: guest state message
|
||
|
*
|
||
|
* Returns the size required for the message.
|
||
|
*/
|
||
|
void kvmppc_gsm_free(struct kvmppc_gs_msg *gsm)
|
||
|
{
|
||
|
kfree(gsm);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_free);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_fill_info() - serialises message to guest state buffer format
|
||
|
* @gsm: self
|
||
|
* @gsb: buffer to serialise into
|
||
|
*/
|
||
|
int kvmppc_gsm_fill_info(struct kvmppc_gs_msg *gsm, struct kvmppc_gs_buff *gsb)
|
||
|
{
|
||
|
if (!gsm->ops->fill_info)
|
||
|
return -EINVAL;
|
||
|
|
||
|
return gsm->ops->fill_info(gsb, gsm);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_fill_info);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsm_refresh_info() - deserialises from guest state buffer
|
||
|
* @gsm: self
|
||
|
* @gsb: buffer to serialise from
|
||
|
*/
|
||
|
int kvmppc_gsm_refresh_info(struct kvmppc_gs_msg *gsm,
|
||
|
struct kvmppc_gs_buff *gsb)
|
||
|
{
|
||
|
if (!gsm->ops->fill_info)
|
||
|
return -EINVAL;
|
||
|
|
||
|
return gsm->ops->refresh_info(gsm, gsb);
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsm_refresh_info);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsb_send - send all elements in the buffer to the hypervisor.
|
||
|
* @gsb: guest state buffer
|
||
|
* @flags: guest wide or thread wide
|
||
|
*
|
||
|
* Performs the H_GUEST_SET_STATE hcall for the guest state buffer.
|
||
|
*/
|
||
|
int kvmppc_gsb_send(struct kvmppc_gs_buff *gsb, unsigned long flags)
|
||
|
{
|
||
|
unsigned long hflags = 0;
|
||
|
unsigned long i;
|
||
|
int rc;
|
||
|
|
||
|
if (kvmppc_gsb_nelems(gsb) == 0)
|
||
|
return 0;
|
||
|
|
||
|
if (flags & KVMPPC_GS_FLAGS_WIDE)
|
||
|
hflags |= H_GUEST_FLAGS_WIDE;
|
||
|
|
||
|
rc = plpar_guest_set_state(hflags, gsb->guest_id, gsb->vcpu_id,
|
||
|
__pa(gsb->hdr), gsb->capacity, &i);
|
||
|
return rc;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsb_send);
|
||
|
|
||
|
/**
|
||
|
* kvmppc_gsb_recv - request all elements in the buffer have their value
|
||
|
* updated.
|
||
|
* @gsb: guest state buffer
|
||
|
* @flags: guest wide or thread wide
|
||
|
*
|
||
|
* Performs the H_GUEST_GET_STATE hcall for the guest state buffer.
|
||
|
* After returning from the hcall the guest state elements that were
|
||
|
* present in the buffer will have updated values from the hypervisor.
|
||
|
*/
|
||
|
int kvmppc_gsb_recv(struct kvmppc_gs_buff *gsb, unsigned long flags)
|
||
|
{
|
||
|
unsigned long hflags = 0;
|
||
|
unsigned long i;
|
||
|
int rc;
|
||
|
|
||
|
if (flags & KVMPPC_GS_FLAGS_WIDE)
|
||
|
hflags |= H_GUEST_FLAGS_WIDE;
|
||
|
|
||
|
rc = plpar_guest_get_state(hflags, gsb->guest_id, gsb->vcpu_id,
|
||
|
__pa(gsb->hdr), gsb->capacity, &i);
|
||
|
return rc;
|
||
|
}
|
||
|
EXPORT_SYMBOL_GPL(kvmppc_gsb_recv);
|