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> 9*5202bbb4SLawrence 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. 15e407b4c8SLawrence Tang json_object * 16e407b4c8SLawrence Tang cper_section_pci_bus_to_ir(void *section, 17e407b4c8SLawrence Tang EFI_ERROR_SECTION_DESCRIPTOR *descriptor) 18214a1542SLawrence Tang { 19e407b4c8SLawrence Tang EFI_PCI_PCIX_BUS_ERROR_DATA *bus_error = 20e407b4c8SLawrence Tang (EFI_PCI_PCIX_BUS_ERROR_DATA *)section; 21214a1542SLawrence Tang json_object *section_ir = json_object_new_object(); 22214a1542SLawrence Tang 23214a1542SLawrence Tang //Validation bits. 24e407b4c8SLawrence Tang json_object *validation = bitfield_to_ir( 25e407b4c8SLawrence 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. 29e407b4c8SLawrence Tang json_object *error_status = 30e407b4c8SLawrence 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. 34e407b4c8SLawrence Tang json_object *error_type = integer_to_readable_pair( 35e407b4c8SLawrence Tang bus_error->Type, 8, PCI_BUS_ERROR_TYPES_KEYS, 36e407b4c8SLawrence 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(); 41e407b4c8SLawrence Tang json_object_object_add(bus_id, "busNumber", 42e407b4c8SLawrence Tang json_object_new_int(bus_error->BusId & 0xFF)); 43e407b4c8SLawrence Tang json_object_object_add(bus_id, "segmentNumber", 44e407b4c8SLawrence 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. 48e407b4c8SLawrence Tang UINT8 command_type = (bus_error->BusCommand >> 56) & 49e407b4c8SLawrence Tang 0b1; //Byte 7, bit 0. 50e407b4c8SLawrence Tang json_object_object_add(section_ir, "busAddress", 51e407b4c8SLawrence Tang json_object_new_uint64(bus_error->BusAddress)); 52e407b4c8SLawrence Tang json_object_object_add(section_ir, "busData", 53e407b4c8SLawrence Tang json_object_new_uint64(bus_error->BusData)); 54e407b4c8SLawrence Tang json_object_object_add( 55e407b4c8SLawrence Tang section_ir, "busCommandType", 56e407b4c8SLawrence Tang json_object_new_string(command_type == 0 ? "PCI" : "PCI-X")); 57e407b4c8SLawrence Tang json_object_object_add(section_ir, "busRequestorID", 58e407b4c8SLawrence Tang json_object_new_uint64(bus_error->RequestorId)); 59e407b4c8SLawrence Tang json_object_object_add(section_ir, "busCompleterID", 60e407b4c8SLawrence Tang json_object_new_uint64(bus_error->ResponderId)); 61e407b4c8SLawrence Tang json_object_object_add(section_ir, "targetID", 62e407b4c8SLawrence 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 = 72e407b4c8SLawrence Tang (EFI_PCI_PCIX_BUS_ERROR_DATA *)calloc( 73e407b4c8SLawrence Tang 1, sizeof(EFI_PCI_PCIX_BUS_ERROR_DATA)); 74205dd1d7SLawrence Tang 75205dd1d7SLawrence Tang //Validation bits. 76e407b4c8SLawrence Tang section_cper->ValidFields = ir_to_bitfield( 77e407b4c8SLawrence Tang json_object_object_get(section, "validationBits"), 9, 78e407b4c8SLawrence Tang PCI_BUS_ERROR_VALID_BITFIELD_NAMES); 79205dd1d7SLawrence Tang 80205dd1d7SLawrence Tang //Error status. 81e407b4c8SLawrence Tang ir_generic_error_status_to_cper(json_object_object_get(section, 82e407b4c8SLawrence Tang "errorStatus"), 83e407b4c8SLawrence Tang §ion_cper->ErrorStatus); 84205dd1d7SLawrence Tang 85205dd1d7SLawrence Tang //Bus ID. 86205dd1d7SLawrence Tang json_object *bus_id = json_object_object_get(section, "busID"); 87e407b4c8SLawrence Tang UINT16 bus_number = (UINT8)json_object_get_int( 88e407b4c8SLawrence Tang json_object_object_get(bus_id, "busNumber")); 89e407b4c8SLawrence Tang UINT16 segment_number = (UINT8)json_object_get_int( 90e407b4c8SLawrence 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; 95e407b4c8SLawrence Tang const char *bus_command = json_object_get_string( 96e407b4c8SLawrence Tang json_object_object_get(section, "busCommandType")); 97e407b4c8SLawrence Tang section_cper->Type = (UINT16)readable_pair_to_integer( 98e407b4c8SLawrence Tang json_object_object_get(section, "errorType")); 99e407b4c8SLawrence Tang section_cper->BusAddress = json_object_get_uint64( 100e407b4c8SLawrence Tang json_object_object_get(section, "busAddress")); 101e407b4c8SLawrence Tang section_cper->BusData = json_object_get_uint64( 102e407b4c8SLawrence Tang json_object_object_get(section, "busData")); 103e407b4c8SLawrence Tang section_cper->BusCommand = 104e407b4c8SLawrence Tang strcmp(bus_command, "PCI") == 0 ? 0 : pcix_command; 105e407b4c8SLawrence Tang section_cper->RequestorId = json_object_get_uint64( 106e407b4c8SLawrence Tang json_object_object_get(section, "busRequestorID")); 107e407b4c8SLawrence Tang section_cper->ResponderId = json_object_get_uint64( 108e407b4c8SLawrence Tang json_object_object_get(section, "busCompleterID")); 109e407b4c8SLawrence Tang section_cper->TargetId = json_object_get_uint64( 110e407b4c8SLawrence 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 }