bios_setting: Add a write handler

Tested:
```
// Test that it writes a new file if it doesn't exist
~# rm /run/oem_bios_setting

// Invalid command (no size / payload)
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x19
Unable to send RAW command (channel=0x0 netfn=0x2e lun=0x0 cmd=0x32 rsp=0xc7): Request data length invalid

// Invalid command (size doesn't match)
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x19 0x01 0x30 0x31
Unable to send RAW command (channel=0x0 netfn=0x2e lun=0x0 cmd=0x32 rsp=0xc7): Request data length invalid

// Command success
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x19 0x02 0x30 0x31
 79 2b 00 19 02

// Read back
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x18
 79 2b 00 18 02 30 31
~# cat /run/oem_bios_setting
01~#

// Verify overwrite works (truncates the previous bytes)
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x19 0x01 0x33
 79 2b 00 19 01
~# cat /run/oem_bios_setting
3~#
~# ipmitool raw 0x2e 0x32 0x79 0x2b 0x00 0x18
 79 2b 00 18 01 33
```

Signed-off-by: Brandon Kim <brandonkim@google.com>
Change-Id: I84db782da9b2f121c0a81a855692b5ca25ffda54
6 files changed
tree: 10931ca7a9248b5ed26ab173ab27aef0d883f401
  1. bifurcation/
  2. subprojects/
  3. test/
  4. .clang-format
  5. .gitignore
  6. .lcovrc
  7. bios_setting.cpp
  8. bios_setting.hpp
  9. bm_instance.cpp
  10. bm_instance.hpp
  11. bmc_mode.cpp
  12. bmc_mode.hpp
  13. bmc_mode_enum.hpp
  14. cable.cpp
  15. cable.hpp
  16. commands.hpp
  17. cpld.cpp
  18. cpld.hpp
  19. entity_name.cpp
  20. entity_name.hpp
  21. errors.hpp
  22. eth.cpp
  23. eth.hpp
  24. file_system_wrapper.cpp
  25. file_system_wrapper.hpp
  26. file_system_wrapper_impl.hpp
  27. flash_size.cpp
  28. flash_size.hpp
  29. gbmc-enqueue-powercycle.sh
  30. gbmc-host-poweroff.target
  31. gbmc-psu-hardreset-pre.target
  32. gbmc-psu-hardreset-time.service
  33. gbmc-psu-hardreset.target
  34. google_accel_oob.cpp
  35. google_accel_oob.hpp
  36. handler.cpp
  37. handler.hpp
  38. handler_impl.hpp
  39. host_power_off.cpp
  40. host_power_off.hpp
  41. ipmi.cpp
  42. ipmi.hpp
  43. LICENSE
  44. linux_boot_done.cpp
  45. linux_boot_done.hpp
  46. machine_name.cpp
  47. machine_name.hpp
  48. main.cpp
  49. MAINTAINERS
  50. meson.build
  51. meson.options
  52. OWNERS
  53. pcie_bifurcation.cpp
  54. pcie_bifurcation.hpp
  55. pcie_i2c.cpp
  56. pcie_i2c.hpp
  57. psu.cpp
  58. psu.hpp
  59. README.md
  60. util.cpp
  61. util.hpp
README.md

Sys Commands (0x32)

There are and will be a variety of sys specific commands.

Cablecheck - SubCommand 0x00

The cablecheck command checks whether the BMC is seeing traffic between itself and the host's NIC. Sys specifies which if_name is expected to be connected. The BMC presently only checks traffic on the interface specified. There are now ethernet statistics available over IPMI, which can be checked directly in lieu of this.

Request

Byte(s)ValueData
0x000x00Subcommand
0x01If_name lengthWhere you expect the cable, eth0 or eth1, etc
0x02 ...The nameThe string, not null-terminated

Response

Byte(s)ValueData
0x000x00Subcommand
0x010x00/0x010 for false, 1 for true

CpldVersion - SubCommand 0x01

Any CPLD on the system that can only be read directly via the BMC can have its version exported to Sys via the cpld version command.

Request

Byte(s)ValueData
0x000x01Subcommand
0x01CPLD IDA one-byte identifier for the CPLD file to read, unsigned byte.

Response

