1de54f486SWilly Tu /*
2de54f486SWilly Tu // Copyright (c) 2017-2019 Intel Corporation
3de54f486SWilly Tu //
4de54f486SWilly Tu // Licensed under the Apache License, Version 2.0 (the "License");
5de54f486SWilly Tu // you may not use this file except in compliance with the License.
6de54f486SWilly Tu // You may obtain a copy of the License at
7de54f486SWilly Tu //
8de54f486SWilly Tu // http://www.apache.org/licenses/LICENSE-2.0
9de54f486SWilly Tu //
10de54f486SWilly Tu // Unless required by applicable law or agreed to in writing, software
11de54f486SWilly Tu // distributed under the License is distributed on an "AS IS" BASIS,
12de54f486SWilly Tu // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13de54f486SWilly Tu // See the License for the specific language governing permissions and
14de54f486SWilly Tu // limitations under the License.
15de54f486SWilly Tu */
16de54f486SWilly Tu
17de54f486SWilly Tu #include "dbus-sdr/storagecommands.hpp"
18de54f486SWilly Tu
19de54f486SWilly Tu #include "dbus-sdr/sdrutils.hpp"
20de54f486SWilly Tu #include "selutility.hpp"
21de54f486SWilly Tu
22de54f486SWilly Tu #include <boost/algorithm/string.hpp>
239860505fSEd Tanous #include <boost/asio/detached.hpp>
24de54f486SWilly Tu #include <boost/container/flat_map.hpp>
25de54f486SWilly Tu #include <boost/process.hpp>
26de54f486SWilly Tu #include <ipmid/api.hpp>
27de54f486SWilly Tu #include <ipmid/message.hpp>
28de54f486SWilly Tu #include <ipmid/types.hpp>
29*de1420dcSGeorge Liu #include <ipmid/utils.hpp>
30de6694e2SGeorge Liu #include <phosphor-logging/lg2.hpp>
31de54f486SWilly Tu #include <sdbusplus/message/types.hpp>
32de54f486SWilly Tu #include <sdbusplus/timer.hpp>
33fbc6c9d7SPatrick Williams
34fbc6c9d7SPatrick Williams #include <filesystem>
35fbc6c9d7SPatrick Williams #include <fstream>
36fbc6c9d7SPatrick Williams #include <functional>
37fbc6c9d7SPatrick Williams #include <iostream>
38de54f486SWilly Tu #include <stdexcept>
39de54f486SWilly Tu #include <string_view>
40de54f486SWilly Tu
41de54f486SWilly Tu static constexpr bool DEBUG = false;
42de54f486SWilly Tu
43de54f486SWilly Tu namespace dynamic_sensors::ipmi::sel
44de54f486SWilly Tu {
45de54f486SWilly Tu static const std::filesystem::path selLogDir = "/var/log";
46de54f486SWilly Tu static const std::string selLogFilename = "ipmi_sel";
47de54f486SWilly Tu
getFileTimestamp(const std::filesystem::path & file)48de54f486SWilly Tu static int getFileTimestamp(const std::filesystem::path& file)
49de54f486SWilly Tu {
50de54f486SWilly Tu struct stat st;
51de54f486SWilly Tu
52de54f486SWilly Tu if (stat(file.c_str(), &st) >= 0)
53de54f486SWilly Tu {
54de54f486SWilly Tu return st.st_mtime;
55de54f486SWilly Tu }
56de54f486SWilly Tu return ::ipmi::sel::invalidTimeStamp;
57de54f486SWilly Tu }
58de54f486SWilly Tu
59de54f486SWilly Tu namespace erase_time
60de54f486SWilly Tu {
61de54f486SWilly Tu static constexpr const char* selEraseTimestamp = "/var/lib/ipmi/sel_erase_time";
62de54f486SWilly Tu
get()63de54f486SWilly Tu int get()
64de54f486SWilly Tu {
65de54f486SWilly Tu return getFileTimestamp(selEraseTimestamp);
66de54f486SWilly Tu }
67de54f486SWilly Tu } // namespace erase_time
68de54f486SWilly Tu } // namespace dynamic_sensors::ipmi::sel
69de54f486SWilly Tu
70de54f486SWilly Tu namespace ipmi
71de54f486SWilly Tu {
72de54f486SWilly Tu
73de54f486SWilly Tu namespace storage
74de54f486SWilly Tu {
75de54f486SWilly Tu
76de54f486SWilly Tu constexpr static const size_t maxFruSdrNameSize = 16;
77de54f486SWilly Tu using ObjectType =
78de54f486SWilly Tu boost::container::flat_map<std::string,
79de54f486SWilly Tu boost::container::flat_map<std::string, Value>>;
80de54f486SWilly Tu using ManagedObjectType =
81de54f486SWilly Tu boost::container::flat_map<sdbusplus::message::object_path, ObjectType>;
82de54f486SWilly Tu using ManagedEntry = std::pair<sdbusplus::message::object_path, ObjectType>;
83de54f486SWilly Tu
84de54f486SWilly Tu constexpr static const char* fruDeviceServiceName =
85de54f486SWilly Tu "xyz.openbmc_project.FruDevice";
86de54f486SWilly Tu constexpr static const size_t writeTimeoutSeconds = 10;
87de54f486SWilly Tu constexpr static const char* chassisTypeRackMount = "23";
88f38f9d1bSZev Weiss constexpr static const char* chassisTypeMainServer = "17";
89de54f486SWilly Tu
90de54f486SWilly Tu static std::vector<uint8_t> fruCache;
91d90b3f08Skrishnar4 static constexpr uint16_t invalidBus = 0xFFFF;
92d90b3f08Skrishnar4 static constexpr uint8_t invalidAddr = 0xFF;
93b99de182SJohnathan Mantey static constexpr uint8_t typeASCIILatin8 = 0xC0;
94d90b3f08Skrishnar4 static uint16_t cacheBus = invalidBus;
95d90b3f08Skrishnar4 static uint8_t cacheAddr = invalidAddr;
96de54f486SWilly Tu static uint8_t lastDevId = 0xFF;
97de54f486SWilly Tu
98d90b3f08Skrishnar4 static uint16_t writeBus = invalidBus;
99d90b3f08Skrishnar4 static uint8_t writeAddr = invalidAddr;
100de54f486SWilly Tu
10195655220SPatrick Williams std::unique_ptr<sdbusplus::Timer> writeTimer = nullptr;
1025d82f474SPatrick Williams static std::vector<sdbusplus::bus::match_t> fruMatches;
103de54f486SWilly Tu
104de54f486SWilly Tu ManagedObjectType frus;
105de54f486SWilly Tu
106de54f486SWilly Tu // we unfortunately have to build a map of hashes in case there is a
107de54f486SWilly Tu // collision to verify our dev-id
108d90b3f08Skrishnar4 boost::container::flat_map<uint8_t, std::pair<uint16_t, uint8_t>> deviceHashes;
109de54f486SWilly Tu void registerStorageFunctions() __attribute__((constructor));
110de54f486SWilly Tu
writeFru(const std::vector<uint8_t> & fru)11148fe64e9SWilly Tu bool writeFru(const std::vector<uint8_t>& fru)
112de54f486SWilly Tu {
113d90b3f08Skrishnar4 if (writeBus == invalidBus && writeAddr == invalidAddr)
114de54f486SWilly Tu {
115de54f486SWilly Tu return true;
116de54f486SWilly Tu }
117d934be9dSThang Tran lastDevId = 0xFF;
118de54f486SWilly Tu std::shared_ptr<sdbusplus::asio::connection> dbus = getSdBus();
1195d82f474SPatrick Williams sdbusplus::message_t writeFru = dbus->new_method_call(
120de54f486SWilly Tu fruDeviceServiceName, "/xyz/openbmc_project/FruDevice",
121de54f486SWilly Tu "xyz.openbmc_project.FruDeviceManager", "WriteFru");
12248fe64e9SWilly Tu writeFru.append(writeBus, writeAddr, fru);
123de54f486SWilly Tu try
124de54f486SWilly Tu {
1255d82f474SPatrick Williams sdbusplus::message_t writeFruResp = dbus->call(writeFru);
126de54f486SWilly Tu }
127a2ad2da8SPatrick Williams catch (const sdbusplus::exception_t&)
128de54f486SWilly Tu {
129de54f486SWilly Tu // todo: log sel?
130de6694e2SGeorge Liu lg2::error("error writing fru");
131de54f486SWilly Tu return false;
132de54f486SWilly Tu }
133d90b3f08Skrishnar4 writeBus = invalidBus;
134d90b3f08Skrishnar4 writeAddr = invalidAddr;
135de54f486SWilly Tu return true;
136de54f486SWilly Tu }
137de54f486SWilly Tu
writeFruCache()13852535624SWilliam A. Kennington III void writeFruCache()
13948fe64e9SWilly Tu {
14052535624SWilliam A. Kennington III writeFru(fruCache);
14148fe64e9SWilly Tu }
14248fe64e9SWilly Tu
createTimers()143de54f486SWilly Tu void createTimers()
144de54f486SWilly Tu {
14595655220SPatrick Williams writeTimer = std::make_unique<sdbusplus::Timer>(writeFruCache);
146de54f486SWilly Tu }
147de54f486SWilly Tu
recalculateHashes()148de54f486SWilly Tu void recalculateHashes()
149de54f486SWilly Tu {
150de54f486SWilly Tu deviceHashes.clear();
151de54f486SWilly Tu // hash the object paths to create unique device id's. increment on
152de54f486SWilly Tu // collision
153de54f486SWilly Tu std::hash<std::string> hasher;
154de54f486SWilly Tu for (const auto& fru : frus)
155de54f486SWilly Tu {
156de54f486SWilly Tu auto fruIface = fru.second.find("xyz.openbmc_project.FruDevice");
157de54f486SWilly Tu if (fruIface == fru.second.end())
158de54f486SWilly Tu {
159de54f486SWilly Tu continue;
160de54f486SWilly Tu }
161de54f486SWilly Tu
162de54f486SWilly Tu auto busFind = fruIface->second.find("BUS");
163de54f486SWilly Tu auto addrFind = fruIface->second.find("ADDRESS");
164de54f486SWilly Tu if (busFind == fruIface->second.end() ||
165de54f486SWilly Tu addrFind == fruIface->second.end())
166de54f486SWilly Tu {
167de6694e2SGeorge Liu lg2::info("fru device missing Bus or Address, fru: {FRU}", "FRU",
168de6694e2SGeorge Liu fru.first.str);
169de54f486SWilly Tu continue;
170de54f486SWilly Tu }
171de54f486SWilly Tu
172d90b3f08Skrishnar4 uint16_t fruBus = std::get<uint32_t>(busFind->second);
173de54f486SWilly Tu uint8_t fruAddr = std::get<uint32_t>(addrFind->second);
174de54f486SWilly Tu auto chassisFind = fruIface->second.find("CHASSIS_TYPE");
175de54f486SWilly Tu std::string chassisType;
176de54f486SWilly Tu if (chassisFind != fruIface->second.end())
177de54f486SWilly Tu {
178de54f486SWilly Tu chassisType = std::get<std::string>(chassisFind->second);
179de54f486SWilly Tu }
180de54f486SWilly Tu
181de54f486SWilly Tu uint8_t fruHash = 0;
182f38f9d1bSZev Weiss if (chassisType.compare(chassisTypeRackMount) != 0 &&
183f38f9d1bSZev Weiss chassisType.compare(chassisTypeMainServer) != 0)
184de54f486SWilly Tu {
185de54f486SWilly Tu fruHash = hasher(fru.first.str);
186de54f486SWilly Tu // can't be 0xFF based on spec, and 0 is reserved for baseboard
187de54f486SWilly Tu if (fruHash == 0 || fruHash == 0xFF)
188de54f486SWilly Tu {
189de54f486SWilly Tu fruHash = 1;
190de54f486SWilly Tu }
191de54f486SWilly Tu }
192d90b3f08Skrishnar4 std::pair<uint16_t, uint8_t> newDev(fruBus, fruAddr);
193de54f486SWilly Tu
194de54f486SWilly Tu bool emplacePassed = false;
195de54f486SWilly Tu while (!emplacePassed)
196de54f486SWilly Tu {
197de54f486SWilly Tu auto resp = deviceHashes.emplace(fruHash, newDev);
198de54f486SWilly Tu emplacePassed = resp.second;
199de54f486SWilly Tu if (!emplacePassed)
200de54f486SWilly Tu {
201de54f486SWilly Tu fruHash++;
202de54f486SWilly Tu // can't be 0xFF based on spec, and 0 is reserved for
203de54f486SWilly Tu // baseboard
204de54f486SWilly Tu if (fruHash == 0XFF)
205de54f486SWilly Tu {
206de54f486SWilly Tu fruHash = 0x1;
207de54f486SWilly Tu }
208de54f486SWilly Tu }
209de54f486SWilly Tu }
210de54f486SWilly Tu }
211de54f486SWilly Tu }
212de54f486SWilly Tu
replaceCacheFru(const std::shared_ptr<sdbusplus::asio::connection> & bus,boost::asio::yield_context & yield,const std::optional<std::string> & path=std::nullopt)21311d68897SWilly Tu void replaceCacheFru(
21411d68897SWilly Tu const std::shared_ptr<sdbusplus::asio::connection>& bus,
215de54f486SWilly Tu boost::asio::yield_context& yield,
21611d68897SWilly Tu [[maybe_unused]] const std::optional<std::string>& path = std::nullopt)
217de54f486SWilly Tu {
218de54f486SWilly Tu boost::system::error_code ec;
219de54f486SWilly Tu
220de54f486SWilly Tu frus = bus->yield_method_call<ManagedObjectType>(
221de54f486SWilly Tu yield, ec, fruDeviceServiceName, "/",
222de54f486SWilly Tu "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
223de54f486SWilly Tu if (ec)
224de54f486SWilly Tu {
225de6694e2SGeorge Liu lg2::error("GetMangagedObjects for replaceCacheFru failed: {ERROR}",
226de6694e2SGeorge Liu "ERROR", ec.message());
227de54f486SWilly Tu
228de54f486SWilly Tu return;
229de54f486SWilly Tu }
230de54f486SWilly Tu recalculateHashes();
231de54f486SWilly Tu }
232de54f486SWilly Tu
getFru(ipmi::Context::ptr ctx,uint8_t devId)23369b4c281SPatrick Williams std::pair<ipmi::Cc, std::vector<uint8_t>> getFru(ipmi::Context::ptr ctx,
23469b4c281SPatrick Williams uint8_t devId)
235de54f486SWilly Tu {
236de54f486SWilly Tu if (lastDevId == devId && devId != 0xFF)
237de54f486SWilly Tu {
23848fe64e9SWilly Tu return {ipmi::ccSuccess, fruCache};
239de54f486SWilly Tu }
240de54f486SWilly Tu
241de54f486SWilly Tu auto deviceFind = deviceHashes.find(devId);
242de54f486SWilly Tu if (deviceFind == deviceHashes.end())
243de54f486SWilly Tu {
24448fe64e9SWilly Tu return {IPMI_CC_SENSOR_INVALID, {}};
245de54f486SWilly Tu }
246de54f486SWilly Tu
247de54f486SWilly Tu cacheBus = deviceFind->second.first;
248de54f486SWilly Tu cacheAddr = deviceFind->second.second;
249de54f486SWilly Tu
250de54f486SWilly Tu boost::system::error_code ec;
251*de1420dcSGeorge Liu std::vector<uint8_t> fru = ipmi::callDbusMethod<std::vector<uint8_t>>(
252*de1420dcSGeorge Liu ctx, ec, fruDeviceServiceName, "/xyz/openbmc_project/FruDevice",
253de54f486SWilly Tu "xyz.openbmc_project.FruDeviceManager", "GetRawFru", cacheBus,
254de54f486SWilly Tu cacheAddr);
255*de1420dcSGeorge Liu
256de54f486SWilly Tu if (ec)
257de54f486SWilly Tu {
258de6694e2SGeorge Liu lg2::error("Couldn't get raw fru: {ERROR}", "ERROR", ec.message());
259de54f486SWilly Tu
260d90b3f08Skrishnar4 cacheBus = invalidBus;
261d90b3f08Skrishnar4 cacheAddr = invalidAddr;
26248fe64e9SWilly Tu return {ipmi::ccResponseError, {}};
263de54f486SWilly Tu }
264de54f486SWilly Tu
26548fe64e9SWilly Tu fruCache.clear();
266de54f486SWilly Tu lastDevId = devId;
26748fe64e9SWilly Tu fruCache = fru;
26848fe64e9SWilly Tu
26948fe64e9SWilly Tu return {ipmi::ccSuccess, fru};
270de54f486SWilly Tu }
271de54f486SWilly Tu
writeFruIfRunning()272de54f486SWilly Tu void writeFruIfRunning()
273de54f486SWilly Tu {
274de54f486SWilly Tu if (!writeTimer->isRunning())
275de54f486SWilly Tu {
276de54f486SWilly Tu return;
277de54f486SWilly Tu }
278de54f486SWilly Tu writeTimer->stop();
27948fe64e9SWilly Tu writeFruCache();
280de54f486SWilly Tu }
281de54f486SWilly Tu
startMatch(void)282de54f486SWilly Tu void startMatch(void)
283de54f486SWilly Tu {
284de54f486SWilly Tu if (fruMatches.size())
285de54f486SWilly Tu {
286de54f486SWilly Tu return;
287de54f486SWilly Tu }
288de54f486SWilly Tu
289de54f486SWilly Tu fruMatches.reserve(2);
290de54f486SWilly Tu
291de54f486SWilly Tu auto bus = getSdBus();
2921318a5edSPatrick Williams fruMatches.emplace_back(
2931318a5edSPatrick Williams *bus,
294de54f486SWilly Tu "type='signal',arg0path='/xyz/openbmc_project/"
295de54f486SWilly Tu "FruDevice/',member='InterfacesAdded'",
2965d82f474SPatrick Williams [](sdbusplus::message_t& message) {
297de54f486SWilly Tu sdbusplus::message::object_path path;
298de54f486SWilly Tu ObjectType object;
299de54f486SWilly Tu try
300de54f486SWilly Tu {
301de54f486SWilly Tu message.read(path, object);
302de54f486SWilly Tu }
303a2ad2da8SPatrick Williams catch (const sdbusplus::exception_t&)
304de54f486SWilly Tu {
305de54f486SWilly Tu return;
306de54f486SWilly Tu }
307fbc6c9d7SPatrick Williams auto findType = object.find("xyz.openbmc_project.FruDevice");
308de54f486SWilly Tu if (findType == object.end())
309de54f486SWilly Tu {
310de54f486SWilly Tu return;
311de54f486SWilly Tu }
312de54f486SWilly Tu writeFruIfRunning();
313de54f486SWilly Tu frus[path] = object;
314de54f486SWilly Tu recalculateHashes();
315de54f486SWilly Tu lastDevId = 0xFF;
316de54f486SWilly Tu });
317de54f486SWilly Tu
3181318a5edSPatrick Williams fruMatches.emplace_back(
3191318a5edSPatrick Williams *bus,
320de54f486SWilly Tu "type='signal',arg0path='/xyz/openbmc_project/"
321de54f486SWilly Tu "FruDevice/',member='InterfacesRemoved'",
3225d82f474SPatrick Williams [](sdbusplus::message_t& message) {
323de54f486SWilly Tu sdbusplus::message::object_path path;
324de54f486SWilly Tu std::set<std::string> interfaces;
325de54f486SWilly Tu try
326de54f486SWilly Tu {
327de54f486SWilly Tu message.read(path, interfaces);
328de54f486SWilly Tu }
329a2ad2da8SPatrick Williams catch (const sdbusplus::exception_t&)
330de54f486SWilly Tu {
331de54f486SWilly Tu return;
332de54f486SWilly Tu }
333fbc6c9d7SPatrick Williams auto findType = interfaces.find("xyz.openbmc_project.FruDevice");
334de54f486SWilly Tu if (findType == interfaces.end())
335de54f486SWilly Tu {
336de54f486SWilly Tu return;
337de54f486SWilly Tu }
338de54f486SWilly Tu writeFruIfRunning();
339de54f486SWilly Tu frus.erase(path);
340de54f486SWilly Tu recalculateHashes();
341de54f486SWilly Tu lastDevId = 0xFF;
342de54f486SWilly Tu });
343de54f486SWilly Tu
344de54f486SWilly Tu // call once to populate
3459860505fSEd Tanous boost::asio::spawn(
3469860505fSEd Tanous *getIoContext(),
347531223fbSJayanth Othayoth [](boost::asio::yield_context yield) {
348de54f486SWilly Tu replaceCacheFru(getSdBus(), yield);
349531223fbSJayanth Othayoth },
3509860505fSEd Tanous boost::asio::detached);
351de54f486SWilly Tu }
352de54f486SWilly Tu
353de54f486SWilly Tu /** @brief implements the read FRU data command
354de54f486SWilly Tu * @param fruDeviceId - FRU Device ID
355de54f486SWilly Tu * @param fruInventoryOffset - FRU Inventory Offset to write
356de54f486SWilly Tu * @param countToRead - Count to read
357de54f486SWilly Tu *
358de54f486SWilly Tu * @returns ipmi completion code plus response data
359de54f486SWilly Tu * - countWritten - Count written
360de54f486SWilly Tu */
361de54f486SWilly Tu ipmi::RspType<uint8_t, // Count
362de54f486SWilly Tu std::vector<uint8_t> // Requested data
363de54f486SWilly Tu >
ipmiStorageReadFruData(ipmi::Context::ptr ctx,uint8_t fruDeviceId,uint16_t fruInventoryOffset,uint8_t countToRead)364de54f486SWilly Tu ipmiStorageReadFruData(ipmi::Context::ptr ctx, uint8_t fruDeviceId,
365de54f486SWilly Tu uint16_t fruInventoryOffset, uint8_t countToRead)
366de54f486SWilly Tu {
367de54f486SWilly Tu if (fruDeviceId == 0xFF)
368de54f486SWilly Tu {
369de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
370de54f486SWilly Tu }
371de54f486SWilly Tu
37248fe64e9SWilly Tu auto [status, fru] = getFru(ctx, fruDeviceId);
373de54f486SWilly Tu if (status != ipmi::ccSuccess)
374de54f486SWilly Tu {
375de54f486SWilly Tu return ipmi::response(status);
376de54f486SWilly Tu }
377de54f486SWilly Tu
378de54f486SWilly Tu size_t fromFruByteLen = 0;
37948fe64e9SWilly Tu if (countToRead + fruInventoryOffset < fru.size())
380de54f486SWilly Tu {
381de54f486SWilly Tu fromFruByteLen = countToRead;
382de54f486SWilly Tu }
38348fe64e9SWilly Tu else if (fru.size() > fruInventoryOffset)
384de54f486SWilly Tu {
38548fe64e9SWilly Tu fromFruByteLen = fru.size() - fruInventoryOffset;
386de54f486SWilly Tu }
387de54f486SWilly Tu else
388de54f486SWilly Tu {
389de54f486SWilly Tu return ipmi::responseReqDataLenExceeded();
390de54f486SWilly Tu }
391de54f486SWilly Tu
392de54f486SWilly Tu std::vector<uint8_t> requestedData;
393de54f486SWilly Tu
39448fe64e9SWilly Tu requestedData.insert(requestedData.begin(),
39548fe64e9SWilly Tu fru.begin() + fruInventoryOffset,
39648fe64e9SWilly Tu fru.begin() + fruInventoryOffset + fromFruByteLen);
397de54f486SWilly Tu
398de54f486SWilly Tu return ipmi::responseSuccess(static_cast<uint8_t>(requestedData.size()),
399de54f486SWilly Tu requestedData);
400de54f486SWilly Tu }
401de54f486SWilly Tu
402de54f486SWilly Tu /** @brief implements the write FRU data command
403de54f486SWilly Tu * @param fruDeviceId - FRU Device ID
404de54f486SWilly Tu * @param fruInventoryOffset - FRU Inventory Offset to write
405de54f486SWilly Tu * @param dataToWrite - Data to write
406de54f486SWilly Tu *
407de54f486SWilly Tu * @returns ipmi completion code plus response data
408de54f486SWilly Tu * - countWritten - Count written
409de54f486SWilly Tu */
ipmiStorageWriteFruData(ipmi::Context::ptr ctx,uint8_t fruDeviceId,uint16_t fruInventoryOffset,std::vector<uint8_t> & dataToWrite)4101318a5edSPatrick Williams ipmi::RspType<uint8_t> ipmiStorageWriteFruData(
4111318a5edSPatrick Williams ipmi::Context::ptr ctx, uint8_t fruDeviceId, uint16_t fruInventoryOffset,
412de54f486SWilly Tu std::vector<uint8_t>& dataToWrite)
413de54f486SWilly Tu {
414de54f486SWilly Tu if (fruDeviceId == 0xFF)
415de54f486SWilly Tu {
416de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
417de54f486SWilly Tu }
418de54f486SWilly Tu
419de54f486SWilly Tu size_t writeLen = dataToWrite.size();
420de54f486SWilly Tu
42148fe64e9SWilly Tu auto [status, fru] = getFru(ctx, fruDeviceId);
422de54f486SWilly Tu if (status != ipmi::ccSuccess)
423de54f486SWilly Tu {
424de54f486SWilly Tu return ipmi::response(status);
425de54f486SWilly Tu }
426de54f486SWilly Tu size_t lastWriteAddr = fruInventoryOffset + writeLen;
42748fe64e9SWilly Tu if (fru.size() < lastWriteAddr)
428de54f486SWilly Tu {
42948fe64e9SWilly Tu fru.resize(fruInventoryOffset + writeLen);
430de54f486SWilly Tu }
431de54f486SWilly Tu
432de54f486SWilly Tu std::copy(dataToWrite.begin(), dataToWrite.begin() + writeLen,
43348fe64e9SWilly Tu fru.begin() + fruInventoryOffset);
434de54f486SWilly Tu
435de54f486SWilly Tu bool atEnd = false;
436de54f486SWilly Tu
43748fe64e9SWilly Tu if (fru.size() >= sizeof(FRUHeader))
438de54f486SWilly Tu {
43948fe64e9SWilly Tu FRUHeader* header = reinterpret_cast<FRUHeader*>(fru.data());
440de54f486SWilly Tu
441de54f486SWilly Tu size_t areaLength = 0;
442de54f486SWilly Tu size_t lastRecordStart = std::max(
443de54f486SWilly Tu {header->internalOffset, header->chassisOffset, header->boardOffset,
444de54f486SWilly Tu header->productOffset, header->multiRecordOffset});
445de54f486SWilly Tu lastRecordStart *= 8; // header starts in are multiples of 8 bytes
446de54f486SWilly Tu
447de54f486SWilly Tu if (header->multiRecordOffset)
448de54f486SWilly Tu {
449de54f486SWilly Tu // This FRU has a MultiRecord Area
450de54f486SWilly Tu uint8_t endOfList = 0;
451de54f486SWilly Tu // Walk the MultiRecord headers until the last record
452de54f486SWilly Tu while (!endOfList)
453de54f486SWilly Tu {
454de54f486SWilly Tu // The MSB in the second byte of the MultiRecord header signals
455de54f486SWilly Tu // "End of list"
45648fe64e9SWilly Tu endOfList = fru[lastRecordStart + 1] & 0x80;
457de54f486SWilly Tu // Third byte in the MultiRecord header is the length
45848fe64e9SWilly Tu areaLength = fru[lastRecordStart + 2];
459de54f486SWilly Tu // This length is in bytes (not 8 bytes like other headers)
460de54f486SWilly Tu areaLength += 5; // The length omits the 5 byte header
461de54f486SWilly Tu if (!endOfList)
462de54f486SWilly Tu {
463de54f486SWilly Tu // Next MultiRecord header
464de54f486SWilly Tu lastRecordStart += areaLength;
465de54f486SWilly Tu }
466de54f486SWilly Tu }
467de54f486SWilly Tu }
468de54f486SWilly Tu else
469de54f486SWilly Tu {
470de54f486SWilly Tu // This FRU does not have a MultiRecord Area
471de54f486SWilly Tu // Get the length of the area in multiples of 8 bytes
472de54f486SWilly Tu if (lastWriteAddr > (lastRecordStart + 1))
473de54f486SWilly Tu {
474de54f486SWilly Tu // second byte in record area is the length
47548fe64e9SWilly Tu areaLength = fru[lastRecordStart + 1];
476de54f486SWilly Tu areaLength *= 8; // it is in multiples of 8 bytes
477de54f486SWilly Tu }
478de54f486SWilly Tu }
479de54f486SWilly Tu if (lastWriteAddr >= (areaLength + lastRecordStart))
480de54f486SWilly Tu {
481de54f486SWilly Tu atEnd = true;
482de54f486SWilly Tu }
483de54f486SWilly Tu }
484de54f486SWilly Tu uint8_t countWritten = 0;
485de54f486SWilly Tu
486de54f486SWilly Tu writeBus = cacheBus;
487de54f486SWilly Tu writeAddr = cacheAddr;
488de54f486SWilly Tu if (atEnd)
489de54f486SWilly Tu {
490de54f486SWilly Tu // cancel timer, we're at the end so might as well send it
491de54f486SWilly Tu writeTimer->stop();
49248fe64e9SWilly Tu if (!writeFru(fru))
493de54f486SWilly Tu {
494de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
495de54f486SWilly Tu }
49648fe64e9SWilly Tu countWritten = std::min(fru.size(), static_cast<size_t>(0xFF));
497de54f486SWilly Tu }
498de54f486SWilly Tu else
499de54f486SWilly Tu {
500548d1a27SSui Chen fruCache = fru; // Write-back
501de54f486SWilly Tu // start a timer, if no further data is sent to check to see if it is
502de54f486SWilly Tu // valid
503de54f486SWilly Tu writeTimer->start(std::chrono::duration_cast<std::chrono::microseconds>(
504de54f486SWilly Tu std::chrono::seconds(writeTimeoutSeconds)));
505de54f486SWilly Tu countWritten = 0;
506de54f486SWilly Tu }
507de54f486SWilly Tu
508de54f486SWilly Tu return ipmi::responseSuccess(countWritten);
509de54f486SWilly Tu }
510de54f486SWilly Tu
511de54f486SWilly Tu /** @brief implements the get FRU inventory area info command
512de54f486SWilly Tu * @param fruDeviceId - FRU Device ID
513de54f486SWilly Tu *
514de54f486SWilly Tu * @returns IPMI completion code plus response data
515de54f486SWilly Tu * - inventorySize - Number of possible allocation units
516de54f486SWilly Tu * - accessType - Allocation unit size in bytes.
517de54f486SWilly Tu */
518de54f486SWilly Tu ipmi::RspType<uint16_t, // inventorySize
519de54f486SWilly Tu uint8_t> // accessType
ipmiStorageGetFruInvAreaInfo(ipmi::Context::ptr ctx,uint8_t fruDeviceId)520de54f486SWilly Tu ipmiStorageGetFruInvAreaInfo(ipmi::Context::ptr ctx, uint8_t fruDeviceId)
521de54f486SWilly Tu {
522de54f486SWilly Tu if (fruDeviceId == 0xFF)
523de54f486SWilly Tu {
524de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
525de54f486SWilly Tu }
526de54f486SWilly Tu
52748fe64e9SWilly Tu auto [ret, fru] = getFru(ctx, fruDeviceId);
528de54f486SWilly Tu if (ret != ipmi::ccSuccess)
529de54f486SWilly Tu {
530de54f486SWilly Tu return ipmi::response(ret);
531de54f486SWilly Tu }
532de54f486SWilly Tu
533de54f486SWilly Tu constexpr uint8_t accessType =
534de54f486SWilly Tu static_cast<uint8_t>(GetFRUAreaAccessType::byte);
535de54f486SWilly Tu
53648fe64e9SWilly Tu return ipmi::responseSuccess(fru.size(), accessType);
537de54f486SWilly Tu }
538de54f486SWilly Tu
getFruSdrCount(ipmi::Context::ptr,size_t & count)53911d68897SWilly Tu ipmi_ret_t getFruSdrCount(ipmi::Context::ptr, size_t& count)
540de54f486SWilly Tu {
541de54f486SWilly Tu count = deviceHashes.size();
542de54f486SWilly Tu return IPMI_CC_OK;
543de54f486SWilly Tu }
544de54f486SWilly Tu
getFruSdrs(ipmi::Context::ptr ctx,size_t index,get_sdr::SensorDataFruRecord & resp)54523a722caSJohnathan Mantey ipmi_ret_t getFruSdrs([[maybe_unused]] ipmi::Context::ptr ctx, size_t index,
546de54f486SWilly Tu get_sdr::SensorDataFruRecord& resp)
547de54f486SWilly Tu {
548de54f486SWilly Tu if (deviceHashes.size() < index)
549de54f486SWilly Tu {
550de54f486SWilly Tu return IPMI_CC_INVALID_FIELD_REQUEST;
551de54f486SWilly Tu }
552de54f486SWilly Tu auto device = deviceHashes.begin() + index;
553d90b3f08Skrishnar4 uint16_t& bus = device->second.first;
554de54f486SWilly Tu uint8_t& address = device->second.second;
555de54f486SWilly Tu
556de54f486SWilly Tu boost::container::flat_map<std::string, Value>* fruData = nullptr;
5571318a5edSPatrick Williams auto fru = std::find_if(
5581318a5edSPatrick Williams frus.begin(), frus.end(),
559de54f486SWilly Tu [bus, address, &fruData](ManagedEntry& entry) {
5601318a5edSPatrick Williams auto findFruDevice =
5611318a5edSPatrick Williams entry.second.find("xyz.openbmc_project.FruDevice");
562de54f486SWilly Tu if (findFruDevice == entry.second.end())
563de54f486SWilly Tu {
564de54f486SWilly Tu return false;
565de54f486SWilly Tu }
566de54f486SWilly Tu fruData = &(findFruDevice->second);
567de54f486SWilly Tu auto findBus = findFruDevice->second.find("BUS");
568fbc6c9d7SPatrick Williams auto findAddress = findFruDevice->second.find("ADDRESS");
569de54f486SWilly Tu if (findBus == findFruDevice->second.end() ||
570de54f486SWilly Tu findAddress == findFruDevice->second.end())
571de54f486SWilly Tu {
572de54f486SWilly Tu return false;
573de54f486SWilly Tu }
574de54f486SWilly Tu if (std::get<uint32_t>(findBus->second) != bus)
575de54f486SWilly Tu {
576de54f486SWilly Tu return false;
577de54f486SWilly Tu }
578de54f486SWilly Tu if (std::get<uint32_t>(findAddress->second) != address)
579de54f486SWilly Tu {
580de54f486SWilly Tu return false;
581de54f486SWilly Tu }
582de54f486SWilly Tu return true;
583de54f486SWilly Tu });
584de54f486SWilly Tu if (fru == frus.end())
585de54f486SWilly Tu {
586de54f486SWilly Tu return IPMI_CC_RESPONSE_ERROR;
587de54f486SWilly Tu }
588eacad3c4SShakeeb Pasha std::string name;
589de54f486SWilly Tu
590de54f486SWilly Tu #ifdef USING_ENTITY_MANAGER_DECORATORS
591de54f486SWilly Tu
592de54f486SWilly Tu boost::container::flat_map<std::string, Value>* entityData = nullptr;
593de54f486SWilly Tu
594de54f486SWilly Tu // todo: this should really use caching, this is a very inefficient lookup
595de54f486SWilly Tu boost::system::error_code ec;
596*de1420dcSGeorge Liu ManagedObjectType entities = ipmi::callDbusMethod<ManagedObjectType>(
597*de1420dcSGeorge Liu ctx, ec, "xyz.openbmc_project.EntityManager",
598947da1b0SNan Zhou "/xyz/openbmc_project/inventory", "org.freedesktop.DBus.ObjectManager",
599947da1b0SNan Zhou "GetManagedObjects");
600de54f486SWilly Tu
601de54f486SWilly Tu if (ec)
602de54f486SWilly Tu {
603de6694e2SGeorge Liu lg2::error("GetMangagedObjects for ipmiStorageGetFruInvAreaInfo "
604de6694e2SGeorge Liu "failed: {ERROR}",
605de6694e2SGeorge Liu "ERROR", ec.message());
606de54f486SWilly Tu
607de54f486SWilly Tu return ipmi::ccResponseError;
608de54f486SWilly Tu }
609de54f486SWilly Tu
6101318a5edSPatrick Williams auto entity = std::find_if(
6111318a5edSPatrick Williams entities.begin(), entities.end(),
612eacad3c4SShakeeb Pasha [bus, address, &entityData, &name](ManagedEntry& entry) {
613de54f486SWilly Tu auto findFruDevice = entry.second.find(
614d2ee9867SWilly Tu "xyz.openbmc_project.Inventory.Decorator.I2CDevice");
615de54f486SWilly Tu if (findFruDevice == entry.second.end())
616de54f486SWilly Tu {
617de54f486SWilly Tu return false;
618de54f486SWilly Tu }
619de54f486SWilly Tu
620de54f486SWilly Tu // Integer fields added via Entity-Manager json are uint64_ts by
621de54f486SWilly Tu // default.
622de54f486SWilly Tu auto findBus = findFruDevice->second.find("Bus");
623de54f486SWilly Tu auto findAddress = findFruDevice->second.find("Address");
624de54f486SWilly Tu
625de54f486SWilly Tu if (findBus == findFruDevice->second.end() ||
626de54f486SWilly Tu findAddress == findFruDevice->second.end())
627de54f486SWilly Tu {
628de54f486SWilly Tu return false;
629de54f486SWilly Tu }
630de54f486SWilly Tu if ((std::get<uint64_t>(findBus->second) != bus) ||
631de54f486SWilly Tu (std::get<uint64_t>(findAddress->second) != address))
632de54f486SWilly Tu {
633de54f486SWilly Tu return false;
634de54f486SWilly Tu }
635de54f486SWilly Tu
636eacad3c4SShakeeb Pasha auto fruName = findFruDevice->second.find("Name");
637eacad3c4SShakeeb Pasha if (fruName != findFruDevice->second.end())
638eacad3c4SShakeeb Pasha {
639eacad3c4SShakeeb Pasha name = std::get<std::string>(fruName->second);
640eacad3c4SShakeeb Pasha }
641eacad3c4SShakeeb Pasha
642de54f486SWilly Tu // At this point we found the device entry and should return
643de54f486SWilly Tu // true.
6441318a5edSPatrick Williams auto findIpmiDevice = entry.second.find(
6451318a5edSPatrick Williams "xyz.openbmc_project.Inventory.Decorator.Ipmi");
646de54f486SWilly Tu if (findIpmiDevice != entry.second.end())
647de54f486SWilly Tu {
648de54f486SWilly Tu entityData = &(findIpmiDevice->second);
649de54f486SWilly Tu }
650de54f486SWilly Tu
651de54f486SWilly Tu return true;
652de54f486SWilly Tu });
653de54f486SWilly Tu
654de54f486SWilly Tu if (entity == entities.end())
655de54f486SWilly Tu {
656de54f486SWilly Tu if constexpr (DEBUG)
657de54f486SWilly Tu {
658de54f486SWilly Tu std::fprintf(stderr, "Ipmi or FruDevice Decorator interface "
659de54f486SWilly Tu "not found for Fru\n");
660de54f486SWilly Tu }
661de54f486SWilly Tu }
662de54f486SWilly Tu
663de54f486SWilly Tu #endif
664de54f486SWilly Tu
665ea46f3caSAlexander Hansen std::vector<std::string> nameProperties = {
666ea46f3caSAlexander Hansen "PRODUCT_PRODUCT_NAME", "BOARD_PRODUCT_NAME", "PRODUCT_PART_NUMBER",
667ea46f3caSAlexander Hansen "BOARD_PART_NUMBER", "PRODUCT_MANUFACTURER", "BOARD_MANUFACTURER",
668ea46f3caSAlexander Hansen "PRODUCT_SERIAL_NUMBER", "BOARD_SERIAL_NUMBER"};
669ea46f3caSAlexander Hansen
670ea46f3caSAlexander Hansen for (const std::string& prop : nameProperties)
671ea46f3caSAlexander Hansen {
672ea46f3caSAlexander Hansen auto findProp = fruData->find(prop);
673ea46f3caSAlexander Hansen if (findProp != fruData->end())
674ea46f3caSAlexander Hansen {
675ea46f3caSAlexander Hansen name = std::get<std::string>(findProp->second);
676ea46f3caSAlexander Hansen break;
677ea46f3caSAlexander Hansen }
678ea46f3caSAlexander Hansen }
679ea46f3caSAlexander Hansen
680eacad3c4SShakeeb Pasha if (name.empty())
681de54f486SWilly Tu {
682de54f486SWilly Tu name = "UNKNOWN";
683de54f486SWilly Tu }
684de54f486SWilly Tu if (name.size() > maxFruSdrNameSize)
685de54f486SWilly Tu {
686de54f486SWilly Tu name = name.substr(0, maxFruSdrNameSize);
687de54f486SWilly Tu }
688de54f486SWilly Tu size_t sizeDiff = maxFruSdrNameSize - name.size();
689de54f486SWilly Tu
690de54f486SWilly Tu resp.header.record_id_lsb = 0x0; // calling code is to implement these
691de54f486SWilly Tu resp.header.record_id_msb = 0x0;
692de54f486SWilly Tu resp.header.sdr_version = ipmiSdrVersion;
693de54f486SWilly Tu resp.header.record_type = get_sdr::SENSOR_DATA_FRU_RECORD;
694de54f486SWilly Tu resp.header.record_length = sizeof(resp.body) + sizeof(resp.key) - sizeDiff;
695de54f486SWilly Tu resp.key.deviceAddress = 0x20;
696de54f486SWilly Tu resp.key.fruID = device->first;
697de54f486SWilly Tu resp.key.accessLun = 0x80; // logical / physical fru device
698de54f486SWilly Tu resp.key.channelNumber = 0x0;
699de54f486SWilly Tu resp.body.reserved = 0x0;
700de54f486SWilly Tu resp.body.deviceType = 0x10;
701de54f486SWilly Tu resp.body.deviceTypeModifier = 0x0;
702de54f486SWilly Tu
703de54f486SWilly Tu uint8_t entityID = 0;
704de54f486SWilly Tu uint8_t entityInstance = 0x1;
705de54f486SWilly Tu
706de54f486SWilly Tu #ifdef USING_ENTITY_MANAGER_DECORATORS
707de54f486SWilly Tu if (entityData)
708de54f486SWilly Tu {
709de54f486SWilly Tu auto entityIdProperty = entityData->find("EntityId");
710de54f486SWilly Tu auto entityInstanceProperty = entityData->find("EntityInstance");
711de54f486SWilly Tu
712de54f486SWilly Tu if (entityIdProperty != entityData->end())
713de54f486SWilly Tu {
714de54f486SWilly Tu entityID = static_cast<uint8_t>(
715de54f486SWilly Tu std::get<uint64_t>(entityIdProperty->second));
716de54f486SWilly Tu }
717de54f486SWilly Tu if (entityInstanceProperty != entityData->end())
718de54f486SWilly Tu {
719de54f486SWilly Tu entityInstance = static_cast<uint8_t>(
720de54f486SWilly Tu std::get<uint64_t>(entityInstanceProperty->second));
721de54f486SWilly Tu }
722de54f486SWilly Tu }
723de54f486SWilly Tu #endif
724de54f486SWilly Tu
725de54f486SWilly Tu resp.body.entityID = entityID;
726de54f486SWilly Tu resp.body.entityInstance = entityInstance;
727de54f486SWilly Tu
728de54f486SWilly Tu resp.body.oem = 0x0;
729b99de182SJohnathan Mantey resp.body.deviceIDLen = ipmi::storage::typeASCIILatin8 | name.size();
730de54f486SWilly Tu name.copy(resp.body.deviceID, name.size());
731de54f486SWilly Tu
732de54f486SWilly Tu return IPMI_CC_OK;
733de54f486SWilly Tu }
734de54f486SWilly Tu
getSELLogFiles(std::vector<std::filesystem::path> & selLogFiles)735de54f486SWilly Tu static bool getSELLogFiles(std::vector<std::filesystem::path>& selLogFiles)
736de54f486SWilly Tu {
737de54f486SWilly Tu // Loop through the directory looking for ipmi_sel log files
738de54f486SWilly Tu for (const std::filesystem::directory_entry& dirEnt :
739de54f486SWilly Tu std::filesystem::directory_iterator(
740de54f486SWilly Tu dynamic_sensors::ipmi::sel::selLogDir))
741de54f486SWilly Tu {
742de54f486SWilly Tu std::string filename = dirEnt.path().filename();
743de54f486SWilly Tu if (boost::starts_with(filename,
744de54f486SWilly Tu dynamic_sensors::ipmi::sel::selLogFilename))
745de54f486SWilly Tu {
746de54f486SWilly Tu // If we find an ipmi_sel log file, save the path
7471318a5edSPatrick Williams selLogFiles.emplace_back(
7481318a5edSPatrick Williams dynamic_sensors::ipmi::sel::selLogDir / filename);
749de54f486SWilly Tu }
750de54f486SWilly Tu }
751de54f486SWilly Tu // As the log files rotate, they are appended with a ".#" that is higher for
752de54f486SWilly Tu // the older logs. Since we don't expect more than 10 log files, we
753de54f486SWilly Tu // can just sort the list to get them in order from newest to oldest
754de54f486SWilly Tu std::sort(selLogFiles.begin(), selLogFiles.end());
755de54f486SWilly Tu
756de54f486SWilly Tu return !selLogFiles.empty();
757de54f486SWilly Tu }
758de54f486SWilly Tu
countSELEntries()759de54f486SWilly Tu static int countSELEntries()
760de54f486SWilly Tu {
761de54f486SWilly Tu // Get the list of ipmi_sel log files
762de54f486SWilly Tu std::vector<std::filesystem::path> selLogFiles;
763de54f486SWilly Tu if (!getSELLogFiles(selLogFiles))
764de54f486SWilly Tu {
765de54f486SWilly Tu return 0;
766de54f486SWilly Tu }
767de54f486SWilly Tu int numSELEntries = 0;
768de54f486SWilly Tu // Loop through each log file and count the number of logs
769de54f486SWilly Tu for (const std::filesystem::path& file : selLogFiles)
770de54f486SWilly Tu {
771de54f486SWilly Tu std::ifstream logStream(file);
772de54f486SWilly Tu if (!logStream.is_open())
773de54f486SWilly Tu {
774de54f486SWilly Tu continue;
775de54f486SWilly Tu }
776de54f486SWilly Tu
777de54f486SWilly Tu std::string line;
778de54f486SWilly Tu while (std::getline(logStream, line))
779de54f486SWilly Tu {
780de54f486SWilly Tu numSELEntries++;
781de54f486SWilly Tu }
782de54f486SWilly Tu }
783de54f486SWilly Tu return numSELEntries;
784de54f486SWilly Tu }
785de54f486SWilly Tu
findSELEntry(const int recordID,const std::vector<std::filesystem::path> & selLogFiles,std::string & entry)786de54f486SWilly Tu static bool findSELEntry(const int recordID,
787de54f486SWilly Tu const std::vector<std::filesystem::path>& selLogFiles,
788de54f486SWilly Tu std::string& entry)
789de54f486SWilly Tu {
790de54f486SWilly Tu // Record ID is the first entry field following the timestamp. It is
791de54f486SWilly Tu // preceded by a space and followed by a comma
792de54f486SWilly Tu std::string search = " " + std::to_string(recordID) + ",";
793de54f486SWilly Tu
794de54f486SWilly Tu // Loop through the ipmi_sel log entries
795de54f486SWilly Tu for (const std::filesystem::path& file : selLogFiles)
796de54f486SWilly Tu {
797de54f486SWilly Tu std::ifstream logStream(file);
798de54f486SWilly Tu if (!logStream.is_open())
799de54f486SWilly Tu {
800de54f486SWilly Tu continue;
801de54f486SWilly Tu }
802de54f486SWilly Tu
803de54f486SWilly Tu while (std::getline(logStream, entry))
804de54f486SWilly Tu {
805de54f486SWilly Tu // Check if the record ID matches
806de54f486SWilly Tu if (entry.find(search) != std::string::npos)
807de54f486SWilly Tu {
808de54f486SWilly Tu return true;
809de54f486SWilly Tu }
810de54f486SWilly Tu }
811de54f486SWilly Tu }
812de54f486SWilly Tu return false;
813de54f486SWilly Tu }
814de54f486SWilly Tu
getNextRecordID(const uint16_t recordID,const std::vector<std::filesystem::path> & selLogFiles)81569b4c281SPatrick Williams static uint16_t getNextRecordID(
81669b4c281SPatrick Williams const uint16_t recordID,
817de54f486SWilly Tu const std::vector<std::filesystem::path>& selLogFiles)
818de54f486SWilly Tu {
819de54f486SWilly Tu uint16_t nextRecordID = recordID + 1;
820de54f486SWilly Tu std::string entry;
821de54f486SWilly Tu if (findSELEntry(nextRecordID, selLogFiles, entry))
822de54f486SWilly Tu {
823de54f486SWilly Tu return nextRecordID;
824de54f486SWilly Tu }
825de54f486SWilly Tu else
826de54f486SWilly Tu {
827de54f486SWilly Tu return ipmi::sel::lastEntry;
828de54f486SWilly Tu }
829de54f486SWilly Tu }
830de54f486SWilly Tu
fromHexStr(const std::string & hexStr,std::vector<uint8_t> & data)831de54f486SWilly Tu static int fromHexStr(const std::string& hexStr, std::vector<uint8_t>& data)
832de54f486SWilly Tu {
833de54f486SWilly Tu for (unsigned int i = 0; i < hexStr.size(); i += 2)
834de54f486SWilly Tu {
835de54f486SWilly Tu try
836de54f486SWilly Tu {
837de54f486SWilly Tu data.push_back(static_cast<uint8_t>(
838de54f486SWilly Tu std::stoul(hexStr.substr(i, 2), nullptr, 16)));
839de54f486SWilly Tu }
840a2ad2da8SPatrick Williams catch (const std::invalid_argument& e)
841de54f486SWilly Tu {
842de6694e2SGeorge Liu lg2::error("Invalid argument: {ERROR}", "ERROR", e);
843de54f486SWilly Tu return -1;
844de54f486SWilly Tu }
845a2ad2da8SPatrick Williams catch (const std::out_of_range& e)
846de54f486SWilly Tu {
847de6694e2SGeorge Liu lg2::error("Out of range: {ERROR}", "ERROR", e);
848de54f486SWilly Tu return -1;
849de54f486SWilly Tu }
850de54f486SWilly Tu }
851de54f486SWilly Tu return 0;
852de54f486SWilly Tu }
853de54f486SWilly Tu
854de54f486SWilly Tu ipmi::RspType<uint8_t, // SEL version
855de54f486SWilly Tu uint16_t, // SEL entry count
856de54f486SWilly Tu uint16_t, // free space
857de54f486SWilly Tu uint32_t, // last add timestamp
858de54f486SWilly Tu uint32_t, // last erase timestamp
859de54f486SWilly Tu uint8_t> // operation support
ipmiStorageGetSELInfo()860de54f486SWilly Tu ipmiStorageGetSELInfo()
861de54f486SWilly Tu {
862de54f486SWilly Tu constexpr uint8_t selVersion = ipmi::sel::selVersion;
863de54f486SWilly Tu uint16_t entries = countSELEntries();
864de54f486SWilly Tu uint32_t addTimeStamp = dynamic_sensors::ipmi::sel::getFileTimestamp(
865de54f486SWilly Tu dynamic_sensors::ipmi::sel::selLogDir /
866de54f486SWilly Tu dynamic_sensors::ipmi::sel::selLogFilename);
867de54f486SWilly Tu uint32_t eraseTimeStamp = dynamic_sensors::ipmi::sel::erase_time::get();
868de54f486SWilly Tu constexpr uint8_t operationSupport =
869de54f486SWilly Tu dynamic_sensors::ipmi::sel::selOperationSupport;
870de54f486SWilly Tu constexpr uint16_t freeSpace =
871de54f486SWilly Tu 0xffff; // Spec indicates that more than 64kB is free
872de54f486SWilly Tu
873de54f486SWilly Tu return ipmi::responseSuccess(selVersion, entries, freeSpace, addTimeStamp,
874de54f486SWilly Tu eraseTimeStamp, operationSupport);
875de54f486SWilly Tu }
876de54f486SWilly Tu
877de54f486SWilly Tu using systemEventType = std::tuple<
878de54f486SWilly Tu uint32_t, // Timestamp
879de54f486SWilly Tu uint16_t, // Generator ID
880de54f486SWilly Tu uint8_t, // EvM Rev
881de54f486SWilly Tu uint8_t, // Sensor Type
882de54f486SWilly Tu uint8_t, // Sensor Number
883de54f486SWilly Tu uint7_t, // Event Type
884de54f486SWilly Tu bool, // Event Direction
885de54f486SWilly Tu std::array<uint8_t, dynamic_sensors::ipmi::sel::systemEventSize>>; // Event
886de54f486SWilly Tu // Data
887de54f486SWilly Tu using oemTsEventType = std::tuple<
888de54f486SWilly Tu uint32_t, // Timestamp
889de54f486SWilly Tu std::array<uint8_t, dynamic_sensors::ipmi::sel::oemTsEventSize>>; // Event
890de54f486SWilly Tu // Data
891de54f486SWilly Tu using oemEventType =
892de54f486SWilly Tu std::array<uint8_t, dynamic_sensors::ipmi::sel::oemEventSize>; // Event Data
893de54f486SWilly Tu
894de54f486SWilly Tu ipmi::RspType<uint16_t, // Next Record ID
895de54f486SWilly Tu uint16_t, // Record ID
896de54f486SWilly Tu uint8_t, // Record Type
897de54f486SWilly Tu std::variant<systemEventType, oemTsEventType,
898de54f486SWilly Tu oemEventType>> // Record Content
ipmiStorageGetSELEntry(uint16_t reservationID,uint16_t targetID,uint8_t offset,uint8_t size)899de54f486SWilly Tu ipmiStorageGetSELEntry(uint16_t reservationID, uint16_t targetID,
900de54f486SWilly Tu uint8_t offset, uint8_t size)
901de54f486SWilly Tu {
902de54f486SWilly Tu // Only support getting the entire SEL record. If a partial size or non-zero
903de54f486SWilly Tu // offset is requested, return an error
904de54f486SWilly Tu if (offset != 0 || size != ipmi::sel::entireRecord)
905de54f486SWilly Tu {
906de54f486SWilly Tu return ipmi::responseRetBytesUnavailable();
907de54f486SWilly Tu }
908de54f486SWilly Tu
909de54f486SWilly Tu // Check the reservation ID if one is provided or required (only if the
910de54f486SWilly Tu // offset is non-zero)
911de54f486SWilly Tu if (reservationID != 0 || offset != 0)
912de54f486SWilly Tu {
913de54f486SWilly Tu if (!checkSELReservation(reservationID))
914de54f486SWilly Tu {
915de54f486SWilly Tu return ipmi::responseInvalidReservationId();
916de54f486SWilly Tu }
917de54f486SWilly Tu }
918de54f486SWilly Tu
919de54f486SWilly Tu // Get the ipmi_sel log files
920de54f486SWilly Tu std::vector<std::filesystem::path> selLogFiles;
921de54f486SWilly Tu if (!getSELLogFiles(selLogFiles))
922de54f486SWilly Tu {
923de54f486SWilly Tu return ipmi::responseSensorInvalid();
924de54f486SWilly Tu }
925de54f486SWilly Tu
926de54f486SWilly Tu std::string targetEntry;
927de54f486SWilly Tu
928de54f486SWilly Tu if (targetID == ipmi::sel::firstEntry)
929de54f486SWilly Tu {
930de54f486SWilly Tu // The first entry will be at the top of the oldest log file
931de54f486SWilly Tu std::ifstream logStream(selLogFiles.back());
932de54f486SWilly Tu if (!logStream.is_open())
933de54f486SWilly Tu {
934de54f486SWilly Tu return ipmi::responseUnspecifiedError();
935de54f486SWilly Tu }
936de54f486SWilly Tu
937de54f486SWilly Tu if (!std::getline(logStream, targetEntry))
938de54f486SWilly Tu {
939de54f486SWilly Tu return ipmi::responseUnspecifiedError();
940de54f486SWilly Tu }
941de54f486SWilly Tu }
942de54f486SWilly Tu else if (targetID == ipmi::sel::lastEntry)
943de54f486SWilly Tu {
944de54f486SWilly Tu // The last entry will be at the bottom of the newest log file
945de54f486SWilly Tu std::ifstream logStream(selLogFiles.front());
946de54f486SWilly Tu if (!logStream.is_open())
947de54f486SWilly Tu {
948de54f486SWilly Tu return ipmi::responseUnspecifiedError();
949de54f486SWilly Tu }
950de54f486SWilly Tu
951de54f486SWilly Tu std::string line;
952de54f486SWilly Tu while (std::getline(logStream, line))
953de54f486SWilly Tu {
954de54f486SWilly Tu targetEntry = line;
955de54f486SWilly Tu }
956de54f486SWilly Tu }
957de54f486SWilly Tu else
958de54f486SWilly Tu {
959de54f486SWilly Tu if (!findSELEntry(targetID, selLogFiles, targetEntry))
960de54f486SWilly Tu {
961de54f486SWilly Tu return ipmi::responseSensorInvalid();
962de54f486SWilly Tu }
963de54f486SWilly Tu }
964de54f486SWilly Tu
965de54f486SWilly Tu // The format of the ipmi_sel message is "<Timestamp>
966de54f486SWilly Tu // <ID>,<Type>,<EventData>,[<Generator ID>,<Path>,<Direction>]".
967de54f486SWilly Tu // First get the Timestamp
968de54f486SWilly Tu size_t space = targetEntry.find_first_of(" ");
969de54f486SWilly Tu if (space == std::string::npos)
970de54f486SWilly Tu {
971de54f486SWilly Tu return ipmi::responseUnspecifiedError();
972de54f486SWilly Tu }
973de54f486SWilly Tu std::string entryTimestamp = targetEntry.substr(0, space);
974de54f486SWilly Tu // Then get the log contents
975de54f486SWilly Tu size_t entryStart = targetEntry.find_first_not_of(" ", space);
976de54f486SWilly Tu if (entryStart == std::string::npos)
977de54f486SWilly Tu {
978de54f486SWilly Tu return ipmi::responseUnspecifiedError();
979de54f486SWilly Tu }
980de54f486SWilly Tu std::string_view entry(targetEntry);
981de54f486SWilly Tu entry.remove_prefix(entryStart);
982de54f486SWilly Tu // Use split to separate the entry into its fields
983de54f486SWilly Tu std::vector<std::string> targetEntryFields;
984de54f486SWilly Tu boost::split(targetEntryFields, entry, boost::is_any_of(","),
985de54f486SWilly Tu boost::token_compress_on);
986de54f486SWilly Tu if (targetEntryFields.size() < 3)
987de54f486SWilly Tu {
988de54f486SWilly Tu return ipmi::responseUnspecifiedError();
989de54f486SWilly Tu }
990de54f486SWilly Tu std::string& recordIDStr = targetEntryFields[0];
991de54f486SWilly Tu std::string& recordTypeStr = targetEntryFields[1];
992de54f486SWilly Tu std::string& eventDataStr = targetEntryFields[2];
993de54f486SWilly Tu
994de54f486SWilly Tu uint16_t recordID;
995de54f486SWilly Tu uint8_t recordType;
996de54f486SWilly Tu try
997de54f486SWilly Tu {
998de54f486SWilly Tu recordID = std::stoul(recordIDStr);
999de54f486SWilly Tu recordType = std::stoul(recordTypeStr, nullptr, 16);
1000de54f486SWilly Tu }
1001de54f486SWilly Tu catch (const std::invalid_argument&)
1002de54f486SWilly Tu {
1003de54f486SWilly Tu return ipmi::responseUnspecifiedError();
1004de54f486SWilly Tu }
1005de54f486SWilly Tu uint16_t nextRecordID = getNextRecordID(recordID, selLogFiles);
1006de54f486SWilly Tu std::vector<uint8_t> eventDataBytes;
1007de54f486SWilly Tu if (fromHexStr(eventDataStr, eventDataBytes) < 0)
1008de54f486SWilly Tu {
1009de54f486SWilly Tu return ipmi::responseUnspecifiedError();
1010de54f486SWilly Tu }
1011de54f486SWilly Tu
1012de54f486SWilly Tu if (recordType == dynamic_sensors::ipmi::sel::systemEvent)
1013de54f486SWilly Tu {
1014de54f486SWilly Tu // Get the timestamp
1015de54f486SWilly Tu std::tm timeStruct = {};
1016de54f486SWilly Tu std::istringstream entryStream(entryTimestamp);
1017de54f486SWilly Tu
1018de54f486SWilly Tu uint32_t timestamp = ipmi::sel::invalidTimeStamp;
1019de54f486SWilly Tu if (entryStream >> std::get_time(&timeStruct, "%Y-%m-%dT%H:%M:%S"))
1020de54f486SWilly Tu {
10217bb412f5SWilly Tu timeStruct.tm_isdst = -1;
1022de54f486SWilly Tu timestamp = std::mktime(&timeStruct);
1023de54f486SWilly Tu }
1024de54f486SWilly Tu
1025de54f486SWilly Tu // Set the event message revision
1026de54f486SWilly Tu uint8_t evmRev = dynamic_sensors::ipmi::sel::eventMsgRev;
1027de54f486SWilly Tu
1028de54f486SWilly Tu uint16_t generatorID = 0;
1029de54f486SWilly Tu uint8_t sensorType = 0;
1030de54f486SWilly Tu uint16_t sensorAndLun = 0;
1031de54f486SWilly Tu uint8_t sensorNum = 0xFF;
1032de54f486SWilly Tu uint7_t eventType = 0;
1033de54f486SWilly Tu bool eventDir = 0;
1034de54f486SWilly Tu // System type events should have six fields
1035de54f486SWilly Tu if (targetEntryFields.size() >= 6)
1036de54f486SWilly Tu {
1037de54f486SWilly Tu std::string& generatorIDStr = targetEntryFields[3];
1038de54f486SWilly Tu std::string& sensorPath = targetEntryFields[4];
1039de54f486SWilly Tu std::string& eventDirStr = targetEntryFields[5];
1040de54f486SWilly Tu
1041de54f486SWilly Tu // Get the generator ID
1042de54f486SWilly Tu try
1043de54f486SWilly Tu {
1044de54f486SWilly Tu generatorID = std::stoul(generatorIDStr, nullptr, 16);
1045de54f486SWilly Tu }
1046de54f486SWilly Tu catch (const std::invalid_argument&)
1047de54f486SWilly Tu {
1048de54f486SWilly Tu std::cerr << "Invalid Generator ID\n";
1049de54f486SWilly Tu }
1050de54f486SWilly Tu
1051de54f486SWilly Tu // Get the sensor type, sensor number, and event type for the sensor
1052de54f486SWilly Tu sensorType = getSensorTypeFromPath(sensorPath);
1053de54f486SWilly Tu sensorAndLun = getSensorNumberFromPath(sensorPath);
1054de54f486SWilly Tu sensorNum = static_cast<uint8_t>(sensorAndLun);
10554376cdf8SHarvey.Wu if ((generatorID & 0x0001) == 0)
10564376cdf8SHarvey.Wu {
10574376cdf8SHarvey.Wu // IPMB Address
10584376cdf8SHarvey.Wu generatorID |= sensorAndLun & 0x0300;
10594376cdf8SHarvey.Wu }
10604376cdf8SHarvey.Wu else
10614376cdf8SHarvey.Wu {
10624376cdf8SHarvey.Wu // system software
1063de54f486SWilly Tu generatorID |= sensorAndLun >> 8;
10644376cdf8SHarvey.Wu }
1065de54f486SWilly Tu eventType = getSensorEventTypeFromPath(sensorPath);
1066de54f486SWilly Tu
1067de54f486SWilly Tu // Get the event direction
1068de54f486SWilly Tu try
1069de54f486SWilly Tu {
1070de54f486SWilly Tu eventDir = std::stoul(eventDirStr) ? 0 : 1;
1071de54f486SWilly Tu }
1072de54f486SWilly Tu catch (const std::invalid_argument&)
1073de54f486SWilly Tu {
1074de54f486SWilly Tu std::cerr << "Invalid Event Direction\n";
1075de54f486SWilly Tu }
1076de54f486SWilly Tu }
1077de54f486SWilly Tu
1078de54f486SWilly Tu // Only keep the eventData bytes that fit in the record
1079de54f486SWilly Tu std::array<uint8_t, dynamic_sensors::ipmi::sel::systemEventSize>
1080de54f486SWilly Tu eventData{};
1081de54f486SWilly Tu std::copy_n(eventDataBytes.begin(),
1082de54f486SWilly Tu std::min(eventDataBytes.size(), eventData.size()),
1083de54f486SWilly Tu eventData.begin());
1084de54f486SWilly Tu
1085de54f486SWilly Tu return ipmi::responseSuccess(
1086de54f486SWilly Tu nextRecordID, recordID, recordType,
1087de54f486SWilly Tu systemEventType{timestamp, generatorID, evmRev, sensorType,
1088de54f486SWilly Tu sensorNum, eventType, eventDir, eventData});
1089de54f486SWilly Tu }
1090de54f486SWilly Tu
1091e70c59b8SThang Tran if (recordType >= dynamic_sensors::ipmi::sel::oemTsEventFirst &&
1092e70c59b8SThang Tran recordType <= dynamic_sensors::ipmi::sel::oemTsEventLast)
1093e70c59b8SThang Tran {
1094e70c59b8SThang Tran // Get the timestamp
1095e70c59b8SThang Tran std::tm timeStruct = {};
1096e70c59b8SThang Tran std::istringstream entryStream(entryTimestamp);
1097e70c59b8SThang Tran
1098e70c59b8SThang Tran uint32_t timestamp = ipmi::sel::invalidTimeStamp;
1099e70c59b8SThang Tran if (entryStream >> std::get_time(&timeStruct, "%Y-%m-%dT%H:%M:%S"))
1100e70c59b8SThang Tran {
1101e70c59b8SThang Tran timeStruct.tm_isdst = -1;
1102e70c59b8SThang Tran timestamp = std::mktime(&timeStruct);
1103e70c59b8SThang Tran }
1104e70c59b8SThang Tran
1105e70c59b8SThang Tran // Only keep the bytes that fit in the record
1106e70c59b8SThang Tran std::array<uint8_t, dynamic_sensors::ipmi::sel::oemTsEventSize>
1107e70c59b8SThang Tran eventData{};
1108e70c59b8SThang Tran std::copy_n(eventDataBytes.begin(),
1109e70c59b8SThang Tran std::min(eventDataBytes.size(), eventData.size()),
1110e70c59b8SThang Tran eventData.begin());
1111e70c59b8SThang Tran
1112e70c59b8SThang Tran return ipmi::responseSuccess(nextRecordID, recordID, recordType,
1113e70c59b8SThang Tran oemTsEventType{timestamp, eventData});
1114e70c59b8SThang Tran }
1115e70c59b8SThang Tran
1116e70c59b8SThang Tran if (recordType >= dynamic_sensors::ipmi::sel::oemEventFirst)
1117e70c59b8SThang Tran {
1118e70c59b8SThang Tran // Only keep the bytes that fit in the record
1119e70c59b8SThang Tran std::array<uint8_t, dynamic_sensors::ipmi::sel::oemEventSize>
1120e70c59b8SThang Tran eventData{};
1121e70c59b8SThang Tran std::copy_n(eventDataBytes.begin(),
1122e70c59b8SThang Tran std::min(eventDataBytes.size(), eventData.size()),
1123e70c59b8SThang Tran eventData.begin());
1124e70c59b8SThang Tran
1125e70c59b8SThang Tran return ipmi::responseSuccess(nextRecordID, recordID, recordType,
1126e70c59b8SThang Tran eventData);
1127e70c59b8SThang Tran }
1128e70c59b8SThang Tran
1129de54f486SWilly Tu return ipmi::responseUnspecifiedError();
1130de54f486SWilly Tu }
1131de54f486SWilly Tu
113211d68897SWilly Tu /*
113311d68897SWilly Tu Unused arguments
1134de54f486SWilly Tu uint16_t recordID, uint8_t recordType, uint32_t timestamp,
1135de54f486SWilly Tu uint16_t generatorID, uint8_t evmRev, uint8_t sensorType, uint8_t sensorNum,
1136de54f486SWilly Tu uint8_t eventType, uint8_t eventData1, uint8_t eventData2,
113711d68897SWilly Tu uint8_t eventData3
113811d68897SWilly Tu */
ipmiStorageAddSELEntry(uint16_t,uint8_t,uint32_t,uint16_t,uint8_t,uint8_t,uint8_t,uint8_t,uint8_t,uint8_t,uint8_t)113969b4c281SPatrick Williams ipmi::RspType<uint16_t> ipmiStorageAddSELEntry(
114069b4c281SPatrick Williams uint16_t, uint8_t, uint32_t, uint16_t, uint8_t, uint8_t, uint8_t, uint8_t,
114169b4c281SPatrick Williams uint8_t, uint8_t, uint8_t)
1142de54f486SWilly Tu {
1143de54f486SWilly Tu // Per the IPMI spec, need to cancel any reservation when a SEL entry is
1144de54f486SWilly Tu // added
1145de54f486SWilly Tu cancelSELReservation();
1146de54f486SWilly Tu
1147de54f486SWilly Tu uint16_t responseID = 0xFFFF;
1148de54f486SWilly Tu return ipmi::responseSuccess(responseID);
1149de54f486SWilly Tu }
1150de54f486SWilly Tu
ipmiStorageClearSEL(ipmi::Context::ptr ctx,uint16_t reservationID,const std::array<uint8_t,3> & clr,uint8_t eraseOperation)11511318a5edSPatrick Williams ipmi::RspType<uint8_t> ipmiStorageClearSEL(
11521318a5edSPatrick Williams ipmi::Context::ptr ctx, uint16_t reservationID,
11531318a5edSPatrick Williams const std::array<uint8_t, 3>& clr, uint8_t eraseOperation)
1154de54f486SWilly Tu {
1155de54f486SWilly Tu if (!checkSELReservation(reservationID))
1156de54f486SWilly Tu {
1157de54f486SWilly Tu return ipmi::responseInvalidReservationId();
1158de54f486SWilly Tu }
1159de54f486SWilly Tu
1160de54f486SWilly Tu static constexpr std::array<uint8_t, 3> clrExpected = {'C', 'L', 'R'};
1161de54f486SWilly Tu if (clr != clrExpected)
1162de54f486SWilly Tu {
1163de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
1164de54f486SWilly Tu }
1165de54f486SWilly Tu
1166de54f486SWilly Tu // Erasure status cannot be fetched, so always return erasure status as
1167de54f486SWilly Tu // `erase completed`.
1168de54f486SWilly Tu if (eraseOperation == ipmi::sel::getEraseStatus)
1169de54f486SWilly Tu {
1170de54f486SWilly Tu return ipmi::responseSuccess(ipmi::sel::eraseComplete);
1171de54f486SWilly Tu }
1172de54f486SWilly Tu
1173de54f486SWilly Tu // Check that initiate erase is correct
1174de54f486SWilly Tu if (eraseOperation != ipmi::sel::initiateErase)
1175de54f486SWilly Tu {
1176de54f486SWilly Tu return ipmi::responseInvalidFieldRequest();
1177de54f486SWilly Tu }
1178de54f486SWilly Tu
1179de54f486SWilly Tu // Per the IPMI spec, need to cancel any reservation when the SEL is
1180de54f486SWilly Tu // cleared
1181de54f486SWilly Tu cancelSELReservation();
1182de54f486SWilly Tu
1183*de1420dcSGeorge Liu boost::system::error_code ec =
1184*de1420dcSGeorge Liu ipmi::callDbusMethod(ctx, "xyz.openbmc_project.Logging.IPMI",
1185818bea1fSCharles Boyer "/xyz/openbmc_project/Logging/IPMI",
1186818bea1fSCharles Boyer "xyz.openbmc_project.Logging.IPMI", "Clear");
1187818bea1fSCharles Boyer if (ec)
1188818bea1fSCharles Boyer {
1189*de1420dcSGeorge Liu std::cerr << "error in clear SEL: " << ec.message() << std::endl;
1190818bea1fSCharles Boyer return ipmi::responseUnspecifiedError();
1191818bea1fSCharles Boyer }
1192de54f486SWilly Tu
1193de54f486SWilly Tu return ipmi::responseSuccess(ipmi::sel::eraseComplete);
1194de54f486SWilly Tu }
1195de54f486SWilly Tu
getType8SDRs(ipmi::sensor::EntityInfoMap::const_iterator & entity,uint16_t recordId)11961318a5edSPatrick Williams std::vector<uint8_t> getType8SDRs(
11971318a5edSPatrick Williams ipmi::sensor::EntityInfoMap::const_iterator& entity, uint16_t recordId)
119805d17c03SHarvey Wu {
119905d17c03SHarvey Wu std::vector<uint8_t> resp;
120005d17c03SHarvey Wu get_sdr::SensorDataEntityRecord data{};
120105d17c03SHarvey Wu
120205d17c03SHarvey Wu /* Header */
120305d17c03SHarvey Wu get_sdr::header::set_record_id(recordId, &(data.header));
120405d17c03SHarvey Wu // Based on IPMI Spec v2.0 rev 1.1
120505d17c03SHarvey Wu data.header.sdr_version = SDR_VERSION;
120605d17c03SHarvey Wu data.header.record_type = 0x08;
120705d17c03SHarvey Wu data.header.record_length = sizeof(data.key) + sizeof(data.body);
120805d17c03SHarvey Wu
120905d17c03SHarvey Wu /* Key */
121005d17c03SHarvey Wu data.key.containerEntityId = entity->second.containerEntityId;
121105d17c03SHarvey Wu data.key.containerEntityInstance = entity->second.containerEntityInstance;
121205d17c03SHarvey Wu get_sdr::key::set_flags(entity->second.isList, entity->second.isLinked,
121305d17c03SHarvey Wu &(data.key));
121405d17c03SHarvey Wu data.key.entityId1 = entity->second.containedEntities[0].first;
121505d17c03SHarvey Wu data.key.entityInstance1 = entity->second.containedEntities[0].second;
121605d17c03SHarvey Wu
121705d17c03SHarvey Wu /* Body */
121805d17c03SHarvey Wu data.body.entityId2 = entity->second.containedEntities[1].first;
121905d17c03SHarvey Wu data.body.entityInstance2 = entity->second.containedEntities[1].second;
122005d17c03SHarvey Wu data.body.entityId3 = entity->second.containedEntities[2].first;
122105d17c03SHarvey Wu data.body.entityInstance3 = entity->second.containedEntities[2].second;
122205d17c03SHarvey Wu data.body.entityId4 = entity->second.containedEntities[3].first;
122305d17c03SHarvey Wu data.body.entityInstance4 = entity->second.containedEntities[3].second;
122405d17c03SHarvey Wu
122505d17c03SHarvey Wu resp.insert(resp.end(), (uint8_t*)&data, ((uint8_t*)&data) + sizeof(data));
122605d17c03SHarvey Wu
122705d17c03SHarvey Wu return resp;
122805d17c03SHarvey Wu }
122905d17c03SHarvey Wu
getType12SDRs(uint16_t index,uint16_t recordId)1230de54f486SWilly Tu std::vector<uint8_t> getType12SDRs(uint16_t index, uint16_t recordId)
1231de54f486SWilly Tu {
1232de54f486SWilly Tu std::vector<uint8_t> resp;
1233de54f486SWilly Tu if (index == 0)
1234de54f486SWilly Tu {
1235de54f486SWilly Tu std::string bmcName = "Basbrd Mgmt Ctlr";
1236cd1c4963SJohnathan Mantey Type12Record bmc(recordId, 0x20, 0, 0, 0xbf, 0x2e, 1, 0, bmcName);
1237de54f486SWilly Tu uint8_t* bmcPtr = reinterpret_cast<uint8_t*>(&bmc);
1238de54f486SWilly Tu resp.insert(resp.end(), bmcPtr, bmcPtr + sizeof(Type12Record));
1239de54f486SWilly Tu }
1240de54f486SWilly Tu else if (index == 1)
1241de54f486SWilly Tu {
1242de54f486SWilly Tu std::string meName = "Mgmt Engine";
1243cd1c4963SJohnathan Mantey Type12Record me(recordId, 0x2c, 6, 0x24, 0x21, 0x2e, 2, 0, meName);
1244de54f486SWilly Tu uint8_t* mePtr = reinterpret_cast<uint8_t*>(&me);
1245de54f486SWilly Tu resp.insert(resp.end(), mePtr, mePtr + sizeof(Type12Record));
1246de54f486SWilly Tu }
1247de54f486SWilly Tu else
1248de54f486SWilly Tu {
12491318a5edSPatrick Williams throw std::runtime_error(
12501318a5edSPatrick Williams "getType12SDRs:: Illegal index " + std::to_string(index));
1251de54f486SWilly Tu }
1252de54f486SWilly Tu
1253de54f486SWilly Tu return resp;
1254de54f486SWilly Tu }
1255de54f486SWilly Tu
registerStorageFunctions()1256de54f486SWilly Tu void registerStorageFunctions()
1257de54f486SWilly Tu {
1258de54f486SWilly Tu createTimers();
1259de54f486SWilly Tu startMatch();
1260de54f486SWilly Tu
1261de54f486SWilly Tu // <Get FRU Inventory Area Info>
1262d351a729SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1263de54f486SWilly Tu ipmi::storage::cmdGetFruInventoryAreaInfo,
1264de54f486SWilly Tu ipmi::Privilege::User, ipmiStorageGetFruInvAreaInfo);
1265de54f486SWilly Tu // <READ FRU Data>
1266de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1267de54f486SWilly Tu ipmi::storage::cmdReadFruData, ipmi::Privilege::User,
1268de54f486SWilly Tu ipmiStorageReadFruData);
1269de54f486SWilly Tu
1270de54f486SWilly Tu // <WRITE FRU Data>
1271de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1272de54f486SWilly Tu ipmi::storage::cmdWriteFruData,
1273de54f486SWilly Tu ipmi::Privilege::Operator, ipmiStorageWriteFruData);
1274de54f486SWilly Tu
1275de54f486SWilly Tu // <Get SEL Info>
1276de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1277de54f486SWilly Tu ipmi::storage::cmdGetSelInfo, ipmi::Privilege::User,
1278de54f486SWilly Tu ipmiStorageGetSELInfo);
1279de54f486SWilly Tu
1280de54f486SWilly Tu // <Get SEL Entry>
1281de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1282de54f486SWilly Tu ipmi::storage::cmdGetSelEntry, ipmi::Privilege::User,
1283de54f486SWilly Tu ipmiStorageGetSELEntry);
1284de54f486SWilly Tu
1285de54f486SWilly Tu // <Add SEL Entry>
1286de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1287de54f486SWilly Tu ipmi::storage::cmdAddSelEntry,
1288de54f486SWilly Tu ipmi::Privilege::Operator, ipmiStorageAddSELEntry);
1289de54f486SWilly Tu
1290de54f486SWilly Tu // <Clear SEL>
1291de54f486SWilly Tu ipmi::registerHandler(ipmi::prioOpenBmcBase, ipmi::netFnStorage,
1292de54f486SWilly Tu ipmi::storage::cmdClearSel, ipmi::Privilege::Operator,
1293de54f486SWilly Tu ipmiStorageClearSEL);
1294de54f486SWilly Tu }
1295de54f486SWilly Tu } // namespace storage
1296de54f486SWilly Tu } // namespace ipmi
1297