CRIS support by Hinko Kocevar (hinko.kocevar AT cetrtapot.si)
* configure.ac: Make it recognize cris.
* process.c: Define ARG_xxx constants for cris.
(change_syscall): Add support for cris.
(struct_user_offsets): Add cris-specific data.
* signal.c (sys_sigreturn): Add support for cris.
* syscall.c (get_scno): Add support for cris.
(syscall_fixup): Add support for cris.
(get_error): Add support for cris.
(syscall_enter): Add support for cris.
(force_result): While at it, fix cpp directives indentation.
* util.c (printcall): Add support for cris.
diff --git a/syscall.c b/syscall.c
index 9722519..128622d 100644
--- a/syscall.c
+++ b/syscall.c
@@ -774,6 +774,8 @@
 	static long r9;
 #elif defined(X86_64)
 	static long rax;
+#elif defined(CRISV10) || defined(CRISV32)
+	static long r10;
 #endif
 #endif /* LINUX */
 #ifdef FREEBSD
@@ -786,7 +788,7 @@
 	long scno = 0;
 
 #ifdef LINUX
-#if defined(S390) || defined(S390X)
+# if defined(S390) || defined(S390X)
 	if (tcp->flags & TCB_WAITEXECVE) {
 		/*
 		 * When the execve system call completes successfully, the
@@ -874,11 +876,11 @@
 			scno = ptrace(PTRACE_PEEKTEXT, tcp->pid, svc_addr, 0);
 			if (errno)
 				return -1;
-#if defined(S390X)
+#  if defined(S390X)
 			scno >>= 48;
-#else
+#  else
 			scno >>= 16;
-#endif
+#  endif
 			tmp = 0;
 			offset_reg = (opcode & 0x00f00000) >> 20;
 			if (offset_reg && (upeek(tcp, gpr_offset[offset_reg], &tmp) < 0))
@@ -887,7 +889,7 @@
 			scno = (scno | tmp) & 0xff;
 		}
 	}
-#elif defined (POWERPC)
+# elif defined (POWERPC)
 	if (upeek(tcp, sizeof(unsigned long)*PT_R0, &scno) < 0)
 		return -1;
 	if (!(tcp->flags & TCB_INSYSCALL)) {
@@ -897,16 +899,16 @@
 			return 0;
 		}
 	}
-#elif defined(BFIN)
+# elif defined(BFIN)
 	if (upeek(tcp, PT_ORIG_P0, &scno))
 		return -1;
 	/* Check if we return from execve. */
 	if (tcp->flags & TCB_WAITEXECVE && tcp->flags & TCB_INSYSCALL)
 		tcp->flags &= ~(TCB_INSYSCALL | TCB_WAITEXECVE);
-#elif defined (I386)
+# elif defined (I386)
 	if (upeek(tcp, 4*ORIG_EAX, &scno) < 0)
 		return -1;
-#elif defined (X86_64)
+# elif defined (X86_64)
 	if (upeek(tcp, 8*ORIG_RAX, &scno) < 0)
 		return -1;
 
@@ -933,7 +935,7 @@
 				currpers = current_personality;
 				break;
 		}
-#if 0
+#  if 0
 		/* This version analyzes the opcode of a syscall instruction.
 		 * (int 0x80 on i386 vs. syscall on x86-64)
 		 * It works, but is too complicated.
@@ -964,7 +966,7 @@
 					"PID=%d\n", (int)call, pid);
 				break;
 		}
-#endif
+#  endif
 		if (currpers != current_personality) {
 			static const char *const names[] = {"64 bit", "32 bit"};
 			set_personality(currpers);
@@ -972,7 +974,7 @@
 					pid, names[current_personality]);
 		}
 	}
-#elif defined(IA64)
+# elif defined(IA64)
 #	define IA64_PSR_IS	((long)1 << 34)
 	if (upeek(tcp, PT_CR_IPSR, &psr) >= 0)
 		ia32 = (psr & IA64_PSR_IS) != 0;
@@ -986,7 +988,7 @@
 		}
 		/* Check if we return from execve. */
 		if (tcp->flags & TCB_WAITEXECVE) {
-#if defined PTRACE_GETSIGINFO
+#  if defined PTRACE_GETSIGINFO
 			siginfo_t si;
 
 			tcp->flags &= ~TCB_WAITEXECVE;
@@ -1005,10 +1007,10 @@
 			ptrace(PTRACE_GETSIGINFO, tcp->pid, (void*) 0, (void*) &si);
 			if (si.si_code == SI_USER)
 				return 0;
-#else
+#  else
 			tcp->flags &= ~TCB_WAITEXECVE;
 			return 0;
-#endif
+#  endif
 		}
 	} else {
 		/* syscall in progress */
@@ -1017,7 +1019,7 @@
 		if (upeek(tcp, PT_R10, &r10) < 0)
 			return -1;
 	}
