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