cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

ssdt-overlays.rst (6805B)


      1.. SPDX-License-Identifier: GPL-2.0
      2
      3=============
      4SSDT Overlays
      5=============
      6
      7In order to support ACPI open-ended hardware configurations (e.g. development
      8boards) we need a way to augment the ACPI configuration provided by the firmware
      9image. A common example is connecting sensors on I2C / SPI buses on development
     10boards.
     11
     12Although this can be accomplished by creating a kernel platform driver or
     13recompiling the firmware image with updated ACPI tables, neither is practical:
     14the former proliferates board specific kernel code while the latter requires
     15access to firmware tools which are often not publicly available.
     16
     17Because ACPI supports external references in AML code a more practical
     18way to augment firmware ACPI configuration is by dynamically loading
     19user defined SSDT tables that contain the board specific information.
     20
     21For example, to enumerate a Bosch BMA222E accelerometer on the I2C bus of the
     22Minnowboard MAX development board exposed via the LSE connector [1], the
     23following ASL code can be used::
     24
     25    DefinitionBlock ("minnowmax.aml", "SSDT", 1, "Vendor", "Accel", 0x00000003)
     26    {
     27        External (\_SB.I2C6, DeviceObj)
     28
     29        Scope (\_SB.I2C6)
     30        {
     31            Device (STAC)
     32            {
     33                Name (_HID, "BMA222E")
     34                Name (RBUF, ResourceTemplate ()
     35                {
     36                    I2cSerialBus (0x0018, ControllerInitiated, 0x00061A80,
     37                                AddressingMode7Bit, "\\_SB.I2C6", 0x00,
     38                                ResourceConsumer, ,)
     39                    GpioInt (Edge, ActiveHigh, Exclusive, PullDown, 0x0000,
     40                            "\\_SB.GPO2", 0x00, ResourceConsumer, , )
     41                    { // Pin list
     42                        0
     43                    }
     44                })
     45
     46                Method (_CRS, 0, Serialized)
     47                {
     48                    Return (RBUF)
     49                }
     50            }
     51        }
     52    }
     53
     54which can then be compiled to AML binary format::
     55
     56    $ iasl minnowmax.asl
     57
     58    Intel ACPI Component Architecture
     59    ASL Optimizing Compiler version 20140214-64 [Mar 29 2014]
     60    Copyright (c) 2000 - 2014 Intel Corporation
     61
     62    ASL Input:     minnomax.asl - 30 lines, 614 bytes, 7 keywords
     63    AML Output:    minnowmax.aml - 165 bytes, 6 named objects, 1 executable opcodes
     64
     65[1] https://www.elinux.org/Minnowboard:MinnowMax#Low_Speed_Expansion_.28Top.29
     66
     67The resulting AML code can then be loaded by the kernel using one of the methods
     68below.
     69
     70Loading ACPI SSDTs from initrd
     71==============================
     72
     73This option allows loading of user defined SSDTs from initrd and it is useful
     74when the system does not support EFI or when there is not enough EFI storage.
     75
     76It works in a similar way with initrd based ACPI tables override/upgrade: SSDT
     77AML code must be placed in the first, uncompressed, initrd under the
     78"kernel/firmware/acpi" path. Multiple files can be used and this will translate
     79in loading multiple tables. Only SSDT and OEM tables are allowed. See
     80initrd_table_override.txt for more details.
     81
     82Here is an example::
     83
     84    # Add the raw ACPI tables to an uncompressed cpio archive.
     85    # They must be put into a /kernel/firmware/acpi directory inside the
     86    # cpio archive.
     87    # The uncompressed cpio archive must be the first.
     88    # Other, typically compressed cpio archives, must be
     89    # concatenated on top of the uncompressed one.
     90    mkdir -p kernel/firmware/acpi
     91    cp ssdt.aml kernel/firmware/acpi
     92
     93    # Create the uncompressed cpio archive and concatenate the original initrd
     94    # on top:
     95    find kernel | cpio -H newc --create > /boot/instrumented_initrd
     96    cat /boot/initrd >>/boot/instrumented_initrd
     97
     98Loading ACPI SSDTs from EFI variables
     99=====================================
    100
    101This is the preferred method, when EFI is supported on the platform, because it
    102allows a persistent, OS independent way of storing the user defined SSDTs. There
    103is also work underway to implement EFI support for loading user defined SSDTs
    104and using this method will make it easier to convert to the EFI loading
    105mechanism when that will arrive. To enable it, the
    106CONFIG_EFI_CUSTOM_SSDT_OVERLAYS shoyld be chosen to y.
    107
    108In order to load SSDTs from an EFI variable the ``"efivar_ssdt=..."`` kernel
    109command line parameter can be used (the name has a limitation of 16 characters).
    110The argument for the option is the variable name to use. If there are multiple
    111variables with the same name but with different vendor GUIDs, all of them will
    112be loaded.
    113
    114In order to store the AML code in an EFI variable the efivarfs filesystem can be
    115used. It is enabled and mounted by default in /sys/firmware/efi/efivars in all
    116recent distribution.
    117
    118Creating a new file in /sys/firmware/efi/efivars will automatically create a new
    119EFI variable. Updating a file in /sys/firmware/efi/efivars will update the EFI
    120variable. Please note that the file name needs to be specially formatted as
    121"Name-GUID" and that the first 4 bytes in the file (little-endian format)
    122represent the attributes of the EFI variable (see EFI_VARIABLE_MASK in
    123include/linux/efi.h). Writing to the file must also be done with one write
    124operation.
    125
    126For example, you can use the following bash script to create/update an EFI
    127variable with the content from a given file::
    128
    129    #!/bin/sh -e
    130
    131    while [ -n "$1" ]; do
    132            case "$1" in
    133            "-f") filename="$2"; shift;;
    134            "-g") guid="$2"; shift;;
    135            *) name="$1";;
    136            esac
    137            shift
    138    done
    139
    140    usage()
    141    {
    142            echo "Syntax: ${0##*/} -f filename [ -g guid ] name"
    143            exit 1
    144    }
    145
    146    [ -n "$name" -a -f "$filename" ] || usage
    147
    148    EFIVARFS="/sys/firmware/efi/efivars"
    149
    150    [ -d "$EFIVARFS" ] || exit 2
    151
    152    if stat -tf $EFIVARFS | grep -q -v de5e81e4; then
    153            mount -t efivarfs none $EFIVARFS
    154    fi
    155
    156    # try to pick up an existing GUID
    157    [ -n "$guid" ] || guid=$(find "$EFIVARFS" -name "$name-*" | head -n1 | cut -f2- -d-)
    158
    159    # use a randomly generated GUID
    160    [ -n "$guid" ] || guid="$(cat /proc/sys/kernel/random/uuid)"
    161
    162    # efivarfs expects all of the data in one write
    163    tmp=$(mktemp)
    164    /bin/echo -ne "\007\000\000\000" | cat - $filename > $tmp
    165    dd if=$tmp of="$EFIVARFS/$name-$guid" bs=$(stat -c %s $tmp)
    166    rm $tmp
    167
    168Loading ACPI SSDTs from configfs
    169================================
    170
    171This option allows loading of user defined SSDTs from user space via the configfs
    172interface. The CONFIG_ACPI_CONFIGFS option must be select and configfs must be
    173mounted. In the following examples, we assume that configfs has been mounted in
    174/sys/kernel/config.
    175
    176New tables can be loading by creating new directories in /sys/kernel/config/acpi/table
    177and writing the SSDT AML code in the aml attribute::
    178
    179    cd /sys/kernel/config/acpi/table
    180    mkdir my_ssdt
    181    cat ~/ssdt.aml > my_ssdt/aml