-#elif defined (ARM)
+# elif defined (ARM)
 	/*
 	 * Read complete register set in one go.
 	 */
@@ -1097,10 +1099,10 @@
 			tcp->flags |= TCB_INSYSCALL;
 		}
 	}
-#elif defined (M68K)
+# elif defined (M68K)
 	if (upeek(tcp, 4*PT_ORIG_D0, &scno) < 0)
 		return -1;
-#elif defined (LINUX_MIPSN32)
+# elif defined (LINUX_MIPSN32)
 	unsigned long long regs[38];
 
 	if (do_ptrace(PTRACE_GETREGS, tcp, NULL, (long) &regs) < 0)
@@ -1125,7 +1127,7 @@
 			}
 		}
 	}
-#elif defined (MIPS)
+# elif defined (MIPS)
 	if (upeek(tcp, REG_A3, &a3) < 0)
 	  	return -1;
 	if(!(tcp->flags & TCB_INSYSCALL)) {
@@ -1149,7 +1151,7 @@
 	  	if (upeek(tcp, REG_V0, &r2) < 0)
 	    		return -1;
 	}
-#elif defined (ALPHA)
+# elif defined (ALPHA)
 	if (upeek(tcp, REG_A3, &a3) < 0)
 		return -1;
 
@@ -1179,7 +1181,7 @@
 		if (upeek(tcp, REG_R0, &r0) < 0)
 			return -1;
 	}
-#elif defined (SPARC) || defined (SPARC64)
+# elif defined (SPARC) || defined (SPARC64)
 	/* Everything we need is in the current register set. */
 	if (ptrace(PTRACE_GETREGS, tcp->pid, (char *)&regs, 0) < 0)
 		return -1;
@@ -1189,9 +1191,9 @@
 		/* Retrieve the syscall trap instruction. */
 		errno = 0;
 		trap = ptrace(PTRACE_PEEKTEXT, tcp->pid, (char *)regs.r_pc, 0);
-#if defined(SPARC64)
+#  if defined(SPARC64)
 		trap >>= 32;
-#endif
+#  endif
 		if (errno)
 			return -1;
 
@@ -1227,11 +1229,11 @@
 				tcp->flags &= ~TCB_WAITEXECVE;
 				return 0;
 			}
-#if defined (SPARC64)
+#  if defined (SPARC64)
 			fprintf(stderr,"syscall: unknown syscall trap %08lx %016lx\n", trap, regs.r_tpc);
-#else
+#  else
 			fprintf(stderr,"syscall: unknown syscall trap %08x %08x\n", trap, regs.r_pc);
-#endif
+#  endif
 			return -1;
 		}
 
@@ -1245,7 +1247,7 @@
 			memmove (&regs.r_o0, &regs.r_o1, 7*sizeof(regs.r_o0));
 		}
 	}
-#elif defined(HPPA)
+# elif defined(HPPA)
 	if (upeek(tcp, PT_GR20, &scno) < 0)
 		return -1;
 	if (!(tcp->flags & TCB_INSYSCALL)) {
@@ -1255,7 +1257,7 @@
 			return 0;
 		}
 	}
