aboutsummaryrefslogtreecommitdiff
path: root/sim/d10v
diff options
context:
space:
mode:
Diffstat (limited to 'sim/d10v')
-rw-r--r--sim/d10v/simops.c208
1 files changed, 192 insertions, 16 deletions
diff --git a/sim/d10v/simops.c b/sim/d10v/simops.c
index 6df10cf..2c825a5 100644
--- a/sim/d10v/simops.c
+++ b/sim/d10v/simops.c
@@ -1,6 +1,12 @@
#include <signal.h>
+#include <errno.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <unistd.h>
+
#include "d10v_sim.h"
#include "simops.h"
+#include "syscall.h"
/* #define DEBUG 1 */
@@ -1454,8 +1460,6 @@ OP_5201 ()
else
tmp = ((State.a[0] << 16) | (State.a[1] & 0xffff)) >> -shift;
tmp = ( SEXT60(tmp) + 0x8000 ) >> 16;
- printf("tmp=0x%llx\n",tmp);
- /*
if (tmp > MAX32)
{
State.regs[OP[0]] = 0x7fff;
@@ -1469,7 +1473,6 @@ OP_5201 ()
State.F0 = 1;
}
else
- */
{
State.regs[OP[0]] = (tmp >> 16) & 0xffff;
State.regs[OP[0]+1] = tmp & 0xffff;
@@ -1492,8 +1495,7 @@ OP_4201 ()
else
tmp = SEXT44 (State.a[1]) >> -shift;
tmp += 0x8000;
- printf("tmp=0x%llx\n",tmp);
- /*
+
if (tmp > MAX32)
{
State.regs[OP[0]] = 0x7fff;
@@ -1505,7 +1507,6 @@ OP_4201 ()
State.F0 = 1;
}
else
- */
{
State.regs[OP[0]] = (tmp >> 16) & 0xffff;
State.F0 = 0;
@@ -2146,17 +2147,192 @@ OP_5F00 ()
printf(" trap\t%d\n",OP[0]);
#endif
- /* for now, trap is used for simulating IO */
-
- if (OP[0] == 0)
+ switch (OP[0])
{
- char *fstr = State.regs[2] + State.imem;
- printf (fstr,State.regs[3],State.regs[4],State.regs[5]);
- }
- else if (OP[0] == 1 )
- {
- char *fstr = State.regs[2] + State.imem;
- puts (fstr);
+ default:
+ fprintf (stderr, "Unknown trap code %d\n", OP[0]);
+ State.exception = SIGILL;
+
+ case 0:
+ /* Trap 0 is used for simulating low-level I/O */
+ {
+ int save_errno = errno;
+ errno = 0;
+
+/* Registers passed to trap 0 */
+
+#define FUNC State.regs[2] /* function number, return value */
+#define PARM1 State.regs[3] /* optional parm 1 */
+#define PARM2 State.regs[4] /* optional parm 2 */
+#define PARM3 State.regs[5] /* optional parm 3 */
+
+/* Registers set by trap 0 */
+
+#define RETVAL State.regs[2] /* return value */
+#define RETERR State.regs[3] /* return error code */
+
+/* Turn a pointer in a register into a pointer into real memory. */
+
+#define MEMPTR(x) ((char *)((x) + State.imem))
+
+ switch (FUNC)
+ {
+#if !defined(__GO32__) && !defined(_WIN32)
+#ifdef SYS_fork
+ case SYS_fork:
+ RETVAL = fork ();
+ break;
+#endif
+#ifdef SYS_execve
+ case SYS_execve:
+ RETVAL = execve (MEMPTR (PARM1), (char **) MEMPTR (PARM2),
+ (char **)MEMPTR (PARM3));
+ break;
+#endif
+#ifdef SYS_execv
+ case SYS_execv:
+ RETVAL = execve (MEMPTR (PARM1), (char **) MEMPTR (PARM2), NULL);
+ break;
+#endif
+#ifdef SYS_pipe
+ case SYS_pipe:
+ {
+ reg_t buf;
+ int host_fd[2];
+
+ buf = PARM1;
+ RETVAL = pipe (host_fd);
+ SW (buf, host_fd[0]);
+ buf += sizeof(uint16);
+ SW (buf, host_fd[1]);
+ }
+ break;
+#endif
+#ifdef SYS_wait
+ case SYS_wait:
+ {
+ int status;
+
+ RETVAL = wait (&status);
+ SW (PARM1, status);
+ }
+ break;
+#endif
+#endif
+
+#ifdef SYS_read
+ case SYS_read:
+ RETVAL = d10v_callback->read (d10v_callback, PARM1, MEMPTR (PARM2),
+ PARM3);
+ break;
+#endif
+#ifdef SYS_write
+ case SYS_write:
+ if (PARM1 == 1)
+ RETVAL = (int)d10v_callback->write_stdout (d10v_callback,
+ MEMPTR (PARM2), PARM3);
+ else
+ RETVAL = (int)d10v_callback->write (d10v_callback, PARM1,
+ MEMPTR (PARM2), PARM3);
+ break;
+#endif
+#ifdef SYS_lseek
+ case SYS_lseek:
+ RETVAL = d10v_callback->lseek (d10v_callback, PARM1, PARM2, PARM3);
+ break;
+#endif
+#ifdef SYS_close
+ case SYS_close:
+ RETVAL = d10v_callback->close (d10v_callback, PARM1);
+ break;
+#endif
+#ifdef SYS_open
+ case SYS_open:
+ RETVAL = d10v_callback->open (d10v_callback, MEMPTR (PARM1), PARM2);
+ break;
+#endif
+#ifdef SYS_exit
+ case SYS_exit:
+ /* EXIT - caller can look in PARM1 to work out the
+ reason */
+ State.exception = SIGQUIT;
+ break;
+#endif
+
+#ifdef SYS_stat: /* added at hmsi *
+ case SYS_stat: /* added at hmsi */
+ /* stat system call */
+ {
+ struct stat host_stat;
+ reg_t buf;
+
+ RETVAL = stat (MEMPTR (PARM1), &host_stat);
+
+ buf = PARM2;
+
+ /* The hard-coded offsets and sizes were determined by using
+ * the D10V compiler on a test program that used struct stat.
+ */
+ SW (buf, host_stat.st_dev);
+ SW (buf+2, host_stat.st_ino);
+ SW (buf+4, host_stat.st_mode);
+ SW (buf+6, host_stat.st_nlink);
+ SW (buf+8, host_stat.st_uid);
+ SW (buf+10, host_stat.st_gid);
+ SW (buf+12, host_stat.st_rdev);
+ SLW (buf+16, host_stat.st_size);
+ SLW (buf+20, host_stat.st_atime);
+ SLW (buf+28, host_stat.st_mtime);
+ SLW (buf+36, host_stat.st_ctime);
+ }
+ break;
+#endif
+
+#ifdef SYS_chown
+ case SYS_chown:
+ RETVAL = chown (MEMPTR (PARM1), PARM2, PARM3);
+ break;
+#endif
+#ifdef SYS_chmod
+ case SYS_chmod:
+ RETVAL = chmod (MEMPTR (PARM1), PARM2);
+ break;
+#endif
+#ifdef SYS_utime
+ case SYS_utime:
+ /* Cast the second argument to void *, to avoid type mismatch
+ if a prototype is present. */
+ RETVAL = utime (MEMPTR (PARM1), (void *) MEMPTR (PARM2));
+ break;
+#endif
+ default:
+ abort ();
+ }
+ RETERR = errno;
+ errno = save_errno;
+ break;
+ }
+
+ case 1:
+ /* Trap 1 prints a string */
+ {
+ char *fstr = State.regs[2] + State.imem;
+ fputs (fstr, stdout);
+ break;
+ }
+
+ case 2:
+ /* Trap 2 calls printf */
+ {
+ char *fstr = State.regs[2] + State.imem;
+ printf (fstr,State.regs[3],State.regs[4],State.regs[5]);
+ break;
+ }
+
+ case 3:
+ /* Trap 3 writes a character */
+ putchar (State.regs[2]);
+ break;
}
}