1db1b7ce2SLawrence Tang /**
2db1b7ce2SLawrence Tang  * Describes functions for converting VT-d specific DMAr CPER sections from binary and JSON format
3db1b7ce2SLawrence Tang  * into an intermediate format.
4db1b7ce2SLawrence Tang  *
5db1b7ce2SLawrence Tang  * Author: Lawrence.Tang@arm.com
6db1b7ce2SLawrence Tang  **/
7db1b7ce2SLawrence Tang #include <stdio.h>
8db1b7ce2SLawrence Tang #include "json.h"
9*ce0f82bfSLawrence Tang #include "b64.h"
10db1b7ce2SLawrence Tang #include "../edk/Cper.h"
11db1b7ce2SLawrence Tang #include "../cper-utils.h"
12db1b7ce2SLawrence Tang #include "cper-section-dmar-vtd.h"
13db1b7ce2SLawrence Tang 
14db1b7ce2SLawrence Tang //Converts a single VT-d specific DMAr CPER section into JSON IR.
15db1b7ce2SLawrence Tang json_object* cper_section_dmar_vtd_to_ir(void* section, EFI_ERROR_SECTION_DESCRIPTOR* descriptor)
16db1b7ce2SLawrence Tang {
17db1b7ce2SLawrence Tang     EFI_DIRECTED_IO_DMAR_ERROR_DATA* vtd_error = (EFI_DIRECTED_IO_DMAR_ERROR_DATA*)section;
18db1b7ce2SLawrence Tang     json_object* section_ir = json_object_new_object();
19db1b7ce2SLawrence Tang 
20db1b7ce2SLawrence Tang     //Version, revision and OEM ID, as defined in the VT-d architecture.
21db1b7ce2SLawrence Tang     UINT64 oem_id = (vtd_error->OemId[0] << 16) + (vtd_error->OemId[1] << 8) + vtd_error->OemId[2];
22db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "version", json_object_new_int(vtd_error->Version));
23db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "revision", json_object_new_int(vtd_error->Revision));
24db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "oemID", json_object_new_uint64(oem_id));
25db1b7ce2SLawrence Tang 
26db1b7ce2SLawrence Tang     //Registers.
27db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "capabilityRegister", json_object_new_uint64(vtd_error->Capability));
28db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "extendedCapabilityRegister", json_object_new_uint64(vtd_error->CapabilityEx));
29db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "globalCommandRegister", json_object_new_uint64(vtd_error->GlobalCommand));
30db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "globalStatusRegister", json_object_new_uint64(vtd_error->GlobalStatus));
31db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "faultStatusRegister", json_object_new_uint64(vtd_error->FaultStatus));
32db1b7ce2SLawrence Tang 
33*ce0f82bfSLawrence Tang     //Fault record basic fields.
34*ce0f82bfSLawrence Tang     json_object* fault_record_ir = json_object_new_object();
35*ce0f82bfSLawrence Tang     EFI_VTD_FAULT_RECORD* fault_record = (EFI_VTD_FAULT_RECORD*)vtd_error->FaultRecord;
36*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "faultInformation", json_object_new_uint64(fault_record->FaultInformation));
37*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "sourceIdentifier", json_object_new_uint64(fault_record->SourceIdentifier));
38*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "privelegeModeRequested",
39*ce0f82bfSLawrence Tang         json_object_new_boolean(fault_record->PrivelegeModeRequested));
40*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "executePermissionRequested",
41*ce0f82bfSLawrence Tang         json_object_new_boolean(fault_record->ExecutePermissionRequested));
42*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "pasidPresent", json_object_new_boolean(fault_record->PasidPresent));
43*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "faultReason", json_object_new_uint64(fault_record->FaultReason));
44*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "pasidValue", json_object_new_uint64(fault_record->PasidValue));
45*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "addressType", json_object_new_uint64(fault_record->AddressType));
46*ce0f82bfSLawrence Tang 
47*ce0f82bfSLawrence Tang     //Fault record address type.
48*ce0f82bfSLawrence Tang     json_object* fault_record_type = integer_to_readable_pair(fault_record->Type, 2,
49*ce0f82bfSLawrence Tang         VTD_FAULT_RECORD_TYPES_KEYS,
50*ce0f82bfSLawrence Tang         VTD_FAULT_RECORD_TYPES_VALUES,
51*ce0f82bfSLawrence Tang         "Unknown");
52*ce0f82bfSLawrence Tang     json_object_object_add(fault_record_ir, "type", fault_record_type);
53*ce0f82bfSLawrence Tang     json_object_object_add(section_ir, "faultRecord", fault_record_ir);
54*ce0f82bfSLawrence Tang 
55*ce0f82bfSLawrence Tang     //Root entry.
56*ce0f82bfSLawrence Tang     char* encoded = b64_encode((unsigned char*)vtd_error->RootEntry, 16);
57*ce0f82bfSLawrence Tang     json_object_object_add(section_ir, "rootEntry", json_object_new_string(encoded));
58*ce0f82bfSLawrence Tang     free(encoded);
59*ce0f82bfSLawrence Tang 
60*ce0f82bfSLawrence Tang     //Context entry.
61*ce0f82bfSLawrence Tang     encoded = b64_encode((unsigned char*)vtd_error->ContextEntry, 16);
62*ce0f82bfSLawrence Tang     json_object_object_add(section_ir, "contextEntry", json_object_new_string(encoded));
63*ce0f82bfSLawrence Tang     free(encoded);
64db1b7ce2SLawrence Tang 
65db1b7ce2SLawrence Tang     //PTE entry for all page levels.
66db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level6", json_object_new_uint64(vtd_error->PteL6));
67db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level5", json_object_new_uint64(vtd_error->PteL5));
68db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level4", json_object_new_uint64(vtd_error->PteL4));
69db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level3", json_object_new_uint64(vtd_error->PteL3));
70db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level2", json_object_new_uint64(vtd_error->PteL2));
71db1b7ce2SLawrence Tang     json_object_object_add(section_ir, "pageTableEntry_Level1", json_object_new_uint64(vtd_error->PteL1));
72db1b7ce2SLawrence Tang 
73db1b7ce2SLawrence Tang     return section_ir;
74db1b7ce2SLawrence Tang }