/** * 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;
if ((iden >= KVMPPC_GSE_HOSTWIDE_START) &&
(iden <= KVMPPC_GSE_HOSTWIDE_END)) return KVMPPC_GS_CLASS_HOSTWIDE;
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;
}
staticint kvmppc_gsid_type(u16 iden)
{ int type = -1;
switch (kvmppc_gsid_class(iden)) { case KVMPPC_GS_CLASS_HOSTWIDE: switch (iden) { case KVMPPC_GSID_L0_GUEST_HEAP:
fallthrough; case KVMPPC_GSID_L0_GUEST_HEAP_MAX:
fallthrough; case KVMPPC_GSID_L0_GUEST_PGTABLE_SIZE:
fallthrough; case KVMPPC_GSID_L0_GUEST_PGTABLE_SIZE_MAX:
fallthrough; case KVMPPC_GSID_L0_GUEST_PGTABLE_RECLAIM:
type = KVMPPC_GSE_BE64; break;
} break; 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.
*/ unsignedlong kvmppc_gsid_flags(u16 iden)
{ unsignedlong flags = 0;
switch (kvmppc_gsid_class(iden)) { case KVMPPC_GS_CLASS_GUESTWIDE:
flags = KVMPPC_GS_FLAGS_WIDE; break; case KVMPPC_GS_CLASS_HOSTWIDE:
flags = KVMPPC_GS_FLAGS_HOST_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;
}
/** * 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;
/** * 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;
/** * __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, constvoid *data)
{ struct kvmppc_gs_elem *gse;
u16 total_size;
/** * 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_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;
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, unsignedlong 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, unsignedlong 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;
/** * 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, unsignedlong flags)
{ unsignedlong hflags = 0; unsignedlong i; int rc;
if (kvmppc_gsb_nelems(gsb) == 0) return 0;
if (flags & KVMPPC_GS_FLAGS_WIDE)
hflags |= H_GUEST_FLAGS_WIDE; if (flags & KVMPPC_GS_FLAGS_HOST_WIDE)
hflags |= H_GUEST_FLAGS_HOST_WIDE;
/** * 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, unsignedlong flags)
{ unsignedlong hflags = 0; unsignedlong i; int rc;
if (flags & KVMPPC_GS_FLAGS_WIDE)
hflags |= H_GUEST_FLAGS_WIDE; if (flags & KVMPPC_GS_FLAGS_HOST_WIDE)
hflags |= H_GUEST_FLAGS_HOST_WIDE;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.