// SPDX-License-Identifier: GPL-2.0 #include "asm/hvcall.h" #include #include #include 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);