Loading...
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 | /* SPDX-License-Identifier: GPL-2.0+ */ /* * (C) Copyright 2002-2010 * Wolfgang Denk, DENX Software Engineering, wd@denx.de. */ #ifndef __ASM_GBL_DATA_H #define __ASM_GBL_DATA_H #ifndef __ASSEMBLY__ #include <linux/types.h> #include <asm/msr-index.h> #include <asm/processor.h> #include <asm/mrccache.h> #include <asm/u-boot.h> enum pei_boot_mode_t { PEI_BOOT_NONE = 0, PEI_BOOT_SOFT_RESET, PEI_BOOT_RESUME, }; struct dimm_info { uint32_t dimm_size; uint16_t ddr_type; uint16_t ddr_frequency; uint8_t rank_per_dimm; uint8_t channel_num; uint8_t dimm_num; uint8_t bank_locator; /* The 5th byte is '\0' for the end of string */ uint8_t serial[5]; /* The 19th byte is '\0' for the end of string */ uint8_t module_part_number[19]; uint16_t mod_id; uint8_t mod_type; uint8_t bus_width; } __packed; struct pei_memory_info { uint8_t dimm_cnt; /* Maximum num of dimm is 8 */ struct dimm_info dimm[8]; } __packed; struct memory_area { uint64_t start; uint64_t size; }; struct memory_info { int num_areas; uint64_t total_memory; uint64_t total_32bit_memory; struct memory_area area[CONFIG_NR_DRAM_BANKS]; }; #define MAX_MTRR_REQUESTS 8 /** * A request for a memory region to be set up in a particular way. These * requests are processed before board_init_r() is called. They are generally * optional and can be ignored with some performance impact. */ struct mtrr_request { int type; /* MTRR_TYPE_... */ uint64_t start; uint64_t size; }; /** * struct mrc_output - holds the MRC data * * @buf: MRC training data to save for the next boot. This is set to point to * the raw data after SDRAM init is complete. Then mrccache_setup() * turns it into a proper cache record with a checksum * @len: Length of @buf * @cache: Resulting cache record */ struct mrc_output { char *buf; uint len; struct mrc_data_container *cache; }; /* Architecture-specific global data */ struct arch_global_data { u64 gdt[X86_GDT_NUM_ENTRIES] __aligned(16); struct global_data *gd_addr; /* Location of Global Data */ uint8_t x86; /* CPU family */ uint8_t x86_vendor; /* CPU vendor */ uint8_t x86_model; uint8_t x86_mask; uint32_t x86_device; uint64_t tsc_base; /* Initial value returned by rdtsc() */ bool tsc_inited; /* true if tsc is ready for use */ unsigned long clock_rate; /* Clock rate of timer in Hz */ void *new_fdt; /* Relocated FDT */ uint32_t bist; /* Built-in self test value */ enum pei_boot_mode_t pei_boot_mode; const struct pch_gpio_map *gpio_map; /* board GPIO map */ struct memory_info meminfo; /* Memory information */ struct pei_memory_info pei_meminfo; /* PEI memory information */ #ifdef CONFIG_USE_HOB void *hob_list; /* FSP HOB list */ #endif struct mtrr_request mtrr_req[MAX_MTRR_REQUESTS]; int mtrr_req_count; int has_mtrr; /* MRC training data */ struct mrc_output mrc[MRC_TYPE_COUNT]; ulong table; /* coreboot/EFI table address from previous loader */ int turbo_state; /* Current turbo state */ struct irq_routing_table *pirq_routing_table; int dw_i2c_num_cards; /* Used by designware i2c driver */ #ifdef CONFIG_SEABIOS u32 high_table_ptr; u32 high_table_limit; #endif int prev_sleep_state; /* Previous sleep state ACPI_S0/1../5 */ ulong backup_mem; /* Backup memory address for S3 */ #ifdef CONFIG_FSP_VERSION2 struct fsp_header *fsp_s_hdr; /* Pointer to FSP-S header */ #endif void *itss_priv; /* Private ITSS data pointer */ ulong coreboot_table; /* Address of coreboot table */ ulong table_start; /* Start address of x86 tables */ ulong table_end; /* End address of x86 tables */ ulong table_start_high; /* Start address of high x86 tables */ ulong table_end_high; /* End address of high x86 tables */ ulong smbios_start; /* Start address of SMBIOS table */ }; #endif #include <asm-generic/global_data.h> #ifndef __ASSEMBLY__ # if defined(CONFIG_EFI_APP) #define gd global_data_ptr static inline void set_gd(volatile gd_t *gd_ptr) { extern struct global_data *global_data_ptr; global_data_ptr = (void *)gd_ptr; } #define DECLARE_GLOBAL_DATA_PTR extern struct global_data *global_data_ptr # else static inline notrace gd_t *get_fs_gd_ptr(void) { gd_t *gd_ptr; #if CONFIG_IS_ENABLED(X86_64) asm volatile("fs mov 0, %0\n" : "=r" (gd_ptr)); #else asm volatile("fs movl 0, %0\n" : "=r" (gd_ptr)); #endif return gd_ptr; } #define gd get_fs_gd_ptr() #if CONFIG_IS_ENABLED(X86_64) /* * On x86_64, use MSR_FS_BASE to hold the address of gd->arch.gd_addr, mirroring * how 32-bit x86 uses the FS segment descriptor base. This avoids the need for * a writable global_data_ptr variable, which would require the .data section to * be in RAM before relocation. */ static inline void set_gd(volatile gd_t *gd_ptr) { gd_t *p = (gd_t *)gd_ptr; unsigned long addr; p->arch.gd_addr = p; addr = (unsigned long)&p->arch.gd_addr; asm volatile("wrmsr" : : "c" (MSR_FS_BASE), "a" ((unsigned int)addr), "d" ((unsigned int)(addr >> 32)) : "memory"); } #endif #define DECLARE_GLOBAL_DATA_PTR # endif #endif #endif /* __ASM_GBL_DATA_H */ |