mirror of https://github.com/acidanthera/audk.git
828 lines
24 KiB
ArmAsm
828 lines
24 KiB
ArmAsm
|
#------------------------------------------------------------------------------
|
||
|
#
|
||
|
# Copyright (c) 2013 Intel Corporation.
|
||
|
#
|
||
|
# This program and the accompanying materials
|
||
|
# are licensed and made available under the terms and conditions of the BSD License
|
||
|
# which accompanies this distribution. The full text of the license may be found at
|
||
|
# http://opensource.org/licenses/bsd-license.php
|
||
|
#
|
||
|
# THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
||
|
# WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
||
|
#
|
||
|
# Module Name:
|
||
|
#
|
||
|
# Flat32.S
|
||
|
#
|
||
|
# Abstract:
|
||
|
#
|
||
|
# This is the code that goes from real-mode to protected mode.
|
||
|
# It consumes the reset vector, configures the stack.
|
||
|
#
|
||
|
#
|
||
|
#------------------------------------------------------------------------------
|
||
|
|
||
|
.macro RET32
|
||
|
jmp *%esp
|
||
|
.endm
|
||
|
|
||
|
#
|
||
|
# ROM/SPI/MEMORY Definitions
|
||
|
#
|
||
|
.equ QUARK_DDR3_MEM_BASE_ADDRESS, (0x000000000) # Memory Base Address = 0
|
||
|
.equ QUARK_MAX_DDR3_MEM_SIZE_BYTES, (0x80000000) # DDR3 Memory Size = 2GB
|
||
|
.equ QUARK_ESRAM_MEM_SIZE_BYTES, (0x00080000) # eSRAM Memory Size = 512K
|
||
|
.equ QUARK_STACK_SIZE_BYTES, (0x008000) # Quark stack size = 32K
|
||
|
|
||
|
#
|
||
|
# RTC/CMOS definitions
|
||
|
#
|
||
|
.equ RTC_INDEX, (0x70)
|
||
|
.equ NMI_DISABLE, (0x80) # Bit7=1 disables NMI
|
||
|
.equ NMI_ENABLE, (0x00) # Bit7=0 disables NMI
|
||
|
.equ RTC_DATA, (0x71)
|
||
|
|
||
|
#
|
||
|
# PCI Configuration definitions
|
||
|
#
|
||
|
.equ PCI_CFG, (0x80000000) # PCI configuration access mechanism
|
||
|
.equ PCI_ADDRESS_PORT, (0xCF8)
|
||
|
.equ PCI_DATA_PORT, (0xCFC)
|
||
|
|
||
|
#
|
||
|
# Quark PCI devices
|
||
|
#
|
||
|
.equ HOST_BRIDGE_PFA, (0x0000) # B0:D0:F0 (Host Bridge)
|
||
|
.equ ILB_PFA, (0x00F8) # B0:D31:F0 (Legacy Block)
|
||
|
|
||
|
#
|
||
|
# ILB PCI Config Registers
|
||
|
#
|
||
|
.equ BDE, (0x0D4) # BIOS Decode Enable register
|
||
|
.equ DECODE_ALL_REGIONS_ENABLE, (0xFF000000) # Decode all BIOS decode ranges
|
||
|
|
||
|
#
|
||
|
# iLB Reset Register
|
||
|
#
|
||
|
.equ ILB_RESET_REG, (0x0CF9)
|
||
|
.equ CF9_WARM_RESET, (0x02)
|
||
|
.equ CF9_COLD_RESET, (0x08)
|
||
|
|
||
|
#
|
||
|
# Host Bridge PCI Config Registers
|
||
|
#
|
||
|
.equ MESSAGE_BUS_CONTROL_REG, (0xD0) # Message Bus Control Register
|
||
|
.equ SB_OPCODE_FIELD, (0x18) # Bit location of Opcode field
|
||
|
.equ OPCODE_SIDEBAND_REG_READ, (0x10) # Read opcode
|
||
|
.equ OPCODE_SIDEBAND_REG_WRITE, (0x11) # Write opcode
|
||
|
.equ OPCODE_SIDEBAND_ALT_REG_READ, (0x06) # Alternate Read opcode
|
||
|
.equ OPCODE_SIDEBAND_ALT_REG_WRITE, (0x07) # Alternate Write opcode
|
||
|
.equ OPCODE_WARM_RESET_REQUEST, (0xF4) # Reset Warm
|
||
|
.equ OPCODE_COLD_RESET_REQUEST, (0xF5) # Reset Cold
|
||
|
.equ SB_PORT_FIELD, (0x10) # Bit location of Port ID field
|
||
|
.equ MEMORY_ARBITER_PORT_ID, (0x00)
|
||
|
.equ HOST_BRIDGE_PORT_ID, (0x03)
|
||
|
.equ RMU_PORT_ID, (0x04)
|
||
|
.equ MEMORY_MANAGER_PORT_ID, (0x05)
|
||
|
.equ SOC_UNIT_PORT_ID, (0x31)
|
||
|
.equ SB_ADDR_FIELD, (0x08) # Bit location of Register field
|
||
|
.equ SB_BE_FIELD, (0x04) # Bit location of Byte Enables field
|
||
|
.equ ALL_BYTE_EN, (0x0F) # All Byte Enables
|
||
|
.equ MESSAGE_DATA_REG, (0xD4) # Message Data Register
|
||
|
|
||
|
#
|
||
|
# Memory Arbiter Config Registers
|
||
|
#
|
||
|
.equ AEC_CTRL_OFFSET, (0x00)
|
||
|
|
||
|
#
|
||
|
# Host Bridge Config Registers
|
||
|
#
|
||
|
.equ HMISC2_OFFSET, (0x03) # PCI configuration access mechanism
|
||
|
.equ OR_PM_FIELD, (0x10)
|
||
|
.equ SMI_EN, (0x00080000)
|
||
|
|
||
|
.equ HMBOUND_OFFSET, (0x08)
|
||
|
.equ HMBOUND_ADDRESS, (QUARK_DDR3_MEM_BASE_ADDRESS + QUARK_MAX_DDR3_MEM_SIZE_BYTES + QUARK_ESRAM_MEM_SIZE_BYTES)
|
||
|
.equ HMBOUND_LOCK, (0x01)
|
||
|
.equ HECREG_OFFSET, (0x09)
|
||
|
.equ EC_BASE, (0xE0000000)
|
||
|
.equ EC_ENABLE, (0x01)
|
||
|
.equ HLEGACY_OFFSET, (0x0A)
|
||
|
.equ NMI, (0x00004000)
|
||
|
.equ SMI, (0x00001000)
|
||
|
.equ INTR, (0x00000400)
|
||
|
|
||
|
#
|
||
|
# Memory Manager Config Registers
|
||
|
#
|
||
|
.equ ESRAMPGCTRL_BLOCK_OFFSET, (0x82)
|
||
|
.equ BLOCK_ENABLE_PG, (0x10000000)
|
||
|
.equ BIMRVCTL_OFFSET, (0x19)
|
||
|
.equ ENABLE_IMR_INTERRUPT, (0x80000000)
|
||
|
|
||
|
#
|
||
|
# SOC UNIT Debug Registers
|
||
|
#
|
||
|
.equ CFGSTICKY_W1_OFFSET, (0x50)
|
||
|
.equ FORCE_COLD_RESET, (0x00000001)
|
||
|
.equ CFGSTICKY_RW_OFFSET, (0x51)
|
||
|
.equ RESET_FOR_ESRAM_LOCK, (0x00000020)
|
||
|
.equ RESET_FOR_HMBOUND_LOCK, (0x00000040)
|
||
|
.equ CFGNONSTICKY_W1_OFFSET, (0x52)
|
||
|
.equ FORCE_WARM_RESET, (0x00000001)
|
||
|
|
||
|
#
|
||
|
# CR0 cache control bit definition
|
||
|
#
|
||
|
.equ CR0_CACHE_DISABLE, 0x040000000
|
||
|
.equ CR0_NO_WRITE, 0x020000000
|
||
|
|
||
|
ASM_GLOBAL ASM_PFX(PcdGet32(PcdEsramStage1Base))
|
||
|
|
||
|
|
||
|
#
|
||
|
# Contrary to the name, this file contains 16 bit code as well.
|
||
|
#
|
||
|
.text
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: _ModuleEntryPoint
|
||
|
#
|
||
|
# Input: None
|
||
|
#
|
||
|
# Output: None
|
||
|
#
|
||
|
# Destroys: Assume all registers
|
||
|
#
|
||
|
# Description:
|
||
|
#
|
||
|
# Transition to non-paged flat-model protected mode from a
|
||
|
# hard-coded GDT that provides exactly two descriptors.
|
||
|
# This is a bare bones transition to protected mode only
|
||
|
# used for a while in PEI and possibly DXE.
|
||
|
#
|
||
|
# After enabling protected mode, a far jump is executed to
|
||
|
# transfer to PEI using the newly loaded GDT.
|
||
|
#
|
||
|
# Return: None
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
ASM_GLOBAL ASM_PFX(_ModuleEntryPoint)
|
||
|
ASM_PFX(_ModuleEntryPoint):
|
||
|
|
||
|
#
|
||
|
# Warm Reset (INIT#) check.
|
||
|
#
|
||
|
.byte 0xbe,0x00,0xf0 #movw $0xF000, %si
|
||
|
.byte 0x8e,0xde #movw %si, %ds
|
||
|
.byte 0xbe,0xf0,0xff #movw $0xFFF0, %si
|
||
|
.byte 0x80,0x3c,0xea #cmpb $0xEA, (%si) # Is it warm reset ?
|
||
|
jne NotWarmReset # Jump if not.
|
||
|
.byte 0xb0,0x08 #movb $0x08, %al
|
||
|
.byte 0xba,0xf9,0x0c #movw $0xcf9, %dx
|
||
|
.byte 0xee #outb %al, %dx
|
||
|
.byte 0xb0,0x55 #movb $0x55, %al
|
||
|
.byte 0xe6,0x80 #outb %al, $0x80
|
||
|
jmp .
|
||
|
NotWarmReset:
|
||
|
.byte 0x66,0x8b,0xe8 #movl %eax, %ebp
|
||
|
|
||
|
#
|
||
|
# Load the GDT table in GdtDesc
|
||
|
#
|
||
|
.byte 0x66,0xbe #movl $GdtDesc, %esi
|
||
|
.long GdtDesc
|
||
|
|
||
|
.byte 0x66,0x2e,0x0f,0x01,0x14 #lgdt %cs:(%si)
|
||
|
|
||
|
#
|
||
|
# Transition to 16 bit protected mode
|
||
|
#
|
||
|
.byte 0x0f,0x20,0xc0 #movl %cr0, %eax # Get control register 0
|
||
|
.byte 0x66,0x83,0xc8,0x03 #orl $0x0000003, %eax # Set PE bit (bit #0) & MP bit (bit #1)
|
||
|
.byte 0x0f,0x22,0xc0 #movl %eax, %cr0 # Activate protected mode
|
||
|
|
||
|
#
|
||
|
# Now we're in 16 bit protected mode
|
||
|
# Set up the selectors for 32 bit protected mode entry
|
||
|
#
|
||
|
.byte 0xb8 #movw SYS_DATA_SEL, %ax
|
||
|
.word SYS_DATA_SEL
|
||
|
|
||
|
.byte 0x8e,0xd8 #movw %ax, %ds
|
||
|
.byte 0x8e,0xc0 #movw %ax, %es
|
||
|
.byte 0x8e,0xe0 #movw %ax, %fs
|
||
|
.byte 0x8e,0xe8 #movw %ax, %gs
|
||
|
.byte 0x8e,0xd0 #movw %ax, %ss
|
||
|
|
||
|
#
|
||
|
# Transition to Flat 32 bit protected mode
|
||
|
# The jump to a far pointer causes the transition to 32 bit mode
|
||
|
#
|
||
|
.byte 0x66,0xbe #movl ProtectedModeEntryLinearAddress, %esi
|
||
|
.long ProtectedModeEntryLinearAddress
|
||
|
.byte 0x66,0x2e,0xff,0x2c #jmp %cs:(%esi)
|
||
|
|
||
|
#
|
||
|
# Protected mode portion initializes stack, configures cache, and calls C entry point
|
||
|
#
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: ProtectedModeEntryPoint
|
||
|
#
|
||
|
# Input: Executing in 32 Bit Protected (flat) mode
|
||
|
# cs: 0-4GB
|
||
|
# ds: 0-4GB
|
||
|
# es: 0-4GB
|
||
|
# fs: 0-4GB
|
||
|
# gs: 0-4GB
|
||
|
# ss: 0-4GB
|
||
|
#
|
||
|
# Output: This function never returns
|
||
|
#
|
||
|
# Destroys:
|
||
|
# ecx
|
||
|
# edi
|
||
|
# esi
|
||
|
# esp
|
||
|
#
|
||
|
# Description:
|
||
|
# Perform any essential early platform initilaisation
|
||
|
# Setup a stack
|
||
|
# Transfer control to EDKII code in eSRAM
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
ProtectedModeEntryPoint:
|
||
|
leal L0, %esp
|
||
|
jmp stackless_EarlyPlatformInit
|
||
|
L0:
|
||
|
|
||
|
#
|
||
|
# Set up stack pointer
|
||
|
#
|
||
|
movl ASM_PFX(PcdGet32(PcdEsramStage1Base)), %esp
|
||
|
movl $QUARK_STACK_SIZE_BYTES, %esi
|
||
|
addl %esi, %esp # ESP = top of stack (stack grows downwards).
|
||
|
|
||
|
#
|
||
|
# Store the the BIST value in EBP
|
||
|
#
|
||
|
movl $0, %ebp # No processor BIST on Quark
|
||
|
|
||
|
#
|
||
|
# Push processor count to stack first, then BIST status (AP then BSP)
|
||
|
#
|
||
|
movl $1, %eax
|
||
|
cpuid
|
||
|
shrl $16, %ebx
|
||
|
andl $0x000000FF, %ebx
|
||
|
cmpb $1, %bl
|
||
|
jae PushProcessorCount
|
||
|
|
||
|
#
|
||
|
# Some processors report 0 logical processors. Effectively 0 = 1.
|
||
|
# So we fix up the processor count
|
||
|
#
|
||
|
incl %ebx
|
||
|
|
||
|
PushProcessorCount:
|
||
|
pushl %ebx
|
||
|
|
||
|
#
|
||
|
# We need to implement a long-term solution for BIST capture. For now, we just copy BSP BIST
|
||
|
# for all processor threads
|
||
|
#
|
||
|
xorl %ecx, %ecx
|
||
|
movb %bl, %cl
|
||
|
|
||
|
PushBist:
|
||
|
pushl %ebp
|
||
|
loop PushBist
|
||
|
|
||
|
#
|
||
|
# Pass entry point of the PEI core
|
||
|
#
|
||
|
movl $0xFFFFFFE0, %edi
|
||
|
pushl %ds:(%edi)
|
||
|
|
||
|
#
|
||
|
# Pass BFV into the PEI Core
|
||
|
#
|
||
|
movl $0xFFFFFFFC, %edi
|
||
|
pushl %ds:(%edi)
|
||
|
|
||
|
#
|
||
|
# Pass Temp Ram Base into the PEI Core
|
||
|
#
|
||
|
movl ASM_PFX(PcdGet32(PcdEsramStage1Base)), %eax
|
||
|
addl $(QUARK_ESRAM_MEM_SIZE_BYTES - QUARK_STACK_SIZE_BYTES), %eax
|
||
|
pushl %eax
|
||
|
|
||
|
|
||
|
#
|
||
|
# Pass stack size into the PEI Core
|
||
|
#
|
||
|
pushl $QUARK_STACK_SIZE_BYTES
|
||
|
|
||
|
#
|
||
|
# Pass Control into the PEI Core
|
||
|
#
|
||
|
call SecStartup
|
||
|
|
||
|
#
|
||
|
# PEI Core should never return to here, this is just to capture an invalid return.
|
||
|
#
|
||
|
jmp .
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: stackless_EarlyPlatformInit
|
||
|
#
|
||
|
# Input: esp - Return address
|
||
|
#
|
||
|
# Output: None
|
||
|
#
|
||
|
# Destroys: Assume all registers
|
||
|
#
|
||
|
# Description:
|
||
|
# Any early platform initialisation required
|
||
|
#
|
||
|
# Return:
|
||
|
# None
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
stackless_EarlyPlatformInit:
|
||
|
|
||
|
#
|
||
|
# Save return address
|
||
|
#
|
||
|
movl %esp, %ebp
|
||
|
|
||
|
#
|
||
|
# Ensure cache is disabled.
|
||
|
#
|
||
|
movl %cr0, %eax
|
||
|
orl $(CR0_CACHE_DISABLE + CR0_NO_WRITE), %eax
|
||
|
invd
|
||
|
movl %eax, %cr0
|
||
|
|
||
|
#
|
||
|
# Disable NMI operation
|
||
|
# Good convention suggests you should read back RTC data port after
|
||
|
# accessing the RTC index port.
|
||
|
#
|
||
|
movb $(NMI_DISABLE), %al
|
||
|
movw $(RTC_INDEX), %dx
|
||
|
outb %al, %dx
|
||
|
movw $(RTC_DATA), %dx
|
||
|
inb %dx, %al
|
||
|
|
||
|
#
|
||
|
# Disable SMI (Disables SMI wire, not SMI messages)
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_REG_READ << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HMISC2_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L1, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L1:
|
||
|
andl $(~SMI_EN), %eax
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HMISC2_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L2, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L2:
|
||
|
|
||
|
#
|
||
|
# Before we get going, check SOC Unit Registers to see if we are required to issue a warm/cold reset
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_READ << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGNONSTICKY_W1_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L3, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L3:
|
||
|
andl $(FORCE_WARM_RESET), %eax
|
||
|
jz TestForceColdReset # Zero means bit clear, we're not requested to warm reset so continue as normal
|
||
|
jmp IssueWarmReset
|
||
|
|
||
|
TestForceColdReset:
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_READ << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGNONSTICKY_W1_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L4, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L4:
|
||
|
andl $(FORCE_COLD_RESET), %eax
|
||
|
jz TestHmboundLock # Zero means bit clear, we're not requested to cold reset so continue as normal
|
||
|
jmp IssueColdReset
|
||
|
|
||
|
#
|
||
|
# Before setting HMBOUND, check it's not locked
|
||
|
#
|
||
|
TestHmboundLock:
|
||
|
movl $((OPCODE_SIDEBAND_REG_READ << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HMBOUND_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L5, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L5:
|
||
|
andl $(HMBOUND_LOCK), %eax
|
||
|
jz ConfigHmbound # Zero means bit clear, we have the config we want so continue as normal
|
||
|
#
|
||
|
# Failed to config - store sticky bit debug
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_READ << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGSTICKY_RW_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L6, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L6:
|
||
|
orl $(RESET_FOR_HMBOUND_LOCK), %eax
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_WRITE << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGSTICKY_RW_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L7, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L7:
|
||
|
jmp IssueWarmReset
|
||
|
|
||
|
#
|
||
|
# Set up the HMBOUND register
|
||
|
#
|
||
|
ConfigHmbound:
|
||
|
movl $(HMBOUND_ADDRESS), %eax # Data (Set HMBOUND location)
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HMBOUND_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L8, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L8:
|
||
|
|
||
|
#
|
||
|
# Enable interrupts to Remote Management Unit when a IMR/SMM/HMBOUND violation occurs.
|
||
|
#
|
||
|
movl $(ENABLE_IMR_INTERRUPT), %eax # Data (Set interrupt enable mask)
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (MEMORY_MANAGER_PORT_ID << SB_PORT_FIELD) | (BIMRVCTL_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L9, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L9:
|
||
|
|
||
|
#
|
||
|
# Set eSRAM address
|
||
|
#
|
||
|
movl ASM_PFX(PcdGet32(PcdEsramStage1Base)), %eax # Data (Set eSRAM location)
|
||
|
shr $(0x18), %eax
|
||
|
addl $(BLOCK_ENABLE_PG), %eax
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (MEMORY_MANAGER_PORT_ID << SB_PORT_FIELD) | (ESRAMPGCTRL_BLOCK_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L10, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L10:
|
||
|
|
||
|
#
|
||
|
# Check that we're not blocked from setting the config that we want.
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_REG_READ << SB_OPCODE_FIELD) | (MEMORY_MANAGER_PORT_ID << SB_PORT_FIELD) | (ESRAMPGCTRL_BLOCK_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L11, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L11:
|
||
|
andl $(BLOCK_ENABLE_PG), %eax
|
||
|
jnz ConfigPci # Non-zero means bit set, we have the config we want so continue as normal
|
||
|
#
|
||
|
# Failed to config - store sticky bit debug
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_READ << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGSTICKY_RW_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L12, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L12:
|
||
|
orl $(RESET_FOR_ESRAM_LOCK), %eax # Set the bit we're interested in
|
||
|
movl $((OPCODE_SIDEBAND_ALT_REG_WRITE << SB_OPCODE_FIELD) | (SOC_UNIT_PORT_ID << SB_PORT_FIELD) | (CFGSTICKY_RW_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L13, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L13:
|
||
|
jmp IssueWarmReset
|
||
|
|
||
|
#
|
||
|
# Enable PCIEXBAR
|
||
|
#
|
||
|
ConfigPci:
|
||
|
movl $(EC_BASE + EC_ENABLE), %eax # Data
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (MEMORY_ARBITER_PORT_ID << SB_PORT_FIELD) | (AEC_CTRL_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L14, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L14:
|
||
|
|
||
|
movl $(EC_BASE + EC_ENABLE), %eax # Data
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HECREG_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L15, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L15:
|
||
|
|
||
|
#
|
||
|
# Open up full 8MB SPI decode
|
||
|
#
|
||
|
movl $(PCI_CFG | (ILB_PFA << 8) | BDE), %ebx # PCI Configuration address
|
||
|
movl $(DECODE_ALL_REGIONS_ENABLE), %eax
|
||
|
leal L16, %esp
|
||
|
jmp stackless_PCIConfig_Write
|
||
|
L16:
|
||
|
|
||
|
#
|
||
|
# Enable NMI operation
|
||
|
# Good convention suggests you should read back RTC data port after
|
||
|
# accessing the RTC index port.
|
||
|
#
|
||
|
movb $(NMI_ENABLE), %al
|
||
|
movw $(RTC_INDEX), %dx
|
||
|
outb %al, %dx
|
||
|
movw $(RTC_DATA), %dx
|
||
|
inb %dx, %al
|
||
|
|
||
|
#
|
||
|
# Clear Host Bridge SMI, NMI, INTR fields
|
||
|
#
|
||
|
movl $((OPCODE_SIDEBAND_REG_READ << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HLEGACY_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L21, %esp
|
||
|
jmp stackless_SideBand_Read
|
||
|
L21:
|
||
|
andl $~(NMI + SMI + INTR), %eax # Clear NMI, SMI, INTR fields
|
||
|
movl $((OPCODE_SIDEBAND_REG_WRITE << SB_OPCODE_FIELD) | (HOST_BRIDGE_PORT_ID << SB_PORT_FIELD) | (HLEGACY_OFFSET << SB_ADDR_FIELD)), %ecx
|
||
|
leal L22, %esp
|
||
|
jmp stackless_SideBand_Write
|
||
|
L22:
|
||
|
|
||
|
#
|
||
|
# Restore return address
|
||
|
#
|
||
|
movl %ebp, %esp
|
||
|
RET32
|
||
|
|
||
|
IssueWarmReset:
|
||
|
#
|
||
|
# Issue Warm Reset request to Remote Management Unit via iLB
|
||
|
#
|
||
|
movw $(CF9_WARM_RESET), %ax
|
||
|
movw $(ILB_RESET_REG), %dx
|
||
|
outw %ax, %dx
|
||
|
jmp . # Stay here until we are reset.
|
||
|
|
||
|
IssueColdReset:
|
||
|
#
|
||
|
# Issue Cold Reset request to Remote Management Unit via iLB
|
||
|
#
|
||
|
movw $(CF9_COLD_RESET), %ax
|
||
|
movw $(ILB_RESET_REG), %dx
|
||
|
outw %ax, %dx
|
||
|
jmp . # Stay here until we are reset.
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: stackless_SideBand_Read
|
||
|
#
|
||
|
# Input: esp - return address
|
||
|
# ecx[15:8] - Register offset
|
||
|
# ecx[23:16] - Port ID
|
||
|
# ecx[31:24] - Opcode
|
||
|
#
|
||
|
# Output: eax - Data read
|
||
|
#
|
||
|
# Destroys:
|
||
|
# eax
|
||
|
# ebx
|
||
|
# cl
|
||
|
# esi
|
||
|
#
|
||
|
# Description:
|
||
|
# Perform requested sideband read
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
stackless_SideBand_Read:
|
||
|
|
||
|
movl %esp, %esi # Save the return address
|
||
|
|
||
|
#
|
||
|
# Load the SideBand Packet Register to generate the transaction
|
||
|
#
|
||
|
movl $((PCI_CFG) | (HOST_BRIDGE_PFA << 8) | (MESSAGE_BUS_CONTROL_REG)), %ebx # PCI Configuration address
|
||
|
movb $(ALL_BYTE_EN << SB_BE_FIELD), %cl # Set all Byte Enable bits
|
||
|
xchgl %ecx, %eax
|
||
|
leal L17, %esp
|
||
|
jmp stackless_PCIConfig_Write
|
||
|
L17:
|
||
|
xchgl %ecx, %eax
|
||
|
|
||
|
#
|
||
|
# Read the SideBand Data Register
|
||
|
#
|
||
|
movl $((PCI_CFG) | (HOST_BRIDGE_PFA << 8) | (MESSAGE_DATA_REG)), %ebx # PCI Configuration address
|
||
|
leal L18, %esp
|
||
|
jmp stackless_PCIConfig_Read
|
||
|
L18:
|
||
|
|
||
|
movl %esi, %esp # Restore the return address
|
||
|
RET32
|
||
|
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: stackless_SideBand_Write
|
||
|
#
|
||
|
# Input: esp - return address
|
||
|
# eax - Data
|
||
|
# ecx[15:8] - Register offset
|
||
|
# ecx[23:16] - Port ID
|
||
|
# ecx[31:24] - Opcode
|
||
|
#
|
||
|
# Output: None
|
||
|
#
|
||
|
# Destroys:
|
||
|
# ebx
|
||
|
# cl
|
||
|
# esi
|
||
|
#
|
||
|
# Description:
|
||
|
# Perform requested sideband write
|
||
|
#
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
stackless_SideBand_Write:
|
||
|
|
||
|
movl %esp, %esi # Save the return address
|
||
|
|
||
|
#
|
||
|
# Load the SideBand Data Register with the data
|
||
|
#
|
||
|
movl $((PCI_CFG) | (HOST_BRIDGE_PFA << 8) | (MESSAGE_DATA_REG)), %ebx # PCI Configuration address
|
||
|
leal L19, %esp
|
||
|
jmp stackless_PCIConfig_Write
|
||
|
L19:
|
||
|
|
||
|
#
|
||
|
# Load the SideBand Packet Register to generate the transaction
|
||
|
#
|
||
|
movl $((PCI_CFG) | (HOST_BRIDGE_PFA << 8) | (MESSAGE_BUS_CONTROL_REG)), %ebx # PCI Configuration address
|
||
|
movb $(ALL_BYTE_EN << SB_BE_FIELD), %cl # Set all Byte Enable bits
|
||
|
xchgl %ecx, %eax
|
||
|
leal L20, %esp
|
||
|
jmp stackless_PCIConfig_Write
|
||
|
L20:
|
||
|
xchgl %ecx, %eax
|
||
|
|
||
|
movl %esi, %esp # Restore the return address
|
||
|
RET32
|
||
|
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: stackless_PCIConfig_Write
|
||
|
#
|
||
|
# Input: esp - return address
|
||
|
# eax - Data to write
|
||
|
# ebx - PCI Config Address
|
||
|
#
|
||
|
# Output: None
|
||
|
#
|
||
|
# Destroys:
|
||
|
# dx
|
||
|
#
|
||
|
# Description:
|
||
|
# Perform a DWORD PCI Configuration write
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
stackless_PCIConfig_Write:
|
||
|
|
||
|
#
|
||
|
# Write the PCI Config Address to the address port
|
||
|
#
|
||
|
xchgl %ebx, %eax
|
||
|
movw $(PCI_ADDRESS_PORT), %dx
|
||
|
outl %eax, %dx
|
||
|
xchgl %ebx, %eax
|
||
|
|
||
|
#
|
||
|
# Write the PCI DWORD Data to the data port
|
||
|
#
|
||
|
movw $(PCI_DATA_PORT), %dx
|
||
|
outl %eax, %dx
|
||
|
|
||
|
RET32
|
||
|
|
||
|
|
||
|
#----------------------------------------------------------------------------
|
||
|
#
|
||
|
# Procedure: stackless_PCIConfig_Read
|
||
|
#
|
||
|
# Input: esp - return address
|
||
|
# ebx - PCI Config Address
|
||
|
#
|
||
|
# Output: eax - Data read
|
||
|
#
|
||
|
# Destroys:
|
||
|
# eax
|
||
|
# dx
|
||
|
#
|
||
|
# Description:
|
||
|
# Perform a DWORD PCI Configuration read
|
||
|
#
|
||
|
#----------------------------------------------------------------------------
|
||
|
stackless_PCIConfig_Read:
|
||
|
|
||
|
#
|
||
|
# Write the PCI Config Address to the address port
|
||
|
#
|
||
|
xchgl %ebx, %eax
|
||
|
movw $(PCI_ADDRESS_PORT), %dx
|
||
|
outl %eax, %dx
|
||
|
xchgl %ebx, %eax
|
||
|
|
||
|
#
|
||
|
# Read the PCI DWORD Data from the data port
|
||
|
#
|
||
|
movw $(PCI_DATA_PORT), %dx
|
||
|
inl %dx, %eax
|
||
|
|
||
|
RET32
|
||
|
|
||
|
|
||
|
#
|
||
|
# ROM-based Global-Descriptor Table for the Tiano PEI Phase
|
||
|
#
|
||
|
.align 16
|
||
|
#
|
||
|
# GDT[0]: 000h: Null entry, never used.
|
||
|
#
|
||
|
|
||
|
GDT_BASE:
|
||
|
BootGdtTable:
|
||
|
# null descriptor
|
||
|
.equ NULL_SEL, . - GDT_BASE # Selector [0]
|
||
|
.word 0 # limit 15:0
|
||
|
.word 0 # base 15:0
|
||
|
.byte 0 # base 23:16
|
||
|
.byte 0 # type
|
||
|
.byte 0 # limit 19:16, flags
|
||
|
.byte 0 # base 31:24
|
||
|
|
||
|
# linear data segment descriptor
|
||
|
.equ LINEAR_SEL, . - GDT_BASE # Selector [0x8]
|
||
|
.word 0xFFFF # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0x92 # present, ring 0, data, expand-up, writable
|
||
|
.byte 0xCF # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# linear code segment descriptor
|
||
|
.equ LINEAR_CODE_SEL, . - GDT_BASE # Selector [0x10]
|
||
|
.word 0xFFFF # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0x9A # present, ring 0, data, expand-up, writable
|
||
|
.byte 0xCF # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# system data segment descriptor
|
||
|
.equ SYS_DATA_SEL, . - GDT_BASE # Selector [0x18]
|
||
|
.word 0xFFFF # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0x92 # present, ring 0, data, expand-up, writable
|
||
|
.byte 0xCF # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# system code segment descriptor
|
||
|
.equ SYS_CODE_SEL, . - GDT_BASE
|
||
|
.word 0xFFFF # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0x9A # present, ring 0, data, expand-up, writable
|
||
|
.byte 0xCF # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# spare segment descriptor
|
||
|
.equ SYS16_CODE_SEL, . - GDT_BASE
|
||
|
.word 0xffff # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0x0f
|
||
|
.byte 0x9b # present, ring 0, data, expand-up, writable
|
||
|
.byte 0 # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# spare segment descriptor
|
||
|
.equ SYS16_DATA_SEL, . - GDT_BASE
|
||
|
.word 0xffff # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0x93 # present, ring 0, data, expand-up, not-writable
|
||
|
.byte 0 # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
|
||
|
# spare segment descriptor
|
||
|
.equ SPARE5_SEL, . - GDT_BASE
|
||
|
.word 0 # limit 0xFFFFF
|
||
|
.word 0 # base 0
|
||
|
.byte 0
|
||
|
.byte 0 # present, ring 0, data, expand-up, writable
|
||
|
.byte 0 # page-granular, 32-bit
|
||
|
.byte 0
|
||
|
.equ GDT_SIZE, . - GDT_BASE
|
||
|
|
||
|
#
|
||
|
# GDT Descriptor
|
||
|
#
|
||
|
GdtDesc: # GDT descriptor
|
||
|
.word GDT_SIZE - 1
|
||
|
.long BootGdtTable
|
||
|
|
||
|
ProtectedModeEntryLinearAddress:
|
||
|
ProtectedModeEntryLinearOffset:
|
||
|
.long ProtectedModeEntryPoint
|
||
|
.word LINEAR_CODE_SEL
|