blob: 5f4018315e1065e99440252fe5792ccaec537c58 [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.
4 *
5 * Author: Lawrence.Tang@arm.com
6 **/
7#include <stdio.h>
Lawrence Tang5202bbb2022-08-12 14:54:36 +01008#include <json.h>
Lawrence Tanga416ec92022-07-06 14:34:40 +01009#include "../edk/Cper.h"
10#include "../cper-utils.h"
11#include "cper-section-pci-dev.h"
12
13//Converts a single PCI/PCI-X device CPER section into JSON IR.
Lawrence Tange407b4c2022-07-21 13:54:01 +010014json_object *
15cper_section_pci_dev_to_ir(void *section,
16 EFI_ERROR_SECTION_DESCRIPTOR *descriptor)
Lawrence Tanga416ec92022-07-06 14:34:40 +010017{
Lawrence Tange407b4c2022-07-21 13:54:01 +010018 EFI_PCI_PCIX_DEVICE_ERROR_DATA *dev_error =
19 (EFI_PCI_PCIX_DEVICE_ERROR_DATA *)section;
20 json_object *section_ir = json_object_new_object();
Lawrence Tanga416ec92022-07-06 14:34:40 +010021
Lawrence Tange407b4c2022-07-21 13:54:01 +010022 //Validation bits.
23 json_object *validation = bitfield_to_ir(
24 dev_error->ValidFields, 5, PCI_DEV_ERROR_VALID_BITFIELD_NAMES);
25 json_object_object_add(section_ir, "validationBits", validation);
Lawrence Tanga416ec92022-07-06 14:34:40 +010026
Lawrence Tange407b4c2022-07-21 13:54:01 +010027 //Error status.
28 json_object *error_status =
29 cper_generic_error_status_to_ir(&dev_error->ErrorStatus);
30 json_object_object_add(section_ir, "errorStatus", error_status);
Lawrence Tanga416ec92022-07-06 14:34:40 +010031
Lawrence Tange407b4c2022-07-21 13:54:01 +010032 //ID information.
33 json_object *id_info = json_object_new_object();
34 json_object_object_add(
35 id_info, "vendorID",
36 json_object_new_uint64(dev_error->IdInfo.VendorId));
37 json_object_object_add(
38 id_info, "deviceID",
39 json_object_new_uint64(dev_error->IdInfo.DeviceId));
40 json_object_object_add(
41 id_info, "classCode",
42 json_object_new_uint64(dev_error->IdInfo.ClassCode));
43 json_object_object_add(
44 id_info, "functionNumber",
45 json_object_new_uint64(dev_error->IdInfo.FunctionNumber));
46 json_object_object_add(
47 id_info, "deviceNumber",
48 json_object_new_uint64(dev_error->IdInfo.DeviceNumber));
49 json_object_object_add(
50 id_info, "busNumber",
51 json_object_new_uint64(dev_error->IdInfo.BusNumber));
52 json_object_object_add(
53 id_info, "segmentNumber",
54 json_object_new_uint64(dev_error->IdInfo.SegmentNumber));
55 json_object_object_add(section_ir, "idInfo", id_info);
Lawrence Tanga416ec92022-07-06 14:34:40 +010056
Lawrence Tange407b4c2022-07-21 13:54:01 +010057 //Number of following register data pairs.
58 json_object_object_add(section_ir, "memoryNumber",
59 json_object_new_uint64(dev_error->MemoryNumber));
60 json_object_object_add(section_ir, "ioNumber",
61 json_object_new_uint64(dev_error->IoNumber));
62 int num_data_pairs = dev_error->MemoryNumber + dev_error->IoNumber;
Lawrence Tanga416ec92022-07-06 14:34:40 +010063
Lawrence Tange407b4c2022-07-21 13:54:01 +010064 //Register pairs, described by the numeric fields.
65 //The actual "pairs" of address and data aren't necessarily 8 bytes long, so can't assume the contents.
66 //Hence the naming "firstHalf" and "secondHalf" rather than "address" and "data".
67 json_object *register_data_pair_array = json_object_new_array();
68 UINT64 *cur_pos = (UINT64 *)(dev_error + 1);
69 for (int i = 0; i < num_data_pairs; i++) {
70 //Save current pair to array.
71 json_object *register_data_pair = json_object_new_object();
72 json_object_object_add(register_data_pair, "firstHalf",
73 json_object_new_uint64(*cur_pos));
74 json_object_object_add(register_data_pair, "secondHalf",
75 json_object_new_uint64(*(cur_pos + 1)));
76 json_object_array_add(register_data_pair_array,
77 register_data_pair);
Lawrence Tanga416ec92022-07-06 14:34:40 +010078
Lawrence Tange407b4c2022-07-21 13:54:01 +010079 //Move to next pair.
80 cur_pos += 2;
81 }
82 json_object_object_add(section_ir, "registerDataPairs",
83 register_data_pair_array);
Lawrence Tanga416ec92022-07-06 14:34:40 +010084
Lawrence Tange407b4c2022-07-21 13:54:01 +010085 return section_ir;
Lawrence Tang205dd1d2022-07-14 16:23:38 +010086}
87
Lawrence Tange407b4c2022-07-21 13:54:01 +010088void ir_section_pci_dev_to_cper(json_object *section, FILE *out)
Lawrence Tang205dd1d2022-07-14 16:23:38 +010089{
Lawrence Tange407b4c2022-07-21 13:54:01 +010090 EFI_PCI_PCIX_DEVICE_ERROR_DATA *section_cper =
91 (EFI_PCI_PCIX_DEVICE_ERROR_DATA *)calloc(
92 1, sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA));
Lawrence Tang205dd1d2022-07-14 16:23:38 +010093
Lawrence Tange407b4c2022-07-21 13:54:01 +010094 //Validation bits.
95 section_cper->ValidFields = ir_to_bitfield(
96 json_object_object_get(section, "validationBits"), 5,
97 PCI_DEV_ERROR_VALID_BITFIELD_NAMES);
Lawrence Tang205dd1d2022-07-14 16:23:38 +010098
Lawrence Tange407b4c2022-07-21 13:54:01 +010099 //Error status.
100 ir_generic_error_status_to_cper(json_object_object_get(section,
101 "errorStatus"),
102 &section_cper->ErrorStatus);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100103
Lawrence Tange407b4c2022-07-21 13:54:01 +0100104 //Device ID information.
105 json_object *id_info = json_object_object_get(section, "idInfo");
106 section_cper->IdInfo.VendorId = json_object_get_uint64(
107 json_object_object_get(id_info, "vendorID"));
108 section_cper->IdInfo.DeviceId = json_object_get_uint64(
109 json_object_object_get(id_info, "deviceID"));
110 section_cper->IdInfo.ClassCode = json_object_get_uint64(
111 json_object_object_get(id_info, "classCode"));
112 section_cper->IdInfo.FunctionNumber = json_object_get_uint64(
113 json_object_object_get(id_info, "functionNumber"));
114 section_cper->IdInfo.DeviceNumber = json_object_get_uint64(
115 json_object_object_get(id_info, "deviceNumber"));
116 section_cper->IdInfo.BusNumber = json_object_get_uint64(
117 json_object_object_get(id_info, "busNumber"));
118 section_cper->IdInfo.SegmentNumber = json_object_get_uint64(
119 json_object_object_get(id_info, "segmentNumber"));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100120
Lawrence Tange407b4c2022-07-21 13:54:01 +0100121 //Amount of following data pairs.
122 section_cper->MemoryNumber = (UINT32)json_object_get_uint64(
123 json_object_object_get(section, "memoryNumber"));
124 section_cper->IoNumber = (UINT32)json_object_get_uint64(
125 json_object_object_get(section, "ioNumber"));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100126
Lawrence Tange407b4c2022-07-21 13:54:01 +0100127 //Write header out to stream, free it.
128 fwrite(section_cper, sizeof(EFI_PCI_PCIX_DEVICE_ERROR_DATA), 1, out);
129 fflush(out);
130 free(section_cper);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100131
Lawrence Tange407b4c2022-07-21 13:54:01 +0100132 //Begin writing register pairs.
133 json_object *register_pairs =
134 json_object_object_get(section, "registerDataPairs");
135 int num_pairs = json_object_array_length(register_pairs);
136 for (int i = 0; i < num_pairs; i++) {
137 //Get the pair array item out.
138 json_object *register_pair =
139 json_object_array_get_idx(register_pairs, i);
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100140
Lawrence Tange407b4c2022-07-21 13:54:01 +0100141 //Create the pair array.
142 UINT64 pair[2];
143 pair[0] = json_object_get_uint64(
144 json_object_object_get(register_pair, "firstHalf"));
145 pair[1] = json_object_get_uint64(
146 json_object_object_get(register_pair, "secondHalf"));
Lawrence Tang205dd1d2022-07-14 16:23:38 +0100147
Lawrence Tange407b4c2022-07-21 13:54:01 +0100148 //Push to stream.
149 fwrite(pair, sizeof(UINT64), 2, out);
150 fflush(out);
151 }
Lawrence Tanga416ec92022-07-06 14:34:40 +0100152}