Implement set chassis capabilities IPMI commands

Create a new dbus interface for the chassis capabilities, and
store the chassis capabilities using phosphor settings manager

Tested By:
ipmitool -H $BMCIP  -U root -P 0penBmc -I lanplus raw 0 5 0 2 4 6 8 0xa
ipmitool -H $BMCIP  -U root -P 0penBmc -I lanplus raw 0 0

Change-Id: Id01555358adfec3814fd03b3e6d422916c861039
Signed-off-by: Yong Li <yong.b.li@linux.intel.com>
diff --git a/chassishandler.cpp b/chassishandler.cpp
index a58eee0..d4ce0e4 100644
--- a/chassishandler.cpp
+++ b/chassishandler.cpp
@@ -103,6 +103,16 @@
     "xyz.openbmc_project.State.PowerOnHours";
 static constexpr auto pOHCounterProperty = "POHCounter";
 static constexpr auto match = "chassis0";
+const static constexpr char chassisCapIntf[] =
+    "xyz.openbmc_project.Control.ChassisCapabilities";
+const static constexpr char chassisCapFlagsProp[] = "CapabilitiesFlags";
+const static constexpr char chassisFRUDevAddrProp[] = "FRUDeviceAddress";
+const static constexpr char chassisSDRDevAddrProp[] = "SDRDeviceAddress";
+const static constexpr char chassisSELDevAddrProp[] = "SELDeviceAddress";
+const static constexpr char chassisSMDevAddrProp[] = "SMDeviceAddress";
+const static constexpr char chassisBridgeDevAddrProp[] = "BridgeDeviceAddress";
+static constexpr uint8_t chassisCapFlagMask = 0x0f;
+static constexpr uint8_t chassisCapAddrMask = 0xfe;
 
 typedef struct
 {
@@ -666,41 +676,184 @@
 
     ipmi_chassis_cap_t chassis_cap{};
 
+    if (*data_len != 0)
+    {
+        return IPMI_CC_REQ_DATA_LEN_INVALID;
+    }
+
     *data_len = sizeof(ipmi_chassis_cap_t);
 
-    // TODO: need future work. Get those flag from MRW.
+    try
+    {
+        sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()};
 
-    // capabilities flags
-    // [7..4] - reserved
-    // [3] – 1b = provides power interlock  (IPM 1.5)
-    // [2] – 1b = provides Diagnostic Interrupt (FP NMI)
-    // [1] – 1b = provides “Front Panel Lockout” (indicates that the chassis has
-    // capabilities
-    //            to lock out external power control and reset button or front
-    //            panel interfaces and/or detect tampering with those
-    //            interfaces).
-    // [0] -1b = Chassis provides intrusion (physical security) sensor.
-    // set to default value 0x0.
-    chassis_cap.cap_flags = 0x0;
+        ipmi::DbusObjectInfo chassisCapObject =
+            ipmi::getDbusObject(bus, chassisCapIntf);
 
-    // Since we do not have a separate SDR Device/SEL Device/ FRU repository.
-    // The 20h was given as those 5 device addresses.
-    // Chassis FRU info Device Address
-    chassis_cap.fru_info_dev_addr = 0x20;
+        // capabilities flags
+        // [7..4] - reserved
+        // [3] – 1b = provides power interlock  (IPM 1.5)
+        // [2] – 1b = provides Diagnostic Interrupt (FP NMI)
+        // [1] – 1b = provides “Front Panel Lockout” (indicates that the chassis
+        // has capabilities
+        //            to lock out external power control and reset button or
+        //            front panel interfaces and/or detect tampering with those
+        //            interfaces).
+        // [0] -1b = Chassis provides intrusion (physical security) sensor.
+        // set to default value 0x0.
+        ipmi::Value variant = ipmi::getDbusProperty(
+            bus, chassisCapObject.second, chassisCapObject.first,
+            chassisCapIntf, chassisCapFlagsProp);
+        chassis_cap.cap_flags = variant_ns::get<uint8_t>(variant);
 
-    // Chassis SDR Device Address
-    chassis_cap.sdr_dev_addr = 0x20;
+        variant = ipmi::getDbusProperty(bus, chassisCapObject.second,
+                                        chassisCapObject.first, chassisCapIntf,
+                                        chassisFRUDevAddrProp);
+        // Chassis FRU info Device Address.
+        chassis_cap.fru_info_dev_addr = variant_ns::get<uint8_t>(variant);
 
-    // Chassis SEL Device Address
-    chassis_cap.sel_dev_addr = 0x20;
+        variant = ipmi::getDbusProperty(bus, chassisCapObject.second,
+                                        chassisCapObject.first, chassisCapIntf,
+                                        chassisSDRDevAddrProp);
+        // Chassis SDR Device Address.
+        chassis_cap.sdr_dev_addr = variant_ns::get<uint8_t>(variant);
 
-    // Chassis System Management Device Address
-    chassis_cap.system_management_dev_addr = 0x20;
+        variant = ipmi::getDbusProperty(bus, chassisCapObject.second,
+                                        chassisCapObject.first, chassisCapIntf,
+                                        chassisSELDevAddrProp);
+        // Chassis SEL Device Address.
+        chassis_cap.sel_dev_addr = variant_ns::get<uint8_t>(variant);
 
-    // Chassis Bridge Device Address.
-    chassis_cap.bridge_dev_addr = 0x20;
+        variant = ipmi::getDbusProperty(bus, chassisCapObject.second,
+                                        chassisCapObject.first, chassisCapIntf,
+                                        chassisSMDevAddrProp);
+        // Chassis System Management Device Address.
+        chassis_cap.system_management_dev_addr =
+            variant_ns::get<uint8_t>(variant);
 
-    std::memcpy(response, &chassis_cap, *data_len);
+        variant = ipmi::getDbusProperty(bus, chassisCapObject.second,
+                                        chassisCapObject.first, chassisCapIntf,
+                                        chassisBridgeDevAddrProp);
+        // Chassis Bridge Device Address.
+        chassis_cap.bridge_dev_addr = variant_ns::get<uint8_t>(variant);
+        uint8_t* respP = reinterpret_cast<uint8_t*>(response);
+        uint8_t* chassisP = reinterpret_cast<uint8_t*>(&chassis_cap);
+        std::copy(chassisP, chassisP + *data_len, respP);
+    }
+    catch (std::exception& e)
+    {
+        log<level::ERR>(e.what());
+        rc = IPMI_CC_UNSPECIFIED_ERROR;
+        *data_len = 0;
+        return rc;
+    }
+
+    return rc;
+}
+
+ipmi_ret_t ipmi_set_chassis_cap(ipmi_netfn_t netfn, ipmi_cmd_t cmd,
+                                ipmi_request_t request,
+                                ipmi_response_t response,
+                                ipmi_data_len_t data_len,
+                                ipmi_context_t context)
+{
+    ipmi_ret_t rc = IPMI_CC_OK;
+
+    if (*data_len != sizeof(ipmi_chassis_cap_t))
+    {
+        log<level::ERR>("Unsupported request length",
+                        entry("LEN=0x%x", *data_len));
+        *data_len = 0;
+        return IPMI_CC_REQ_DATA_LEN_INVALID;
+    }
+
+    ipmi_chassis_cap_t* chassisCap = static_cast<ipmi_chassis_cap_t*>(request);
+
+    *data_len = 0;
+
+    // check input data
+    if (0 != (chassisCap->cap_flags & ~chassisCapFlagMask))
+    {
+        log<level::ERR>("Unsupported request parameter(CAP Flags)",
+                        entry("REQ=0x%x", chassisCap->cap_flags));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    if (0 != (chassisCap->fru_info_dev_addr & ~chassisCapAddrMask))
+    {
+        log<level::ERR>("Unsupported request parameter(FRU Addr)",
+                        entry("REQ=0x%x", chassisCap->fru_info_dev_addr));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    if (0 != (chassisCap->sdr_dev_addr & ~chassisCapAddrMask))
+    {
+        log<level::ERR>("Unsupported request parameter(SDR Addr)",
+                        entry("REQ=0x%x", chassisCap->sdr_dev_addr));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    if (0 != (chassisCap->sel_dev_addr & ~chassisCapAddrMask))
+    {
+        log<level::ERR>("Unsupported request parameter(SEL Addr)",
+                        entry("REQ=0x%x", chassisCap->sel_dev_addr));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    if (0 != (chassisCap->system_management_dev_addr & ~chassisCapAddrMask))
+    {
+        log<level::ERR>(
+            "Unsupported request parameter(SM Addr)",
+            entry("REQ=0x%x", chassisCap->system_management_dev_addr));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    if (0 != (chassisCap->bridge_dev_addr & ~chassisCapAddrMask))
+    {
+        log<level::ERR>("Unsupported request parameter(Bridge Addr)",
+                        entry("REQ=0x%x", chassisCap->bridge_dev_addr));
+        return IPMI_CC_INVALID_FIELD_REQUEST;
+    }
+
+    try
+    {
+        sdbusplus::bus::bus bus(ipmid_get_sd_bus_connection());
+        ipmi::DbusObjectInfo chassisCapObject =
+            ipmi::getDbusObject(bus, chassisCapIntf);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisCapFlagsProp, chassisCap->cap_flags);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisFRUDevAddrProp,
+                              chassisCap->fru_info_dev_addr);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisSDRDevAddrProp, chassisCap->sdr_dev_addr);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisSELDevAddrProp, chassisCap->sel_dev_addr);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisSMDevAddrProp,
+                              chassisCap->system_management_dev_addr);
+
+        ipmi::setDbusProperty(bus, chassisCapObject.second,
+                              chassisCapObject.first, chassisCapIntf,
+                              chassisBridgeDevAddrProp,
+                              chassisCap->bridge_dev_addr);
+    }
+    catch (std::exception& e)
+    {
+        log<level::ERR>(e.what());
+        rc = IPMI_CC_UNSPECIFIED_ERROR;
+        return rc;
+    }
 
     return rc;
 }
