blob: ab6e921c9ede435755990a5b1031a85310c8054e [file] [log] [blame]
Lawrence Tang214a1542022-07-06 14:11:28 +01001/**
2 * Describes functions for converting PCI/PCI-X bus CPER sections from binary and JSON format
3 * into an intermediate format.
Ed Tanousfedd4572024-07-12 13:56:00 -07004 *
Lawrence Tang214a1542022-07-06 14:11:28 +01005 * Author: Lawrence.Tang@arm.com
6 **/
7#include <stdio.h>
Lawrence Tang0a4b3f22022-07-21 10:40:10 +01008#include <string.h>
Lawrence Tang5202bbb2022-08-12 14:54:36 +01009#include <json.h>
Thu Nguyene42fb482024-10-15 14:43:11 +000010#include <libcper/Cper.h>
11#include <libcper/cper-utils.h>
12#include <libcper/sections/cper-section-pci-bus.h>
Lawrence Tang214a1542022-07-06 14:11:28 +010013
14//Converts a single PCI/PCI-X bus CPER section into JSON IR.
John Chungf8fc7052024-05-03 20:05:29 +080015json_object *cper_section_pci_bus_to_ir(void *section)
Lawrence Tang214a1542022-07-06 14:11:28 +010016{
Lawrence Tange407b4c2022-07-21 13:54:01 +010017 EFI_PCI_PCIX_BUS_ERROR_DATA *bus_error =
18 (EFI_PCI_PCIX_BUS_ERROR_DATA *)section;
19 json_object *section_ir = json_object_new_object();
Lawrence Tang214a1542022-07-06 14:11:28 +010020
Lawrence Tange407b4c2022-07-21 13:54:01 +010021 //Validation bits.
22 json_object *validation = bitfield_to_ir(
23 bus_error->ValidFields, 9, PCI_BUS_ERROR_VALID_BITFIELD_NAMES);
24 json_object_object_add(section_ir, "validationBits", validation);
Lawrence Tang214a1542022-07-06 14:11:28 +010025
Lawrence Tange407b4c2022-07-21 13:54:01 +010026 //Error status.
27 json_object *error_status =
28 cper_generic_error_status_to_ir(&bus_error->ErrorStatus);
29 json_object_object_add(section_ir, "errorStatus", error_status);
Lawrence Tang214a1542022-07-06 14:11:28 +010030
Lawrence Tange407b4c2022-07-21 13:54:01 +010031 //PCI bus error type.
32 json_object *error_type = integer_to_readable_pair(
33 bus_error->Type, 8, PCI_BUS_ERROR_TYPES_KEYS,
34 PCI_BUS_ERROR_TYPES_VALUES, "Unknown (Reserved)");
35 json_object_object_add(section_ir, "errorType", error_type);
Lawrence Tang214a1542022-07-06 14:11:28 +010036
Lawrence Tange407b4c2022-07-21 13:54:01 +010037 //Bus ID.
38 json_object *bus_id = json_object_new_object();
39 json_object_object_add(bus_id, "busNumber",
40 json_object_new_int(bus_error->BusId & 0xFF));
41 json_object_object_add(bus_id, "segmentNumber",
42 json_object_new_int(bus_error->BusId >> 8));
43 json_object_object_add(section_ir, "busID", bus_id);
Lawrence Tang214a1542022-07-06 14:11:28 +010044
Lawrence Tange407b4c2022-07-21 13:54:01 +010045 //Miscellaneous numeric fields.
46 UINT8 command_type = (bus_error->BusCommand >> 56) &
John Chungf8fc7052024-05-03 20:05:29 +080047 0x1; //Byte 7, bit 0.
Lawrence Tange407b4c2022-07-21 13:54:01 +010048 json_object_object_add(section_ir, "busAddress",
49 json_object_new_uint64(bus_error->BusAddress));
50 json_object_object_add(section_ir, "busData",
51 json_object_new_uint64(bus_error->BusData));
52 json_object_object_add(
53 section_ir, "busCommandType",
54 json_object_new_string(command_type == 0 ? "PCI" : "PCI-X"));
Aushim Nagarkatticc367012024-12-05 18:17:27 -080055
Lawrence Tange407b4c2022-07-21 13:54:01 +010056 json_object_object_add(section_ir, "busRequestorID",
57 json_object_new_uint64(bus_error->RequestorId));
Aushim Nagarkatticc367012024-12-05 18:17:27 -080058
59 char hexstring_buf[EFI_UINT64_HEX_STRING_LEN];
60 snprintf(hexstring_buf, EFI_UINT64_HEX_STRING_LEN, "0x%016llX",
61 bus_error->RequestorId);
62 json_object_object_add(section_ir, "busRequestorIDHex",
63 json_object_new_string(hexstring_buf));
64
Lawrence Tange407b4c2022-07-21 13:54:01 +010065 json_object_object_add(section_ir, "busCompleterID",
66 json_object_new_uint64(bus_error->ResponderId));
Aushim Nagarkatticc367012024-12-05 18:17:27 -080067 snprintf(hexstring_buf, EFI_UINT64_HEX_STRING_LEN, "0x%016llX",
68 bus_error->ResponderId);
69 json_object_object_add(section_ir, "busCompleterIDHex",
70 json_object_new_string(hexstring_buf));
71
Lawrence Tange407b4c2022-07-21 13:54:01 +010072 json_object_object_add(section_ir, "targetID",
73 json_object_new_uint64(bus_error->TargetId));
Lawrence Tang214a1542022-07-06 14:11:28 +010074
Lawrence Tange407b4c2022-07-21 13:54:01 +010075 return section_ir;
Lawrence Tang205dd1d2022-07-14 16:23:38 +010076}
77
78//Converts a single provided PCI/PCI-X bus CPER-JSON section into CPER binary, outputting to the
79//provided stream.
Lawrence Tange407b4c2022-07-21 13:54:01 +010080void ir_section_pci_bus_to_cper(json_object *section, FILE *out)
Lawrence Tang205dd1d2022-07-14 16:23:38 +010081{
Lawrence Tange407b4c2022-07-21 13:54:01 +010082 EFI_PCI_PCIX_BUS_ERROR_DATA *section_cper =
83 (EFI_PCI_PCIX_BUS_ERROR_DATA *)calloc(
84 1, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA));
Lawrence Tang205dd1d2022-07-14 16:23:38 +010085
Lawrence Tange407b4c2022-07-21 13:54:01 +010086 //Validation bits.
87 section_cper->ValidFields = ir_to_bitfield(
88 json_object_object_get(section, "validationBits"), 9,
89 PCI_BUS_ERROR_VALID_BITFIELD_NAMES);
Lawrence Tang205dd1d2022-07-14 16:23:38 +010090
Lawrence Tange407b4c2022-07-21 13:54:01 +010091 //Error status.
92 ir_generic_error_status_to_cper(json_object_object_get(section,
93 "errorStatus"),
94 &section_cper->ErrorStatus);
Lawrence Tang205dd1d2022-07-14 16:23:38 +010095
Lawrence Tange407b4c2022-07-21 13:54:01 +010096 //Bus ID.
97 json_object *bus_id = json_object_object_get(section, "busID");
98 UINT16 bus_number = (UINT8)json_object_get_int(
99 json_object_object_get(bus_id, "busNumber"));
100 UINT16 segment_number = (UINT8)json_object_get_int(
101 json_object_object_get(bus_id, "segmentNumber"));
102 section_cper->BusId = bus_number + (segment_number << 8);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100103
Lawrence Tange407b4c2022-07-21 13:54:01 +0100104 //Remaining fields.
105 UINT64 pcix_command = (UINT64)0x1 << 56;
106 const char *bus_command = json_object_get_string(
107 json_object_object_get(section, "busCommandType"));
108 section_cper->Type = (UINT16)readable_pair_to_integer(
109 json_object_object_get(section, "errorType"));
110 section_cper->BusAddress = json_object_get_uint64(
111 json_object_object_get(section, "busAddress"));
112 section_cper->BusData = json_object_get_uint64(
113 json_object_object_get(section, "busData"));
114 section_cper->BusCommand =
115 strcmp(bus_command, "PCI") == 0 ? 0 : pcix_command;
116 section_cper->RequestorId = json_object_get_uint64(
117 json_object_object_get(section, "busRequestorID"));
118 section_cper->ResponderId = json_object_get_uint64(
119 json_object_object_get(section, "busCompleterID"));
120 section_cper->TargetId = json_object_get_uint64(
121 json_object_object_get(section, "targetID"));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100122
Lawrence Tange407b4c2022-07-21 13:54:01 +0100123 //Write to stream, free resources.
124 fwrite(section_cper, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA), 1, out);
125 fflush(out);
126 free(section_cper);
John Chungf8fc7052024-05-03 20:05:29 +0800127}