blob: 2459a892f50567d9dc7913bc0d85d0d7356dd1f7 [file] [log] [blame]
Cheng C Yang3e3269a2019-12-02 15:11:30 +08001/*
Zhikui Ren18a5ab92020-09-01 21:35:20 -07002// Copyright (c) 2018 Intel Corporation
Cheng C Yang3e3269a2019-12-02 15:11:30 +08003//
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
17#pragma once
Cheng C Yang43c6a1d2019-12-19 00:48:34 +080018#include "cpu.hpp"
Cheng C Yang2ca7a0f2019-12-19 10:46:42 +080019#include "dimm.hpp"
Jie Yang08e4a6d2021-08-23 13:07:41 -070020#include "pcieslot.hpp"
Zhikui Ren18a5ab92020-09-01 21:35:20 -070021#include "smbios_mdrv2.hpp"
Cheng C Yang2ca7a0f2019-12-19 10:46:42 +080022#include "system.hpp"
Prithvi Pai5af42bb2025-02-11 17:59:07 +053023#include "tpm.hpp"
Cheng C Yang3e3269a2019-12-02 15:11:30 +080024
25#include <sys/stat.h>
26#include <sys/types.h>
27#include <unistd.h>
28
Zhikui Ren18a5ab92020-09-01 21:35:20 -070029#include <boost/asio/io_context.hpp>
30#include <boost/asio/steady_timer.hpp>
31#include <boost/container/flat_map.hpp>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080032#include <phosphor-logging/elog-errors.hpp>
Alex Schendel5f2d6272021-09-22 10:35:40 -070033#include <phosphor-logging/lg2.hpp>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080034#include <phosphor-logging/log.hpp>
Zhikui Ren18a5ab92020-09-01 21:35:20 -070035#include <sdbusplus/asio/object_server.hpp>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080036#include <sdbusplus/server.hpp>
37#include <sdbusplus/timer.hpp>
Cheng C Yangeecaf822019-12-19 00:34:23 +080038#include <xyz/openbmc_project/Smbios/MDR_V2/server.hpp>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080039
Josh Lehan027277a2023-09-11 05:28:19 -070040#include <filesystem>
41#include <memory>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080042
Cheng C Yang3e3269a2019-12-02 15:11:30 +080043namespace phosphor
44{
45namespace smbios
46{
47
Josh Lehan027277a2023-09-11 05:28:19 -070048using RecordVariant =
49 std::variant<std::string, uint64_t, uint32_t, uint16_t, uint8_t>;
50
51static constexpr const char* defaultObjectPath =
52 "/xyz/openbmc_project/Smbios/MDR_V2";
Zhikui Ren18a5ab92020-09-01 21:35:20 -070053static constexpr const char* smbiosInterfaceName =
54 "xyz.openbmc_project.Smbios.GetRecordType";
Gobinath Krishnamoorthybb9c6222022-11-03 20:08:01 +000055static constexpr const char* mapperBusName = "xyz.openbmc_project.ObjectMapper";
56static constexpr const char* mapperPath = "/xyz/openbmc_project/object_mapper";
57static constexpr const char* mapperInterface =
58 "xyz.openbmc_project.ObjectMapper";
Josh Lehan027277a2023-09-11 05:28:19 -070059static constexpr const char* defaultInventoryPath =
Gobinath Krishnamoorthybb9c6222022-11-03 20:08:01 +000060 "/xyz/openbmc_project/inventory/system";
61static constexpr const char* systemInterface =
62 "xyz.openbmc_project.Inventory.Item.System";
Josh Lehanabdccd32024-01-26 15:49:50 -080063static constexpr const char* boardInterface =
64 "xyz.openbmc_project.Inventory.Item.Board";
Zhikui Ren18a5ab92020-09-01 21:35:20 -070065constexpr const int limitEntryLen = 0xff;
66
Josh Lehan027277a2023-09-11 05:28:19 -070067// Avoid putting multiple interfaces with same name on same object
68static std::string placeGetRecordType(const std::string& objectPath)
69{
70 if (objectPath != defaultObjectPath)
71 {
72 // Place GetRecordType interface on object itself, not parent
73 return objectPath;
74 }
75
76 std::filesystem::path path(objectPath);
77
78 // As there is only one default, safe to place it on the common parent
79 return path.parent_path().string();
80}
81
Jason M. Billsa3f5b382023-04-26 08:17:28 -070082class MDRV2 :
Patrick Williams77b9c472022-07-22 19:26:57 -050083 sdbusplus::server::object_t<
Jason M. Bills33ae81f2023-04-26 09:06:08 -070084 sdbusplus::server::xyz::openbmc_project::smbios::MDRV2>
Cheng C Yang3e3269a2019-12-02 15:11:30 +080085{
86 public:
Jason M. Billsa3f5b382023-04-26 08:17:28 -070087 MDRV2() = delete;
88 MDRV2(const MDRV2&) = delete;
89 MDRV2& operator=(const MDRV2&) = delete;
90 MDRV2(MDRV2&&) = delete;
91 MDRV2& operator=(MDRV2&&) = delete;
Cheng C Yang3e3269a2019-12-02 15:11:30 +080092
Josh Lehan027277a2023-09-11 05:28:19 -070093 virtual ~MDRV2()
Cheng C Yang3e3269a2019-12-02 15:11:30 +080094 {
Josh Lehan027277a2023-09-11 05:28:19 -070095 if (smbiosInterface)
96 {
97 if (objServer)
98 {
99 // Must manually undo add_interface()
100 objServer->remove_interface(smbiosInterface);
101 }
102 }
103 }
104
105 MDRV2(std::shared_ptr<boost::asio::io_context> io,
106 std::shared_ptr<sdbusplus::asio::connection> conn,
107 std::shared_ptr<sdbusplus::asio::object_server> obj,
108 std::string filePath, std::string objectPath,
109 std::string inventoryPath) :
110 sdbusplus::server::object_t<
111 sdbusplus::server::xyz::openbmc_project::smbios::MDRV2>(
112 *conn, objectPath.c_str()),
113 timer(*io), bus(conn), objServer(std::move(obj)),
114 smbiosInterface(objServer->add_interface(placeGetRecordType(objectPath),
115 smbiosInterfaceName)),
116 smbiosFilePath(std::move(filePath)),
117 smbiosObjectPath(std::move(objectPath)),
118 smbiosInventoryPath(std::move(inventoryPath))
119 {
120 lg2::info("SMBIOS data file path: {F}", "F", smbiosFilePath);
121 lg2::info("SMBIOS control object: {O}", "O", smbiosObjectPath);
122 lg2::info("SMBIOS inventory path: {I}", "I", smbiosInventoryPath);
123
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800124 smbiosDir.agentVersion = smbiosAgentVersion;
Josh Lehanf079e832023-09-19 15:52:52 -0700125 smbiosDir.dirVersion = smbiosDirVersion;
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800126 smbiosDir.dirEntries = 1;
127 directoryEntries(smbiosDir.dirEntries);
128 smbiosDir.status = 1;
129 smbiosDir.remoteDirVersion = 0;
130
131 std::copy(smbiosTableId.begin(), smbiosTableId.end(),
132 smbiosDir.dir[smbiosDirIndex].common.id.dataInfo);
133
134 smbiosDir.dir[smbiosDirIndex].dataStorage = smbiosTableStorage;
135
136 agentSynchronizeData();
Zhikui Ren18a5ab92020-09-01 21:35:20 -0700137
138 smbiosInterface->register_method("GetRecordType", [this](size_t type) {
139 return getRecordType(type);
140 });
141 smbiosInterface->initialize();
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800142 }
143
Cheng C Yangeecaf822019-12-19 00:34:23 +0800144 std::vector<uint8_t> getDirectoryInformation(uint8_t dirIndex) override;
145
146 std::vector<uint8_t> getDataInformation(uint8_t idIndex) override;
147
Patrick Williams1d73dcc2024-08-16 15:21:42 -0400148 bool sendDirectoryInformation(
149 uint8_t dirVersion, uint8_t dirIndex, uint8_t returnedEntries,
150 uint8_t remainingEntries, std::vector<uint8_t> dirEntry) override;
Cheng C Yangeecaf822019-12-19 00:34:23 +0800151
152 std::vector<uint8_t> getDataOffer() override;
153
154 bool sendDataInformation(uint8_t idIndex, uint8_t flag, uint32_t dataLen,
155 uint32_t dataVer, uint32_t timeStamp) override;
156
157 int findIdIndex(std::vector<uint8_t> dataInfo) override;
158
159 bool agentSynchronizeData() override;
160
Patrick Williams07a2c9a2025-02-01 08:23:10 -0500161 std::vector<uint32_t> synchronizeDirectoryCommonData(
162 uint8_t idIndex, uint32_t size) override;
Cheng C Yangeecaf822019-12-19 00:34:23 +0800163
164 uint8_t directoryEntries(uint8_t value) override;
165
Zhikui Ren18a5ab92020-09-01 21:35:20 -0700166 std::vector<boost::container::flat_map<std::string, RecordVariant>>
167 getRecordType(size_t type);
168
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800169 private:
Zhikui Ren18a5ab92020-09-01 21:35:20 -0700170 boost::asio::steady_timer timer;
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800171
Josh Lehan027277a2023-09-11 05:28:19 -0700172 std::shared_ptr<sdbusplus::asio::connection> bus;
173 std::shared_ptr<sdbusplus::asio::object_server> objServer;
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800174
175 Mdr2DirStruct smbiosDir;
176
Cheng C Yangec634252019-12-19 00:42:36 +0800177 bool readDataFromFlash(MDRSMBIOSHeader* mdrHdr, uint8_t* data);
Arun P. Mohanan0435a482022-02-01 21:45:55 +0530178 bool checkSMBIOSVersion(uint8_t* dataIn);
Cheng C Yangec634252019-12-19 00:42:36 +0800179
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800180 const std::array<uint8_t, 16> smbiosTableId{
181 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 0x42};
Josh Lehanc6d87a52024-02-08 19:01:29 -0800182 uint8_t smbiosTableStorage[smbiosTableStorageSize] = {};
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800183
184 bool smbiosIsUpdating(uint8_t index);
185 bool smbiosIsAvailForUpdate(uint8_t index);
186 inline uint8_t smbiosValidFlag(uint8_t index);
187 void systemInfoUpdate(void);
Cheng C Yang43c6a1d2019-12-19 00:48:34 +0800188
Jonathan Domanf2d8bb42023-07-26 10:13:34 -0700189 std::optional<size_t> getTotalCpuSlot(void);
190 std::optional<size_t> getTotalDimmSlot(void);
191 std::optional<size_t> getTotalPcieSlot(void);
Prithvi Pai5af42bb2025-02-11 17:59:07 +0530192 std::optional<size_t> getTotalTpm(void);
Cheng C Yang43c6a1d2019-12-19 00:48:34 +0800193 std::vector<std::unique_ptr<Cpu>> cpus;
Cheng C Yang8c3fab62019-12-19 00:51:06 +0800194 std::vector<std::unique_ptr<Dimm>> dimms;
Jie Yang08e4a6d2021-08-23 13:07:41 -0700195 std::vector<std::unique_ptr<Pcie>> pcies;
Prithvi Pai5af42bb2025-02-11 17:59:07 +0530196 std::vector<std::unique_ptr<Tpm>> tpms;
Cheng C Yangb4651b92019-12-19 00:59:01 +0800197 std::unique_ptr<System> system;
Zhikui Ren18a5ab92020-09-01 21:35:20 -0700198 std::shared_ptr<sdbusplus::asio::dbus_interface> smbiosInterface;
Josh Lehan027277a2023-09-11 05:28:19 -0700199
200 std::string smbiosFilePath;
201 std::string smbiosObjectPath;
202 std::string smbiosInventoryPath;
203 std::unique_ptr<sdbusplus::bus::match_t> motherboardConfigMatch;
Cheng C Yang3e3269a2019-12-02 15:11:30 +0800204};
205
206} // namespace smbios
207} // namespace phosphor