xref: /openbmc/libcper/generator/sections/gen-section-ia32x64.c (revision 2d4d3b65396596d8939bacaea54ed529530362f9)
102c801a5SLawrence Tang /**
2efe17e2cSLawrence Tang  * Functions for generating pseudo-random CPER IA32/x64 sections.
302c801a5SLawrence Tang  *
402c801a5SLawrence Tang  * Author: Lawrence.Tang@arm.com
502c801a5SLawrence Tang  **/
602c801a5SLawrence Tang 
702c801a5SLawrence Tang #include <stdlib.h>
802c801a5SLawrence Tang #include <string.h>
9e42fb487SThu Nguyen #include <libcper/Cper.h>
10e42fb487SThu Nguyen #include <libcper/generator/gen-utils.h>
11e42fb487SThu Nguyen #include <libcper/generator/sections/gen-section.h>
1267cbed6bSLawrence Tang #define IA32X64_ERROR_STRUCTURE_SIZE 64
1302c801a5SLawrence Tang 
14ae8f6d9aSAushim Nagarkatti void *generate_ia32x64_error_structure(GEN_VALID_BITS_TEST_TYPE validBitsType);
1502c801a5SLawrence Tang size_t generate_ia32x64_context_structure(void **location);
1602c801a5SLawrence Tang 
17efe17e2cSLawrence Tang //Generates a single pseudo-random IA32/x64 section, saving the resulting address to the given
1802c801a5SLawrence Tang //location. Returns the size of the newly created section.
generate_section_ia32x64(void ** location,GEN_VALID_BITS_TEST_TYPE validBitsType)19ae8f6d9aSAushim Nagarkatti size_t generate_section_ia32x64(void **location,
20ae8f6d9aSAushim Nagarkatti 				GEN_VALID_BITS_TEST_TYPE validBitsType)
2102c801a5SLawrence Tang {
2202c801a5SLawrence Tang 	//Set up for generation of error/context structures.
23*2d4d3b65SEd Tanous 	UINT16 error_structure_num = cper_rand() % 4 + 1;
24*2d4d3b65SEd Tanous 	UINT16 context_structure_num = cper_rand() % 4 + 1;
2502c801a5SLawrence Tang 	void *error_structures[error_structure_num];
2602c801a5SLawrence Tang 	void *context_structures[context_structure_num];
2702c801a5SLawrence Tang 	size_t context_structure_lengths[context_structure_num];
2802c801a5SLawrence Tang 
2902c801a5SLawrence Tang 	//Generate the structures.
30f8fc7052SJohn Chung 	for (int i = 0; i < error_structure_num; i++) {
31ae8f6d9aSAushim Nagarkatti 		error_structures[i] =
32ae8f6d9aSAushim Nagarkatti 			generate_ia32x64_error_structure(validBitsType);
33f8fc7052SJohn Chung 	}
34f8fc7052SJohn Chung 	for (int i = 0; i < context_structure_num; i++) {
35f8fc7052SJohn Chung 		context_structure_lengths[i] =
36f8fc7052SJohn Chung 			generate_ia32x64_context_structure(context_structures +
37f8fc7052SJohn Chung 							   i);
38f8fc7052SJohn Chung 	}
3902c801a5SLawrence Tang 
4002c801a5SLawrence Tang 	//Create a valid IA32/x64 section.
41f8fc7052SJohn Chung 	size_t total_len =
42f8fc7052SJohn Chung 		64 + (IA32X64_ERROR_STRUCTURE_SIZE * error_structure_num);
43f8fc7052SJohn Chung 	for (int i = 0; i < context_structure_num; i++) {
4402c801a5SLawrence Tang 		total_len += context_structure_lengths[i];
45f8fc7052SJohn Chung 	}
4602c801a5SLawrence Tang 	UINT8 *section = generate_random_bytes(total_len);
4702c801a5SLawrence Tang 
48aacf0e26SLawrence Tang 	//Null extend the end of the CPUID in the header.
49f8fc7052SJohn Chung 	for (int i = 0; i < 16; i++) {
50aacf0e26SLawrence Tang 		*(section + 48 + i) = 0x0;
51f8fc7052SJohn Chung 	}
52aacf0e26SLawrence Tang 
5302c801a5SLawrence Tang 	//Set header information.
5402c801a5SLawrence Tang 	UINT64 *validation = (UINT64 *)section;
55f8fc7052SJohn Chung 	*validation &= 0x3;
56ae8f6d9aSAushim Nagarkatti 	if (validBitsType == ALL_VALID) {
57ae8f6d9aSAushim Nagarkatti 		*validation = 0x3;
58ae8f6d9aSAushim Nagarkatti 	} else if (validBitsType == SOME_VALID) {
59ae8f6d9aSAushim Nagarkatti 		*validation = 0x2;
60ae8f6d9aSAushim Nagarkatti 	}
6102c801a5SLawrence Tang 	*validation |= error_structure_num << 2;
6202c801a5SLawrence Tang 	*validation |= context_structure_num << 8;
6302c801a5SLawrence Tang 
6402c801a5SLawrence Tang 	//Copy in structures, free resources.
6502c801a5SLawrence Tang 	UINT8 *cur_pos = section + 64;
66f8fc7052SJohn Chung 	for (int i = 0; i < error_structure_num; i++) {
67f8fc7052SJohn Chung 		memcpy(cur_pos, error_structures[i],
68f8fc7052SJohn Chung 		       IA32X64_ERROR_STRUCTURE_SIZE);
6902c801a5SLawrence Tang 		free(error_structures[i]);
7002c801a5SLawrence Tang 		cur_pos += IA32X64_ERROR_STRUCTURE_SIZE;
7102c801a5SLawrence Tang 	}
72f8fc7052SJohn Chung 	for (int i = 0; i < context_structure_num; i++) {
73f8fc7052SJohn Chung 		memcpy(cur_pos, context_structures[i],
74f8fc7052SJohn Chung 		       context_structure_lengths[i]);
7502c801a5SLawrence Tang 		free(context_structures[i]);
7602c801a5SLawrence Tang 		cur_pos += context_structure_lengths[i];
7702c801a5SLawrence Tang 	}
7802c801a5SLawrence Tang 
7902c801a5SLawrence Tang 	//Set return values, exist.
8002c801a5SLawrence Tang 	*location = section;
8102c801a5SLawrence Tang 	return total_len;
8202c801a5SLawrence Tang }
8302c801a5SLawrence Tang 
8402c801a5SLawrence Tang //Generates a single IA32/x64 error structure. Must later be freed.
generate_ia32x64_error_structure(GEN_VALID_BITS_TEST_TYPE validBitsType)85ae8f6d9aSAushim Nagarkatti void *generate_ia32x64_error_structure(GEN_VALID_BITS_TEST_TYPE validBitsType)
8602c801a5SLawrence Tang {
87f8fc7052SJohn Chung 	UINT8 *error_structure =
88f8fc7052SJohn Chung 		generate_random_bytes(IA32X64_ERROR_STRUCTURE_SIZE);
8902c801a5SLawrence Tang 
90aacf0e26SLawrence Tang 	//Set error structure reserved space to zero.
91aacf0e26SLawrence Tang 	UINT64 *validation = (UINT64 *)(error_structure + 16);
92aacf0e26SLawrence Tang 	*validation &= 0x1F;
93ae8f6d9aSAushim Nagarkatti 	if (validBitsType == ALL_VALID) {
94ae8f6d9aSAushim Nagarkatti 		*validation = 0x1F;
95ae8f6d9aSAushim Nagarkatti 	} else if (validBitsType == SOME_VALID) {
96ae8f6d9aSAushim Nagarkatti 		*validation = 0x15;
97ae8f6d9aSAushim Nagarkatti 	}
98aacf0e26SLawrence Tang 
9902c801a5SLawrence Tang 	//Create a random type of error structure.
10002c801a5SLawrence Tang 	EFI_GUID *guid = (EFI_GUID *)error_structure;
101aacf0e26SLawrence Tang 	UINT64 *check_info = (UINT64 *)(error_structure + 24);
102*2d4d3b65SEd Tanous 	int error_structure_type = cper_rand() % 4;
103f8fc7052SJohn Chung 	switch (error_structure_type) {
10402c801a5SLawrence Tang 	//Cache
10502c801a5SLawrence Tang 	case 0:
106f8fc7052SJohn Chung 		memcpy(guid, &gEfiIa32x64ErrorTypeCacheCheckGuid,
107f8fc7052SJohn Chung 		       sizeof(EFI_GUID));
108aacf0e26SLawrence Tang 
109aacf0e26SLawrence Tang 		//Set reserved space to zero.
110ae8f6d9aSAushim Nagarkatti 		*check_info = ~0x20FF00;
111aacf0e26SLawrence Tang 		*check_info &= 0x3FFFFFFF;
11202c801a5SLawrence Tang 		break;
11302c801a5SLawrence Tang 
11402c801a5SLawrence Tang 	//TLB
11502c801a5SLawrence Tang 	case 1:
116f8fc7052SJohn Chung 		memcpy(guid, &gEfiIa32x64ErrorTypeTlbCheckGuid,
117f8fc7052SJohn Chung 		       sizeof(EFI_GUID));
118aacf0e26SLawrence Tang 
119aacf0e26SLawrence Tang 		//Set reserved space to zero.
120ae8f6d9aSAushim Nagarkatti 		*check_info = ~0x20FF00;
121aacf0e26SLawrence Tang 		*check_info &= 0x3FFFFFFF;
12202c801a5SLawrence Tang 		break;
12302c801a5SLawrence Tang 
12402c801a5SLawrence Tang 	//Bus
12502c801a5SLawrence Tang 	case 2:
126f8fc7052SJohn Chung 		memcpy(guid, &gEfiIa32x64ErrorTypeBusCheckGuid,
127f8fc7052SJohn Chung 		       sizeof(EFI_GUID));
128aacf0e26SLawrence Tang 
129aacf0e26SLawrence Tang 		//Set reserved space to zero.
130ae8f6d9aSAushim Nagarkatti 		*check_info = ~0x20F800;
131aacf0e26SLawrence Tang 		*check_info &= 0x7FFFFFFFF;
13202c801a5SLawrence Tang 		break;
13302c801a5SLawrence Tang 
13402c801a5SLawrence Tang 	//MS
13502c801a5SLawrence Tang 	case 3:
136f8fc7052SJohn Chung 		memcpy(guid, &gEfiIa32x64ErrorTypeMsCheckGuid,
137f8fc7052SJohn Chung 		       sizeof(EFI_GUID));
138aacf0e26SLawrence Tang 
139aacf0e26SLawrence Tang 		//Set reserved space to zero.
140ae8f6d9aSAushim Nagarkatti 		*check_info = ~0xFFC0;
141aacf0e26SLawrence Tang 		*check_info &= 0xFFFFFF;
14202c801a5SLawrence Tang 		break;
14302c801a5SLawrence Tang 	}
14402c801a5SLawrence Tang 
14502c801a5SLawrence Tang 	return error_structure;
14602c801a5SLawrence Tang }
14702c801a5SLawrence Tang 
14802c801a5SLawrence Tang //Generates a single IA32/x64 context structure. Must later be freed.
generate_ia32x64_context_structure(void ** location)14902c801a5SLawrence Tang size_t generate_ia32x64_context_structure(void **location)
15002c801a5SLawrence Tang {
15102c801a5SLawrence Tang 	//Initial length is 16 bytes. Add extra based on type.
152*2d4d3b65SEd Tanous 	int reg_type = cper_rand() % 8;
15302c801a5SLawrence Tang 	int reg_size = 0;
15402c801a5SLawrence Tang 
15502c801a5SLawrence Tang 	//Set register size.
156f8fc7052SJohn Chung 	if (reg_type == 2) {
15702c801a5SLawrence Tang 		reg_size = 92;			       //IA32 registers.
158f8fc7052SJohn Chung 	} else if (reg_type == 3) {
15902c801a5SLawrence Tang 		reg_size = 244;			       //x64 registers.
160f8fc7052SJohn Chung 	} else {
161*2d4d3b65SEd Tanous 		reg_size = (cper_rand() % 5 + 1) * 32; //Not table defined.
16204750a9eSLawrence Tang 	}
16302c801a5SLawrence Tang 
16402c801a5SLawrence Tang 	//Create structure randomly.
16502c801a5SLawrence Tang 	int total_size = 16 + reg_size;
16602c801a5SLawrence Tang 	UINT16 *context_structure = (UINT16 *)generate_random_bytes(total_size);
16702c801a5SLawrence Tang 
16804750a9eSLawrence Tang 	//If it is x64 registers, set reserved area accordingly.
169f8fc7052SJohn Chung 	if (reg_type == 3) {
17004750a9eSLawrence Tang 		UINT8 *reg_bytes = (UINT8 *)(context_structure + 8);
17104750a9eSLawrence Tang 		UINT32 *reserved = (UINT32 *)(reg_bytes + 140);
17204750a9eSLawrence Tang 		*reserved = 0;
17304750a9eSLawrence Tang 	}
17404750a9eSLawrence Tang 
17502c801a5SLawrence Tang 	//Set header information.
17602c801a5SLawrence Tang 	*(context_structure) = reg_type;
17702c801a5SLawrence Tang 	*(context_structure + 1) = reg_size;
17802c801a5SLawrence Tang 
17902c801a5SLawrence Tang 	//Set return values and exit.
18002c801a5SLawrence Tang 	*location = context_structure;
18102c801a5SLawrence Tang 	return total_size;
18202c801a5SLawrence Tang }
183