-#elif defined(SH)
+# elif defined(SH)
 	/*
 	 * In the new syscall ABI, the system call number is in R3.
 	 */
@@ -1283,7 +1285,7 @@
 			return 0;
 		}
 	}
-#elif defined(SH64)
+# elif defined(SH64)
 	if (upeek(tcp, REG_SYSCALL, &scno) < 0)
 		return -1;
 	scno &= 0xFFFF;
@@ -1295,8 +1297,12 @@
 			return 0;
 		}
 	}
-#endif /* SH64 */
+# elif defined(CRISV10) || defined(CRISV32)
+	if (upeek(tcp, 4*PT_R9, &scno) < 0)
+		return -1;
+# endif
 #endif /* LINUX */
+
 #ifdef SUNOS4
 	if (upeek(tcp, uoff(u_arg[7]), &scno) < 0)
 		return -1;
@@ -1308,15 +1314,15 @@
 	/* ABI defines result returned in r9 */
 	if (upeek(tcp, REG_GENERAL(9), (long *)&r9) < 0)
 		return -1;
-
 #endif
+
 #ifdef USE_PROCFS
-#ifdef HAVE_PR_SYSCALL
+# ifdef HAVE_PR_SYSCALL
 	scno = tcp->status.PR_SYSCALL;
-#else /* !HAVE_PR_SYSCALL */
-#ifndef FREEBSD
+# else
+#  ifndef FREEBSD
 	scno = tcp->status.PR_WHAT;
-#else /* FREEBSD */
+#  else
 	if (pread(tcp->pfd_reg, &regs, sizeof(regs), 0) < 0) {
 		perror("pread");
 		return -1;
@@ -1330,9 +1336,10 @@
 		scno = regs.r_eax;
 		break;
 	}
-#endif /* FREEBSD */
-#endif /* !HAVE_PR_SYSCALL */
+#  endif /* FREEBSD */
+# endif /* !HAVE_PR_SYSCALL */
 #endif /* USE_PROCFS */
+
 	if (!(tcp->flags & TCB_INSYSCALL))
 		tcp->scno = scno;
 	return 1;
@@ -1359,8 +1366,7 @@
  *    ("????" etc) and bail out.
  */
 static int
-syscall_fixup(tcp)
-struct tcb *tcp;
+syscall_fixup(struct tcb *tcp)
 {
 #ifdef USE_PROCFS
 	int scno = known_scno(tcp);
@@ -1498,6 +1504,14 @@
 			fprintf(stderr, "stray syscall exit: r8 = %ld\n", r8);
 		return 0;
 	}
+#elif defined(CRISV10) || defined(CRISV32)
+	if (upeek(tcp->pid, 4*PT_R10, &r10) < 0)
+		return -1;
+	if (r10 != -ENOSYS && !(tcp->flags & TCB_INSYSCALL)) {
+		if (debug)
+			fprintf(stderr, "stray syscall exit: r10 = %ld\n", r10);
+		return 0;
+	}
 #endif
 #endif /* LINUX */
 	return 1;
@@ -1521,12 +1535,11 @@
 #endif
 
 static int
-get_error(tcp)
-struct tcb *tcp;
+get_error(struct tcb *tcp)
 {
 	int u_error = 0;
 #ifdef LINUX
-#if defined(S390) || defined(S390X)
+# if defined(S390) || defined(S390X)
 	if (is_negated_errno(gpr2)) {
 		tcp->u_rval = -1;
 		u_error = -gpr2;
@@ -1535,8 +1548,7 @@
 		tcp->u_rval = gpr2;
 		u_error = 0;
 	}
-#else /* !S390 && !S390X */
-#ifdef I386
+# elif defined(I386)
 	if (is_negated_errno(eax)) {
 		tcp->u_rval = -1;
 		u_error = -eax;
@@ -1545,8 +1557,7 @@
 		tcp->u_rval = eax;
 		u_error = 0;
 	}
-#else /* !I386 */
-#ifdef X86_64
+# elif defined(X86_64)
 	if (is_negated_errno(rax)) {
 		tcp->u_rval = -1;
 		u_error = -rax;
@@ -1555,8 +1566,7 @@
 		tcp->u_rval = rax;
 		u_error = 0;
 	}
-#else
-#ifdef IA64
+# elif defined(IA64)
 	if (ia32) {
 		int err;
 
@@ -1578,8 +1588,7 @@
 			u_error = 0;
 		}
 	}
