blob: 2cfe71cecee0ffe923ba4fa1815e48e1c96007cd [file] [log] [blame]
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01001/**
Ed Tanousfedd4572024-07-12 13:56:00 -07002 * Describes high level functions for converting an entire CPER log, and functions for parsing
Lawrence Tang2800cd82022-07-05 16:08:20 +01003 * CPER headers and section descriptions into an intermediate JSON format.
Ed Tanousfedd4572024-07-12 13:56:00 -07004 *
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01005 * Author: Lawrence.Tang@arm.com
6 **/
7
Ed Tanous73498f62025-03-05 18:03:36 -08008#include <limits.h>
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01009#include <stdio.h>
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -070010#include <string.h>
Lawrence Tang5202bbb2022-08-12 14:54:36 +010011#include <json.h>
Ed Tanous73498f62025-03-05 18:03:36 -080012
Thu Nguyene42fb482024-10-15 14:43:11 +000013#include <libcper/base64.h>
14#include <libcper/Cper.h>
Ed Tanous50b966f2025-03-11 09:06:19 -070015#include <libcper/log.h>
Thu Nguyene42fb482024-10-15 14:43:11 +000016#include <libcper/cper-parse.h>
17#include <libcper/cper-parse-str.h>
18#include <libcper/cper-utils.h>
19#include <libcper/sections/cper-section.h>
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010020
21//Private pre-definitions.
Lawrence Tange407b4c2022-07-21 13:54:01 +010022json_object *cper_header_to_ir(EFI_COMMON_ERROR_RECORD_HEADER *header);
23json_object *
24cper_section_descriptor_to_ir(EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010025
Ed Tanous73498f62025-03-05 18:03:36 -080026json_object *cper_buf_section_to_ir(const void *cper_section_buf, size_t size,
27 EFI_ERROR_SECTION_DESCRIPTOR *descriptor);
28
29json_object *cper_buf_to_ir(const unsigned char *cper_buf, size_t size)
Ed Tanous8d47a372025-03-05 15:55:36 -080030{
Ed Tanous73498f62025-03-05 18:03:36 -080031 json_object *parent = NULL;
32 json_object *header_ir = NULL;
33 json_object *section_descriptors_ir = NULL;
34 json_object *sections_ir = NULL;
35
36 const unsigned char *pos = cper_buf;
37 unsigned int remaining = size;
38
39 if (remaining < sizeof(EFI_COMMON_ERROR_RECORD_HEADER)) {
Ed Tanous50b966f2025-03-11 09:06:19 -070040 cper_print_log(
41 "Invalid CPER file: Invalid header (incorrect signature).\n");
Ed Tanous73498f62025-03-05 18:03:36 -080042 goto fail;
Ed Tanous8d47a372025-03-05 15:55:36 -080043 }
Ed Tanous73498f62025-03-05 18:03:36 -080044
45 EFI_COMMON_ERROR_RECORD_HEADER *header = NULL;
46 header = (EFI_COMMON_ERROR_RECORD_HEADER *)cper_buf;
47 pos += sizeof(EFI_COMMON_ERROR_RECORD_HEADER);
48 remaining -= sizeof(EFI_COMMON_ERROR_RECORD_HEADER);
49 if (header->SignatureStart != EFI_ERROR_RECORD_SIGNATURE_START) {
Ed Tanous50b966f2025-03-11 09:06:19 -070050 cper_print_log(
51 "Invalid CPER file: Invalid header (incorrect signature).\n");
Ed Tanous73498f62025-03-05 18:03:36 -080052 goto fail;
53 }
54 if (header->SectionCount == 0) {
Ed Tanous50b966f2025-03-11 09:06:19 -070055 cper_print_log(
56 "Invalid CPER file: Invalid section count (0).\n");
Ed Tanous73498f62025-03-05 18:03:36 -080057 goto fail;
58 }
59 if (remaining < sizeof(EFI_ERROR_SECTION_DESCRIPTOR)) {
Ed Tanous50b966f2025-03-11 09:06:19 -070060 cper_print_log(
61 "Invalid CPER file: Invalid section descriptor (section offset + length > size).\n");
Ed Tanous73498f62025-03-05 18:03:36 -080062 goto fail;
63 }
64
65 //Create the header JSON object from the read bytes.
66 parent = json_object_new_object();
67 header_ir = cper_header_to_ir(header);
68
69 json_object_object_add(parent, "header", header_ir);
70
71 //Read the appropriate number of section descriptors & sections, and convert them into IR format.
72 section_descriptors_ir = json_object_new_array();
73 sections_ir = json_object_new_array();
74 for (int i = 0; i < header->SectionCount; i++) {
75 //Create the section descriptor.
76 if (remaining < sizeof(EFI_ERROR_SECTION_DESCRIPTOR)) {
Ed Tanous50b966f2025-03-11 09:06:19 -070077 cper_print_log(
78 "Invalid number of section headers: Header states %d sections, could not read section %d.\n",
79 header->SectionCount, i + 1);
Ed Tanous73498f62025-03-05 18:03:36 -080080 goto fail;
81 }
82
83 EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor;
84 section_descriptor = (EFI_ERROR_SECTION_DESCRIPTOR *)(pos);
85 pos += sizeof(EFI_ERROR_SECTION_DESCRIPTOR);
86 remaining -= sizeof(EFI_ERROR_SECTION_DESCRIPTOR);
87
88 if (section_descriptor->SectionOffset > size) {
Ed Tanous50b966f2025-03-11 09:06:19 -070089 cper_print_log(
90 "Invalid section descriptor: Section offset > size.\n");
Ed Tanous73498f62025-03-05 18:03:36 -080091 goto fail;
92 }
93
94 if (section_descriptor->SectionLength <= 0) {
Ed Tanous50b966f2025-03-11 09:06:19 -070095 cper_print_log(
96 "Invalid section descriptor: Section length <= 0.\n");
Ed Tanous73498f62025-03-05 18:03:36 -080097 goto fail;
98 }
99
100 if (section_descriptor->SectionOffset >
101 UINT_MAX - section_descriptor->SectionLength) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700102 cper_print_log(
103 "Invalid section descriptor: Section offset + length would overflow.\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800104 goto fail;
105 }
106
107 if (section_descriptor->SectionOffset +
108 section_descriptor->SectionLength >
109 size) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700110 cper_print_log(
111 "Invalid section descriptor: Section offset + length > size.\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800112 goto fail;
113 }
114
115 const unsigned char *section_begin =
116 cper_buf + section_descriptor->SectionOffset;
Ed Tanousd6b62632025-03-14 15:30:07 -0700117 json_object *section_descriptor_ir =
118 cper_section_descriptor_to_ir(section_descriptor);
Ed Tanous8e423942025-03-14 23:11:06 -0700119
Ed Tanousd6b62632025-03-14 15:30:07 -0700120 json_object_array_add(section_descriptors_ir,
121 section_descriptor_ir);
Ed Tanous73498f62025-03-05 18:03:36 -0800122
123 //Read the section itself.
124 json_object *section_ir = cper_buf_section_to_ir(
125 section_begin, section_descriptor->SectionLength,
126 section_descriptor);
127 json_object_array_add(sections_ir, section_ir);
128 }
129
130 //Add the header, section descriptors, and sections to a parent object.
131 json_object_object_add(parent, "sectionDescriptors",
132 section_descriptors_ir);
133 json_object_object_add(parent, "sections", sections_ir);
134
135 return parent;
136
137fail:
138 json_object_put(sections_ir);
139 json_object_put(section_descriptors_ir);
140 json_object_put(parent);
Ed Tanous50b966f2025-03-11 09:06:19 -0700141 cper_print_log("Failed to parse CPER file.\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800142 return NULL;
Ed Tanous8d47a372025-03-05 15:55:36 -0800143}
144
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100145//Reads a CPER log file at the given file location, and returns an intermediate
146//JSON representation of this CPER record.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100147json_object *cper_to_ir(FILE *cper_file)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100148{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100149 //Ensure this is really a CPER log.
150 EFI_COMMON_ERROR_RECORD_HEADER header;
Lawrence Tange407b4c2022-07-21 13:54:01 +0100151 if (fread(&header, sizeof(EFI_COMMON_ERROR_RECORD_HEADER), 1,
152 cper_file) != 1) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700153 cper_print_log(
154 "Invalid CPER file: Invalid length (log too short).\n");
Lawrence Tange407b4c2022-07-21 13:54:01 +0100155 return NULL;
156 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100157
Lawrence Tange407b4c2022-07-21 13:54:01 +0100158 //Check if the header contains the magic bytes ("CPER").
159 if (header.SignatureStart != EFI_ERROR_RECORD_SIGNATURE_START) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700160 cper_print_log(
161 "Invalid CPER file: Invalid header (incorrect signature).\n");
Lawrence Tange407b4c2022-07-21 13:54:01 +0100162 return NULL;
163 }
Ed Tanous73498f62025-03-05 18:03:36 -0800164 fseek(cper_file, -sizeof(EFI_COMMON_ERROR_RECORD_HEADER), SEEK_CUR);
165 unsigned char *cper_buf = malloc(header.RecordLength);
166 if (fread(cper_buf, header.RecordLength, 1, cper_file) != 1) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700167 cper_print_log("File read failed\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800168 free(cper_buf);
169 return NULL;
Lawrence Tange407b4c2022-07-21 13:54:01 +0100170 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100171
Ed Tanous73498f62025-03-05 18:03:36 -0800172 json_object *ir = cper_buf_to_ir(cper_buf, header.RecordLength);
173 free(cper_buf);
174 return ir;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100175}
176
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700177char *cper_to_str_ir(FILE *cper_file)
178{
179 json_object *jobj = cper_to_ir(cper_file);
180 char *str = jobj ? strdup(json_object_to_json_string(jobj)) : NULL;
181
182 json_object_put(jobj);
183 return str;
184}
185
186char *cperbuf_to_str_ir(const unsigned char *cper, size_t size)
187{
188 FILE *cper_file = fmemopen((void *)cper, size, "r");
189
190 return cper_file ? cper_to_str_ir(cper_file) : NULL;
191}
192
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100193//Converts a parsed CPER record header into intermediate JSON object format.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100194json_object *cper_header_to_ir(EFI_COMMON_ERROR_RECORD_HEADER *header)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100195{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100196 json_object *header_ir = json_object_new_object();
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100197
Lawrence Tange407b4c2022-07-21 13:54:01 +0100198 //Revision/version information.
199 json_object_object_add(header_ir, "revision",
200 revision_to_ir(header->Revision));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100201
Lawrence Tange407b4c2022-07-21 13:54:01 +0100202 //Section count.
203 json_object_object_add(header_ir, "sectionCount",
204 json_object_new_int(header->SectionCount));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100205
Lawrence Tange407b4c2022-07-21 13:54:01 +0100206 //Error severity (with interpreted string version).
207 json_object *error_severity = json_object_new_object();
208 json_object_object_add(error_severity, "code",
209 json_object_new_uint64(header->ErrorSeverity));
210 json_object_object_add(error_severity, "name",
211 json_object_new_string(severity_to_string(
212 header->ErrorSeverity)));
213 json_object_object_add(header_ir, "severity", error_severity);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100214
Lawrence Tange407b4c2022-07-21 13:54:01 +0100215 //Total length of the record (including headers) in bytes.
216 json_object_object_add(header_ir, "recordLength",
217 json_object_new_uint64(header->RecordLength));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100218
Lawrence Tange407b4c2022-07-21 13:54:01 +0100219 //If a timestamp exists according to validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800220 if (header->ValidationBits & 0x2) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100221 char timestamp_string[TIMESTAMP_LENGTH];
Ed Tanous596c59e2025-03-10 13:15:58 -0700222 if (timestamp_to_string(timestamp_string, TIMESTAMP_LENGTH,
Ed Tanousd6b62632025-03-14 15:30:07 -0700223 &header->TimeStamp) >= 0) {
224 json_object_object_add(
225 header_ir, "timestamp",
226 json_object_new_string(timestamp_string));
Ed Tanous596c59e2025-03-10 13:15:58 -0700227
Ed Tanousd6b62632025-03-14 15:30:07 -0700228 json_object_object_add(header_ir, "timestampIsPrecise",
229 json_object_new_boolean(
230 header->TimeStamp.Flag));
231 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100232 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100233
Lawrence Tange407b4c2022-07-21 13:54:01 +0100234 //If a platform ID exists according to the validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800235 if (header->ValidationBits & 0x1) {
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700236 add_guid(header_ir, "platformID", &header->PlatformID);
Lawrence Tange407b4c2022-07-21 13:54:01 +0100237 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100238
Lawrence Tange407b4c2022-07-21 13:54:01 +0100239 //If a partition ID exists according to the validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800240 if (header->ValidationBits & 0x4) {
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700241 add_guid(header_ir, "partitionID", &header->PartitionID);
Lawrence Tange407b4c2022-07-21 13:54:01 +0100242 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100243
Lawrence Tange407b4c2022-07-21 13:54:01 +0100244 //Creator ID of the header.
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700245 add_guid(header_ir, "creatorID", &header->CreatorID);
Lawrence Tange407b4c2022-07-21 13:54:01 +0100246 //Notification type for the header. Some defined types are available.
247 json_object *notification_type = json_object_new_object();
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700248 add_guid(notification_type, "guid", &header->NotificationType);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100249
Lawrence Tange407b4c2022-07-21 13:54:01 +0100250 //Add the human readable notification type if possible.
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700251 const char *notification_type_readable = "Unknown";
Ed Tanous1a648562025-03-10 15:23:38 -0700252
253 EFI_GUID *guids[] = {
254 &gEfiEventNotificationTypeCmcGuid,
255 &gEfiEventNotificationTypeCpeGuid,
256 &gEfiEventNotificationTypeMceGuid,
257 &gEfiEventNotificationTypePcieGuid,
258 &gEfiEventNotificationTypeInitGuid,
259 &gEfiEventNotificationTypeNmiGuid,
260 &gEfiEventNotificationTypeBootGuid,
261 &gEfiEventNotificationTypeDmarGuid,
262 &gEfiEventNotificationTypeSeaGuid,
263 &gEfiEventNotificationTypeSeiGuid,
264 &gEfiEventNotificationTypePeiGuid,
265 &gEfiEventNotificationTypeCxlGuid,
266 };
267
268 const char *readable_names[] = {
269 "CMC", "CPE", "MCE", "PCIe", "INIT", "NMI",
270 "Boot", "DMAr", "SEA", "SEI", "PEI", "CXL Component"
271 };
272
273 int index = select_guid_from_list(&header->NotificationType, guids,
274 sizeof(guids) / sizeof(EFI_GUID *));
275 if (index < (int)(sizeof(readable_names) / sizeof(char *))) {
276 notification_type_readable = readable_names[index];
John Chungf8fc7052024-05-03 20:05:29 +0800277 }
Ed Tanous1a648562025-03-10 15:23:38 -0700278
Lawrence Tange407b4c2022-07-21 13:54:01 +0100279 json_object_object_add(
280 notification_type, "type",
281 json_object_new_string(notification_type_readable));
282 json_object_object_add(header_ir, "notificationType",
283 notification_type);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100284
Lawrence Tange407b4c2022-07-21 13:54:01 +0100285 //The record ID for this record, unique on a given system.
286 json_object_object_add(header_ir, "recordID",
287 json_object_new_uint64(header->RecordID));
288
289 //Flag for the record, and a human readable form.
290 json_object *flags = integer_to_readable_pair(
291 header->Flags,
292 sizeof(CPER_HEADER_FLAG_TYPES_KEYS) / sizeof(int),
293 CPER_HEADER_FLAG_TYPES_KEYS, CPER_HEADER_FLAG_TYPES_VALUES,
294 "Unknown");
295 json_object_object_add(header_ir, "flags", flags);
296
297 //Persistence information. Outside the scope of specification, so just a uint32 here.
298 json_object_object_add(header_ir, "persistenceInfo",
299 json_object_new_uint64(header->PersistenceInfo));
300 return header_ir;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100301}
302
303//Converts the given EFI section descriptor into JSON IR format.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100304json_object *
305cper_section_descriptor_to_ir(EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100306{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100307 json_object *section_descriptor_ir = json_object_new_object();
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100308
Lawrence Tange407b4c2022-07-21 13:54:01 +0100309 //The offset of the section from the base of the record header, length.
310 json_object_object_add(
311 section_descriptor_ir, "sectionOffset",
312 json_object_new_uint64(section_descriptor->SectionOffset));
313 json_object_object_add(
314 section_descriptor_ir, "sectionLength",
315 json_object_new_uint64(section_descriptor->SectionLength));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100316
Lawrence Tange407b4c2022-07-21 13:54:01 +0100317 //Revision.
318 json_object_object_add(section_descriptor_ir, "revision",
319 revision_to_ir(section_descriptor->Revision));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100320
Lawrence Tange407b4c2022-07-21 13:54:01 +0100321 //Flag bits.
322 json_object *flags =
323 bitfield_to_ir(section_descriptor->SectionFlags, 8,
324 CPER_SECTION_DESCRIPTOR_FLAGS_BITFIELD_NAMES);
325 json_object_object_add(section_descriptor_ir, "flags", flags);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100326
Lawrence Tange407b4c2022-07-21 13:54:01 +0100327 //Section type (GUID).
328 json_object *section_type = json_object_new_object();
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100329
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700330 add_guid(section_type, "data", &section_descriptor->SectionType);
Lawrence Tange407b4c2022-07-21 13:54:01 +0100331 //Readable section type, if possible.
Lawrence Tang580423f2022-08-24 09:37:53 +0100332 const char *section_type_readable = "Unknown";
Ed Tanous1a648562025-03-10 15:23:38 -0700333
334 CPER_SECTION_DEFINITION *section =
335 select_section_by_guid(&section_descriptor->SectionType);
336 if (section != NULL) {
337 section_type_readable = section->ReadableName;
Lawrence Tang580423f2022-08-24 09:37:53 +0100338 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100339
Lawrence Tange407b4c2022-07-21 13:54:01 +0100340 json_object_object_add(section_type, "type",
341 json_object_new_string(section_type_readable));
342 json_object_object_add(section_descriptor_ir, "sectionType",
343 section_type);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100344
Lawrence Tange407b4c2022-07-21 13:54:01 +0100345 //If validation bits indicate it exists, add FRU ID.
John Chungf8fc7052024-05-03 20:05:29 +0800346 if (section_descriptor->SecValidMask & 0x1) {
Ed Tanousc2ebddd2025-03-09 10:07:01 -0700347 add_guid(section_descriptor_ir, "fruID",
348 &section_descriptor->FruId);
Lawrence Tange407b4c2022-07-21 13:54:01 +0100349 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100350
Lawrence Tange407b4c2022-07-21 13:54:01 +0100351 //If validation bits indicate it exists, add FRU text.
John Chungf8fc7052024-05-03 20:05:29 +0800352 if ((section_descriptor->SecValidMask & 0x2) >> 1) {
Ed Tanous8121f7e2025-03-06 14:39:07 -0800353 int fru_text_len = 0;
354 for (;
355 fru_text_len < (int)sizeof(section_descriptor->FruString);
356 fru_text_len++) {
357 char c = section_descriptor->FruString[fru_text_len];
358 if (c < 0) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700359 //cper_print_log("Fru text contains non-ASCII character\n");
Ed Tanous8121f7e2025-03-06 14:39:07 -0800360 fru_text_len = -1;
361 break;
362 }
363 if (c == '\0') {
364 break;
365 }
366 }
367 if (fru_text_len >= 0) {
368 json_object_object_add(
369 section_descriptor_ir, "fruText",
370 json_object_new_string_len(
371 section_descriptor->FruString,
372 fru_text_len));
373 }
John Chungf8fc7052024-05-03 20:05:29 +0800374 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100375
Lawrence Tange407b4c2022-07-21 13:54:01 +0100376 //Section severity.
377 json_object *section_severity = json_object_new_object();
378 json_object_object_add(
379 section_severity, "code",
380 json_object_new_uint64(section_descriptor->Severity));
381 json_object_object_add(section_severity, "name",
382 json_object_new_string(severity_to_string(
383 section_descriptor->Severity)));
384 json_object_object_add(section_descriptor_ir, "severity",
385 section_severity);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100386
Lawrence Tange407b4c2022-07-21 13:54:01 +0100387 return section_descriptor_ir;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100388}
389
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800390json_object *read_section(const unsigned char *cper_section_buf, size_t size,
391 CPER_SECTION_DEFINITION *definition)
Ed Tanousd759a182025-03-07 15:51:45 -0800392{
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800393 if (definition->ToIR == NULL) {
394 return NULL;
395 }
396 json_object *section_ir = definition->ToIR(cper_section_buf, size);
Ed Tanousd6b62632025-03-14 15:30:07 -0700397 if (section_ir == NULL) {
398 return NULL;
399 }
Ed Tanousd759a182025-03-07 15:51:45 -0800400 json_object *result = json_object_new_object();
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800401 json_object_object_add(result, definition->ShortName, section_ir);
Ed Tanousd759a182025-03-07 15:51:45 -0800402 return result;
403}
404
Ed Tanous1a648562025-03-10 15:23:38 -0700405CPER_SECTION_DEFINITION *select_section_by_guid(EFI_GUID *guid)
406{
407 size_t i = 0;
408 for (; i < section_definitions_len; i++) {
409 if (guid_equal(guid, section_definitions[i].Guid)) {
410 break;
411 }
412 }
413 // It's unlikely fuzzing can reliably come up with a correct guid, given how
414 // much entropy there is. If we're in fuzzing mode, and if we haven't found
415 // a match, try to force a match so we get some coverage. Note, we still
416 // want coverage of the section failed to convert code, so treat index ==
417 // size as section failed to convert.
418#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
419 if (i == section_definitions_len) {
420 i = guid->Data1 % (section_definitions_len + 1);
421 }
422#endif
423 if (i < section_definitions_len) {
424 return &section_definitions[i];
425 }
426
427 return NULL;
428}
429
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100430//Converts the section described by a single given section descriptor.
Ed Tanous73498f62025-03-05 18:03:36 -0800431json_object *cper_buf_section_to_ir(const void *cper_section_buf, size_t size,
432 EFI_ERROR_SECTION_DESCRIPTOR *descriptor)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100433{
Ed Tanous73498f62025-03-05 18:03:36 -0800434 if (descriptor->SectionLength > size) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700435 cper_print_log(
436 "Invalid CPER file: Invalid header (incorrect signature).\n");
Lawrence Tange407b4c2022-07-21 13:54:01 +0100437 return NULL;
438 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100439
Lawrence Tange407b4c2022-07-21 13:54:01 +0100440 //Parse section to IR based on GUID.
441 json_object *result = NULL;
Ed Tanousb07061a2024-09-22 10:33:29 -0700442 json_object *section_ir = NULL;
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800443
Ed Tanous1a648562025-03-10 15:23:38 -0700444 CPER_SECTION_DEFINITION *section =
445 select_section_by_guid(&descriptor->SectionType);
Ed Tanousd6b62632025-03-14 15:30:07 -0700446 if (section == NULL) {
447 cper_print_log("Unknown section type guid\n");
448 } else {
Ed Tanous1a648562025-03-10 15:23:38 -0700449 result = read_section(cper_section_buf, size, section);
Ed Tanousd759a182025-03-07 15:51:45 -0800450 }
Ed Tanousb07061a2024-09-22 10:33:29 -0700451
Lawrence Tang580423f2022-08-24 09:37:53 +0100452 //Was it an unknown GUID/failed read?
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800453 if (result == NULL) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100454 //Output the data as formatted base64.
Ed Tanousa7d2cdd2024-07-15 11:07:27 -0700455 int32_t encoded_len = 0;
Ed Tanous73498f62025-03-05 18:03:36 -0800456 char *encoded = base64_encode(cper_section_buf,
457 descriptor->SectionLength,
458 &encoded_len);
Ed Tanousa7d2cdd2024-07-15 11:07:27 -0700459 if (encoded == NULL) {
Ed Tanousd6b62632025-03-14 15:30:07 -0700460 cper_print_log(
461 "Failed to allocate encode output buffer. \n");
John Chungf8fc7052024-05-03 20:05:29 +0800462 } else {
Ed Tanousb07061a2024-09-22 10:33:29 -0700463 section_ir = json_object_new_object();
464 json_object_object_add(section_ir, "data",
John Chungf8fc7052024-05-03 20:05:29 +0800465 json_object_new_string_len(
466 encoded, encoded_len));
467 free(encoded);
Ed Tanousb07061a2024-09-22 10:33:29 -0700468
469 result = json_object_new_object();
470 json_object_object_add(result, "Unknown", section_ir);
John Chungf8fc7052024-05-03 20:05:29 +0800471 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100472 }
Ed Tanousd6b62632025-03-14 15:30:07 -0700473 if (result == NULL) {
474 cper_print_log("RETURNING NULL!! !!\n");
475 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100476 return result;
Lawrence Tang617949e2022-08-08 14:21:42 +0100477}
478
Ed Tanous73498f62025-03-05 18:03:36 -0800479json_object *cper_buf_single_section_to_ir(const unsigned char *cper_buf,
480 size_t size)
Ed Tanous8d47a372025-03-05 15:55:36 -0800481{
Ed Tanous8121f7e2025-03-06 14:39:07 -0800482 const unsigned char *cper_end;
483 const unsigned char *section_begin;
484 json_object *ir;
485
486 cper_end = cper_buf + size;
Ed Tanous73498f62025-03-05 18:03:36 -0800487
488 //Read the section descriptor out.
489 EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor;
490 if (sizeof(EFI_ERROR_SECTION_DESCRIPTOR) > size) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700491 cper_print_log(
492 "Size of cper buffer was too small to read section descriptor %zu\n",
493 size);
Ed Tanous8d47a372025-03-05 15:55:36 -0800494 return NULL;
495 }
Ed Tanous8121f7e2025-03-06 14:39:07 -0800496
497 ir = json_object_new_object();
Ed Tanous73498f62025-03-05 18:03:36 -0800498 section_descriptor = (EFI_ERROR_SECTION_DESCRIPTOR *)cper_buf;
499 //Convert the section descriptor to IR.
500 json_object *section_descriptor_ir =
501 cper_section_descriptor_to_ir(section_descriptor);
502 json_object_object_add(ir, "sectionDescriptor", section_descriptor_ir);
Ed Tanous8121f7e2025-03-06 14:39:07 -0800503 section_begin = cper_buf + section_descriptor->SectionOffset;
Ed Tanous73498f62025-03-05 18:03:36 -0800504
Ed Tanous8121f7e2025-03-06 14:39:07 -0800505 if (section_begin + section_descriptor->SectionLength >= cper_end) {
506 json_object_put(ir);
Ed Tanous50b966f2025-03-11 09:06:19 -0700507 //cper_print_log("Invalid CPER file: Invalid section descriptor (section offset + length > size).\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800508 return NULL;
509 }
510
511 const unsigned char *section =
512 cper_buf + section_descriptor->SectionOffset;
513
514 //Parse the single section.
515 json_object *section_ir = cper_buf_section_to_ir(
516 section, section_descriptor->SectionLength, section_descriptor);
Ed Tanousd6b62632025-03-14 15:30:07 -0700517 if (section_ir == NULL) {
518 cper_print_log("RETURNING NULL2!! !!\n");
519 }
Ed Tanous73498f62025-03-05 18:03:36 -0800520 json_object_object_add(ir, "section", section_ir);
Ed Tanous8d47a372025-03-05 15:55:36 -0800521 return ir;
522}
523
Lawrence Tang617949e2022-08-08 14:21:42 +0100524//Converts a single CPER section, without a header but with a section descriptor, to JSON.
525json_object *cper_single_section_to_ir(FILE *cper_section_file)
526{
527 json_object *ir = json_object_new_object();
528
Lawrence Tang94153492022-09-05 13:07:54 +0100529 //Read the current file pointer location as base record position.
530 long base_pos = ftell(cper_section_file);
531
Lawrence Tang617949e2022-08-08 14:21:42 +0100532 //Read the section descriptor out.
533 EFI_ERROR_SECTION_DESCRIPTOR section_descriptor;
534 if (fread(&section_descriptor, sizeof(EFI_ERROR_SECTION_DESCRIPTOR), 1,
535 cper_section_file) != 1) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700536 cper_print_log(
537 "Failed to read section descriptor for CPER single section (fread() returned an unexpected value).\n");
Ed Tanous8121f7e2025-03-06 14:39:07 -0800538 json_object_put(ir);
Lawrence Tang617949e2022-08-08 14:21:42 +0100539 return NULL;
540 }
541
542 //Convert the section descriptor to IR.
543 json_object *section_descriptor_ir =
544 cper_section_descriptor_to_ir(&section_descriptor);
545 json_object_object_add(ir, "sectionDescriptor", section_descriptor_ir);
546
Ed Tanous73498f62025-03-05 18:03:36 -0800547 //Save our current position in the stream.
548 long position = ftell(cper_section_file);
549
550 //Read section as described by the section descriptor.
551 fseek(cper_section_file, base_pos + section_descriptor.SectionOffset,
552 SEEK_SET);
553 void *section = malloc(section_descriptor.SectionLength);
554 if (fread(section, section_descriptor.SectionLength, 1,
555 cper_section_file) != 1) {
Ed Tanous50b966f2025-03-11 09:06:19 -0700556 cper_print_log(
557 "Section read failed: Could not read %u bytes from global offset %d.\n",
558 section_descriptor.SectionLength,
559 section_descriptor.SectionOffset);
Ed Tanous8121f7e2025-03-06 14:39:07 -0800560 json_object_put(ir);
Ed Tanous73498f62025-03-05 18:03:36 -0800561 free(section);
562 return NULL;
563 }
564
565 //Seek back to our original position.
566 fseek(cper_section_file, position, SEEK_SET);
567
Lawrence Tang617949e2022-08-08 14:21:42 +0100568 //Parse the single section.
Ed Tanous73498f62025-03-05 18:03:36 -0800569 json_object *section_ir = cper_buf_section_to_ir(
570 section, section_descriptor.SectionLength, &section_descriptor);
Lawrence Tang617949e2022-08-08 14:21:42 +0100571 json_object_object_add(ir, "section", section_ir);
Ed Tanous73498f62025-03-05 18:03:36 -0800572 free(section);
Lawrence Tang617949e2022-08-08 14:21:42 +0100573 return ir;
John Chungf8fc7052024-05-03 20:05:29 +0800574}
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700575
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700576char *cperbuf_single_section_to_str_ir(const unsigned char *cper_section,
577 size_t size)
578{
Ed Tanous73498f62025-03-05 18:03:36 -0800579 json_object *jobj = cper_buf_single_section_to_ir(cper_section, size);
580 char *str = jobj ? strdup(json_object_to_json_string(jobj)) : NULL;
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700581
Ed Tanous73498f62025-03-05 18:03:36 -0800582 json_object_put(jobj);
583 return str;
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700584}