918 lines
27 KiB
C
918 lines
27 KiB
C
/* $NetBSD: awin_machdep.c,v 1.36 2014/12/11 23:35:11 jmcneill Exp $ */
|
|
|
|
/*
|
|
* Machine dependent functions for kernel setup for TI OSK5912 board.
|
|
* Based on lubbock_machdep.c which in turn was based on iq80310_machhdep.c
|
|
*
|
|
* Copyright (c) 2002, 2003, 2005 Genetec Corporation. All rights reserved.
|
|
* Written by Hiroyuki Bessho for Genetec Corporation.
|
|
*
|
|
* 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 Genetec Corporation may not be used to endorse or
|
|
* promote products derived from this software without specific prior
|
|
* written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY GENETEC CORPORATION ``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 GENETEC CORPORATION
|
|
* 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.
|
|
*
|
|
* Copyright (c) 2001 Wasabi Systems, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Written by Jason R. Thorpe for Wasabi Systems, Inc.
|
|
*
|
|
* 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. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed for the NetBSD Project by
|
|
* Wasabi Systems, Inc.
|
|
* 4. The name of Wasabi Systems, Inc. may not be used to endorse
|
|
* or promote products derived from this software without specific prior
|
|
* written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``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 WASABI SYSTEMS, INC
|
|
* 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.
|
|
*
|
|
* Copyright (c) 1997,1998 Mark Brinicombe.
|
|
* Copyright (c) 1997,1998 Causality Limited.
|
|
* 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by Mark Brinicombe
|
|
* for the NetBSD Project.
|
|
* 4. The name of the company nor the name of the author may be used to
|
|
* endorse or promote products derived from this software without specific
|
|
* prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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.
|
|
*
|
|
* Copyright (c) 2007 Microsoft
|
|
* 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.
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
* must display the following acknowledgement:
|
|
* This product includes software developed by Microsoft
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 CONTRIBUTERS 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>
|
|
__KERNEL_RCSID(0, "$NetBSD: awin_machdep.c,v 1.36 2014/12/11 23:35:11 jmcneill Exp $");
|
|
|
|
#include "opt_machdep.h"
|
|
#include "opt_ddb.h"
|
|
#include "opt_kgdb.h"
|
|
#include "opt_ipkdb.h"
|
|
#include "opt_md.h"
|
|
#include "opt_com.h"
|
|
#include "opt_allwinner.h"
|
|
#include "opt_arm_debug.h"
|
|
|
|
#include "com.h"
|
|
#include "ukbd.h"
|
|
#include "genfb.h"
|
|
#include "ether.h"
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/cpu.h>
|
|
#include <sys/device.h>
|
|
#include <sys/exec.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/ksyms.h>
|
|
#include <sys/msgbuf.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/reboot.h>
|
|
#include <sys/termios.h>
|
|
#include <sys/gpio.h>
|
|
|
|
#include <uvm/uvm_extern.h>
|
|
|
|
#include <sys/conf.h>
|
|
#include <dev/cons.h>
|
|
#include <dev/md.h>
|
|
|
|
#include <machine/db_machdep.h>
|
|
#include <ddb/db_sym.h>
|
|
#include <ddb/db_extern.h>
|
|
#ifdef KGDB
|
|
#include <sys/kgdb.h>
|
|
#endif
|
|
|
|
#include <machine/bootconfig.h>
|
|
#include <arm/armreg.h>
|
|
#include <arm/undefined.h>
|
|
|
|
#include <arm/arm32/machdep.h>
|
|
#include <arm/mainbus/mainbus.h>
|
|
|
|
#include <dev/ic/ns16550reg.h>
|
|
#include <dev/ic/comreg.h>
|
|
|
|
#include <arm/allwinner/awin_reg.h>
|
|
#include <arm/allwinner/awin_var.h>
|
|
|
|
#include <evbarm/include/autoconf.h>
|
|
#include <evbarm/awin/platform.h>
|
|
|
|
#ifdef AWIN_SYSCONFIG
|
|
#include <evbarm/awin/awin_sysconfig.h>
|
|
#endif
|
|
|
|
#include <dev/i2c/i2cvar.h>
|
|
#include <dev/i2c/ddcreg.h>
|
|
|
|
#include <dev/usb/ukbdvar.h>
|
|
#include <net/if_ether.h>
|
|
|
|
BootConfig bootconfig; /* Boot config storage */
|
|
static char bootargs[MAX_BOOT_STRING];
|
|
char *boot_args = NULL;
|
|
char *boot_file = NULL;
|
|
|
|
#if AWIN_board == AWIN_cubieboard
|
|
bool cubietruck_p;
|
|
#elif AWIN_board == AWIN_cubietruck
|
|
#define cubietruck_p true
|
|
#else
|
|
#define cubietruck_p false
|
|
#endif
|
|
|
|
#ifdef AWIN_SYSCONFIG
|
|
bool awin_sysconfig_p;
|
|
#else
|
|
#define awin_sysconfig_p false
|
|
#endif
|
|
|
|
/*
|
|
* uboot_args are filled in by awin_start.S and must be in .data
|
|
* and not .bss since .bss is cleared after uboot_args are filled in.
|
|
*/
|
|
uintptr_t uboot_args[4] = { 0 };
|
|
|
|
/* Same things, but for the free (unused by the kernel) memory. */
|
|
|
|
extern char KERNEL_BASE_phys[]; /* physical start of kernel */
|
|
extern char _end[]; /* physical end of kernel */
|
|
|
|
#if NAWIN_FB > 0
|
|
#if NCOM > 0
|
|
int use_fb_console = false;
|
|
#else
|
|
int use_fb_console = true;
|
|
#endif
|
|
#endif
|
|
|
|
/*
|
|
* Macros to translate between physical and virtual for a subset of the
|
|
* kernel address space. *Not* for general use.
|
|
*/
|
|
#define KERNEL_BASE_PHYS ((paddr_t)KERNEL_BASE_phys)
|
|
#define AWIN_CORE_VOFFSET (AWIN_CORE_VBASE - AWIN_CORE_PBASE)
|
|
|
|
/* Prototypes */
|
|
|
|
void consinit(void);
|
|
#ifdef KGDB
|
|
static void kgdb_port_init(void);
|
|
#endif
|
|
|
|
static void awin_device_register(device_t, void *);
|
|
|
|
#ifdef AWIN_SYSCONFIG
|
|
static void awin_gpio_sysconfig(prop_dictionary_t);
|
|
#endif
|
|
|
|
#if NCOM > 0
|
|
#include <dev/ic/comreg.h>
|
|
#include <dev/ic/comvar.h>
|
|
#endif
|
|
|
|
/*
|
|
* Static device mappings. These peripheral registers are mapped at
|
|
* fixed virtual addresses very early in initarm() so that we can use
|
|
* them while booting the kernel, and stay at the same address
|
|
* throughout whole kernel's life time.
|
|
*
|
|
* We use this table twice; once with bootstrap page table, and once
|
|
* with kernel's page table which we build up in initarm().
|
|
*
|
|
* Since we map these registers into the bootstrap page table using
|
|
* pmap_devmap_bootstrap() which calls pmap_map_chunk(), we map
|
|
* registers segment-aligned and segment-rounded in order to avoid
|
|
* using the 2nd page tables.
|
|
*/
|
|
|
|
#define _A(a) ((a) & ~L1_S_OFFSET)
|
|
#define _S(s) (((s) + L1_S_SIZE - 1) & ~(L1_S_SIZE-1))
|
|
|
|
static const struct pmap_devmap devmap[] = {
|
|
{
|
|
/*
|
|
* Map all of core area, this gets us everything and
|
|
* it's only 3MB.
|
|
*/
|
|
.pd_va = _A(AWIN_CORE_VBASE),
|
|
.pd_pa = _A(AWIN_CORE_PBASE),
|
|
.pd_size = _S(AWIN_CORE_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_NOCACHE
|
|
},
|
|
#if defined(ALLWINNER_A80)
|
|
{
|
|
/*
|
|
* A80 SYS_CTRL, HS TIMER, DMA, MSG-BOX, SPINLOCK
|
|
*/
|
|
.pd_va = _A(AWIN_A80_CORE2_VBASE),
|
|
.pd_pa = _A(AWIN_A80_CORE2_PBASE),
|
|
.pd_size = _S(AWIN_A80_CORE2_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_NOCACHE
|
|
},
|
|
{
|
|
/*
|
|
* A80 USB-EHCI0/OHCI0, USB-EHCI1, USB-EHCI2/OHCI2
|
|
*/
|
|
.pd_va = _A(AWIN_A80_USB_VBASE),
|
|
.pd_pa = _A(AWIN_A80_USB_PBASE),
|
|
.pd_size = _S(AWIN_A80_USB_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_NOCACHE
|
|
},
|
|
{
|
|
/*
|
|
* A80 RSB, RPRCM
|
|
*/
|
|
.pd_va = _A(AWIN_A80_RCPUS_VBASE),
|
|
.pd_pa = _A(AWIN_A80_RCPUS_PBASE),
|
|
.pd_size = _S(AWIN_A80_RCPUS_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_NOCACHE
|
|
},
|
|
{
|
|
/*
|
|
* A80 CPUCFG
|
|
*/
|
|
.pd_va = _A(AWIN_A80_RCPUCFG_VBASE),
|
|
.pd_pa = _A(AWIN_A80_RCPUCFG_PBASE),
|
|
.pd_size = _S(AWIN_A80_RCPUCFG_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_NOCACHE
|
|
},
|
|
#endif
|
|
{
|
|
/*
|
|
* Map all 1MB of SRAM area.
|
|
*/
|
|
.pd_va = _A(AWIN_SRAM_VBASE),
|
|
.pd_pa = _A(AWIN_SRAM_PBASE),
|
|
.pd_size = _S(AWIN_SRAM_SIZE),
|
|
.pd_prot = VM_PROT_READ|VM_PROT_WRITE,
|
|
.pd_cache = PTE_CACHE
|
|
},
|
|
{0}
|
|
};
|
|
|
|
#undef _A
|
|
#undef _S
|
|
|
|
#ifdef PMAP_NEED_ALLOC_POOLPAGE
|
|
static struct boot_physmem bp_highgig = {
|
|
.bp_start = AWIN_SDRAM_PBASE / NBPG,
|
|
.bp_pages = (KERNEL_VM_BASE - KERNEL_BASE) / NBPG,
|
|
.bp_freelist = VM_FREELIST_ISADMA,
|
|
.bp_flags = 0,
|
|
};
|
|
#endif
|
|
|
|
/*
|
|
* u_int initarm(...)
|
|
*
|
|
* Initial entry point on startup. This gets called before main() is
|
|
* entered.
|
|
* It should be responsible for setting up everything that must be
|
|
* in place when main is called.
|
|
* This includes
|
|
* Taking a copy of the boot configuration structure.
|
|
* Initialising the physical console so characters can be printed.
|
|
* Setting up page tables for the kernel
|
|
* Relocating the kernel to the bottom of physical memory
|
|
*/
|
|
u_int
|
|
initarm(void *arg)
|
|
{
|
|
pmap_devmap_register(devmap);
|
|
awin_bootstrap(AWIN_CORE_VBASE, CONADDR_VA);
|
|
|
|
/* Heads up ... Setup the CPU / MMU / TLB functions. */
|
|
if (set_cpufuncs())
|
|
panic("cpu not recognized!");
|
|
|
|
/* The console is going to try to map things. Give pmap a devmap. */
|
|
consinit();
|
|
|
|
#ifdef VERBOSE_INIT_ARM
|
|
printf("\nuboot arg = %#"PRIxPTR", %#"PRIxPTR", %#"PRIxPTR", %#"PRIxPTR"\n",
|
|
uboot_args[0], uboot_args[1], uboot_args[2], uboot_args[3]);
|
|
#endif
|
|
|
|
#ifdef KGDB
|
|
kgdb_port_init();
|
|
#endif
|
|
|
|
cpu_reset_address = awin_wdog_reset;
|
|
|
|
#ifdef VERBOSE_INIT_ARM
|
|
/* Talk to the user */
|
|
printf("\nNetBSD/evbarm (" BOARDTYPE ") booting ...\n");
|
|
#endif
|
|
|
|
#ifdef BOOT_ARGS
|
|
char mi_bootargs[] = BOOT_ARGS;
|
|
parse_mi_bootargs(mi_bootargs);
|
|
#endif
|
|
|
|
#ifdef VERBOSE_INIT_ARM
|
|
printf("initarm: Configuring system ...\n");
|
|
|
|
#if defined(CPU_CORTEXA7) || defined(CPU_CORTEXA9) || defined(CPU_CORTEXA15)
|
|
if (!CPU_ID_CORTEX_A8_P(curcpu()->ci_arm_cpuid)) {
|
|
printf("initarm: cbar=%#x\n", armreg_cbar_read());
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
/*
|
|
* Set up the variables that define the availability of physical
|
|
* memory.
|
|
*/
|
|
psize_t ram_size = awin_memprobe();
|
|
|
|
#if AWIN_board == AWIN_cubieboard
|
|
/* the cubietruck has 2GB whereas the cubieboards only has 1GB */
|
|
cubietruck_p = (ram_size == 0x80000000);
|
|
#endif
|
|
|
|
/*
|
|
* If MEMSIZE specified less than what we really have, limit ourselves
|
|
* to that.
|
|
*/
|
|
#ifdef MEMSIZE
|
|
if (ram_size == 0 || ram_size > (unsigned)MEMSIZE * 1024 * 1024)
|
|
ram_size = (unsigned)MEMSIZE * 1024 * 1024;
|
|
#else
|
|
KASSERTMSG(ram_size > 0, "RAM size unknown and MEMSIZE undefined");
|
|
#endif
|
|
|
|
/*
|
|
* Configure DMA tags
|
|
*/
|
|
awin_dma_bootstrap(ram_size);
|
|
|
|
/* Fake bootconfig structure for the benefit of pmap.c. */
|
|
bootconfig.dramblocks = 1;
|
|
bootconfig.dram[0].address = AWIN_SDRAM_PBASE;
|
|
bootconfig.dram[0].pages = ram_size / PAGE_SIZE;
|
|
|
|
#ifdef __HAVE_MM_MD_DIRECT_MAPPED_PHYS
|
|
const bool mapallmem_p = true;
|
|
#ifndef PMAP_NEED_ALLOC_POOLPAGE
|
|
if (ram_size > KERNEL_VM_BASE - KERNEL_BASE) {
|
|
printf("%s: dropping RAM size from %luMB to %uMB\n",
|
|
__func__, (unsigned long) (ram_size >> 20),
|
|
(KERNEL_VM_BASE - KERNEL_BASE) >> 20);
|
|
ram_size = KERNEL_VM_BASE - KERNEL_BASE;
|
|
}
|
|
#endif
|
|
#else
|
|
const bool mapallmem_p = false;
|
|
#endif
|
|
KASSERT((armreg_pfr1_read() & ARM_PFR1_SEC_MASK) != 0);
|
|
|
|
arm32_bootmem_init(bootconfig.dram[0].address, ram_size,
|
|
KERNEL_BASE_PHYS);
|
|
arm32_kernel_vm_init(KERNEL_VM_BASE, ARM_VECTORS_LOW, 0, devmap,
|
|
mapallmem_p);
|
|
|
|
if (mapallmem_p) {
|
|
/*
|
|
* "bootargs" env variable is passed as 4th argument
|
|
* to kernel but it's using the physical address and
|
|
* we to convert that to a virtual address.
|
|
*/
|
|
if (uboot_args[3] - AWIN_SDRAM_PBASE < ram_size) {
|
|
const char * const args = (const char *)
|
|
(uboot_args[3] + KERNEL_BASE_VOFFSET);
|
|
strlcpy(bootargs, args, sizeof(bootargs));
|
|
}
|
|
|
|
}
|
|
|
|
boot_args = bootargs;
|
|
parse_mi_bootargs(boot_args);
|
|
|
|
#ifdef AWIN_SYSCONFIG
|
|
if (mapallmem_p) {
|
|
awin_sysconfig_p = awin_sysconfig_init();
|
|
}
|
|
#endif
|
|
|
|
/* we've a specific device_register routine */
|
|
evbarm_device_register = awin_device_register;
|
|
|
|
#if NAWIN_FB > 0
|
|
char *ptr;
|
|
if (get_bootconf_option(boot_args, "console",
|
|
BOOTOPT_TYPE_STRING, &ptr) && strncmp(ptr, "fb", 2) == 0) {
|
|
use_fb_console = true;
|
|
}
|
|
#endif
|
|
/*
|
|
* If we couldn't map all of memory via TTBR1, limit the memory the
|
|
* kernel can allocate from to be from the highest available 1GB.
|
|
*/
|
|
#ifdef PMAP_NEED_ALLOC_POOLPAGE
|
|
if (atop(ram_size) > bp_highgig.bp_pages) {
|
|
bp_highgig.bp_start += atop(ram_size) - bp_highgig.bp_pages;
|
|
arm_poolpage_vmfreelist = bp_highgig.bp_freelist;
|
|
return initarm_common(KERNEL_VM_BASE, KERNEL_VM_SIZE,
|
|
&bp_highgig, 1);
|
|
}
|
|
#endif
|
|
|
|
return initarm_common(KERNEL_VM_BASE, KERNEL_VM_SIZE, NULL, 0);
|
|
|
|
}
|
|
|
|
#if NCOM > 0
|
|
#ifndef CONADDR
|
|
#define CONADDR (AWIN_CORE_PBASE + AWIN_UART0_OFFSET)
|
|
#endif
|
|
#ifndef CONSPEED
|
|
#define CONSPEED 115200
|
|
#endif
|
|
#ifndef CONMODE
|
|
#define CONMODE ((TTYDEF_CFLAG & ~(CSIZE | CSTOPB | PARENB | HUPCL)) | CS8) /* 8N1 */
|
|
#endif
|
|
|
|
#ifdef ALLWINNER_A80
|
|
__CTASSERT(AWIN_CORE_PBASE + AWIN_A80_UART0_OFFSET <= CONADDR);
|
|
__CTASSERT(CONADDR <= AWIN_CORE_PBASE + AWIN_A80_UART5_OFFSET);
|
|
#else
|
|
__CTASSERT(AWIN_CORE_PBASE + AWIN_UART0_OFFSET <= CONADDR);
|
|
__CTASSERT(CONADDR <= AWIN_CORE_PBASE + AWIN_UART7_OFFSET);
|
|
#endif
|
|
__CTASSERT(CONADDR % AWIN_UART_SIZE == 0);
|
|
static const bus_addr_t conaddr = CONADDR;
|
|
static const int conspeed = CONSPEED;
|
|
static const int conmode = CONMODE;
|
|
#endif
|
|
|
|
void
|
|
consinit(void)
|
|
{
|
|
bus_space_tag_t bst = &awin_a4x_bs_tag;
|
|
#if NCOM > 0
|
|
bus_space_handle_t bh;
|
|
#endif
|
|
static int consinit_called = 0;
|
|
|
|
if (consinit_called != 0)
|
|
return;
|
|
|
|
consinit_called = 1;
|
|
|
|
#if NCOM > 0
|
|
if (bus_space_map(bst, conaddr, AWIN_UART_SIZE, 0, &bh))
|
|
panic("Serial console can not be mapped.");
|
|
|
|
if (comcnattach(bst, conaddr, conspeed, AWIN_UART_FREQ,
|
|
COM_TYPE_NORMAL, conmode))
|
|
panic("Serial console can not be initialized.");
|
|
|
|
bus_space_unmap(bst, bh, AWIN_UART_SIZE);
|
|
#else
|
|
#error only COM console is supported
|
|
|
|
#if NUKBD > 0
|
|
ukbd_cnattach(); /* allow USB keyboard to become console */
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
#ifdef KGDB
|
|
#ifndef KGDB_DEVADDR
|
|
#error Specify the address of the kgdb UART with the KGDB_DEVADDR option.
|
|
#endif
|
|
#ifndef KGDB_DEVRATE
|
|
#define KGDB_DEVRATE 115200
|
|
#endif
|
|
|
|
#ifndef KGDB_DEVMODE
|
|
#define KGDB_DEVMODE ((TTYDEF_CFLAG & ~(CSIZE | CSTOPB | PARENB)) | CS8) /* 8N1 */
|
|
#endif
|
|
|
|
__CTASSERT(KGDB_DEVADDR != CONADDR);
|
|
__CTASSERT(AWIN_CORE_PBASE + AWIN_UART0_OFFSET <= KGDB_DEVADDR);
|
|
__CTASSERT(KGDB_DEVADDR <= AWIN_CORE_PBASE + AWIN_UART7_OFFSET);
|
|
__CTASSERT(KGDB_DEVADDR % AWIN_UART_SIZE == 0);
|
|
|
|
static const vaddr_t comkgdbaddr = KGDB_DEVADDR;
|
|
static const int comkgdbspeed = KGDB_DEVRATE;
|
|
static const int comkgdbmode = KGDB_DEVMODE;
|
|
|
|
void
|
|
static kgdb_port_init(void)
|
|
{
|
|
bus_space_tag_t bst = &awin_a4x_bs_tag;
|
|
static bool kgdbsinit_called;
|
|
|
|
if (kgdbsinit_called)
|
|
return;
|
|
|
|
kgdbsinit_called = true;
|
|
|
|
bus_space_handle_t bh;
|
|
if (bus_space_map(bst, comkgdbaddr, AWIN_UART_SIZE, 0, &bh))
|
|
panic("kgdb port can not be mapped.");
|
|
|
|
if (com_kgdb_attach(bst, comkgdbaddr, comkgdbspeed, AWIN_REF_FREQ,
|
|
COM_TYPE_NORMAL, comkgdbmode))
|
|
panic("KGDB uart can not be initialized.");
|
|
|
|
bus_space_unmap(bst, bh, AWIN_UART_SIZE);
|
|
}
|
|
#endif
|
|
|
|
void
|
|
awin_device_register(device_t self, void *aux)
|
|
{
|
|
prop_dictionary_t dict = device_properties(self);
|
|
|
|
if (device_is_a(self, "armperiph")
|
|
&& device_is_a(device_parent(self), "mainbus")) {
|
|
|
|
#if defined(ALLWINNER_A80)
|
|
/* XXX Cubie4 SDK u-boot wrongly sets cbar to 0x01c80000 */
|
|
if (armreg_cbar_read() != AWIN_A80_GIC_BASE) {
|
|
aprint_normal("fixup: cbar %#x -> %#x\n",
|
|
armreg_cbar_read(), AWIN_A80_GIC_BASE);
|
|
prop_dictionary_set_uint32(dict, "cbar",
|
|
AWIN_A80_GIC_BASE);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* XXX KLUDGE ALERT XXX
|
|
* The iot mainbus supplies is completely wrong since it scales
|
|
* addresses by 2. The simpliest remedy is to replace with our
|
|
* bus space used for the armcore regisers (which armperiph uses).
|
|
*/
|
|
struct mainbus_attach_args * const mb = aux;
|
|
mb->mb_iot = &awin_bs_tag;
|
|
return;
|
|
}
|
|
|
|
#if defined(CPU_CORTEXA7) || defined(CPU_CORTEXA15)
|
|
if (device_is_a(self, "armgtmr")) {
|
|
/*
|
|
* The frequency of the generic timer is the reference
|
|
* frequency.
|
|
*/
|
|
prop_dictionary_set_uint32(dict, "frequency", AWIN_REF_FREQ);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
if (device_is_a(self, "awinio")) {
|
|
#if AWIN_board == AWIN_cubieboard || AWIN_board == AWIN_cubietruck
|
|
if (awin_chip_id() == AWIN_CHIP_ID_A20) {
|
|
prop_dictionary_set_bool(dict, "no-awe", true);
|
|
} else {
|
|
prop_dictionary_set_bool(dict, "no-awge", true);
|
|
}
|
|
#elif AWIN_board == AWIN_bpi
|
|
prop_dictionary_set_bool(dict, "no-awe", true);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "awingpio")) {
|
|
#ifdef AWIN_SYSCONFIG
|
|
/*
|
|
* Configure GPIOs using FEX script
|
|
*/
|
|
if (awin_sysconfig_p) {
|
|
awin_gpio_sysconfig(dict);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* These are GPIOs being used for various functions.
|
|
*/
|
|
prop_dictionary_set_cstring(dict, "satapwren",
|
|
(cubietruck_p ? ">PH12" : ">PB8"));
|
|
#if AWIN_board == AWIN_hummingbird_a31
|
|
prop_dictionary_set_cstring(dict, "usb0iddet", "<PA15");
|
|
prop_dictionary_set_cstring(dict, "usb0vbusdet", "<PA16");
|
|
prop_dictionary_set_cstring(dict, "usb0drv", ">PA17");
|
|
prop_dictionary_set_cstring(dict, "usb0restrict", ">PA18");
|
|
prop_dictionary_set_cstring(dict, "usb1drv", ">PH27");
|
|
prop_dictionary_set_cstring(dict, "usb1restrict", ">PH26");
|
|
prop_dictionary_set_cstring(dict, "usb2drv", ">PH24");
|
|
#elif AWIN_board == AWIN_allwinner_a80
|
|
prop_dictionary_set_cstring(dict, "usb1drv", ">PH14");
|
|
prop_dictionary_set_cstring(dict, "usb2drv", ">PH15");
|
|
#else
|
|
if (cubietruck_p) {
|
|
prop_dictionary_set_cstring(dict, "usb0drv", ">PH17");
|
|
} else if (awin_chip_id() == AWIN_CHIP_ID_A20) {
|
|
prop_dictionary_set_cstring(dict, "usb0drv", ">PB9");
|
|
} else {
|
|
prop_dictionary_set_cstring(dict, "usb0drv", ">PB2");
|
|
}
|
|
prop_dictionary_set_cstring(dict, "usb2drv", ">PH3");
|
|
prop_dictionary_set_cstring(dict, "usb0iddet",
|
|
(cubietruck_p ? "<PH19" : "<PH4"));
|
|
prop_dictionary_set_cstring(dict, "usb0vbusdet",
|
|
(cubietruck_p ? "<PH22" : "<PH5"));
|
|
prop_dictionary_set_cstring(dict, "usb1drv", ">PH6");
|
|
#endif
|
|
#if AWIN_board == AWIN_cubietruck
|
|
prop_dictionary_set_cstring(dict, "usb0restrict", ">PH0");
|
|
#endif
|
|
#if AWIN_board == AWIN_allwinner_a80
|
|
prop_dictionary_set_cstring(dict, "status-led1", ">PH06");
|
|
prop_dictionary_set_cstring(dict, "status-led2", ">PH17");
|
|
#else
|
|
prop_dictionary_set_cstring(dict, "status-led1", ">PH21");
|
|
prop_dictionary_set_cstring(dict, "status-led2", ">PH20");
|
|
if (cubietruck_p) {
|
|
prop_dictionary_set_cstring(dict, "status-led3", ">PH11");
|
|
prop_dictionary_set_cstring(dict, "status-led4", ">PH7");
|
|
} else {
|
|
prop_dictionary_set_cstring(dict, "hdd5ven", ">PH17");
|
|
prop_dictionary_set_cstring(dict, "emacpwren", ">PH19");
|
|
}
|
|
#endif
|
|
#if AWIN_board == AWIN_cubieboard || AWIN_board == AWIN_cubietruck
|
|
prop_dictionary_set_cstring(dict, "mmc0detect", "<PH1");
|
|
#elif AWIN_board == AWIN_bpi
|
|
prop_dictionary_set_cstring(dict, "mmc0detect", "<PH10");
|
|
#elif AWIN_board == AWIN_hummingbird_a31
|
|
prop_dictionary_set_cstring(dict, "mmc0detect", "<PH8");
|
|
#elif AWIN_board == AWIN_allwinner_a80
|
|
prop_dictionary_set_cstring(dict, "mmc0detect", "<PH18");
|
|
#endif
|
|
|
|
#if AWIN_board == AWIN_hummingbird_a31
|
|
prop_dictionary_set_cstring(dict, "audiopactrl", ">PH22");
|
|
#else
|
|
prop_dictionary_set_cstring(dict, "audiopactrl", ">PH15");
|
|
#endif
|
|
|
|
#if AWIN_board == AWIN_bpi
|
|
prop_dictionary_set_cstring(dict, "gmacpwren", ">PH23");
|
|
#endif
|
|
|
|
/*
|
|
* These pins have no connections.
|
|
*/
|
|
prop_dictionary_set_uint32(dict, "nc-b", 0x0003d0e8);
|
|
prop_dictionary_set_uint32(dict, "nc-c", 0x00ff0000);
|
|
prop_dictionary_set_uint32(dict, "nc-h", 0x03c53f04);
|
|
prop_dictionary_set_uint32(dict, "nc-i", 0x003fc03f);
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "ehci")) {
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "ahcisata")) {
|
|
/* PIO PB<8> / PIO PH<12> output */
|
|
prop_dictionary_set_cstring(dict, "power-gpio", "satapwren");
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "awinmmc")) {
|
|
struct awinio_attach_args * const aio = aux;
|
|
if (aio->aio_loc.loc_port == 0) {
|
|
prop_dictionary_set_cstring(dict,
|
|
"detect-gpio", "mmc0detect");
|
|
#if !(AWIN_board == AWIN_hummingbird_a31)
|
|
prop_dictionary_set_cstring(dict,
|
|
"led-gpio", "status-led2");
|
|
#endif
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "awinac")) {
|
|
prop_dictionary_set_cstring(dict, "pactrl-gpio", "audiopactrl");
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "awge")) {
|
|
#if NETHER > 0
|
|
/*
|
|
* Get the GMAC MAC address from cmdline.
|
|
*/
|
|
uint8_t enaddr[ETHER_ADDR_LEN];
|
|
char argname[strlen("awge?.mac-address") + 1];
|
|
char *mac_addr;
|
|
snprintf(argname, sizeof(argname), "%s.mac-address",
|
|
device_xname(self));
|
|
|
|
if (get_bootconf_option(boot_args, argname,
|
|
BOOTOPT_TYPE_STRING, &mac_addr)) {
|
|
char mac[strlen("XX:XX:XX:XX:XX:XX") + 1];
|
|
strlcpy(mac, mac_addr, sizeof(mac));
|
|
if (!ether_aton_r(enaddr, sizeof(enaddr), mac)) {
|
|
prop_data_t pd;
|
|
pd = prop_data_create_data(enaddr, sizeof(enaddr));
|
|
KASSERT(pd != NULL);
|
|
prop_dictionary_set(dict, "mac-address", pd);
|
|
prop_object_release(pd);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if AWIN_board == AWIN_cubieboard
|
|
if (awin_chip_id() == AWIN_CHIP_ID_A20) {
|
|
/* Cubieboard2 uses GMAC with a 100Mbit PHY */
|
|
prop_dictionary_set_cstring(dict, "phy-type", "mii");
|
|
}
|
|
#endif
|
|
#if AWIN_BOARD == AWIN_bpi
|
|
prop_dictionary_set_cstring(dict, "phy-power", "gmacpwren");
|
|
prop_dictionary_set_cstring(dict, "phy-type", "rgmii-bpi");
|
|
prop_dictionary_set_uint8(dict, "pinset-func", 5);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "com")) {
|
|
#if NAWIN_FB > 0
|
|
if (use_fb_console)
|
|
prop_dictionary_set_bool(dict, "is_console", false);
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
if (device_is_a(self, "awindebe")) {
|
|
int margin;
|
|
if (get_bootconf_option(boot_args, "fb.margin",
|
|
BOOTOPT_TYPE_INT, &margin) && margin > 0) {
|
|
prop_dictionary_set_uint16(dict, "margin", margin);
|
|
}
|
|
}
|
|
|
|
if (device_is_a(self, "awinhdmi")) {
|
|
char *display_mode;
|
|
if (get_bootconf_option(boot_args, "hdmi.forcemode",
|
|
BOOTOPT_TYPE_STRING, &display_mode)) {
|
|
if (strcasecmp(display_mode, "hdmi") == 0) {
|
|
prop_dictionary_set_cstring(dict,
|
|
"display-mode", "hdmi");
|
|
} else if (strcasecmp(display_mode, "dvi") == 0) {
|
|
prop_dictionary_set_cstring(dict,
|
|
"display-mode", "dvi");
|
|
}
|
|
}
|
|
}
|
|
|
|
#if NGENFB > 0
|
|
if (device_is_a(self, "genfb")) {
|
|
#ifdef DDB
|
|
db_trap_callback = awin_fb_ddb_trap_callback;
|
|
#endif
|
|
char *ptr;
|
|
if (get_bootconf_option(boot_args, "console",
|
|
BOOTOPT_TYPE_STRING, &ptr) && strncmp(ptr, "fb", 2) == 0) {
|
|
prop_dictionary_set_bool(dict, "is_console", true);
|
|
#if NUKBD > 0
|
|
ukbd_cnattach();
|
|
#endif
|
|
} else {
|
|
prop_dictionary_set_bool(dict, "is_console", false);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef AWIN_SYSCONFIG
|
|
static void
|
|
awin_gpio_sysconfig(prop_dictionary_t dict)
|
|
{
|
|
static const struct {
|
|
const char *prop;
|
|
const char *key;
|
|
const char *subkey;
|
|
} gpios[] = {
|
|
{ "satapwren", "sata_para", "sata_power_en" },
|
|
{ "usb0drv", "usbc0", "usb_drv_vbus_gpio" },
|
|
{ "usb0iddet", "usbc0", "usb_id_gpio" },
|
|
{ "usb0vbusdet", "usbc0", "usb_det_vbus_gpio" },
|
|
{ "usb0restrict", "usbc0", "usb_restrict_gpio" },
|
|
{ "usb1drv", "usbc1", "usb_drv_vbus_gpio" },
|
|
{ "usb1iddet", "usbc1", "usb_id_gpio" },
|
|
{ "usb1vbusdet", "usbc1", "usb_det_vbus_gpio" },
|
|
{ "usb1restrict", "usbc1", "usb_restrict_gpio" },
|
|
{ "usb2drv", "usbc2", "usb_drv_vbus_gpio" },
|
|
{ "usb2iddet", "usbc2", "usb_id_gpio" },
|
|
{ "usb2vbusdet", "usbc2", "usb_det_vbus_gpio" },
|
|
{ "usb2restrict", "usbc2", "usb_restrict_gpio" },
|
|
{ "status-led1", "leds_para", "leds_pin_1" },
|
|
{ "status-led2", "leds_para", "leds_pin_2" },
|
|
{ "status-led3", "leds_para", "leds_pin_3" },
|
|
{ "status-led4", "leds_para", "leds_pin_4" },
|
|
{ "mmc0detect", "mmc0_para", "sdc_det" },
|
|
{ "audiopactrl", "audio_para", "audio_pa_ctrl" },
|
|
{ "gmacpwren", "gmac_phy_power", "gmac_phy_power_en" },
|
|
};
|
|
unsigned int n;
|
|
|
|
aprint_normal(":");
|
|
|
|
for (n = 0; n < __arraycount(gpios); n++) {
|
|
const char *cfg = awin_sysconfig_get_gpio(
|
|
gpios[n].key, gpios[n].subkey);
|
|
if (cfg) {
|
|
aprint_normal(" [%s %s]", gpios[n].prop, cfg);
|
|
prop_dictionary_set_cstring(dict, gpios[n].prop, cfg);
|
|
}
|
|
}
|
|
}
|
|
#endif
|