-#else /* !IA64 */
-#ifdef MIPS
+# elif defined(MIPS)
 		if (a3) {
 		  	tcp->u_rval = -1;
 			u_error = r2;
@@ -1587,8 +1596,7 @@
 		  	tcp->u_rval = r2;
 			u_error = 0;
 		}
-#else
-#ifdef POWERPC
+# elif defined(POWERPC)
 		if (is_negated_errno(result)) {
 			tcp->u_rval = -1;
 			u_error = -result;
@@ -1597,8 +1605,7 @@
 			tcp->u_rval = result;
 			u_error = 0;
 		}
-#else /* !POWERPC */
-#ifdef M68K
+# elif defined(M68K)
 		if (is_negated_errno(d0)) {
 			tcp->u_rval = -1;
 			u_error = -d0;
@@ -1607,8 +1614,7 @@
 			tcp->u_rval = d0;
 			u_error = 0;
 		}
-#else /* !M68K */
-#ifdef ARM
+# elif defined(ARM)
 		if (is_negated_errno(regs.ARM_r0)) {
 			tcp->u_rval = -1;
 			u_error = -regs.ARM_r0;
@@ -1617,8 +1623,7 @@
 			tcp->u_rval = regs.ARM_r0;
 			u_error = 0;
 		}
-#else /* !ARM */
-#ifdef BFIN
+# elif defined(BFIN)
 		if (is_negated_errno(r0)) {
 			tcp->u_rval = -1;
 			u_error = -r0;
@@ -1626,8 +1631,7 @@
 			tcp->u_rval = r0;
 			u_error = 0;
 		}
-#else /* !BFIN */
-#ifdef ALPHA
+# elif defined(ALPHA)
 		if (a3) {
 			tcp->u_rval = -1;
 			u_error = r0;
@@ -1636,8 +1640,7 @@
 			tcp->u_rval = r0;
 			u_error = 0;
 		}
-#else /* !ALPHA */
-#ifdef SPARC
+# elif defined(SPARC)
 		if (regs.r_psr & PSR_C) {
 			tcp->u_rval = -1;
 			u_error = regs.r_o0;
@@ -1646,8 +1649,7 @@
 			tcp->u_rval = regs.r_o0;
 			u_error = 0;
 		}
-#else /* !SPARC */
-#ifdef SPARC64
+# elif defined(SPARC64)
 		if (regs.r_tstate & 0x1100000000UL) {
 			tcp->u_rval = -1;
 			u_error = regs.r_o0;
@@ -1656,8 +1658,7 @@
 			tcp->u_rval = regs.r_o0;
 			u_error = 0;
 		}
-#else /* !SPARC64 */
-#ifdef HPPA
+# elif defined(HPPA)
 		if (is_negated_errno(r28)) {
 			tcp->u_rval = -1;
 			u_error = -r28;
@@ -1666,8 +1667,7 @@
 			tcp->u_rval = r28;
 			u_error = 0;
 		}
-#else
-#ifdef SH
+# elif defined(SH)
 		/* interpret R0 as return value or error number */
 		if (is_negated_errno(r0)) {
 			tcp->u_rval = -1;
@@ -1677,8 +1677,7 @@
 			tcp->u_rval = r0;
 			u_error = 0;
 		}
-#else
-#ifdef SH64
+# elif defined(SH64)
 		/* interpret result as return value or error number */
 		if (is_negated_errno(r9)) {
 			tcp->u_rval = -1;
@@ -1688,21 +1687,16 @@
 			tcp->u_rval = r9;
 			u_error = 0;
 		}
