diff options
| author | Allen Martin <amartin@nvidia.com> | 2012-12-19 13:02:36 -0800 | 
|---|---|---|
| committer | Allen Martin <amartin@nvidia.com> | 2012-12-19 13:02:36 -0800 | 
| commit | a098cf41fdb2a6607c675f7fe4f3164617c9367e (patch) | |
| tree | b37acb36f65909e6f74cc537d73efd883a1485a6 /arch/x86/cpu | |
| parent | b8a7c467960ffb4d5a5e1eef5f7783fb6f594542 (diff) | |
| parent | 095728803eedfce850a2f85828f79500cb09979e (diff) | |
Merge remote-tracking branch 'u-boot/master' into u-boot-arm-merged
Conflicts:
	README
	arch/arm/cpu/armv7/exynos/clock.c
	board/samsung/universal_c210/universal.c
	drivers/misc/Makefile
	drivers/power/power_fsl.c
	include/configs/mx35pdk.h
	include/configs/mx53loco.h
	include/configs/seaboard.h
Diffstat (limited to 'arch/x86/cpu')
| -rw-r--r-- | arch/x86/cpu/Makefile | 7 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/Makefile | 7 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/car.S (renamed from arch/x86/cpu/coreboot/coreboot_car.S) | 0 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/config.mk | 23 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/coreboot.c | 140 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/pci.c | 65 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/sdram.c | 72 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/sysinfo.c | 39 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/tables.c | 117 | ||||
| -rw-r--r-- | arch/x86/cpu/coreboot/timestamp.c | 61 | ||||
| -rw-r--r-- | arch/x86/cpu/cpu.c | 49 | ||||
| -rw-r--r-- | arch/x86/cpu/interrupts.c | 99 | ||||
| -rw-r--r-- | arch/x86/cpu/start.S | 77 | ||||
| -rw-r--r-- | arch/x86/cpu/start16.S | 3 | ||||
| -rw-r--r-- | arch/x86/cpu/timer.c | 17 | ||||
| -rw-r--r-- | arch/x86/cpu/u-boot.lds | 3 | 
16 files changed, 624 insertions, 155 deletions
| diff --git a/arch/x86/cpu/Makefile b/arch/x86/cpu/Makefile index 7f1fc188cb7..57324b61749 100644 --- a/arch/x86/cpu/Makefile +++ b/arch/x86/cpu/Makefile @@ -28,12 +28,13 @@ include $(TOPDIR)/config.mk  LIB	= $(obj)lib$(CPU).o -START	= start.o start16.o resetvec.o -COBJS	= interrupts.o cpu.o +START-y	= start.o +RESET_OBJS-$(CONFIG_X86_NO_RESET_VECTOR) += resetvec.o start16.o +COBJS	= interrupts.o cpu.o timer.o  SRCS	:= $(START:.o=.S) $(SOBJS:.o=.S) $(COBJS:.o=.c)  OBJS	:= $(addprefix $(obj),$(SOBJS) $(COBJS)) -START	:= $(addprefix $(obj),$(START)) +START	:= $(addprefix $(obj),$(START-y) $(RESET_OBJS-))  all:	$(obj).depend $(START) $(LIB) diff --git a/arch/x86/cpu/coreboot/Makefile b/arch/x86/cpu/coreboot/Makefile index 13f5f8a2f05..b1d3e959f8f 100644 --- a/arch/x86/cpu/coreboot/Makefile +++ b/arch/x86/cpu/coreboot/Makefile @@ -33,12 +33,13 @@ include $(TOPDIR)/config.mk  LIB	:= $(obj)lib$(SOC).o +SOBJS-$(CONFIG_SYS_COREBOOT) += car.o +COBJS-$(CONFIG_SYS_COREBOOT) += coreboot.o  COBJS-$(CONFIG_SYS_COREBOOT) += tables.o  COBJS-$(CONFIG_SYS_COREBOOT) += ipchecksum.o  COBJS-$(CONFIG_SYS_COREBOOT) += sdram.o -COBJS-$(CONFIG_SYS_COREBOOT) += sysinfo.o - -SOBJS-$(CONFIG_SYS_COREBOOT) += coreboot_car.o +COBJS-$(CONFIG_SYS_COREBOOT) += timestamp.o +COBJS-$(CONFIG_PCI) += pci.o  SRCS	:= $(SOBJS-y:.o=.S) $(COBJS-y:.o=.c)  OBJS	:= $(addprefix $(obj),$(SOBJS-y) $(COBJS-y)) diff --git a/arch/x86/cpu/coreboot/coreboot_car.S b/arch/x86/cpu/coreboot/car.S index 3cc25755fdb..3cc25755fdb 100644 --- a/arch/x86/cpu/coreboot/coreboot_car.S +++ b/arch/x86/cpu/coreboot/car.S diff --git a/arch/x86/cpu/coreboot/config.mk b/arch/x86/cpu/coreboot/config.mk new file mode 100644 index 00000000000..4858fc37284 --- /dev/null +++ b/arch/x86/cpu/coreboot/config.mk @@ -0,0 +1,23 @@ +# +# Copyright (c) 2012 The Chromium OS Authors. +# +# See file CREDITS for list of people who contributed to this +# project. +# +# This program is free software; you can redistribute it and/or +# modify it under the terms of the GNU General Public License as +# published by the Free Software Foundation; either version 2 of +# the License, or (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 59 Temple Place, Suite 330, Boston, +# MA 02111-1307 USA +# + +CONFIG_ARCH_DEVICE_TREE := coreboot diff --git a/arch/x86/cpu/coreboot/coreboot.c b/arch/x86/cpu/coreboot/coreboot.c new file mode 100644 index 00000000000..9c9431e0d9d --- /dev/null +++ b/arch/x86/cpu/coreboot/coreboot.c @@ -0,0 +1,140 @@ +/* + * Copyright (c) 2011 The Chromium OS Authors. + * (C) Copyright 2008 + * Graeme Russ, graeme.russ@gmail.com. + * + * See file CREDITS for list of people who contributed to this + * project. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, + * MA 02111-1307 USA + */ + +#include <common.h> +#include <asm/u-boot-x86.h> +#include <flash.h> +#include <netdev.h> +#include <asm/msr.h> +#include <asm/cache.h> +#include <asm/io.h> +#include <asm/arch-coreboot/tables.h> +#include <asm/arch-coreboot/sysinfo.h> +#include <asm/arch/timestamp.h> + +DECLARE_GLOBAL_DATA_PTR; + +/* + * Miscellaneous platform dependent initializations + */ +int cpu_init_f(void) +{ +	int ret = get_coreboot_info(&lib_sysinfo); +	if (ret != 0) +		printf("Failed to parse coreboot tables.\n"); + +	timestamp_init(); + +	return ret; +} + +int board_early_init_f(void) +{ +	return 0; +} + +int board_early_init_r(void) +{ +	/* CPU Speed to 100MHz */ +	gd->cpu_clk = 100000000; + +	/* Crystal is 33.000MHz */ +	gd->bus_clk = 33000000; + +	return 0; +} + +void show_boot_progress(int val) +{ +#if MIN_PORT80_KCLOCKS_DELAY +	static uint32_t prev_stamp; +	static uint32_t base; + +	/* +	 * Scale the time counter reading to avoid using 64 bit arithmetics. +	 * Can't use get_timer() here becuase it could be not yet +	 * initialized or even implemented. +	 */ +	if (!prev_stamp) { +		base = rdtsc() / 1000; +		prev_stamp = 0; +	} else { +		uint32_t now; + +		do { +			now = rdtsc() / 1000 - base; +		} while (now < (prev_stamp + MIN_PORT80_KCLOCKS_DELAY)); +		prev_stamp = now; +	} +#endif +	outb(val, 0x80); +} + +int last_stage_init(void) +{ +	return 0; +} + +#ifndef CONFIG_SYS_NO_FLASH +ulong board_flash_get_legacy(ulong base, int banknum, flash_info_t *info) +{ +	return 0; +} +#endif + +int board_eth_init(bd_t *bis) +{ +	return pci_eth_init(bis); +} + +#define MTRR_TYPE_WP          5 +#define MTRRcap_MSR           0xfe +#define MTRRphysBase_MSR(reg) (0x200 + 2 * (reg)) +#define MTRRphysMask_MSR(reg) (0x200 + 2 * (reg) + 1) + +int board_final_cleanup(void) +{ +	/* Un-cache the ROM so the kernel has one +	 * more MTRR available. +	 * +	 * Coreboot should have assigned this to the +	 * top available variable MTRR. +	 */ +	u8 top_mtrr = (native_read_msr(MTRRcap_MSR) & 0xff) - 1; +	u8 top_type = native_read_msr(MTRRphysBase_MSR(top_mtrr)) & 0xff; + +	/* Make sure this MTRR is the correct Write-Protected type */ +	if (top_type == MTRR_TYPE_WP) { +		disable_caches(); +		wrmsrl(MTRRphysBase_MSR(top_mtrr), 0); +		wrmsrl(MTRRphysMask_MSR(top_mtrr), 0); +		enable_caches(); +	} + +	/* Issue SMI to Coreboot to lock down ME and registers */ +	printf("Finalizing Coreboot\n"); +	outb(0xcb, 0xb2); + +	return 0; +} diff --git a/arch/x86/cpu/coreboot/pci.c b/arch/x86/cpu/coreboot/pci.c new file mode 100644 index 00000000000..8f941674806 --- /dev/null +++ b/arch/x86/cpu/coreboot/pci.c @@ -0,0 +1,65 @@ +/* + * Copyright (c) 2011 The Chromium OS Authors. + * (C) Copyright 2008,2009 + * Graeme Russ, <graeme.russ@gmail.com> + * + * (C) Copyright 2002 + * Daniel Engström, Omicron Ceti AB, <daniel@omicron.se> + * + * See file CREDITS for list of people who contributed to this + * project. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of + * the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, + * MA 02111-1307 USA + */ + +#include <common.h> +#include <pci.h> +#include <asm/pci.h> + +static struct pci_controller coreboot_hose; + +static void config_pci_bridge(struct pci_controller *hose, pci_dev_t dev, +			      struct pci_config_table *table) +{ +	u8 secondary; +	hose->read_byte(hose, dev, PCI_SECONDARY_BUS, &secondary); +	hose->last_busno = max(hose->last_busno, secondary); +	pci_hose_scan_bus(hose, secondary); +} + +static struct pci_config_table pci_coreboot_config_table[] = { +	/* vendor, device, class, bus, dev, func */ +	{ PCI_ANY_ID, PCI_ANY_ID, PCI_CLASS_BRIDGE_PCI, +		PCI_ANY_ID, PCI_ANY_ID, PCI_ANY_ID, &config_pci_bridge }, +	{} +}; + +void pci_init_board(void) +{ +	coreboot_hose.config_table = pci_coreboot_config_table; +	coreboot_hose.first_busno = 0; +	coreboot_hose.last_busno = 0; + +	pci_set_region(coreboot_hose.regions + 0, 0x0, 0x0, 0xffffffff, +		PCI_REGION_MEM); +	coreboot_hose.region_count = 1; + +	pci_setup_type1(&coreboot_hose); + +	pci_register_hose(&coreboot_hose); + +	pci_hose_scan(&coreboot_hose); +} diff --git a/arch/x86/cpu/coreboot/sdram.c b/arch/x86/cpu/coreboot/sdram.c index f8fdac6319e..76274cb88e3 100644 --- a/arch/x86/cpu/coreboot/sdram.c +++ b/arch/x86/cpu/coreboot/sdram.c @@ -27,8 +27,9 @@  #include <asm/e820.h>  #include <asm/u-boot-x86.h>  #include <asm/global_data.h> -#include <asm/arch-coreboot/sysinfo.h> -#include <asm/arch-coreboot/tables.h> +#include <asm/processor.h> +#include <asm/arch/sysinfo.h> +#include <asm/arch/tables.h>  DECLARE_GLOBAL_DATA_PTR; @@ -51,6 +52,58 @@ unsigned install_e820_map(unsigned max_entries, struct e820entry *entries)  	return num_entries;  } +/* + * This function looks for the highest region of memory lower than 4GB which + * has enough space for U-Boot where U-Boot is aligned on a page boundary. It + * overrides the default implementation found elsewhere which simply picks the + * end of ram, wherever that may be. The location of the stack, the relocation + * address, and how far U-Boot is moved by relocation are set in the global + * data structure. + */ +int calculate_relocation_address(void) +{ +	const uint64_t uboot_size = (uintptr_t)&__bss_end - +			(uintptr_t)&__text_start; +	const uint64_t total_size = uboot_size + CONFIG_SYS_MALLOC_LEN + +		CONFIG_SYS_STACK_SIZE; +	uintptr_t dest_addr = 0; +	int i; + +	for (i = 0; i < lib_sysinfo.n_memranges; i++) { +		struct memrange *memrange = &lib_sysinfo.memrange[i]; +		/* Force U-Boot to relocate to a page aligned address. */ +		uint64_t start = roundup(memrange->base, 1 << 12); +		uint64_t end = memrange->base + memrange->size; + +		/* Ignore non-memory regions. */ +		if (memrange->type != CB_MEM_RAM) +			continue; + +		/* Filter memory over 4GB. */ +		if (end > 0xffffffffULL) +			end = 0x100000000ULL; +		/* Skip this region if it's too small. */ +		if (end - start < total_size) +			continue; + +		/* Use this address if it's the largest so far. */ +		if (end - uboot_size > dest_addr) +			dest_addr = end; +	} + +	/* If no suitable area was found, return an error. */ +	if (!dest_addr) +		return 1; + +	dest_addr -= uboot_size; +	dest_addr &= ~((1 << 12) - 1); +	gd->relocaddr = dest_addr; +	gd->reloc_off = dest_addr - (uintptr_t)&__text_start; +	gd->start_addr_sp = dest_addr - CONFIG_SYS_MALLOC_LEN; + +	return 0; +} +  int dram_init_f(void)  {  	int i; @@ -71,5 +124,20 @@ int dram_init_f(void)  int dram_init(void)  { +	int i, j; + +	if (CONFIG_NR_DRAM_BANKS) { +		for (i = 0, j = 0; i < lib_sysinfo.n_memranges; i++) { +			struct memrange *memrange = &lib_sysinfo.memrange[i]; + +			if (memrange->type == CB_MEM_RAM) { +				gd->bd->bi_dram[j].start = memrange->base; +				gd->bd->bi_dram[j].size = memrange->size; +				j++; +				if (j >= CONFIG_NR_DRAM_BANKS) +					break; +			} +		} +	}  	return 0;  } diff --git a/arch/x86/cpu/coreboot/sysinfo.c b/arch/x86/cpu/coreboot/sysinfo.c deleted file mode 100644 index 9b3e660ddeb..00000000000 --- a/arch/x86/cpu/coreboot/sysinfo.c +++ /dev/null @@ -1,39 +0,0 @@ -/* - * This file is part of the libpayload project. - * - * Copyright (C) 2008 Advanced Micro Devices, Inc. - * Copyright (C) 2009 coresystems GmbH - * - * Redistribution and use in source and binary forms, with or without - * modification, are permitted provided that the following conditions - * are met: - * 1. Redistributions of source code must retain the above copyright - *    notice, this list of conditions and the following disclaimer. - * 2. Redistributions in binary form must reproduce the above copyright - *    notice, this list of conditions and the following disclaimer in the - *    documentation and/or other materials provided with the distribution. - * 3. The name of the author may not be used to endorse or promote products - *    derived from this software without specific prior written permission. - * - * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND - * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE - * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE - * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE - * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL - * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS - * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) - * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT - * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY - * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF - * SUCH DAMAGE. - */ - -#include <asm/arch-coreboot/sysinfo.h> - -/* - * This needs to be in the .data section so that it's copied over during - * relocation. By default it's put in the .bss section which is simply filled - * with zeroes when transitioning from "ROM", which is really RAM, to other - * RAM. - */ -struct sysinfo_t lib_sysinfo __attribute__((section(".data"))); diff --git a/arch/x86/cpu/coreboot/tables.c b/arch/x86/cpu/coreboot/tables.c index 0e3451bb223..b116d595550 100644 --- a/arch/x86/cpu/coreboot/tables.c +++ b/arch/x86/cpu/coreboot/tables.c @@ -28,11 +28,20 @@   * SUCH DAMAGE.   */ +#include <common.h>  #include <asm/arch-coreboot/ipchecksum.h>  #include <asm/arch-coreboot/sysinfo.h>  #include <asm/arch-coreboot/tables.h>  /* + * This needs to be in the .data section so that it's copied over during + * relocation. By default it's put in the .bss section which is simply filled + * with zeroes when transitioning from "ROM", which is really RAM, to other + * RAM. + */ +struct sysinfo_t lib_sysinfo __attribute__((section(".data"))); + +/*   * Some of this is x86 specific, and the rest of it is generic. Right now,   * since we only support x86, we'll avoid trying to make lots of infrastructure   * we don't need. If in the future, we want to use coreboot on some other @@ -72,22 +81,45 @@ static void cb_parse_memory(unsigned char *ptr, struct sysinfo_t *info)  static void cb_parse_serial(unsigned char *ptr, struct sysinfo_t *info)  {  	struct cb_serial *ser = (struct cb_serial *)ptr; -	if (ser->type != CB_SERIAL_TYPE_IO_MAPPED) -		return; -	info->ser_ioport = ser->baseaddr; +	info->serial = ser;  } -static void cb_parse_optiontable(unsigned char *ptr, struct sysinfo_t *info) +static void cb_parse_vbnv(unsigned char *ptr, struct sysinfo_t *info)  { -	info->option_table = (struct cb_cmos_option_table *)ptr; +	struct cb_vbnv *vbnv = (struct cb_vbnv *)ptr; + +	info->vbnv_start = vbnv->vbnv_start; +	info->vbnv_size = vbnv->vbnv_size;  } -static void cb_parse_checksum(unsigned char *ptr, struct sysinfo_t *info) +static void cb_parse_gpios(unsigned char *ptr, struct sysinfo_t *info)  { -	struct cb_cmos_checksum *cmos_cksum = (struct cb_cmos_checksum *)ptr; -	info->cmos_range_start = cmos_cksum->range_start; -	info->cmos_range_end = cmos_cksum->range_end; -	info->cmos_checksum_location = cmos_cksum->location; +	int i; +	struct cb_gpios *gpios = (struct cb_gpios *)ptr; + +	info->num_gpios = (gpios->count < SYSINFO_MAX_GPIOS) ? +				(gpios->count) : SYSINFO_MAX_GPIOS; + +	for (i = 0; i < info->num_gpios; i++) +		info->gpios[i] = gpios->gpios[i]; +} + +static void cb_parse_vdat(unsigned char *ptr, struct sysinfo_t *info) +{ +	struct cb_vdat *vdat = (struct cb_vdat *) ptr; + +	info->vdat_addr = vdat->vdat_addr; +	info->vdat_size = vdat->vdat_size; +} + +static void cb_parse_tstamp(unsigned char *ptr, struct sysinfo_t *info) +{ +	info->tstamp_table = ((struct cb_cbmem_tab *)ptr)->cbmem_tab; +} + +static void cb_parse_cbmem_cons(unsigned char *ptr, struct sysinfo_t *info) +{ +	info->cbmem_cons = ((struct cb_cbmem_tab *)ptr)->cbmem_tab;  }  static void cb_parse_framebuffer(unsigned char *ptr, struct sysinfo_t *info) @@ -95,6 +127,11 @@ static void cb_parse_framebuffer(unsigned char *ptr, struct sysinfo_t *info)  	info->framebuffer = (struct cb_framebuffer *)ptr;  } +static void cb_parse_string(unsigned char *ptr, char **info) +{ +	*info = (char *)((struct cb_string *)ptr)->string; +} +  static int cb_parse_header(void *addr, int len, struct sysinfo_t *info)  {  	struct cb_header *header; @@ -125,6 +162,9 @@ static int cb_parse_header(void *addr, int len, struct sysinfo_t *info)  	/* Now, walk the tables. */  	ptr += header->header_bytes; +	/* Inintialize some fields to sentinel values. */ +	info->vbnv_start = info->vbnv_size = (uint32_t)(-1); +  	for (i = 0; i < header->table_entries; i++) {  		struct cb_record *rec = (struct cb_record *)ptr; @@ -142,11 +182,35 @@ static int cb_parse_header(void *addr, int len, struct sysinfo_t *info)  		case CB_TAG_SERIAL:  			cb_parse_serial(ptr, info);  			break; -		case CB_TAG_CMOS_OPTION_TABLE: -			cb_parse_optiontable(ptr, info); +		case CB_TAG_VERSION: +			cb_parse_string(ptr, &info->version); +			break; +		case CB_TAG_EXTRA_VERSION: +			cb_parse_string(ptr, &info->extra_version); +			break; +		case CB_TAG_BUILD: +			cb_parse_string(ptr, &info->build); +			break; +		case CB_TAG_COMPILE_TIME: +			cb_parse_string(ptr, &info->compile_time); +			break; +		case CB_TAG_COMPILE_BY: +			cb_parse_string(ptr, &info->compile_by); +			break; +		case CB_TAG_COMPILE_HOST: +			cb_parse_string(ptr, &info->compile_host); +			break; +		case CB_TAG_COMPILE_DOMAIN: +			cb_parse_string(ptr, &info->compile_domain); +			break; +		case CB_TAG_COMPILER: +			cb_parse_string(ptr, &info->compiler);  			break; -		case CB_TAG_OPTION_CHECKSUM: -			cb_parse_checksum(ptr, info); +		case CB_TAG_LINKER: +			cb_parse_string(ptr, &info->linker); +			break; +		case CB_TAG_ASSEMBLER: +			cb_parse_string(ptr, &info->assembler);  			break;  		/*  		 * FIXME we should warn on serial if coreboot set up a @@ -155,6 +219,21 @@ static int cb_parse_header(void *addr, int len, struct sysinfo_t *info)  		case CB_TAG_FRAMEBUFFER:  			cb_parse_framebuffer(ptr, info);  			break; +		case CB_TAG_GPIO: +			cb_parse_gpios(ptr, info); +			break; +		case CB_TAG_VDAT: +			cb_parse_vdat(ptr, info); +			break; +		case CB_TAG_TIMESTAMPS: +			cb_parse_tstamp(ptr, info); +			break; +		case CB_TAG_CBMEM_CONSOLE: +			cb_parse_cbmem_cons(ptr, info); +			break; +		case CB_TAG_VBNV: +			cb_parse_vbnv(ptr, info); +			break;  		}  		ptr += rec->size; @@ -166,18 +245,12 @@ static int cb_parse_header(void *addr, int len, struct sysinfo_t *info)  /* == Architecture specific == */  /* This is the x86 specific stuff. */ -/* Assume no translation or that memory is identity mapped. */ -static void *phys_to_virt(unsigned long virt) -{ -	return (void *)(uintptr_t)virt; -} -  int get_coreboot_info(struct sysinfo_t *info)  { -	int ret = cb_parse_header(phys_to_virt(0x00000000), 0x1000, info); +	int ret = cb_parse_header((void *)0x00000000, 0x1000, info);  	if (ret != 1) -		ret = cb_parse_header(phys_to_virt(0x000f0000), 0x1000, info); +		ret = cb_parse_header((void *)0x000f0000, 0x1000, info);  	return (ret == 1) ? 0 : -1;  } diff --git a/arch/x86/cpu/coreboot/timestamp.c b/arch/x86/cpu/coreboot/timestamp.c new file mode 100644 index 00000000000..2ca7a57bce5 --- /dev/null +++ b/arch/x86/cpu/coreboot/timestamp.c @@ -0,0 +1,61 @@ +/* + * This file is part of the coreboot project. + * + * Copyright (C) 2011 The ChromiumOS Authors.  All rights reserved. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; version 2 of the License. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA, 02110-1301 USA + */ + +#include <common.h> +#include <asm/arch/timestamp.h> +#include <asm/arch/sysinfo.h> +#include <linux/compiler.h> + +struct timestamp_entry { +	uint32_t	entry_id; +	uint64_t	entry_stamp; +} __packed; + +struct timestamp_table { +	uint64_t	base_time; +	uint32_t	max_entries; +	uint32_t	num_entries; +	struct timestamp_entry entries[0]; /* Variable number of entries */ +} __packed; + +static struct timestamp_table *ts_table  __attribute__((section(".data"))); + +void timestamp_init(void) +{ +	ts_table = lib_sysinfo.tstamp_table; +	timer_set_tsc_base(ts_table->base_time); +	timestamp_add_now(TS_U_BOOT_INITTED); +} + +void timestamp_add(enum timestamp_id id, uint64_t ts_time) +{ +	struct timestamp_entry *tse; + +	if (!ts_table || (ts_table->num_entries == ts_table->max_entries)) +		return; + +	tse = &ts_table->entries[ts_table->num_entries++]; +	tse->entry_id = id; +	tse->entry_stamp = ts_time - ts_table->base_time; +} + +void timestamp_add_now(enum timestamp_id id) +{ +	timestamp_add(id, rdtsc()); +} diff --git a/arch/x86/cpu/cpu.c b/arch/x86/cpu/cpu.c index e9bb0d770a3..315e87afeb2 100644 --- a/arch/x86/cpu/cpu.c +++ b/arch/x86/cpu/cpu.c @@ -34,6 +34,7 @@  #include <common.h>  #include <command.h> +#include <asm/control_regs.h>  #include <asm/processor.h>  #include <asm/processor-flags.h>  #include <asm/interrupt.h> @@ -90,12 +91,6 @@ static void load_gdt(const u64 *boot_gdt, u16 num_entries)  	asm volatile("lgdtl %0\n" : : "m" (gdt));  } -void init_gd(gd_t *id, u64 *gdt_addr) -{ -	id->gd_addr = (ulong)id; -	setup_gdt(id, gdt_addr); -} -  void setup_gdt(gd_t *id, u64 *gdt_addr)  {  	/* CS: code, read/execute, 4 GB, base 0 */ @@ -121,6 +116,11 @@ void setup_gdt(gd_t *id, u64 *gdt_addr)  	load_fs(X86_GDT_ENTRY_32BIT_FS);  } +int __weak x86_cleanup_before_linux(void) +{ +	return 0; +} +  int x86_cpu_init_f(void)  {  	const u32 em_rst = ~X86_CR0_EM; @@ -148,16 +148,27 @@ int cpu_init_r(void) __attribute__((weak, alias("x86_cpu_init_r")));  void x86_enable_caches(void)  { -	const u32 nw_cd_rst = ~(X86_CR0_NW | X86_CR0_CD); +	unsigned long cr0; -	/* turn on the cache and disable write through */ -	asm("movl	%%cr0, %%eax\n" -	    "andl	%0, %%eax\n" -	    "movl	%%eax, %%cr0\n" -	    "wbinvd\n" : : "i" (nw_cd_rst) : "eax"); +	cr0 = read_cr0(); +	cr0 &= ~(X86_CR0_NW | X86_CR0_CD); +	write_cr0(cr0); +	wbinvd();  }  void enable_caches(void) __attribute__((weak, alias("x86_enable_caches"))); +void x86_disable_caches(void) +{ +	unsigned long cr0; + +	cr0 = read_cr0(); +	cr0 |= X86_CR0_NW | X86_CR0_CD; +	wbinvd(); +	write_cr0(cr0); +	wbinvd(); +} +void disable_caches(void) __attribute__((weak, alias("x86_disable_caches"))); +  int x86_init_cache(void)  {  	enable_caches(); @@ -201,3 +212,17 @@ void __reset_cpu(ulong addr)  	generate_gpf();			/* start the show */  }  void reset_cpu(ulong addr) __attribute__((weak, alias("__reset_cpu"))); + +int dcache_status(void) +{ +	return !(read_cr0() & 0x40000000); +} + +/* Define these functions to allow ehch-hcd to function */ +void flush_dcache_range(unsigned long start, unsigned long stop) +{ +} + +void invalidate_dcache_range(unsigned long start, unsigned long stop) +{ +} diff --git a/arch/x86/cpu/interrupts.c b/arch/x86/cpu/interrupts.c index 43ec3f8b081..dd30a05a9dd 100644 --- a/arch/x86/cpu/interrupts.c +++ b/arch/x86/cpu/interrupts.c @@ -28,10 +28,14 @@   */  #include <common.h> +#include <asm/cache.h> +#include <asm/control_regs.h>  #include <asm/interrupt.h>  #include <asm/io.h>  #include <asm/processor-flags.h>  #include <linux/compiler.h> +#include <asm/msr.h> +#include <asm/u-boot-x86.h>  #define DECLARE_INTERRUPT(x) \  	".globl irq_"#x"\n" \ @@ -41,72 +45,6 @@  	"pushl $"#x"\n" \  	"jmp irq_common_entry\n" -/* - * Volatile isn't enough to prevent the compiler from reordering the - * read/write functions for the control registers and messing everything up. - * A memory clobber would solve the problem, but would prevent reordering of - * all loads stores around it, which can hurt performance. Solution is to - * use a variable and mimic reads and writes to it to enforce serialisation - */ -static unsigned long __force_order; - -static inline unsigned long read_cr0(void) -{ -	unsigned long val; -	asm volatile("mov %%cr0,%0\n\t" : "=r" (val), "=m" (__force_order)); -	return val; -} - -static inline unsigned long read_cr2(void) -{ -	unsigned long val; -	asm volatile("mov %%cr2,%0\n\t" : "=r" (val), "=m" (__force_order)); -	return val; -} - -static inline unsigned long read_cr3(void) -{ -	unsigned long val; -	asm volatile("mov %%cr3,%0\n\t" : "=r" (val), "=m" (__force_order)); -	return val; -} - -static inline unsigned long read_cr4(void) -{ -	unsigned long val; -	asm volatile("mov %%cr4,%0\n\t" : "=r" (val), "=m" (__force_order)); -	return val; -} - -static inline unsigned long get_debugreg(int regno) -{ -	unsigned long val = 0;	/* Damn you, gcc! */ - -	switch (regno) { -	case 0: -		asm("mov %%db0, %0" : "=r" (val)); -		break; -	case 1: -		asm("mov %%db1, %0" : "=r" (val)); -		break; -	case 2: -		asm("mov %%db2, %0" : "=r" (val)); -		break; -	case 3: -		asm("mov %%db3, %0" : "=r" (val)); -		break; -	case 6: -		asm("mov %%db6, %0" : "=r" (val)); -		break; -	case 7: -		asm("mov %%db7, %0" : "=r" (val)); -		break; -	default: -		val = 0; -	} -	return val; -} -  void dump_regs(struct irq_regs *regs)  {  	unsigned long cr0 = 0L, cr2 = 0L, cr3 = 0L, cr4 = 0L; @@ -679,3 +617,32 @@ asm(".globl irq_common_entry\n" \  	DECLARE_INTERRUPT(253) \  	DECLARE_INTERRUPT(254) \  	DECLARE_INTERRUPT(255)); + +#if defined(CONFIG_INTEL_CORE_ARCH) +/* + * Get the number of CPU time counter ticks since it was read first time after + * restart. This yields a free running counter guaranteed to take almost 6 + * years to wrap around even at 100GHz clock rate. + */ +u64 get_ticks(void) +{ +	static u64 tick_base; +	u64 now_tick = rdtsc(); + +	if (!tick_base) +		tick_base = now_tick; + +	return now_tick - tick_base; +} + +#define PLATFORM_INFO_MSR 0xce + +unsigned long get_tbclk(void) +{ +	u32 ratio; +	u64 platform_info = native_read_msr(PLATFORM_INFO_MSR); + +	ratio = (platform_info >> 8) & 0xff; +	return 100 * 1000 * 1000 * ratio; /* 100MHz times Max Non Turbo ratio */ +} +#endif diff --git a/arch/x86/cpu/start.S b/arch/x86/cpu/start.S index ee0dabe4bcd..e960e21f6e4 100644 --- a/arch/x86/cpu/start.S +++ b/arch/x86/cpu/start.S @@ -55,8 +55,16 @@ _x86boot_start:  	movl	%eax, %cr0  	wbinvd +	/* Tell 32-bit code it is being entered from an in-RAM copy */ +	movw	$GD_FLG_WARM_BOOT, %bx +	jmp	1f  _start: -	/* This is the 32-bit cold-reset entry point */ +	/* +	 * This is the 32-bit cold-reset entry point. Initialize %bx to 0 +	 * in case we're preceeded by some sort of boot stub. +	 */ +	movw	$GD_FLG_COLD_BOOT, %bx +1:  	/* Load the segement registes to match the gdt loaded in start16.S */  	movl	$(X86_GDT_ENTRY_32BIT_DS * X86_GDT_ENTRY_SIZE), %eax @@ -83,13 +91,33 @@ car_init_ret:  	 * or fully initialised SDRAM - we really don't care which)  	 * starting at CONFIG_SYS_CAR_ADDR to be used as a temporary stack  	 */ -	movl	$CONFIG_SYS_INIT_SP_ADDR, %esp -	/* Initialise the Global Data Pointer */ -	movl	$CONFIG_SYS_INIT_GD_ADDR, %eax -	movl	%eax, %edx -	addl	$GENERATED_GBL_DATA_SIZE, %edx -	call	init_gd; +	/* Stack grows down from top of CAR */ +	movl	$(CONFIG_SYS_CAR_ADDR + CONFIG_SYS_CAR_SIZE), %esp + +	/* Reserve space on stack for global data */ +	subl	$GENERATED_GBL_DATA_SIZE, %esp + +	/* Align global data to 16-byte boundary */ +	andl	$0xfffffff0, %esp + +	/* Setup first parameter to setup_gdt */ +	movl	%esp, %eax + +	/* Reserve space for global descriptor table */ +	subl	$X86_GDT_SIZE, %esp + +	/* Align temporary global descriptor table to 16-byte boundary */ +	andl	$0xfffffff0, %esp + +	/* Set second parameter to setup_gdt */ +	movl	%esp, %edx + +	/* gd->gd_addr = gd (Required to allow gd->xyz to work) */ +	movl	%eax, (%eax) + +	/* Setup global descriptor table so gd->xyz works */ +	call	setup_gdt  	/* Set parameter to board_init_f() to boot flags */  	xorl	%eax, %eax @@ -113,9 +141,42 @@ board_init_f_r_trampoline:  	 * %eax = Address of top of new stack  	 */ -	/* Setup stack in RAM */ +	/* Stack grows down from top of SDRAM */  	movl	%eax, %esp +	/* Reserve space on stack for global data */ +	subl	$GENERATED_GBL_DATA_SIZE, %esp + +	/* Align global data to 16-byte boundary */ +	andl	$0xfffffff0, %esp + +	/* Setup first parameter to memcpy (and setup_gdt) */ +	movl	%esp, %eax + +	/* Setup second parameter to memcpy */ +	fs movl 0, %edx + +	/* Set third parameter to memcpy */ +	movl	$GENERATED_GBL_DATA_SIZE, %ecx + +	/* Copy global data from CAR to SDRAM stack */ +	call	memcpy + +	/* Reserve space for global descriptor table */ +	subl	$X86_GDT_SIZE, %esp + +	/* Align global descriptor table to 16-byte boundary */ +	andl	$0xfffffff0, %esp + +	/* Set second parameter to setup_gdt */ +	movl	%esp, %edx + +	/* gd->gd_addr = gd (Required to allow gd->xyz to work) */ +	movl	%eax, (%eax) + +	/* Setup global descriptor table so gd->xyz works */ +	call	setup_gdt +  	/* Re-enter U-Boot by calling board_init_f_r */  	call	board_init_f_r diff --git a/arch/x86/cpu/start16.S b/arch/x86/cpu/start16.S index cc393ff54fc..603bf1d2d3e 100644 --- a/arch/x86/cpu/start16.S +++ b/arch/x86/cpu/start16.S @@ -37,6 +37,9 @@  .code16  .globl start16  start16: +	/* Set the Cold Boot / Hard Reset flag */ +	movl	$GD_FLG_COLD_BOOT, %ebx +  	/*  	 * First we let the BSP do some early initialization  	 * this code have to map the flash to its final position diff --git a/arch/x86/cpu/timer.c b/arch/x86/cpu/timer.c new file mode 100644 index 00000000000..149109d4f45 --- /dev/null +++ b/arch/x86/cpu/timer.c @@ -0,0 +1,17 @@ +/* + * Copyright (c) 2011 The Chromium OS Authors. All rights reserved. + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + * + * Alternatively, this software may be distributed under the terms of the + * GNU General Public License ("GPL") version 2 as published by the Free + * Software Foundation. + */ + +#include <common.h> + +unsigned long timer_get_us(void) +{ +	printf("timer_get_us used but not implemented.\n"); +	return 0; +} diff --git a/arch/x86/cpu/u-boot.lds b/arch/x86/cpu/u-boot.lds index a1ecefafc64..0c6f0e31d83 100644 --- a/arch/x86/cpu/u-boot.lds +++ b/arch/x86/cpu/u-boot.lds @@ -86,6 +86,8 @@ SECTIONS  	__bios_start = LOADADDR(.bios);  	__bios_size = SIZEOF(.bios); +#ifndef CONFIG_X86_NO_RESET_VECTOR +  	/*  	 * The following expressions place the 16-bit Real-Mode code and  	 * Reset Vector at the end of the Flash ROM @@ -95,4 +97,5 @@ SECTIONS  	. = RESET_VEC_LOC;  	.resetvec : AT (CONFIG_SYS_TEXT_BASE + (CONFIG_SYS_MONITOR_LEN - RESET_SEG_SIZE + RESET_VEC_LOC)) { KEEP(*(.resetvec)); } +#endif  } | 