Byte(s)ValueData
0x000x01Subcommand
0x01MajorMajor version
0x02MinorMinor Version
0x03Sub 1Third version number
0x04Sub 2Fourth version number

Per the above, if the version number doesn't fit in a byte it'll be cast to size.

GetEthDevice - SubCommand 0x02

The BMC itself must have hard-coded into the image, which ethernet device is connected to the host NIC. This is true also in the mapping of ethernet device to channel number. Alternatively, you can pass a specific interface name for channel lookup. The channel number is used to configure the ethernet device over IPMI, instead of the interface name. This is because we leverage the current IPMI command set to read and write the networking configuration. Sys can be programmed already to have this information in the board protobuf, however, this information -- can be read from the BMC over IPMI.

Request

Byte(s)ValueData
0x000x02Subcommand
0x01...if_name(optional) The interface name, not null-terminated

Response

Byte(s)ValueData
0x000x02Subcommand
0x01Channel numberThe IPMI channel number for use with the network configuration commands (such as reading the MAC or IP address of the BMC).
0x02if_name lengthThe length of the if_name in bytes.
0x03...if_nameThe interface name, not null-terminated

DelayedHardReset - SubCommand 0x03

Sys needs to be able to tell the BMC to reset the host but given a delay in seconds.

Request

Byte(s)ValueData
0x000x03Subcommand
0x01..0x04Seconds to delay (uint32)

Response

Byte(s)ValueData
0x000x03Subcommand

GetPCIeSlotsCount - SubCommand 0x04

Sys can get the total number of PCIe slots from BMC using this command. When BMC receives this command, BMC can enumerate over all the PCIe slots and create a hashmap with all the available PCIe slot name - I2C bus number mappings. BMC can then send the total number of PCIe slots as part of this command response.

Request

Byte(s)ValueData
0x000x04Subcommand

Response

Byte(s)ValueData
0x000x04Subcommand
0x01Total number of PCIe slotsTotal number of PCIe slots

GetPCIeSlotI2cBusMapping - SubCommand 0x05

If Sys gets N total slots as part of the above command, then Sys can send this command N times with Entry IDs ranging from 0 to N - 1. Say, Sys sends this command with Entry ID as 1, BMC can go and fetch the first PCIe slot name - I2C bus number mapping from the hashmap created above and then send the PCIe slot name and I2C bus number as part of the command response.

Request

Byte(s)ValueData
0x000x05Subcommand
0x01Entry IDEntry ID ranging from 0 to N - 1

Response

Byte(s)ValueData
0x000x05Subcommand
0x01I2C bus numberThe I2C bus number which is input to the above PCIe slot
0x02PCIe slot name lengthThe PCIe slot name length
0x03...PCIe slot nameThe PCIe slot name without null terminator

GetEntityName - SubCommand 0x06

Gsys can get the "Entity ID:Entity Instance" to Entity name mapping from BMC using this command. When BMC receives this command, BMC can check the related JSON file and then send the name for that particular entity as this command response.

Request

Byte(s)ValueData
0x000x06Subcommand
0x01Entity IDEntity ID
0x02Entity InstanceEntity Instance

Response

Byte(s)ValueData
0x000x06Subcommand
0x01Entity name length (say N)Entity name length
0x02...0x02 + N - 1Entity nameEntity name without null terminator

GetMachineName - SubCommand 0x07

The BMC parses /etc/os-release for the OPENBMC_TARGET_MACHINE field and returns its value.

Request

Byte(s)ValueData
0x000x06Subcommand

Response

Byte(s)ValueData
0x000x06Subcommand
0x01Model name length (say N)Model name length
0x02...0x02 + N - 1Model nameModel name without null terminator

HardResetOnShutdown - SubCommand 0x08

Tells the BMC to powercycle the next time the host shuts down.

Request

Byte(s)ValueData
0x000x08Subcommand

Response

Byte(s)ValueData
0x000x08Subcommand

GetFlashSize - SubCommand 0x09

Request the physical size of the BMC flash.

Request

Byte(s)ValueData
0x000x09Subcommand

Response

Byte(s)ValueData
0x000x09Subcommand
0x01...0x04Flash sizeFlash size

HostPowerOff - SubCommand 0x0A

