ac2071063e
implementations or no implementation on all platforms. Some of these functions might be good ideas, but their semantics were unclear given the lack of implementation, and an unlucky porter could be fooled into trying to implement them or, worse, being baffled when something like platform_trap_enter() failed to be called.
288 lines
7.2 KiB
C
288 lines
7.2 KiB
C
/*-
|
|
* Copyright (c) 2009 Oleksandr Tymoshenko
|
|
* All rights reserved.
|
|
*
|
|
* 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.
|
|
*
|
|
* 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 <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include "opt_ddb.h"
|
|
#include "opt_ar71xx.h"
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/conf.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/cons.h>
|
|
#include <sys/kdb.h>
|
|
#include <sys/reboot.h>
|
|
|
|
#include <vm/vm.h>
|
|
#include <vm/vm_page.h>
|
|
|
|
#include <net/ethernet.h>
|
|
|
|
#include <machine/clock.h>
|
|
#include <machine/cpu.h>
|
|
#include <machine/cpuregs.h>
|
|
#include <machine/hwfunc.h>
|
|
#include <machine/md_var.h>
|
|
#include <machine/trap.h>
|
|
#include <machine/vmparam.h>
|
|
|
|
#include <mips/atheros/ar71xxreg.h>
|
|
|
|
#include <mips/atheros/ar71xx_setup.h>
|
|
#include <mips/atheros/ar71xx_cpudef.h>
|
|
|
|
#include <mips/sentry5/s5reg.h>
|
|
|
|
extern char edata[], end[];
|
|
|
|
uint32_t ar711_base_mac[ETHER_ADDR_LEN];
|
|
/* 4KB static data aread to keep a copy of the bootload env until
|
|
the dynamic kenv is setup */
|
|
char boot1_env[4096];
|
|
|
|
/*
|
|
* We get a string in from Redboot with the all the arguments together,
|
|
* "foo=bar bar=baz". Split them up and save in kenv.
|
|
*/
|
|
static void
|
|
parse_argv(char *str)
|
|
{
|
|
char *n, *v;
|
|
|
|
while ((v = strsep(&str, " ")) != NULL) {
|
|
if (*v == '\0')
|
|
continue;
|
|
if (*v == '-') {
|
|
while (*v != '\0') {
|
|
v++;
|
|
switch (*v) {
|
|
case 'a': boothowto |= RB_ASKNAME; break;
|
|
case 'd': boothowto |= RB_KDB; break;
|
|
case 'g': boothowto |= RB_GDB; break;
|
|
case 's': boothowto |= RB_SINGLE; break;
|
|
case 'v': boothowto |= RB_VERBOSE; break;
|
|
}
|
|
}
|
|
} else {
|
|
n = strsep(&v, "=");
|
|
if (v == NULL)
|
|
setenv(n, "1");
|
|
else
|
|
setenv(n, v);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
platform_cpu_init()
|
|
{
|
|
/* Nothing special */
|
|
}
|
|
|
|
void
|
|
platform_reset(void)
|
|
{
|
|
ar71xx_device_stop(RST_RESET_FULL_CHIP);
|
|
/* Wait for reset */
|
|
while(1)
|
|
;
|
|
}
|
|
|
|
/*
|
|
* Obtain the MAC address via the Redboot environment.
|
|
*/
|
|
static void
|
|
ar71xx_redboot_get_macaddr(void)
|
|
{
|
|
char *var;
|
|
int count = 0;
|
|
|
|
/*
|
|
* "ethaddr" is passed via envp on RedBoot platforms
|
|
* "kmac" is passed via argv on RouterBOOT platforms
|
|
*/
|
|
if ((var = getenv("ethaddr")) != NULL ||
|
|
(var = getenv("kmac")) != NULL) {
|
|
count = sscanf(var, "%x%*c%x%*c%x%*c%x%*c%x%*c%x",
|
|
&ar711_base_mac[0], &ar711_base_mac[1],
|
|
&ar711_base_mac[2], &ar711_base_mac[3],
|
|
&ar711_base_mac[4], &ar711_base_mac[5]);
|
|
if (count < 6)
|
|
memset(ar711_base_mac, 0,
|
|
sizeof(ar711_base_mac));
|
|
freeenv(var);
|
|
}
|
|
}
|
|
|
|
void
|
|
platform_start(__register_t a0 __unused, __register_t a1 __unused,
|
|
__register_t a2 __unused, __register_t a3 __unused)
|
|
{
|
|
uint64_t platform_counter_freq;
|
|
int argc = 0, i;
|
|
char **argv = NULL, **envp = NULL;
|
|
vm_offset_t kernend;
|
|
|
|
/*
|
|
* clear the BSS and SBSS segments, this should be first call in
|
|
* the function
|
|
*/
|
|
kernend = (vm_offset_t)&end;
|
|
memset(&edata, 0, kernend - (vm_offset_t)(&edata));
|
|
|
|
mips_postboot_fixup();
|
|
|
|
/* Initialize pcpu stuff */
|
|
mips_pcpu0_init();
|
|
|
|
/*
|
|
* Until some more sensible abstractions for uboot/redboot
|
|
* environment handling, we have to make this a compile-time
|
|
* hack. The existing code handles the uboot environment
|
|
* very incorrectly so we should just ignore initialising
|
|
* the relevant pointers.
|
|
*/
|
|
#ifndef AR71XX_ENV_UBOOT
|
|
argc = a0;
|
|
argv = (char**)a1;
|
|
envp = (char**)a2;
|
|
#endif
|
|
/*
|
|
* Protect ourselves from garbage in registers
|
|
*/
|
|
if (MIPS_IS_VALID_PTR(envp)) {
|
|
for (i = 0; envp[i]; i += 2) {
|
|
if (strcmp(envp[i], "memsize") == 0)
|
|
realmem = btoc(strtoul(envp[i+1], NULL, 16));
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Just wild guess. RedBoot let us down and didn't reported
|
|
* memory size
|
|
*/
|
|
if (realmem == 0)
|
|
realmem = btoc(32*1024*1024);
|
|
|
|
/*
|
|
* Allow build-time override in case Redboot lies
|
|
* or in other situations (eg where there's u-boot)
|
|
* where there isn't (yet) a convienent method of
|
|
* being told how much RAM is available.
|
|
*
|
|
* This happens on at least the Ubiquiti LS-SR71A
|
|
* board, where redboot says there's 16mb of RAM
|
|
* but in fact there's 32mb.
|
|
*/
|
|
#if defined(AR71XX_REALMEM)
|
|
realmem = btoc(AR71XX_REALMEM);
|
|
#endif
|
|
|
|
/* phys_avail regions are in bytes */
|
|
phys_avail[0] = MIPS_KSEG0_TO_PHYS(kernel_kseg0_end);
|
|
phys_avail[1] = ctob(realmem);
|
|
|
|
dump_avail[0] = phys_avail[0];
|
|
dump_avail[1] = phys_avail[1] - phys_avail[0];
|
|
|
|
physmem = realmem;
|
|
|
|
/*
|
|
* ns8250 uart code uses DELAY so ticker should be inititalized
|
|
* before cninit. And tick_init_params refers to hz, so * init_param1
|
|
* should be called first.
|
|
*/
|
|
init_param1();
|
|
|
|
/* Detect the system type - this is needed for subsequent chipset-specific calls */
|
|
ar71xx_detect_sys_type();
|
|
ar71xx_detect_sys_frequency();
|
|
|
|
platform_counter_freq = ar71xx_cpu_freq();
|
|
mips_timer_init_params(platform_counter_freq, 1);
|
|
cninit();
|
|
init_static_kenv(boot1_env, sizeof(boot1_env));
|
|
|
|
printf("CPU platform: %s\n", ar71xx_get_system_type());
|
|
printf("CPU Frequency=%d MHz\n", u_ar71xx_cpu_freq / 1000000);
|
|
printf("CPU DDR Frequency=%d MHz\n", u_ar71xx_ddr_freq / 1000000);
|
|
printf("CPU AHB Frequency=%d MHz\n", u_ar71xx_ahb_freq / 1000000);
|
|
|
|
printf("platform frequency: %lld\n", platform_counter_freq);
|
|
printf("arguments: \n");
|
|
printf(" a0 = %08x\n", a0);
|
|
printf(" a1 = %08x\n", a1);
|
|
printf(" a2 = %08x\n", a2);
|
|
printf(" a3 = %08x\n", a3);
|
|
|
|
/*
|
|
* XXX this code is very redboot specific.
|
|
*/
|
|
printf("Cmd line:");
|
|
if (MIPS_IS_VALID_PTR(argv)) {
|
|
for (i = 0; i < argc; i++) {
|
|
printf(" %s", argv[i]);
|
|
parse_argv(argv[i]);
|
|
}
|
|
}
|
|
else
|
|
printf ("argv is invalid");
|
|
printf("\n");
|
|
|
|
printf("Environment:\n");
|
|
if (MIPS_IS_VALID_PTR(envp)) {
|
|
for (i = 0; envp[i]; i+=2) {
|
|
printf(" %s = %s\n", envp[i], envp[i+1]);
|
|
setenv(envp[i], envp[i+1]);
|
|
}
|
|
}
|
|
else
|
|
printf ("envp is invalid\n");
|
|
|
|
/* Redboot if_arge MAC address is in the environment */
|
|
ar71xx_redboot_get_macaddr();
|
|
|
|
init_param2(physmem);
|
|
mips_cpu_init();
|
|
pmap_bootstrap();
|
|
mips_proc0_init();
|
|
mutex_init();
|
|
|
|
/*
|
|
* Reset USB devices
|
|
*/
|
|
ar71xx_init_usb_peripheral();
|
|
|
|
kdb_init();
|
|
#ifdef KDB
|
|
if (boothowto & RB_KDB)
|
|
kdb_enter(KDB_WHY_BOOTFLAGS, "Boot flags requested debugger");
|
|
#endif
|
|
}
|