-#endif /* SH64 */
-#endif /* SH */
-#endif /* HPPA */
-#endif /* SPARC */
-#endif /* SPARC64 */
-#endif /* ALPHA */
-#endif /* BFIN */
-#endif /* ARM */
-#endif /* M68K */
-#endif /* POWERPC */
-#endif /* MIPS */
-#endif /* IA64 */
-#endif /* X86_64 */
-#endif /* I386 */
-#endif /* S390 || S390X */
+# elif defined(CRISV10) || defined(CRISV32)
+		if (r10 && (unsigned) -r10 < nerrnos) {
+			tcp->u_rval = -1;
+			u_error = -r10;
+		}
+		else {
+			tcp->u_rval = r10;
+			u_error = 0;
+		}
+# endif
 #endif /* LINUX */
 #ifdef SUNOS4
 		/* get error code from user struct */
@@ -1786,22 +1780,19 @@
 	long rval;
 {
 #ifdef LINUX
-#if defined(S390) || defined(S390X)
+# if defined(S390) || defined(S390X)
 	gpr2 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)PT_GPR2, gpr2) < 0)
 		return -1;
-#else /* !S390 && !S390X */
-#ifdef I386
+# elif defined(I386)
 	eax = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(EAX * 4), eax) < 0)
 		return -1;
-#else /* !I386 */
-#ifdef X86_64
+# elif defined(X86_64)
 	rax = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(RAX * 8), rax) < 0)
 		return -1;
-#else
-#ifdef IA64
+# elif defined(IA64)
 	if (ia32) {
 		r8 = error ? -error : rval;
 		if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(PT_R8), r8) < 0)
@@ -1820,13 +1811,11 @@
 		    ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(PT_R10), r10) < 0)
 			return -1;
 	}
-#else /* !IA64 */
-#ifdef BFIN
+# elif defined(BFIN)
 	r0 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)PT_R0, r0) < 0)
 		return -1;
-#else /* !BFIN */
-#ifdef MIPS
+# elif defined(MIPS)
 	if (error) {
 		r2 = error;
 		a3 = -1;
@@ -1839,8 +1828,7 @@
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(REG_A3), a3) < 0 ||
 	    ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(REG_V0), r2) < 0)
 	    	return -1;
-#else
-#ifdef POWERPC
+# elif defined(POWERPC)
 	if (upeek(tcp, sizeof(unsigned long)*PT_CCR, &flags) < 0)
 		return -1;
 	if (error) {
@@ -1854,18 +1842,15 @@
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(sizeof(unsigned long)*PT_CCR), flags) < 0 ||
 	    ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(sizeof(unsigned long)*PT_R3), result) < 0)
 		return -1;
-#else /* !POWERPC */
-#ifdef M68K
+# elif defined(M68K)
 	d0 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(4*PT_D0), d0) < 0)
 		return -1;
-#else /* !M68K */
-#ifdef ARM
+# elif defined(ARM)
 	regs.ARM_r0 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(4*0), regs.ARM_r0) < 0)
 		return -1;
-#else /* !ARM */
-#ifdef ALPHA
+# elif defined(ALPHA)
 	if (error) {
 		a3 = -1;
 		r0 = error;
@@ -1877,8 +1862,7 @@
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(REG_A3), a3) < 0 ||
 	    ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(REG_R0), r0) < 0)
 		return -1;
-#else /* !ALPHA */
-#ifdef SPARC
+# elif defined(SPARC)
 	if (ptrace(PTRACE_GETREGS, tcp->pid, (char *)&regs, 0) < 0)
 		return -1;
 	if (error) {
@@ -1891,8 +1875,7 @@
 	}
 	if (ptrace(PTRACE_SETREGS, tcp->pid, (char *)&regs, 0) < 0)
 		return -1;
-#else /* !SPARC */
-#ifdef SPARC64
+# elif defined(SPARC64)
 	if (ptrace(PTRACE_GETREGS, tcp->pid, (char *)&regs, 0) < 0)
 		return -1;
 	if (error) {
@@ -1905,37 +1888,21 @@
 	}
 	if (ptrace(PTRACE_SETREGS, tcp->pid, (char *)&regs, 0) < 0)
 		return -1;