@@ -1714,6 +1867,10 @@
     ipmi_register_callback(NETFUN_CHASSIS, IPMI_CMD_GET_CHASSIS_CAP, NULL,
                            ipmi_get_chassis_cap, PRIVILEGE_USER);
 
+    // Set Chassis Capabilities
+    ipmi_register_callback(NETFUN_CHASSIS, IPMI_CMD_SET_CHASSIS_CAP, NULL,
+                           ipmi_set_chassis_cap, PRIVILEGE_USER);
+
     // <Get System Boot Options>
     ipmi_register_callback(NETFUN_CHASSIS, IPMI_CMD_GET_SYS_BOOT_OPTIONS, NULL,
                            ipmi_chassis_get_sys_boot_options,
diff --git a/chassishandler.hpp b/chassishandler.hpp
index 6535c0e..49b5ef8 100644
--- a/chassishandler.hpp
+++ b/chassishandler.hpp
@@ -13,6 +13,7 @@
     // Chassis Control
     IPMI_CMD_CHASSIS_CONTROL = 0x02,
     IPMI_CMD_CHASSIS_IDENTIFY = 0x04,
+    IPMI_CMD_SET_CHASSIS_CAP = 0x05,
     // Set Power Restore Policy
     IPMI_CMD_SET_RESTORE_POLICY = 0x06,
     // Get capability bits
diff --git a/host-ipmid-whitelist.conf b/host-ipmid-whitelist.conf
index 9dff10a..e5cd0b5 100644
--- a/host-ipmid-whitelist.conf
+++ b/host-ipmid-whitelist.conf
@@ -2,6 +2,7 @@
 0x00:0x00    //<Chassis>:<Chassis Capabiliti>
 0x00:0x01    //<Chassis>:<Get Chassis Status>
 0x00:0x02    //<Chassis>:<Chassis Control>
+0x00:0x05    //<Chassis>:<Set Chassis Capabilities>
 0x00:0x06    //<Chassis>:<Set Power Restore Policy>
 0x00:0x08    //<Chassis>:<Set System Boot Options>
 0x00:0x09    //<Chassis>:<Get System Boot Options>