xref: /openbmc/bmcweb/redfish-core/lib/processor.hpp (revision 5b90429a75c58797ec29ac9a8ed18c2dcd7d4950)
1 /*
2 // Copyright (c) 2018 Intel Corporation
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 //      http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 */
16 #pragma once
17 
18 #include "app.hpp"
19 #include "dbus_singleton.hpp"
20 #include "dbus_utility.hpp"
21 #include "error_messages.hpp"
22 #include "generated/enums/processor.hpp"
23 #include "query.hpp"
24 #include "registries/privilege_registry.hpp"
25 #include "utils/collection.hpp"
26 #include "utils/dbus_utils.hpp"
27 #include "utils/json_utils.hpp"
28 
29 #include <boost/container/flat_map.hpp>
30 #include <boost/system/error_code.hpp>
31 #include <boost/url/format.hpp>
32 #include <sdbusplus/asio/property.hpp>
33 #include <sdbusplus/message/native_types.hpp>
34 #include <sdbusplus/unpack_properties.hpp>
35 #include <sdbusplus/utility/dedup_variant.hpp>
36 
37 #include <array>
38 #include <limits>
39 #include <ranges>
40 #include <string>
41 #include <string_view>
42 
43 namespace redfish
44 {
45 
46 // Interfaces which imply a D-Bus object represents a Processor
47 constexpr std::array<std::string_view, 2> processorInterfaces = {
48     "xyz.openbmc_project.Inventory.Item.Cpu",
49     "xyz.openbmc_project.Inventory.Item.Accelerator"};
50 
51 /**
52  * @brief Fill out uuid info of a processor by
53  * requesting data from the given D-Bus object.
54  *
55  * @param[in,out]   asyncResp       Async HTTP response.
56  * @param[in]       service     D-Bus service to query.
57  * @param[in]       objPath     D-Bus object to query.
58  */
59 inline void getProcessorUUID(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
60                              const std::string& service,
61                              const std::string& objPath)
62 {
63     BMCWEB_LOG_DEBUG("Get Processor UUID");
64     sdbusplus::asio::getProperty<std::string>(
65         *crow::connections::systemBus, service, objPath,
66         "xyz.openbmc_project.Common.UUID", "UUID",
67         [objPath, asyncResp{std::move(asyncResp)}](
68             const boost::system::error_code& ec, const std::string& property) {
69         if (ec)
70         {
71             BMCWEB_LOG_DEBUG("DBUS response error");
72             messages::internalError(asyncResp->res);
73             return;
74         }
75         asyncResp->res.jsonValue["UUID"] = property;
76     });
77 }
78 
79 inline void getCpuDataByInterface(
80     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
81     const dbus::utility::DBusInterfacesMap& cpuInterfacesProperties)
82 {
83     BMCWEB_LOG_DEBUG("Get CPU resources by interface.");
84 
85     // Set the default value of state
86     asyncResp->res.jsonValue["Status"]["State"] = "Enabled";
87     asyncResp->res.jsonValue["Status"]["Health"] = "OK";
88 
89     for (const auto& interface : cpuInterfacesProperties)
90     {
91         for (const auto& property : interface.second)
92         {
93             if (property.first == "Present")
94             {
95                 const bool* cpuPresent = std::get_if<bool>(&property.second);
96                 if (cpuPresent == nullptr)
97                 {
98                     // Important property not in desired type
99                     messages::internalError(asyncResp->res);
100                     return;
101                 }
102                 if (!*cpuPresent)
103                 {
104                     // Slot is not populated
105                     asyncResp->res.jsonValue["Status"]["State"] = "Absent";
106                 }
107             }
108             else if (property.first == "Functional")
109             {
110                 const bool* cpuFunctional = std::get_if<bool>(&property.second);
111                 if (cpuFunctional == nullptr)
112                 {
113                     messages::internalError(asyncResp->res);
114                     return;
115                 }
116                 if (!*cpuFunctional)
117                 {
118                     asyncResp->res.jsonValue["Status"]["Health"] = "Critical";
119                 }
120             }
121             else if (property.first == "CoreCount")
122             {
123                 const uint16_t* coresCount =
124                     std::get_if<uint16_t>(&property.second);
125                 if (coresCount == nullptr)
126                 {
127                     messages::internalError(asyncResp->res);
128                     return;
129                 }
130                 asyncResp->res.jsonValue["TotalCores"] = *coresCount;
131             }
132             else if (property.first == "MaxSpeedInMhz")
133             {
134                 const uint32_t* value = std::get_if<uint32_t>(&property.second);
135                 if (value != nullptr)
136                 {
137                     asyncResp->res.jsonValue["MaxSpeedMHz"] = *value;
138                 }
139             }
140             else if (property.first == "Socket")
141             {
142                 const std::string* value =
143                     std::get_if<std::string>(&property.second);
144                 if (value != nullptr)
145                 {
146                     asyncResp->res.jsonValue["Socket"] = *value;
147                 }
148             }
149             else if (property.first == "ThreadCount")
150             {
151                 const uint16_t* value = std::get_if<uint16_t>(&property.second);
152                 if (value != nullptr)
153                 {
154                     asyncResp->res.jsonValue["TotalThreads"] = *value;
155                 }
156             }
157             else if (property.first == "EffectiveFamily")
158             {
159                 const uint16_t* value = std::get_if<uint16_t>(&property.second);
160                 if (value != nullptr && *value != 2)
161                 {
162                     asyncResp->res.jsonValue["ProcessorId"]["EffectiveFamily"] =
163                         "0x" + intToHexString(*value, 4);
164                 }
165             }
166             else if (property.first == "EffectiveModel")
167             {
168                 const uint16_t* value = std::get_if<uint16_t>(&property.second);
169                 if (value == nullptr)
170                 {
171                     messages::internalError(asyncResp->res);
172                     return;
173                 }
174                 if (*value != 0)
175                 {
176                     asyncResp->res.jsonValue["ProcessorId"]["EffectiveModel"] =
177                         "0x" + intToHexString(*value, 4);
178                 }
179             }
180             else if (property.first == "Id")
181             {
182                 const uint64_t* value = std::get_if<uint64_t>(&property.second);
183                 if (value != nullptr && *value != 0)
184                 {
185                     asyncResp->res
186                         .jsonValue["ProcessorId"]["IdentificationRegisters"] =
187                         "0x" + intToHexString(*value, 16);
188                 }
189             }
190             else if (property.first == "Microcode")
191             {
192                 const uint32_t* value = std::get_if<uint32_t>(&property.second);
193                 if (value == nullptr)
194                 {
195                     messages::internalError(asyncResp->res);
196                     return;
197                 }
198                 if (*value != 0)
199                 {
200                     asyncResp->res.jsonValue["ProcessorId"]["MicrocodeInfo"] =
201                         "0x" + intToHexString(*value, 8);
202                 }
203             }
204             else if (property.first == "Step")
205             {
206                 const uint16_t* value = std::get_if<uint16_t>(&property.second);
207                 if (value == nullptr)
208                 {
209                     messages::internalError(asyncResp->res);
210                     return;
211                 }
212                 if (*value != std::numeric_limits<uint16_t>::max())
213                 {
214                     asyncResp->res.jsonValue["ProcessorId"]["Step"] =
215                         "0x" + intToHexString(*value, 4);
216                 }
217             }
218         }
219     }
220 }
221 
222 inline void getCpuDataByService(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
223                                 const std::string& cpuId,
224                                 const std::string& service,
225                                 const std::string& objPath)
226 {
227     BMCWEB_LOG_DEBUG("Get available system cpu resources by service.");
228 
229     sdbusplus::message::object_path path("/xyz/openbmc_project/inventory");
230     dbus::utility::getManagedObjects(
231         service, path,
232         [cpuId, service, objPath, asyncResp{std::move(asyncResp)}](
233             const boost::system::error_code& ec,
234             const dbus::utility::ManagedObjectType& dbusData) {
235         if (ec)
236         {
237             BMCWEB_LOG_DEBUG("DBUS response error");
238             messages::internalError(asyncResp->res);
239             return;
240         }
241         asyncResp->res.jsonValue["Id"] = cpuId;
242         asyncResp->res.jsonValue["Name"] = "Processor";
243         asyncResp->res.jsonValue["ProcessorType"] = "CPU";
244 
245         bool slotPresent = false;
246         std::string corePath = objPath + "/core";
247         size_t totalCores = 0;
248         for (const auto& object : dbusData)
249         {
250             if (object.first.str == objPath)
251             {
252                 getCpuDataByInterface(asyncResp, object.second);
253             }
254             else if (object.first.str.starts_with(corePath))
255             {
256                 for (const auto& interface : object.second)
257                 {
258                     if (interface.first == "xyz.openbmc_project.Inventory.Item")
259                     {
260                         for (const auto& property : interface.second)
261                         {
262                             if (property.first == "Present")
263                             {
264                                 const bool* present =
265                                     std::get_if<bool>(&property.second);
266                                 if (present != nullptr)
267                                 {
268                                     if (*present)
269                                     {
270                                         slotPresent = true;
271                                         totalCores++;
272                                     }
273                                 }
274                             }
275                         }
276                     }
277                 }
278             }
279         }
280         // In getCpuDataByInterface(), state and health are set
281         // based on the present and functional status. If core
282         // count is zero, then it has a higher precedence.
283         if (slotPresent)
284         {
285             asyncResp->res.jsonValue["TotalCores"] = totalCores;
286         }
287         return;
288     });
289 }
290 
291 /**
292  * @brief Translates throttle cause DBUS property to redfish.
293  *
294  * @param[in] dbusSource    The throttle cause from DBUS
295  *
296  * @return Returns as a string, the throttle cause in Redfish terms. If
297  * translation cannot be done, returns "Unknown" throttle reason.
298  */
299 inline processor::ThrottleCause
300     dbusToRfThrottleCause(const std::string& dbusSource)
301 {
302     if (dbusSource ==
303         "xyz.openbmc_project.Control.Power.Throttle.ThrottleReasons.ClockLimit")
304     {
305         return processor::ThrottleCause::ClockLimit;
306     }
307     if (dbusSource ==
308         "xyz.openbmc_project.Control.Power.Throttle.ThrottleReasons.ManagementDetectedFault")
309     {
310         return processor::ThrottleCause::ManagementDetectedFault;
311     }
312     if (dbusSource ==
313         "xyz.openbmc_project.Control.Power.Throttle.ThrottleReasons.PowerLimit")
314     {
315         return processor::ThrottleCause::PowerLimit;
316     }
317     if (dbusSource ==
318         "xyz.openbmc_project.Control.Power.Throttle.ThrottleReasons.ThermalLimit")
319     {
320         return processor::ThrottleCause::ThermalLimit;
321     }
322     if (dbusSource ==
323         "xyz.openbmc_project.Control.Power.Throttle.ThrottleReasons.Unknown")
324     {
325         return processor::ThrottleCause::Unknown;
326     }
327     return processor::ThrottleCause::Invalid;
328 }
329 
330 inline void
331     readThrottleProperties(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
332                            const boost::system::error_code& ec,
333                            const dbus::utility::DBusPropertiesMap& properties)
334 {
335     if (ec)
336     {
337         BMCWEB_LOG_ERROR("Processor Throttle getAllProperties error {}", ec);
338         messages::internalError(asyncResp->res);
339         return;
340     }
341 
342     const bool* status = nullptr;
343     const std::vector<std::string>* causes = nullptr;
344 
345     if (!sdbusplus::unpackPropertiesNoThrow(dbus_utils::UnpackErrorPrinter(),
346                                             properties, "Throttled", status,
347                                             "ThrottleCauses", causes))
348     {
349         messages::internalError(asyncResp->res);
350         return;
351     }
352 
353     asyncResp->res.jsonValue["Throttled"] = *status;
354     nlohmann::json::array_t rCauses;
355     for (const std::string& cause : *causes)
356     {
357         processor::ThrottleCause rfCause = dbusToRfThrottleCause(cause);
358         if (rfCause == processor::ThrottleCause::Invalid)
359         {
360             messages::internalError(asyncResp->res);
361             return;
362         }
363 
364         rCauses.emplace_back(rfCause);
365     }
366     asyncResp->res.jsonValue["ThrottleCauses"] = std::move(rCauses);
367 }
368 
369 inline void
370     getThrottleProperties(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
371                           const std::string& service,
372                           const std::string& objectPath)
373 {
374     BMCWEB_LOG_DEBUG("Get processor throttle resources");
375 
376     sdbusplus::asio::getAllProperties(
377         *crow::connections::systemBus, service, objectPath,
378         "xyz.openbmc_project.Control.Power.Throttle",
379         [asyncResp](const boost::system::error_code& ec,
380                     const dbus::utility::DBusPropertiesMap& properties) {
381         readThrottleProperties(asyncResp, ec, properties);
382     });
383 }
384 
385 inline void getCpuAssetData(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
386                             const std::string& service,
387                             const std::string& objPath)
388 {
389     BMCWEB_LOG_DEBUG("Get Cpu Asset Data");
390     sdbusplus::asio::getAllProperties(
391         *crow::connections::systemBus, service, objPath,
392         "xyz.openbmc_project.Inventory.Decorator.Asset",
393         [objPath, asyncResp{std::move(asyncResp)}](
394             const boost::system::error_code& ec,
395             const dbus::utility::DBusPropertiesMap& properties) {
396         if (ec)
397         {
398             BMCWEB_LOG_DEBUG("DBUS response error");
399             messages::internalError(asyncResp->res);
400             return;
401         }
402 
403         const std::string* serialNumber = nullptr;
404         const std::string* model = nullptr;
405         const std::string* manufacturer = nullptr;
406         const std::string* partNumber = nullptr;
407         const std::string* sparePartNumber = nullptr;
408 
409         const bool success = sdbusplus::unpackPropertiesNoThrow(
410             dbus_utils::UnpackErrorPrinter(), properties, "SerialNumber",
411             serialNumber, "Model", model, "Manufacturer", manufacturer,
412             "PartNumber", partNumber, "SparePartNumber", sparePartNumber);
413 
414         if (!success)
415         {
416             messages::internalError(asyncResp->res);
417             return;
418         }
419 
420         if (serialNumber != nullptr && !serialNumber->empty())
421         {
422             asyncResp->res.jsonValue["SerialNumber"] = *serialNumber;
423         }
424 
425         if ((model != nullptr) && !model->empty())
426         {
427             asyncResp->res.jsonValue["Model"] = *model;
428         }
429 
430         if (manufacturer != nullptr)
431         {
432             asyncResp->res.jsonValue["Manufacturer"] = *manufacturer;
433 
434             // Otherwise would be unexpected.
435             if (manufacturer->find("Intel") != std::string::npos)
436             {
437                 asyncResp->res.jsonValue["ProcessorArchitecture"] = "x86";
438                 asyncResp->res.jsonValue["InstructionSet"] = "x86-64";
439             }
440             else if (manufacturer->find("IBM") != std::string::npos)
441             {
442                 asyncResp->res.jsonValue["ProcessorArchitecture"] = "Power";
443                 asyncResp->res.jsonValue["InstructionSet"] = "PowerISA";
444             }
445         }
446 
447         if (partNumber != nullptr)
448         {
449             asyncResp->res.jsonValue["PartNumber"] = *partNumber;
450         }
451 
452         if (sparePartNumber != nullptr && !sparePartNumber->empty())
453         {
454             asyncResp->res.jsonValue["SparePartNumber"] = *sparePartNumber;
455         }
456     });
457 }
458 
459 inline void getCpuRevisionData(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
460                                const std::string& service,
461                                const std::string& objPath)
462 {
463     BMCWEB_LOG_DEBUG("Get Cpu Revision Data");
464     sdbusplus::asio::getAllProperties(
465         *crow::connections::systemBus, service, objPath,
466         "xyz.openbmc_project.Inventory.Decorator.Revision",
467         [objPath, asyncResp{std::move(asyncResp)}](
468             const boost::system::error_code& ec,
469             const dbus::utility::DBusPropertiesMap& properties) {
470         if (ec)
471         {
472             BMCWEB_LOG_DEBUG("DBUS response error");
473             messages::internalError(asyncResp->res);
474             return;
475         }
476 
477         const std::string* version = nullptr;
478 
479         const bool success = sdbusplus::unpackPropertiesNoThrow(
480             dbus_utils::UnpackErrorPrinter(), properties, "Version", version);
481 
482         if (!success)
483         {
484             messages::internalError(asyncResp->res);
485             return;
486         }
487 
488         if (version != nullptr)
489         {
490             asyncResp->res.jsonValue["Version"] = *version;
491         }
492     });
493 }
494 
495 inline void getAcceleratorDataByService(
496     std::shared_ptr<bmcweb::AsyncResp> asyncResp, const std::string& acclrtrId,
497     const std::string& service, const std::string& objPath)
498 {
499     BMCWEB_LOG_DEBUG("Get available system Accelerator resources by service.");
500     sdbusplus::asio::getAllProperties(
501         *crow::connections::systemBus, service, objPath, "",
502         [acclrtrId, asyncResp{std::move(asyncResp)}](
503             const boost::system::error_code& ec,
504             const dbus::utility::DBusPropertiesMap& properties) {
505         if (ec)
506         {
507             BMCWEB_LOG_DEBUG("DBUS response error");
508             messages::internalError(asyncResp->res);
509             return;
510         }
511 
512         const bool* functional = nullptr;
513         const bool* present = nullptr;
514 
515         const bool success = sdbusplus::unpackPropertiesNoThrow(
516             dbus_utils::UnpackErrorPrinter(), properties, "Functional",
517             functional, "Present", present);
518 
519         if (!success)
520         {
521             messages::internalError(asyncResp->res);
522             return;
523         }
524 
525         std::string state = "Enabled";
526         std::string health = "OK";
527 
528         if (present != nullptr && !*present)
529         {
530             state = "Absent";
531         }
532 
533         if (functional != nullptr && !*functional)
534         {
535             if (state == "Enabled")
536             {
537                 health = "Critical";
538             }
539         }
540 
541         asyncResp->res.jsonValue["Id"] = acclrtrId;
542         asyncResp->res.jsonValue["Name"] = "Processor";
543         asyncResp->res.jsonValue["Status"]["State"] = state;
544         asyncResp->res.jsonValue["Status"]["Health"] = health;
545         asyncResp->res.jsonValue["ProcessorType"] = "Accelerator";
546     });
547 }
548 
549 // OperatingConfig D-Bus Types
550 using TurboProfileProperty = std::vector<std::tuple<uint32_t, size_t>>;
551 using BaseSpeedPrioritySettingsProperty =
552     std::vector<std::tuple<uint32_t, std::vector<uint32_t>>>;
553 // uint32_t and size_t may or may not be the same type, requiring a dedup'd
554 // variant
555 
556 /**
557  * Fill out the HighSpeedCoreIDs in a Processor resource from the given
558  * OperatingConfig D-Bus property.
559  *
560  * @param[in,out]   asyncResp           Async HTTP response.
561  * @param[in]       baseSpeedSettings   Full list of base speed priority groups,
562  *                                      to use to determine the list of high
563  *                                      speed cores.
564  */
565 inline void highSpeedCoreIdsHandler(
566     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
567     const BaseSpeedPrioritySettingsProperty& baseSpeedSettings)
568 {
569     // The D-Bus property does not indicate which bucket is the "high
570     // priority" group, so let's discern that by looking for the one with
571     // highest base frequency.
572     auto highPriorityGroup = baseSpeedSettings.cend();
573     uint32_t highestBaseSpeed = 0;
574     for (auto it = baseSpeedSettings.cbegin(); it != baseSpeedSettings.cend();
575          ++it)
576     {
577         const uint32_t baseFreq = std::get<uint32_t>(*it);
578         if (baseFreq > highestBaseSpeed)
579         {
580             highestBaseSpeed = baseFreq;
581             highPriorityGroup = it;
582         }
583     }
584 
585     nlohmann::json& jsonCoreIds = asyncResp->res.jsonValue["HighSpeedCoreIDs"];
586     jsonCoreIds = nlohmann::json::array();
587 
588     // There may not be any entries in the D-Bus property, so only populate
589     // if there was actually something there.
590     if (highPriorityGroup != baseSpeedSettings.cend())
591     {
592         jsonCoreIds = std::get<std::vector<uint32_t>>(*highPriorityGroup);
593     }
594 }
595 
596 /**
597  * Fill out OperatingConfig related items in a Processor resource by requesting
598  * data from the given D-Bus object.
599  *
600  * @param[in,out]   asyncResp       Async HTTP response.
601  * @param[in]       cpuId       CPU D-Bus name.
602  * @param[in]       service     D-Bus service to query.
603  * @param[in]       objPath     D-Bus object to query.
604  */
605 inline void
606     getCpuConfigData(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
607                      const std::string& cpuId, const std::string& service,
608                      const std::string& objPath)
609 {
610     BMCWEB_LOG_INFO("Getting CPU operating configs for {}", cpuId);
611 
612     // First, GetAll CurrentOperatingConfig properties on the object
613     sdbusplus::asio::getAllProperties(
614         *crow::connections::systemBus, service, objPath,
615         "xyz.openbmc_project.Control.Processor.CurrentOperatingConfig",
616         [asyncResp, cpuId,
617          service](const boost::system::error_code& ec,
618                   const dbus::utility::DBusPropertiesMap& properties) {
619         if (ec)
620         {
621             BMCWEB_LOG_WARNING("D-Bus error: {}, {}", ec, ec.message());
622             messages::internalError(asyncResp->res);
623             return;
624         }
625 
626         nlohmann::json& json = asyncResp->res.jsonValue;
627 
628         const sdbusplus::message::object_path* appliedConfig = nullptr;
629         const bool* baseSpeedPriorityEnabled = nullptr;
630 
631         const bool success = sdbusplus::unpackPropertiesNoThrow(
632             dbus_utils::UnpackErrorPrinter(), properties, "AppliedConfig",
633             appliedConfig, "BaseSpeedPriorityEnabled",
634             baseSpeedPriorityEnabled);
635 
636         if (!success)
637         {
638             messages::internalError(asyncResp->res);
639             return;
640         }
641 
642         if (appliedConfig != nullptr)
643         {
644             const std::string& dbusPath = appliedConfig->str;
645             nlohmann::json::object_t operatingConfig;
646             operatingConfig["@odata.id"] = boost::urls::format(
647                 "/redfish/v1/Systems/system/Processors/{}/OperatingConfigs",
648                 cpuId);
649             json["OperatingConfigs"] = std::move(operatingConfig);
650 
651             // Reuse the D-Bus config object name for the Redfish
652             // URI
653             size_t baseNamePos = dbusPath.rfind('/');
654             if (baseNamePos == std::string::npos ||
655                 baseNamePos == (dbusPath.size() - 1))
656             {
657                 // If the AppliedConfig was somehow not a valid path,
658                 // skip adding any more properties, since everything
659                 // else is tied to this applied config.
660                 messages::internalError(asyncResp->res);
661                 return;
662             }
663             nlohmann::json::object_t appliedOperatingConfig;
664             appliedOperatingConfig["@odata.id"] = boost::urls::format(
665                 "/redfish/v1/Systems/system/Processors/{}/OperatingConfigs/{}",
666                 cpuId, dbusPath.substr(baseNamePos + 1));
667             json["AppliedOperatingConfig"] = std::move(appliedOperatingConfig);
668 
669             // Once we found the current applied config, queue another
670             // request to read the base freq core ids out of that
671             // config.
672             sdbusplus::asio::getProperty<BaseSpeedPrioritySettingsProperty>(
673                 *crow::connections::systemBus, service, dbusPath,
674                 "xyz.openbmc_project.Inventory.Item.Cpu."
675                 "OperatingConfig",
676                 "BaseSpeedPrioritySettings",
677                 [asyncResp](
678                     const boost::system::error_code& ec2,
679                     const BaseSpeedPrioritySettingsProperty& baseSpeedList) {
680                 if (ec2)
681                 {
682                     BMCWEB_LOG_WARNING("D-Bus Property Get error: {}", ec2);
683                     messages::internalError(asyncResp->res);
684                     return;
685                 }
686 
687                 highSpeedCoreIdsHandler(asyncResp, baseSpeedList);
688             });
689         }
690 
691         if (baseSpeedPriorityEnabled != nullptr)
692         {
693             json["BaseSpeedPriorityState"] =
694                 *baseSpeedPriorityEnabled ? "Enabled" : "Disabled";
695         }
696     });
697 }
698 
699 /**
700  * @brief Fill out location info of a processor by
701  * requesting data from the given D-Bus object.
702  *
703  * @param[in,out]   asyncResp       Async HTTP response.
704  * @param[in]       service     D-Bus service to query.
705  * @param[in]       objPath     D-Bus object to query.
706  */
707 inline void getCpuLocationCode(std::shared_ptr<bmcweb::AsyncResp> asyncResp,
708                                const std::string& service,
709                                const std::string& objPath)
710 {
711     BMCWEB_LOG_DEBUG("Get Cpu Location Data");
712     sdbusplus::asio::getProperty<std::string>(
713         *crow::connections::systemBus, service, objPath,
714         "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode",
715         [objPath, asyncResp{std::move(asyncResp)}](
716             const boost::system::error_code& ec, const std::string& property) {
717         if (ec)
718         {
719             BMCWEB_LOG_DEBUG("DBUS response error");
720             messages::internalError(asyncResp->res);
721             return;
722         }
723 
724         asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] =
725             property;
726     });
727 }
728 
729 /**
730  * Populate the unique identifier in a Processor resource by requesting data
731  * from the given D-Bus object.
732  *
733  * @param[in,out]   asyncResp   Async HTTP response.
734  * @param[in]       service     D-Bus service to query.
735  * @param[in]       objPath     D-Bus object to query.
736  */
737 inline void getCpuUniqueId(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
738                            const std::string& service,
739                            const std::string& objectPath)
740 {
741     BMCWEB_LOG_DEBUG("Get CPU UniqueIdentifier");
742     sdbusplus::asio::getProperty<std::string>(
743         *crow::connections::systemBus, service, objectPath,
744         "xyz.openbmc_project.Inventory.Decorator.UniqueIdentifier",
745         "UniqueIdentifier",
746         [asyncResp](const boost::system::error_code& ec,
747                     const std::string& id) {
748         if (ec)
749         {
750             BMCWEB_LOG_ERROR("Failed to read cpu unique id: {}", ec);
751             messages::internalError(asyncResp->res);
752             return;
753         }
754         asyncResp->res
755             .jsonValue["ProcessorId"]["ProtectedIdentificationNumber"] = id;
756     });
757 }
758 
759 /**
760  * Find the D-Bus object representing the requested Processor, and call the
761  * handler with the results. If matching object is not found, add 404 error to
762  * response and don't call the handler.
763  *
764  * @param[in,out]   resp            Async HTTP response.
765  * @param[in]       processorId     Redfish Processor Id.
766  * @param[in]       handler         Callback to continue processing request upon
767  *                                  successfully finding object.
768  */
769 template <typename Handler>
770 inline void getProcessorObject(const std::shared_ptr<bmcweb::AsyncResp>& resp,
771                                const std::string& processorId,
772                                Handler&& handler)
773 {
774     BMCWEB_LOG_DEBUG("Get available system processor resources.");
775 
776     // GetSubTree on all interfaces which provide info about a Processor
777     constexpr std::array<std::string_view, 9> interfaces = {
778         "xyz.openbmc_project.Common.UUID",
779         "xyz.openbmc_project.Inventory.Decorator.Asset",
780         "xyz.openbmc_project.Inventory.Decorator.Revision",
781         "xyz.openbmc_project.Inventory.Item.Cpu",
782         "xyz.openbmc_project.Inventory.Decorator.LocationCode",
783         "xyz.openbmc_project.Inventory.Item.Accelerator",
784         "xyz.openbmc_project.Control.Processor.CurrentOperatingConfig",
785         "xyz.openbmc_project.Inventory.Decorator.UniqueIdentifier",
786         "xyz.openbmc_project.Control.Power.Throttle"};
787     dbus::utility::getSubTree(
788         "/xyz/openbmc_project/inventory", 0, interfaces,
789         [resp, processorId, handler = std::forward<Handler>(handler)](
790             const boost::system::error_code& ec,
791             const dbus::utility::MapperGetSubTreeResponse& subtree) {
792         if (ec)
793         {
794             BMCWEB_LOG_DEBUG("DBUS response error: {}", ec);
795             messages::internalError(resp->res);
796             return;
797         }
798         for (const auto& [objectPath, serviceMap] : subtree)
799         {
800             // Ignore any objects which don't end with our desired cpu name
801             if (!objectPath.ends_with(processorId))
802             {
803                 continue;
804             }
805 
806             bool found = false;
807             // Filter out objects that don't have the CPU-specific
808             // interfaces to make sure we can return 404 on non-CPUs
809             // (e.g. /redfish/../Processors/dimm0)
810             for (const auto& [serviceName, interfaceList] : serviceMap)
811             {
812                 if (std::ranges::find_first_of(interfaceList,
813                                                processorInterfaces) !=
814                     std::end(interfaceList))
815                 {
816                     found = true;
817                     break;
818                 }
819             }
820 
821             if (!found)
822             {
823                 continue;
824             }
825 
826             // Process the first object which does match our cpu name and
827             // required interfaces, and potentially ignore any other
828             // matching objects. Assume all interfaces we want to process
829             // must be on the same object path.
830 
831             handler(objectPath, serviceMap);
832             return;
833         }
834         messages::resourceNotFound(resp->res, "Processor", processorId);
835     });
836 }
837 
838 inline void
839     getProcessorData(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
840                      const std::string& processorId,
841                      const std::string& objectPath,
842                      const dbus::utility::MapperServiceMap& serviceMap)
843 {
844     for (const auto& [serviceName, interfaceList] : serviceMap)
845     {
846         for (const auto& interface : interfaceList)
847         {
848             if (interface == "xyz.openbmc_project.Inventory.Decorator.Asset")
849             {
850                 getCpuAssetData(asyncResp, serviceName, objectPath);
851             }
852             else if (interface ==
853                      "xyz.openbmc_project.Inventory.Decorator.Revision")
854             {
855                 getCpuRevisionData(asyncResp, serviceName, objectPath);
856             }
857             else if (interface == "xyz.openbmc_project.Inventory.Item.Cpu")
858             {
859                 getCpuDataByService(asyncResp, processorId, serviceName,
860                                     objectPath);
861             }
862             else if (interface ==
863                      "xyz.openbmc_project.Inventory.Item.Accelerator")
864             {
865                 getAcceleratorDataByService(asyncResp, processorId, serviceName,
866                                             objectPath);
867             }
868             else if (
869                 interface ==
870                 "xyz.openbmc_project.Control.Processor.CurrentOperatingConfig")
871             {
872                 getCpuConfigData(asyncResp, processorId, serviceName,
873                                  objectPath);
874             }
875             else if (interface ==
876                      "xyz.openbmc_project.Inventory.Decorator.LocationCode")
877             {
878                 getCpuLocationCode(asyncResp, serviceName, objectPath);
879             }
880             else if (interface == "xyz.openbmc_project.Common.UUID")
881             {
882                 getProcessorUUID(asyncResp, serviceName, objectPath);
883             }
884             else if (interface ==
885                      "xyz.openbmc_project.Inventory.Decorator.UniqueIdentifier")
886             {
887                 getCpuUniqueId(asyncResp, serviceName, objectPath);
888             }
889             else if (interface == "xyz.openbmc_project.Control.Power.Throttle")
890             {
891                 getThrottleProperties(asyncResp, serviceName, objectPath);
892             }
893         }
894     }
895 }
896 
897 /**
898  * Request all the properties for the given D-Bus object and fill out the
899  * related entries in the Redfish OperatingConfig response.
900  *
901  * @param[in,out]   asyncResp       Async HTTP response.
902  * @param[in]       service     D-Bus service name to query.
903  * @param[in]       objPath     D-Bus object to query.
904  */
905 inline void
906     getOperatingConfigData(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
907                            const std::string& service,
908                            const std::string& objPath)
909 {
910     sdbusplus::asio::getAllProperties(
911         *crow::connections::systemBus, service, objPath,
912         "xyz.openbmc_project.Inventory.Item.Cpu.OperatingConfig",
913         [asyncResp](const boost::system::error_code& ec,
914                     const dbus::utility::DBusPropertiesMap& properties) {
915         if (ec)
916         {
917             BMCWEB_LOG_WARNING("D-Bus error: {}, {}", ec, ec.message());
918             messages::internalError(asyncResp->res);
919             return;
920         }
921 
922         const size_t* availableCoreCount = nullptr;
923         const uint32_t* baseSpeed = nullptr;
924         const uint32_t* maxJunctionTemperature = nullptr;
925         const uint32_t* maxSpeed = nullptr;
926         const uint32_t* powerLimit = nullptr;
927         const TurboProfileProperty* turboProfile = nullptr;
928         const BaseSpeedPrioritySettingsProperty* baseSpeedPrioritySettings =
929             nullptr;
930 
931         const bool success = sdbusplus::unpackPropertiesNoThrow(
932             dbus_utils::UnpackErrorPrinter(), properties, "AvailableCoreCount",
933             availableCoreCount, "BaseSpeed", baseSpeed,
934             "MaxJunctionTemperature", maxJunctionTemperature, "MaxSpeed",
935             maxSpeed, "PowerLimit", powerLimit, "TurboProfile", turboProfile,
936             "BaseSpeedPrioritySettings", baseSpeedPrioritySettings);
937 
938         if (!success)
939         {
940             messages::internalError(asyncResp->res);
941             return;
942         }
943 
944         nlohmann::json& json = asyncResp->res.jsonValue;
945 
946         if (availableCoreCount != nullptr)
947         {
948             json["TotalAvailableCoreCount"] = *availableCoreCount;
949         }
950 
951         if (baseSpeed != nullptr)
952         {
953             json["BaseSpeedMHz"] = *baseSpeed;
954         }
955 
956         if (maxJunctionTemperature != nullptr)
957         {
958             json["MaxJunctionTemperatureCelsius"] = *maxJunctionTemperature;
959         }
960 
961         if (maxSpeed != nullptr)
962         {
963             json["MaxSpeedMHz"] = *maxSpeed;
964         }
965 
966         if (powerLimit != nullptr)
967         {
968             json["TDPWatts"] = *powerLimit;
969         }
970 
971         if (turboProfile != nullptr)
972         {
973             nlohmann::json& turboArray = json["TurboProfile"];
974             turboArray = nlohmann::json::array();
975             for (const auto& [turboSpeed, coreCount] : *turboProfile)
976             {
977                 nlohmann::json::object_t turbo;
978                 turbo["ActiveCoreCount"] = coreCount;
979                 turbo["MaxSpeedMHz"] = turboSpeed;
980                 turboArray.emplace_back(std::move(turbo));
981             }
982         }
983 
984         if (baseSpeedPrioritySettings != nullptr)
985         {
986             nlohmann::json& baseSpeedArray = json["BaseSpeedPrioritySettings"];
987             baseSpeedArray = nlohmann::json::array();
988             for (const auto& [baseSpeedMhz, coreList] :
989                  *baseSpeedPrioritySettings)
990             {
991                 nlohmann::json::object_t speed;
992                 speed["CoreCount"] = coreList.size();
993                 speed["CoreIDs"] = coreList;
994                 speed["BaseSpeedMHz"] = baseSpeedMhz;
995                 baseSpeedArray.emplace_back(std::move(speed));
996             }
997         }
998     });
999 }
1000 
1001 /**
1002  * Handle the PATCH operation of the AppliedOperatingConfig property. Do basic
1003  * validation of the input data, and then set the D-Bus property.
1004  *
1005  * @param[in,out]   resp            Async HTTP response.
1006  * @param[in]       processorId     Processor's Id.
1007  * @param[in]       appliedConfigUri    New property value to apply.
1008  * @param[in]       cpuObjectPath   Path of CPU object to modify.
1009  * @param[in]       serviceMap      Service map for CPU object.
1010  */
1011 inline void patchAppliedOperatingConfig(
1012     const std::shared_ptr<bmcweb::AsyncResp>& resp,
1013     const std::string& processorId, const std::string& appliedConfigUri,
1014     const std::string& cpuObjectPath,
1015     const dbus::utility::MapperServiceMap& serviceMap)
1016 {
1017     // Check that the property even exists by checking for the interface
1018     const std::string* controlService = nullptr;
1019     for (const auto& [serviceName, interfaceList] : serviceMap)
1020     {
1021         if (std::ranges::find(interfaceList,
1022                               "xyz.openbmc_project.Control.Processor."
1023                               "CurrentOperatingConfig") != interfaceList.end())
1024         {
1025             controlService = &serviceName;
1026             break;
1027         }
1028     }
1029 
1030     if (controlService == nullptr)
1031     {
1032         messages::internalError(resp->res);
1033         return;
1034     }
1035 
1036     // Check that the config URI is a child of the cpu URI being patched.
1037     std::string expectedPrefix("/redfish/v1/Systems/system/Processors/");
1038     expectedPrefix += processorId;
1039     expectedPrefix += "/OperatingConfigs/";
1040     if (!appliedConfigUri.starts_with(expectedPrefix) ||
1041         expectedPrefix.size() == appliedConfigUri.size())
1042     {
1043         messages::propertyValueIncorrect(resp->res, "AppliedOperatingConfig",
1044                                          appliedConfigUri);
1045         return;
1046     }
1047 
1048     // Generate the D-Bus path of the OperatingConfig object, by assuming it's a
1049     // direct child of the CPU object.
1050     // Strip the expectedPrefix from the config URI to get the "filename", and
1051     // append to the CPU's path.
1052     std::string configBaseName = appliedConfigUri.substr(expectedPrefix.size());
1053     sdbusplus::message::object_path configPath(cpuObjectPath);
1054     configPath /= configBaseName;
1055 
1056     BMCWEB_LOG_INFO("Setting config to {}", configPath.str);
1057 
1058     // Set the property, with handler to check error responses
1059     setDbusProperty(
1060         resp, *controlService, cpuObjectPath,
1061         "xyz.openbmc_project.Control.Processor.CurrentOperatingConfig",
1062         "AppliedConfig", "AppliedOperatingConfig", configPath);
1063 }
1064 
1065 inline void
1066     handleProcessorHead(crow::App& app, const crow::Request& req,
1067                         const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1068                         const std::string& /* systemName */,
1069                         const std::string& /* processorId */)
1070 {
1071     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1072     {
1073         return;
1074     }
1075     asyncResp->res.addHeader(
1076         boost::beast::http::field::link,
1077         "</redfish/v1/JsonSchemas/Processor/Processor.json>; rel=describedby");
1078 }
1079 
1080 inline void handleProcessorCollectionHead(
1081     crow::App& app, const crow::Request& req,
1082     const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1083     const std::string& /* systemName */)
1084 {
1085     if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1086     {
1087         return;
1088     }
1089     asyncResp->res.addHeader(
1090         boost::beast::http::field::link,
1091         "</redfish/v1/JsonSchemas/ProcessorCollection/ProcessorCollection.json>; rel=describedby");
1092 }
1093 
1094 inline void requestRoutesOperatingConfigCollection(App& app)
1095 {
1096     BMCWEB_ROUTE(app,
1097                  "/redfish/v1/Systems/<str>/Processors/<str>/OperatingConfigs/")
1098         .privileges(redfish::privileges::getOperatingConfigCollection)
1099         .methods(boost::beast::http::verb::get)(
1100             [&app](const crow::Request& req,
1101                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1102                    const std::string& systemName, const std::string& cpuName) {
1103         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1104         {
1105             return;
1106         }
1107 
1108         if constexpr (bmcwebEnableMultiHost)
1109         {
1110             // Option currently returns no systems.  TBD
1111             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1112                                        systemName);
1113             return;
1114         }
1115 
1116         if (systemName != "system")
1117         {
1118             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1119                                        systemName);
1120             return;
1121         }
1122         asyncResp->res.jsonValue["@odata.type"] =
1123             "#OperatingConfigCollection.OperatingConfigCollection";
1124         asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
1125             "/redfish/v1/Systems/system/Processors/{}/OperatingConfigs",
1126             cpuName);
1127         asyncResp->res.jsonValue["Name"] = "Operating Config Collection";
1128 
1129         // First find the matching CPU object so we know how to
1130         // constrain our search for related Config objects.
1131         const std::array<std::string_view, 1> interfaces = {
1132             "xyz.openbmc_project.Control.Processor.CurrentOperatingConfig"};
1133         dbus::utility::getSubTreePaths(
1134             "/xyz/openbmc_project/inventory", 0, interfaces,
1135             [asyncResp, cpuName](
1136                 const boost::system::error_code& ec,
1137                 const dbus::utility::MapperGetSubTreePathsResponse& objects) {
1138             if (ec)
1139             {
1140                 BMCWEB_LOG_WARNING("D-Bus error: {}, {}", ec, ec.message());
1141                 messages::internalError(asyncResp->res);
1142                 return;
1143             }
1144 
1145             for (const std::string& object : objects)
1146             {
1147                 if (!object.ends_with(cpuName))
1148                 {
1149                     continue;
1150                 }
1151 
1152                 // Not expected that there will be multiple matching
1153                 // CPU objects, but if there are just use the first
1154                 // one.
1155 
1156                 // Use the common search routine to construct the
1157                 // Collection of all Config objects under this CPU.
1158                 constexpr std::array<std::string_view, 1> interface{
1159                     "xyz.openbmc_project.Inventory.Item.Cpu.OperatingConfig"};
1160                 collection_util::getCollectionMembers(
1161                     asyncResp,
1162                     boost::urls::format(
1163                         "/redfish/v1/Systems/system/Processors/{}/OperatingConfigs",
1164                         cpuName),
1165                     interface, object);
1166                 return;
1167             }
1168         });
1169     });
1170 }
1171 
1172 inline void requestRoutesOperatingConfig(App& app)
1173 {
1174     BMCWEB_ROUTE(
1175         app,
1176         "/redfish/v1/Systems/<str>/Processors/<str>/OperatingConfigs/<str>/")
1177         .privileges(redfish::privileges::getOperatingConfig)
1178         .methods(boost::beast::http::verb::get)(
1179             [&app](const crow::Request& req,
1180                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1181                    const std::string& systemName, const std::string& cpuName,
1182                    const std::string& configName) {
1183         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1184         {
1185             return;
1186         }
1187         if constexpr (bmcwebEnableMultiHost)
1188         {
1189             // Option currently returns no systems.  TBD
1190             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1191                                        systemName);
1192             return;
1193         }
1194 
1195         if (systemName != "system")
1196         {
1197             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1198                                        systemName);
1199             return;
1200         }
1201         // Ask for all objects implementing OperatingConfig so we can search
1202         // for one with a matching name
1203         constexpr std::array<std::string_view, 1> interfaces = {
1204             "xyz.openbmc_project.Inventory.Item.Cpu.OperatingConfig"};
1205         dbus::utility::getSubTree(
1206             "/xyz/openbmc_project/inventory", 0, interfaces,
1207             [asyncResp, cpuName, configName](
1208                 const boost::system::error_code& ec,
1209                 const dbus::utility::MapperGetSubTreeResponse& subtree) {
1210             if (ec)
1211             {
1212                 BMCWEB_LOG_WARNING("D-Bus error: {}, {}", ec, ec.message());
1213                 messages::internalError(asyncResp->res);
1214                 return;
1215             }
1216             const std::string expectedEnding = cpuName + '/' + configName;
1217             for (const auto& [objectPath, serviceMap] : subtree)
1218             {
1219                 // Ignore any configs without matching cpuX/configY
1220                 if (!objectPath.ends_with(expectedEnding) || serviceMap.empty())
1221                 {
1222                     continue;
1223                 }
1224 
1225                 nlohmann::json& json = asyncResp->res.jsonValue;
1226                 json["@odata.type"] = "#OperatingConfig.v1_0_0.OperatingConfig";
1227                 json["@odata.id"] = boost::urls::format(
1228                     "/redfish/v1/Systems/system/Processors/{}/OperatingConfigs/{}",
1229                     cpuName, configName);
1230                 json["Name"] = "Processor Profile";
1231                 json["Id"] = configName;
1232 
1233                 // Just use the first implementation of the object - not
1234                 // expected that there would be multiple matching
1235                 // services
1236                 getOperatingConfigData(asyncResp, serviceMap.begin()->first,
1237                                        objectPath);
1238                 return;
1239             }
1240             messages::resourceNotFound(asyncResp->res, "OperatingConfig",
1241                                        configName);
1242         });
1243     });
1244 }
1245 
1246 inline void requestRoutesProcessorCollection(App& app)
1247 {
1248     /**
1249      * Functions triggers appropriate requests on DBus
1250      */
1251     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Processors/")
1252         .privileges(redfish::privileges::headProcessorCollection)
1253         .methods(boost::beast::http::verb::head)(
1254             std::bind_front(handleProcessorCollectionHead, std::ref(app)));
1255 
1256     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Processors/")
1257         .privileges(redfish::privileges::getProcessorCollection)
1258         .methods(boost::beast::http::verb::get)(
1259             [&app](const crow::Request& req,
1260                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1261                    const std::string& systemName) {
1262         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1263         {
1264             return;
1265         }
1266         if constexpr (bmcwebEnableMultiHost)
1267         {
1268             // Option currently returns no systems.  TBD
1269             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1270                                        systemName);
1271             return;
1272         }
1273 
1274         if (systemName != "system")
1275         {
1276             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1277                                        systemName);
1278             return;
1279         }
1280 
1281         asyncResp->res.addHeader(
1282             boost::beast::http::field::link,
1283             "</redfish/v1/JsonSchemas/ProcessorCollection/ProcessorCollection.json>; rel=describedby");
1284 
1285         asyncResp->res.jsonValue["@odata.type"] =
1286             "#ProcessorCollection.ProcessorCollection";
1287         asyncResp->res.jsonValue["Name"] = "Processor Collection";
1288 
1289         asyncResp->res.jsonValue["@odata.id"] =
1290             "/redfish/v1/Systems/system/Processors";
1291 
1292         collection_util::getCollectionMembers(
1293             asyncResp,
1294             boost::urls::url("/redfish/v1/Systems/system/Processors"),
1295             processorInterfaces, "/xyz/openbmc_project/inventory");
1296     });
1297 }
1298 
1299 inline void requestRoutesProcessor(App& app)
1300 {
1301     /**
1302      * Functions triggers appropriate requests on DBus
1303      */
1304 
1305     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Processors/<str>/")
1306         .privileges(redfish::privileges::headProcessor)
1307         .methods(boost::beast::http::verb::head)(
1308             std::bind_front(handleProcessorHead, std::ref(app)));
1309 
1310     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Processors/<str>/")
1311         .privileges(redfish::privileges::getProcessor)
1312         .methods(boost::beast::http::verb::get)(
1313             [&app](const crow::Request& req,
1314                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1315                    const std::string& systemName,
1316                    const std::string& processorId) {
1317         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1318         {
1319             return;
1320         }
1321         if constexpr (bmcwebEnableMultiHost)
1322         {
1323             // Option currently returns no systems.  TBD
1324             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1325                                        systemName);
1326             return;
1327         }
1328         if (systemName != "system")
1329         {
1330             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1331                                        systemName);
1332             return;
1333         }
1334 
1335         asyncResp->res.addHeader(
1336             boost::beast::http::field::link,
1337             "</redfish/v1/JsonSchemas/Processor/Processor.json>; rel=describedby");
1338         asyncResp->res.jsonValue["@odata.type"] =
1339             "#Processor.v1_18_0.Processor";
1340         asyncResp->res.jsonValue["@odata.id"] = boost::urls::format(
1341             "/redfish/v1/Systems/system/Processors/{}", processorId);
1342 
1343         getProcessorObject(
1344             asyncResp, processorId,
1345             std::bind_front(getProcessorData, asyncResp, processorId));
1346     });
1347 
1348     BMCWEB_ROUTE(app, "/redfish/v1/Systems/<str>/Processors/<str>/")
1349         .privileges(redfish::privileges::patchProcessor)
1350         .methods(boost::beast::http::verb::patch)(
1351             [&app](const crow::Request& req,
1352                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
1353                    const std::string& systemName,
1354                    const std::string& processorId) {
1355         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
1356         {
1357             return;
1358         }
1359         if constexpr (bmcwebEnableMultiHost)
1360         {
1361             // Option currently returns no systems.  TBD
1362             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1363                                        systemName);
1364             return;
1365         }
1366         if (systemName != "system")
1367         {
1368             messages::resourceNotFound(asyncResp->res, "ComputerSystem",
1369                                        systemName);
1370             return;
1371         }
1372 
1373         std::optional<std::string> appliedConfigUri;
1374         if (!json_util::readJsonPatch(req, asyncResp->res,
1375                                       "AppliedOperatingConfig/@odata.id",
1376                                       appliedConfigUri))
1377         {
1378             return;
1379         }
1380 
1381         if (appliedConfigUri)
1382         {
1383             // Check for 404 and find matching D-Bus object, then run
1384             // property patch handlers if that all succeeds.
1385             getProcessorObject(asyncResp, processorId,
1386                                std::bind_front(patchAppliedOperatingConfig,
1387                                                asyncResp, processorId,
1388                                                *appliedConfigUri));
1389         }
1390     });
1391 }
1392 
1393 } // namespace redfish
1394