blob: 8b62ef629f69aca7370412d1db292777821e12fc [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>
15#include <libcper/cper-parse.h>
16#include <libcper/cper-parse-str.h>
17#include <libcper/cper-utils.h>
18#include <libcper/sections/cper-section.h>
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010019
20//Private pre-definitions.
Lawrence Tange407b4c2022-07-21 13:54:01 +010021json_object *cper_header_to_ir(EFI_COMMON_ERROR_RECORD_HEADER *header);
22json_object *
23cper_section_descriptor_to_ir(EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010024
Ed Tanous73498f62025-03-05 18:03:36 -080025json_object *cper_buf_section_to_ir(const void *cper_section_buf, size_t size,
26 EFI_ERROR_SECTION_DESCRIPTOR *descriptor);
27
28json_object *cper_buf_to_ir(const unsigned char *cper_buf, size_t size)
Ed Tanous8d47a372025-03-05 15:55:36 -080029{
Ed Tanous73498f62025-03-05 18:03:36 -080030 json_object *parent = NULL;
31 json_object *header_ir = NULL;
32 json_object *section_descriptors_ir = NULL;
33 json_object *sections_ir = NULL;
34
35 const unsigned char *pos = cper_buf;
36 unsigned int remaining = size;
37
38 if (remaining < sizeof(EFI_COMMON_ERROR_RECORD_HEADER)) {
39 printf("Invalid CPER file: Invalid header (incorrect signature).\n");
40 goto fail;
Ed Tanous8d47a372025-03-05 15:55:36 -080041 }
Ed Tanous73498f62025-03-05 18:03:36 -080042
43 EFI_COMMON_ERROR_RECORD_HEADER *header = NULL;
44 header = (EFI_COMMON_ERROR_RECORD_HEADER *)cper_buf;
45 pos += sizeof(EFI_COMMON_ERROR_RECORD_HEADER);
46 remaining -= sizeof(EFI_COMMON_ERROR_RECORD_HEADER);
47 if (header->SignatureStart != EFI_ERROR_RECORD_SIGNATURE_START) {
48 printf("Invalid CPER file: Invalid header (incorrect signature).\n");
49 goto fail;
50 }
51 if (header->SectionCount == 0) {
52 printf("Invalid CPER file: Invalid section count (0).\n");
53 goto fail;
54 }
55 if (remaining < sizeof(EFI_ERROR_SECTION_DESCRIPTOR)) {
56 printf("Invalid CPER file: Invalid section descriptor (section offset + length > size).\n");
57 goto fail;
58 }
59
60 //Create the header JSON object from the read bytes.
61 parent = json_object_new_object();
62 header_ir = cper_header_to_ir(header);
63
64 json_object_object_add(parent, "header", header_ir);
65
66 //Read the appropriate number of section descriptors & sections, and convert them into IR format.
67 section_descriptors_ir = json_object_new_array();
68 sections_ir = json_object_new_array();
69 for (int i = 0; i < header->SectionCount; i++) {
70 //Create the section descriptor.
71 if (remaining < sizeof(EFI_ERROR_SECTION_DESCRIPTOR)) {
72 printf("Invalid number of section headers: Header states %d sections, could not read section %d.\n",
73 header->SectionCount, i + 1);
74 goto fail;
75 }
76
77 EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor;
78 section_descriptor = (EFI_ERROR_SECTION_DESCRIPTOR *)(pos);
79 pos += sizeof(EFI_ERROR_SECTION_DESCRIPTOR);
80 remaining -= sizeof(EFI_ERROR_SECTION_DESCRIPTOR);
81
82 if (section_descriptor->SectionOffset > size) {
83 printf("Invalid section descriptor: Section offset > size.\n");
84 goto fail;
85 }
86
87 if (section_descriptor->SectionLength <= 0) {
88 printf("Invalid section descriptor: Section length <= 0.\n");
89 goto fail;
90 }
91
92 if (section_descriptor->SectionOffset >
93 UINT_MAX - section_descriptor->SectionLength) {
94 printf("Invalid section descriptor: Section offset + length would overflow.\n");
95 goto fail;
96 }
97
98 if (section_descriptor->SectionOffset +
99 section_descriptor->SectionLength >
100 size) {
101 printf("Invalid section descriptor: Section offset + length > size.\n");
102 goto fail;
103 }
104
105 const unsigned char *section_begin =
106 cper_buf + section_descriptor->SectionOffset;
107
108 json_object_array_add(
109 section_descriptors_ir,
110 cper_section_descriptor_to_ir(section_descriptor));
111
112 //Read the section itself.
113 json_object *section_ir = cper_buf_section_to_ir(
114 section_begin, section_descriptor->SectionLength,
115 section_descriptor);
116 json_object_array_add(sections_ir, section_ir);
117 }
118
119 //Add the header, section descriptors, and sections to a parent object.
120 json_object_object_add(parent, "sectionDescriptors",
121 section_descriptors_ir);
122 json_object_object_add(parent, "sections", sections_ir);
123
124 return parent;
125
126fail:
127 json_object_put(sections_ir);
128 json_object_put(section_descriptors_ir);
129 json_object_put(parent);
130 printf("Failed to parse CPER file.\n");
131 return NULL;
Ed Tanous8d47a372025-03-05 15:55:36 -0800132}
133
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100134//Reads a CPER log file at the given file location, and returns an intermediate
135//JSON representation of this CPER record.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100136json_object *cper_to_ir(FILE *cper_file)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100137{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100138 //Ensure this is really a CPER log.
139 EFI_COMMON_ERROR_RECORD_HEADER header;
Lawrence Tange407b4c2022-07-21 13:54:01 +0100140 if (fread(&header, sizeof(EFI_COMMON_ERROR_RECORD_HEADER), 1,
141 cper_file) != 1) {
142 printf("Invalid CPER file: Invalid length (log too short).\n");
143 return NULL;
144 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100145
Lawrence Tange407b4c2022-07-21 13:54:01 +0100146 //Check if the header contains the magic bytes ("CPER").
147 if (header.SignatureStart != EFI_ERROR_RECORD_SIGNATURE_START) {
148 printf("Invalid CPER file: Invalid header (incorrect signature).\n");
149 return NULL;
150 }
Ed Tanous73498f62025-03-05 18:03:36 -0800151 fseek(cper_file, -sizeof(EFI_COMMON_ERROR_RECORD_HEADER), SEEK_CUR);
152 unsigned char *cper_buf = malloc(header.RecordLength);
153 if (fread(cper_buf, header.RecordLength, 1, cper_file) != 1) {
154 printf("File read failed\n");
155 free(cper_buf);
156 return NULL;
Lawrence Tange407b4c2022-07-21 13:54:01 +0100157 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100158
Ed Tanous73498f62025-03-05 18:03:36 -0800159 json_object *ir = cper_buf_to_ir(cper_buf, header.RecordLength);
160 free(cper_buf);
161 return ir;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100162}
163
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700164char *cper_to_str_ir(FILE *cper_file)
165{
166 json_object *jobj = cper_to_ir(cper_file);
167 char *str = jobj ? strdup(json_object_to_json_string(jobj)) : NULL;
168
169 json_object_put(jobj);
170 return str;
171}
172
173char *cperbuf_to_str_ir(const unsigned char *cper, size_t size)
174{
175 FILE *cper_file = fmemopen((void *)cper, size, "r");
176
177 return cper_file ? cper_to_str_ir(cper_file) : NULL;
178}
179
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100180//Converts a parsed CPER record header into intermediate JSON object format.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100181json_object *cper_header_to_ir(EFI_COMMON_ERROR_RECORD_HEADER *header)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100182{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100183 json_object *header_ir = json_object_new_object();
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100184
Lawrence Tange407b4c2022-07-21 13:54:01 +0100185 //Revision/version information.
186 json_object_object_add(header_ir, "revision",
187 revision_to_ir(header->Revision));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100188
Lawrence Tange407b4c2022-07-21 13:54:01 +0100189 //Section count.
190 json_object_object_add(header_ir, "sectionCount",
191 json_object_new_int(header->SectionCount));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100192
Lawrence Tange407b4c2022-07-21 13:54:01 +0100193 //Error severity (with interpreted string version).
194 json_object *error_severity = json_object_new_object();
195 json_object_object_add(error_severity, "code",
196 json_object_new_uint64(header->ErrorSeverity));
197 json_object_object_add(error_severity, "name",
198 json_object_new_string(severity_to_string(
199 header->ErrorSeverity)));
200 json_object_object_add(header_ir, "severity", error_severity);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100201
Lawrence Tange407b4c2022-07-21 13:54:01 +0100202 //Total length of the record (including headers) in bytes.
203 json_object_object_add(header_ir, "recordLength",
204 json_object_new_uint64(header->RecordLength));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100205
Lawrence Tange407b4c2022-07-21 13:54:01 +0100206 //If a timestamp exists according to validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800207 if (header->ValidationBits & 0x2) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100208 char timestamp_string[TIMESTAMP_LENGTH];
Ed Tanous596c59e2025-03-10 13:15:58 -0700209 if (timestamp_to_string(timestamp_string, TIMESTAMP_LENGTH,
210 &header->TimeStamp) < 0) {
211 goto fail;
212 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100213 json_object_object_add(
214 header_ir, "timestamp",
215 json_object_new_string(timestamp_string));
Ed Tanous596c59e2025-03-10 13:15:58 -0700216
Lawrence Tange407b4c2022-07-21 13:54:01 +0100217 json_object_object_add(
218 header_ir, "timestampIsPrecise",
219 json_object_new_boolean(header->TimeStamp.Flag));
220 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100221
Lawrence Tange407b4c2022-07-21 13:54:01 +0100222 //If a platform ID exists according to the validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800223 if (header->ValidationBits & 0x1) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100224 char platform_string[GUID_STRING_LENGTH];
225 guid_to_string(platform_string, &header->PlatformID);
226 json_object_object_add(header_ir, "platformID",
227 json_object_new_string(platform_string));
228 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100229
Lawrence Tange407b4c2022-07-21 13:54:01 +0100230 //If a partition ID exists according to the validation bits, then add it.
John Chungf8fc7052024-05-03 20:05:29 +0800231 if (header->ValidationBits & 0x4) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100232 char partition_string[GUID_STRING_LENGTH];
233 guid_to_string(partition_string, &header->PartitionID);
234 json_object_object_add(
235 header_ir, "partitionID",
236 json_object_new_string(partition_string));
237 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100238
Lawrence Tange407b4c2022-07-21 13:54:01 +0100239 //Creator ID of the header.
240 char creator_string[GUID_STRING_LENGTH];
241 guid_to_string(creator_string, &header->CreatorID);
242 json_object_object_add(header_ir, "creatorID",
243 json_object_new_string(creator_string));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100244
Lawrence Tange407b4c2022-07-21 13:54:01 +0100245 //Notification type for the header. Some defined types are available.
246 json_object *notification_type = json_object_new_object();
247 char notification_type_string[GUID_STRING_LENGTH];
248 guid_to_string(notification_type_string, &header->NotificationType);
249 json_object_object_add(
250 notification_type, "guid",
251 json_object_new_string(notification_type_string));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100252
Lawrence Tange407b4c2022-07-21 13:54:01 +0100253 //Add the human readable notification type if possible.
254 char *notification_type_readable = "Unknown";
255 if (guid_equal(&header->NotificationType,
John Chungf8fc7052024-05-03 20:05:29 +0800256 &gEfiEventNotificationTypeCmcGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100257 notification_type_readable = "CMC";
John Chungf8fc7052024-05-03 20:05:29 +0800258 } else if (guid_equal(&header->NotificationType,
259 &gEfiEventNotificationTypeCpeGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100260 notification_type_readable = "CPE";
John Chungf8fc7052024-05-03 20:05:29 +0800261 } else if (guid_equal(&header->NotificationType,
262 &gEfiEventNotificationTypeMceGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100263 notification_type_readable = "MCE";
John Chungf8fc7052024-05-03 20:05:29 +0800264 } else if (guid_equal(&header->NotificationType,
265 &gEfiEventNotificationTypePcieGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100266 notification_type_readable = "PCIe";
John Chungf8fc7052024-05-03 20:05:29 +0800267 } else if (guid_equal(&header->NotificationType,
268 &gEfiEventNotificationTypeInitGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100269 notification_type_readable = "INIT";
John Chungf8fc7052024-05-03 20:05:29 +0800270 } else if (guid_equal(&header->NotificationType,
271 &gEfiEventNotificationTypeNmiGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100272 notification_type_readable = "NMI";
John Chungf8fc7052024-05-03 20:05:29 +0800273 } else if (guid_equal(&header->NotificationType,
274 &gEfiEventNotificationTypeBootGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100275 notification_type_readable = "Boot";
John Chungf8fc7052024-05-03 20:05:29 +0800276 } else if (guid_equal(&header->NotificationType,
277 &gEfiEventNotificationTypeDmarGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100278 notification_type_readable = "DMAr";
John Chungf8fc7052024-05-03 20:05:29 +0800279 } else if (guid_equal(&header->NotificationType,
280 &gEfiEventNotificationTypeSeaGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100281 notification_type_readable = "SEA";
John Chungf8fc7052024-05-03 20:05:29 +0800282 } else if (guid_equal(&header->NotificationType,
283 &gEfiEventNotificationTypeSeiGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100284 notification_type_readable = "SEI";
John Chungf8fc7052024-05-03 20:05:29 +0800285 } else if (guid_equal(&header->NotificationType,
286 &gEfiEventNotificationTypePeiGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100287 notification_type_readable = "PEI";
John Chungf8fc7052024-05-03 20:05:29 +0800288 } else if (guid_equal(&header->NotificationType,
289 &gEfiEventNotificationTypeCxlGuid)) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100290 notification_type_readable = "CXL Component";
John Chungf8fc7052024-05-03 20:05:29 +0800291 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100292 json_object_object_add(
293 notification_type, "type",
294 json_object_new_string(notification_type_readable));
295 json_object_object_add(header_ir, "notificationType",
296 notification_type);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100297
Lawrence Tange407b4c2022-07-21 13:54:01 +0100298 //The record ID for this record, unique on a given system.
299 json_object_object_add(header_ir, "recordID",
300 json_object_new_uint64(header->RecordID));
301
302 //Flag for the record, and a human readable form.
303 json_object *flags = integer_to_readable_pair(
304 header->Flags,
305 sizeof(CPER_HEADER_FLAG_TYPES_KEYS) / sizeof(int),
306 CPER_HEADER_FLAG_TYPES_KEYS, CPER_HEADER_FLAG_TYPES_VALUES,
307 "Unknown");
308 json_object_object_add(header_ir, "flags", flags);
309
310 //Persistence information. Outside the scope of specification, so just a uint32 here.
311 json_object_object_add(header_ir, "persistenceInfo",
312 json_object_new_uint64(header->PersistenceInfo));
313 return header_ir;
Ed Tanous596c59e2025-03-10 13:15:58 -0700314
315fail:
316 json_object_put(header_ir);
317 return NULL;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100318}
319
320//Converts the given EFI section descriptor into JSON IR format.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100321json_object *
322cper_section_descriptor_to_ir(EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100323{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100324 json_object *section_descriptor_ir = json_object_new_object();
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100325
Lawrence Tange407b4c2022-07-21 13:54:01 +0100326 //The offset of the section from the base of the record header, length.
327 json_object_object_add(
328 section_descriptor_ir, "sectionOffset",
329 json_object_new_uint64(section_descriptor->SectionOffset));
330 json_object_object_add(
331 section_descriptor_ir, "sectionLength",
332 json_object_new_uint64(section_descriptor->SectionLength));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100333
Lawrence Tange407b4c2022-07-21 13:54:01 +0100334 //Revision.
335 json_object_object_add(section_descriptor_ir, "revision",
336 revision_to_ir(section_descriptor->Revision));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100337
Lawrence Tange407b4c2022-07-21 13:54:01 +0100338 //Flag bits.
339 json_object *flags =
340 bitfield_to_ir(section_descriptor->SectionFlags, 8,
341 CPER_SECTION_DESCRIPTOR_FLAGS_BITFIELD_NAMES);
342 json_object_object_add(section_descriptor_ir, "flags", flags);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100343
Lawrence Tange407b4c2022-07-21 13:54:01 +0100344 //Section type (GUID).
345 json_object *section_type = json_object_new_object();
346 char section_type_string[GUID_STRING_LENGTH];
347 guid_to_string(section_type_string, &section_descriptor->SectionType);
348 json_object_object_add(section_type, "data",
349 json_object_new_string(section_type_string));
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100350
Lawrence Tange407b4c2022-07-21 13:54:01 +0100351 //Readable section type, if possible.
Lawrence Tang580423f2022-08-24 09:37:53 +0100352 const char *section_type_readable = "Unknown";
John Chungf8fc7052024-05-03 20:05:29 +0800353 for (size_t i = 0; i < section_definitions_len; i++) {
Lawrence Tangf1f3b832022-08-24 09:42:39 +0100354 if (guid_equal(section_definitions[i].Guid,
355 &section_descriptor->SectionType)) {
356 section_type_readable =
357 section_definitions[i].ReadableName;
Lawrence Tang580423f2022-08-24 09:37:53 +0100358 break;
359 }
360 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100361
Lawrence Tange407b4c2022-07-21 13:54:01 +0100362 json_object_object_add(section_type, "type",
363 json_object_new_string(section_type_readable));
364 json_object_object_add(section_descriptor_ir, "sectionType",
365 section_type);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100366
Lawrence Tange407b4c2022-07-21 13:54:01 +0100367 //If validation bits indicate it exists, add FRU ID.
John Chungf8fc7052024-05-03 20:05:29 +0800368 if (section_descriptor->SecValidMask & 0x1) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100369 char fru_id_string[GUID_STRING_LENGTH];
370 guid_to_string(fru_id_string, &section_descriptor->FruId);
371 json_object_object_add(section_descriptor_ir, "fruID",
372 json_object_new_string(fru_id_string));
373 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100374
Lawrence Tange407b4c2022-07-21 13:54:01 +0100375 //If validation bits indicate it exists, add FRU text.
John Chungf8fc7052024-05-03 20:05:29 +0800376 if ((section_descriptor->SecValidMask & 0x2) >> 1) {
Ed Tanous8121f7e2025-03-06 14:39:07 -0800377 int fru_text_len = 0;
378 for (;
379 fru_text_len < (int)sizeof(section_descriptor->FruString);
380 fru_text_len++) {
381 char c = section_descriptor->FruString[fru_text_len];
382 if (c < 0) {
383 //printf("Fru text contains non-ASCII character\n");
384 fru_text_len = -1;
385 break;
386 }
387 if (c == '\0') {
388 break;
389 }
390 }
391 if (fru_text_len >= 0) {
392 json_object_object_add(
393 section_descriptor_ir, "fruText",
394 json_object_new_string_len(
395 section_descriptor->FruString,
396 fru_text_len));
397 }
John Chungf8fc7052024-05-03 20:05:29 +0800398 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100399
Lawrence Tange407b4c2022-07-21 13:54:01 +0100400 //Section severity.
401 json_object *section_severity = json_object_new_object();
402 json_object_object_add(
403 section_severity, "code",
404 json_object_new_uint64(section_descriptor->Severity));
405 json_object_object_add(section_severity, "name",
406 json_object_new_string(severity_to_string(
407 section_descriptor->Severity)));
408 json_object_object_add(section_descriptor_ir, "severity",
409 section_severity);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100410
Lawrence Tange407b4c2022-07-21 13:54:01 +0100411 return section_descriptor_ir;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100412}
413
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800414json_object *read_section(const unsigned char *cper_section_buf, size_t size,
415 CPER_SECTION_DEFINITION *definition)
Ed Tanousd759a182025-03-07 15:51:45 -0800416{
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800417 if (definition->ToIR == NULL) {
418 return NULL;
419 }
420 json_object *section_ir = definition->ToIR(cper_section_buf, size);
Ed Tanousd759a182025-03-07 15:51:45 -0800421 json_object *result = json_object_new_object();
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800422 json_object_object_add(result, definition->ShortName, section_ir);
Ed Tanousd759a182025-03-07 15:51:45 -0800423 return result;
424}
425
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100426//Converts the section described by a single given section descriptor.
Ed Tanous73498f62025-03-05 18:03:36 -0800427json_object *cper_buf_section_to_ir(const void *cper_section_buf, size_t size,
428 EFI_ERROR_SECTION_DESCRIPTOR *descriptor)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100429{
Ed Tanous73498f62025-03-05 18:03:36 -0800430 if (descriptor->SectionLength > size) {
431 printf("Invalid CPER file: Invalid header (incorrect signature).\n");
Lawrence Tange407b4c2022-07-21 13:54:01 +0100432 return NULL;
433 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100434
Lawrence Tange407b4c2022-07-21 13:54:01 +0100435 //Parse section to IR based on GUID.
436 json_object *result = NULL;
Ed Tanousb07061a2024-09-22 10:33:29 -0700437
438 json_object *section_ir = NULL;
John Chungf8fc7052024-05-03 20:05:29 +0800439 for (size_t i = 0; i < section_definitions_len; i++) {
Ed Tanousd759a182025-03-07 15:51:45 -0800440 if (!guid_equal(section_definitions[i].Guid,
441 &descriptor->SectionType)) {
442 continue;
443 }
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800444 result = read_section(cper_section_buf, size,
445 &section_definitions[i]);
446
Ed Tanousd759a182025-03-07 15:51:45 -0800447 break;
448 }
Ed Tanousb07061a2024-09-22 10:33:29 -0700449
Ed Tanousd759a182025-03-07 15:51:45 -0800450#ifdef FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION
451 // It's unlikely fuzzing can reliably come up with a correct guid, given how
452 // much entropy there is. If we're in fuzzing mode, and if we haven't found
453 // a match, try to force a match so we get some coverage. Note, we still
454 // want coverage of the section failed to convert code, so treat index ==
455 // size as section failed to convert.
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800456 if (result == NULL) {
Ed Tanousd759a182025-03-07 15:51:45 -0800457 unsigned char index = 0;
458 if (index > 0) {
459 index = descriptor->SectionType.Data1 %
460 section_definitions_len;
461 }
462 if (index < section_definitions_len) {
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800463 result = read_section(cper_section_buf, size,
464 &section_definitions[index]);
Lawrence Tang580423f2022-08-24 09:37:53 +0100465 }
466 }
Ed Tanousd759a182025-03-07 15:51:45 -0800467#endif
Lawrence Tang580423f2022-08-24 09:37:53 +0100468
469 //Was it an unknown GUID/failed read?
Ed Tanous12dbd4f2025-03-08 19:05:01 -0800470 if (result == NULL) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100471 //Output the data as formatted base64.
Ed Tanousa7d2cdd2024-07-15 11:07:27 -0700472 int32_t encoded_len = 0;
Ed Tanous73498f62025-03-05 18:03:36 -0800473 char *encoded = base64_encode(cper_section_buf,
474 descriptor->SectionLength,
475 &encoded_len);
Ed Tanousa7d2cdd2024-07-15 11:07:27 -0700476 if (encoded == NULL) {
Ed Tanous8121f7e2025-03-06 14:39:07 -0800477 //printf("Failed to allocate encode output buffer. \n");
John Chungf8fc7052024-05-03 20:05:29 +0800478 } else {
Ed Tanousb07061a2024-09-22 10:33:29 -0700479 section_ir = json_object_new_object();
480 json_object_object_add(section_ir, "data",
John Chungf8fc7052024-05-03 20:05:29 +0800481 json_object_new_string_len(
482 encoded, encoded_len));
483 free(encoded);
Ed Tanousb07061a2024-09-22 10:33:29 -0700484
485 result = json_object_new_object();
486 json_object_object_add(result, "Unknown", section_ir);
John Chungf8fc7052024-05-03 20:05:29 +0800487 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100488 }
Ed Tanousb07061a2024-09-22 10:33:29 -0700489
Lawrence Tange407b4c2022-07-21 13:54:01 +0100490 return result;
Lawrence Tang617949e2022-08-08 14:21:42 +0100491}
492
Ed Tanous73498f62025-03-05 18:03:36 -0800493json_object *cper_buf_single_section_to_ir(const unsigned char *cper_buf,
494 size_t size)
Ed Tanous8d47a372025-03-05 15:55:36 -0800495{
Ed Tanous8121f7e2025-03-06 14:39:07 -0800496 const unsigned char *cper_end;
497 const unsigned char *section_begin;
498 json_object *ir;
499
500 cper_end = cper_buf + size;
Ed Tanous73498f62025-03-05 18:03:36 -0800501
502 //Read the section descriptor out.
503 EFI_ERROR_SECTION_DESCRIPTOR *section_descriptor;
504 if (sizeof(EFI_ERROR_SECTION_DESCRIPTOR) > size) {
Ed Tanousd759a182025-03-07 15:51:45 -0800505 printf("Size of cper buffer was too small to read section descriptor %zu\n",
506 size);
Ed Tanous8d47a372025-03-05 15:55:36 -0800507 return NULL;
508 }
Ed Tanous8121f7e2025-03-06 14:39:07 -0800509
510 ir = json_object_new_object();
Ed Tanous73498f62025-03-05 18:03:36 -0800511 section_descriptor = (EFI_ERROR_SECTION_DESCRIPTOR *)cper_buf;
512 //Convert the section descriptor to IR.
513 json_object *section_descriptor_ir =
514 cper_section_descriptor_to_ir(section_descriptor);
515 json_object_object_add(ir, "sectionDescriptor", section_descriptor_ir);
Ed Tanous8121f7e2025-03-06 14:39:07 -0800516 section_begin = cper_buf + section_descriptor->SectionOffset;
Ed Tanous73498f62025-03-05 18:03:36 -0800517
Ed Tanous8121f7e2025-03-06 14:39:07 -0800518 if (section_begin + section_descriptor->SectionLength >= cper_end) {
519 json_object_put(ir);
520 //printf("Invalid CPER file: Invalid section descriptor (section offset + length > size).\n");
Ed Tanous73498f62025-03-05 18:03:36 -0800521 return NULL;
522 }
523
524 const unsigned char *section =
525 cper_buf + section_descriptor->SectionOffset;
526
527 //Parse the single section.
528 json_object *section_ir = cper_buf_section_to_ir(
529 section, section_descriptor->SectionLength, section_descriptor);
530 json_object_object_add(ir, "section", section_ir);
Ed Tanous8d47a372025-03-05 15:55:36 -0800531 return ir;
532}
533
Lawrence Tang617949e2022-08-08 14:21:42 +0100534//Converts a single CPER section, without a header but with a section descriptor, to JSON.
535json_object *cper_single_section_to_ir(FILE *cper_section_file)
536{
537 json_object *ir = json_object_new_object();
538
Lawrence Tang94153492022-09-05 13:07:54 +0100539 //Read the current file pointer location as base record position.
540 long base_pos = ftell(cper_section_file);
541
Lawrence Tang617949e2022-08-08 14:21:42 +0100542 //Read the section descriptor out.
543 EFI_ERROR_SECTION_DESCRIPTOR section_descriptor;
544 if (fread(&section_descriptor, sizeof(EFI_ERROR_SECTION_DESCRIPTOR), 1,
545 cper_section_file) != 1) {
546 printf("Failed to read section descriptor for CPER single section (fread() returned an unexpected value).\n");
Ed Tanous8121f7e2025-03-06 14:39:07 -0800547 json_object_put(ir);
Lawrence Tang617949e2022-08-08 14:21:42 +0100548 return NULL;
549 }
550
551 //Convert the section descriptor to IR.
552 json_object *section_descriptor_ir =
553 cper_section_descriptor_to_ir(&section_descriptor);
554 json_object_object_add(ir, "sectionDescriptor", section_descriptor_ir);
555
Ed Tanous73498f62025-03-05 18:03:36 -0800556 //Save our current position in the stream.
557 long position = ftell(cper_section_file);
558
559 //Read section as described by the section descriptor.
560 fseek(cper_section_file, base_pos + section_descriptor.SectionOffset,
561 SEEK_SET);
562 void *section = malloc(section_descriptor.SectionLength);
563 if (fread(section, section_descriptor.SectionLength, 1,
564 cper_section_file) != 1) {
565 printf("Section read failed: Could not read %u bytes from global offset %d.\n",
566 section_descriptor.SectionLength,
567 section_descriptor.SectionOffset);
Ed Tanous8121f7e2025-03-06 14:39:07 -0800568 json_object_put(ir);
Ed Tanous73498f62025-03-05 18:03:36 -0800569 free(section);
570 return NULL;
571 }
572
573 //Seek back to our original position.
574 fseek(cper_section_file, position, SEEK_SET);
575
Lawrence Tang617949e2022-08-08 14:21:42 +0100576 //Parse the single section.
Ed Tanous73498f62025-03-05 18:03:36 -0800577 json_object *section_ir = cper_buf_section_to_ir(
578 section, section_descriptor.SectionLength, &section_descriptor);
Lawrence Tang617949e2022-08-08 14:21:42 +0100579 json_object_object_add(ir, "section", section_ir);
Ed Tanous73498f62025-03-05 18:03:36 -0800580 free(section);
Lawrence Tang617949e2022-08-08 14:21:42 +0100581 return ir;
John Chungf8fc7052024-05-03 20:05:29 +0800582}
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700583
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700584char *cperbuf_single_section_to_str_ir(const unsigned char *cper_section,
585 size_t size)
586{
Ed Tanous73498f62025-03-05 18:03:36 -0800587 json_object *jobj = cper_buf_single_section_to_ir(cper_section, size);
588 char *str = jobj ? strdup(json_object_to_json_string(jobj)) : NULL;
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700589
Ed Tanous73498f62025-03-05 18:03:36 -0800590 json_object_put(jobj);
591 return str;
Karthik Rajagopalan5220c9b2024-08-08 00:24:44 -0700592}