-#else /* !SPARC64 */
-#ifdef HPPA
+# elif defined(HPPA)
 	r28 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(PT_GR28), r28) < 0)
 		return -1;
-#else
-#ifdef SH
+# elif defined(SH)
 	r0 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)(4*REG_REG0), r0) < 0)
 		return -1;
-#else
-#ifdef SH64
+# elif defined(SH64)
 	r9 = error ? -error : rval;
 	if (ptrace(PTRACE_POKEUSER, tcp->pid, (char*)REG_GENERAL(9), r9) < 0)
 		return -1;
-#endif /* SH64 */
-#endif /* SH */
-#endif /* HPPA */
-#endif /* SPARC */
-#endif /* SPARC64 */
-#endif /* ALPHA */
-#endif /* ARM */
-#endif /* M68K */
-#endif /* POWERPC */
-#endif /* MIPS */
-#endif /* BFIN */
-#endif /* IA64 */
-#endif /* X86_64 */
-#endif /* I386 */
-#endif /* S390 || S390X */
+# endif
 #endif /* LINUX */
+
 #ifdef SUNOS4
 	if (do_ptrace(PTRACE_POKEUSER, tcp->pid, (char*)uoff(u_error), error << 24) < 0
 	 || do_ptrace(PTRACE_POKEUSER, tcp->pid, (char*)uoff(u_rval1), rval) < 0
@@ -1943,10 +1910,12 @@
 		return -1;
 	}
 #endif /* SUNOS4 */
+
 #ifdef SVR4
 	/* XXX no clue */
 	return -1;
 #endif /* SVR4 */
+
 #ifdef FREEBSD
 	if (pread(tcp->pfd_reg, &regs, sizeof(regs), 0) < 0) {
 		perror("pread");
@@ -1973,8 +1942,7 @@
 }
 
 static int
-syscall_enter(tcp)
-struct tcb *tcp;
+syscall_enter(struct tcb *tcp)
 {
 #ifdef LINUX
 #if defined(S390) || defined(S390X)
@@ -2108,9 +2076,9 @@
 		}
 	}
 #elif defined (POWERPC)
-#ifndef PT_ORIG_R3
-#define PT_ORIG_R3 34
-#endif
+# ifndef PT_ORIG_R3
+#  define PT_ORIG_R3 34
+# endif
 	{
 		int i;
 		if (tcp->scno >= 0 && tcp->scno < nsyscalls && sysent[tcp->scno].nargs != -1)
@@ -2227,6 +2195,23 @@
 				return -1;
 		}
 	}
+#elif defined(CRISV10) || defined(CRISV32)
+	{
+		int i;
+		static const int crisregs[] = {
+			4*PT_ORIG_R10, 4*PT_R11, 4*PT_R12,
+			4*PT_R13, 4*PT_MOF, 4*PT_SRP
+		};
+
+		if (tcp->scno >= 0 && tcp->scno < nsyscalls)
+			tcp->u_nargs = sysent[tcp->scno].nargs;
+		else
+			tcp->u_nargs = 0;
+		for (i = 0; i < tcp->u_nargs; i++) {
+			if (upeek(tcp->pid, crisregs[i], &tcp->u_arg[i]) < 0)
+				return -1;
+		}
+	}
 #else /* Other architecture (like i386) (32bits specific) */
 	{
 		int i;
@@ -2692,7 +2677,7 @@
 #elif defined(IA64)
 	if (upeek(tcp, PT_R9, &val) < 0)
 		return -1;
-#endif /* SPARC || SPARC64 */
+#endif
 #endif /* LINUX */
 
 #ifdef SUNOS4
@@ -2714,6 +2699,7 @@
 	val = tcp->status.PR_REG[CTX_V1];
 #endif /* MIPS */
 #endif /* SVR4 */
+
 #ifdef FREEBSD
 	struct reg regs;
 	pread(tcp->pfd_reg, &regs, sizeof(regs), 0);