2005-01-07 02:29:27 +00:00
|
|
|
/*-
|
2001-06-10 02:39:37 +00:00
|
|
|
* Copyright (C) 1995, 1996 Wolfgang Solfrank.
|
|
|
|
* Copyright (C) 1995, 1996 TooLs GmbH.
|
|
|
|
* 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 TooLs GmbH.
|
|
|
|
* 4. The name of TooLs GmbH may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``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 TOOLS GMBH 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.
|
|
|
|
*/
|
2005-01-07 02:29:27 +00:00
|
|
|
/*-
|
2001-06-10 02:39:37 +00:00
|
|
|
* Copyright (C) 2001 Benno Rice
|
|
|
|
* 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 Benno Rice ``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 TOOLS GMBH 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.
|
|
|
|
* $NetBSD: machdep.c,v 1.74.2.1 2000/11/01 16:13:48 tv Exp $
|
|
|
|
*/
|
|
|
|
|
2003-04-03 21:36:33 +00:00
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
#include "opt_compat.h"
|
2004-07-27 03:41:34 +00:00
|
|
|
#include "opt_ddb.h"
|
2003-07-31 01:31:32 +00:00
|
|
|
#include "opt_kstack_pages.h"
|
2001-06-10 02:39:37 +00:00
|
|
|
#include "opt_msgbuf.h"
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/proc.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <sys/systm.h>
|
2005-02-04 06:58:09 +00:00
|
|
|
#include <sys/bio.h>
|
|
|
|
#include <sys/buf.h>
|
|
|
|
#include <sys/bus.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/cons.h>
|
2005-02-04 05:32:56 +00:00
|
|
|
#include <sys/cpu.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <sys/eventhandler.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/exec.h>
|
2002-08-29 06:17:48 +00:00
|
|
|
#include <sys/imgact.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/kdb.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <sys/kernel.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/ktr.h>
|
|
|
|
#include <sys/linker.h>
|
|
|
|
#include <sys/lock.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <sys/malloc.h>
|
|
|
|
#include <sys/mbuf.h>
|
|
|
|
#include <sys/msgbuf.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/mutex.h>
|
|
|
|
#include <sys/ptrace.h>
|
|
|
|
#include <sys/reboot.h>
|
|
|
|
#include <sys/signalvar.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <sys/sysctl.h>
|
2002-09-19 04:30:43 +00:00
|
|
|
#include <sys/sysent.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <sys/sysproto.h>
|
|
|
|
#include <sys/ucontext.h>
|
|
|
|
#include <sys/uio.h>
|
|
|
|
#include <sys/vmmeter.h>
|
|
|
|
#include <sys/vnode.h>
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <net/netisr.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <vm/vm.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <vm/vm_extern.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <vm/vm_kern.h>
|
|
|
|
#include <vm/vm_page.h>
|
|
|
|
#include <vm/vm_map.h>
|
|
|
|
#include <vm/vm_object.h>
|
|
|
|
#include <vm/vm_pager.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <machine/bat.h>
|
|
|
|
#include <machine/clock.h>
|
2004-02-11 07:27:34 +00:00
|
|
|
#include <machine/cpu.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <machine/elf.h>
|
|
|
|
#include <machine/fpu.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <machine/md_var.h>
|
2002-07-10 12:16:48 +00:00
|
|
|
#include <machine/metadata.h>
|
2004-11-27 06:51:39 +00:00
|
|
|
#include <machine/pcb.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <machine/powerpc.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <machine/reg.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
#include <machine/sigframe.h>
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <machine/trap.h>
|
|
|
|
#include <machine/vmparam.h>
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-12 22:25:09 +00:00
|
|
|
#include <ddb/ddb.h>
|
|
|
|
|
2005-02-06 01:55:08 +00:00
|
|
|
#include <dev/ofw/openfirm.h>
|
|
|
|
|
2004-07-27 03:41:34 +00:00
|
|
|
#ifdef DDB
|
|
|
|
extern vm_offset_t ksym_start, ksym_end;
|
|
|
|
#endif
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
int cold = 1;
|
|
|
|
|
2005-02-28 08:47:51 +00:00
|
|
|
struct pcpu __pcpu[MAXCPU];
|
2002-02-14 01:39:11 +00:00
|
|
|
struct trapframe frame0;
|
|
|
|
|
|
|
|
vm_offset_t kstack0;
|
|
|
|
vm_offset_t kstack0_phys;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
char machine[] = "powerpc";
|
|
|
|
SYSCTL_STRING(_hw, HW_MACHINE, machine, CTLFLAG_RD, machine, 0, "");
|
|
|
|
|
|
|
|
static char model[128];
|
|
|
|
SYSCTL_STRING(_hw, HW_MODEL, model, CTLFLAG_RD, model, 0, "");
|
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
static int cacheline_size = CACHELINESIZE;
|
|
|
|
SYSCTL_INT(_machdep, CPU_CACHELINE, cacheline_size,
|
|
|
|
CTLFLAG_RD, &cacheline_size, 0, "");
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
static void cpu_startup(void *);
|
|
|
|
SYSINIT(cpu, SI_SUB_CPU, SI_ORDER_FIRST, cpu_startup, NULL)
|
|
|
|
|
2002-07-10 12:16:48 +00:00
|
|
|
void powerpc_init(u_int, u_int, u_int, void *);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
int save_ofw_mapping(void);
|
|
|
|
int restore_ofw_mapping(void);
|
|
|
|
|
|
|
|
void install_extint(void (*)(void));
|
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
int setfault(faultbuf); /* defined in locore.S */
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
static int grab_mcontext(struct thread *, mcontext_t *, int);
|
|
|
|
|
2004-01-21 05:18:08 +00:00
|
|
|
void asm_panic(char *);
|
|
|
|
|
2002-08-30 04:04:37 +00:00
|
|
|
long Maxmem = 0;
|
2005-03-01 07:59:24 +00:00
|
|
|
long realmem = 0;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
struct pmap ofw_pmap;
|
|
|
|
extern int ofmsr;
|
|
|
|
|
|
|
|
struct bat battable[16];
|
|
|
|
|
2001-09-04 08:42:35 +00:00
|
|
|
struct kva_md_info kmi;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
powerpc_ofw_shutdown(void *junk, int howto)
|
|
|
|
{
|
|
|
|
if (howto & RB_HALT) {
|
2003-12-09 14:45:39 +00:00
|
|
|
OF_halt();
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
2003-12-09 14:45:39 +00:00
|
|
|
OF_reboot();
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
cpu_startup(void *dummy)
|
|
|
|
{
|
|
|
|
|
2003-02-05 11:16:36 +00:00
|
|
|
/*
|
|
|
|
* Initialise the decrementer-based clock.
|
|
|
|
*/
|
|
|
|
decr_init();
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
/*
|
|
|
|
* Good {morning,afternoon,evening,night}.
|
|
|
|
*/
|
2003-02-05 12:10:46 +00:00
|
|
|
cpu_setup(PCPU_GET(cpuid));
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
/* startrtclock(); */
|
|
|
|
#ifdef PERFMON
|
|
|
|
perfmon_init();
|
|
|
|
#endif
|
2005-03-07 01:52:24 +00:00
|
|
|
printf("real memory = %ld (%ld MB)\n", ptoa(physmem),
|
|
|
|
ptoa(physmem) / 1048576);
|
|
|
|
realmem = physmem;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Display any holes after the first chunk of extended memory.
|
|
|
|
*/
|
|
|
|
if (bootverbose) {
|
|
|
|
int indx;
|
|
|
|
|
|
|
|
printf("Physical memory chunk(s):\n");
|
|
|
|
for (indx = 0; phys_avail[indx + 1] != 0; indx += 2) {
|
|
|
|
int size1 = phys_avail[indx + 1] - phys_avail[indx];
|
|
|
|
|
|
|
|
printf("0x%08x - 0x%08x, %d bytes (%d pages)\n",
|
|
|
|
phys_avail[indx], phys_avail[indx + 1] - 1, size1,
|
|
|
|
size1 / PAGE_SIZE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-09-04 08:42:35 +00:00
|
|
|
vm_ksubmap_init(&kmi);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-11-09 16:19:14 +00:00
|
|
|
printf("avail memory = %ld (%ld MB)\n", ptoa(cnt.v_free_count),
|
|
|
|
ptoa(cnt.v_free_count) / 1048576);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up buffers, so they can be used to read disk labels.
|
|
|
|
*/
|
|
|
|
bufinit();
|
|
|
|
vm_pager_bufferinit();
|
2002-07-09 13:40:37 +00:00
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
EVENTHANDLER_REGISTER(shutdown_final, powerpc_ofw_shutdown, 0,
|
|
|
|
SHUTDOWN_PRI_LAST);
|
|
|
|
|
|
|
|
#ifdef SMP
|
|
|
|
/*
|
|
|
|
* OK, enough kmem_alloc/malloc state should be up, lets get on with it!
|
|
|
|
*/
|
|
|
|
mp_start(); /* fire up the secondaries */
|
|
|
|
mp_announce();
|
|
|
|
#endif /* SMP */
|
|
|
|
}
|
|
|
|
|
|
|
|
extern char kernel_text[], _end[];
|
|
|
|
|
|
|
|
extern void *trapcode, *trapsize;
|
|
|
|
extern void *alitrap, *alisize;
|
|
|
|
extern void *dsitrap, *dsisize;
|
|
|
|
extern void *decrint, *decrsize;
|
2002-07-09 13:40:37 +00:00
|
|
|
extern void *extint, *extsize;
|
2004-07-12 22:25:09 +00:00
|
|
|
extern void *dblow, *dbsize;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-02-14 01:39:11 +00:00
|
|
|
void
|
2002-07-10 12:16:48 +00:00
|
|
|
powerpc_init(u_int startkernel, u_int endkernel, u_int basekernel, void *mdp)
|
2002-02-14 01:39:11 +00:00
|
|
|
{
|
|
|
|
struct pcpu *pc;
|
2002-07-10 12:16:48 +00:00
|
|
|
vm_offset_t end, off;
|
|
|
|
void *kmdp;
|
2005-02-28 07:14:13 +00:00
|
|
|
char *env;
|
2002-07-10 12:16:48 +00:00
|
|
|
|
|
|
|
end = 0;
|
|
|
|
kmdp = NULL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Parse metadata if present and fetch parameters. Must be done
|
|
|
|
* before console is inited so cninit gets the right value of
|
|
|
|
* boothowto.
|
|
|
|
*/
|
|
|
|
if (mdp != NULL) {
|
|
|
|
preload_metadata = mdp;
|
2003-05-01 03:33:28 +00:00
|
|
|
kmdp = preload_search_by_type("elf kernel");
|
2002-07-10 12:16:48 +00:00
|
|
|
if (kmdp != NULL) {
|
|
|
|
boothowto = MD_FETCH(kmdp, MODINFOMD_HOWTO, int);
|
|
|
|
kern_envp = MD_FETCH(kmdp, MODINFOMD_ENVP, char *);
|
|
|
|
end = MD_FETCH(kmdp, MODINFOMD_KERNEND, vm_offset_t);
|
2004-07-27 03:41:34 +00:00
|
|
|
#ifdef DDB
|
|
|
|
ksym_start = MD_FETCH(kmdp, MODINFOMD_SSYM, uintptr_t);
|
|
|
|
ksym_end = MD_FETCH(kmdp, MODINFOMD_ESYM, uintptr_t);
|
|
|
|
#endif
|
2002-07-10 12:16:48 +00:00
|
|
|
}
|
|
|
|
}
|
2002-02-14 01:39:11 +00:00
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
/*
|
|
|
|
* Init params/tunables that can be overridden by the loader
|
|
|
|
*/
|
|
|
|
init_param1();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Start initializing proc0 and thread0.
|
|
|
|
*/
|
2004-09-05 02:09:54 +00:00
|
|
|
proc_linkup(&proc0, &ksegrp0, &thread0);
|
2004-03-02 06:13:09 +00:00
|
|
|
thread0.td_frame = &frame0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up per-cpu data.
|
|
|
|
*/
|
2005-02-28 08:47:51 +00:00
|
|
|
pc = &__pcpu[0];
|
2004-03-02 06:13:09 +00:00
|
|
|
pcpu_init(pc, 0, sizeof(struct pcpu));
|
|
|
|
pc->pc_curthread = &thread0;
|
|
|
|
pc->pc_curpcb = thread0.td_pcb;
|
|
|
|
pc->pc_cpuid = 0;
|
|
|
|
|
|
|
|
__asm __volatile("mtsprg 0, %0" :: "r"(pc));
|
|
|
|
|
|
|
|
mutex_init();
|
|
|
|
|
2002-02-14 01:39:11 +00:00
|
|
|
/*
|
|
|
|
* Initialize the console before printing anything.
|
|
|
|
*/
|
|
|
|
cninit();
|
|
|
|
|
2002-07-10 12:16:48 +00:00
|
|
|
/*
|
|
|
|
* Complain if there is no metadata.
|
|
|
|
*/
|
|
|
|
if (mdp == NULL || kmdp == NULL) {
|
|
|
|
printf("powerpc_init: no loader metadata.\n");
|
|
|
|
}
|
|
|
|
|
2002-07-10 12:21:54 +00:00
|
|
|
kdb_init();
|
2004-07-12 22:25:09 +00:00
|
|
|
|
2002-02-14 01:39:11 +00:00
|
|
|
/*
|
|
|
|
* XXX: Initialize the interrupt tables.
|
2004-07-12 22:25:09 +00:00
|
|
|
* Disable translation in case the vector area
|
|
|
|
* hasn't been mapped (G5)
|
2002-02-14 01:39:11 +00:00
|
|
|
*/
|
2004-07-12 22:25:09 +00:00
|
|
|
mtmsr(mfmsr() & ~(PSL_IR | PSL_DR));
|
2004-07-20 02:22:36 +00:00
|
|
|
isync();
|
2004-02-04 13:00:56 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_RST, (size_t)&trapsize);
|
2003-03-19 08:33:21 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_MCHK, (size_t)&trapsize);
|
2002-07-09 13:40:37 +00:00
|
|
|
bcopy(&dsitrap, (void *)EXC_DSI, (size_t)&dsisize);
|
2004-02-04 13:00:56 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_ISI, (size_t)&trapsize);
|
2002-07-09 13:40:37 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_EXI, (size_t)&trapsize);
|
2002-09-19 04:30:43 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_ALI, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_PGM, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_FPU, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_DECR, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_SC, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_TRC, (size_t)&trapsize);
|
2004-02-04 13:00:56 +00:00
|
|
|
bcopy(&trapcode, (void *)EXC_FPA, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_THRM, (size_t)&trapsize);
|
|
|
|
bcopy(&trapcode, (void *)EXC_BPT, (size_t)&trapsize);
|
2004-07-12 22:25:09 +00:00
|
|
|
#ifdef KDB
|
|
|
|
bcopy(&dblow, (void *)EXC_RST, (size_t)&dbsize);
|
|
|
|
bcopy(&dblow, (void *)EXC_MCHK, (size_t)&dbsize);
|
|
|
|
bcopy(&dblow, (void *)EXC_PGM, (size_t)&dbsize);
|
|
|
|
bcopy(&dblow, (void *)EXC_TRC, (size_t)&dbsize);
|
|
|
|
bcopy(&dblow, (void *)EXC_BPT, (size_t)&dbsize);
|
2004-02-04 13:00:56 +00:00
|
|
|
#endif
|
2003-01-08 12:27:47 +00:00
|
|
|
__syncicache(EXC_RSVD, EXC_LAST - EXC_RSVD);
|
2002-02-14 01:39:11 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
/*
|
|
|
|
* Make sure translation has been enabled
|
|
|
|
*/
|
|
|
|
mtmsr(mfmsr() | PSL_IR|PSL_DR|PSL_ME|PSL_RI);
|
2004-07-20 02:22:36 +00:00
|
|
|
isync();
|
2002-09-19 04:30:43 +00:00
|
|
|
|
2002-02-28 03:15:49 +00:00
|
|
|
/*
|
|
|
|
* Initialise virtual memory.
|
|
|
|
*/
|
|
|
|
pmap_bootstrap(startkernel, endkernel);
|
|
|
|
|
2002-03-07 10:15:17 +00:00
|
|
|
/*
|
2004-03-02 06:13:09 +00:00
|
|
|
* Initialize params/tunables that are derived from memsize
|
2002-03-07 10:15:17 +00:00
|
|
|
*/
|
|
|
|
init_param2(physmem);
|
|
|
|
|
2005-02-28 07:14:13 +00:00
|
|
|
/*
|
|
|
|
* Grab booted kernel's name
|
|
|
|
*/
|
|
|
|
env = getenv("kernelname");
|
|
|
|
if (env != NULL) {
|
|
|
|
strlcpy(kernelname, env, sizeof(kernelname));
|
|
|
|
freeenv(env);
|
|
|
|
}
|
|
|
|
|
2002-02-28 03:15:49 +00:00
|
|
|
/*
|
|
|
|
* Finish setting up thread0.
|
|
|
|
*/
|
|
|
|
thread0.td_kstack = kstack0;
|
|
|
|
thread0.td_pcb = (struct pcb *)
|
|
|
|
(thread0.td_kstack + KSTACK_PAGES * PAGE_SIZE) - 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map and initialise the message buffer.
|
|
|
|
*/
|
|
|
|
for (off = 0; off < round_page(MSGBUF_SIZE); off += PAGE_SIZE)
|
|
|
|
pmap_kenter((vm_offset_t)msgbufp + off, msgbuf_phys + off);
|
|
|
|
msgbufinit(msgbufp, MSGBUF_SIZE);
|
2004-07-12 22:25:09 +00:00
|
|
|
|
|
|
|
#ifdef KDB
|
|
|
|
if (boothowto & RB_KDB)
|
|
|
|
kdb_enter("Boot flags requested debugger");
|
|
|
|
#endif
|
2002-02-14 01:39:11 +00:00
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
void
|
|
|
|
bzero(void *buf, size_t len)
|
|
|
|
{
|
|
|
|
caddr_t p;
|
|
|
|
|
|
|
|
p = buf;
|
|
|
|
|
|
|
|
while (((vm_offset_t) p & (sizeof(u_long) - 1)) && len) {
|
|
|
|
*p++ = 0;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (len >= sizeof(u_long) * 8) {
|
|
|
|
*(u_long*) p = 0;
|
|
|
|
*((u_long*) p + 1) = 0;
|
|
|
|
*((u_long*) p + 2) = 0;
|
|
|
|
*((u_long*) p + 3) = 0;
|
|
|
|
len -= sizeof(u_long) * 8;
|
|
|
|
*((u_long*) p + 4) = 0;
|
|
|
|
*((u_long*) p + 5) = 0;
|
|
|
|
*((u_long*) p + 6) = 0;
|
|
|
|
*((u_long*) p + 7) = 0;
|
|
|
|
p += sizeof(u_long) * 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (len >= sizeof(u_long)) {
|
|
|
|
*(u_long*) p = 0;
|
|
|
|
len -= sizeof(u_long);
|
|
|
|
p += sizeof(u_long);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (len) {
|
|
|
|
*p++ = 0;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2002-09-19 04:30:43 +00:00
|
|
|
sendsig(sig_t catcher, int sig, sigset_t *mask, u_long code)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2002-09-19 04:30:43 +00:00
|
|
|
struct trapframe *tf;
|
|
|
|
struct sigframe *sfp;
|
|
|
|
struct sigacts *psp;
|
|
|
|
struct sigframe sf;
|
|
|
|
struct thread *td;
|
|
|
|
struct proc *p;
|
|
|
|
int oonstack, rndfsize;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
td = curthread;
|
|
|
|
p = td->td_proc;
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
PROC_LOCK_ASSERT(p, MA_OWNED);
|
2002-09-19 04:30:43 +00:00
|
|
|
psp = p->p_sigacts;
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
mtx_assert(&psp->ps_mtx, MA_OWNED);
|
2002-09-19 04:30:43 +00:00
|
|
|
tf = td->td_frame;
|
|
|
|
oonstack = sigonstack(tf->fixreg[1]);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
rndfsize = ((sizeof(sf) + 15) / 16) * 16;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
CTR4(KTR_SIG, "sendsig: td=%p (%s) catcher=%p sig=%d", td, p->p_comm,
|
|
|
|
catcher, sig);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
/*
|
|
|
|
* Save user context
|
|
|
|
*/
|
|
|
|
memset(&sf, 0, sizeof(sf));
|
2004-03-02 06:13:09 +00:00
|
|
|
grab_mcontext(td, &sf.sf_uc.uc_mcontext, 0);
|
2002-09-19 04:30:43 +00:00
|
|
|
sf.sf_uc.uc_sigmask = *mask;
|
2004-01-03 02:02:26 +00:00
|
|
|
sf.sf_uc.uc_stack = td->td_sigstk;
|
|
|
|
sf.sf_uc.uc_stack.ss_flags = (td->td_pflags & TDP_ALTSTACK)
|
2002-09-19 04:30:43 +00:00
|
|
|
? ((oonstack) ? SS_ONSTACK : 0) : SS_DISABLE;
|
|
|
|
|
|
|
|
sf.sf_uc.uc_mcontext.mc_onstack = (oonstack) ? 1 : 0;
|
|
|
|
|
|
|
|
/*
|
2004-03-02 06:13:09 +00:00
|
|
|
* Allocate and validate space for the signal handler context.
|
2002-09-19 04:30:43 +00:00
|
|
|
*/
|
2004-01-03 02:02:26 +00:00
|
|
|
if ((td->td_pflags & TDP_ALTSTACK) != 0 && !oonstack &&
|
2002-09-19 04:30:43 +00:00
|
|
|
SIGISMEMBER(psp->ps_sigonstack, sig)) {
|
2004-01-03 02:02:26 +00:00
|
|
|
sfp = (struct sigframe *)((caddr_t)td->td_sigstk.ss_sp +
|
|
|
|
td->td_sigstk.ss_size - rndfsize);
|
2002-09-19 04:30:43 +00:00
|
|
|
} else {
|
|
|
|
sfp = (struct sigframe *)(tf->fixreg[1] - rndfsize);
|
|
|
|
}
|
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
/*
|
2002-09-19 04:30:43 +00:00
|
|
|
* Translate the signal if appropriate (Linux emu ?)
|
|
|
|
*/
|
|
|
|
if (p->p_sysent->sv_sigtbl && sig <= p->p_sysent->sv_sigsize)
|
2004-03-02 06:13:09 +00:00
|
|
|
sig = p->p_sysent->sv_sigtbl[_SIG_IDX(sig)];
|
2002-09-19 04:30:43 +00:00
|
|
|
|
|
|
|
/*
|
2004-03-02 06:13:09 +00:00
|
|
|
* Save the floating-point state, if necessary, then copy it.
|
2002-09-19 04:30:43 +00:00
|
|
|
*/
|
|
|
|
/* XXX */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up the registers to return to sigcode.
|
|
|
|
*
|
|
|
|
* r1/sp - sigframe ptr
|
|
|
|
* lr - sig function, dispatched to by blrl in trampoline
|
|
|
|
* r3 - sig number
|
|
|
|
* r4 - SIGINFO ? &siginfo : exception code
|
|
|
|
* r5 - user context
|
|
|
|
* srr0 - trampoline function addr
|
|
|
|
*/
|
|
|
|
tf->lr = (register_t)catcher;
|
|
|
|
tf->fixreg[1] = (register_t)sfp;
|
|
|
|
tf->fixreg[FIRSTARG] = sig;
|
|
|
|
tf->fixreg[FIRSTARG+2] = (register_t)&sfp->sf_uc;
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
if (SIGISMEMBER(psp->ps_siginfo, sig)) {
|
2004-03-02 06:13:09 +00:00
|
|
|
/*
|
2002-09-19 04:30:43 +00:00
|
|
|
* Signal handler installed with SA_SIGINFO.
|
|
|
|
*/
|
|
|
|
tf->fixreg[FIRSTARG+1] = (register_t)&sfp->sf_si;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fill siginfo structure.
|
|
|
|
*/
|
|
|
|
sf.sf_si.si_signo = sig;
|
|
|
|
sf.sf_si.si_code = code;
|
|
|
|
sf.sf_si.si_addr = (void *)tf->srr0;
|
|
|
|
} else {
|
|
|
|
/* Old FreeBSD-style arguments. */
|
|
|
|
tf->fixreg[FIRSTARG+1] = code;
|
|
|
|
}
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
mtx_unlock(&psp->ps_mtx);
|
2002-09-19 04:30:43 +00:00
|
|
|
PROC_UNLOCK(p);
|
|
|
|
|
|
|
|
tf->srr0 = (register_t)(PS_STRINGS - *(p->p_sysent->sv_szsigcode));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* copy the frame out to userland.
|
|
|
|
*/
|
|
|
|
if (copyout((caddr_t)&sf, (caddr_t)sfp, sizeof(sf)) != 0) {
|
|
|
|
/*
|
|
|
|
* Process has trashed its stack. Kill it.
|
|
|
|
*/
|
|
|
|
CTR2(KTR_SIG, "sendsig: sigexit td=%p sfp=%p", td, sfp);
|
|
|
|
PROC_LOCK(p);
|
|
|
|
sigexit(td, SIGILL);
|
|
|
|
}
|
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
CTR3(KTR_SIG, "sendsig: return td=%p pc=%#x sp=%#x", td,
|
2002-09-19 04:30:43 +00:00
|
|
|
tf->srr0, tf->fixreg[1]);
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
PROC_LOCK(p);
|
- Merge struct procsig with struct sigacts.
- Move struct sigacts out of the u-area and malloc() it using the
M_SUBPROC malloc bucket.
- Add a small sigacts_*() API for managing sigacts structures: sigacts_alloc(),
sigacts_free(), sigacts_copy(), sigacts_share(), and sigacts_shared().
- Remove the p_sigignore, p_sigacts, and p_sigcatch macros.
- Add a mutex to struct sigacts that protects all the members of the struct.
- Add sigacts locking.
- Remove Giant from nosys(), kill(), killpg(), and kern_sigaction() now
that sigacts is locked.
- Several in-kernel functions such as psignal(), tdsignal(), trapsignal(),
and thread_stopped() are now MP safe.
Reviewed by: arch@
Approved by: re (rwatson)
2003-05-13 20:36:02 +00:00
|
|
|
mtx_lock(&psp->ps_mtx);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
2003-06-28 06:34:08 +00:00
|
|
|
/*
|
|
|
|
* Build siginfo_t for SA thread
|
|
|
|
*/
|
|
|
|
void
|
2003-07-15 00:11:04 +00:00
|
|
|
cpu_thread_siginfo(int sig, u_long code, siginfo_t *si)
|
2003-06-28 06:34:08 +00:00
|
|
|
{
|
|
|
|
struct proc *p;
|
|
|
|
struct thread *td;
|
|
|
|
|
|
|
|
td = curthread;
|
|
|
|
p = td->td_proc;
|
|
|
|
PROC_LOCK_ASSERT(p, MA_OWNED);
|
|
|
|
|
|
|
|
bzero(si, sizeof(*si));
|
|
|
|
si->si_signo = sig;
|
|
|
|
si->si_code = code;
|
|
|
|
/* XXXKSE fill other fields */
|
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
int
|
2001-09-20 00:47:17 +00:00
|
|
|
sigreturn(struct thread *td, struct sigreturn_args *uap)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2002-09-19 04:30:43 +00:00
|
|
|
struct proc *p;
|
|
|
|
ucontext_t uc;
|
2004-03-02 06:13:09 +00:00
|
|
|
int error;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2002-09-19 04:30:43 +00:00
|
|
|
CTR2(KTR_SIG, "sigreturn: td=%p ucp=%p", td, uap->sigcntxp);
|
|
|
|
|
|
|
|
if (copyin(uap->sigcntxp, &uc, sizeof(uc)) != 0) {
|
|
|
|
CTR1(KTR_SIG, "sigreturn: efault td=%p", td);
|
|
|
|
return (EFAULT);
|
|
|
|
}
|
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
error = set_mcontext(td, &uc.uc_mcontext);
|
|
|
|
if (error != 0)
|
|
|
|
return (error);
|
2002-09-19 04:30:43 +00:00
|
|
|
|
|
|
|
p = td->td_proc;
|
|
|
|
PROC_LOCK(p);
|
2003-03-31 22:49:17 +00:00
|
|
|
td->td_sigmask = uc.uc_sigmask;
|
|
|
|
SIG_CANTMASK(td->td_sigmask);
|
|
|
|
signotify(td);
|
2002-09-19 04:30:43 +00:00
|
|
|
PROC_UNLOCK(p);
|
|
|
|
|
|
|
|
CTR3(KTR_SIG, "sigreturn: return td=%p pc=%#x sp=%#x",
|
2004-03-02 06:13:09 +00:00
|
|
|
td, uc.uc_mcontext.mc_srr0, uc.uc_mcontext.mc_gpr[1]);
|
2002-09-19 04:30:43 +00:00
|
|
|
|
|
|
|
return (EJUSTRETURN);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
2002-10-25 19:10:58 +00:00
|
|
|
#ifdef COMPAT_FREEBSD4
|
|
|
|
int
|
|
|
|
freebsd4_sigreturn(struct thread *td, struct freebsd4_sigreturn_args *uap)
|
|
|
|
{
|
|
|
|
|
|
|
|
return sigreturn(td, (struct sigreturn_args *)uap);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-07-12 22:25:09 +00:00
|
|
|
/*
|
|
|
|
* Construct a PCB from a trapframe. This is called from kdb_trap() where
|
|
|
|
* we want to start a backtrace from the function that caused us to enter
|
|
|
|
* the debugger. We have the context in the trapframe, but base the trace
|
|
|
|
* on the PCB. The PCB doesn't have to be perfect, as long as it contains
|
|
|
|
* enough for a backtrace.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
makectx(struct trapframe *tf, struct pcb *pcb)
|
|
|
|
{
|
|
|
|
|
|
|
|
pcb->pcb_lr = tf->srr0;
|
|
|
|
pcb->pcb_sp = tf->fixreg[1];
|
|
|
|
}
|
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
/*
|
|
|
|
* get_mcontext/sendsig helper routine that doesn't touch the
|
|
|
|
* proc lock
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
grab_mcontext(struct thread *td, mcontext_t *mcp, int flags)
|
|
|
|
{
|
|
|
|
struct pcb *pcb;
|
|
|
|
|
|
|
|
pcb = td->td_pcb;
|
|
|
|
|
|
|
|
memset(mcp, 0, sizeof(mcontext_t));
|
|
|
|
|
|
|
|
mcp->mc_vers = _MC_VERSION;
|
|
|
|
mcp->mc_flags = 0;
|
|
|
|
memcpy(&mcp->mc_frame, td->td_frame, sizeof(struct trapframe));
|
|
|
|
if (flags & GET_MC_CLEAR_RET) {
|
|
|
|
mcp->mc_gpr[3] = 0;
|
|
|
|
mcp->mc_gpr[4] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This assumes that floating-point context is *not* lazy,
|
|
|
|
* so if the thread has used FP there would have been a
|
|
|
|
* FP-unavailable exception that would have set things up
|
|
|
|
* correctly.
|
|
|
|
*/
|
|
|
|
if (pcb->pcb_flags & PCB_FPU) {
|
|
|
|
KASSERT(td == curthread,
|
|
|
|
("get_mcontext: fp save not curthread"));
|
|
|
|
critical_enter();
|
|
|
|
save_fpu(td);
|
|
|
|
critical_exit();
|
|
|
|
mcp->mc_flags |= _MC_FP_VALID;
|
|
|
|
memcpy(&mcp->mc_fpscr, &pcb->pcb_fpu.fpscr, sizeof(double));
|
|
|
|
memcpy(mcp->mc_fpreg, pcb->pcb_fpu.fpr, 32*sizeof(double));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX Altivec context ? */
|
|
|
|
|
|
|
|
mcp->mc_len = sizeof(*mcp);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2002-11-16 06:35:53 +00:00
|
|
|
int
|
2003-11-09 20:31:04 +00:00
|
|
|
get_mcontext(struct thread *td, mcontext_t *mcp, int flags)
|
2002-11-16 06:35:53 +00:00
|
|
|
{
|
2004-03-02 06:13:09 +00:00
|
|
|
int error;
|
2002-11-16 06:35:53 +00:00
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
error = grab_mcontext(td, mcp, flags);
|
|
|
|
if (error == 0) {
|
|
|
|
PROC_LOCK(curthread->td_proc);
|
|
|
|
mcp->mc_onstack = sigonstack(td->td_frame->fixreg[1]);
|
|
|
|
PROC_UNLOCK(curthread->td_proc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (error);
|
2002-11-16 06:35:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
set_mcontext(struct thread *td, const mcontext_t *mcp)
|
|
|
|
{
|
2004-03-02 06:13:09 +00:00
|
|
|
struct pcb *pcb;
|
|
|
|
struct trapframe *tf;
|
2002-11-16 06:35:53 +00:00
|
|
|
|
2004-03-02 06:13:09 +00:00
|
|
|
pcb = td->td_pcb;
|
|
|
|
tf = td->td_frame;
|
|
|
|
|
|
|
|
if (mcp->mc_vers != _MC_VERSION ||
|
|
|
|
mcp->mc_len != sizeof(*mcp))
|
|
|
|
return (EINVAL);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Don't let the user set privileged MSR bits
|
|
|
|
*/
|
|
|
|
if ((mcp->mc_srr1 & PSL_USERSTATIC) != (tf->srr1 & PSL_USERSTATIC)) {
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(tf, mcp->mc_frame, sizeof(mcp->mc_frame));
|
|
|
|
|
|
|
|
if (mcp->mc_flags & _MC_FP_VALID) {
|
|
|
|
if ((pcb->pcb_flags & PCB_FPU) != PCB_FPU) {
|
|
|
|
critical_enter();
|
|
|
|
enable_fpu(td);
|
|
|
|
critical_exit();
|
|
|
|
}
|
|
|
|
memcpy(&pcb->pcb_fpu.fpscr, &mcp->mc_fpscr, sizeof(double));
|
|
|
|
memcpy(pcb->pcb_fpu.fpr, mcp->mc_fpreg, 32*sizeof(double));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX Altivec context? */
|
|
|
|
|
|
|
|
return (0);
|
2002-11-16 06:35:53 +00:00
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
void
|
|
|
|
cpu_boot(int howto)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2005-02-04 05:32:56 +00:00
|
|
|
/* Get current clock frequency for the given cpu id. */
|
|
|
|
int
|
|
|
|
cpu_est_clockrate(int cpu_id, uint64_t *rate)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (ENXIO);
|
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
/*
|
|
|
|
* Shutdown the CPU as much as possible.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
cpu_halt(void)
|
|
|
|
{
|
|
|
|
|
|
|
|
OF_exit();
|
|
|
|
}
|
|
|
|
|
2003-10-19 02:36:07 +00:00
|
|
|
void
|
|
|
|
cpu_idle(void)
|
|
|
|
{
|
2005-02-28 09:49:00 +00:00
|
|
|
/* TODO: Insert code to halt (until next interrupt) */
|
|
|
|
|
|
|
|
#ifdef INVARIANTS
|
|
|
|
if ((mfmsr() & PSL_EE) != PSL_EE) {
|
|
|
|
struct thread *td = curthread;
|
|
|
|
printf("td crit %x\n", td->td_md.md_savecrit);
|
|
|
|
panic("ints disabled in idleproc!");
|
|
|
|
}
|
|
|
|
#endif
|
2003-10-19 02:36:07 +00:00
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
/*
|
|
|
|
* Set set up registers on exec.
|
|
|
|
*/
|
|
|
|
void
|
2002-08-29 06:17:48 +00:00
|
|
|
exec_setregs(struct thread *td, u_long entry, u_long stack, u_long ps_strings)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
|
|
|
struct trapframe *tf;
|
|
|
|
struct ps_strings arginfo;
|
|
|
|
|
2001-09-20 00:47:17 +00:00
|
|
|
tf = trapframe(td);
|
2001-06-10 02:39:37 +00:00
|
|
|
bzero(tf, sizeof *tf);
|
|
|
|
tf->fixreg[1] = -roundup(-stack + 8, 16);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX Machine-independent code has already copied arguments and
|
|
|
|
* XXX environment to userland. Get them back here.
|
|
|
|
*/
|
|
|
|
(void)copyin((char *)PS_STRINGS, &arginfo, sizeof(arginfo));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up arguments for _start():
|
|
|
|
* _start(argc, argv, envp, obj, cleanup, ps_strings);
|
|
|
|
*
|
|
|
|
* Notes:
|
|
|
|
* - obj and cleanup are the auxilliary and termination
|
|
|
|
* vectors. They are fixed up by ld.elf_so.
|
|
|
|
* - ps_strings is a NetBSD extention, and will be
|
|
|
|
* ignored by executables which are strictly
|
|
|
|
* compliant with the SVR4 ABI.
|
|
|
|
*
|
|
|
|
* XXX We have to set both regs and retval here due to different
|
|
|
|
* XXX calling convention in trap.c and init_main.c.
|
|
|
|
*/
|
2002-09-19 04:30:43 +00:00
|
|
|
/*
|
|
|
|
* XXX PG: these get overwritten in the syscall return code.
|
|
|
|
* execve() should return EJUSTRETURN, like it does on NetBSD.
|
|
|
|
* Emulate by setting the syscall return value cells. The
|
|
|
|
* registers still have to be set for init's fork trampoline.
|
|
|
|
*/
|
|
|
|
td->td_retval[0] = arginfo.ps_nargvstr;
|
|
|
|
td->td_retval[1] = (register_t)arginfo.ps_argvstr;
|
2001-06-10 02:39:37 +00:00
|
|
|
tf->fixreg[3] = arginfo.ps_nargvstr;
|
|
|
|
tf->fixreg[4] = (register_t)arginfo.ps_argvstr;
|
|
|
|
tf->fixreg[5] = (register_t)arginfo.ps_envstr;
|
|
|
|
tf->fixreg[6] = 0; /* auxillary vector */
|
|
|
|
tf->fixreg[7] = 0; /* termination vector */
|
|
|
|
tf->fixreg[8] = (register_t)PS_STRINGS; /* NetBSD extension */
|
|
|
|
|
|
|
|
tf->srr0 = entry;
|
|
|
|
tf->srr1 = PSL_MBO | PSL_USERSET | PSL_FE_DFLT;
|
2001-09-20 00:47:17 +00:00
|
|
|
td->td_pcb->pcb_flags = 0;
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
fill_regs(struct thread *td, struct reg *regs)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct trapframe *tf;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
tf = td->td_frame;
|
|
|
|
memcpy(regs, tf, sizeof(struct reg));
|
|
|
|
|
|
|
|
return (0);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
fill_dbregs(struct thread *td, struct dbreg *dbregs)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
/* No debug registers on PowerPC */
|
2001-06-10 02:39:37 +00:00
|
|
|
return (ENOSYS);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
fill_fpregs(struct thread *td, struct fpreg *fpregs)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct pcb *pcb;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
pcb = td->td_pcb;
|
|
|
|
|
|
|
|
if ((pcb->pcb_flags & PCB_FPU) == 0)
|
|
|
|
memset(fpregs, 0, sizeof(struct fpreg));
|
|
|
|
else
|
|
|
|
memcpy(fpregs, &pcb->pcb_fpu, sizeof(struct fpreg));
|
|
|
|
|
|
|
|
return (0);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
set_regs(struct thread *td, struct reg *regs)
|
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct trapframe *tf;
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
tf = td->td_frame;
|
|
|
|
memcpy(tf, regs, sizeof(struct reg));
|
|
|
|
|
|
|
|
return (0);
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
set_dbregs(struct thread *td, struct dbreg *dbregs)
|
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
/* No debug registers on PowerPC */
|
[partially forced commit due to pilot error in earlier commit attempt]
{set,fill}_{,fp,db}regs() fixup:
- Add dummy {set,fill}_dbregs() on architectures that don't have them.
- KSEfy the powerpc versions (struct proc -> struct thread).
- Some architectures had the prototypes in md_var.h, some in reg.h, and
some in both; for consistency, move them to reg.h on all platforms.
These functions aren't really MD (the implementation is MD, but the interface
is MI), so they should move to an MI header, but I haven't figured out which
one yet.
Run-tested on i386, build-tested on Alpha, untested on other platforms.
2001-10-21 22:16:48 +00:00
|
|
|
return (ENOSYS);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
set_fpregs(struct thread *td, struct fpreg *fpregs)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct pcb *pcb;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
pcb = td->td_pcb;
|
|
|
|
if ((pcb->pcb_flags & PCB_FPU) == 0)
|
|
|
|
enable_fpu(td);
|
|
|
|
memcpy(&pcb->pcb_fpu, fpregs, sizeof(struct fpreg));
|
|
|
|
|
|
|
|
return (0);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-20 00:47:17 +00:00
|
|
|
ptrace_set_pc(struct thread *td, unsigned long addr)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct trapframe *tf;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
tf = td->td_frame;
|
|
|
|
tf->srr0 = (register_t)addr;
|
|
|
|
|
|
|
|
return (0);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2001-09-20 00:47:17 +00:00
|
|
|
ptrace_single_step(struct thread *td)
|
2001-06-10 02:39:37 +00:00
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct trapframe *tf;
|
|
|
|
|
|
|
|
tf = td->td_frame;
|
|
|
|
tf->srr1 |= PSL_SE;
|
2001-06-10 02:39:37 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
return (0);
|
2001-06-10 02:39:37 +00:00
|
|
|
}
|
|
|
|
|
2004-07-13 07:22:56 +00:00
|
|
|
int
|
|
|
|
ptrace_clear_single_step(struct thread *td)
|
|
|
|
{
|
2004-07-29 13:34:50 +00:00
|
|
|
struct trapframe *tf;
|
2004-07-13 07:22:56 +00:00
|
|
|
|
2004-07-29 13:34:50 +00:00
|
|
|
tf = td->td_frame;
|
|
|
|
tf->srr1 &= ~PSL_SE;
|
|
|
|
|
|
|
|
return (0);
|
2004-07-13 07:22:56 +00:00
|
|
|
}
|
|
|
|
|
2001-06-10 02:39:37 +00:00
|
|
|
/*
|
2001-12-11 23:33:44 +00:00
|
|
|
* Initialise a struct pcpu.
|
2001-06-10 02:39:37 +00:00
|
|
|
*/
|
|
|
|
void
|
2001-12-11 23:33:44 +00:00
|
|
|
cpu_pcpu_init(struct pcpu *pcpu, int cpuid, size_t sz)
|
|
|
|
{
|
2001-06-10 02:39:37 +00:00
|
|
|
|
|
|
|
}
|
2002-05-28 07:36:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* kcopy(const void *src, void *dst, size_t len);
|
|
|
|
*
|
|
|
|
* Copy len bytes from src to dst, aborting if we encounter a fatal
|
|
|
|
* page fault.
|
|
|
|
*
|
|
|
|
* kcopy() _must_ save and restore the old fault handler since it is
|
|
|
|
* called by uiomove(), which may be in the path of servicing a non-fatal
|
|
|
|
* page fault.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
kcopy(const void *src, void *dst, size_t len)
|
|
|
|
{
|
|
|
|
struct thread *td;
|
|
|
|
faultbuf env, *oldfault;
|
|
|
|
int rv;
|
|
|
|
|
|
|
|
td = PCPU_GET(curthread);
|
|
|
|
oldfault = td->td_pcb->pcb_onfault;
|
|
|
|
if ((rv = setfault(env)) != 0) {
|
|
|
|
td->td_pcb->pcb_onfault = oldfault;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(dst, src, len);
|
|
|
|
|
|
|
|
td->td_pcb->pcb_onfault = oldfault;
|
|
|
|
return (0);
|
|
|
|
}
|
2003-04-01 00:18:55 +00:00
|
|
|
|
2004-01-21 05:18:08 +00:00
|
|
|
void
|
|
|
|
asm_panic(char *pstr)
|
|
|
|
{
|
|
|
|
panic(pstr);
|
|
|
|
}
|
2004-07-12 22:25:09 +00:00
|
|
|
|
|
|
|
int db_trap_glue(struct trapframe *); /* Called from trap_subr.S */
|
|
|
|
|
|
|
|
int
|
|
|
|
db_trap_glue(struct trapframe *frame)
|
|
|
|
{
|
|
|
|
if (!(frame->srr1 & PSL_PR)
|
|
|
|
&& (frame->exc == EXC_TRC || frame->exc == EXC_RUNMODETRC
|
|
|
|
|| (frame->exc == EXC_PGM
|
|
|
|
&& (frame->srr1 & 0x20000))
|
2004-07-23 05:27:17 +00:00
|
|
|
|| frame->exc == EXC_BPT
|
|
|
|
|| frame->exc == EXC_DSI)) {
|
2004-07-12 22:25:09 +00:00
|
|
|
int type = frame->exc;
|
|
|
|
if (type == EXC_PGM && (frame->srr1 & 0x20000)) {
|
|
|
|
type = T_BREAKPOINT;
|
|
|
|
}
|
|
|
|
return (kdb_trap(type, 0, frame));
|
|
|
|
}
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|