1214a1542SLawrence Tang /**
2214a1542SLawrence Tang  * Describes functions for converting PCI/PCI-X bus CPER sections from binary and JSON format
3214a1542SLawrence Tang  * into an intermediate format.
4214a1542SLawrence Tang  *
5214a1542SLawrence Tang  * Author: Lawrence.Tang@arm.com
6214a1542SLawrence Tang  **/
7214a1542SLawrence Tang #include <stdio.h>
8*0a4b3f2dSLawrence Tang #include <string.h>
9214a1542SLawrence Tang #include "json.h"
10214a1542SLawrence Tang #include "../edk/Cper.h"
11214a1542SLawrence Tang #include "../cper-utils.h"
12214a1542SLawrence Tang #include "cper-section-pci-bus.h"
13214a1542SLawrence Tang 
14214a1542SLawrence Tang //Converts a single PCI/PCI-X bus CPER section into JSON IR.
15214a1542SLawrence Tang json_object* cper_section_pci_bus_to_ir(void* section, EFI_ERROR_SECTION_DESCRIPTOR* descriptor)
16214a1542SLawrence Tang {
17214a1542SLawrence Tang     EFI_PCI_PCIX_BUS_ERROR_DATA* bus_error = (EFI_PCI_PCIX_BUS_ERROR_DATA*)section;
18214a1542SLawrence Tang     json_object* section_ir = json_object_new_object();
19214a1542SLawrence Tang 
20214a1542SLawrence Tang     //Validation bits.
21214a1542SLawrence Tang     json_object* validation = bitfield_to_ir(bus_error->ValidFields, 9, PCI_BUS_ERROR_VALID_BITFIELD_NAMES);
22214a1542SLawrence Tang     json_object_object_add(section_ir, "validationBits", validation);
23214a1542SLawrence Tang 
24214a1542SLawrence Tang     //Error status.
25214a1542SLawrence Tang     json_object* error_status = cper_generic_error_status_to_ir(&bus_error->ErrorStatus);
26214a1542SLawrence Tang     json_object_object_add(section_ir, "errorStatus", error_status);
27214a1542SLawrence Tang 
28214a1542SLawrence Tang     //PCI bus error type.
29214a1542SLawrence Tang     json_object* error_type = integer_to_readable_pair(bus_error->Type, 8,
30214a1542SLawrence Tang         PCI_BUS_ERROR_TYPES_KEYS,
31214a1542SLawrence Tang         PCI_BUS_ERROR_TYPES_VALUES,
32214a1542SLawrence Tang         "Unknown (Reserved)");
33214a1542SLawrence Tang     json_object_object_add(section_ir, "errorType", error_type);
34214a1542SLawrence Tang 
35214a1542SLawrence Tang     //Bus ID.
36214a1542SLawrence Tang     json_object* bus_id = json_object_new_object();
37214a1542SLawrence Tang     json_object_object_add(bus_id, "busNumber", json_object_new_int(bus_error->BusId & 0xFF));
38214a1542SLawrence Tang     json_object_object_add(bus_id, "segmentNumber", json_object_new_int(bus_error->BusId >> 8));
39214a1542SLawrence Tang     json_object_object_add(section_ir, "busID", bus_id);
40214a1542SLawrence Tang 
41214a1542SLawrence Tang     //Miscellaneous numeric fields.
42de9707f9SLawrence Tang     UINT8 command_type = (bus_error->BusCommand >> 56) & 0b1; //Byte 7, bit 0.
43214a1542SLawrence Tang     json_object_object_add(section_ir, "busAddress", json_object_new_uint64(bus_error->BusAddress));
44214a1542SLawrence Tang     json_object_object_add(section_ir, "busData", json_object_new_uint64(bus_error->BusData));
45de9707f9SLawrence Tang     json_object_object_add(section_ir, "busCommandType", json_object_new_string(command_type == 0 ? "PCI" : "PCI-X"));
46214a1542SLawrence Tang     json_object_object_add(section_ir, "busRequestorID", json_object_new_uint64(bus_error->RequestorId));
47214a1542SLawrence Tang     json_object_object_add(section_ir, "busCompleterID", json_object_new_uint64(bus_error->ResponderId));
48214a1542SLawrence Tang     json_object_object_add(section_ir, "targetID", json_object_new_uint64(bus_error->TargetId));
49214a1542SLawrence Tang 
50214a1542SLawrence Tang     return section_ir;
51214a1542SLawrence Tang }
52205dd1d7SLawrence Tang 
53205dd1d7SLawrence Tang //Converts a single provided PCI/PCI-X bus CPER-JSON section into CPER binary, outputting to the
54205dd1d7SLawrence Tang //provided stream.
55205dd1d7SLawrence Tang void ir_section_pci_bus_to_cper(json_object* section, FILE* out)
56205dd1d7SLawrence Tang {
57205dd1d7SLawrence Tang     EFI_PCI_PCIX_BUS_ERROR_DATA* section_cper =
58205dd1d7SLawrence Tang         (EFI_PCI_PCIX_BUS_ERROR_DATA*)calloc(1, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA));
59205dd1d7SLawrence Tang 
60205dd1d7SLawrence Tang     //Validation bits.
61205dd1d7SLawrence Tang     section_cper->ValidFields = ir_to_bitfield(json_object_object_get(section, "validationBits"),
62205dd1d7SLawrence Tang         9, PCI_BUS_ERROR_VALID_BITFIELD_NAMES);
63205dd1d7SLawrence Tang 
64205dd1d7SLawrence Tang     //Error status.
65205dd1d7SLawrence Tang     ir_generic_error_status_to_cper(json_object_object_get(section, "errorStatus"), &section_cper->ErrorStatus);
66205dd1d7SLawrence Tang 
67205dd1d7SLawrence Tang     //Bus ID.
68205dd1d7SLawrence Tang     json_object* bus_id = json_object_object_get(section, "busID");
69205dd1d7SLawrence Tang     UINT16 bus_number = (UINT8)json_object_get_int(json_object_object_get(bus_id, "busNumber"));
70205dd1d7SLawrence Tang     UINT16 segment_number = (UINT8)json_object_get_int(json_object_object_get(bus_id, "segmentNumber"));
71205dd1d7SLawrence Tang     section_cper->BusId = bus_number + (segment_number << 8);
72205dd1d7SLawrence Tang 
73205dd1d7SLawrence Tang     //Remaining fields.
74*0a4b3f2dSLawrence Tang     UINT64 pcix_command = (UINT64)0x1 << 56;
75*0a4b3f2dSLawrence Tang     const char* bus_command = json_object_get_string(json_object_object_get(section, "busCommandType"));
76205dd1d7SLawrence Tang     section_cper->Type = (UINT16)readable_pair_to_integer(json_object_object_get(section, "errorType"));
77205dd1d7SLawrence Tang     section_cper->BusAddress = json_object_get_uint64(json_object_object_get(section, "busAddress"));
78205dd1d7SLawrence Tang     section_cper->BusData = json_object_get_uint64(json_object_object_get(section, "busData"));
79*0a4b3f2dSLawrence Tang     section_cper->BusCommand = strcmp(bus_command, "PCI") == 0 ? 0 : pcix_command;
80*0a4b3f2dSLawrence Tang     section_cper->RequestorId = json_object_get_uint64(json_object_object_get(section, "busRequestorID"));
81*0a4b3f2dSLawrence Tang     section_cper->ResponderId = json_object_get_uint64(json_object_object_get(section, "busCompleterID"));
82205dd1d7SLawrence Tang     section_cper->TargetId = json_object_get_uint64(json_object_object_get(section, "targetID"));
83205dd1d7SLawrence Tang 
84205dd1d7SLawrence Tang     //Write to stream, free resources.
853ab351feSLawrence Tang     fwrite(section_cper, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA), 1, out);
86205dd1d7SLawrence Tang     fflush(out);
87205dd1d7SLawrence Tang     free(section_cper);
88205dd1d7SLawrence Tang }