x86: acpi: let the firmware handle pending "CPU remove" events in SMM
if firmware and QEMU negotiated CPU hotunplug support, generate _EJ0 method so that it will mark CPU for removal by firmware and pass control to it by triggering SMI. Signed-off-by: Igor Mammedov <imammedo@redhat.com> Message-Id: <20201207140739.3829993-6-imammedo@redhat.com> Reviewed-by: Michael S. Tsirkin <mst@redhat.com> Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
This commit is contained in:
parent
414aa64fda
commit
69dea9d6b3
@ -342,6 +342,7 @@ const VMStateDescription vmstate_cpu_hotplug = {
|
||||
#define CPU_INSERT_EVENT "CINS"
|
||||
#define CPU_REMOVE_EVENT "CRMV"
|
||||
#define CPU_EJECT_EVENT "CEJ0"
|
||||
#define CPU_FW_EJECT_EVENT "CEJF"
|
||||
|
||||
void build_cpus_aml(Aml *table, MachineState *machine, CPUHotplugFeatures opts,
|
||||
hwaddr io_base,
|
||||
@ -394,7 +395,9 @@ void build_cpus_aml(Aml *table, MachineState *machine, CPUHotplugFeatures opts,
|
||||
aml_append(field, aml_named_field(CPU_REMOVE_EVENT, 1));
|
||||
/* initiates device eject, write only */
|
||||
aml_append(field, aml_named_field(CPU_EJECT_EVENT, 1));
|
||||
aml_append(field, aml_reserved_field(4));
|
||||
/* tell firmware to do device eject, write only */
|
||||
aml_append(field, aml_named_field(CPU_FW_EJECT_EVENT, 1));
|
||||
aml_append(field, aml_reserved_field(3));
|
||||
aml_append(field, aml_named_field(CPU_COMMAND, 8));
|
||||
aml_append(cpu_ctrl_dev, field);
|
||||
|
||||
@ -429,6 +432,7 @@ void build_cpus_aml(Aml *table, MachineState *machine, CPUHotplugFeatures opts,
|
||||
Aml *ins_evt = aml_name("%s.%s", cphp_res_path, CPU_INSERT_EVENT);
|
||||
Aml *rm_evt = aml_name("%s.%s", cphp_res_path, CPU_REMOVE_EVENT);
|
||||
Aml *ej_evt = aml_name("%s.%s", cphp_res_path, CPU_EJECT_EVENT);
|
||||
Aml *fw_ej_evt = aml_name("%s.%s", cphp_res_path, CPU_FW_EJECT_EVENT);
|
||||
|
||||
aml_append(cpus_dev, aml_name_decl("_HID", aml_string("ACPI0010")));
|
||||
aml_append(cpus_dev, aml_name_decl("_CID", aml_eisaid("PNP0A05")));
|
||||
@ -471,7 +475,13 @@ void build_cpus_aml(Aml *table, MachineState *machine, CPUHotplugFeatures opts,
|
||||
|
||||
aml_append(method, aml_acquire(ctrl_lock, 0xFFFF));
|
||||
aml_append(method, aml_store(idx, cpu_selector));
|
||||
aml_append(method, aml_store(one, ej_evt));
|
||||
if (opts.fw_unplugs_cpu) {
|
||||
aml_append(method, aml_store(one, fw_ej_evt));
|
||||
aml_append(method, aml_store(aml_int(OVMF_CPUHP_SMI_CMD),
|
||||
aml_name("%s", opts.smi_path)));
|
||||
} else {
|
||||
aml_append(method, aml_store(one, ej_evt));
|
||||
}
|
||||
aml_append(method, aml_release(ctrl_lock));
|
||||
}
|
||||
aml_append(cpus_dev, method);
|
||||
|
@ -1293,6 +1293,7 @@ build_dsdt(GArray *table_data, BIOSLinker *linker,
|
||||
CPUHotplugFeatures opts = {
|
||||
.acpi_1_compatible = true, .has_legacy_cphp = true,
|
||||
.smi_path = pm->smi_on_cpuhp ? "\\_SB.PCI0.SMI0.SMIC" : NULL,
|
||||
.fw_unplugs_cpu = pm->smi_on_cpu_unplug,
|
||||
};
|
||||
build_cpus_aml(dsdt, machine, opts, pm->cpu_hp_io_base,
|
||||
"\\_SB.PCI0", "\\_GPE._E02");
|
||||
|
@ -51,6 +51,7 @@ void cpu_hotplug_hw_init(MemoryRegion *as, Object *owner,
|
||||
typedef struct CPUHotplugFeatures {
|
||||
bool acpi_1_compatible;
|
||||
bool has_legacy_cphp;
|
||||
bool fw_unplugs_cpu;
|
||||
const char *smi_path;
|
||||
} CPUHotplugFeatures;
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user