blob: c777974ba009ac8c64b6fc708ee0bef799746a50 [file] [log] [blame]
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01001/**
Ed Tanousfedd4572024-07-12 13:56:00 -07002 * Describes utility functions for parsing CPER into JSON IR.
3 *
Lawrence Tang1b0b00e2022-07-05 10:33:10 +01004 * Author: Lawrence.Tang@arm.com
5 **/
6
7#include <stdio.h>
Lawrence Tang5202bbb2022-08-12 14:54:36 +01008#include <json.h>
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -08009#include <string.h>
Thu Nguyene42fb482024-10-15 14:43:11 +000010#include <libcper/Cper.h>
11#include <libcper/cper-utils.h>
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010012
13//The available severity types for CPER.
Lawrence Tange407b4c2022-07-21 13:54:01 +010014const char *CPER_SEVERITY_TYPES[4] = { "Recoverable", "Fatal", "Corrected",
15 "Informational" };
Lawrence Tang1b0b00e2022-07-05 10:33:10 +010016
Lawrence Tanga0865e32022-07-06 11:59:52 +010017//Converts the given generic CPER error status to JSON IR.
Lawrence Tange407b4c2022-07-21 13:54:01 +010018json_object *
19cper_generic_error_status_to_ir(EFI_GENERIC_ERROR_STATUS *error_status)
Lawrence Tanga0865e32022-07-06 11:59:52 +010020{
Lawrence Tange407b4c2022-07-21 13:54:01 +010021 json_object *error_status_ir = json_object_new_object();
Lawrence Tanga0865e32022-07-06 11:59:52 +010022
Lawrence Tange407b4c2022-07-21 13:54:01 +010023 //Error type.
24 json_object_object_add(error_status_ir, "errorType",
25 integer_to_readable_pair_with_desc(
26 error_status->Type, 18,
27 CPER_GENERIC_ERROR_TYPES_KEYS,
28 CPER_GENERIC_ERROR_TYPES_VALUES,
29 CPER_GENERIC_ERROR_TYPES_DESCRIPTIONS,
30 "Unknown (Reserved)"));
Lawrence Tanga0865e32022-07-06 11:59:52 +010031
Lawrence Tange407b4c2022-07-21 13:54:01 +010032 //Boolean bit fields.
33 json_object_object_add(
34 error_status_ir, "addressSignal",
35 json_object_new_boolean(error_status->AddressSignal));
36 json_object_object_add(
37 error_status_ir, "controlSignal",
38 json_object_new_boolean(error_status->ControlSignal));
39 json_object_object_add(
40 error_status_ir, "dataSignal",
41 json_object_new_boolean(error_status->DataSignal));
42 json_object_object_add(
43 error_status_ir, "detectedByResponder",
44 json_object_new_boolean(error_status->DetectedByResponder));
45 json_object_object_add(
46 error_status_ir, "detectedByRequester",
47 json_object_new_boolean(error_status->DetectedByRequester));
48 json_object_object_add(
49 error_status_ir, "firstError",
50 json_object_new_boolean(error_status->FirstError));
51 json_object_object_add(
52 error_status_ir, "overflowDroppedLogs",
53 json_object_new_boolean(error_status->OverflowNotLogged));
54
55 return error_status_ir;
Lawrence Tanga0865e32022-07-06 11:59:52 +010056}
57
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010058//Converts the given CPER-JSON generic error status into a CPER structure.
Lawrence Tange407b4c2022-07-21 13:54:01 +010059void ir_generic_error_status_to_cper(
60 json_object *error_status, EFI_GENERIC_ERROR_STATUS *error_status_cper)
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010061{
Lawrence Tange407b4c2022-07-21 13:54:01 +010062 error_status_cper->Type = readable_pair_to_integer(
63 json_object_object_get(error_status, "errorType"));
64 error_status_cper->AddressSignal = json_object_get_boolean(
65 json_object_object_get(error_status, "addressSignal"));
66 error_status_cper->ControlSignal = json_object_get_boolean(
67 json_object_object_get(error_status, "controlSignal"));
68 error_status_cper->DataSignal = json_object_get_boolean(
69 json_object_object_get(error_status, "dataSignal"));
70 error_status_cper->DetectedByResponder = json_object_get_boolean(
71 json_object_object_get(error_status, "detectedByResponder"));
72 error_status_cper->DetectedByRequester = json_object_get_boolean(
73 json_object_object_get(error_status, "detectedByRequester"));
74 error_status_cper->FirstError = json_object_get_boolean(
75 json_object_object_get(error_status, "firstError"));
76 error_status_cper->OverflowNotLogged = json_object_get_boolean(
77 json_object_object_get(error_status, "overflowDroppedLogs"));
Lawrence Tang3b7f45b2022-07-14 14:14:30 +010078}
79
Lawrence Tang7f21db62022-07-06 11:09:39 +010080//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 +010081json_object *uniform_struct64_to_ir(UINT64 *start, int len, const char *names[])
Lawrence Tang7f21db62022-07-06 11:09:39 +010082{
Lawrence Tange407b4c2022-07-21 13:54:01 +010083 json_object *result = json_object_new_object();
Lawrence Tang7f21db62022-07-06 11:09:39 +010084
Lawrence Tange407b4c2022-07-21 13:54:01 +010085 UINT64 *cur = start;
86 for (int i = 0; i < len; i++) {
87 json_object_object_add(result, names[i],
88 json_object_new_uint64(*cur));
89 cur++;
90 }
Lawrence Tang7f21db62022-07-06 11:09:39 +010091
Lawrence Tange407b4c2022-07-21 13:54:01 +010092 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +010093}
94
95//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 +010096json_object *uniform_struct_to_ir(UINT32 *start, int len, const char *names[])
Lawrence Tang7f21db62022-07-06 11:09:39 +010097{
Lawrence Tange407b4c2022-07-21 13:54:01 +010098 json_object *result = json_object_new_object();
Lawrence Tang7f21db62022-07-06 11:09:39 +010099
Lawrence Tange407b4c2022-07-21 13:54:01 +0100100 UINT32 *cur = start;
101 for (int i = 0; i < len; i++) {
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800102 UINT32 value;
103 memcpy(&value, cur, sizeof(UINT32));
Lawrence Tange407b4c2022-07-21 13:54:01 +0100104 json_object_object_add(result, names[i],
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800105 json_object_new_uint64(value));
Lawrence Tange407b4c2022-07-21 13:54:01 +0100106 cur++;
107 }
Lawrence Tang7f21db62022-07-06 11:09:39 +0100108
Lawrence Tange407b4c2022-07-21 13:54:01 +0100109 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +0100110}
111
Lawrence Tang71570a22022-07-14 11:45:28 +0100112//Converts a single object containing UINT32s into a uniform struct.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100113void ir_to_uniform_struct64(json_object *ir, UINT64 *start, int len,
114 const char *names[])
Lawrence Tang71570a22022-07-14 11:45:28 +0100115{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100116 UINT64 *cur = start;
117 for (int i = 0; i < len; i++) {
118 *cur = json_object_get_uint64(
119 json_object_object_get(ir, names[i]));
120 cur++;
121 }
Lawrence Tang71570a22022-07-14 11:45:28 +0100122}
123
124//Converts a single object containing UINT32s into a uniform struct.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100125void ir_to_uniform_struct(json_object *ir, UINT32 *start, int len,
126 const char *names[])
Lawrence Tang71570a22022-07-14 11:45:28 +0100127{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100128 UINT32 *cur = start;
129 for (int i = 0; i < len; i++) {
130 *cur = (UINT32)json_object_get_uint64(
131 json_object_object_get(ir, names[i]));
132 cur++;
133 }
Lawrence Tang71570a22022-07-14 11:45:28 +0100134}
135
Lawrence Tang3c43f742022-07-05 11:37:17 +0100136//Converts a single integer value to an object containing a value, and a readable name if possible.
Ed Tanousb35d9572024-06-18 13:17:22 -0700137json_object *integer_to_readable_pair(UINT64 value, int len, const int keys[],
Lawrence Tange407b4c2022-07-21 13:54:01 +0100138 const char *values[],
139 const char *default_value)
Lawrence Tang3c43f742022-07-05 11:37:17 +0100140{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100141 json_object *result = json_object_new_object();
142 json_object_object_add(result, "value", json_object_new_uint64(value));
Lawrence Tang3c43f742022-07-05 11:37:17 +0100143
Lawrence Tange407b4c2022-07-21 13:54:01 +0100144 //Search for human readable name, add.
145 const char *name = default_value;
146 for (int i = 0; i < len; i++) {
John Chungf8fc7052024-05-03 20:05:29 +0800147 if ((UINT64)keys[i] == value) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100148 name = values[i];
John Chungf8fc7052024-05-03 20:05:29 +0800149 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100150 }
Lawrence Tang3c43f742022-07-05 11:37:17 +0100151
Lawrence Tange407b4c2022-07-21 13:54:01 +0100152 json_object_object_add(result, "name", json_object_new_string(name));
153 return result;
Lawrence Tang3c43f742022-07-05 11:37:17 +0100154}
155
Lawrence Tang7f21db62022-07-06 11:09:39 +0100156//Converts a single integer value to an object containing a value, readable name and description if possible.
Ed Tanousb35d9572024-06-18 13:17:22 -0700157json_object *integer_to_readable_pair_with_desc(int value, int len,
158 const int keys[],
Lawrence Tange407b4c2022-07-21 13:54:01 +0100159 const char *values[],
160 const char *descriptions[],
161 const char *default_value)
Lawrence Tang7f21db62022-07-06 11:09:39 +0100162{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100163 json_object *result = json_object_new_object();
164 json_object_object_add(result, "value", json_object_new_int(value));
Lawrence Tang7f21db62022-07-06 11:09:39 +0100165
Lawrence Tange407b4c2022-07-21 13:54:01 +0100166 //Search for human readable name, add.
167 const char *name = default_value;
168 for (int i = 0; i < len; i++) {
169 if (keys[i] == value) {
170 name = values[i];
171 json_object_object_add(
172 result, "description",
173 json_object_new_string(descriptions[i]));
174 }
175 }
Lawrence Tang7f21db62022-07-06 11:09:39 +0100176
Lawrence Tange407b4c2022-07-21 13:54:01 +0100177 json_object_object_add(result, "name", json_object_new_string(name));
178 return result;
Lawrence Tang7f21db62022-07-06 11:09:39 +0100179}
180
Lawrence Tangb44314c2022-07-13 11:45:22 +0100181//Returns a single UINT64 value from the given readable pair object.
182//Assumes the integer value is held in the "value" field.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100183UINT64 readable_pair_to_integer(json_object *pair)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100184{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100185 return json_object_get_uint64(json_object_object_get(pair, "value"));
Lawrence Tangb44314c2022-07-13 11:45:22 +0100186}
187
Lawrence Tang794312c2022-07-05 14:46:10 +0100188//Converts the given 64 bit bitfield to IR, assuming bit 0 starts on the left.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100189json_object *bitfield_to_ir(UINT64 bitfield, int num_fields,
190 const char *names[])
Lawrence Tang794312c2022-07-05 14:46:10 +0100191{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100192 json_object *result = json_object_new_object();
193 for (int i = 0; i < num_fields; i++) {
194 json_object_object_add(result, names[i],
195 json_object_new_boolean((bitfield >> i) &
John Chungf8fc7052024-05-03 20:05:29 +0800196 0x1));
Lawrence Tange407b4c2022-07-21 13:54:01 +0100197 }
Lawrence Tang794312c2022-07-05 14:46:10 +0100198
Lawrence Tange407b4c2022-07-21 13:54:01 +0100199 return result;
Lawrence Tang794312c2022-07-05 14:46:10 +0100200}
201
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800202//Filters properties based on Validation Bits.
203// Refer to CPER spec for vbit_idx to be passed here.
204void add_to_valid_bitfield(ValidationTypes *val, int vbit_idx)
205{
206 switch (val->size) {
207 case UINT_8T:
208 val->value.ui8 |= (0x01 << vbit_idx);
209 break;
210 case UINT_16T:
211 val->value.ui16 |= (0x01 << vbit_idx);
212 break;
213 case UINT_32T:
214 val->value.ui32 |= (0x01 << vbit_idx);
215 break;
216 case UINT_64T:
217 val->value.ui64 |= (0x01 << vbit_idx);
218 break;
219 default:
220 printf("IR to CPER: Unknown validation bits size passed, Enum IntType=%d",
221 val->size);
222 }
223}
224
Lawrence Tangb44314c2022-07-13 11:45:22 +0100225//Converts the given IR bitfield into a standard UINT64 bitfield, with fields beginning from bit 0.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100226UINT64 ir_to_bitfield(json_object *ir, int num_fields, const char *names[])
Lawrence Tangb44314c2022-07-13 11:45:22 +0100227{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100228 UINT64 result = 0x0;
229 for (int i = 0; i < num_fields; i++) {
230 if (json_object_get_boolean(
John Chungf8fc7052024-05-03 20:05:29 +0800231 json_object_object_get(ir, names[i]))) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100232 result |= (0x1 << i);
John Chungf8fc7052024-05-03 20:05:29 +0800233 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100234 }
Lawrence Tangb44314c2022-07-13 11:45:22 +0100235
Lawrence Tange407b4c2022-07-21 13:54:01 +0100236 return result;
Lawrence Tangb44314c2022-07-13 11:45:22 +0100237}
238
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800239// Filters properties based on Validation Bits.
240// Refer to CPER spec for vbit_idx to be passed here.
241// Overload function for 16, 32, 64b
242bool isvalid_prop_to_ir(ValidationTypes *val, int vbit_idx)
243{
244 UINT64 vbit_mask = 0x01 << vbit_idx;
245 switch (val->size) {
246 case UINT_16T:
247 return (vbit_mask & val->value.ui16);
248
249 case UINT_32T:
250 return (vbit_mask & val->value.ui32);
251
252 case UINT_64T:
253 return (vbit_mask & val->value.ui64);
254
255 default:
256 printf("CPER to IR:Unknown validation bits size passed. Enum IntType: %d",
257 val->size);
258 }
259 return 0;
260}
261
262void print_val(ValidationTypes *val)
263{
264 switch (val->size) {
265 case UINT_8T:
266 printf("Validation bits: %x\n", val->value.ui8);
267 break;
268 case UINT_16T:
269 printf("Validation bits: %x\n", val->value.ui16);
270 break;
271
272 case UINT_32T:
273 printf("Validation bits: %x\n", val->value.ui32);
274 break;
275
276 case UINT_64T:
277 printf("Validation bits: %llx\n", val->value.ui64);
278 break;
279
280 default:
281 printf("CPER to IR:Unknown validation bits size passed. Enum IntType: %d",
282 val->size);
283 }
284}
285
Lawrence Tange18aaee2022-07-07 09:01:30 +0100286//Converts the given UINT64 array into a JSON IR array, given the length.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100287json_object *uint64_array_to_ir_array(UINT64 *array, int len)
Lawrence Tange18aaee2022-07-07 09:01:30 +0100288{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100289 json_object *array_ir = json_object_new_array();
John Chungf8fc7052024-05-03 20:05:29 +0800290 for (int i = 0; i < len; i++) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100291 json_object_array_add(array_ir,
292 json_object_new_uint64(array[i]));
John Chungf8fc7052024-05-03 20:05:29 +0800293 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100294 return array_ir;
Lawrence Tange18aaee2022-07-07 09:01:30 +0100295}
Lawrence Tang794312c2022-07-05 14:46:10 +0100296
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100297//Converts a single UINT16 revision number into JSON IR representation.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100298json_object *revision_to_ir(UINT16 revision)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100299{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100300 json_object *revision_info = json_object_new_object();
301 json_object_object_add(revision_info, "major",
302 json_object_new_int(revision >> 8));
303 json_object_object_add(revision_info, "minor",
304 json_object_new_int(revision & 0xFF));
305 return revision_info;
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100306}
307
308//Returns the appropriate string for the given integer severity.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100309const char *severity_to_string(UINT32 severity)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100310{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100311 return severity < 4 ? CPER_SEVERITY_TYPES[severity] : "Unknown";
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100312}
313
Lawrence Tangb44314c2022-07-13 11:45:22 +0100314//Converts a single EFI timestamp to string, at the given output.
315//Output must be at least TIMESTAMP_LENGTH bytes long.
Ed Tanous13f099f2024-11-20 11:10:30 -0800316void timestamp_to_string(char *out, int out_len,
317 EFI_ERROR_TIME_STAMP *timestamp)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100318{
Ed Tanous13f099f2024-11-20 11:10:30 -0800319 int written = snprintf(
320 out, out_len,
321 "%02hhu%02hhu-%02hhu-%02hhuT%02hhu:%02hhu:%02hhu+00:00",
Lawrence Tange407b4c2022-07-21 13:54:01 +0100322 bcd_to_int(timestamp->Century) %
John Chungf8fc7052024-05-03 20:05:29 +0800323 100, //Cannot go to three digits.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100324 bcd_to_int(timestamp->Year) % 100, //Cannot go to three digits.
325 bcd_to_int(timestamp->Month), bcd_to_int(timestamp->Day),
326 bcd_to_int(timestamp->Hours), bcd_to_int(timestamp->Minutes),
327 bcd_to_int(timestamp->Seconds));
Ed Tanous13f099f2024-11-20 11:10:30 -0800328
329 if (written < 0 || written >= out_len) {
330 printf("Timestamp buffer of insufficient size\n");
331 }
Lawrence Tangb44314c2022-07-13 11:45:22 +0100332}
333
334//Converts a single timestamp string to an EFI timestamp.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100335void string_to_timestamp(EFI_ERROR_TIME_STAMP *out, const char *timestamp)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100336{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100337 //Ignore invalid timestamps.
John Chungf8fc7052024-05-03 20:05:29 +0800338 if (timestamp == NULL) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100339 return;
John Chungf8fc7052024-05-03 20:05:29 +0800340 }
Lawrence Tang0cb33792022-07-13 13:51:39 +0100341
Aushim Nagarkattide7dd062024-11-04 15:23:39 -0800342 sscanf(timestamp, "%2hhu%2hhu-%hhu-%hhuT%hhu:%hhu:%hhu+00:00",
Lawrence Tange407b4c2022-07-21 13:54:01 +0100343 &out->Century, &out->Year, &out->Month, &out->Day, &out->Hours,
344 &out->Minutes, &out->Seconds);
Lawrence Tangaacf0e22022-07-20 13:28:52 +0100345
Lawrence Tange407b4c2022-07-21 13:54:01 +0100346 //Convert back to BCD.
347 out->Century = int_to_bcd(out->Century);
348 out->Year = int_to_bcd(out->Year);
349 out->Month = int_to_bcd(out->Month);
350 out->Day = int_to_bcd(out->Day);
351 out->Hours = int_to_bcd(out->Hours);
352 out->Minutes = int_to_bcd(out->Minutes);
353 out->Seconds = int_to_bcd(out->Seconds);
Lawrence Tangb44314c2022-07-13 11:45:22 +0100354}
355
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100356//Helper function to convert an EDK EFI GUID into a string for intermediate use.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100357void guid_to_string(char *out, EFI_GUID *guid)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100358{
Karthik Rajagopalan7623a6e2024-09-20 19:24:35 -0700359 sprintf(out, "%08x-%04x-%04x-%02x%02x-%02x%02x%02x%02x%02x%02x",
Lawrence Tange407b4c2022-07-21 13:54:01 +0100360 guid->Data1, guid->Data2, guid->Data3, guid->Data4[0],
361 guid->Data4[1], guid->Data4[2], guid->Data4[3], guid->Data4[4],
362 guid->Data4[5], guid->Data4[6], guid->Data4[7]);
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100363}
364
Lawrence Tangb44314c2022-07-13 11:45:22 +0100365//Helper function to convert a string into an EDK EFI GUID.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100366void string_to_guid(EFI_GUID *out, const char *guid)
Lawrence Tangb44314c2022-07-13 11:45:22 +0100367{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100368 //Ignore invalid GUIDs.
John Chungf8fc7052024-05-03 20:05:29 +0800369 if (guid == NULL) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100370 return;
John Chungf8fc7052024-05-03 20:05:29 +0800371 }
Lawrence Tang0cb33792022-07-13 13:51:39 +0100372
Lawrence Tange407b4c2022-07-21 13:54:01 +0100373 sscanf(guid,
Karthik Rajagopalan7623a6e2024-09-20 19:24:35 -0700374 "%08x-%04hx-%04hx-%02hhx%02hhx-%02hhx%02hhx%02hhx%02hhx%02hhx%02hhx",
Lawrence Tange407b4c2022-07-21 13:54:01 +0100375 &out->Data1, &out->Data2, &out->Data3, out->Data4,
376 out->Data4 + 1, out->Data4 + 2, out->Data4 + 3, out->Data4 + 4,
377 out->Data4 + 5, out->Data4 + 6, out->Data4 + 7);
Lawrence Tangb44314c2022-07-13 11:45:22 +0100378}
379
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100380//Returns one if two EFI GUIDs are equal, zero otherwise.
Lawrence Tange407b4c2022-07-21 13:54:01 +0100381int guid_equal(EFI_GUID *a, EFI_GUID *b)
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100382{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100383 //Check top base 3 components.
384 if (a->Data1 != b->Data1 || a->Data2 != b->Data2 ||
385 a->Data3 != b->Data3) {
386 return 0;
387 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100388
Lawrence Tange407b4c2022-07-21 13:54:01 +0100389 //Check Data4 array for equality.
390 for (int i = 0; i < 8; i++) {
John Chungf8fc7052024-05-03 20:05:29 +0800391 if (a->Data4[i] != b->Data4[i]) {
Lawrence Tange407b4c2022-07-21 13:54:01 +0100392 return 0;
John Chungf8fc7052024-05-03 20:05:29 +0800393 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100394 }
Lawrence Tang1b0b00e2022-07-05 10:33:10 +0100395
Lawrence Tange407b4c2022-07-21 13:54:01 +0100396 return 1;
John Chungf8fc7052024-05-03 20:05:29 +0800397}