blob: 519349f643108489f65d3b7303acdec374565e72 [file] [log] [blame]
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01001/**
2 * Describes utility functions for parsing CPER into JSON IR.
3 *
4 * Author: Lawrence.Tang@arm.com
5 **/
6
7#include <stdio.h>
8#include "json.h"
9#include "edk/Cper.h"
10#include "cper-utils.h"
11
12//The available severity types for CPER.
Lawrence Tange407b4c2022-07-21 13:54:01 +010013const char *CPER_SEVERITY_TYPES[4] = { "Recoverable", "Fatal", "Corrected",
14 "Informational" };
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010015
Lawrence Tanga0865e32022-07-06 11:59:52 +010016//Converts the given generic CPER error status to JSON IR.
Lawrence Tange407b4c2022-07-21 13:54:01 +010017json_object *
18cper_generic_error_status_to_ir(EFI_GENERIC_ERROR_STATUS *error_status)
Lawrence Tanga0865e32022-07-06 11:59:52 +010019{
Lawrence Tange407b4c2022-07-21 13:54:01 +010020 json_object *error_status_ir = json_object_new_object();
Lawrence Tanga0865e32022-07-06 11:59:52 +010021
Lawrence Tange407b4c2022-07-21 13:54:01 +010022 //Error type.
23 json_object_object_add(error_status_ir, "errorType",
24 integer_to_readable_pair_with_desc(
25 error_status->Type, 18,
26 CPER_GENERIC_ERROR_TYPES_KEYS,
27 CPER_GENERIC_ERROR_TYPES_VALUES,
28 CPER_GENERIC_ERROR_TYPES_DESCRIPTIONS,
29 "Unknown (Reserved)"));
Lawrence Tanga0865e32022-07-06 11:59:52 +010030
Lawrence Tange407b4c2022-07-21 13:54:01 +010031 //Boolean bit fields.
32 json_object_object_add(
33 error_status_ir, "addressSignal",
34 json_object_new_boolean(error_status->AddressSignal));
35 json_object_object_add(
36 error_status_ir, "controlSignal",
37 json_object_new_boolean(error_status->ControlSignal));
38 json_object_object_add(
39 error_status_ir, "dataSignal",
40 json_object_new_boolean(error_status->DataSignal));
41 json_object_object_add(
42 error_status_ir, "detectedByResponder",
43 json_object_new_boolean(error_status->DetectedByResponder));
44 json_object_object_add(
45 error_status_ir, "detectedByRequester",
46 json_object_new_boolean(error_status->DetectedByRequester));
47 json_object_object_add(
48 error_status_ir, "firstError",
49 json_object_new_boolean(error_status->FirstError));
50 json_object_object_add(
51 error_status_ir, "overflowDroppedLogs",
52 json_object_new_boolean(error_status->OverflowNotLogged));
53
54 return error_status_ir;
Lawrence Tanga0865e32022-07-06 11:59:52 +010055}
56
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010057//Converts the given CPER-JSON generic error status into a CPER structure.
Lawrence Tange407b4c2022-07-21 13:54:01 +010058void ir_generic_error_status_to_cper(
59 json_object *error_status, EFI_GENERIC_ERROR_STATUS *error_status_cper)
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010060{
Lawrence Tange407b4c2022-07-21 13:54:01 +010061 error_status_cper->Type = readable_pair_to_integer(
62 json_object_object_get(error_status, "errorType"));
63 error_status_cper->AddressSignal = json_object_get_boolean(
64 json_object_object_get(error_status, "addressSignal"));
65 error_status_cper->ControlSignal = json_object_get_boolean(
66 json_object_object_get(error_status, "controlSignal"));
67 error_status_cper->DataSignal = json_object_get_boolean(
68 json_object_object_get(error_status, "dataSignal"));
69 error_status_cper->DetectedByResponder = json_object_get_boolean(
70 json_object_object_get(error_status, "detectedByResponder"));
71 error_status_cper->DetectedByRequester = json_object_get_boolean(
72 json_object_object_get(error_status, "detectedByRequester"));
73 error_status_cper->FirstError = json_object_get_boolean(
74 json_object_object_get(error_status, "firstError"));
75 error_status_cper->OverflowNotLogged = json_object_get_boolean(
76 json_object_object_get(error_status, "overflowDroppedLogs"));
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010077}
78
Lawrence Tang7f21db62022-07-06 11:09:39 +010079//Converts a single uniform struct of UINT64s into intermediate JSON IR format, given names for each field in byte order.
Lawrence Tange407b4c2022-07-21 13:54:01 +010080json_object *uniform_struct64_to_ir(UINT64 *start, int len, const char *names[])
Lawrence Tang7f21db62022-07-06 11:09:39 +010081{
Lawrence Tange407b4c2022-07-21 13:54:01 +010082 json_object *result = json_object_new_object();
Lawrence Tang7f21db62022-07-06 11:09:39 +010083
Lawrence Tange407b4c2022-07-21 13:54:01 +010084 UINT64 *cur = start;
85 for (int i = 0; i < len; i++) {
86 json_object_object_add(result, names[i],
87 json_object_new_uint64(*cur));
88 cur++;
89 }
Lawrence Tang7f21db62022-07-06 11:09:39 +010090
Lawrence Tange407b4c2022-07-21 13:54:01 +010091 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +010092}
93
94//Converts a single uniform struct of UINT32s into intermediate JSON IR format, given names for each field in byte order.
Lawrence Tange407b4c2022-07-21 13:54:01 +010095json_object *uniform_struct_to_ir(UINT32 *start, int len, const char *names[])
Lawrence Tang7f21db62022-07-06 11:09:39 +010096{
Lawrence Tange407b4c2022-07-21 13:54:01 +010097 json_object *result = json_object_new_object();
Lawrence Tang7f21db62022-07-06 11:09:39 +010098
Lawrence Tange407b4c2022-07-21 13:54:01 +010099 UINT32 *cur = start;
100 for (int i = 0; i < len; i++) {
101 json_object_object_add(result, names[i],
102 json_object_new_uint64(*cur));
103 cur++;
104 }
Lawrence Tang7f21db62022-07-06 11:09:39 +0100105
Lawrence Tange407b4c2022-07-21 13:54:01 +0100106 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +0100107}
108
Lawrence Tang71570a22022-07-14 11:45:28 +0100109//Converts a single object containing UINT32s into a uniform struct.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100110void ir_to_uniform_struct64(json_object *ir, UINT64 *start, int len,
111 const char *names[])
Lawrence Tang71570a22022-07-14 11:45:28 +0100112{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100113 UINT64 *cur = start;
114 for (int i = 0; i < len; i++) {
115 *cur = json_object_get_uint64(
116 json_object_object_get(ir, names[i]));
117 cur++;
118 }
Lawrence Tang71570a22022-07-14 11:45:28 +0100119}
120
121//Converts a single object containing UINT32s into a uniform struct.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100122void ir_to_uniform_struct(json_object *ir, UINT32 *start, int len,
123 const char *names[])
Lawrence Tang71570a22022-07-14 11:45:28 +0100124{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100125 UINT32 *cur = start;
126 for (int i = 0; i < len; i++) {
127 *cur = (UINT32)json_object_get_uint64(
128 json_object_object_get(ir, names[i]));
129 cur++;
130 }
Lawrence Tang71570a22022-07-14 11:45:28 +0100131}
132
Lawrence Tang3c43f742022-07-05 11:37:17 +0100133//Converts a single integer value to an object containing a value, and a readable name if possible.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100134json_object *integer_to_readable_pair(UINT64 value, int len, int keys[],
135 const char *values[],
136 const char *default_value)
Lawrence Tang3c43f742022-07-05 11:37:17 +0100137{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100138 json_object *result = json_object_new_object();
139 json_object_object_add(result, "value", json_object_new_uint64(value));
Lawrence Tang3c43f742022-07-05 11:37:17 +0100140
Lawrence Tange407b4c2022-07-21 13:54:01 +0100141 //Search for human readable name, add.
142 const char *name = default_value;
143 for (int i = 0; i < len; i++) {
144 if (keys[i] == value)
145 name = values[i];
146 }
Lawrence Tang3c43f742022-07-05 11:37:17 +0100147
Lawrence Tange407b4c2022-07-21 13:54:01 +0100148 json_object_object_add(result, "name", json_object_new_string(name));
149 return result;
Lawrence Tang3c43f742022-07-05 11:37:17 +0100150}
151
Lawrence Tang7f21db62022-07-06 11:09:39 +0100152//Converts a single integer value to an object containing a value, readable name and description if possible.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100153json_object *integer_to_readable_pair_with_desc(int value, int len, int keys[],
154 const char *values[],
155 const char *descriptions[],
156 const char *default_value)
Lawrence Tang7f21db62022-07-06 11:09:39 +0100157{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100158 json_object *result = json_object_new_object();
159 json_object_object_add(result, "value", json_object_new_int(value));
Lawrence Tang7f21db62022-07-06 11:09:39 +0100160
Lawrence Tange407b4c2022-07-21 13:54:01 +0100161 //Search for human readable name, add.
162 const char *name = default_value;
163 for (int i = 0; i < len; i++) {
164 if (keys[i] == value) {
165 name = values[i];
166 json_object_object_add(
167 result, "description",
168 json_object_new_string(descriptions[i]));
169 }
170 }
Lawrence Tang7f21db62022-07-06 11:09:39 +0100171
Lawrence Tange407b4c2022-07-21 13:54:01 +0100172 json_object_object_add(result, "name", json_object_new_string(name));
173 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +0100174}
175
Lawrence Tangb44314c2022-07-13 11:45:22 +0100176//Returns a single UINT64 value from the given readable pair object.
177//Assumes the integer value is held in the "value" field.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100178UINT64 readable_pair_to_integer(json_object *pair)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100179{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100180 return json_object_get_uint64(json_object_object_get(pair, "value"));
Lawrence Tangb44314c2022-07-13 11:45:22 +0100181}
182
Lawrence Tang794312c2022-07-05 14:46:10 +0100183//Converts the given 64 bit bitfield to IR, assuming bit 0 starts on the left.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100184json_object *bitfield_to_ir(UINT64 bitfield, int num_fields,
185 const char *names[])
Lawrence Tang794312c2022-07-05 14:46:10 +0100186{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100187 json_object *result = json_object_new_object();
188 for (int i = 0; i < num_fields; i++) {
189 json_object_object_add(result, names[i],
190 json_object_new_boolean((bitfield >> i) &
191 0b1));
192 }
Lawrence Tang794312c2022-07-05 14:46:10 +0100193
Lawrence Tange407b4c2022-07-21 13:54:01 +0100194 return result;
Lawrence Tang794312c2022-07-05 14:46:10 +0100195}
196
Lawrence Tangb44314c2022-07-13 11:45:22 +0100197//Converts the given IR bitfield into a standard UINT64 bitfield, with fields beginning from bit 0.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100198UINT64 ir_to_bitfield(json_object *ir, int num_fields, const char *names[])
Lawrence Tangb44314c2022-07-13 11:45:22 +0100199{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100200 UINT64 result = 0x0;
201 for (int i = 0; i < num_fields; i++) {
202 if (json_object_get_boolean(
203 json_object_object_get(ir, names[i])))
204 result |= (0x1 << i);
205 }
Lawrence Tangb44314c2022-07-13 11:45:22 +0100206
Lawrence Tange407b4c2022-07-21 13:54:01 +0100207 return result;
Lawrence Tangb44314c2022-07-13 11:45:22 +0100208}
209
Lawrence Tange18aaee2022-07-07 09:01:30 +0100210//Converts the given UINT64 array into a JSON IR array, given the length.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100211json_object *uint64_array_to_ir_array(UINT64 *array, int len)
Lawrence Tange18aaee2022-07-07 09:01:30 +0100212{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100213 json_object *array_ir = json_object_new_array();
214 for (int i = 0; i < len; i++)
215 json_object_array_add(array_ir,
216 json_object_new_uint64(array[i]));
217 return array_ir;
Lawrence Tange18aaee2022-07-07 09:01:30 +0100218}
Lawrence Tang794312c2022-07-05 14:46:10 +0100219
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100220//Converts a single UINT16 revision number into JSON IR representation.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100221json_object *revision_to_ir(UINT16 revision)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100222{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100223 json_object *revision_info = json_object_new_object();
224 json_object_object_add(revision_info, "major",
225 json_object_new_int(revision >> 8));
226 json_object_object_add(revision_info, "minor",
227 json_object_new_int(revision & 0xFF));
228 return revision_info;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100229}
230
231//Returns the appropriate string for the given integer severity.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100232const char *severity_to_string(UINT32 severity)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100233{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100234 return severity < 4 ? CPER_SEVERITY_TYPES[severity] : "Unknown";
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100235}
236
Lawrence Tangb44314c2022-07-13 11:45:22 +0100237//Converts a single EFI timestamp to string, at the given output.
238//Output must be at least TIMESTAMP_LENGTH bytes long.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100239void timestamp_to_string(char *out, EFI_ERROR_TIME_STAMP *timestamp)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100240{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100241 sprintf(out, "%02hhu%02hhu-%02hhu-%02hhuT%02hhu:%02hhu:%02hhu.000",
242 bcd_to_int(timestamp->Century) %
243 100, //Cannot go to three digits.
244 bcd_to_int(timestamp->Year) % 100, //Cannot go to three digits.
245 bcd_to_int(timestamp->Month), bcd_to_int(timestamp->Day),
246 bcd_to_int(timestamp->Hours), bcd_to_int(timestamp->Minutes),
247 bcd_to_int(timestamp->Seconds));
Lawrence Tangb44314c2022-07-13 11:45:22 +0100248}
249
250//Converts a single timestamp string to an EFI timestamp.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100251void string_to_timestamp(EFI_ERROR_TIME_STAMP *out, const char *timestamp)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100252{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100253 //Ignore invalid timestamps.
254 if (timestamp == NULL)
255 return;
Lawrence Tang0cb33792022-07-13 13:51:39 +0100256
Lawrence Tange407b4c2022-07-21 13:54:01 +0100257 sscanf(timestamp, "%2hhu%2hhu-%hhu-%hhuT%hhu:%hhu:%hhu.000",
258 &out->Century, &out->Year, &out->Month, &out->Day, &out->Hours,
259 &out->Minutes, &out->Seconds);
Lawrence Tangaacf0e22022-07-20 13:28:52 +0100260
Lawrence Tange407b4c2022-07-21 13:54:01 +0100261 //Convert back to BCD.
262 out->Century = int_to_bcd(out->Century);
263 out->Year = int_to_bcd(out->Year);
264 out->Month = int_to_bcd(out->Month);
265 out->Day = int_to_bcd(out->Day);
266 out->Hours = int_to_bcd(out->Hours);
267 out->Minutes = int_to_bcd(out->Minutes);
268 out->Seconds = int_to_bcd(out->Seconds);
Lawrence Tangb44314c2022-07-13 11:45:22 +0100269}
270
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100271//Helper function to convert an EDK EFI GUID into a string for intermediate use.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100272void guid_to_string(char *out, EFI_GUID *guid)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100273{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100274 sprintf(out, "%08x-%04x-%04x-%02x%02x%02x%02x%02x%02x%02x%02x",
275 guid->Data1, guid->Data2, guid->Data3, guid->Data4[0],
276 guid->Data4[1], guid->Data4[2], guid->Data4[3], guid->Data4[4],
277 guid->Data4[5], guid->Data4[6], guid->Data4[7]);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100278}
279
Lawrence Tangb44314c2022-07-13 11:45:22 +0100280//Helper function to convert a string into an EDK EFI GUID.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100281void string_to_guid(EFI_GUID *out, const char *guid)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100282{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100283 //Ignore invalid GUIDs.
284 if (guid == NULL)
285 return;
Lawrence Tang0cb33792022-07-13 13:51:39 +0100286
Lawrence Tange407b4c2022-07-21 13:54:01 +0100287 sscanf(guid,
288 "%08x-%04hx-%04hx-%02hhx%02hhx%02hhx%02hhx%02hhx%02hhx%02hhx%02hhx",
289 &out->Data1, &out->Data2, &out->Data3, out->Data4,
290 out->Data4 + 1, out->Data4 + 2, out->Data4 + 3, out->Data4 + 4,
291 out->Data4 + 5, out->Data4 + 6, out->Data4 + 7);
Lawrence Tangb44314c2022-07-13 11:45:22 +0100292}
293
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100294//Returns one if two EFI GUIDs are equal, zero otherwise.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100295int guid_equal(EFI_GUID *a, EFI_GUID *b)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100296{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100297 //Check top base 3 components.
298 if (a->Data1 != b->Data1 || a->Data2 != b->Data2 ||
299 a->Data3 != b->Data3) {
300 return 0;
301 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100302
Lawrence Tange407b4c2022-07-21 13:54:01 +0100303 //Check Data4 array for equality.
304 for (int i = 0; i < 8; i++) {
305 if (a->Data4[i] != b->Data4[i])
306 return 0;
307 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100308
Lawrence Tange407b4c2022-07-21 13:54:01 +0100309 return 1;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100310}