Add a function to "struct device_operations" to return the ACPI name
for the device, and helper functions to find this name (either from
the device or its parent) and to build a fully qualified ACPI path
from the root device.
This addition will allow device drivers to generate their ACPI AML in
the SSDT at boot, with customization supplied by devicetree.cb,
instead of needing custom DSDT ASL for every mainboard.
The root device acpi_name is defined as "\_SB" and is used to start
the path when building a fully qualified name.
This requires SOC support to provide handlers for returning the ACPI
name for devices that it owns, and those names must match the objects
declared in the DSDT. The handler can be done either in each device
driver or with a global handler for the entire SOC.
Simplified example of how this can be used for an i2c device declared
in devicetree.cb with:
chip soc/intel/skylake # "_SB" (from root device)
device domain 0 on # "PCI0"
device pci 19.2 on # "I2C4"
chip drivers/i2c/test0
device i2c 1a.0 on end # "TST0"
end
end
end
end
And basic SSDT generating code in the device driver:
acpigen_write_scope(acpi_device_scope(dev));
acpigen_write_device(acpi_device_name(dev));
acpigen_write_string("_HID", "TEST0000");
acpigen_write_byte("_UID", 0);
acpigen_pop_len(); /bin /boot /cdrom /dev /etc /home /initrd.img /initrd.img.old /lib /lib32 /lib64 /libx32 /lost+found /media /mnt /opt /proc /root /run /sbin /srv /sys /tmp /usr /var /vmlinuz /vmlinuz.old device */
acpigen_pop_len(); /bin /boot /cdrom /dev /etc /home /initrd.img /initrd.img.old /lib /lib32 /lib64 /libx32 /lost+found /media /mnt /opt /proc /root /run /sbin /srv /sys /tmp /usr /var /vmlinuz /vmlinuz.old scope */
Will produce this ACPI code:
Scope (_SB.PCI0.I2C4) {
Device (TST0) {
Name (_HID, "TEST0000")
Name (_UID, 0)
}
}
BUG=None
BRANCH=None
TEST=None
Change-Id: Ie149595aeab96266fa5f006e7934339f0119ac54
Original-Signed-off-by: Duncan Laurie <dlaurie@chromium.org>
Original-Reviewed-on: https://review.coreboot.org/14840
Original-Tested-by: build bot (Jenkins)
Original-Reviewed-by: Aaron Durbin <adurbin@chromium.org>
Signed-off-by: Aaron Durbin <adurbin@chromium.org>
Reviewed-on: https://chromium-review.googlesource.com/346989
Commit-Ready: Martin Roth <martinroth@chromium.org>
Tested-by: Martin Roth <martinroth@chromium.org>
Reviewed-by: Martin Roth <martinroth@chromium.org>
157 lines
4.5 KiB
C
157 lines
4.5 KiB
C
/*
|
|
* This file is part of the coreboot project.
|
|
*
|
|
* Copyright (C) 2003-2004 Linux Networx
|
|
* (Written by Eric Biederman <ebiederman@lnxi.com> for Linux Networx)
|
|
* Copyright (C) 2003 Ronald G. Minnich <rminnich@gmail.com>
|
|
* Copyright (C) 2004-2005 Li-Ta Lo <ollie@lanl.gov>
|
|
* Copyright (C) 2005 Tyan
|
|
* (Written by Yinghai Lu <yhlu@tyan.com> for Tyan)
|
|
*
|
|
* 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; version 2 of the License.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <console/console.h>
|
|
#include <device/device.h>
|
|
#include <device/pci.h>
|
|
#include <reset.h>
|
|
|
|
const char mainboard_name[] = CONFIG_MAINBOARD_VENDOR " " CONFIG_MAINBOARD_PART_NUMBER;
|
|
|
|
/**
|
|
* Scan devices on static buses.
|
|
*
|
|
* The enumeration of certain buses is purely static. The existence of
|
|
* devices on those buses can be completely determined at compile time
|
|
* and is specified in the config file. Typical examples are the 'PNP'
|
|
* devices on a legacy ISA/LPC bus. There is no need of probing of any kind,
|
|
* the only thing we have to do is to walk through the bus and
|
|
* enable or disable devices as indicated in the config file.
|
|
*
|
|
* On the other hand, some devices are virtual and their existence is
|
|
* artificial. They can not be probed at run time. One example is the
|
|
* debug device. Those virtual devices have to be listed in the config
|
|
* file under some static bus in order to be enumerated at run time.
|
|
*
|
|
* @param bus Pointer to the device to which the static buses are attached to.
|
|
*/
|
|
|
|
void scan_static_bus(device_t bus)
|
|
{
|
|
device_t child;
|
|
struct bus *link;
|
|
|
|
for (link = bus->link_list; link; link = link->next) {
|
|
for (child = link->children; child; child = child->sibling) {
|
|
|
|
if (child->chip_ops && child->chip_ops->enable_dev)
|
|
child->chip_ops->enable_dev(child);
|
|
|
|
if (child->ops && child->ops->enable)
|
|
child->ops->enable(child);
|
|
|
|
printk(BIOS_DEBUG, "%s %s\n", dev_path(child),
|
|
child->enabled ? "enabled" : "disabled");
|
|
}
|
|
}
|
|
}
|
|
|
|
void scan_lpc_bus(device_t bus)
|
|
{
|
|
printk(BIOS_SPEW, "%s for %s\n", __func__, dev_path(bus));
|
|
|
|
scan_static_bus(bus);
|
|
|
|
printk(BIOS_SPEW, "%s for %s done\n", __func__, dev_path(bus));
|
|
}
|
|
|
|
void scan_smbus(device_t bus)
|
|
{
|
|
device_t child;
|
|
struct bus *link;
|
|
static int smbus_max = 0;
|
|
|
|
printk(BIOS_SPEW, "%s for %s\n", __func__, dev_path(bus));
|
|
|
|
for (link = bus->link_list; link; link = link->next) {
|
|
|
|
link->secondary = ++smbus_max;
|
|
|
|
for (child = link->children; child; child = child->sibling) {
|
|
|
|
if (child->chip_ops && child->chip_ops->enable_dev)
|
|
child->chip_ops->enable_dev(child);
|
|
|
|
if (child->ops && child->ops->enable)
|
|
child->ops->enable(child);
|
|
|
|
printk(BIOS_DEBUG, "smbus: %s[%d]->", dev_path(child->bus->dev),
|
|
child->bus->link_num);
|
|
|
|
printk(BIOS_DEBUG, "%s %s\n", dev_path(child),
|
|
child->enabled ? "enabled" : "disabled");
|
|
}
|
|
}
|
|
|
|
printk(BIOS_SPEW, "%s for %s done\n", __func__, dev_path(bus));
|
|
}
|
|
|
|
/**
|
|
* Scan root bus for generic systems.
|
|
*
|
|
* This function is the default scan_bus() method of the root device.
|
|
*
|
|
* @param root The root device structure.
|
|
*/
|
|
static void root_dev_scan_bus(device_t bus)
|
|
{
|
|
struct bus *link;
|
|
|
|
printk(BIOS_SPEW, "%s for %s\n", __func__, dev_path(bus));
|
|
|
|
scan_static_bus(bus);
|
|
|
|
for (link = bus->link_list; link; link = link->next)
|
|
scan_bridges(link);
|
|
|
|
printk(BIOS_SPEW, "%s for %s done\n", __func__, dev_path(bus));
|
|
}
|
|
|
|
static void root_dev_reset(struct bus *bus)
|
|
{
|
|
printk(BIOS_INFO, "Resetting board...\n");
|
|
hard_reset();
|
|
}
|
|
|
|
#if IS_ENABLED(CONFIG_HAVE_ACPI_TABLES)
|
|
static const char *root_dev_acpi_name(struct device *dev)
|
|
{
|
|
return "\\_SB";
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Default device operation for root device.
|
|
*
|
|
* This is the default device operation for root devices. These operations
|
|
* should be fully usable as is. However the chip_operations::enable_dev()
|
|
* of a motherboard can override this if you want non-default behavior.
|
|
*/
|
|
struct device_operations default_dev_ops_root = {
|
|
.read_resources = DEVICE_NOOP,
|
|
.set_resources = DEVICE_NOOP,
|
|
.enable_resources = DEVICE_NOOP,
|
|
.init = DEVICE_NOOP,
|
|
.scan_bus = root_dev_scan_bus,
|
|
.reset_bus = root_dev_reset,
|
|
#if IS_ENABLED(CONFIG_HAVE_ACPI_TABLES)
|
|
.acpi_name = root_dev_acpi_name,
|
|
#endif
|
|
};
|