blob: b0b3c75a905c316a169a1b18a505cd6674a25e60 [file] [log] [blame]
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -05001extern "C"
2{
3#include <libpdbg.h>
4#include <libpdbg_sbe.h>
5}
6
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -05007#include "create_pel.hpp"
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -05008#include "sbe_consts.hpp"
9#include "sbe_dump_collector.hpp"
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -050010#include "sbe_type.hpp"
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050011
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -050012#include <ekb/hwpf/fapi2/include/target_types.H>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050013#include <libphal.H>
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -050014#include <phal_exception.H>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050015
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -050016#include <phosphor-logging/elog-errors.hpp>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050017#include <phosphor-logging/lg2.hpp>
18#include <phosphor-logging/log.hpp>
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -050019#include <sbe_consts.hpp>
20#include <xyz/openbmc_project/Common/File/error.hpp>
21#include <xyz/openbmc_project/Common/error.hpp>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050022
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -050023#include <cstdint>
24#include <filesystem>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050025#include <format>
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -050026#include <fstream>
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050027#include <stdexcept>
28
29namespace openpower::dump::sbe_chipop
30{
31
32using namespace phosphor::logging;
33using namespace openpower::dump::SBE;
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -050034using Severity = sdbusplus::xyz::openbmc_project::Logging::server::Entry::Level;
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050035
36void SbeDumpCollector::collectDump(uint8_t type, uint32_t id,
37 uint64_t failingUnit,
38 const std::filesystem::path& path)
39{
40 lg2::error("Starting dump collection: type:{TYPE} id:{ID} "
41 "failingUnit:{FAILINGUNIT}, path:{PATH}",
42 "TYPE", type, "ID", id, "FAILINGUNIT", failingUnit, "PATH",
43 path.string());
44
45 initializePdbg();
46
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -050047 TargetMap targets;
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050048
49 struct pdbg_target* target = nullptr;
50 pdbg_for_each_class_target("proc", target)
51 {
52 if (pdbg_target_probe(target) != PDBG_TARGET_ENABLED ||
53 !openpower::phal::pdbg::isTgtFunctional(target))
54 {
55 continue;
56 }
57
Dhruvaraj Subhashchandranf9f65b82022-10-13 06:46:43 -050058 bool includeTarget = true;
59 // if the dump type is hostboot then call stop instructions
60 if (type == SBE_DUMP_TYPE_HOSTBOOT)
61 {
62 includeTarget = executeThreadStop(target);
63 }
64 if (includeTarget)
65 {
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -050066 targets[target] = std::vector<struct pdbg_target*>();
67
68 // Hardware dump needs OCMB data if present
69 if (type == openpower::dump::SBE::SBE_DUMP_TYPE_HARDWARE)
70 {
71 struct pdbg_target* ocmbTarget;
72 pdbg_for_each_target("ocmb", target, ocmbTarget)
73 {
74 if (!is_ody_ocmb_chip(ocmbTarget))
75 {
76 continue;
77 }
78
79 if (pdbg_target_probe(ocmbTarget) != PDBG_TARGET_ENABLED)
80 {
81 continue;
82 }
83
84 if (!openpower::phal::pdbg::isTgtFunctional(ocmbTarget))
85 {
86 continue;
87 }
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -050088 targets[target].push_back(ocmbTarget);
89 }
90 }
Dhruvaraj Subhashchandranf9f65b82022-10-13 06:46:43 -050091 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -050092 }
93
94 std::vector<uint8_t> clockStates = {SBE_CLOCK_ON, SBE_CLOCK_OFF};
95 for (auto cstate : clockStates)
96 {
Dhruvaraj Subhashchandran9098d8c2022-12-01 00:40:20 -060097 // Skip collection for performance dump if clock state is not ON
98 if (type == SBE_DUMP_TYPE_PERFORMANCE && cstate != SBE_CLOCK_ON)
99 {
100 continue;
101 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500102 auto futures = spawnDumpCollectionProcesses(type, id, path, failingUnit,
103 cstate, targets);
104
105 // Wait for all asynchronous tasks to complete
106 for (auto& future : futures)
107 {
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500108 try
109 {
110 future.wait();
111 }
112 catch (const std::exception& e)
113 {
114 lg2::error("Failed to collect dump from SBE ErrorMsg({ERROR})",
115 "ERROR", e);
116 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500117 }
118 lg2::info(
119 "Dump collection completed for clock state({CSTATE}): type({TYPE}) "
120 "id({ID}) failingUnit({FAILINGUNIT}), path({PATH})",
121 "CSTATE", cstate, "TYPE", type, "ID", id, "FAILINGUNIT",
122 failingUnit, "PATH", path.string());
123 }
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500124 if (std::filesystem::is_empty(path))
125 {
126 lg2::error("Failed to collect the dump");
127 throw std::runtime_error("Failed to collect the dump");
128 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500129 lg2::info("Dump collection completed");
130}
131
132void SbeDumpCollector::initializePdbg()
133{
134 openpower::phal::pdbg::init();
135}
136
137std::vector<std::future<void>> SbeDumpCollector::spawnDumpCollectionProcesses(
138 uint8_t type, uint32_t id, const std::filesystem::path& path,
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500139 uint64_t failingUnit, uint8_t cstate, const TargetMap& targetMap)
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500140{
141 std::vector<std::future<void>> futures;
142
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500143 for (const auto& [procTarget, ocmbTargets] : targetMap)
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500144 {
Patrick Williams540521e2024-08-16 15:20:03 -0400145 auto future = std::async(std::launch::async, [this, procTarget,
146 ocmbTargets, path, id,
147 type, cstate,
148 failingUnit]() {
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500149 try
150 {
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500151 this->collectDumpFromSBE(procTarget, path, id, type, cstate,
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500152 failingUnit);
153 }
154 catch (const std::exception& e)
155 {
156 lg2::error(
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500157 "Failed to collect dump from SBE on Proc-({PROCINDEX}) {ERROR}",
158 "PROCINDEX", pdbg_target_index(procTarget), "ERROR", e);
159 }
160
161 // Collect OCMBs only with clock on
162 if (cstate == SBE_CLOCK_ON)
163 {
164 // Handle OCMBs serially after handling the proc
165 for (auto ocmbTarget : ocmbTargets)
166 {
167 try
168 {
169 this->collectDumpFromSBE(ocmbTarget, path, id, type,
170 cstate, failingUnit);
171 }
172 catch (const std::exception& e)
173 {
174 lg2::error(
175 "Failed to collect dump from OCMB -({OCMBINDEX}) {ERROR}",
176 "OCMBINDEX", pdbg_target_index(ocmbTarget), "ERROR",
177 e);
178 }
179 }
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500180 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500181 });
182
183 futures.push_back(std::move(future));
184 }
185
186 return futures;
187}
188
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500189bool SbeDumpCollector::logErrorAndCreatePEL(
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500190 const openpower::phal::sbeError_t& sbeError, uint64_t chipPos,
191 SBETypes sbeType, uint32_t cmdClass, uint32_t cmdType)
192{
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500193 namespace fs = std::filesystem;
194
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500195 std::string chipName;
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500196 std::string event;
197 bool dumpIsRequired = false;
198 bool isDumpFailure = true;
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500199 try
200 {
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500201 chipName = sbeTypeAttributes.at(sbeType).chipName;
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500202 event = sbeTypeAttributes.at(sbeType).chipOpFailure;
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500203
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500204 lg2::info("log error {CHIP} {POSITION}", "CHIP", chipName, "POSITION",
205 chipPos);
206
207 // Common FFDC data
208 openpower::dump::pel::FFDCData pelAdditionalData = {
Dhruvaraj Subhashchandran778d1db2024-05-20 09:13:08 -0500209 {"SRC6", std::format("0x{:X}{:X}", chipPos, (cmdClass | cmdType))}};
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500210
211 if (sbeType == SBETypes::OCMB)
212 {
213 pelAdditionalData.emplace_back(
214 "CHIP_TYPE", std::to_string(fapi2::TARGET_TYPE_OCMB_CHIP));
215 }
216
217 // Check the error type
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500218 if (sbeError.errType() == openpower::phal::exception::SBE_CMD_TIMEOUT)
219 {
220 event = sbeTypeAttributes.at(sbeType).chipOpTimeout;
221 dumpIsRequired = true;
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500222 // For timeout, we do not expect any FFDC packets
223 }
224 else if (sbeError.errType() ==
225 openpower::phal::exception::SBE_FFDC_NO_DATA)
226 {
227 // We will create a PEL without FFDC with the common information we
228 // added
229 lg2::error("No FFDC data after a chip-op failure {CHIP} {POSITION}",
230 "CHIP", chipName, "POSITION", chipPos);
231 event = sbeTypeAttributes.at(sbeType).noFfdc;
232 }
233 else
234 {
235 if (sbeError.errType() ==
236 openpower::phal::exception::SBE_INTERNAL_FFDC_DATA)
237 {
238 lg2::info(
239 "FFDC Not related to chip-op present {CHIP} {POSITION}",
240 "CHIP", chipName, "POSITION", chipPos);
241 event = sbeTypeAttributes.at(sbeType).sbeInternalFFDCData;
242 isDumpFailure = false;
243 }
244 else
245 {
246 lg2::error("Process FFDC {CHIP} {POSITION}", "CHIP", chipName,
247 "POSITION", chipPos);
248 }
249 // Processor FFDC Packets
250 openpower::dump::pel::processFFDCPackets(sbeError, event,
251 pelAdditionalData);
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500252 }
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500253
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500254 // If dump is required, request it
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500255 if (dumpIsRequired)
256 {
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500257 auto logId = openpower::dump::pel::createSbeErrorPEL(
258 event, sbeError, pelAdditionalData);
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500259 util::requestSBEDump(chipPos, logId, sbeType);
260 }
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500261 }
262 catch (const std::out_of_range& e)
263 {
264 lg2::error("Unknown SBE Type({SBETYPE}) ErrorMsg({ERROR})", "SBETYPE",
265 sbeType, "ERROR", e);
266 }
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500267 catch (const std::exception& e)
268 {
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500269 lg2::error("SBE Dump request failed, chip type({CHIPTYPE}) "
270 "position({CHIPPOS}), Error: {ERROR}",
271 "CHIPTYPE", chipName, "CHIPPOS", chipPos, "ERROR", e);
Dhruvaraj Subhashchandran5f5c94d2021-10-19 07:18:30 -0500272 }
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500273
274 return isDumpFailure;
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500275}
276
Patrick Williams540521e2024-08-16 15:20:03 -0400277void SbeDumpCollector::collectDumpFromSBE(
278 struct pdbg_target* chip, const std::filesystem::path& path, uint32_t id,
279 uint8_t type, uint8_t clockState, uint64_t failingUnit)
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500280{
281 auto chipPos = pdbg_target_index(chip);
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500282 SBETypes sbeType = getSBEType(chip);
283 auto chipName = sbeTypeAttributes.at(sbeType).chipName;
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500284 lg2::info(
Dhruvaraj Subhashchandrane74e9162024-04-01 09:53:13 -0500285 "Collecting dump from ({CHIPTYPE}) ({POSITION}): path({PATH}) id({ID}) "
286 "type({TYPE}) clockState({CLOCKSTATE}) failingUnit({FAILINGUNIT})",
287 "CHIPTYPE", chipName, "POSITION", chipPos, "PATH", path.string(), "ID",
288 id, "TYPE", type, "CLOCKSTATE", clockState, "FAILINGUNIT", failingUnit);
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500289
290 util::DumpDataPtr dataPtr;
291 uint32_t len = 0;
292 uint8_t collectFastArray =
293 checkFastarrayCollectionNeeded(clockState, type, failingUnit, chipPos);
294
295 try
296 {
297 openpower::phal::sbe::getDump(chip, type, clockState, collectFastArray,
298 dataPtr.getPtr(), &len);
299 }
300 catch (const openpower::phal::sbeError_t& sbeError)
301 {
302 if (sbeError.errType() ==
303 openpower::phal::exception::SBE_CHIPOP_NOT_ALLOWED)
304 {
305 // SBE is not ready to accept chip-ops,
306 // Skip the request, no additional error handling required.
307 lg2::info("Collect dump: Skipping ({ERROR}) dump({TYPE}) "
308 "on proc({PROC}) clock state({CLOCKSTATE})",
309 "ERROR", sbeError, "TYPE", type, "PROC", chipPos,
310 "CLOCKSTATE", clockState);
311 return;
312 }
313
Dhruvaraj Subhashchandranf2298892024-04-21 04:42:55 -0500314 // If the FFDC is from actual chip-op failure this function will
315 // return true, if the chip-op is not failed but FFDC is present
316 // then create PELs with FFDC but write the dump contents to the
317 // file.
318 if (logErrorAndCreatePEL(sbeError, chipPos, sbeType,
319 SBEFIFO_CMD_CLASS_DUMP, SBEFIFO_CMD_GET_DUMP))
320 {
321 lg2::error("Error in collecting dump dump type({TYPE}), "
322 "clockstate({CLOCKSTATE}), chip type({CHIPTYPE}) "
323 "position({POSITION}), "
324 "collectFastArray({COLLECTFASTARRAY}) error({ERROR})",
325 "TYPE", type, "CLOCKSTATE", clockState, "CHIPTYPE",
326 chipName, "POSITION", chipPos, "COLLECTFASTARRAY",
327 collectFastArray, "ERROR", sbeError);
328 return;
329 }
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500330 }
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500331 writeDumpFile(path, id, clockState, 0, chipName, chipPos, dataPtr, len);
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500332}
333
334void SbeDumpCollector::writeDumpFile(
335 const std::filesystem::path& path, const uint32_t id,
Dhruvaraj Subhashchandran6feeebd2021-10-19 05:03:59 -0500336 const uint8_t clockState, const uint8_t nodeNum,
337 const std::string& chipName, const uint8_t chipPos,
338 util::DumpDataPtr& dataPtr, const uint32_t len)
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500339{
340 using namespace sdbusplus::xyz::openbmc_project::Common::Error;
341 namespace fileError = sdbusplus::xyz::openbmc_project::Common::File::Error;
342
343 // Construct the filename
344 std::ostringstream filenameBuilder;
Dhruvaraj Subhashchandranb394bf22024-05-18 15:38:56 -0500345 filenameBuilder << std::hex << std::setw(8) << std::setfill('0') << id
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500346 << ".SbeDataClocks"
347 << (clockState == SBE_CLOCK_ON ? "On" : "Off") << ".node"
Dhruvaraj Subhashchandranb394bf22024-05-18 15:38:56 -0500348 << std::dec << static_cast<int>(nodeNum) << "." << chipName
Dhruvaraj Subhashchandrana699e312021-10-27 07:20:34 -0500349 << static_cast<int>(chipPos);
350
351 auto dumpPath = path / filenameBuilder.str();
352
353 // Attempt to open the file
354 std::ofstream outfile(dumpPath, std::ios::out | std::ios::binary);
355 if (!outfile)
356 {
357 using namespace sdbusplus::xyz::openbmc_project::Common::File::Error;
358 using metadata = xyz::openbmc_project::Common::File::Open;
359 // Unable to open the file for writing
360 auto err = errno;
361 lg2::error("Error opening file to write dump, "
362 "errno({ERRNO}), filepath({FILEPATH})",
363 "ERRNO", err, "FILEPATH", dumpPath.string());
364
365 report<Open>(metadata::ERRNO(err), metadata::PATH(dumpPath.c_str()));
366 // Just return here, so that the dumps collected from other
367 // SBEs can be packaged.
368 return;
369 }
370
371 // Write to the file
372 try
373 {
374 outfile.write(reinterpret_cast<const char*>(dataPtr.getData()), len);
375
376 lg2::info("Successfully wrote dump file "
377 "path=({PATH}) size=({SIZE})",
378 "PATH", dumpPath.string(), "SIZE", len);
379 }
380 catch (const std::ofstream::failure& oe)
381 {
382 using namespace sdbusplus::xyz::openbmc_project::Common::File::Error;
383 using metadata = xyz::openbmc_project::Common::File::Write;
384
385 lg2::error(
386 "Failed to write to dump file, "
387 "errorMsg({ERROR}), error({ERRORCODE}), filepath({FILEPATH})",
388 "ERROR", oe, "ERRORCODE", oe.code().value(), "FILEPATH",
389 dumpPath.string());
390 report<Write>(metadata::ERRNO(oe.code().value()),
391 metadata::PATH(dumpPath.c_str()));
392 // Just return here so dumps collected from other SBEs can be
393 // packaged.
394 }
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500395}
396
Dhruvaraj Subhashchandranf9f65b82022-10-13 06:46:43 -0500397bool SbeDumpCollector::executeThreadStop(struct pdbg_target* target)
398{
399 try
400 {
401 openpower::phal::sbe::threadStopProc(target);
402 return true;
403 }
404 catch (const openpower::phal::sbeError_t& sbeError)
405 {
406 uint64_t chipPos = pdbg_target_index(target);
407 if (sbeError.errType() ==
408 openpower::phal::exception::SBE_CHIPOP_NOT_ALLOWED)
409 {
410 lg2::info("SBE is not ready to accept chip-op: Skipping "
411 "stop instruction on proc-({POSITION}) error({ERROR}) ",
412 "POSITION", chipPos, "ERROR", sbeError);
413 return false; // Do not include the target for dump collection
414 }
415
416 lg2::error("Stop instructions failed on "
417 "proc-({POSITION}) error({ERROR}) ",
418 "POSITION", chipPos, "ERROR", sbeError);
419
420 logErrorAndCreatePEL(sbeError, chipPos, SBETypes::PROC,
421 SBEFIFO_CMD_CLASS_INSTRUCTION,
422 SBEFIFO_CMD_CONTROL_INSN);
423 // For TIMEOUT, log the error and skip adding the processor for dump
424 // collection
425 if (sbeError.errType() == openpower::phal::exception::SBE_CMD_TIMEOUT)
426 {
427 return false;
428 }
429 }
430 // Include the target for dump collection for SBE_CMD_FAILED or any other
431 // non-critical errors
432 return true;
433}
434
Dhruvaraj Subhashchandran858d1aa2021-10-27 03:26:06 -0500435} // namespace openpower::dump::sbe_chipop