blob: 359e56e383d64ab6facfefcfa08f5c6c50584a4d [file] [log] [blame]
Lawrence Tanga416ec92022-07-06 14:34:40 +01001/**
2 * Describes functions for converting PCI/PCI-X device CPER sections from binary and JSON format
3 * into an intermediate format.
Ed Tanousfedd4572024-07-12 13:56:00 -07004 *
Lawrence Tanga416ec92022-07-06 14:34:40 +01005 * Author: Lawrence.Tang@arm.com
6 **/
7#include <stdio.h>
Lawrence Tang5202bbb2022-08-12 14:54:36 +01008#include <json.h>
Thu Nguyene42fb482024-10-15 14:43:11 +00009#include <libcper/Cper.h>
10#include <libcper/cper-utils.h>
11#include <libcper/sections/cper-section-pci-dev.h>
Ed Tanous50b966f2025-03-11 09:06:19 -070012#include <libcper/log.h>
Lawrence Tanga416ec92022-07-06 14:34:40 +010013
14//Converts a single PCI/PCI-X device CPER section into JSON IR.
Ed Tanous12dbd4f2025-03-08 19:05:01 -080015json_object *cper_section_pci_dev_to_ir(const UINT8 *section, UINT32 size)
Lawrence Tanga416ec92022-07-06 14:34:40 +010016{
Ed Tanous12dbd4f2025-03-08 19:05:01 -080017 if (size < sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA)) {
18 return NULL;
19 }
20
Lawrence Tange407b4c2022-07-21 13:54:01 +010021 EFI_PCI_PCIX_DEVICE_ERROR_DATA *dev_error =
22 (EFI_PCI_PCIX_DEVICE_ERROR_DATA *)section;
Ed Tanous12dbd4f2025-03-08 19:05:01 -080023
24 if (size < sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA) +
25 ((dev_error->MemoryNumber + dev_error->IoNumber) *
26 sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA_REGISTER))) {
27 return NULL;
28 }
29
Lawrence Tange407b4c2022-07-21 13:54:01 +010030 json_object *section_ir = json_object_new_object();
Lawrence Tanga416ec92022-07-06 14:34:40 +010031
Lawrence Tange407b4c2022-07-21 13:54:01 +010032 //Validation bits.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -080033 ValidationTypes ui64Type = { UINT_64T,
34 .value.ui64 = dev_error->ValidFields };
Lawrence Tanga416ec92022-07-06 14:34:40 +010035
Lawrence Tange407b4c2022-07-21 13:54:01 +010036 //Error status.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -080037 if (isvalid_prop_to_ir(&ui64Type, 0)) {
38 json_object *error_status = cper_generic_error_status_to_ir(
39 &dev_error->ErrorStatus);
40 json_object_object_add(section_ir, "errorStatus", error_status);
41 }
Lawrence Tanga416ec92022-07-06 14:34:40 +010042
Lawrence Tange407b4c2022-07-21 13:54:01 +010043 //ID information.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -080044 if (isvalid_prop_to_ir(&ui64Type, 1)) {
45 json_object *id_info = json_object_new_object();
46 json_object_object_add(
47 id_info, "vendorID",
48 json_object_new_uint64(dev_error->IdInfo.VendorId));
49 json_object_object_add(
50 id_info, "deviceID",
51 json_object_new_uint64(dev_error->IdInfo.DeviceId));
52 json_object_object_add(
53 id_info, "classCode",
54 json_object_new_uint64(dev_error->IdInfo.ClassCode));
55 json_object_object_add(
56 id_info, "functionNumber",
57 json_object_new_uint64(
58 dev_error->IdInfo.FunctionNumber));
59 json_object_object_add(
60 id_info, "deviceNumber",
61 json_object_new_uint64(dev_error->IdInfo.DeviceNumber));
62 json_object_object_add(
63 id_info, "busNumber",
64 json_object_new_uint64(dev_error->IdInfo.BusNumber));
65 json_object_object_add(
66 id_info, "segmentNumber",
67 json_object_new_uint64(
68 dev_error->IdInfo.SegmentNumber));
69 json_object_object_add(section_ir, "idInfo", id_info);
70 }
Lawrence Tanga416ec92022-07-06 14:34:40 +010071
Lawrence Tange407b4c2022-07-21 13:54:01 +010072 //Number of following register data pairs.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -080073 if (isvalid_prop_to_ir(&ui64Type, 2)) {
74 json_object_object_add(
75 section_ir, "memoryNumber",
76 json_object_new_uint64(dev_error->MemoryNumber));
Lawrence Tange407b4c2022-07-21 13:54:01 +010077 }
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -080078 if (isvalid_prop_to_ir(&ui64Type, 3)) {
79 json_object_object_add(
80 section_ir, "ioNumber",
81 json_object_new_uint64(dev_error->IoNumber));
82 }
83
84 if (isvalid_prop_to_ir(&ui64Type, 4)) {
85 int num_data_pairs =
86 dev_error->MemoryNumber + dev_error->IoNumber;
87
88 //Register pairs, described by the numeric fields.
89 //The actual "pairs" of address and data aren't necessarily 8 bytes long, so can't assume the contents.
90 //Hence the naming "firstHalf" and "secondHalf" rather than "address" and "data".
91 json_object *register_data_pair_array = json_object_new_array();
92 UINT64 *cur_pos = (UINT64 *)(dev_error + 1);
93 for (int i = 0; i < num_data_pairs; i++) {
94 //Save current pair to array.
95 json_object *register_data_pair =
96 json_object_new_object();
97 json_object_object_add(
98 register_data_pair, "firstHalf",
99 json_object_new_uint64(*cur_pos));
100 json_object_object_add(
101 register_data_pair, "secondHalf",
102 json_object_new_uint64(*(cur_pos + 1)));
103 json_object_array_add(register_data_pair_array,
104 register_data_pair);
105
106 //Move to next pair.
107 cur_pos += 2;
108 }
109 json_object_object_add(section_ir, "registerDataPairs",
110 register_data_pair_array);
111 }
Lawrence Tanga416ec92022-07-06 14:34:40 +0100112
Lawrence Tange407b4c2022-07-21 13:54:01 +0100113 return section_ir;
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100114}
115
Lawrence Tange407b4c2022-07-21 13:54:01 +0100116void ir_section_pci_dev_to_cper(json_object *section, FILE *out)
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100117{
Lawrence Tange407b4c2022-07-21 13:54:01 +0100118 EFI_PCI_PCIX_DEVICE_ERROR_DATA *section_cper =
119 (EFI_PCI_PCIX_DEVICE_ERROR_DATA *)calloc(
120 1, sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100121
Lawrence Tange407b4c2022-07-21 13:54:01 +0100122 //Validation bits.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800123 ValidationTypes ui64Type = { UINT_64T, .value.ui64 = 0 };
124 struct json_object *obj = NULL;
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100125
Lawrence Tange407b4c2022-07-21 13:54:01 +0100126 //Error status.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800127 if (json_object_object_get_ex(section, "errorStatus", &obj)) {
128 ir_generic_error_status_to_cper(obj,
129 &section_cper->ErrorStatus);
130 add_to_valid_bitfield(&ui64Type, 0);
131 }
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100132
Lawrence Tange407b4c2022-07-21 13:54:01 +0100133 //Device ID information.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800134 if (json_object_object_get_ex(section, "idInfo", &obj)) {
135 json_object *id_info = obj;
136 section_cper->IdInfo.VendorId = json_object_get_uint64(
137 json_object_object_get(id_info, "vendorID"));
138 section_cper->IdInfo.DeviceId = json_object_get_uint64(
139 json_object_object_get(id_info, "deviceID"));
140 section_cper->IdInfo.ClassCode = json_object_get_uint64(
141 json_object_object_get(id_info, "classCode"));
142 section_cper->IdInfo.FunctionNumber = json_object_get_uint64(
143 json_object_object_get(id_info, "functionNumber"));
144 section_cper->IdInfo.DeviceNumber = json_object_get_uint64(
145 json_object_object_get(id_info, "deviceNumber"));
146 section_cper->IdInfo.BusNumber = json_object_get_uint64(
147 json_object_object_get(id_info, "busNumber"));
148 section_cper->IdInfo.SegmentNumber = json_object_get_uint64(
149 json_object_object_get(id_info, "segmentNumber"));
150 add_to_valid_bitfield(&ui64Type, 1);
151 }
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100152
Lawrence Tange407b4c2022-07-21 13:54:01 +0100153 //Amount of following data pairs.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800154 if (json_object_object_get_ex(section, "memoryNumber", &obj)) {
155 section_cper->MemoryNumber =
156 (UINT32)json_object_get_uint64(obj);
157 add_to_valid_bitfield(&ui64Type, 2);
158 }
159 if (json_object_object_get_ex(section, "ioNumber", &obj)) {
160 section_cper->IoNumber = (UINT32)json_object_get_uint64(obj);
161 add_to_valid_bitfield(&ui64Type, 3);
162 }
163 json_object *register_pairs = NULL;
164 if (json_object_object_get_ex(section, "registerDataPairs", &obj)) {
165 register_pairs = obj;
166 add_to_valid_bitfield(&ui64Type, 4);
167 }
168
169 section_cper->ValidFields = ui64Type.value.ui64;
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100170
Lawrence Tange407b4c2022-07-21 13:54:01 +0100171 //Write header out to stream, free it.
172 fwrite(section_cper, sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA), 1, out);
173 fflush(out);
174 free(section_cper);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100175
Lawrence Tange407b4c2022-07-21 13:54:01 +0100176 //Begin writing register pairs.
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800177 if (register_pairs != NULL) {
178 int num_pairs = json_object_array_length(register_pairs);
179 for (int i = 0; i < num_pairs; i++) {
180 //Get the pair array item out.
181 json_object *register_pair =
182 json_object_array_get_idx(register_pairs, i);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100183
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800184 //Create the pair array.
185 UINT64 pair[2];
186 pair[0] = json_object_get_uint64(json_object_object_get(
187 register_pair, "firstHalf"));
188 pair[1] = json_object_get_uint64(json_object_object_get(
189 register_pair, "secondHalf"));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100190
Aushim Nagarkattiae8f6d92025-01-29 17:34:44 -0800191 //Push to stream.
192 fwrite(pair, sizeof(UINT64), 2, out);
193 fflush(out);
194 }
Lawrence Tange407b4c2022-07-21 13:54:01 +0100195 }
John Chungf8fc7052024-05-03 20:05:29 +0800196}