|
|
9ae3a8 |
From b70ff3a2191f959098f12965c4c7e5adb60be59e Mon Sep 17 00:00:00 2001
|
|
|
9ae3a8 |
Message-Id: <b70ff3a2191f959098f12965c4c7e5adb60be59e.1387298827.git.minovotn@redhat.com>
|
|
|
9ae3a8 |
In-Reply-To: <3ed0fb61a3dc912ef036d7ef450bed192090709e.1387298827.git.minovotn@redhat.com>
|
|
|
9ae3a8 |
References: <3ed0fb61a3dc912ef036d7ef450bed192090709e.1387298827.git.minovotn@redhat.com>
|
|
|
9ae3a8 |
From: "Michael S. Tsirkin" <mst@redhat.com>
|
|
|
9ae3a8 |
Date: Tue, 17 Dec 2013 15:17:24 +0100
|
|
|
9ae3a8 |
Subject: [PATCH 14/56] pc: pass PCI hole ranges to Guests
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
RH-Author: Michael S. Tsirkin <mst@redhat.com>
|
|
|
9ae3a8 |
Message-id: <1387293161-4085-15-git-send-email-mst@redhat.com>
|
|
|
9ae3a8 |
Patchwork-id: 56320
|
|
|
9ae3a8 |
O-Subject: [PATCH qemu-kvm RHEL7.0 v2 14/57] pc: pass PCI hole ranges to Guests
|
|
|
9ae3a8 |
Bugzilla: 1034876
|
|
|
9ae3a8 |
RH-Acked-by: Igor Mammedov <imammedo@redhat.com>
|
|
|
9ae3a8 |
RH-Acked-by: Marcel Apfelbaum <marcel.a@redhat.com>
|
|
|
9ae3a8 |
RH-Acked-by: Laszlo Ersek <lersek@redhat.com>
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
Guest currently has to jump through lots of hoops to guess the PCI hole
|
|
|
9ae3a8 |
ranges. It's fragile, and makes us change BIOS each time we add a new
|
|
|
9ae3a8 |
chipset. Let's report the window in a ROM file, to make BIOS do exactly
|
|
|
9ae3a8 |
what QEMU intends.
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
Signed-off-by: Michael S. Tsirkin <mst@redhat.com>
|
|
|
9ae3a8 |
(cherry picked from commit f8c457b88d72a48989f190bc3d7b79f4f3b7d11c)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
Conflicts:
|
|
|
9ae3a8 |
hw/i386/pc_piix.c
|
|
|
9ae3a8 |
hw/i386/pc_q35.c
|
|
|
9ae3a8 |
---
|
|
|
9ae3a8 |
include/hw/i386/pc.h | 1 +
|
|
|
9ae3a8 |
hw/i386/pc.c | 26 ++++++++++++++++++++++++++
|
|
|
9ae3a8 |
hw/i386/pc_piix.c | 16 +++++++++++++++-
|
|
|
9ae3a8 |
hw/i386/pc_q35.c | 12 ++++++++++--
|
|
|
9ae3a8 |
4 files changed, 52 insertions(+), 3 deletions(-)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
Signed-off-by: Michal Novotny <minovotn@redhat.com>
|
|
|
9ae3a8 |
---
|
|
|
9ae3a8 |
hw/i386/pc.c | 26 ++++++++++++++++++++++++++
|
|
|
9ae3a8 |
hw/i386/pc_piix.c | 16 +++++++++++++++-
|
|
|
9ae3a8 |
hw/i386/pc_q35.c | 12 ++++++++++--
|
|
|
9ae3a8 |
include/hw/i386/pc.h | 1 +
|
|
|
9ae3a8 |
4 files changed, 52 insertions(+), 3 deletions(-)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
diff --git a/hw/i386/pc.c b/hw/i386/pc.c
|
|
|
9ae3a8 |
index 68a8e1b..238f6a0 100644
|
|
|
9ae3a8 |
--- a/hw/i386/pc.c
|
|
|
9ae3a8 |
+++ b/hw/i386/pc.c
|
|
|
9ae3a8 |
@@ -985,6 +985,31 @@ void pc_cpus_init(const char *cpu_model, DeviceState *icc_bridge)
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
+/* pci-info ROM file. Little endian format */
|
|
|
9ae3a8 |
+typedef struct PcRomPciInfo {
|
|
|
9ae3a8 |
+ uint64_t w32_min;
|
|
|
9ae3a8 |
+ uint64_t w32_max;
|
|
|
9ae3a8 |
+ uint64_t w64_min;
|
|
|
9ae3a8 |
+ uint64_t w64_max;
|
|
|
9ae3a8 |
+} PcRomPciInfo;
|
|
|
9ae3a8 |
+
|
|
|
9ae3a8 |
+static void pc_fw_cfg_guest_info(PcGuestInfo *guest_info)
|
|
|
9ae3a8 |
+{
|
|
|
9ae3a8 |
+ PcRomPciInfo *info;
|
|
|
9ae3a8 |
+ if (!guest_info->has_pci_info) {
|
|
|
9ae3a8 |
+ return;
|
|
|
9ae3a8 |
+ }
|
|
|
9ae3a8 |
+
|
|
|
9ae3a8 |
+ info = g_malloc(sizeof *info);
|
|
|
9ae3a8 |
+ info->w32_min = cpu_to_le64(guest_info->pci_info.w32.begin);
|
|
|
9ae3a8 |
+ info->w32_max = cpu_to_le64(guest_info->pci_info.w32.end);
|
|
|
9ae3a8 |
+ info->w64_min = cpu_to_le64(guest_info->pci_info.w64.begin);
|
|
|
9ae3a8 |
+ info->w64_max = cpu_to_le64(guest_info->pci_info.w64.end);
|
|
|
9ae3a8 |
+ /* Pass PCI hole info to guest via a side channel.
|
|
|
9ae3a8 |
+ * Required so guest PCI enumeration does the right thing. */
|
|
|
9ae3a8 |
+ fw_cfg_add_file(guest_info->fw_cfg, "etc/pci-info", info, sizeof *info);
|
|
|
9ae3a8 |
+}
|
|
|
9ae3a8 |
+
|
|
|
9ae3a8 |
typedef struct PcGuestInfoState {
|
|
|
9ae3a8 |
PcGuestInfo info;
|
|
|
9ae3a8 |
Notifier machine_done;
|
|
|
9ae3a8 |
@@ -996,6 +1021,7 @@ void pc_guest_info_machine_done(Notifier *notifier, void *data)
|
|
|
9ae3a8 |
PcGuestInfoState *guest_info_state = container_of(notifier,
|
|
|
9ae3a8 |
PcGuestInfoState,
|
|
|
9ae3a8 |
machine_done);
|
|
|
9ae3a8 |
+ pc_fw_cfg_guest_info(&guest_info_state->info);
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
PcGuestInfo *pc_guest_info_init(ram_addr_t below_4g_mem_size,
|
|
|
9ae3a8 |
diff --git a/hw/i386/pc_piix.c b/hw/i386/pc_piix.c
|
|
|
9ae3a8 |
index 3a77998..6d50a4e 100644
|
|
|
9ae3a8 |
--- a/hw/i386/pc_piix.c
|
|
|
9ae3a8 |
+++ b/hw/i386/pc_piix.c
|
|
|
9ae3a8 |
@@ -59,6 +59,7 @@ static const int ide_iobase2[MAX_IDE_BUS] = { 0x3f6, 0x376 };
|
|
|
9ae3a8 |
static const int ide_irq[MAX_IDE_BUS] = { 14, 15 };
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
static bool smbios_type1_defaults = true;
|
|
|
9ae3a8 |
+static bool has_pci_info = true;
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
/* PC hardware initialisation */
|
|
|
9ae3a8 |
static void pc_init1(QEMUMachineInitArgs *args,
|
|
|
9ae3a8 |
@@ -122,6 +123,7 @@ static void pc_init1(QEMUMachineInitArgs *args,
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
guest_info = pc_guest_info_init(below_4g_mem_size, above_4g_mem_size);
|
|
|
9ae3a8 |
+ guest_info->has_pci_info = has_pci_info;
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
/* Set PCI window size the way seabios has always done it. */
|
|
|
9ae3a8 |
/* Power of 2 so bios can cover it with a single MTRR */
|
|
|
9ae3a8 |
@@ -248,8 +250,15 @@ static void pc_init_pci(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
#if 0 /* Disabled for Red Hat Enterprise Linux */
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
+static void pc_init_pci_1_5(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
+{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
+ pc_init_pci(args);
|
|
|
9ae3a8 |
+}
|
|
|
9ae3a8 |
+
|
|
|
9ae3a8 |
static void pc_init_pci_1_4(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
x86_cpu_compat_set_features("n270", FEAT_1_ECX, 0, CPUID_EXT_MOVBE);
|
|
|
9ae3a8 |
x86_cpu_compat_set_features("Westmere", FEAT_1_ECX, 0, CPUID_EXT_PCLMULQDQ);
|
|
|
9ae3a8 |
pc_init_pci(args);
|
|
|
9ae3a8 |
@@ -257,6 +266,7 @@ static void pc_init_pci_1_4(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
static void pc_init_pci_1_3(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
enable_compat_apic_id_mode();
|
|
|
9ae3a8 |
pc_init_pci(args);
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
@@ -264,6 +274,7 @@ static void pc_init_pci_1_3(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
/* PC machine init function for pc-1.1 to pc-1.2 */
|
|
|
9ae3a8 |
static void pc_init_pci_1_2(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
disable_kvm_pv_eoi();
|
|
|
9ae3a8 |
enable_compat_apic_id_mode();
|
|
|
9ae3a8 |
pc_init_pci(args);
|
|
|
9ae3a8 |
@@ -272,6 +283,7 @@ static void pc_init_pci_1_2(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
/* PC machine init function for pc-0.14 to pc-1.0 */
|
|
|
9ae3a8 |
static void pc_init_pci_1_0(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
disable_kvm_pv_eoi();
|
|
|
9ae3a8 |
enable_compat_apic_id_mode();
|
|
|
9ae3a8 |
pc_init_pci(args);
|
|
|
9ae3a8 |
@@ -280,6 +292,7 @@ static void pc_init_pci_1_0(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
/* PC init function for pc-0.10 to pc-0.13, and reused by xenfv */
|
|
|
9ae3a8 |
static void pc_init_pci_no_kvmclock(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
disable_kvm_pv_eoi();
|
|
|
9ae3a8 |
enable_compat_apic_id_mode();
|
|
|
9ae3a8 |
pc_init1(args, get_system_memory(), get_system_io(), 1, 0);
|
|
|
9ae3a8 |
@@ -290,6 +303,7 @@ static void pc_init_isa(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
if (!args->cpu_model) {
|
|
|
9ae3a8 |
args->cpu_model = "486";
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
disable_kvm_pv_eoi();
|
|
|
9ae3a8 |
enable_compat_apic_id_mode();
|
|
|
9ae3a8 |
pc_init1(args, get_system_memory(), get_system_io(), 0, 1);
|
|
|
9ae3a8 |
@@ -310,7 +324,7 @@ static QEMUMachine pc_i440fx_machine_v1_5 = {
|
|
|
9ae3a8 |
.name = "pc-i440fx-1.5",
|
|
|
9ae3a8 |
.alias = "pc",
|
|
|
9ae3a8 |
.desc = "Standard PC (i440FX + PIIX, 1996)",
|
|
|
9ae3a8 |
- .init = pc_init_pci,
|
|
|
9ae3a8 |
+ .init = pc_init_pci_1_5,
|
|
|
9ae3a8 |
.hot_add_cpu = pc_hot_add_cpu,
|
|
|
9ae3a8 |
.max_cpus = 255,
|
|
|
9ae3a8 |
.is_default = 1,
|
|
|
9ae3a8 |
diff --git a/hw/i386/pc_q35.c b/hw/i386/pc_q35.c
|
|
|
9ae3a8 |
index 9fab93c..7a58b61 100644
|
|
|
9ae3a8 |
--- a/hw/i386/pc_q35.c
|
|
|
9ae3a8 |
+++ b/hw/i386/pc_q35.c
|
|
|
9ae3a8 |
@@ -49,6 +49,7 @@
|
|
|
9ae3a8 |
#define MAX_SATA_PORTS 6
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
static bool smbios_type1_defaults = true;
|
|
|
9ae3a8 |
+static bool has_pci_info = true;
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
/* PC hardware initialisation */
|
|
|
9ae3a8 |
static void pc_q35_init(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
@@ -108,6 +109,7 @@ static void pc_q35_init(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
guest_info = pc_guest_info_init(below_4g_mem_size, above_4g_mem_size);
|
|
|
9ae3a8 |
+ guest_info->has_pci_info = has_pci_info;
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
/* allocate ram and load rom/bios */
|
|
|
9ae3a8 |
if (!xen_enabled()) {
|
|
|
9ae3a8 |
@@ -213,18 +215,24 @@ static void pc_q35_init(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
#if 0 /* Disabled for Red Hat Enterprise Linux */
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
+static void pc_q35_init_1_5(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
+{
|
|
|
9ae3a8 |
+ has_pci_info = false;
|
|
|
9ae3a8 |
+ pc_q35_init(args);
|
|
|
9ae3a8 |
+}
|
|
|
9ae3a8 |
+
|
|
|
9ae3a8 |
static void pc_q35_init_1_4(QEMUMachineInitArgs *args)
|
|
|
9ae3a8 |
{
|
|
|
9ae3a8 |
x86_cpu_compat_set_features("n270", FEAT_1_ECX, 0, CPUID_EXT_MOVBE);
|
|
|
9ae3a8 |
x86_cpu_compat_set_features("Westmere", FEAT_1_ECX, 0, CPUID_EXT_PCLMULQDQ);
|
|
|
9ae3a8 |
- pc_q35_init(args);
|
|
|
9ae3a8 |
+ pc_q35_init_1_5(args);
|
|
|
9ae3a8 |
}
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
static QEMUMachine pc_q35_machine_v1_5 = {
|
|
|
9ae3a8 |
.name = "pc-q35-1.5",
|
|
|
9ae3a8 |
.alias = "q35",
|
|
|
9ae3a8 |
.desc = "Standard PC (Q35 + ICH9, 2009)",
|
|
|
9ae3a8 |
- .init = pc_q35_init,
|
|
|
9ae3a8 |
+ .init = pc_q35_init_1_5,
|
|
|
9ae3a8 |
.hot_add_cpu = pc_hot_add_cpu,
|
|
|
9ae3a8 |
.max_cpus = 255,
|
|
|
9ae3a8 |
.compat_props = (GlobalProperty[]) {
|
|
|
9ae3a8 |
diff --git a/include/hw/i386/pc.h b/include/hw/i386/pc.h
|
|
|
9ae3a8 |
index 2518db6..2992464 100644
|
|
|
9ae3a8 |
--- a/include/hw/i386/pc.h
|
|
|
9ae3a8 |
+++ b/include/hw/i386/pc.h
|
|
|
9ae3a8 |
@@ -21,6 +21,7 @@ typedef struct PcPciInfo {
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
struct PcGuestInfo {
|
|
|
9ae3a8 |
PcPciInfo pci_info;
|
|
|
9ae3a8 |
+ bool has_pci_info;
|
|
|
9ae3a8 |
FWCfgState *fw_cfg;
|
|
|
9ae3a8 |
};
|
|
|
9ae3a8 |
|
|
|
9ae3a8 |
--
|
|
|
9ae3a8 |
1.7.11.7
|
|
|
9ae3a8 |
|