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> 80a4b3f2dSLawrence 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. 15*e407b4c8SLawrence Tang json_object * 16*e407b4c8SLawrence Tang cper_section_pci_bus_to_ir(void *section, 17*e407b4c8SLawrence Tang EFI_ERROR_SECTION_DESCRIPTOR *descriptor) 18214a1542SLawrence Tang { 19*e407b4c8SLawrence Tang EFI_PCI_PCIX_BUS_ERROR_DATA *bus_error = 20*e407b4c8SLawrence Tang (EFI_PCI_PCIX_BUS_ERROR_DATA *)section; 21214a1542SLawrence Tang json_object *section_ir = json_object_new_object(); 22214a1542SLawrence Tang 23214a1542SLawrence Tang //Validation bits. 24*e407b4c8SLawrence Tang json_object *validation = bitfield_to_ir( 25*e407b4c8SLawrence Tang bus_error->ValidFields, 9, PCI_BUS_ERROR_VALID_BITFIELD_NAMES); 26214a1542SLawrence Tang json_object_object_add(section_ir, "validationBits", validation); 27214a1542SLawrence Tang 28214a1542SLawrence Tang //Error status. 29*e407b4c8SLawrence Tang json_object *error_status = 30*e407b4c8SLawrence Tang cper_generic_error_status_to_ir(&bus_error->ErrorStatus); 31214a1542SLawrence Tang json_object_object_add(section_ir, "errorStatus", error_status); 32214a1542SLawrence Tang 33214a1542SLawrence Tang //PCI bus error type. 34*e407b4c8SLawrence Tang json_object *error_type = integer_to_readable_pair( 35*e407b4c8SLawrence Tang bus_error->Type, 8, PCI_BUS_ERROR_TYPES_KEYS, 36*e407b4c8SLawrence Tang PCI_BUS_ERROR_TYPES_VALUES, "Unknown (Reserved)"); 37214a1542SLawrence Tang json_object_object_add(section_ir, "errorType", error_type); 38214a1542SLawrence Tang 39214a1542SLawrence Tang //Bus ID. 40214a1542SLawrence Tang json_object *bus_id = json_object_new_object(); 41*e407b4c8SLawrence Tang json_object_object_add(bus_id, "busNumber", 42*e407b4c8SLawrence Tang json_object_new_int(bus_error->BusId & 0xFF)); 43*e407b4c8SLawrence Tang json_object_object_add(bus_id, "segmentNumber", 44*e407b4c8SLawrence Tang json_object_new_int(bus_error->BusId >> 8)); 45214a1542SLawrence Tang json_object_object_add(section_ir, "busID", bus_id); 46214a1542SLawrence Tang 47214a1542SLawrence Tang //Miscellaneous numeric fields. 48*e407b4c8SLawrence Tang UINT8 command_type = (bus_error->BusCommand >> 56) & 49*e407b4c8SLawrence Tang 0b1; //Byte 7, bit 0. 50*e407b4c8SLawrence Tang json_object_object_add(section_ir, "busAddress", 51*e407b4c8SLawrence Tang json_object_new_uint64(bus_error->BusAddress)); 52*e407b4c8SLawrence Tang json_object_object_add(section_ir, "busData", 53*e407b4c8SLawrence Tang json_object_new_uint64(bus_error->BusData)); 54*e407b4c8SLawrence Tang json_object_object_add( 55*e407b4c8SLawrence Tang section_ir, "busCommandType", 56*e407b4c8SLawrence Tang json_object_new_string(command_type == 0 ? "PCI" : "PCI-X")); 57*e407b4c8SLawrence Tang json_object_object_add(section_ir, "busRequestorID", 58*e407b4c8SLawrence Tang json_object_new_uint64(bus_error->RequestorId)); 59*e407b4c8SLawrence Tang json_object_object_add(section_ir, "busCompleterID", 60*e407b4c8SLawrence Tang json_object_new_uint64(bus_error->ResponderId)); 61*e407b4c8SLawrence Tang json_object_object_add(section_ir, "targetID", 62*e407b4c8SLawrence Tang json_object_new_uint64(bus_error->TargetId)); 63214a1542SLawrence Tang 64214a1542SLawrence Tang return section_ir; 65214a1542SLawrence Tang } 66205dd1d7SLawrence Tang 67205dd1d7SLawrence Tang //Converts a single provided PCI/PCI-X bus CPER-JSON section into CPER binary, outputting to the 68205dd1d7SLawrence Tang //provided stream. 69205dd1d7SLawrence Tang void ir_section_pci_bus_to_cper(json_object *section, FILE *out) 70205dd1d7SLawrence Tang { 71205dd1d7SLawrence Tang EFI_PCI_PCIX_BUS_ERROR_DATA *section_cper = 72*e407b4c8SLawrence Tang (EFI_PCI_PCIX_BUS_ERROR_DATA *)calloc( 73*e407b4c8SLawrence Tang 1, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA)); 74205dd1d7SLawrence Tang 75205dd1d7SLawrence Tang //Validation bits. 76*e407b4c8SLawrence Tang section_cper->ValidFields = ir_to_bitfield( 77*e407b4c8SLawrence Tang json_object_object_get(section, "validationBits"), 9, 78*e407b4c8SLawrence Tang PCI_BUS_ERROR_VALID_BITFIELD_NAMES); 79205dd1d7SLawrence Tang 80205dd1d7SLawrence Tang //Error status. 81*e407b4c8SLawrence Tang ir_generic_error_status_to_cper(json_object_object_get(section, 82*e407b4c8SLawrence Tang "errorStatus"), 83*e407b4c8SLawrence Tang §ion_cper->ErrorStatus); 84205dd1d7SLawrence Tang 85205dd1d7SLawrence Tang //Bus ID. 86205dd1d7SLawrence Tang json_object *bus_id = json_object_object_get(section, "busID"); 87*e407b4c8SLawrence Tang UINT16 bus_number = (UINT8)json_object_get_int( 88*e407b4c8SLawrence Tang json_object_object_get(bus_id, "busNumber")); 89*e407b4c8SLawrence Tang UINT16 segment_number = (UINT8)json_object_get_int( 90*e407b4c8SLawrence Tang json_object_object_get(bus_id, "segmentNumber")); 91205dd1d7SLawrence Tang section_cper->BusId = bus_number + (segment_number << 8); 92205dd1d7SLawrence Tang 93205dd1d7SLawrence Tang //Remaining fields. 940a4b3f2dSLawrence Tang UINT64 pcix_command = (UINT64)0x1 << 56; 95*e407b4c8SLawrence Tang const char *bus_command = json_object_get_string( 96*e407b4c8SLawrence Tang json_object_object_get(section, "busCommandType")); 97*e407b4c8SLawrence Tang section_cper->Type = (UINT16)readable_pair_to_integer( 98*e407b4c8SLawrence Tang json_object_object_get(section, "errorType")); 99*e407b4c8SLawrence Tang section_cper->BusAddress = json_object_get_uint64( 100*e407b4c8SLawrence Tang json_object_object_get(section, "busAddress")); 101*e407b4c8SLawrence Tang section_cper->BusData = json_object_get_uint64( 102*e407b4c8SLawrence Tang json_object_object_get(section, "busData")); 103*e407b4c8SLawrence Tang section_cper->BusCommand = 104*e407b4c8SLawrence Tang strcmp(bus_command, "PCI") == 0 ? 0 : pcix_command; 105*e407b4c8SLawrence Tang section_cper->RequestorId = json_object_get_uint64( 106*e407b4c8SLawrence Tang json_object_object_get(section, "busRequestorID")); 107*e407b4c8SLawrence Tang section_cper->ResponderId = json_object_get_uint64( 108*e407b4c8SLawrence Tang json_object_object_get(section, "busCompleterID")); 109*e407b4c8SLawrence Tang section_cper->TargetId = json_object_get_uint64( 110*e407b4c8SLawrence Tang json_object_object_get(section, "targetID")); 111205dd1d7SLawrence Tang 112205dd1d7SLawrence Tang //Write to stream, free resources. 1133ab351feSLawrence Tang fwrite(section_cper, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA), 1, out); 114205dd1d7SLawrence Tang fflush(out); 115205dd1d7SLawrence Tang free(section_cper); 116205dd1d7SLawrence Tang }