/* SPDX-License-Identifier: GPL-2.0-or-later */ /* * CPU Topology * * Copyright IBM Corp. 2022, 2023 * Author(s): Pierre Morel * * S390 topology handling can be divided in two parts: * * - The first part in this file is taking care of all common functions * used by KVM and TCG to create and modify the topology. * * - The second part, building the topology information data for the * guest with CPU and KVM specificity will be implemented inside * the target/s390/kvm sub tree. */ #include "qemu/osdep.h" #include "qapi/error.h" #include "qemu/error-report.h" #include "hw/qdev-properties.h" #include "hw/boards.h" #include "target/s390x/cpu.h" #include "hw/s390x/s390-virtio-ccw.h" #include "hw/s390x/cpu-topology.h" #include "qapi/qapi-commands-machine-target.h" #include "qapi/qapi-events-machine-target.h" /* * s390_topology is used to keep the topology information. * .cores_per_socket: tracks information on the count of cores * per socket. * .polarization: tracks machine polarization. */ S390Topology s390_topology = { /* will be initialized after the CPU model is realized */ .cores_per_socket = NULL, .polarization = S390_CPU_POLARIZATION_HORIZONTAL, }; /** * s390_socket_nb: * @cpu: s390x CPU * * Returns the socket number used inside the cores_per_socket array * for a topology tree entry */ static int s390_socket_nb_from_ids(int drawer_id, int book_id, int socket_id) { return (drawer_id * current_machine->smp.books + book_id) * current_machine->smp.sockets + socket_id; } /** * s390_socket_nb: * @cpu: s390x CPU * * Returns the socket number used inside the cores_per_socket array * for a cpu. */ static int s390_socket_nb(S390CPU *cpu) { return s390_socket_nb_from_ids(cpu->env.drawer_id, cpu->env.book_id, cpu->env.socket_id); } /** * s390_has_topology: * * Return: true if the topology is supported by the machine. */ bool s390_has_topology(void) { return s390_has_feat(S390_FEAT_CONFIGURATION_TOPOLOGY); } /** * s390_topology_init: * @ms: the machine state where the machine topology is defined * * Keep track of the machine topology. * * Allocate an array to keep the count of cores per socket. * The index of the array starts at socket 0 from book 0 and * drawer 0 up to the maximum allowed by the machine topology. */ static void s390_topology_init(MachineState *ms) { CpuTopology *smp = &ms->smp; s390_topology.cores_per_socket = g_new0(uint8_t, smp->sockets * smp->books * smp->drawers); } /* * s390_handle_ptf: * * @register 1: contains the function code * * Function codes 0 (horizontal) and 1 (vertical) define the CPU * polarization requested by the guest. * * Function code 2 is handling topology changes and is interpreted * by the SIE. */ void s390_handle_ptf(S390CPU *cpu, uint8_t r1, uintptr_t ra) { S390CpuPolarization polarization; CPUS390XState *env = &cpu->env; uint64_t reg = env->regs[r1]; int fc = reg & S390_TOPO_FC_MASK; if (!s390_has_feat(S390_FEAT_CONFIGURATION_TOPOLOGY)) { s390_program_interrupt(env, PGM_OPERATION, ra); return; } if (env->psw.mask & PSW_MASK_PSTATE) { s390_program_interrupt(env, PGM_PRIVILEGED, ra); return; } if (reg & ~S390_TOPO_FC_MASK) { s390_program_interrupt(env, PGM_SPECIFICATION, ra); return; } polarization = S390_CPU_POLARIZATION_VERTICAL; switch (fc) { case 0: polarization = S390_CPU_POLARIZATION_HORIZONTAL; /* fallthrough */ case 1: if (s390_topology.polarization == polarization) { env->regs[r1] |= S390_PTF_REASON_DONE; setcc(cpu, 2); } else { s390_topology.polarization = polarization; s390_cpu_topology_set_changed(true); qapi_event_send_cpu_polarization_change(polarization); setcc(cpu, 0); } break; default: /* Note that fc == 2 is interpreted by the SIE */ s390_program_interrupt(env, PGM_SPECIFICATION, ra); } } /** * s390_topology_reset: * * Generic reset for CPU topology, calls s390_topology_reset() * to reset the kernel Modified Topology Change Record. */ void s390_topology_reset(void) { s390_cpu_topology_set_changed(false); s390_topology.polarization = S390_CPU_POLARIZATION_HORIZONTAL; } /** * s390_topology_cpu_default: * @cpu: pointer to a S390CPU * @errp: Error pointer * * Setup the default topology if no attributes are already set. * Passing a CPU with some, but not all, attributes set is considered * an error. * * The function calculates the (drawer_id, book_id, socket_id) * topology by filling the cores starting from the first socket * (0, 0, 0) up to the last (smp->drawers, smp->books, smp->sockets). * * CPU type and dedication have defaults values set in the * s390x_cpu_properties, entitlement must be adjust depending on the * dedication. * * Returns false if it is impossible to setup a default topology * true otherwise. */ static bool s390_topology_cpu_default(S390CPU *cpu, Error **errp) { CpuTopology *smp = ¤t_machine->smp; CPUS390XState *env = &cpu->env; /* All geometry topology attributes must be set or all unset */ if ((env->socket_id < 0 || env->book_id < 0 || env->drawer_id < 0) && (env->socket_id >= 0 || env->book_id >= 0 || env->drawer_id >= 0)) { error_setg(errp, "Please define all or none of the topology geometry attributes"); return false; } /* If one value is unset all are unset -> calculate defaults */ if (env->socket_id < 0) { env->socket_id = s390_std_socket(env->core_id, smp); env->book_id = s390_std_book(env->core_id, smp); env->drawer_id = s390_std_drawer(env->core_id, smp); } /* * When the user specifies the entitlement as 'auto' on the command line, * QEMU will set the entitlement as: * Medium when the CPU is not dedicated. * High when dedicated is true. */ if (env->entitlement == S390_CPU_ENTITLEMENT_AUTO) { if (env->dedicated) { env->entitlement = S390_CPU_ENTITLEMENT_HIGH; } else { env->entitlement = S390_CPU_ENTITLEMENT_MEDIUM; } } return true; } /** * s390_topology_check: * @socket_id: socket to check * @book_id: book to check * @drawer_id: drawer to check * @entitlement: entitlement to check * @dedicated: dedication to check * @errp: Error pointer * * The function checks if the topology * attributes fits inside the system topology. * * Returns false if the specified topology does not match with * the machine topology. */ static bool s390_topology_check(uint16_t socket_id, uint16_t book_id, uint16_t drawer_id, uint16_t entitlement, bool dedicated, Error **errp) { CpuTopology *smp = ¤t_machine->smp; if (socket_id >= smp->sockets) { error_setg(errp, "Unavailable socket: %d", socket_id); return false; } if (book_id >= smp->books) { error_setg(errp, "Unavailable book: %d", book_id); return false; } if (drawer_id >= smp->drawers) { error_setg(errp, "Unavailable drawer: %d", drawer_id); return false; } if (entitlement >= S390_CPU_ENTITLEMENT__MAX) { error_setg(errp, "Unknown entitlement: %d", entitlement); return false; } if (dedicated && (entitlement == S390_CPU_ENTITLEMENT_LOW || entitlement == S390_CPU_ENTITLEMENT_MEDIUM)) { error_setg(errp, "A dedicated CPU implies high entitlement"); return false; } return true; } /** * s390_topology_need_report * @cpu: Current cpu * @drawer_id: future drawer ID * @book_id: future book ID * @socket_id: future socket ID * @entitlement: future entitlement * @dedicated: future dedicated * * A modified topology change report is needed if the topology * tree or the topology attributes change. */ static bool s390_topology_need_report(S390CPU *cpu, int drawer_id, int book_id, int socket_id, uint16_t entitlement, bool dedicated) { return cpu->env.drawer_id != drawer_id || cpu->env.book_id != book_id || cpu->env.socket_id != socket_id || cpu->env.entitlement != entitlement || cpu->env.dedicated != dedicated; } /** * s390_update_cpu_props: * @ms: the machine state * @cpu: the CPU for which to update the properties from the environment. * */ static void s390_update_cpu_props(MachineState *ms, S390CPU *cpu) { CpuInstanceProperties *props; props = &ms->possible_cpus->cpus[cpu->env.core_id].props; props->socket_id = cpu->env.socket_id; props->book_id = cpu->env.book_id; props->drawer_id = cpu->env.drawer_id; } /** * s390_topology_setup_cpu: * @ms: MachineState used to initialize the topology structure on * first call. * @cpu: the new S390CPU to insert in the topology structure * @errp: the error pointer * * Called from CPU hotplug to check and setup the CPU attributes * before the CPU is inserted in the topology. * There is no need to update the MTCR explicitly here because it * will be updated by KVM on creation of the new CPU. */ void s390_topology_setup_cpu(MachineState *ms, S390CPU *cpu, Error **errp) { int entry; /* * We do not want to initialize the topology if the CPU model * does not support topology, consequently, we have to wait for * the first CPU to be realized, which realizes the CPU model * to initialize the topology structures. * * s390_topology_setup_cpu() is called from the CPU hotplug. */ if (!s390_topology.cores_per_socket) { s390_topology_init(ms); } if (!s390_topology_cpu_default(cpu, errp)) { return; } if (!s390_topology_check(cpu->env.socket_id, cpu->env.book_id, cpu->env.drawer_id, cpu->env.entitlement, cpu->env.dedicated, errp)) { return; } /* Do we still have space in the socket */ entry = s390_socket_nb(cpu); if (s390_topology.cores_per_socket[entry] >= ms->smp.cores) { error_setg(errp, "No more space on this socket"); return; } /* Update the count of cores in sockets */ s390_topology.cores_per_socket[entry] += 1; /* topology tree is reflected in props */ s390_update_cpu_props(ms, cpu); } static void s390_change_topology(uint16_t core_id, bool has_socket_id, uint16_t socket_id, bool has_book_id, uint16_t book_id, bool has_drawer_id, uint16_t drawer_id, bool has_entitlement, S390CpuEntitlement entitlement, bool has_dedicated, bool dedicated, Error **errp) { MachineState *ms = current_machine; int old_socket_entry; int new_socket_entry; bool report_needed; S390CPU *cpu; cpu = s390_cpu_addr2state(core_id); if (!cpu) { error_setg(errp, "Core-id %d does not exist!", core_id); return; } /* Get attributes not provided from cpu and verify the new topology */ if (!has_socket_id) { socket_id = cpu->env.socket_id; } if (!has_book_id) { book_id = cpu->env.book_id; } if (!has_drawer_id) { drawer_id = cpu->env.drawer_id; } if (!has_dedicated) { dedicated = cpu->env.dedicated; } /* * When the user specifies the entitlement as 'auto' on the command line, * QEMU will set the entitlement as: * Medium when the CPU is not dedicated. * High when dedicated is true. */ if (!has_entitlement || entitlement == S390_CPU_ENTITLEMENT_AUTO) { if (dedicated) { entitlement = S390_CPU_ENTITLEMENT_HIGH; } else { entitlement = S390_CPU_ENTITLEMENT_MEDIUM; } } if (!s390_topology_check(socket_id, book_id, drawer_id, entitlement, dedicated, errp)) { return; } /* Check for space on new socket */ old_socket_entry = s390_socket_nb(cpu); new_socket_entry = s390_socket_nb_from_ids(drawer_id, book_id, socket_id); if (new_socket_entry != old_socket_entry) { if (s390_topology.cores_per_socket[new_socket_entry] >= ms->smp.cores) { error_setg(errp, "No more space on this socket"); return; } /* Update the count of cores in sockets */ s390_topology.cores_per_socket[new_socket_entry] += 1; s390_topology.cores_per_socket[old_socket_entry] -= 1; } /* Check if we will need to report the modified topology */ report_needed = s390_topology_need_report(cpu, drawer_id, book_id, socket_id, entitlement, dedicated); /* All checks done, report new topology into the vCPU */ cpu->env.drawer_id = drawer_id; cpu->env.book_id = book_id; cpu->env.socket_id = socket_id; cpu->env.dedicated = dedicated; cpu->env.entitlement = entitlement; /* topology tree is reflected in props */ s390_update_cpu_props(ms, cpu); /* Advertise the topology change */ if (report_needed) { s390_cpu_topology_set_changed(true); } } void qmp_set_cpu_topology(uint16_t core, bool has_socket, uint16_t socket, bool has_book, uint16_t book, bool has_drawer, uint16_t drawer, bool has_entitlement, S390CpuEntitlement entitlement, bool has_dedicated, bool dedicated, Error **errp) { if (!s390_has_topology()) { error_setg(errp, "This machine doesn't support topology"); return; } s390_change_topology(core, has_socket, socket, has_book, book, has_drawer, drawer, has_entitlement, entitlement, has_dedicated, dedicated, errp); } CpuPolarizationInfo *qmp_query_s390x_cpu_polarization(Error **errp) { CpuPolarizationInfo *info = g_new0(CpuPolarizationInfo, 1); info->polarization = s390_topology.polarization; return info; }