xref: /openbmc/bmcweb/features/redfish/lib/pcie.hpp (revision bad2c4a9a1095e5e41959bfc5bc56102bbd2a79a)
1f5c9f8bdSJason M. Bills /*
2f5c9f8bdSJason M. Bills // Copyright (c) 2018 Intel Corporation
3f5c9f8bdSJason M. Bills //
4f5c9f8bdSJason M. Bills // Licensed under the Apache License, Version 2.0 (the "License");
5f5c9f8bdSJason M. Bills // you may not use this file except in compliance with the License.
6f5c9f8bdSJason M. Bills // You may obtain a copy of the License at
7f5c9f8bdSJason M. Bills //
8f5c9f8bdSJason M. Bills //      http://www.apache.org/licenses/LICENSE-2.0
9f5c9f8bdSJason M. Bills //
10f5c9f8bdSJason M. Bills // Unless required by applicable law or agreed to in writing, software
11f5c9f8bdSJason M. Bills // distributed under the License is distributed on an "AS IS" BASIS,
12f5c9f8bdSJason M. Bills // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f5c9f8bdSJason M. Bills // See the License for the specific language governing permissions and
14f5c9f8bdSJason M. Bills // limitations under the License.
15f5c9f8bdSJason M. Bills */
16f5c9f8bdSJason M. Bills 
17f5c9f8bdSJason M. Bills #pragma once
18f5c9f8bdSJason M. Bills 
193ccb3adbSEd Tanous #include "app.hpp"
207a1dbc48SGeorge Liu #include "dbus_utility.hpp"
210ec8b83dSEd Tanous #include "generated/enums/pcie_device.hpp"
223ccb3adbSEd Tanous #include "query.hpp"
233ccb3adbSEd Tanous #include "registries/privilege_registry.hpp"
24b38fa2abSLakshmi Yadlapati #include "utils/collection.hpp"
253ccb3adbSEd Tanous #include "utils/dbus_utils.hpp"
260ec8b83dSEd Tanous 
27f5c9f8bdSJason M. Bills #include <boost/system/linux_error.hpp>
28d1bde9e5SKrzysztof Grobelny #include <sdbusplus/asio/property.hpp>
29d1bde9e5SKrzysztof Grobelny #include <sdbusplus/unpack_properties.hpp>
30f5c9f8bdSJason M. Bills 
31f5c9f8bdSJason M. Bills namespace redfish
32f5c9f8bdSJason M. Bills {
33f5c9f8bdSJason M. Bills 
3489492a15SPatrick Williams static constexpr const char* inventoryPath = "/xyz/openbmc_project/inventory";
3594c3a10bSLakshmi Yadlapati static constexpr std::array<std::string_view, 1> pcieDeviceInterface = {
3694c3a10bSLakshmi Yadlapati     "xyz.openbmc_project.Inventory.Item.PCIeDevice"};
37f5c9f8bdSJason M. Bills 
38543f9a75SLakshmi Yadlapati static inline void handlePCIeDevicePath(
39543f9a75SLakshmi Yadlapati     const std::string& pcieDeviceId,
40543f9a75SLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
41543f9a75SLakshmi Yadlapati     const dbus::utility::MapperGetSubTreePathsResponse& pcieDevicePaths,
42543f9a75SLakshmi Yadlapati     const std::function<void(const std::string& pcieDevicePath,
43543f9a75SLakshmi Yadlapati                              const std::string& service)>& callback)
44543f9a75SLakshmi Yadlapati 
45543f9a75SLakshmi Yadlapati {
46543f9a75SLakshmi Yadlapati     for (const std::string& pcieDevicePath : pcieDevicePaths)
47543f9a75SLakshmi Yadlapati     {
48543f9a75SLakshmi Yadlapati         std::string pciecDeviceName =
49543f9a75SLakshmi Yadlapati             sdbusplus::message::object_path(pcieDevicePath).filename();
50543f9a75SLakshmi Yadlapati         if (pciecDeviceName.empty() || pciecDeviceName != pcieDeviceId)
51543f9a75SLakshmi Yadlapati         {
52543f9a75SLakshmi Yadlapati             continue;
53543f9a75SLakshmi Yadlapati         }
54543f9a75SLakshmi Yadlapati 
55543f9a75SLakshmi Yadlapati         dbus::utility::getDbusObject(
56543f9a75SLakshmi Yadlapati             pcieDevicePath, {},
57543f9a75SLakshmi Yadlapati             [pcieDevicePath, aResp,
58543f9a75SLakshmi Yadlapati              callback](const boost::system::error_code& ec,
59543f9a75SLakshmi Yadlapati                        const dbus::utility::MapperGetObject& object) {
60543f9a75SLakshmi Yadlapati             if (ec || object.empty())
61543f9a75SLakshmi Yadlapati             {
62543f9a75SLakshmi Yadlapati                 BMCWEB_LOG_ERROR << "DBUS response error " << ec;
63543f9a75SLakshmi Yadlapati                 messages::internalError(aResp->res);
64543f9a75SLakshmi Yadlapati                 return;
65543f9a75SLakshmi Yadlapati             }
66543f9a75SLakshmi Yadlapati             callback(pcieDevicePath, object.begin()->first);
67543f9a75SLakshmi Yadlapati             });
68543f9a75SLakshmi Yadlapati         return;
69543f9a75SLakshmi Yadlapati     }
70543f9a75SLakshmi Yadlapati 
71543f9a75SLakshmi Yadlapati     BMCWEB_LOG_WARNING << "PCIe Device not found";
72543f9a75SLakshmi Yadlapati     messages::resourceNotFound(aResp->res, "PCIeDevice", pcieDeviceId);
73543f9a75SLakshmi Yadlapati }
74543f9a75SLakshmi Yadlapati 
75543f9a75SLakshmi Yadlapati static inline void getValidPCIeDevicePath(
76543f9a75SLakshmi Yadlapati     const std::string& pcieDeviceId,
77543f9a75SLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
78543f9a75SLakshmi Yadlapati     const std::function<void(const std::string& pcieDevicePath,
79543f9a75SLakshmi Yadlapati                              const std::string& service)>& callback)
80543f9a75SLakshmi Yadlapati {
81543f9a75SLakshmi Yadlapati     dbus::utility::getSubTreePaths(
8294c3a10bSLakshmi Yadlapati         inventoryPath, 0, pcieDeviceInterface,
83543f9a75SLakshmi Yadlapati         [pcieDeviceId, aResp,
84543f9a75SLakshmi Yadlapati          callback](const boost::system::error_code& ec,
85543f9a75SLakshmi Yadlapati                    const dbus::utility::MapperGetSubTreePathsResponse&
86543f9a75SLakshmi Yadlapati                        pcieDevicePaths) {
87543f9a75SLakshmi Yadlapati         if (ec)
88543f9a75SLakshmi Yadlapati         {
89543f9a75SLakshmi Yadlapati             BMCWEB_LOG_ERROR << "D-Bus response error on GetSubTree " << ec;
90543f9a75SLakshmi Yadlapati             messages::internalError(aResp->res);
91543f9a75SLakshmi Yadlapati             return;
92543f9a75SLakshmi Yadlapati         }
93543f9a75SLakshmi Yadlapati         handlePCIeDevicePath(pcieDeviceId, aResp, pcieDevicePaths, callback);
94543f9a75SLakshmi Yadlapati         return;
95543f9a75SLakshmi Yadlapati         });
96543f9a75SLakshmi Yadlapati }
97543f9a75SLakshmi Yadlapati 
98b5a76932SEd Tanous static inline void
998d1b46d7Szhanghch05     getPCIeDeviceList(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
100adbe192aSJason M. Bills                       const std::string& name)
101f5c9f8bdSJason M. Bills {
1027a1dbc48SGeorge Liu     dbus::utility::getSubTreePaths(
10338e3d67dSLakshmi Yadlapati         inventoryPath, 0, pcieDeviceInterface,
1047a1dbc48SGeorge Liu         [asyncResp, name](const boost::system::error_code& ec,
105b9d36b47SEd Tanous                           const dbus::utility::MapperGetSubTreePathsResponse&
106b9d36b47SEd Tanous                               pcieDevicePaths) {
107f5c9f8bdSJason M. Bills         if (ec)
108f5c9f8bdSJason M. Bills         {
109a2730f01SAndrew Geissler             BMCWEB_LOG_DEBUG << "no PCIe device paths found ec: "
110f5c9f8bdSJason M. Bills                              << ec.message();
111a2730f01SAndrew Geissler             // Not an error, system just doesn't have PCIe info
112f5c9f8bdSJason M. Bills             return;
113f5c9f8bdSJason M. Bills         }
114adbe192aSJason M. Bills         nlohmann::json& pcieDeviceList = asyncResp->res.jsonValue[name];
115f5c9f8bdSJason M. Bills         pcieDeviceList = nlohmann::json::array();
116f5c9f8bdSJason M. Bills         for (const std::string& pcieDevicePath : pcieDevicePaths)
117f5c9f8bdSJason M. Bills         {
1183174e4dfSEd Tanous             size_t devStart = pcieDevicePath.rfind('/');
119f5c9f8bdSJason M. Bills             if (devStart == std::string::npos)
120f5c9f8bdSJason M. Bills             {
121f5c9f8bdSJason M. Bills                 continue;
122f5c9f8bdSJason M. Bills             }
123f5c9f8bdSJason M. Bills 
124f5c9f8bdSJason M. Bills             std::string devName = pcieDevicePath.substr(devStart + 1);
125f5c9f8bdSJason M. Bills             if (devName.empty())
126f5c9f8bdSJason M. Bills             {
127f5c9f8bdSJason M. Bills                 continue;
128f5c9f8bdSJason M. Bills             }
1291476687dSEd Tanous             nlohmann::json::object_t pcieDevice;
130eddfc437SWilly Tu             pcieDevice["@odata.id"] = crow::utility::urlFromPieces(
131eddfc437SWilly Tu                 "redfish", "v1", "Systems", "system", "PCIeDevices", devName);
132b2ba3072SPatrick Williams             pcieDeviceList.emplace_back(std::move(pcieDevice));
133f5c9f8bdSJason M. Bills         }
134002d39b4SEd Tanous         asyncResp->res.jsonValue[name + "@odata.count"] = pcieDeviceList.size();
1357a1dbc48SGeorge Liu         });
136f5c9f8bdSJason M. Bills }
137f5c9f8bdSJason M. Bills 
138b38fa2abSLakshmi Yadlapati static inline void handlePCIeDeviceCollectionGet(
139b38fa2abSLakshmi Yadlapati     crow::App& app, const crow::Request& req,
140b38fa2abSLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
141b38fa2abSLakshmi Yadlapati     const std::string& systemName)
142b38fa2abSLakshmi Yadlapati {
143b38fa2abSLakshmi Yadlapati     if (!redfish::setUpRedfishRoute(app, req, aResp))
144b38fa2abSLakshmi Yadlapati     {
145b38fa2abSLakshmi Yadlapati         return;
146b38fa2abSLakshmi Yadlapati     }
147b38fa2abSLakshmi Yadlapati     if (systemName != "system")
148b38fa2abSLakshmi Yadlapati     {
149b38fa2abSLakshmi Yadlapati         messages::resourceNotFound(aResp->res, "ComputerSystem", systemName);
150b38fa2abSLakshmi Yadlapati         return;
151b38fa2abSLakshmi Yadlapati     }
152543f9a75SLakshmi Yadlapati 
153b38fa2abSLakshmi Yadlapati     aResp->res.addHeader(boost::beast::http::field::link,
154b38fa2abSLakshmi Yadlapati                          "</redfish/v1/JsonSchemas/PCIeDeviceCollection/"
155b38fa2abSLakshmi Yadlapati                          "PCIeDeviceCollection.json>; rel=describedby");
156b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["@odata.type"] =
157b38fa2abSLakshmi Yadlapati         "#PCIeDeviceCollection.PCIeDeviceCollection";
158b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["@odata.id"] =
159b38fa2abSLakshmi Yadlapati         "/redfish/v1/Systems/system/PCIeDevices";
160b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["Name"] = "PCIe Device Collection";
161b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["Description"] = "Collection of PCIe Devices";
162b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["Members"] = nlohmann::json::array();
163b38fa2abSLakshmi Yadlapati     aResp->res.jsonValue["Members@odata.count"] = 0;
164b38fa2abSLakshmi Yadlapati 
165b38fa2abSLakshmi Yadlapati     collection_util::getCollectionMembers(
166b38fa2abSLakshmi Yadlapati         aResp, boost::urls::url("/redfish/v1/Systems/system/PCIeDevices"),
16794c3a10bSLakshmi Yadlapati         pcieDeviceInterface);
168b38fa2abSLakshmi Yadlapati }
169b38fa2abSLakshmi Yadlapati 
1707e860f15SJohn Edward Broadbent inline void requestRoutesSystemPCIeDeviceCollection(App& app)
171adbe192aSJason M. Bills {
172adbe192aSJason M. Bills     /**
173adbe192aSJason M. Bills      * Functions triggers appropriate requests on DBus
174adbe192aSJason M. Bills      */
17522d268cbSEd Tanous     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/PCIeDevices/")
176ed398213SEd Tanous         .privileges(redfish::privileges::getPCIeDeviceCollection)
1777e860f15SJohn Edward Broadbent         .methods(boost::beast::http::verb::get)(
178b38fa2abSLakshmi Yadlapati             std::bind_front(handlePCIeDeviceCollectionGet, std::ref(app)));
179f5c9f8bdSJason M. Bills }
180f5c9f8bdSJason M. Bills 
1810ec8b83dSEd Tanous inline std::optional<pcie_device::PCIeTypes>
18262cd45afSSpencer Ku     redfishPcieGenerationFromDbus(const std::string& generationInUse)
18362cd45afSSpencer Ku {
18462cd45afSSpencer Ku     if (generationInUse ==
18562cd45afSSpencer Ku         "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Gen1")
18662cd45afSSpencer Ku     {
1870ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Gen1;
18862cd45afSSpencer Ku     }
18962cd45afSSpencer Ku     if (generationInUse ==
19062cd45afSSpencer Ku         "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Gen2")
19162cd45afSSpencer Ku     {
1920ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Gen2;
19362cd45afSSpencer Ku     }
19462cd45afSSpencer Ku     if (generationInUse ==
19562cd45afSSpencer Ku         "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Gen3")
19662cd45afSSpencer Ku     {
1970ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Gen3;
19862cd45afSSpencer Ku     }
19962cd45afSSpencer Ku     if (generationInUse ==
20062cd45afSSpencer Ku         "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Gen4")
20162cd45afSSpencer Ku     {
2020ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Gen4;
20362cd45afSSpencer Ku     }
20462cd45afSSpencer Ku     if (generationInUse ==
20562cd45afSSpencer Ku         "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Gen5")
20662cd45afSSpencer Ku     {
2070ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Gen5;
20862cd45afSSpencer Ku     }
209e825cbc8SEd Tanous     if (generationInUse.empty() ||
210e825cbc8SEd Tanous         generationInUse ==
21162cd45afSSpencer Ku             "xyz.openbmc_project.Inventory.Item.PCIeSlot.Generations.Unknown")
21262cd45afSSpencer Ku     {
2130ec8b83dSEd Tanous         return pcie_device::PCIeTypes::Invalid;
21462cd45afSSpencer Ku     }
21562cd45afSSpencer Ku 
21662cd45afSSpencer Ku     // The value is not unknown or Gen1-5, need return an internal error.
21762cd45afSSpencer Ku     return std::nullopt;
21862cd45afSSpencer Ku }
21962cd45afSSpencer Ku 
220913e7732SSunnySrivastava1984 inline void getPCIeDeviceAsset(const std::shared_ptr<bmcweb::AsyncResp>& aResp,
221913e7732SSunnySrivastava1984                                const std::string& pcieDevicePath,
222913e7732SSunnySrivastava1984                                const std::string& service)
223913e7732SSunnySrivastava1984 {
224913e7732SSunnySrivastava1984     sdbusplus::asio::getAllProperties(
225913e7732SSunnySrivastava1984         *crow::connections::systemBus, service, pcieDevicePath,
226913e7732SSunnySrivastava1984         "xyz.openbmc_project.Inventory.Decorator.Asset",
227913e7732SSunnySrivastava1984         [pcieDevicePath,
228913e7732SSunnySrivastava1984          aResp{aResp}](const boost::system::error_code& ec,
229913e7732SSunnySrivastava1984                        const dbus::utility::DBusPropertiesMap& assetList) {
230913e7732SSunnySrivastava1984         if (ec)
231913e7732SSunnySrivastava1984         {
232913e7732SSunnySrivastava1984             if (ec.value() != EBADR)
233913e7732SSunnySrivastava1984             {
234913e7732SSunnySrivastava1984                 BMCWEB_LOG_ERROR << "DBUS response error for Properties"
235913e7732SSunnySrivastava1984                                  << ec.value();
236913e7732SSunnySrivastava1984                 messages::internalError(aResp->res);
237913e7732SSunnySrivastava1984             }
238913e7732SSunnySrivastava1984             return;
239913e7732SSunnySrivastava1984         }
240913e7732SSunnySrivastava1984 
241913e7732SSunnySrivastava1984         const std::string* manufacturer = nullptr;
242913e7732SSunnySrivastava1984         const std::string* model = nullptr;
243913e7732SSunnySrivastava1984         const std::string* partNumber = nullptr;
244913e7732SSunnySrivastava1984         const std::string* serialNumber = nullptr;
245913e7732SSunnySrivastava1984         const std::string* sparePartNumber = nullptr;
246913e7732SSunnySrivastava1984 
247913e7732SSunnySrivastava1984         const bool success = sdbusplus::unpackPropertiesNoThrow(
248913e7732SSunnySrivastava1984             dbus_utils::UnpackErrorPrinter(), assetList, "Manufacturer",
249913e7732SSunnySrivastava1984             manufacturer, "Model", model, "PartNumber", partNumber,
250913e7732SSunnySrivastava1984             "SerialNumber", serialNumber, "SparePartNumber", sparePartNumber);
251913e7732SSunnySrivastava1984 
252913e7732SSunnySrivastava1984         if (!success)
253913e7732SSunnySrivastava1984         {
254913e7732SSunnySrivastava1984             messages::internalError(aResp->res);
255913e7732SSunnySrivastava1984             return;
256913e7732SSunnySrivastava1984         }
257913e7732SSunnySrivastava1984 
258913e7732SSunnySrivastava1984         if (manufacturer != nullptr)
259913e7732SSunnySrivastava1984         {
260913e7732SSunnySrivastava1984             aResp->res.jsonValue["Manufacturer"] = *manufacturer;
261913e7732SSunnySrivastava1984         }
262913e7732SSunnySrivastava1984         if (model != nullptr)
263913e7732SSunnySrivastava1984         {
264913e7732SSunnySrivastava1984             aResp->res.jsonValue["Model"] = *model;
265913e7732SSunnySrivastava1984         }
266913e7732SSunnySrivastava1984 
267913e7732SSunnySrivastava1984         if (partNumber != nullptr)
268913e7732SSunnySrivastava1984         {
269913e7732SSunnySrivastava1984             aResp->res.jsonValue["PartNumber"] = *partNumber;
270913e7732SSunnySrivastava1984         }
271913e7732SSunnySrivastava1984 
272913e7732SSunnySrivastava1984         if (serialNumber != nullptr)
273913e7732SSunnySrivastava1984         {
274913e7732SSunnySrivastava1984             aResp->res.jsonValue["SerialNumber"] = *serialNumber;
275913e7732SSunnySrivastava1984         }
276913e7732SSunnySrivastava1984 
277913e7732SSunnySrivastava1984         if (sparePartNumber != nullptr && !sparePartNumber->empty())
278913e7732SSunnySrivastava1984         {
279913e7732SSunnySrivastava1984             aResp->res.jsonValue["SparePartNumber"] = *sparePartNumber;
280913e7732SSunnySrivastava1984         }
281913e7732SSunnySrivastava1984         });
282913e7732SSunnySrivastava1984 }
283913e7732SSunnySrivastava1984 
284543f9a75SLakshmi Yadlapati inline void addPCIeDeviceProperties(
28535ad613dSLakshmi Yadlapati     crow::Response& resp, const std::string& pcieDeviceId,
286543f9a75SLakshmi Yadlapati     const dbus::utility::DBusPropertiesMap& pcieDevProperties)
287f5c9f8bdSJason M. Bills {
288d1bde9e5SKrzysztof Grobelny     const std::string* deviceType = nullptr;
289d1bde9e5SKrzysztof Grobelny     const std::string* generationInUse = nullptr;
290543f9a75SLakshmi Yadlapati     const int64_t* lanesInUse = nullptr;
291d1bde9e5SKrzysztof Grobelny 
292d1bde9e5SKrzysztof Grobelny     const bool success = sdbusplus::unpackPropertiesNoThrow(
293543f9a75SLakshmi Yadlapati         dbus_utils::UnpackErrorPrinter(), pcieDevProperties, "DeviceType",
294543f9a75SLakshmi Yadlapati         deviceType, "GenerationInUse", generationInUse, "LanesInUse",
295*bad2c4a9SLakshmi Yadlapati         lanesInUse);
296d1bde9e5SKrzysztof Grobelny 
297d1bde9e5SKrzysztof Grobelny     if (!success)
298d1bde9e5SKrzysztof Grobelny     {
299543f9a75SLakshmi Yadlapati         messages::internalError(resp);
300d1bde9e5SKrzysztof Grobelny         return;
301d1bde9e5SKrzysztof Grobelny     }
302d1bde9e5SKrzysztof Grobelny 
303543f9a75SLakshmi Yadlapati     if (deviceType != nullptr && !deviceType->empty())
304703f6741SMyung Bae     {
305543f9a75SLakshmi Yadlapati         resp.jsonValue["PCIeInterface"]["DeviceType"] = *deviceType;
306703f6741SMyung Bae     }
307703f6741SMyung Bae 
308d1bde9e5SKrzysztof Grobelny     if (generationInUse != nullptr)
309d1bde9e5SKrzysztof Grobelny     {
3100ec8b83dSEd Tanous         std::optional<pcie_device::PCIeTypes> redfishGenerationInUse =
311d1bde9e5SKrzysztof Grobelny             redfishPcieGenerationFromDbus(*generationInUse);
312543f9a75SLakshmi Yadlapati 
313d1bde9e5SKrzysztof Grobelny         if (!redfishGenerationInUse)
314d1bde9e5SKrzysztof Grobelny         {
315543f9a75SLakshmi Yadlapati             messages::internalError(resp);
316d1bde9e5SKrzysztof Grobelny             return;
317d1bde9e5SKrzysztof Grobelny         }
3180ec8b83dSEd Tanous         if (*redfishGenerationInUse != pcie_device::PCIeTypes::Invalid)
319d1bde9e5SKrzysztof Grobelny         {
320543f9a75SLakshmi Yadlapati             resp.jsonValue["PCIeInterface"]["PCIeType"] =
321d1bde9e5SKrzysztof Grobelny                 *redfishGenerationInUse;
322d1bde9e5SKrzysztof Grobelny         }
323a9f68bb5STony Lee     }
324d1bde9e5SKrzysztof Grobelny 
325543f9a75SLakshmi Yadlapati     // The default value of LanesInUse is 0, and the field will be
326543f9a75SLakshmi Yadlapati     // left as off if it is a default value.
327543f9a75SLakshmi Yadlapati     if (lanesInUse != nullptr && *lanesInUse != 0)
328543f9a75SLakshmi Yadlapati     {
329543f9a75SLakshmi Yadlapati         resp.jsonValue["PCIeInterface"]["LanesInUse"] = *lanesInUse;
330543f9a75SLakshmi Yadlapati     }
331543f9a75SLakshmi Yadlapati 
33235ad613dSLakshmi Yadlapati     resp.jsonValue["PCIeFunctions"]["@odata.id"] = crow::utility::urlFromPieces(
33335ad613dSLakshmi Yadlapati         "redfish", "v1", "Systems", "system", "PCIeDevices", pcieDeviceId,
33435ad613dSLakshmi Yadlapati         "PCIeFunctions");
335d1bde9e5SKrzysztof Grobelny }
336d1bde9e5SKrzysztof Grobelny 
337543f9a75SLakshmi Yadlapati inline void getPCIeDeviceProperties(
338543f9a75SLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
339543f9a75SLakshmi Yadlapati     const std::string& pcieDevicePath, const std::string& service,
340543f9a75SLakshmi Yadlapati     const std::function<void(
341543f9a75SLakshmi Yadlapati         const dbus::utility::DBusPropertiesMap& pcieDevProperties)>&& callback)
342d1bde9e5SKrzysztof Grobelny {
343543f9a75SLakshmi Yadlapati     sdbusplus::asio::getAllProperties(
344543f9a75SLakshmi Yadlapati         *crow::connections::systemBus, service, pcieDevicePath,
345543f9a75SLakshmi Yadlapati         "xyz.openbmc_project.Inventory.Item.PCIeDevice",
346543f9a75SLakshmi Yadlapati         [aResp,
347543f9a75SLakshmi Yadlapati          callback](const boost::system::error_code& ec,
348543f9a75SLakshmi Yadlapati                    const dbus::utility::DBusPropertiesMap& pcieDevProperties) {
349543f9a75SLakshmi Yadlapati         if (ec)
350543f9a75SLakshmi Yadlapati         {
351543f9a75SLakshmi Yadlapati             if (ec.value() != EBADR)
352543f9a75SLakshmi Yadlapati             {
353543f9a75SLakshmi Yadlapati                 BMCWEB_LOG_ERROR << "DBUS response error for Properties";
354543f9a75SLakshmi Yadlapati                 messages::internalError(aResp->res);
355543f9a75SLakshmi Yadlapati             }
356543f9a75SLakshmi Yadlapati             return;
357543f9a75SLakshmi Yadlapati         }
358543f9a75SLakshmi Yadlapati         callback(pcieDevProperties);
359543f9a75SLakshmi Yadlapati         });
360d1bde9e5SKrzysztof Grobelny }
361d1bde9e5SKrzysztof Grobelny 
362543f9a75SLakshmi Yadlapati inline void addPCIeDeviceCommonProperties(
363543f9a75SLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
364543f9a75SLakshmi Yadlapati     const std::string& pcieDeviceId)
365543f9a75SLakshmi Yadlapati {
366543f9a75SLakshmi Yadlapati     aResp->res.addHeader(
367543f9a75SLakshmi Yadlapati         boost::beast::http::field::link,
368543f9a75SLakshmi Yadlapati         "</redfish/v1/JsonSchemas/PCIeDevice/PCIeDevice.json>; rel=describedby");
369543f9a75SLakshmi Yadlapati     aResp->res.jsonValue["@odata.type"] = "#PCIeDevice.v1_9_0.PCIeDevice";
370543f9a75SLakshmi Yadlapati     aResp->res.jsonValue["@odata.id"] = crow::utility::urlFromPieces(
371543f9a75SLakshmi Yadlapati         "redfish", "v1", "Systems", "system", "PCIeDevices", pcieDeviceId);
372543f9a75SLakshmi Yadlapati     aResp->res.jsonValue["Name"] = "PCIe Device";
373543f9a75SLakshmi Yadlapati     aResp->res.jsonValue["Id"] = pcieDeviceId;
374543f9a75SLakshmi Yadlapati }
3751476687dSEd Tanous 
376543f9a75SLakshmi Yadlapati inline void handlePCIeDeviceGet(App& app, const crow::Request& req,
377543f9a75SLakshmi Yadlapati                                 const std::shared_ptr<bmcweb::AsyncResp>& aResp,
378543f9a75SLakshmi Yadlapati                                 const std::string& systemName,
379543f9a75SLakshmi Yadlapati                                 const std::string& pcieDeviceId)
380543f9a75SLakshmi Yadlapati {
381543f9a75SLakshmi Yadlapati     if (!redfish::setUpRedfishRoute(app, req, aResp))
382543f9a75SLakshmi Yadlapati     {
383543f9a75SLakshmi Yadlapati         return;
384543f9a75SLakshmi Yadlapati     }
385543f9a75SLakshmi Yadlapati     if (systemName != "system")
386543f9a75SLakshmi Yadlapati     {
387543f9a75SLakshmi Yadlapati         messages::resourceNotFound(aResp->res, "ComputerSystem", systemName);
388543f9a75SLakshmi Yadlapati         return;
389543f9a75SLakshmi Yadlapati     }
390543f9a75SLakshmi Yadlapati 
391543f9a75SLakshmi Yadlapati     getValidPCIeDevicePath(
392543f9a75SLakshmi Yadlapati         pcieDeviceId, aResp,
393543f9a75SLakshmi Yadlapati         [aResp, pcieDeviceId](const std::string& pcieDevicePath,
394543f9a75SLakshmi Yadlapati                               const std::string& service) {
395543f9a75SLakshmi Yadlapati         addPCIeDeviceCommonProperties(aResp, pcieDeviceId);
396913e7732SSunnySrivastava1984         getPCIeDeviceAsset(aResp, pcieDevicePath, service);
397543f9a75SLakshmi Yadlapati         getPCIeDeviceProperties(
398543f9a75SLakshmi Yadlapati             aResp, pcieDevicePath, service,
39935ad613dSLakshmi Yadlapati             [aResp, pcieDeviceId](
40035ad613dSLakshmi Yadlapati                 const dbus::utility::DBusPropertiesMap& pcieDevProperties) {
40135ad613dSLakshmi Yadlapati             addPCIeDeviceProperties(aResp->res, pcieDeviceId,
40235ad613dSLakshmi Yadlapati                                     pcieDevProperties);
4037e860f15SJohn Edward Broadbent             });
404543f9a75SLakshmi Yadlapati         });
405543f9a75SLakshmi Yadlapati }
406543f9a75SLakshmi Yadlapati 
407543f9a75SLakshmi Yadlapati inline void requestRoutesSystemPCIeDevice(App& app)
408543f9a75SLakshmi Yadlapati {
409543f9a75SLakshmi Yadlapati     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/PCIeDevices/<str>/")
410543f9a75SLakshmi Yadlapati         .privileges(redfish::privileges::getPCIeDevice)
411543f9a75SLakshmi Yadlapati         .methods(boost::beast::http::verb::get)(
412543f9a75SLakshmi Yadlapati             std::bind_front(handlePCIeDeviceGet, std::ref(app)));
413dede6a98SJason M. Bills }
414dede6a98SJason M. Bills 
41535ad613dSLakshmi Yadlapati inline void addPCIeFunctionList(
41635ad613dSLakshmi Yadlapati     crow::Response& res, const std::string& pcieDeviceId,
41735ad613dSLakshmi Yadlapati     const dbus::utility::DBusPropertiesMap& pcieDevProperties)
41835ad613dSLakshmi Yadlapati {
41935ad613dSLakshmi Yadlapati     nlohmann::json& pcieFunctionList = res.jsonValue["Members"];
42035ad613dSLakshmi Yadlapati     pcieFunctionList = nlohmann::json::array();
42135ad613dSLakshmi Yadlapati     static constexpr const int maxPciFunctionNum = 8;
42235ad613dSLakshmi Yadlapati 
42335ad613dSLakshmi Yadlapati     for (int functionNum = 0; functionNum < maxPciFunctionNum; functionNum++)
42435ad613dSLakshmi Yadlapati     {
42535ad613dSLakshmi Yadlapati         // Check if this function exists by
42635ad613dSLakshmi Yadlapati         // looking for a device ID
42789492a15SPatrick Williams         std::string devIDProperty = "Function" + std::to_string(functionNum) +
42889492a15SPatrick Williams                                     "DeviceId";
42935ad613dSLakshmi Yadlapati         const std::string* property = nullptr;
43035ad613dSLakshmi Yadlapati         for (const auto& propEntry : pcieDevProperties)
43135ad613dSLakshmi Yadlapati         {
43235ad613dSLakshmi Yadlapati             if (propEntry.first == devIDProperty)
43335ad613dSLakshmi Yadlapati             {
43435ad613dSLakshmi Yadlapati                 property = std::get_if<std::string>(&propEntry.second);
43535ad613dSLakshmi Yadlapati                 break;
43635ad613dSLakshmi Yadlapati             }
43735ad613dSLakshmi Yadlapati         }
43835ad613dSLakshmi Yadlapati         if (property == nullptr || property->empty())
43935ad613dSLakshmi Yadlapati         {
44035ad613dSLakshmi Yadlapati             continue;
44135ad613dSLakshmi Yadlapati         }
44235ad613dSLakshmi Yadlapati 
44335ad613dSLakshmi Yadlapati         nlohmann::json::object_t pcieFunction;
44435ad613dSLakshmi Yadlapati         pcieFunction["@odata.id"] = crow::utility::urlFromPieces(
44535ad613dSLakshmi Yadlapati             "redfish", "v1", "Systems", "system", "PCIeDevices", pcieDeviceId,
44635ad613dSLakshmi Yadlapati             "PCIeFunctions", std::to_string(functionNum));
447b2ba3072SPatrick Williams         pcieFunctionList.emplace_back(std::move(pcieFunction));
44835ad613dSLakshmi Yadlapati     }
44935ad613dSLakshmi Yadlapati     res.jsonValue["PCIeFunctions@odata.count"] = pcieFunctionList.size();
45035ad613dSLakshmi Yadlapati }
45135ad613dSLakshmi Yadlapati 
45235ad613dSLakshmi Yadlapati inline void handlePCIeFunctionCollectionGet(
45335ad613dSLakshmi Yadlapati     App& app, const crow::Request& req,
45435ad613dSLakshmi Yadlapati     const std::shared_ptr<bmcweb::AsyncResp>& aResp,
45535ad613dSLakshmi Yadlapati     const std::string& pcieDeviceId)
45635ad613dSLakshmi Yadlapati {
45735ad613dSLakshmi Yadlapati     if (!redfish::setUpRedfishRoute(app, req, aResp))
45835ad613dSLakshmi Yadlapati     {
45935ad613dSLakshmi Yadlapati         return;
46035ad613dSLakshmi Yadlapati     }
46135ad613dSLakshmi Yadlapati 
46235ad613dSLakshmi Yadlapati     getValidPCIeDevicePath(
46335ad613dSLakshmi Yadlapati         pcieDeviceId, aResp,
46435ad613dSLakshmi Yadlapati         [aResp, pcieDeviceId](const std::string& pcieDevicePath,
46535ad613dSLakshmi Yadlapati                               const std::string& service) {
46635ad613dSLakshmi Yadlapati         aResp->res.addHeader(
46735ad613dSLakshmi Yadlapati             boost::beast::http::field::link,
46835ad613dSLakshmi Yadlapati             "</redfish/v1/JsonSchemas/PCIeFunctionCollection/PCIeFunctionCollection.json>; rel=describedby");
46935ad613dSLakshmi Yadlapati         aResp->res.jsonValue["@odata.type"] =
47035ad613dSLakshmi Yadlapati             "#PCIeFunctionCollection.PCIeFunctionCollection";
47135ad613dSLakshmi Yadlapati         aResp->res.jsonValue["@odata.id"] = crow::utility::urlFromPieces(
47235ad613dSLakshmi Yadlapati             "redfish", "v1", "Systems", "system", "PCIeDevices", pcieDeviceId,
47335ad613dSLakshmi Yadlapati             "PCIeFunctions");
47435ad613dSLakshmi Yadlapati         aResp->res.jsonValue["Name"] = "PCIe Function Collection";
47535ad613dSLakshmi Yadlapati         aResp->res.jsonValue["Description"] =
47635ad613dSLakshmi Yadlapati             "Collection of PCIe Functions for PCIe Device " + pcieDeviceId;
47735ad613dSLakshmi Yadlapati         getPCIeDeviceProperties(
47835ad613dSLakshmi Yadlapati             aResp, pcieDevicePath, service,
47935ad613dSLakshmi Yadlapati             [aResp, pcieDeviceId](
48035ad613dSLakshmi Yadlapati                 const dbus::utility::DBusPropertiesMap& pcieDevProperties) {
48135ad613dSLakshmi Yadlapati             addPCIeFunctionList(aResp->res, pcieDeviceId, pcieDevProperties);
48235ad613dSLakshmi Yadlapati             });
48335ad613dSLakshmi Yadlapati         });
48435ad613dSLakshmi Yadlapati }
48535ad613dSLakshmi Yadlapati 
4867e860f15SJohn Edward Broadbent inline void requestRoutesSystemPCIeFunctionCollection(App& app)
4877e860f15SJohn Edward Broadbent {
488dede6a98SJason M. Bills     /**
489dede6a98SJason M. Bills      * Functions triggers appropriate requests on DBus
490dede6a98SJason M. Bills      */
4917e860f15SJohn Edward Broadbent     BMCWEB_ROUTE(app,
4927e860f15SJohn Edward Broadbent                  "/redfish/v1/Systems/system/PCIeDevices/<str>/PCIeFunctions/")
493ed398213SEd Tanous         .privileges(redfish::privileges::getPCIeFunctionCollection)
494002d39b4SEd Tanous         .methods(boost::beast::http::verb::get)(
49535ad613dSLakshmi Yadlapati             std::bind_front(handlePCIeFunctionCollectionGet, std::ref(app)));
4967e860f15SJohn Edward Broadbent }
4977e860f15SJohn Edward Broadbent 
498727a046cSLakshmi Yadlapati inline bool validatePCIeFunctionId(
499727a046cSLakshmi Yadlapati     const std::string& pcieFunctionId,
500727a046cSLakshmi Yadlapati     const dbus::utility::DBusPropertiesMap& pcieDevProperties)
5017e860f15SJohn Edward Broadbent {
502727a046cSLakshmi Yadlapati     std::string functionName = "Function" + pcieFunctionId;
503b9d36b47SEd Tanous     std::string devIDProperty = functionName + "DeviceId";
504b9d36b47SEd Tanous 
505b9d36b47SEd Tanous     const std::string* devIdProperty = nullptr;
506b9d36b47SEd Tanous     for (const auto& property : pcieDevProperties)
507b9d36b47SEd Tanous     {
508b9d36b47SEd Tanous         if (property.first == devIDProperty)
509b9d36b47SEd Tanous         {
510002d39b4SEd Tanous             devIdProperty = std::get_if<std::string>(&property.second);
511727a046cSLakshmi Yadlapati             break;
512b9d36b47SEd Tanous         }
513b9d36b47SEd Tanous     }
514727a046cSLakshmi Yadlapati     return (devIdProperty != nullptr && !devIdProperty->empty());
515727a046cSLakshmi Yadlapati }
516727a046cSLakshmi Yadlapati 
517727a046cSLakshmi Yadlapati inline void addPCIeFunctionProperties(
518727a046cSLakshmi Yadlapati     crow::Response& resp, const std::string& pcieFunctionId,
519727a046cSLakshmi Yadlapati     const dbus::utility::DBusPropertiesMap& pcieDevProperties)
520f5c9f8bdSJason M. Bills {
521727a046cSLakshmi Yadlapati     std::string functionName = "Function" + pcieFunctionId;
522727a046cSLakshmi Yadlapati     if (!validatePCIeFunctionId(pcieFunctionId, pcieDevProperties))
523727a046cSLakshmi Yadlapati     {
524727a046cSLakshmi Yadlapati         messages::resourceNotFound(resp, "PCIeFunction", pcieFunctionId);
525f5c9f8bdSJason M. Bills         return;
526f5c9f8bdSJason M. Bills     }
527b9d36b47SEd Tanous     for (const auto& property : pcieDevProperties)
528f5c9f8bdSJason M. Bills     {
529b9d36b47SEd Tanous         const std::string* strProperty =
530b9d36b47SEd Tanous             std::get_if<std::string>(&property.second);
531727a046cSLakshmi Yadlapati 
532b9d36b47SEd Tanous         if (property.first == functionName + "DeviceId")
533f5c9f8bdSJason M. Bills         {
534727a046cSLakshmi Yadlapati             resp.jsonValue["DeviceId"] = *strProperty;
535f5c9f8bdSJason M. Bills         }
536b9d36b47SEd Tanous         if (property.first == functionName + "VendorId")
537f5c9f8bdSJason M. Bills         {
538727a046cSLakshmi Yadlapati             resp.jsonValue["VendorId"] = *strProperty;
539f5c9f8bdSJason M. Bills         }
540727a046cSLakshmi Yadlapati         // TODO: FunctionType and DeviceClass are Redfish enums. The D-Bus
541727a046cSLakshmi Yadlapati         // property strings should be mapped correctly to ensure these
542727a046cSLakshmi Yadlapati         // strings are Redfish enum values. For now just check for empty.
543b9d36b47SEd Tanous         if (property.first == functionName + "FunctionType")
544f5c9f8bdSJason M. Bills         {
545727a046cSLakshmi Yadlapati             if (!strProperty->empty())
546727a046cSLakshmi Yadlapati             {
547727a046cSLakshmi Yadlapati                 resp.jsonValue["FunctionType"] = *strProperty;
548727a046cSLakshmi Yadlapati             }
549f5c9f8bdSJason M. Bills         }
550b9d36b47SEd Tanous         if (property.first == functionName + "DeviceClass")
551f5c9f8bdSJason M. Bills         {
552727a046cSLakshmi Yadlapati             if (!strProperty->empty())
553727a046cSLakshmi Yadlapati             {
554727a046cSLakshmi Yadlapati                 resp.jsonValue["DeviceClass"] = *strProperty;
555727a046cSLakshmi Yadlapati             }
556f5c9f8bdSJason M. Bills         }
557b9d36b47SEd Tanous         if (property.first == functionName + "ClassCode")
558f5c9f8bdSJason M. Bills         {
559727a046cSLakshmi Yadlapati             resp.jsonValue["ClassCode"] = *strProperty;
560f5c9f8bdSJason M. Bills         }
561b9d36b47SEd Tanous         if (property.first == functionName + "RevisionId")
562f5c9f8bdSJason M. Bills         {
563727a046cSLakshmi Yadlapati             resp.jsonValue["RevisionId"] = *strProperty;
564f5c9f8bdSJason M. Bills         }
565b9d36b47SEd Tanous         if (property.first == functionName + "SubsystemId")
566b9d36b47SEd Tanous         {
567727a046cSLakshmi Yadlapati             resp.jsonValue["SubsystemId"] = *strProperty;
568b9d36b47SEd Tanous         }
569002d39b4SEd Tanous         if (property.first == functionName + "SubsystemVendorId")
570f5c9f8bdSJason M. Bills         {
571727a046cSLakshmi Yadlapati             resp.jsonValue["SubsystemVendorId"] = *strProperty;
572b9d36b47SEd Tanous         }
573f5c9f8bdSJason M. Bills     }
574727a046cSLakshmi Yadlapati }
575727a046cSLakshmi Yadlapati 
576727a046cSLakshmi Yadlapati inline void addPCIeFunctionCommonProperties(crow::Response& resp,
577727a046cSLakshmi Yadlapati                                             const std::string& pcieDeviceId,
578727a046cSLakshmi Yadlapati                                             const std::string& pcieFunctionId)
579727a046cSLakshmi Yadlapati {
580727a046cSLakshmi Yadlapati     resp.addHeader(
581727a046cSLakshmi Yadlapati         boost::beast::http::field::link,
582727a046cSLakshmi Yadlapati         "</redfish/v1/JsonSchemas/PCIeFunction/PCIeFunction.json>; rel=describedby");
583727a046cSLakshmi Yadlapati     resp.jsonValue["@odata.type"] = "#PCIeFunction.v1_2_3.PCIeFunction";
584727a046cSLakshmi Yadlapati     resp.jsonValue["@odata.id"] = crow::utility::urlFromPieces(
585727a046cSLakshmi Yadlapati         "redfish", "v1", "Systems", "system", "PCIeDevices", pcieDeviceId,
586727a046cSLakshmi Yadlapati         "PCIeFunctions", pcieFunctionId);
587727a046cSLakshmi Yadlapati     resp.jsonValue["Name"] = "PCIe Function";
588727a046cSLakshmi Yadlapati     resp.jsonValue["Id"] = pcieFunctionId;
589727a046cSLakshmi Yadlapati     resp.jsonValue["FunctionId"] = std::stoi(pcieFunctionId);
590727a046cSLakshmi Yadlapati     resp.jsonValue["Links"]["PCIeDevice"]["@odata.id"] =
591727a046cSLakshmi Yadlapati         crow::utility::urlFromPieces("redfish", "v1", "Systems", "system",
592727a046cSLakshmi Yadlapati                                      "PCIeDevices", pcieDeviceId);
593727a046cSLakshmi Yadlapati }
594727a046cSLakshmi Yadlapati 
595727a046cSLakshmi Yadlapati inline void
596727a046cSLakshmi Yadlapati     handlePCIeFunctionGet(App& app, const crow::Request& req,
597727a046cSLakshmi Yadlapati                           const std::shared_ptr<bmcweb::AsyncResp>& aResp,
598727a046cSLakshmi Yadlapati                           const std::string& pcieDeviceId,
599727a046cSLakshmi Yadlapati                           const std::string& pcieFunctionId)
600727a046cSLakshmi Yadlapati {
601727a046cSLakshmi Yadlapati     if (!redfish::setUpRedfishRoute(app, req, aResp))
602727a046cSLakshmi Yadlapati     {
603727a046cSLakshmi Yadlapati         return;
604727a046cSLakshmi Yadlapati     }
605727a046cSLakshmi Yadlapati 
606727a046cSLakshmi Yadlapati     getValidPCIeDevicePath(
607727a046cSLakshmi Yadlapati         pcieDeviceId, aResp,
608727a046cSLakshmi Yadlapati         [aResp, pcieDeviceId, pcieFunctionId](const std::string& pcieDevicePath,
609727a046cSLakshmi Yadlapati                                               const std::string& service) {
610727a046cSLakshmi Yadlapati         getPCIeDeviceProperties(
611727a046cSLakshmi Yadlapati             aResp, pcieDevicePath, service,
612727a046cSLakshmi Yadlapati             [aResp, pcieDeviceId, pcieFunctionId](
613727a046cSLakshmi Yadlapati                 const dbus::utility::DBusPropertiesMap& pcieDevProperties) {
614727a046cSLakshmi Yadlapati             addPCIeFunctionCommonProperties(aResp->res, pcieDeviceId,
615727a046cSLakshmi Yadlapati                                             pcieFunctionId);
616727a046cSLakshmi Yadlapati             addPCIeFunctionProperties(aResp->res, pcieFunctionId,
617727a046cSLakshmi Yadlapati                                       pcieDevProperties);
6187e860f15SJohn Edward Broadbent             });
619727a046cSLakshmi Yadlapati         });
620727a046cSLakshmi Yadlapati }
621727a046cSLakshmi Yadlapati 
622727a046cSLakshmi Yadlapati inline void requestRoutesSystemPCIeFunction(App& app)
623727a046cSLakshmi Yadlapati {
624727a046cSLakshmi Yadlapati     BMCWEB_ROUTE(
625727a046cSLakshmi Yadlapati         app,
626727a046cSLakshmi Yadlapati         "/redfish/v1/Systems/system/PCIeDevices/<str>/PCIeFunctions/<str>/")
627727a046cSLakshmi Yadlapati         .privileges(redfish::privileges::getPCIeFunction)
628727a046cSLakshmi Yadlapati         .methods(boost::beast::http::verb::get)(
629727a046cSLakshmi Yadlapati             std::bind_front(handlePCIeFunctionGet, std::ref(app)));
630f5c9f8bdSJason M. Bills }
631f5c9f8bdSJason M. Bills 
632f5c9f8bdSJason M. Bills } // namespace redfish
633