Sys command needs to be able to let the BMC knows host attempt S5 shutdown, it need power-off the Host gracefully and disable the watchdog with given time delay.

Request

Byte(s)ValueData
0x000x0ASubcommand
0x01..0x04Seconds to delay (uint32)

Response

Byte(s)ValueData
0x000x0ASubcommand

AccelOobDeviceCount - SubCommand 0x0B

Query the number of available devices from the google-accel-oob service.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

Request

Byte(s)ValueData
0x000x0BSubcommand

Response

Byte(s)ValueData
0x000x0BSubcommand
0x01..0x04Number of devices available

AccelOobDeviceName - SubCommand 0x0C

Query the name of a single device from the google-accel-oob service.

This name is used as the identifier for the AccelOobRead and AccelOobWrite commands.

Index values start at zero and go up to (but don't include) the device count.

The name of the device is exactly as it appears in DBus, except for the common "/com/google/customAccel/" prefix. This prefix is removed to reduce the size of the IPMI packet.

DBus requires all element names to be non-empty strings of ASCII characters "[A-Z][a-z][0-9]_", separated by ASCII '/'. Therefore, all device names will be valid ASCII strings (1 byte/character).

For convenience, the name string is followed by a single 0x00 (NULL terminator) byte which is not included as part of the length.

The length field (byte 5) is the number of bytes in the name string (not including the trailing NULL terminator byte).

The maximum length for any name is 43 bytes (not including the trailing NULL).

If a name is longer than 43 bytes, IPMI_CC_REQ_DATA_TRUNCATED is returned. These names will not be usable in the rest of the API. Changing the name requires code changes to the managed_acceld service binary.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

If a name does not begin with the expected "/com/google/customAccel/" prefix, IPMI_CC_INVALID is returned. This indicates a change in the DBus API for the google-accel-oob service that requires a matching code change in the handler.

Request

Byte(s)ValueData
0x000x0CSubcommand
0x05Length of the name
0x06..nName of the device

Response

Byte(s)ValueData
0x000x0CSubcommand
0x01..0x04Index of the device
0x05Length of the name
0x06..nName of the device

AccelOobRead - SubCommand 0x0D

Read a PCIe CSR from a device.

Length is the length of the name, in bytes.

The device name gets prepended with "/com/google/customAccel/" and sent to DBus. This string must NOT have a trailing NULL terminator.

The token is an arbitrary byte that gets echoed back in the reply; it is not interpreted by the service at all. This is used to disambiguate identical requests so clients can check for lost transactions.

Address is the 64b PCIe address to read from.

Number of bytes is the size of the read, in bytes (max 8). The value is subject to hardware limitations (both PCIe and ASIC), so it will generally be 1, 2, 4, or 8.

The register data is always returned in 8 bytes (uint64) in little Endian order. If fewer than than 8 bytes are read, the MSBs are padded with 0s.

On success, the response ends with the data read as a single uint64.

If the number of bytes requested would not fit in a single IPMI payload, IPMI_CC_REQUESTED_TOO_MANY_BYTES is returned.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

Request

Byte(s)ValueData
0x000x0DSubcommand
0x01Number of bytes in the device name
0x02..nName of the device (from AccelOobDeviceName)
n+1Token
n+2..n+10Address
n+11Number of bytes

Response

Byte(s)ValueData
0x000x0DSubcommand
0x01Number of bytes in the device name
0x02..nName of the device (no trailing NULL)
n+1Token
n+2..n+10Address
n+11Number of bytes
n+12..n+20Data

AccelOobWrite - SubCommand 0x0E

Write a PCIe CSR from a device.

All parameters are identical to AccelOobRead (above). The only difference is the register data becomes an input parameter (in the Request) instead of an output value (in the Response).

As with read, the register data must be 8 bytes (uint64) in little Endian order. If fewer than 8 bytes will be written, only the LSBs will be read and the the MSBs will be ignored.

All fields returned in the Response are simply a copy of the Request.

On success, IPMI_CC_OK is returned.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

Request

Byte(s)ValueData
0x000x0DSubcommand
0x01Number of bytes in the device name
0x02..nName of the device (from AccelOobDeviceName)
n+1Token
n+2..n+10Address
n+11Number of bytes
n+12..n+20Data

Response

Byte(s)ValueData
0x000x0DSubcommand
0x01Number of bytes in the device name
0x02..nName of the device (no trailing NULL)
n+1Token
n+2..n+10Address
n+11Number of bytes
n+12..n+20Data

PCIe Bifurcation - SubCommand 0x0F

Sys command to return the highest level of bifurcation for the target PCIe Slot.

Request

Byte(s)ValueData
0x000x0FSubcommand
0x01PE slot numberIndex of the PE slot

Response

Byte(s)ValueData
0x000x0FSubcommand
0x01Config length (N)Number of bytes needed for the bifurcation config
0x02..0x02 + N - 1Lanes per deviceEach byte represents the number of lanes bonded together for each endpoint device

GetBmcMode - SubCommand 0x10

Request the operational mode of the BMC.

Request

Byte(s)ValueData
0x000x10Subcommand

Response

Byte(s)ValueData
0x000x10Subcommand
0x01Current BMC MODE0 -> Non Bare Metal Mode1 -> Bare Metal Mode2 -> Bare Metal Cleaning Mode

LinuxBootDone - SubCommand 0x11

Notify the BMC that LinuxBoot is finished and will kexec into the OS momentarily.

If in bare metal mode, the BMC will disable IPMI upon receiving this command, to protect against a malicious OS. For this reason, the BMC may not respond to this command.

If not in bare metal mode, this command has no effect.

Request

Byte(s)ValueData
0x000x11Subcommand

Response (if applicable)

Byte(s)ValueData
0x000x11Subcommand

SysGetAccelVrSettings - SubCommand 0x15

Get the accel's VR setting value for the given chip and settings ID

Currently 3 settings are supported. [0] IdleMode [1] PowerBrake [2] Loadline

On success, the response contains 2 bytes containing the setting value.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

Request

Byte(s)ValueData
0x000x15Subcommand
0x01Chip ID
0x02SettingsID

Response (if applicable)

Byte(s)ValueData
0x000x15Subcommand
0x01..0x020x15Settings Value

SysSetAccelVrSettings - SubCommand 0x16

Update the VR settings of a given accel device for a specific settings id.

Currently 3 settings are supported. [0] IdleMode [1] PowerBrake [2] Loadline

The settings value parameter is a 2 byte value and is expected in little endian format

On success, IPMI_CC_OK is returned.

If not enough data is proveded, IPMI_CC_REQ_DATA_LEN_INVALID is returned.

Request

Byte(s)ValueData
0x000x16Subcommand
0x01Chip ID
0x02Settings ID
0x03..0x040x16Settings Value

Response (if applicable)

Byte(s)ValueData
0x000x16Subcommand

GetBMInstanceProperty - SubCommand 0x17

Read a BM instance property, specify the property to read following the enum.

The response contains the length of the property string to read in bytes followed by the property string which does NOT have a NULL terminator.

Request

Byte(s)ValueData
0x000x17Subcommand
0x010x0: AssetTag
0x1: BoardSerialNumber
0x2: Family
0x3: ProductName
0x4: SKU
0x5: SystemSerialNumber
0x6: UUID

Response

Byte(s)ValueData
0x000x17Subcommand
0x01String Length (N)Number of bytes to read for property string - Size limited to 64
0x2..0x02 + N - 1String of propertyString, not null-terminated

ReadBiosSetting - SubCommand 0x18

Read a BIOS setting, set at /run/oem_bios_setting.

The response contains the length of the BIOS setting followed by the BIOS setting bytes read.

Request

Byte(s)ValueData
0x000x18Subcommand

Response

Byte(s)ValueData
0x000x18Subcommand
0x01Length (N)Number of payload bytes - Size limited to 64
0x2..0x02 + N - 1Payload bytesPayload bytes

WriteBiosSetting - SubCommand 0x19

Write a BIOS setting, set at /run/oem_bios_setting.

The response contains the length of the BIOS setting followed by the BIOS setting bytes read.

Request

Byte(s)ValueData
0x000x19Subcommand
0x01Length (N)Number of payload bytes - Size limited to 64
0x2..0x02 + N - 1Payload bytesPayload bytes

Response

Byte(s)ValueData
0x000x19Subcommand
0x01Length (N)Number of bytes successfully written