1/* SPDX-License-Identifier: GPL-2.0 */ 2/* 3 * Copyright (c) 2014 Google, Inc 4 * 5 * From Coreboot file cpu/intel/model_206ax/cache_as_ram.inc 6 * 7 * Copyright (C) 2000,2007 Ronald G. Minnich <rminnich@gmail.com> 8 * Copyright (C) 2005 Tyan (written by Yinghai Lu for Tyan) 9 * Copyright (C) 2007-2008 coresystems GmbH 10 * Copyright (C) 2012 Kyösti Mälkki <kyosti.malkki@gmail.com> 11 */ 12 13#include <common.h> 14#include <asm/microcode.h> 15#include <asm/msr-index.h> 16#include <asm/mtrr.h> 17#include <asm/post.h> 18#include <asm/processor.h> 19#include <asm/processor-flags.h> 20 21#define MTRR_PHYS_BASE_MSR(reg) (0x200 + 2 * (reg)) 22#define MTRR_PHYS_MASK_MSR(reg) (0x200 + 2 * (reg) + 1) 23 24#define CACHE_AS_RAM_SIZE CONFIG_DCACHE_RAM_SIZE 25#define CACHE_AS_RAM_BASE CONFIG_DCACHE_RAM_BASE 26 27/* Cache 4GB - MRC_SIZE_KB for MRC */ 28#define CACHE_MRC_BYTES ((CONFIG_CACHE_MRC_SIZE_KB << 10) - 1) 29#define CACHE_MRC_BASE (0xFFFFFFFF - CACHE_MRC_BYTES) 30#define CACHE_MRC_MASK (~CACHE_MRC_BYTES) 31 32#define CPU_PHYSMASK_HI (1 << (CONFIG_CPU_ADDR_BITS - 32) - 1) 33 34#define NOEVICTMOD_MSR 0x2e0 35 36 /* 37 * Note: ebp must not be touched in this code as it holds the BIST 38 * value (built-in self test). We preserve this value until it can 39 * be written to global_data when CAR is ready for use. 40 */ 41.globl car_init 42car_init: 43 post_code(POST_CAR_START) 44 45 /* Send INIT IPI to all excluding ourself */ 46 movl $0x000C4500, %eax 47 movl $0xFEE00300, %esi 48 movl %eax, (%esi) 49 50 /* TODO: Load microcode later - the 'no eviction' mode breaks this */ 51 movl $MSR_IA32_UCODE_WRITE, %ecx 52 xorl %edx, %edx 53 movl $_dt_ucode_base_size, %eax 54 movl (%eax), %eax 55 addl $UCODE_HEADER_LEN, %eax 56 wrmsr 57 58 post_code(POST_CAR_SIPI) 59 /* Zero out all fixed range and variable range MTRRs */ 60 movl $mtrr_table, %esi 61 movl $((mtrr_table_end - mtrr_table) / 2), %edi 62 xorl %eax, %eax 63 xorl %edx, %edx 64clear_mtrrs: 65 movw (%esi), %bx 66 movzx %bx, %ecx 67 wrmsr 68 add $2, %esi 69 dec %edi 70 jnz clear_mtrrs 71 72 post_code(POST_CAR_MTRR) 73 /* Configure the default memory type to uncacheable */ 74 movl $MTRR_DEF_TYPE_MSR, %ecx 75 rdmsr 76 andl $(~0x00000cff), %eax 77 wrmsr 78 79 post_code(POST_CAR_UNCACHEABLE) 80 /* Set Cache-as-RAM base address */ 81 movl $(MTRR_PHYS_BASE_MSR(0)), %ecx 82 movl $(CACHE_AS_RAM_BASE | MTRR_TYPE_WRBACK), %eax 83 xorl %edx, %edx 84 wrmsr 85 86 post_code(POST_CAR_BASE_ADDRESS) 87 /* Set Cache-as-RAM mask */ 88 movl $(MTRR_PHYS_MASK_MSR(0)), %ecx 89 movl $(~(CACHE_AS_RAM_SIZE - 1) | MTRR_PHYS_MASK_VALID), %eax 90 movl $CPU_PHYSMASK_HI, %edx 91 wrmsr 92 93 post_code(POST_CAR_MASK) 94 95 /* Enable MTRR */ 96 movl $MTRR_DEF_TYPE_MSR, %ecx 97 rdmsr 98 orl $MTRR_DEF_TYPE_EN, %eax 99 wrmsr 100 101 /* Enable cache (CR0.CD = 0, CR0.NW = 0) */ 102 movl %cr0, %eax 103 andl $(~(X86_CR0_CD | X86_CR0_NW)), %eax 104 invd 105 movl %eax, %cr0 106 107 /* enable the 'no eviction' mode */ 108 movl $NOEVICTMOD_MSR, %ecx 109 rdmsr 110 orl $1, %eax 111 andl $~2, %eax 112 wrmsr 113 114 /* Clear the cache memory region. This will also fill up the cache */ 115 movl $CACHE_AS_RAM_BASE, %esi 116 movl %esi, %edi 117 movl $(CACHE_AS_RAM_SIZE / 4), %ecx 118 xorl %eax, %eax 119 rep stosl 120 121 /* enable the 'no eviction run' state */ 122 movl $NOEVICTMOD_MSR, %ecx 123 rdmsr 124 orl $3, %eax 125 wrmsr 126 127 post_code(POST_CAR_FILL) 128 /* Enable Cache-as-RAM mode by disabling cache */ 129 movl %cr0, %eax 130 orl $X86_CR0_CD, %eax 131 movl %eax, %cr0 132 133 /* Enable cache for our code in Flash because we do XIP here */ 134 movl $MTRR_PHYS_BASE_MSR(1), %ecx 135 xorl %edx, %edx 136 movl $car_init_ret, %eax 137 andl $(~(CONFIG_XIP_ROM_SIZE - 1)), %eax 138 orl $MTRR_TYPE_WRPROT, %eax 139 wrmsr 140 141 movl $MTRR_PHYS_MASK_MSR(1), %ecx 142 movl $CPU_PHYSMASK_HI, %edx 143 movl $(~(CONFIG_XIP_ROM_SIZE - 1) | MTRR_PHYS_MASK_VALID), %eax 144 wrmsr 145 146 post_code(POST_CAR_ROM_CACHE) 147#ifdef CONFIG_CACHE_MRC_BIN 148 /* Enable caching for ram init code to run faster */ 149 movl $MTRR_PHYS_BASE_MSR(2), %ecx 150 movl $(CACHE_MRC_BASE | MTRR_TYPE_WRPROT), %eax 151 xorl %edx, %edx 152 wrmsr 153 movl $MTRR_PHYS_MASK_MSR(2), %ecx 154 movl $(CACHE_MRC_MASK | MTRR_PHYS_MASK_VALID), %eax 155 movl $CPU_PHYSMASK_HI, %edx 156 wrmsr 157#endif 158 159 post_code(POST_CAR_MRC_CACHE) 160 /* Enable cache */ 161 movl %cr0, %eax 162 andl $(~(X86_CR0_CD | X86_CR0_NW)), %eax 163 movl %eax, %cr0 164 165 post_code(POST_CAR_CPU_CACHE) 166 167 /* All CPUs need to be in Wait for SIPI state */ 168wait_for_sipi: 169 movl (%esi), %eax 170 bt $12, %eax 171 jc wait_for_sipi 172 173 /* return */ 174 jmp car_init_ret 175 176.globl car_uninit 177car_uninit: 178 /* Disable cache */ 179 movl %cr0, %eax 180 orl $X86_CR0_CD, %eax 181 movl %eax, %cr0 182 183 /* Disable MTRRs */ 184 movl $MTRR_DEF_TYPE_MSR, %ecx 185 rdmsr 186 andl $(~MTRR_DEF_TYPE_EN), %eax 187 wrmsr 188 189 /* Disable the no-eviction run state */ 190 movl $NOEVICTMOD_MSR, %ecx 191 rdmsr 192 andl $~2, %eax 193 wrmsr 194 195 invd 196 197 /* Disable the no-eviction mode */ 198 rdmsr 199 andl $~1, %eax 200 wrmsr 201 202#ifdef CONFIG_CACHE_MRC_BIN 203 /* Clear the MTRR that was used to cache MRC */ 204 xorl %eax, %eax 205 xorl %edx, %edx 206 movl $MTRR_PHYS_BASE_MSR(2), %ecx 207 wrmsr 208 movl $MTRR_PHYS_MASK_MSR(2), %ecx 209 wrmsr 210#endif 211 212 /* Enable MTRRs */ 213 movl $MTRR_DEF_TYPE_MSR, %ecx 214 rdmsr 215 orl $MTRR_DEF_TYPE_EN, %eax 216 wrmsr 217 218 invd 219 220 ret 221 222mtrr_table: 223 /* Fixed MTRRs */ 224 .word 0x250, 0x258, 0x259 225 .word 0x268, 0x269, 0x26A 226 .word 0x26B, 0x26C, 0x26D 227 .word 0x26E, 0x26F 228 /* Variable MTRRs */ 229 .word 0x200, 0x201, 0x202, 0x203 230 .word 0x204, 0x205, 0x206, 0x207 231 .word 0x208, 0x209, 0x20A, 0x20B 232 .word 0x20C, 0x20D, 0x20E, 0x20F 233 .word 0x210, 0x211, 0x212, 0x213 234mtrr_table_end: 235 236 .align 4 237_dt_ucode_base_size: 238 /* These next two fields are filled in by binman */ 239.globl ucode_base 240ucode_base: /* Declared in microcode.h */ 241 .long 0 /* microcode base */ 242.globl ucode_size 243ucode_size: /* Declared in microcode.h */ 244 .long 0 /* microcode size */ 245