qemu-devel
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

Re: [Qemu-devel] [PATCH v4 7/7] hw/i386: build ACPI MADT (APIC) for fw_c


From: Michael S. Tsirkin
Subject: Re: [Qemu-devel] [PATCH v4 7/7] hw/i386: build ACPI MADT (APIC) for fw_cfg clients
Date: Wed, 24 Apr 2013 12:42:57 +0300

On Fri, Apr 19, 2013 at 12:58:03PM +0200, Laszlo Ersek wrote:
> On 04/18/13 22:30, Michael S. Tsirkin wrote:
> > On Thu, Apr 18, 2013 at 10:22:24PM +0200, Laszlo Ersek wrote:
> >> This patch reuses some code from SeaBIOS, which was originally under
> >> LGPLv2 and then relicensed to GPLv3 or LGPLv3, in QEMU under GPLv2+. This
> >> relicensing has been acked by all contributors that had contributed to the
> >> code since the v2->v3 relicense. ACKs approving the v2+ relicensing are
> >> listed below. The list might include ACKs from people not holding
> >> copyright on any parts of the reused code, but it's better to err on the
> >> side of caution and include them.
> >>
> >> Affected SeaBIOS files (GPLv2+ license headers added)
> >> <http://thread.gmane.org/gmane.comp.bios.coreboot.seabios/5949>:
> >>
> >>  src/acpi-dsdt-cpu-hotplug.dsl    |   15 +++++++++++++++
> >>  src/acpi-dsdt-dbug.dsl           |   15 +++++++++++++++
> >>  src/acpi-dsdt-hpet.dsl           |   15 +++++++++++++++
> >>  src/acpi-dsdt-isa.dsl            |   15 +++++++++++++++
> >>  src/acpi-dsdt-pci-crs.dsl        |   15 +++++++++++++++
> >>  src/acpi.c                       |   14 +++++++++++++-
> >>  src/acpi.h                       |   14 ++++++++++++++
> >>  src/ssdt-misc.dsl                |   15 +++++++++++++++
> >>  src/ssdt-pcihp.dsl               |   15 +++++++++++++++
> >>  src/ssdt-proc.dsl                |   15 +++++++++++++++
> >>  tools/acpi_extract.py            |   13 ++++++++++++-
> >>  tools/acpi_extract_preprocess.py |   13 ++++++++++++-
> >>  12 files changed, 171 insertions(+), 3 deletions(-)
> >>
> >> Each one of the listed people agreed to the following:
> >>
> >>> If you allow the use of your contribution in QEMU under the
> >>> terms of GPLv2 or later as proposed by this patch,
> >>> please respond to this mail including the line:
> >>>
> >>> Acked-by: Name <email address>
> >>
> >>   Acked-by: Gerd Hoffmann <address@hidden>
> >>   Acked-by: Jan Kiszka <address@hidden>
> >>   Acked-by: Jason Baron <address@hidden>
> >>   Acked-by: David Woodhouse <address@hidden>
> >>   Acked-by: Gleb Natapov <address@hidden>
> >>   Acked-by: Marcelo Tosatti <address@hidden>
> >>   Acked-by: Dave Frodin <address@hidden>
> >>   Acked-by: Paolo Bonzini <address@hidden>
> >>   Acked-by: Kevin O'Connor <address@hidden>
> >>   Acked-by: Laszlo Ersek <address@hidden>
> >>   Acked-by: Kenji Kaneshige <address@hidden>
> >>   Acked-by: Isaku Yamahata <address@hidden>
> >>   Acked-by: Magnus Christensson <address@hidden>
> >>   Acked-by: Hu Tao <address@hidden>
> >>   Acked-by: Eduardo Habkost <address@hidden>
> >>
> >> The patch incorporates ideas/suggestions from Michael Tsirkin's prototype
> >> code:
> >> - "hw/i386/pc.c" is too big, create new file "hw/i386/acpi.c" with
> >>   i386-specific ACPI table stuff,
> >> - separate preparation of individual tables from their installation as
> >>   fw_cfg files,
> >> - install these fw_cfg files inside pc_memory_init(), which is shared by
> >>   piix4/q35,
> >> - add the above licensing-related block to the commit message.
> >>
> >> Signed-off-by: Laszlo Ersek <address@hidden>
> >> ---
> >>  configure             |   12 ++++
> >>  hw/i386/Makefile.objs |    1 +
> >>  hw/i386/acpi.h        |    9 +++
> >>  hw/i386/acpi.c        |  159 
> >> +++++++++++++++++++++++++++++++++++++++++++++++++
> >>  hw/i386/pc.c          |   23 +++++++
> >>  5 files changed, 204 insertions(+), 0 deletions(-)
> >>  create mode 100644 hw/i386/acpi.h
> >>  create mode 100644 hw/i386/acpi.c
> >>
> >> diff --git a/configure b/configure
> >> index ed49f91..45a5f55 100755
> >> --- a/configure
> >> +++ b/configure
> >> @@ -241,6 +241,7 @@ gtk=""
> >>  gtkabi="2.0"
> >>  tpm="no"
> >>  libssh2=""
> >> +dynamic_acpi="no"
> >>  
> >>  # parse CC options first
> >>  for opt do
> >> @@ -928,6 +929,10 @@ for opt do
> >>    ;;
> >>    --enable-libssh2) libssh2="yes"
> >>    ;;
> >> +  --disable-dynamic-acpi) dynamic_acpi="no"
> >> +  ;;
> >> +  --enable-dynamic-acpi) dynamic_acpi="yes"
> >> +  ;;
> >>    *) echo "ERROR: unknown option $opt"; show_help="yes"
> >>    ;;
> >>    esac
> >> @@ -1195,6 +1200,8 @@ echo "  --gcov=GCOV              use specified gcov 
> >> [$gcov_tool]"
> >>  echo "  --enable-tpm             enable TPM support"
> >>  echo "  --disable-libssh2        disable ssh block device support"
> >>  echo "  --enable-libssh2         enable ssh block device support"
> >> +echo "  --disable-dynamic-acpi   disable dynamic ACPI table generation 
> >> (default)"
> >> +echo "  --enable-dynamic-acpi    enable dynamic ACPI table generation 
> >> (work in progress)"
> >>  echo ""
> >>  echo "NOTE: The object files are built at the place where configure is 
> >> launched"
> >>  exit 1
> >> @@ -3573,6 +3580,7 @@ echo "gcov enabled      $gcov"
> >>  echo "TPM support       $tpm"
> >>  echo "libssh2 support   $libssh2"
> >>  echo "TPM passthrough   $tpm_passthrough"
> >> +echo "dynamic ACPI tables $dynamic_acpi"
> >>  
> >>  if test "$sdl_too_old" = "yes"; then
> >>  echo "-> Your SDL version is too old - please upgrade to have SDL support"
> >> @@ -3958,6 +3966,10 @@ if test "$virtio_blk_data_plane" = "yes" ; then
> >>    echo 'CONFIG_VIRTIO_BLK_DATA_PLANE=$(CONFIG_VIRTIO)' >> $config_host_mak
> >>  fi
> >>  
> >> +if test "$dynamic_acpi" = "yes"; then
> >> +  echo "CONFIG_DYN_ACPI=y" >> $config_host_mak
> >> +fi
> >> +
> >>  # USB host support
> >>  case "$usb" in
> >>  linux)
> >> diff --git a/hw/i386/Makefile.objs b/hw/i386/Makefile.objs
> >> index 205d22e..8429d52 100644
> >> --- a/hw/i386/Makefile.objs
> >> +++ b/hw/i386/Makefile.objs
> >> @@ -1,6 +1,7 @@
> >>  obj-$(CONFIG_KVM) += kvm/
> >>  obj-y += multiboot.o smbios.o
> >>  obj-y += pc.o pc_piix.o pc_q35.o
> >> +obj-$(CONFIG_DYN_ACPI) += acpi.o
> >>  obj-$(CONFIG_XEN) += xen_domainbuild.o xen_machine_pv.o
> >>  
> >>  obj-y += kvmvapic.o
> >> diff --git a/hw/i386/acpi.h b/hw/i386/acpi.h
> >> new file mode 100644
> >> index 0000000..94f9ad3
> >> --- /dev/null
> >> +++ b/hw/i386/acpi.h
> >> @@ -0,0 +1,9 @@
> >> +#ifndef QEMU_HW_I386_ACPI_H
> >> +#define QEMU_HW_I386_ACPI_H
> >> +
> >> +#include <stddef.h>
> >> +
> >> +void acpi_build_madt(unsigned char **out_blob, size_t *out_blob_size,
> >> +                     unsigned num_lapic);
> >> +
> >> +#endif
> >> diff --git a/hw/i386/acpi.c b/hw/i386/acpi.c
> >> new file mode 100644
> >> index 0000000..179cdf5
> >> --- /dev/null
> >> +++ b/hw/i386/acpi.c
> >> @@ -0,0 +1,159 @@
> >> +/*
> >> + * Copyright (c) 2013 Red Hat Inc.
> >> + * Copyright (C) 2008-2010  Kevin O'Connor <address@hidden>
> >> + * Copyright (C) 2006 Fabrice Bellard
> >> + *
> >> + * This program is free software; you can redistribute it and/or modify
> >> + * it under the terms of the GNU General Public License as published by
> >> + * the Free Software Foundation; either version 2 of the License, or
> >> + * (at your option) any later version.
> >> + *
> >> + * This program is distributed in the hope that it will be useful,
> >> + * but WITHOUT ANY WARRANTY; without even the implied warranty of
> >> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
> >> + * GNU General Public License for more details.
> >> + *
> >> + * You should have received a copy of the GNU General Public License along
> >> + * with this program; if not, see <http://www.gnu.org/licenses/>.
> >> + *
> >> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 
> >> EXPRESS OR
> >> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 
> >> MERCHANTABILITY,
> >> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
> >> + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR 
> >> OTHER
> >> + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 
> >> ARISING FROM,
> >> + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS 
> >> IN
> >> + * THE SOFTWARE.
> >> + */
> >> +
> >> +#include "hw/acpi/acpi.h"
> >> +#include "hw/i386/acpi.h"
> >> +#include "kvm_i386.h"
> >> +#include "sysemu/sysemu.h"
> >> +
> >> +static void acpi_table_fill_hdr(AcpiTableStdHdr *std_hdr, size_t 
> >> blob_size,
> >> +                                const char *sig)
> >> +{
> >> +    g_assert(blob_size >= sizeof *std_hdr);
> >> +
> >> +    *std_hdr = acpi_dfl_hdr;
> >> +    strncpy(std_hdr->sig, sig, sizeof std_hdr->sig);
> >> +    strncpy(std_hdr->oem_id, "QEMU  ", sizeof std_hdr->oem_id);
> >> +    strncpy(std_hdr->oem_table_id + 4, sig, sizeof std_hdr->oem_table_id 
> >> - 4);
> >> +    std_hdr->length = cpu_to_le32(blob_size);
> >> +    std_hdr->checksum = acpi_checksum((uint8_t *)std_hdr, blob_size);
> >> +}
> >> +
> >> +void acpi_build_madt(unsigned char **out_blob, size_t *out_blob_size,
> >> +                     unsigned num_lapic)
> >> +{
> >> +    typedef struct {
> >> +        uint8_t    type;
> >> +        uint8_t    length;
> >> +    } QEMU_PACKED AcpiSubHdr;
> >> +
> >> +    AcpiTableStdHdr *std_hdr;
> >> +    struct {
> >> +        uint32_t   lapic_addr; /* Local Interrupt Controller Address */
> >> +        uint32_t   flags;      /* Multiple APIC flags */
> >> +    } QEMU_PACKED *madt;
> >> +    struct {
> >> +        AcpiSubHdr hdr;
> >> +        uint8_t    processor_id; /* ACPI Processor ID */
> >> +        uint8_t    apic_id;      /* APIC ID */
> >> +        uint32_t   flags;        /* LOcal APIC flags */
> >> +    } QEMU_PACKED *lapic;
> >> +    struct {
> >> +        AcpiSubHdr hdr;
> >> +        uint8_t    io_apic_id;   /* The I/O APIC's ID */
> >> +        uint8_t    reserved;     /* constant zero */
> >> +        uint32_t   io_apic_addr; /* 32-bit physical address to access */
> >> +        uint32_t   gsi_base;     /* interrupt inputs start here */
> >> +    } QEMU_PACKED *io_apic;
> >> +    struct {
> >> +        AcpiSubHdr hdr;
> >> +        uint8_t    bus;    /* constant zero: ISA */
> >> +        uint8_t    source; /* this bus-relative interrupt source... */
> >> +        uint32_t   gsi;    /* ... will signal this global system 
> >> interrupt */
> >> +        uint16_t   flags;  /* MPS INTI Flags: Polarity, Trigger Mode */
> >> +    } QEMU_PACKED *int_src_ovr;
> >> +    struct {
> >> +        AcpiSubHdr hdr;
> >> +        uint8_t    processor_id; /* ACPI Processor ID */
> >> +        uint16_t   flags;        /* MPS INTI Flags: Polarity, Trigger 
> >> Mode */
> >> +        uint8_t    lint;         /* LAPIC interrupt input for NMI */
> >> +    } QEMU_PACKED *lapic_nmi;
> >> +
> >> +    static const uint8_t pci_isa_irq[] = { 5, 9, 10, 11 };
> >> +
> >> +    unsigned num_int_src_ovr, i;
> >> +    size_t blob_size;
> >> +    char unsigned *blob;
> >> +
> >> +    num_int_src_ovr = sizeof pci_isa_irq + kvm_allows_irq0_override();
> >> +
> >> +    blob_size = (sizeof *std_hdr)     * 1               +
> >> +                (sizeof *madt)        * 1               +
> >> +                (sizeof *lapic)       * num_lapic       +
> >> +                (sizeof *io_apic)     * 1               +
> >> +                (sizeof *int_src_ovr) * num_int_src_ovr +
> >> +                (sizeof *lapic_nmi)   * 1;
> >> +    blob      = g_malloc(blob_size);
> >> +
> >> +    std_hdr     = (void *)blob;
> >> +    madt        = (void *)(std_hdr     + 1              );
> >> +    lapic       = (void *)(madt        + 1              );
> >> +    io_apic     = (void *)(lapic       + num_lapic      );
> >> +    int_src_ovr = (void *)(io_apic     + 1              );
> >> +    lapic_nmi   = (void *)(int_src_ovr + num_int_src_ovr);
> >> +
> >> +    madt->lapic_addr = cpu_to_le32(APIC_DEFAULT_ADDRESS);
> >> +    madt->flags      = cpu_to_le32(1); /* PCAT_COMPAT */
> >> +
> >> +    /* create a Local APIC structure for each possible APIC ID */
> >> +    for (i = 0; i < num_lapic; ++i) {
> >> +        lapic[i].hdr.type     = 0; /* Processor Local APIC */
> >> +        lapic[i].hdr.length   = sizeof *lapic;
> >> +        lapic[i].processor_id = i;
> >> +        lapic[i].apic_id      = i;
> >> +        lapic[i].flags        = cpu_to_le32(0); /* disabled */
> >> +    }
> >> +    /* enable the CPUs with a CPU index in the [0..smp_cpus-1] range */
> >> +    for (i = 0; i < smp_cpus; ++i) {
> >> +        lapic[x86_cpu_apic_id_from_index(i)].flags = cpu_to_le32(1);
> >> +    }
> >> +
> >> +    io_apic->hdr.type     = 1; /* I/O APIC */
> >> +    io_apic->hdr.length   = sizeof *io_apic;
> >> +    io_apic->io_apic_id   = 0;
> >> +    io_apic->reserved     = 0;
> >> +    io_apic->io_apic_addr = cpu_to_le32(IO_APIC_DEFAULT_ADDRESS);
> >> +    io_apic->gsi_base     = cpu_to_le32(0);
> >> +
> >> +    for (i = 0; i < sizeof pci_isa_irq; ++i) {
> >> +        int_src_ovr[i].hdr.type   = 2; /* Interrupt Source Override */
> >> +        int_src_ovr[i].hdr.length = sizeof *int_src_ovr;
> >> +        int_src_ovr[i].bus        = 0;
> >> +        int_src_ovr[i].source     = pci_isa_irq[i];
> >> +        int_src_ovr[i].gsi        = cpu_to_le32(pci_isa_irq[i]);
> >> +        int_src_ovr[i].flags      = cpu_to_le16(0xd);
> >> +                                    /* active high, level-triggered */
> >> +    }
> >> +    if (kvm_allows_irq0_override()) {
> >> +        int_src_ovr[i].hdr.type   = 2; /* Interrupt Source Override */
> >> +        int_src_ovr[i].hdr.length = sizeof *int_src_ovr;
> >> +        int_src_ovr[i].bus        = 0;
> >> +        int_src_ovr[i].source     = 0;
> >> +        int_src_ovr[i].gsi        = cpu_to_le32(2);
> >> +        int_src_ovr[i].flags      = cpu_to_le16(0); /* conforms to bus 
> >> spec */
> >> +    }
> >> +
> >> +    lapic_nmi->hdr.type     = 4; /* Local APIC NMI */
> >> +    lapic_nmi->hdr.length   = sizeof *lapic_nmi;
> >> +    lapic_nmi->processor_id = 0xff; /* all processors */
> >> +    lapic_nmi->flags        = cpu_to_le16(0); /* conforms to bus spec */
> >> +    lapic_nmi->lint         = 1; /* NMI connected to LAPIC input LINT1 */
> >> +
> >> +    acpi_table_fill_hdr(std_hdr, blob_size, "APIC");
> >> +    *out_blob = blob;
> >> +    *out_blob_size = blob_size;
> >> +}
> >> diff --git a/hw/i386/pc.c b/hw/i386/pc.c
> >> index 8727489..15c6284 100644
> >> --- a/hw/i386/pc.c
> >> +++ b/hw/i386/pc.c
> >> @@ -54,6 +54,10 @@
> >>  #include "qemu/config-file.h"
> >>  #include "hw/acpi/acpi.h"
> >>  
> >> +#ifdef CONFIG_DYN_ACPI
> >> +#  include "hw/i386/acpi.h"
> >> +#endif
> >> +
> >>  /* debug PC/ISA interrupts */
> >>  //#define DEBUG_IRQ
> >>  
> >> @@ -922,6 +926,21 @@ void pc_acpi_init(const char *default_dsdt)
> >>      }
> >>  }
> >>  
> >> +#ifdef CONFIG_DYN_ACPI
> >> +static void pc_acpi_madt(FWCfgState *fw_cfg)
> >> +{
> >> +    unsigned num_lapic;
> >> +    char unsigned *blob;
> >> +    size_t blob_size;
> >> +
> >> +    /* see note on FW_CFG_MAX_CPUS in bochs_bios_init() */
> >> +    num_lapic = pc_apic_id_limit(max_cpus);
> >> +
> >> +    acpi_build_madt(&blob, &blob_size, num_lapic);
> >> +    fw_cfg_add_file(fw_cfg, "etc/acpi/APIC", blob, blob_size);
> >> +}
> >> +#endif
> >> +
> >>  FWCfgState *pc_memory_init(MemoryRegion *system_memory,
> >>                             const char *kernel_filename,
> >>                             const char *kernel_cmdline,
> >> @@ -974,6 +993,10 @@ FWCfgState *pc_memory_init(MemoryRegion 
> >> *system_memory,
> >>      fw_cfg = bochs_bios_init();
> >>      rom_set_fw(fw_cfg);
> > 
> > Let's cut out the ifdefs, just use an if below. Compiler is smart enough
> > to do it right.
> 
> I considered that, because it's how SeaBIOS does it.
> 
> There are several "depths" to do this; for example, I could even build
> hw/i386/acpi.o with the switch being off, and pass the object file to
> the linker (ie. no variable substitution in Makefile.objs) and let the
> linker omit it when it finds no references.
> 
> However, I obviously checked how existent parts of the code do it,
> specifically CONFIG_XEN, and I followed that manner. See
> "hw/i386/pc_piix.c" and "arch_init.c". There's not one CONFIG_XXX
> feature test macro in the qemu source (that I can find) that is examined
> with the "if" statement.

Good point.

> IOW with all due respect I disagree.
> 
> Thanks,
> Laszlo

Fine, it's not a big deal.

> > 
> >> +#ifdef CONFIG_DYN_ACPI
> >> +    pc_acpi_madt(fw_cfg);
> >> +#endif
> >> +
> >>      if (linux_boot) {
> >>          load_linux(fw_cfg, kernel_filename, initrd_filename, 
> >> kernel_cmdline, below_4g_mem_size);
> >>      }
> >> -- 
> >> 1.7.1
> >>



reply via email to

[Prev in Thread] Current Thread [Next in Thread]