Commit a0f3a291 authored by Linus Torvalds's avatar Linus Torvalds

Merge bk://are.twiddle.net/axp-2.5

into home.transmeta.com:/home/torvalds/v2.5/linux
parents 2494a44d 35cc9f80
...@@ -530,6 +530,9 @@ config ALPHA_LARGE_VMALLOC ...@@ -530,6 +530,9 @@ config ALPHA_LARGE_VMALLOC
Say N unless you know you need gobs and gobs of vmalloc space. Say N unless you know you need gobs and gobs of vmalloc space.
config VERBOSE_MCHECK
bool "Verbose Machine Checks"
source "drivers/pci/Kconfig" source "drivers/pci/Kconfig"
config HOTPLUG config HOTPLUG
......
...@@ -12,6 +12,7 @@ NM := $(NM) -B ...@@ -12,6 +12,7 @@ NM := $(NM) -B
LDFLAGS_vmlinux = -static -N #-relax LDFLAGS_vmlinux = -static -N #-relax
CFLAGS := $(CFLAGS) -pipe -mno-fp-regs -ffixed-8 CFLAGS := $(CFLAGS) -pipe -mno-fp-regs -ffixed-8
ARCHBLOBLFLAGS := -I binary -O elf64-alpha -B alpha
# Determine if we can use the BWX instructions with GAS. # Determine if we can use the BWX instructions with GAS.
old_gas := $(shell if $(AS) --version 2>&1 | grep 'version 2.7' > /dev/null; then echo y; else echo n; fi) old_gas := $(shell if $(AS) --version 2>&1 | grep 'version 2.7' > /dev/null; then echo y; else echo n; fi)
......
...@@ -59,6 +59,7 @@ CONFIG_ALPHA_GENERIC=y ...@@ -59,6 +59,7 @@ CONFIG_ALPHA_GENERIC=y
# CONFIG_ALPHA_TAKARA is not set # CONFIG_ALPHA_TAKARA is not set
# CONFIG_ALPHA_TITAN is not set # CONFIG_ALPHA_TITAN is not set
# CONFIG_ALPHA_WILDFIRE is not set # CONFIG_ALPHA_WILDFIRE is not set
CONFIG_VERBOSE_MCHECK=y
CONFIG_ISA=y CONFIG_ISA=y
CONFIG_EISA=y CONFIG_EISA=y
# CONFIG_SBUS is not set # CONFIG_SBUS is not set
......
...@@ -10,7 +10,7 @@ export-objs := alpha_ksyms.o ...@@ -10,7 +10,7 @@ export-objs := alpha_ksyms.o
obj-y := entry.o traps.o process.o init_task.o osf_sys.o irq.o \ obj-y := entry.o traps.o process.o init_task.o osf_sys.o irq.o \
irq_alpha.o signal.o setup.o ptrace.o time.o semaphore.o \ irq_alpha.o signal.o setup.o ptrace.o time.o semaphore.o \
alpha_ksyms.o systbls.o alpha_ksyms.o systbls.o err_common.o
# #
# FIXME! # FIXME!
......
...@@ -47,6 +47,15 @@ ...@@ -47,6 +47,15 @@
#define vip volatile int * #define vip volatile int *
/* Save CIA configuration data as the console had it set up. */
struct
{
unsigned int w_base;
unsigned int w_mask;
unsigned int t_base;
} saved_config[4] __attribute((common));
/* /*
* Given a bus, device, and function number, compute resulting * Given a bus, device, and function number, compute resulting
* configuration space address. It is therefore not safe to have * configuration space address. It is therefore not safe to have
...@@ -648,6 +657,24 @@ do_init_arch(int is_pyxis) ...@@ -648,6 +657,24 @@ do_init_arch(int is_pyxis)
hose->dense_io_base = CIA_BW_IO - IDENT_ADDR; hose->dense_io_base = CIA_BW_IO - IDENT_ADDR;
} }
/* Save CIA configuration data as the console had it set up. */
saved_config[0].w_base = *(vip)CIA_IOC_PCI_W0_BASE;
saved_config[0].w_mask = *(vip)CIA_IOC_PCI_W0_MASK;
saved_config[0].t_base = *(vip)CIA_IOC_PCI_T0_BASE;
saved_config[1].w_base = *(vip)CIA_IOC_PCI_W1_BASE;
saved_config[1].w_mask = *(vip)CIA_IOC_PCI_W1_MASK;
saved_config[1].t_base = *(vip)CIA_IOC_PCI_T1_BASE;
saved_config[2].w_base = *(vip)CIA_IOC_PCI_W2_BASE;
saved_config[2].w_mask = *(vip)CIA_IOC_PCI_W2_MASK;
saved_config[2].t_base = *(vip)CIA_IOC_PCI_T2_BASE;
saved_config[3].w_base = *(vip)CIA_IOC_PCI_W3_BASE;
saved_config[3].w_mask = *(vip)CIA_IOC_PCI_W3_MASK;
saved_config[3].t_base = *(vip)CIA_IOC_PCI_T3_BASE;
/* /*
* Set up the PCI to main memory translation windows. * Set up the PCI to main memory translation windows.
* *
...@@ -736,6 +763,26 @@ pyxis_init_arch(void) ...@@ -736,6 +763,26 @@ pyxis_init_arch(void)
do_init_arch(1); do_init_arch(1);
} }
void
cia_kill_arch(int mode)
{
*(vip)CIA_IOC_PCI_W0_BASE = saved_config[0].w_base;
*(vip)CIA_IOC_PCI_W0_MASK = saved_config[0].w_mask;
*(vip)CIA_IOC_PCI_T0_BASE = saved_config[0].t_base;
*(vip)CIA_IOC_PCI_W1_BASE = saved_config[1].w_base;
*(vip)CIA_IOC_PCI_W1_MASK = saved_config[1].w_mask;
*(vip)CIA_IOC_PCI_T1_BASE = saved_config[1].t_base;
*(vip)CIA_IOC_PCI_W2_BASE = saved_config[2].w_base;
*(vip)CIA_IOC_PCI_W2_MASK = saved_config[2].w_mask;
*(vip)CIA_IOC_PCI_T2_BASE = saved_config[2].t_base;
*(vip)CIA_IOC_PCI_W3_BASE = saved_config[3].w_base;
*(vip)CIA_IOC_PCI_W3_MASK = saved_config[3].w_mask;
*(vip)CIA_IOC_PCI_T3_BASE = saved_config[3].t_base;
}
void __init void __init
cia_init_pci(void) cia_init_pci(void)
{ {
...@@ -755,6 +802,7 @@ cia_pci_clr_err(void) ...@@ -755,6 +802,7 @@ cia_pci_clr_err(void)
*(vip)CIA_IOC_CIA_ERR; /* re-read to force write. */ *(vip)CIA_IOC_CIA_ERR; /* re-read to force write. */
} }
#ifdef CONFIG_VERBOSE_MCHECK
static void static void
cia_decode_pci_error(struct el_CIA_sysdata_mcheck *cia, const char *msg) cia_decode_pci_error(struct el_CIA_sysdata_mcheck *cia, const char *msg)
{ {
...@@ -1022,13 +1070,13 @@ cia_decode_parity_error(struct el_CIA_sysdata_mcheck *cia) ...@@ -1022,13 +1070,13 @@ cia_decode_parity_error(struct el_CIA_sysdata_mcheck *cia)
printk(KERN_CRIT " Command: %s, Parity bit: %d\n", cmd, par); printk(KERN_CRIT " Command: %s, Parity bit: %d\n", cmd, par);
printk(KERN_CRIT " Address: %#010lx, Mask: %#lx\n", addr, mask); printk(KERN_CRIT " Address: %#010lx, Mask: %#lx\n", addr, mask);
} }
#endif
static int static int
cia_decode_mchk(unsigned long la_ptr) cia_decode_mchk(unsigned long la_ptr)
{ {
struct el_common *com; struct el_common *com;
struct el_CIA_sysdata_mcheck *cia; struct el_CIA_sysdata_mcheck *cia;
int which;
com = (void *)la_ptr; com = (void *)la_ptr;
cia = (void *)(la_ptr + com->sys_offset); cia = (void *)(la_ptr + com->sys_offset);
...@@ -1036,8 +1084,8 @@ cia_decode_mchk(unsigned long la_ptr) ...@@ -1036,8 +1084,8 @@ cia_decode_mchk(unsigned long la_ptr)
if ((cia->cia_err & CIA_ERR_VALID) == 0) if ((cia->cia_err & CIA_ERR_VALID) == 0)
return 0; return 0;
which = cia->cia_err & 0xfff; #ifdef CONFIG_VERBOSE_MCHECK
switch (ffs(which) - 1) { switch (ffs(cia->cia_err & 0xfff) - 1) {
case 0: /* CIA_ERR_COR_ERR */ case 0: /* CIA_ERR_COR_ERR */
cia_decode_ecc_error(cia, "Corrected ECC error"); cia_decode_ecc_error(cia, "Corrected ECC error");
break; break;
...@@ -1109,6 +1157,7 @@ cia_decode_mchk(unsigned long la_ptr) ...@@ -1109,6 +1157,7 @@ cia_decode_mchk(unsigned long la_ptr)
if (cia->cia_err & CIA_ERR_LOST_IOA_TIMEOUT) if (cia->cia_err & CIA_ERR_LOST_IOA_TIMEOUT)
printk(KERN_CRIT "CIA lost machine check: " printk(KERN_CRIT "CIA lost machine check: "
"I/O timeout\n"); "I/O timeout\n");
#endif
return 1; return 1;
} }
......
...@@ -27,12 +27,14 @@ ...@@ -27,12 +27,14 @@
unsigned TITAN_agp = 0; unsigned TITAN_agp = 0;
static struct /* Save Titan configuration data as the console had it set up. */
struct
{ {
unsigned long wsba[4]; unsigned long wsba[4];
unsigned long wsm[4]; unsigned long wsm[4];
unsigned long tba[4]; unsigned long tba[4];
} saved_pachip_port[4]; } saved_config[4] __attribute__((common));
/* /*
* BIOS32-style PCI interface: * BIOS32-style PCI interface:
...@@ -289,21 +291,21 @@ titan_init_one_pachip_port(titan_pachip_port *port, int index) ...@@ -289,21 +291,21 @@ titan_init_one_pachip_port(titan_pachip_port *port, int index)
* Save the existing PCI window translations. SRM will * Save the existing PCI window translations. SRM will
* need them when we go to reboot. * need them when we go to reboot.
*/ */
saved_pachip_port[index].wsba[0] = port->wsba[0].csr; saved_config[index].wsba[0] = port->wsba[0].csr;
saved_pachip_port[index].wsm[0] = port->wsm[0].csr; saved_config[index].wsm[0] = port->wsm[0].csr;
saved_pachip_port[index].tba[0] = port->tba[0].csr; saved_config[index].tba[0] = port->tba[0].csr;
saved_pachip_port[index].wsba[1] = port->wsba[1].csr; saved_config[index].wsba[1] = port->wsba[1].csr;
saved_pachip_port[index].wsm[1] = port->wsm[1].csr; saved_config[index].wsm[1] = port->wsm[1].csr;
saved_pachip_port[index].tba[1] = port->tba[1].csr; saved_config[index].tba[1] = port->tba[1].csr;
saved_pachip_port[index].wsba[2] = port->wsba[2].csr; saved_config[index].wsba[2] = port->wsba[2].csr;
saved_pachip_port[index].wsm[2] = port->wsm[2].csr; saved_config[index].wsm[2] = port->wsm[2].csr;
saved_pachip_port[index].tba[2] = port->tba[2].csr; saved_config[index].tba[2] = port->tba[2].csr;
saved_pachip_port[index].wsba[3] = port->wsba[3].csr; saved_config[index].wsba[3] = port->wsba[3].csr;
saved_pachip_port[index].wsm[3] = port->wsm[3].csr; saved_config[index].wsm[3] = port->wsm[3].csr;
saved_pachip_port[index].tba[3] = port->tba[3].csr; saved_config[index].tba[3] = port->tba[3].csr;
/* /*
* Set up the PCI to main memory translation windows. * Set up the PCI to main memory translation windows.
...@@ -392,21 +394,21 @@ titan_init_arch(void) ...@@ -392,21 +394,21 @@ titan_init_arch(void)
static void static void
titan_kill_one_pachip_port(titan_pachip_port *port, int index) titan_kill_one_pachip_port(titan_pachip_port *port, int index)
{ {
port->wsba[0].csr = saved_pachip_port[index].wsba[0]; port->wsba[0].csr = saved_config[index].wsba[0];
port->wsm[0].csr = saved_pachip_port[index].wsm[0]; port->wsm[0].csr = saved_config[index].wsm[0];
port->tba[0].csr = saved_pachip_port[index].tba[0]; port->tba[0].csr = saved_config[index].tba[0];
port->wsba[1].csr = saved_pachip_port[index].wsba[1]; port->wsba[1].csr = saved_config[index].wsba[1];
port->wsm[1].csr = saved_pachip_port[index].wsm[1]; port->wsm[1].csr = saved_config[index].wsm[1];
port->tba[1].csr = saved_pachip_port[index].tba[1]; port->tba[1].csr = saved_config[index].tba[1];
port->wsba[2].csr = saved_pachip_port[index].wsba[2]; port->wsba[2].csr = saved_config[index].wsba[2];
port->wsm[2].csr = saved_pachip_port[index].wsm[2]; port->wsm[2].csr = saved_config[index].wsm[2];
port->tba[2].csr = saved_pachip_port[index].tba[2]; port->tba[2].csr = saved_config[index].tba[2];
port->wsba[3].csr = saved_pachip_port[index].wsba[3]; port->wsba[3].csr = saved_config[index].wsba[3];
port->wsm[3].csr = saved_pachip_port[index].wsm[3]; port->wsm[3].csr = saved_config[index].wsm[3];
port->tba[3].csr = saved_pachip_port[index].tba[3]; port->tba[3].csr = saved_config[index].tba[3];
} }
static void static void
......
...@@ -26,13 +26,14 @@ ...@@ -26,13 +26,14 @@
#include "proto.h" #include "proto.h"
#include "pci_impl.h" #include "pci_impl.h"
/* Save Tsunami configuration data as the console had it set up. */
static struct struct
{ {
unsigned long wsba[4]; unsigned long wsba[4];
unsigned long wsm[4]; unsigned long wsm[4];
unsigned long tba[4]; unsigned long tba[4];
} saved_pchip[2]; } saved_config[2] __attribute__((common));
/* /*
* NOTE: Herein lie back-to-back mb instructions. They are magic. * NOTE: Herein lie back-to-back mb instructions. They are magic.
...@@ -293,21 +294,21 @@ tsunami_init_one_pchip(tsunami_pchip *pchip, int index) ...@@ -293,21 +294,21 @@ tsunami_init_one_pchip(tsunami_pchip *pchip, int index)
* need them when we go to reboot. * need them when we go to reboot.
*/ */
saved_pchip[index].wsba[0] = pchip->wsba[0].csr; saved_config[index].wsba[0] = pchip->wsba[0].csr;
saved_pchip[index].wsm[0] = pchip->wsm[0].csr; saved_config[index].wsm[0] = pchip->wsm[0].csr;
saved_pchip[index].tba[0] = pchip->tba[0].csr; saved_config[index].tba[0] = pchip->tba[0].csr;
saved_pchip[index].wsba[1] = pchip->wsba[1].csr; saved_config[index].wsba[1] = pchip->wsba[1].csr;
saved_pchip[index].wsm[1] = pchip->wsm[1].csr; saved_config[index].wsm[1] = pchip->wsm[1].csr;
saved_pchip[index].tba[1] = pchip->tba[1].csr; saved_config[index].tba[1] = pchip->tba[1].csr;
saved_pchip[index].wsba[2] = pchip->wsba[2].csr; saved_config[index].wsba[2] = pchip->wsba[2].csr;
saved_pchip[index].wsm[2] = pchip->wsm[2].csr; saved_config[index].wsm[2] = pchip->wsm[2].csr;
saved_pchip[index].tba[2] = pchip->tba[2].csr; saved_config[index].tba[2] = pchip->tba[2].csr;
saved_pchip[index].wsba[3] = pchip->wsba[3].csr; saved_config[index].wsba[3] = pchip->wsba[3].csr;
saved_pchip[index].wsm[3] = pchip->wsm[3].csr; saved_config[index].wsm[3] = pchip->wsm[3].csr;
saved_pchip[index].tba[3] = pchip->tba[3].csr; saved_config[index].tba[3] = pchip->tba[3].csr;
/* /*
* Set up the PCI to main memory translation windows. * Set up the PCI to main memory translation windows.
...@@ -403,21 +404,21 @@ tsunami_init_arch(void) ...@@ -403,21 +404,21 @@ tsunami_init_arch(void)
static void static void
tsunami_kill_one_pchip(tsunami_pchip *pchip, int index) tsunami_kill_one_pchip(tsunami_pchip *pchip, int index)
{ {
pchip->wsba[0].csr = saved_pchip[index].wsba[0]; pchip->wsba[0].csr = saved_config[index].wsba[0];
pchip->wsm[0].csr = saved_pchip[index].wsm[0]; pchip->wsm[0].csr = saved_config[index].wsm[0];
pchip->tba[0].csr = saved_pchip[index].tba[0]; pchip->tba[0].csr = saved_config[index].tba[0];
pchip->wsba[1].csr = saved_pchip[index].wsba[1]; pchip->wsba[1].csr = saved_config[index].wsba[1];
pchip->wsm[1].csr = saved_pchip[index].wsm[1]; pchip->wsm[1].csr = saved_config[index].wsm[1];
pchip->tba[1].csr = saved_pchip[index].tba[1]; pchip->tba[1].csr = saved_config[index].tba[1];
pchip->wsba[2].csr = saved_pchip[index].wsba[2]; pchip->wsba[2].csr = saved_config[index].wsba[2];
pchip->wsm[2].csr = saved_pchip[index].wsm[2]; pchip->wsm[2].csr = saved_config[index].wsm[2];
pchip->tba[2].csr = saved_pchip[index].tba[2]; pchip->tba[2].csr = saved_config[index].tba[2];
pchip->wsba[3].csr = saved_pchip[index].wsba[3]; pchip->wsba[3].csr = saved_config[index].wsba[3];
pchip->wsm[3].csr = saved_pchip[index].wsm[3]; pchip->wsm[3].csr = saved_config[index].wsm[3];
pchip->tba[3].csr = saved_pchip[index].tba[3]; pchip->tba[3].csr = saved_config[index].tba[3];
} }
void void
......
...@@ -452,7 +452,8 @@ sys_fork: ...@@ -452,7 +452,8 @@ sys_fork:
bsr $1,do_switch_stack bsr $1,do_switch_stack
bis $31,SIGCHLD,$16 bis $31,SIGCHLD,$16
mov $31,$17 mov $31,$17
mov $30,$18 mov $31,$18
mov $30,$19
jsr $26,alpha_clone jsr $26,alpha_clone
bsr $1,undo_switch_stack bsr $1,undo_switch_stack
ret $31,($26),1 ret $31,($26),1
...@@ -463,8 +464,9 @@ sys_fork: ...@@ -463,8 +464,9 @@ sys_fork:
.ent sys_clone .ent sys_clone
sys_clone: sys_clone:
bsr $1,do_switch_stack bsr $1,do_switch_stack
/* arg1 and arg2 come from the user */ /* $16, $17, $18, $19 come from the user; $19 is used later
mov $30,$18 via pt_regs->r19. */
mov $30,$19
jsr $26,alpha_clone jsr $26,alpha_clone
bsr $1,undo_switch_stack bsr $1,undo_switch_stack
ret $31,($26),1 ret $31,($26),1
......
This diff is collapsed.
/*
* linux/arch/alpha/kernel/err_impl.h
*
* Copyright (C) 2000 Jeff Wiedemeier (Compaq Computer Corporation)
*
* Contains declarations and macros to support Alpha error handling
* implementations.
*/
/*
* SCB Vector definitions
*/
#define SCB_Q_SYSERR 0x620
#define SCB_Q_PROCERR 0x630
#define SCB_Q_SYSMCHK 0x660
#define SCB_Q_PROCMCHK 0x670
#define SCB_Q_SYSEVENT 0x680
/*
* Disposition definitions for logout frame parser
*/
#define MCHK_DISPOSITION_UNKNOWN_ERROR 0x00
#define MCHK_DISPOSITION_REPORT 0x01
#define MCHK_DISPOSITION_DISMISS 0x02
/*
* Error Log definitions
*/
/*
* Types
*/
#define EL_CLASS__TERMINATION (0)
# define EL_TYPE__TERMINATION__TERMINATION (0)
#define EL_CLASS__HEADER (5)
# define EL_TYPE__HEADER__SYSTEM_ERROR_FRAME (1)
# define EL_TYPE__HEADER__SYSTEM_EVENT_FRAME (2)
# define EL_TYPE__HEADER__HALT_FRAME (3)
# define EL_TYPE__HEADER__LOGOUT_FRAME (19)
#define EL_CLASS__GENERAL_NOTIFICATION (9)
#define EL_CLASS__PCI_ERROR_FRAME (11)
#define EL_CLASS__REGATTA_FAMILY (12)
# define EL_TYPE__REGATTA__PROCESSOR_ERROR_FRAME (1)
# define EL_TYPE__REGATTA__SYSTEM_ERROR_FRAME (2)
# define EL_TYPE__REGATTA__ENVIRONMENTAL_FRAME (3)
# define EL_TYPE__REGATTA__TITAN_PCHIP0_EXTENDED (8)
# define EL_TYPE__REGATTA__TITAN_PCHIP1_EXTENDED (9)
# define EL_TYPE__REGATTA__TITAN_MEMORY_EXTENDED (10)
# define EL_TYPE__REGATTA__PROCESSOR_DBL_ERROR_HALT (11)
# define EL_TYPE__REGATTA__SYSTEM_DBL_ERROR_HALT (12)
#define EL_CLASS__PAL (14)
# define EL_TYPE__PAL__LOGOUT_FRAME (1)
# define EL_TYPE__PAL__EV7_PROCESSOR (4)
# define EL_TYPE__PAL__EV7_ZBOX (5)
# define EL_TYPE__PAL__EV7_RBOX (6)
# define EL_TYPE__PAL__EV7_IO (7)
union el_timestamp {
struct {
u8 second;
u8 minute;
u8 hour;
u8 day;
u8 month;
u8 year;
} b;
u64 as_int;
};
struct el_subpacket {
u16 length; /* length of header (in bytes) */
u16 class; /* header class and type... */
u16 type; /* ...determine content */
u16 revision; /* header revision */
union {
struct { /* Class 5, Type 1 - System Error */
u32 frame_length;
u32 frame_packet_count;
} sys_err;
struct { /* Class 5, Type 2 - System Event */
union el_timestamp timestamp;
u32 frame_length;
u32 frame_packet_count;
} sys_event;
struct { /* Class 5, Type 3 - Double Error Halt */
u16 halt_code;
u16 reserved;
union el_timestamp timestamp;
u32 frame_length;
u32 frame_packet_count;
} err_halt;
struct { /* Clasee 5, Type 19 - Logout Frame Header */
u32 frame_length;
u32 frame_flags;
u32 cpu_offset;
u32 system_offset;
} logout_header;
struct { /* Class 12 - Regatta */
u64 cpuid;
u64 data_start[1];
} regatta_frame;
struct { /* Raw */
u64 data_start[1];
} raw;
} by_type;
};
struct el_subpacket_annotation {
struct el_subpacket_annotation *next;
u16 class;
u16 type;
u16 revision;
char *description;
char **annotation;
};
#define SUBPACKET_ANNOTATION(c, t, r, d, a) {NULL, (c), (t), (r), (d), (a)}
struct el_subpacket_handler {
struct el_subpacket_handler *next;
u16 class;
struct el_subpacket *(*handler)(struct el_subpacket *);
};
#define SUBPACKET_HANDLER_INIT(c, h) {NULL, (c), (h)}
/*
* Extract a field from a register given it's name. defines
* for the LSB (__S - shift count) and bitmask (__M) are required
*/
#define EXTRACT(u, f) (((u) >> f##__S) & f##__M)
/*
* err_common.c
*/
extern char *err_print_prefix;
extern void mchk_dump_mem(void *, int, char **);
extern void mchk_dump_logout_frame(struct el_common *);
extern void ev7_register_error_handlers(void);
extern void ev7_machine_check(u64, u64, struct pt_regs *);
extern void ev6_register_error_handlers(void);
extern int ev6_process_logout_frame(struct el_common *, int);
extern void ev6_machine_check(u64, u64, struct pt_regs *);
extern struct el_subpacket *el_process_subpacket(struct el_subpacket *);
extern void el_annotate_subpacket(struct el_subpacket *);
extern void cdl_check_console_data_log(void);
extern int cdl_register_subpacket_annotation(struct el_subpacket_annotation *);
extern int cdl_register_subpacket_handler(struct el_subpacket_handler *);
...@@ -590,7 +590,7 @@ handle_irq(int irq, struct pt_regs * regs) ...@@ -590,7 +590,7 @@ handle_irq(int irq, struct pt_regs * regs)
} }
irq_enter(); irq_enter();
kstat_cpu(i).irqs[irq]++; kstat_cpu(cpu).irqs[irq]++;
spin_lock_irq(&desc->lock); /* mask also the higher prio events */ spin_lock_irq(&desc->lock); /* mask also the higher prio events */
desc->handler->ack(irq); desc->handler->ack(irq);
/* /*
......
...@@ -147,10 +147,10 @@ process_mcheck_info(unsigned long vector, unsigned long la_ptr, ...@@ -147,10 +147,10 @@ process_mcheck_info(unsigned long vector, unsigned long la_ptr,
mchk_header = (struct el_common *)la_ptr; mchk_header = (struct el_common *)la_ptr;
printk(KERN_CRIT "%s machine check: vector=0x%lx pc=0x%lx code=0x%lx\n", printk(KERN_CRIT "%s machine check: vector=0x%lx pc=0x%lx code=0x%x\n",
machine, vector, regs->pc, mchk_header->code); machine, vector, regs->pc, mchk_header->code);
switch ((unsigned int) mchk_header->code) { switch (mchk_header->code) {
/* Machine check reasons. Defined according to PALcode sources. */ /* Machine check reasons. Defined according to PALcode sources. */
case 0x80: reason = "tag parity error"; break; case 0x80: reason = "tag parity error"; break;
case 0x82: reason = "tag control parity error"; break; case 0x82: reason = "tag control parity error"; break;
......
...@@ -844,7 +844,7 @@ osf_setsysinfo(unsigned long op, void *buffer, unsigned long nbytes, ...@@ -844,7 +844,7 @@ osf_setsysinfo(unsigned long op, void *buffer, unsigned long nbytes,
{ {
switch (op) { switch (op) {
case SSI_IEEE_FP_CONTROL: { case SSI_IEEE_FP_CONTROL: {
unsigned long swcr, fpcr; unsigned long swcr, fpcr, fex;
/* /*
* Alpha Architecture Handbook 4.7.7.3: * Alpha Architecture Handbook 4.7.7.3:
...@@ -867,9 +867,24 @@ osf_setsysinfo(unsigned long op, void *buffer, unsigned long nbytes, ...@@ -867,9 +867,24 @@ osf_setsysinfo(unsigned long op, void *buffer, unsigned long nbytes,
wrfpcr(fpcr); wrfpcr(fpcr);
/* If any exceptions are now unmasked, send a signal. */ /* If any exceptions are now unmasked, send a signal. */
if (((swcr & IEEE_STATUS_MASK) fex = ((swcr & IEEE_STATUS_MASK)
>> IEEE_STATUS_TO_EXCSUM_SHIFT) & swcr) { >> IEEE_STATUS_TO_EXCSUM_SHIFT) & swcr;
send_sig(SIGFPE, current, 1); if (fex) {
siginfo_t info;
int si_code = 0;
if (fex & IEEE_TRAP_ENABLE_DNO) si_code = FPE_FLTUND;
if (fex & IEEE_TRAP_ENABLE_INE) si_code = FPE_FLTRES;
if (fex & IEEE_TRAP_ENABLE_UNF) si_code = FPE_FLTUND;
if (fex & IEEE_TRAP_ENABLE_OVF) si_code = FPE_FLTOVF;
if (fex & IEEE_TRAP_ENABLE_DZE) si_code = FPE_FLTDIV;
if (fex & IEEE_TRAP_ENABLE_INV) si_code = FPE_FLTINV;
info.si_signo = SIGFPE;
info.si_errno = 0;
info.si_code = si_code;
info.si_addr = 0; /* FIXME */
send_sig_info(SIGFPE, &info, current);
} }
return 0; return 0;
......
...@@ -245,11 +245,10 @@ release_thread(struct task_struct *dead_task) ...@@ -245,11 +245,10 @@ release_thread(struct task_struct *dead_task)
*/ */
int int
alpha_clone(unsigned long clone_flags, unsigned long usp, alpha_clone(unsigned long clone_flags, unsigned long usp,
struct switch_stack * swstack) int *user_tid, struct switch_stack * swstack)
{ {
struct task_struct *p; struct task_struct *p;
struct pt_regs *u_regs = (struct pt_regs *) (swstack+1); struct pt_regs *u_regs = (struct pt_regs *) (swstack+1);
int *user_tid = (int *)u_regs->r19;
if (!usp) if (!usp)
usp = rdusp(); usp = rdusp();
...@@ -314,6 +313,16 @@ copy_thread(int nr, unsigned long clone_flags, unsigned long usp, ...@@ -314,6 +313,16 @@ copy_thread(int nr, unsigned long clone_flags, unsigned long usp,
childti->pcb.ksp = (unsigned long) childstack; childti->pcb.ksp = (unsigned long) childstack;
childti->pcb.flags = 1; /* set FEN, clear everything else */ childti->pcb.flags = 1; /* set FEN, clear everything else */
/* Set a new TLS for the child thread? Peek back into the
syscall arguments that we saved on syscall entry. */
/* Note: if CLONE_SETTLS is not set, then we must inherit the
value from the parent, which will have been set by the block
copy in dup_task_struct. This is non-intuitive, but is
required for proper operation in the case of a threaded
application calling fork. */
if (clone_flags & CLONE_SETTLS)
childti->pcb.unique = regs->r19;
return 0; return 0;
} }
......
...@@ -24,6 +24,7 @@ extern struct pci_ops cia_pci_ops; ...@@ -24,6 +24,7 @@ extern struct pci_ops cia_pci_ops;
extern void cia_init_pci(void); extern void cia_init_pci(void);
extern void cia_init_arch(void); extern void cia_init_arch(void);
extern void pyxis_init_arch(void); extern void pyxis_init_arch(void);
extern void cia_kill_arch(int);
extern void cia_machine_check(u64, u64, struct pt_regs *); extern void cia_machine_check(u64, u64, struct pt_regs *);
extern void cia_pci_tbi(struct pci_controller *, dma_addr_t, dma_addr_t); extern void cia_pci_tbi(struct pci_controller *, dma_addr_t, dma_addr_t);
......
...@@ -293,8 +293,16 @@ do_sigreturn(struct sigframe *frame, struct pt_regs *regs, ...@@ -293,8 +293,16 @@ do_sigreturn(struct sigframe *frame, struct pt_regs *regs,
goto give_sigsegv; goto give_sigsegv;
/* Send SIGTRAP if we're single-stepping: */ /* Send SIGTRAP if we're single-stepping: */
if (ptrace_cancel_bpt (current)) if (ptrace_cancel_bpt (current)) {
send_sig(SIGTRAP, current, 1); siginfo_t info;
info.si_signo = SIGTRAP;
info.si_errno = 0;
info.si_code = TRAP_BRKPT;
info.si_addr = (void *) regs->pc;
info.si_trapno = 0;
send_sig_info(SIGTRAP, &info, current);
}
return; return;
give_sigsegv: give_sigsegv:
...@@ -330,8 +338,16 @@ do_rt_sigreturn(struct rt_sigframe *frame, struct pt_regs *regs, ...@@ -330,8 +338,16 @@ do_rt_sigreturn(struct rt_sigframe *frame, struct pt_regs *regs,
do_sigaltstack(&st, NULL, rdusp()); do_sigaltstack(&st, NULL, rdusp());
/* Send SIGTRAP if we're single-stepping: */ /* Send SIGTRAP if we're single-stepping: */
if (ptrace_cancel_bpt (current)) if (ptrace_cancel_bpt (current)) {
send_sig(SIGTRAP, current, 1); siginfo_t info;
info.si_signo = SIGTRAP;
info.si_errno = 0;
info.si_code = TRAP_BRKPT;
info.si_addr = (void *) regs->pc;
info.si_trapno = 0;
send_sig_info(SIGTRAP, &info, current);
}
return; return;
give_sigsegv: give_sigsegv:
......
...@@ -221,6 +221,8 @@ alcor_map_irq(struct pci_dev *dev, u8 slot, u8 pin) ...@@ -221,6 +221,8 @@ alcor_map_irq(struct pci_dev *dev, u8 slot, u8 pin)
static void static void
alcor_kill_arch(int mode) alcor_kill_arch(int mode)
{ {
cia_kill_arch(mode);
switch(mode) { switch(mode) {
case LINUX_REBOOT_CMD_RESTART: case LINUX_REBOOT_CMD_RESTART:
/* Who said DEC engineer's have no sense of humor? ;-) */ /* Who said DEC engineer's have no sense of humor? ;-) */
......
...@@ -338,7 +338,6 @@ struct alpha_machine_vector cabriolet_mv __initmv = { ...@@ -338,7 +338,6 @@ struct alpha_machine_vector cabriolet_mv __initmv = {
.init_irq = cabriolet_init_irq, .init_irq = cabriolet_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = cabriolet_init_pci, .init_pci = cabriolet_init_pci,
.kill_arch = NULL,
.pci_map_irq = cabriolet_map_irq, .pci_map_irq = cabriolet_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
...@@ -366,6 +365,7 @@ struct alpha_machine_vector eb164_mv __initmv = { ...@@ -366,6 +365,7 @@ struct alpha_machine_vector eb164_mv __initmv = {
.init_irq = cabriolet_init_irq, .init_irq = cabriolet_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = cia_cab_init_pci, .init_pci = cia_cab_init_pci,
.kill_arch = cia_kill_arch,
.pci_map_irq = cabriolet_map_irq, .pci_map_irq = cabriolet_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
...@@ -417,6 +417,7 @@ struct alpha_machine_vector lx164_mv __initmv = { ...@@ -417,6 +417,7 @@ struct alpha_machine_vector lx164_mv __initmv = {
.init_irq = cabriolet_init_irq, .init_irq = cabriolet_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = alphapc164_init_pci, .init_pci = alphapc164_init_pci,
.kill_arch = cia_kill_arch,
.pci_map_irq = alphapc164_map_irq, .pci_map_irq = alphapc164_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
...@@ -442,6 +443,7 @@ struct alpha_machine_vector pc164_mv __initmv = { ...@@ -442,6 +443,7 @@ struct alpha_machine_vector pc164_mv __initmv = {
.init_irq = pc164_init_irq, .init_irq = pc164_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = alphapc164_init_pci, .init_pci = alphapc164_init_pci,
.kill_arch = cia_kill_arch,
.pci_map_irq = alphapc164_map_irq, .pci_map_irq = alphapc164_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
......
...@@ -238,6 +238,8 @@ miata_init_pci(void) ...@@ -238,6 +238,8 @@ miata_init_pci(void)
static void static void
miata_kill_arch(int mode) miata_kill_arch(int mode)
{ {
cia_kill_arch(mode);
switch(mode) { switch(mode) {
case LINUX_REBOOT_CMD_RESTART: case LINUX_REBOOT_CMD_RESTART:
/* Who said DEC engineers have no sense of humor? ;-) */ /* Who said DEC engineers have no sense of humor? ;-) */
......
...@@ -234,7 +234,6 @@ struct alpha_machine_vector mikasa_mv __initmv = { ...@@ -234,7 +234,6 @@ struct alpha_machine_vector mikasa_mv __initmv = {
.init_irq = mikasa_init_irq, .init_irq = mikasa_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = common_init_pci, .init_pci = common_init_pci,
.kill_arch = NULL,
.pci_map_irq = mikasa_map_irq, .pci_map_irq = mikasa_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
...@@ -260,6 +259,7 @@ struct alpha_machine_vector mikasa_primo_mv __initmv = { ...@@ -260,6 +259,7 @@ struct alpha_machine_vector mikasa_primo_mv __initmv = {
.init_irq = mikasa_init_irq, .init_irq = mikasa_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = cia_init_pci, .init_pci = cia_init_pci,
.kill_arch = cia_kill_arch,
.pci_map_irq = mikasa_map_irq, .pci_map_irq = mikasa_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
......
This diff is collapsed.
...@@ -316,7 +316,6 @@ struct alpha_machine_vector noritake_mv __initmv = { ...@@ -316,7 +316,6 @@ struct alpha_machine_vector noritake_mv __initmv = {
.init_irq = noritake_init_irq, .init_irq = noritake_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = common_init_pci, .init_pci = common_init_pci,
.kill_arch = NULL,
.pci_map_irq = noritake_map_irq, .pci_map_irq = noritake_map_irq,
.pci_swizzle = noritake_swizzle, .pci_swizzle = noritake_swizzle,
}; };
...@@ -342,6 +341,7 @@ struct alpha_machine_vector noritake_primo_mv __initmv = { ...@@ -342,6 +341,7 @@ struct alpha_machine_vector noritake_primo_mv __initmv = {
.init_irq = noritake_init_irq, .init_irq = noritake_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = cia_init_pci, .init_pci = cia_init_pci,
.kill_arch = cia_kill_arch,
.pci_map_irq = noritake_map_irq, .pci_map_irq = noritake_map_irq,
.pci_swizzle = noritake_swizzle, .pci_swizzle = noritake_swizzle,
}; };
......
...@@ -85,6 +85,7 @@ ruffian_init_rtc(void) ...@@ -85,6 +85,7 @@ ruffian_init_rtc(void)
static void static void
ruffian_kill_arch (int mode) ruffian_kill_arch (int mode)
{ {
cia_kill_arch(mode);
#if 0 #if 0
/* This only causes re-entry to ARCSBIOS */ /* This only causes re-entry to ARCSBIOS */
/* Perhaps this works for other PYXIS as well? */ /* Perhaps this works for other PYXIS as well? */
......
...@@ -172,7 +172,7 @@ struct alpha_machine_vector sx164_mv __initmv = { ...@@ -172,7 +172,7 @@ struct alpha_machine_vector sx164_mv __initmv = {
.init_irq = sx164_init_irq, .init_irq = sx164_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = sx164_init_pci, .init_pci = sx164_init_pci,
.kill_arch = NULL, .kill_arch = cia_kill_arch,
.pci_map_irq = sx164_map_irq, .pci_map_irq = sx164_map_irq,
.pci_swizzle = common_swizzle, .pci_swizzle = common_swizzle,
}; };
......
...@@ -286,7 +286,7 @@ struct alpha_machine_vector takara_mv __initmv = { ...@@ -286,7 +286,7 @@ struct alpha_machine_vector takara_mv __initmv = {
.init_irq = takara_init_irq, .init_irq = takara_init_irq,
.init_rtc = common_init_rtc, .init_rtc = common_init_rtc,
.init_pci = takara_init_pci, .init_pci = takara_init_pci,
.kill_arch = NULL, .kill_arch = cia_kill_arch,
.pci_map_irq = takara_map_irq, .pci_map_irq = takara_map_irq,
.pci_swizzle = takara_swizzle, .pci_swizzle = takara_swizzle,
}; };
......
...@@ -213,25 +213,25 @@ do_entArith(unsigned long summary, unsigned long write_mask, ...@@ -213,25 +213,25 @@ do_entArith(unsigned long summary, unsigned long write_mask,
unsigned long a2, unsigned long a3, unsigned long a4, unsigned long a2, unsigned long a3, unsigned long a4,
unsigned long a5, struct pt_regs regs) unsigned long a5, struct pt_regs regs)
{ {
long si_code = FPE_FLTINV;
siginfo_t info;
if (summary & 1) { if (summary & 1) {
/* Software-completion summary bit is set, so try to /* Software-completion summary bit is set, so try to
emulate the instruction. */ emulate the instruction. If the processor supports
if (!amask(AMASK_PRECISE_TRAP)) { precise exceptions, we don't have to search. */
/* 21264 (except pass 1) has precise exceptions. */ if (!amask(AMASK_PRECISE_TRAP))
if (alpha_fp_emul(regs.pc - 4)) si_code = alpha_fp_emul(regs.pc - 4);
return; else
} else { si_code = alpha_fp_emul_imprecise(&regs, write_mask);
if (alpha_fp_emul_imprecise(&regs, write_mask))
return;
}
} }
#if 0
printk("%s: arithmetic trap at %016lx: %02lx %016lx\n",
current->comm, regs.pc, summary, write_mask);
#endif
die_if_kernel("Arithmetic fault", &regs, 0, 0); die_if_kernel("Arithmetic fault", &regs, 0, 0);
send_sig(SIGFPE, current, 1);
info.si_signo = SIGFPE;
info.si_errno = 0;
info.si_code = si_code;
info.si_addr = (void *) regs.pc;
send_sig_info(SIGFPE, &info, current);
} }
asmlinkage void asmlinkage void
...@@ -239,6 +239,9 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -239,6 +239,9 @@ do_entIF(unsigned long type, unsigned long a1,
unsigned long a2, unsigned long a3, unsigned long a4, unsigned long a2, unsigned long a3, unsigned long a4,
unsigned long a5, struct pt_regs regs) unsigned long a5, struct pt_regs regs)
{ {
siginfo_t info;
int signo, code;
if (!opDEC_testing || type != 4) { if (!opDEC_testing || type != 4) {
if (type == 1) { if (type == 1) {
const unsigned int *data const unsigned int *data
...@@ -253,30 +256,64 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -253,30 +256,64 @@ do_entIF(unsigned long type, unsigned long a1,
switch (type) { switch (type) {
case 0: /* breakpoint */ case 0: /* breakpoint */
info.si_signo = SIGTRAP;
info.si_errno = 0;
info.si_code = TRAP_BRKPT;
info.si_trapno = 0;
info.si_addr = (void *) regs.pc;
if (ptrace_cancel_bpt(current)) { if (ptrace_cancel_bpt(current)) {
regs.pc -= 4; /* make pc point to former bpt */ regs.pc -= 4; /* make pc point to former bpt */
} }
send_sig(SIGTRAP, current, 1);
send_sig_info(SIGTRAP, &info, current);
return; return;
case 1: /* bugcheck */ case 1: /* bugcheck */
send_sig(SIGTRAP, current, 1); info.si_signo = SIGTRAP;
info.si_errno = 0;
info.si_code = __SI_FAULT;
info.si_addr = (void *) regs.pc;
info.si_trapno = 0;
send_sig_info(SIGTRAP, &info, current);
return; return;
case 2: /* gentrap */ case 2: /* gentrap */
/* info.si_addr = (void *) regs.pc;
* The exception code should be passed on to the signal info.si_trapno = regs.r16;
* handler as the second argument. Linux doesn't do that
* yet (also notice that Linux *always* behaves like
* DEC Unix with SA_SIGINFO off; see DEC Unix man page
* for sigaction(2)).
*/
switch ((long) regs.r16) { switch ((long) regs.r16) {
case GEN_INTOVF: case GEN_INTDIV: case GEN_FLTOVF: case GEN_INTOVF:
case GEN_FLTDIV: case GEN_FLTUND: case GEN_FLTINV: signo = SIGFPE;
case GEN_FLTINE: case GEN_ROPRAND: code = FPE_INTOVF;
send_sig(SIGFPE, current, 1); break;
return; case GEN_INTDIV:
signo = SIGFPE;
code = FPE_INTDIV;
break;
case GEN_FLTOVF:
signo = SIGFPE;
code = FPE_FLTOVF;
break;
case GEN_FLTDIV:
signo = SIGFPE;
code = FPE_FLTDIV;
break;
case GEN_FLTUND:
signo = SIGFPE;
code = FPE_FLTUND;
break;
case GEN_FLTINV:
signo = SIGFPE;
code = FPE_FLTINV;
break;
case GEN_FLTINE:
signo = SIGFPE;
code = FPE_FLTRES;
break;
case GEN_ROPRAND:
signo = SIGFPE;
code = __SI_FAULT;
break;
case GEN_DECOVF: case GEN_DECOVF:
case GEN_DECDIV: case GEN_DECDIV:
...@@ -295,13 +332,23 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -295,13 +332,23 @@ do_entIF(unsigned long type, unsigned long a1,
case GEN_SUBRNG5: case GEN_SUBRNG5:
case GEN_SUBRNG6: case GEN_SUBRNG6:
case GEN_SUBRNG7: case GEN_SUBRNG7:
send_sig(SIGTRAP, current, 1); default:
return; signo = SIGTRAP;
} code = __SI_FAULT;
break; break;
}
info.si_signo = signo;
info.si_errno = 0;
info.si_code = code;
info.si_addr = (void *) regs.pc;
send_sig_info(signo, &info, current);
return;
case 4: /* opDEC */ case 4: /* opDEC */
if (implver() == IMPLVER_EV4) { if (implver() == IMPLVER_EV4) {
long si_code;
/* The some versions of SRM do not handle /* The some versions of SRM do not handle
the opDEC properly - they return the PC of the the opDEC properly - they return the PC of the
opDEC fault, not the instruction after as the opDEC fault, not the instruction after as the
...@@ -309,8 +356,7 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -309,8 +356,7 @@ do_entIF(unsigned long type, unsigned long a1,
We do this by intentionally causing an opDEC We do this by intentionally causing an opDEC
fault during the boot sequence and testing if fault during the boot sequence and testing if
we get the correct PC. If not, we set a flag we get the correct PC. If not, we set a flag
to correct it every time through. to correct it every time through. */
*/
if (opDEC_testing) { if (opDEC_testing) {
if (regs.pc == opDEC_test_pc) { if (regs.pc == opDEC_test_pc) {
opDEC_fix = 4; opDEC_fix = 4;
...@@ -324,9 +370,18 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -324,9 +370,18 @@ do_entIF(unsigned long type, unsigned long a1,
/* EV4 does not implement anything except normal /* EV4 does not implement anything except normal
rounding. Everything else will come here as rounding. Everything else will come here as
an illegal instruction. Emulate them. */ an illegal instruction. Emulate them. */
if (alpha_fp_emul(regs.pc-4)) si_code = alpha_fp_emul(regs.pc - 4);
if (si_code == 0)
return;
if (si_code > 0) {
info.si_signo = SIGFPE;
info.si_errno = 0;
info.si_code = si_code;
info.si_addr = (void *) regs.pc;
send_sig_info(SIGFPE, &info, current);
return; return;
} }
}
break; break;
case 3: /* FEN fault */ case 3: /* FEN fault */
...@@ -347,7 +402,12 @@ do_entIF(unsigned long type, unsigned long a1, ...@@ -347,7 +402,12 @@ do_entIF(unsigned long type, unsigned long a1,
default: /* unexpected instruction-fault type */ default: /* unexpected instruction-fault type */
; ;
} }
send_sig(SIGILL, current, 1);
info.si_signo = SIGILL;
info.si_errno = 0;
info.si_code = ILL_ILLOPC;
info.si_addr = regs.pc;
send_sig_info(SIGILL, &info, current);
} }
/* There is an ifdef in the PALcode in MILO that enables a /* There is an ifdef in the PALcode in MILO that enables a
...@@ -362,8 +422,15 @@ do_entDbg(unsigned long type, unsigned long a1, ...@@ -362,8 +422,15 @@ do_entDbg(unsigned long type, unsigned long a1,
unsigned long a2, unsigned long a3, unsigned long a4, unsigned long a2, unsigned long a3, unsigned long a4,
unsigned long a5, struct pt_regs regs) unsigned long a5, struct pt_regs regs)
{ {
siginfo_t info;
die_if_kernel("Instruction fault", &regs, type, 0); die_if_kernel("Instruction fault", &regs, type, 0);
force_sig(SIGILL, current);
info.si_signo = SIGILL;
info.si_errno = 0;
info.si_code = ILL_ILLOPC;
info.si_addr = regs.pc;
force_sig_info(SIGILL, &info, current);
} }
...@@ -720,6 +787,7 @@ do_entUnaUser(void * va, unsigned long opcode, ...@@ -720,6 +787,7 @@ do_entUnaUser(void * va, unsigned long opcode,
unsigned long tmp1, tmp2, tmp3, tmp4; unsigned long tmp1, tmp2, tmp3, tmp4;
unsigned long fake_reg, *reg_addr = &fake_reg; unsigned long fake_reg, *reg_addr = &fake_reg;
siginfo_t info;
long error; long error;
/* Check the UAC bits to decide what the user wants us to do /* Check the UAC bits to decide what the user wants us to do
...@@ -984,12 +1052,34 @@ do_entUnaUser(void * va, unsigned long opcode, ...@@ -984,12 +1052,34 @@ do_entUnaUser(void * va, unsigned long opcode,
give_sigsegv: give_sigsegv:
regs->pc -= 4; /* make pc point to faulting insn */ regs->pc -= 4; /* make pc point to faulting insn */
send_sig(SIGSEGV, current, 1); info.si_signo = SIGSEGV;
info.si_errno = 0;
/* We need to replicate some of the logic in mm/fault.c,
since we don't have access to the fault code in the
exception handling return path. */
if (!__access_ok((unsigned long)va, 0, USER_DS))
info.si_code = SEGV_ACCERR;
else {
struct mm_struct *mm = current->mm;
down_read(&mm->mmap_sem);
if (find_vma(mm, (unsigned long)va))
info.si_code = SEGV_ACCERR;
else
info.si_code = SEGV_MAPERR;
up_read(&mm->mmap_sem);
}
info.si_addr = va;
send_sig_info(SIGSEGV, &info, current);
return; return;
give_sigbus: give_sigbus:
regs->pc -= 4; regs->pc -= 4;
send_sig(SIGBUS, current, 1); info.si_signo = SIGBUS;
info.si_errno = 0;
info.si_code = BUS_ADRALN;
info.si_addr = va;
send_sig_info(SIGBUS, &info, current);
return; return;
} }
......
/*
* arch/alpha/lib/dbg_current.S
* Contributed by Richard Henderson (rth@cygnus.com)
*
* Trap if we find current not correct.
*/
#include <asm/pal.h>
.text
.set noat
.globl _mcount
.ent _mcount
_mcount:
.frame $30, 0, $28, 0
.prologue 0
lda $0, -0x4000($30)
cmpult $8, $30, $1
cmpule $0, $30, $2
and $1, $2, $3
bne $3, 1f
call_pal PAL_bugchk
1: ret $31, ($28), 1
.end _mcount
...@@ -86,11 +86,13 @@ void cleanup_module(void) ...@@ -86,11 +86,13 @@ void cleanup_module(void)
/* /*
* Emulate the floating point instruction at address PC. Returns 0 if * Emulate the floating point instruction at address PC. Returns -1 if the
* emulation fails. Notice that the kernel does not and cannot use FP * instruction to be emulated is illegal (such as with the opDEC trap), else
* regs. This is good because it means that instead of * the SI_CODE for a SIGFPE signal, else 0 if everything's ok.
* saving/restoring all fp regs, we simply stick the result of the *
* operation into the appropriate register. * Notice that the kernel does not and cannot use FP regs. This is good
* because it means that instead of saving/restoring all fp regs, we simply
* stick the result of the operation into the appropriate register.
*/ */
long long
alpha_fp_emul (unsigned long pc) alpha_fp_emul (unsigned long pc)
...@@ -102,6 +104,7 @@ alpha_fp_emul (unsigned long pc) ...@@ -102,6 +104,7 @@ alpha_fp_emul (unsigned long pc)
unsigned long fa, fb, fc, func, mode, src; unsigned long fa, fb, fc, func, mode, src;
unsigned long res, va, vb, vc, swcr, fpcr; unsigned long res, va, vb, vc, swcr, fpcr;
__u32 insn; __u32 insn;
long si_code;
MOD_INC_USE_COUNT; MOD_INC_USE_COUNT;
...@@ -306,10 +309,19 @@ alpha_fp_emul (unsigned long pc) ...@@ -306,10 +309,19 @@ alpha_fp_emul (unsigned long pc)
wrfpcr(fpcr); wrfpcr(fpcr);
/* Do we generate a signal? */ /* Do we generate a signal? */
if (_fex & swcr & IEEE_TRAP_ENABLE_MASK) { _fex = _fex & swcr & IEEE_TRAP_ENABLE_MASK;
MOD_DEC_USE_COUNT; si_code = 0;
return 0; if (_fex) {
if (_fex & IEEE_TRAP_ENABLE_DNO) si_code = FPE_FLTUND;
if (_fex & IEEE_TRAP_ENABLE_INE) si_code = FPE_FLTRES;
if (_fex & IEEE_TRAP_ENABLE_UNF) si_code = FPE_FLTUND;
if (_fex & IEEE_TRAP_ENABLE_OVF) si_code = FPE_FLTOVF;
if (_fex & IEEE_TRAP_ENABLE_DZE) si_code = FPE_FLTDIV;
if (_fex & IEEE_TRAP_ENABLE_INV) si_code = FPE_FLTINV;
} }
MOD_DEC_USE_COUNT;
return si_code;
} }
/* We used to write the destination register here, but DEC FORTRAN /* We used to write the destination register here, but DEC FORTRAN
...@@ -317,20 +329,20 @@ alpha_fp_emul (unsigned long pc) ...@@ -317,20 +329,20 @@ alpha_fp_emul (unsigned long pc)
immediately after the operations above. */ immediately after the operations above. */
MOD_DEC_USE_COUNT; MOD_DEC_USE_COUNT;
return 1; return 0;
bad_insn: bad_insn:
printk(KERN_ERR "alpha_fp_emul: Invalid FP insn %#x at %#lx\n", printk(KERN_ERR "alpha_fp_emul: Invalid FP insn %#x at %#lx\n",
insn, pc); insn, pc);
MOD_DEC_USE_COUNT; MOD_DEC_USE_COUNT;
return 0; return -1;
} }
long long
alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask) alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask)
{ {
unsigned long trigger_pc = regs->pc - 4; unsigned long trigger_pc = regs->pc - 4;
unsigned long insn, opcode, rc, no_signal = 0; unsigned long insn, opcode, rc, si_code = 0;
MOD_INC_USE_COUNT; MOD_INC_USE_COUNT;
...@@ -384,7 +396,7 @@ alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask) ...@@ -384,7 +396,7 @@ alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask)
if (!write_mask) { if (!write_mask) {
/* Re-execute insns in the trap-shadow. */ /* Re-execute insns in the trap-shadow. */
regs->pc = trigger_pc + 4; regs->pc = trigger_pc + 4;
no_signal = alpha_fp_emul(trigger_pc); si_code = alpha_fp_emul(trigger_pc);
goto egress; goto egress;
} }
trigger_pc -= 4; trigger_pc -= 4;
...@@ -392,5 +404,5 @@ alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask) ...@@ -392,5 +404,5 @@ alpha_fp_emul_imprecise (struct pt_regs *regs, unsigned long write_mask)
egress: egress:
MOD_DEC_USE_COUNT; MOD_DEC_USE_COUNT;
return no_signal; return si_code;
} }
...@@ -89,7 +89,8 @@ do_page_fault(unsigned long address, unsigned long mmcsr, ...@@ -89,7 +89,8 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
struct vm_area_struct * vma; struct vm_area_struct * vma;
struct mm_struct *mm = current->mm; struct mm_struct *mm = current->mm;
unsigned int fixup; unsigned int fixup;
int fault; int fault, si_code = SEGV_MAPERR;
siginfo_t info;
/* As of EV6, a load into $31/$f31 is a prefetch, and never faults /* As of EV6, a load into $31/$f31 is a prefetch, and never faults
(or is suppressed by the PALcode). Support that for older CPUs (or is suppressed by the PALcode). Support that for older CPUs
...@@ -129,6 +130,7 @@ do_page_fault(unsigned long address, unsigned long mmcsr, ...@@ -129,6 +130,7 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
/* Ok, we have a good vm_area for this memory access, so /* Ok, we have a good vm_area for this memory access, so
we can handle it. */ we can handle it. */
good_area: good_area:
si_code = SEGV_ACCERR;
if (cause < 0) { if (cause < 0) {
if (!(vma->vm_flags & VM_EXEC)) if (!(vma->vm_flags & VM_EXEC))
goto bad_area; goto bad_area;
...@@ -148,10 +150,20 @@ do_page_fault(unsigned long address, unsigned long mmcsr, ...@@ -148,10 +150,20 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
fault = handle_mm_fault(mm, vma, address, cause > 0); fault = handle_mm_fault(mm, vma, address, cause > 0);
up_read(&mm->mmap_sem); up_read(&mm->mmap_sem);
if (fault < 0) switch (fault) {
goto out_of_memory; case VM_FAULT_MINOR:
if (fault == 0) current->min_flt++;
break;
case VM_FAULT_MAJOR:
current->maj_flt++;
break;
case VM_FAULT_SIGBUS:
goto do_sigbus; goto do_sigbus;
case VM_FAULT_OOM:
goto out_of_memory;
default:
BUG();
}
return; return;
/* Something tried to access memory that isn't in our memory map. /* Something tried to access memory that isn't in our memory map.
...@@ -159,20 +171,14 @@ do_page_fault(unsigned long address, unsigned long mmcsr, ...@@ -159,20 +171,14 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
bad_area: bad_area:
up_read(&mm->mmap_sem); up_read(&mm->mmap_sem);
if (user_mode(regs)) { if (user_mode(regs))
force_sig(SIGSEGV, current); goto do_sigsegv;
return;
}
no_context: no_context:
/* Are we prepared to handle this fault as an exception? */ /* Are we prepared to handle this fault as an exception? */
if ((fixup = search_exception_table(regs->pc, regs->gp)) != 0) { if ((fixup = search_exception_table(regs->pc, regs->gp)) != 0) {
unsigned long newpc; unsigned long newpc;
newpc = fixup_exception(dpf_reg, fixup, regs->pc); newpc = fixup_exception(dpf_reg, fixup, regs->pc);
#if 0
printk("%s: Exception at [<%lx>] (%lx) handled successfully\n",
current->comm, regs->pc, newpc);
#endif
regs->pc = newpc; regs->pc = newpc;
return; return;
} }
...@@ -201,17 +207,28 @@ do_page_fault(unsigned long address, unsigned long mmcsr, ...@@ -201,17 +207,28 @@ do_page_fault(unsigned long address, unsigned long mmcsr,
do_sigbus: do_sigbus:
/* Send a sigbus, regardless of whether we were in kernel /* Send a sigbus, regardless of whether we were in kernel
or user mode. */ or user mode. */
force_sig(SIGBUS, current); info.si_signo = SIGBUS;
info.si_errno = 0;
info.si_code = BUS_ADRERR;
info.si_addr = (void *) address;
force_sig_info(SIGBUS, &info, current);
if (!user_mode(regs)) if (!user_mode(regs))
goto no_context; goto no_context;
return; return;
do_sigsegv:
info.si_signo = SIGSEGV;
info.si_errno = 0;
info.si_code = si_code;
info.si_addr = (void *) address;
force_sig_info(SIGSEGV, &info, current);
return;
#ifdef CONFIG_ALPHA_LARGE_VMALLOC #ifdef CONFIG_ALPHA_LARGE_VMALLOC
vmalloc_fault: vmalloc_fault:
if (user_mode(regs)) { if (user_mode(regs))
force_sig(SIGSEGV, current); goto do_sigsegv;
return; else {
} else {
/* Synchronize this task's top level page-table /* Synchronize this task's top level page-table
with the "reference" page table from init. */ with the "reference" page table from init. */
long offset = __pgd_offset(address); long offset = __pgd_offset(address);
......
...@@ -67,6 +67,12 @@ SECTIONS ...@@ -67,6 +67,12 @@ SECTIONS
__initcall_end = .; __initcall_end = .;
} }
.init.ramfs ALIGN(8192): {
__initramfs_start = .;
*(.init.initramfs)
__initramfs_end = .;
}
.data.percpu ALIGN(64): { .data.percpu ALIGN(64): {
__per_cpu_start = .; __per_cpu_start = .;
*(.data.percpu) *(.data.percpu)
......
...@@ -109,6 +109,9 @@ struct percpu_struct { ...@@ -109,6 +109,9 @@ struct percpu_struct {
unsigned long ipc_buffer[21]; unsigned long ipc_buffer[21];
unsigned long palcode_avail[16]; unsigned long palcode_avail[16];
unsigned long compatibility; unsigned long compatibility;
unsigned long console_data_log_pa;
unsigned long console_data_log_length;
unsigned long bcache_info;
}; };
struct procdesc_struct { struct procdesc_struct {
......
...@@ -19,11 +19,13 @@ ...@@ -19,11 +19,13 @@
#define _MAP_UNALIGNED 0x0800 #define _MAP_UNALIGNED 0x0800
/* These are linux-specific */ /* These are linux-specific */
#define MAP_GROWSDOWN 0x1000 /* stack-like segment */ #define MAP_GROWSDOWN 0x01000 /* stack-like segment */
#define MAP_DENYWRITE 0x2000 /* ETXTBSY */ #define MAP_DENYWRITE 0x02000 /* ETXTBSY */
#define MAP_EXECUTABLE 0x4000 /* mark it as an executable */ #define MAP_EXECUTABLE 0x04000 /* mark it as an executable */
#define MAP_LOCKED 0x8000 /* lock the mapping */ #define MAP_LOCKED 0x08000 /* lock the mapping */
#define MAP_NORESERVE 0x10000 /* don't check for reservations */ #define MAP_NORESERVE 0x10000 /* don't check for reservations */
#define MAP_POPULATE 0x20000 /* populate (prefault) pagetables */
#define MAP_NONBLOCK 0x40000 /* do not block on IO */
#define MS_ASYNC 1 /* sync memory asynchronously */ #define MS_ASYNC 1 /* sync memory asynchronously */
#define MS_SYNC 2 /* synchronous memory sync */ #define MS_SYNC 2 /* synchronous memory sync */
......
#ifndef _ALPHA_SIGINFO_H #ifndef _ALPHA_SIGINFO_H
#define _ALPHA_SIGINFO_H #define _ALPHA_SIGINFO_H
#define SI_PAD_SIZE ((SI_MAX_SIZE/sizeof(int)) - 4) #define __ARCH_SI_PREAMBLE_SIZE (4 * sizeof(int))
#define __ARCH_SI_TRAPNO
#define SIGEV_PAD_SIZE ((SIGEV_MAX_SIZE/sizeof(int)) - 4) #define SIGEV_PAD_SIZE ((SIGEV_MAX_SIZE/sizeof(int)) - 4)
#define HAVE_ARCH_COPY_SIGINFO
#include <asm-generic/siginfo.h> #include <asm-generic/siginfo.h>
#ifdef __KERNEL__
#include <linux/string.h>
extern inline void copy_siginfo(siginfo_t *to, siginfo_t *from)
{
if (from->si_code < 0)
memcpy(to, from, sizeof(siginfo_t));
else
/* _sigchld is currently the largest know union member */
memcpy(to, from, 4*sizeof(int) + sizeof(from->_sifields._sigchld));
}
#endif /* __KERNEL__ */
#endif #endif
...@@ -61,7 +61,8 @@ struct el_common { ...@@ -61,7 +61,8 @@ struct el_common {
int retry : 1; /* retry flag */ int retry : 1; /* retry flag */
unsigned int proc_offset; /* processor-specific offset */ unsigned int proc_offset; /* processor-specific offset */
unsigned int sys_offset; /* system-specific offset */ unsigned int sys_offset; /* system-specific offset */
unsigned long code; /* machine check code */ unsigned int code; /* machine check code */
unsigned int frame_rev; /* frame revision */
}; };
/* Machine Check Frame for uncorrectable errors (Large format) /* Machine Check Frame for uncorrectable errors (Large format)
...@@ -117,11 +118,11 @@ struct el_common_EV6_mcheck { ...@@ -117,11 +118,11 @@ struct el_common_EV6_mcheck {
unsigned long DC0_SYNDROME; unsigned long DC0_SYNDROME;
unsigned long C_STAT; unsigned long C_STAT;
unsigned long C_STS; unsigned long C_STS;
unsigned long RESERVED0; unsigned long MM_STAT;
unsigned long EXC_ADDR; unsigned long EXC_ADDR;
unsigned long IER_CM; unsigned long IER_CM;
unsigned long ISUM; unsigned long ISUM;
unsigned long MM_STAT; unsigned long RESERVED0;
unsigned long PAL_BASE; unsigned long PAL_BASE;
unsigned long I_CTL; unsigned long I_CTL;
unsigned long PCTX; unsigned long PCTX;
......
...@@ -552,9 +552,9 @@ static inline long close(int fd) ...@@ -552,9 +552,9 @@ static inline long close(int fd)
} }
extern off_t sys_lseek(int, off_t, int); extern off_t sys_lseek(int, off_t, int);
static inline off_t lseek(int fd, off_t off, int whense) static inline off_t lseek(int fd, off_t off, int whence)
{ {
return sys_lseek(fd, off, whense); return sys_lseek(fd, off, whence);
} }
extern long sys_exit(int); extern long sys_exit(int);
...@@ -565,14 +565,14 @@ static inline long _exit(int value) ...@@ -565,14 +565,14 @@ static inline long _exit(int value)
#define exit(x) _exit(x) #define exit(x) _exit(x)
extern long sys_write(int, const char *, int); extern long sys_write(int, const char *, size_t);
static inline long write(int fd, const char * buf, int nr) static inline long write(int fd, const char * buf, size_t nr)
{ {
return sys_write(fd, buf, nr); return sys_write(fd, buf, nr);
} }
extern long sys_read(int, char *, int); extern long sys_read(int, char *, size_t);
static inline long read(int fd, char * buf, int nr) static inline long read(int fd, char * buf, size_t nr)
{ {
return sys_read(fd, buf, nr); return sys_read(fd, buf, nr);
} }
......
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include <linux/mount.h> #include <linux/mount.h>
#include <linux/list.h> #include <linux/list.h>
#include <linux/spinlock.h> #include <linux/spinlock.h>
#include <linux/cache.h>
#include <asm/page.h> /* for BUG() */ #include <asm/page.h> /* for BUG() */
/* /*
......
...@@ -26,7 +26,7 @@ asmlinkage long sys_mkdir(char *name, int mode); ...@@ -26,7 +26,7 @@ asmlinkage long sys_mkdir(char *name, int mode);
asmlinkage long sys_mknod(char *name, int mode, dev_t dev); asmlinkage long sys_mknod(char *name, int mode, dev_t dev);
asmlinkage long sys_symlink(char *old, char *new); asmlinkage long sys_symlink(char *old, char *new);
asmlinkage long sys_link(char *old, char *new); asmlinkage long sys_link(char *old, char *new);
asmlinkage long sys_write(int fd, void *buf, ssize_t size); asmlinkage long sys_write(int fd, const char *buf, size_t size);
asmlinkage long sys_chown(char *name, uid_t uid, gid_t gid); asmlinkage long sys_chown(char *name, uid_t uid, gid_t gid);
asmlinkage long sys_lchown(char *name, uid_t uid, gid_t gid); asmlinkage long sys_lchown(char *name, uid_t uid, gid_t gid);
asmlinkage long sys_fchown(int fd, uid_t uid, gid_t gid); asmlinkage long sys_fchown(int fd, uid_t uid, gid_t gid);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment