mirror of
				https://github.com/espressif/binutils-gdb.git
				synced 2025-10-26 12:05:00 +08:00 
			
		
		
		
	 42a4f53d2b
			
		
	
	42a4f53d2b
	
	
	
		
			
			This commit applies all changes made after running the gdb/copyright.py script. Note that one file was flagged by the script, due to an invalid copyright header (gdb/unittests/basic_string_view/element_access/char/empty.cc). As the file was copied from GCC's libstdc++-v3 testsuite, this commit leaves this file untouched for the time being; a patch to fix the header was sent to gcc-patches first. gdb/ChangeLog: Update copyright year range in all GDB files.
		
			
				
	
	
		
			335 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			335 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* syscalls.c --- implement system calls for the M32C simulator.
 | |
| 
 | |
| Copyright (C) 2005-2019 Free Software Foundation, Inc.
 | |
| Contributed by Red Hat, Inc.
 | |
| 
 | |
| This file is part of the GNU simulators.
 | |
| 
 | |
| This program is free software; you can redistribute it and/or modify
 | |
| it under the terms of the GNU General Public License as published by
 | |
| the Free Software Foundation; either version 3 of the License, or
 | |
| (at your option) any later version.
 | |
| 
 | |
| This program is distributed in the hope that it will be useful,
 | |
| but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| GNU General Public License for more details.
 | |
| 
 | |
| You should have received a copy of the GNU General Public License
 | |
| along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
 | |
| 
 | |
| #include "config.h"
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <sys/time.h>
 | |
| 
 | |
| #include "gdb/callback.h"
 | |
| 
 | |
| #include "cpu.h"
 | |
| #include "mem.h"
 | |
| #include "syscalls.h"
 | |
| 
 | |
| #include "syscall.h"
 | |
| 
 | |
| /* The current syscall callbacks we're using.  */
 | |
| static struct host_callback_struct *callbacks;
 | |
| 
 | |
| void
 | |
| set_callbacks (struct host_callback_struct *cb)
 | |
| {
 | |
|   callbacks = cb;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* A16 ABI: arg1 in r1l (QI) or r1 (HI) or stack
 | |
|             arg2 in r2 (HI) or stack
 | |
| 	    arg3..N on stack
 | |
| 	    padding: none
 | |
| 
 | |
|    A24 ABI: arg1 in r0l (QI) or r0 (HI) or stack
 | |
| 	    arg2..N on stack
 | |
| 	    padding: qi->hi
 | |
| 
 | |
|    return value in r0l (QI) r0 (HI) r2r0 (SI)
 | |
|      structs: pointer pushed on stack last
 | |
| 
 | |
| */
 | |
| 
 | |
| int argp, stackp;
 | |
| 
 | |
| static int
 | |
| arg (int bytes)
 | |
| {
 | |
|   int rv = 0;
 | |
|   argp++;
 | |
|   if (A16)
 | |
|     {
 | |
|       switch (argp)
 | |
| 	{
 | |
| 	case 1:
 | |
| 	  if (bytes == 1)
 | |
| 	    return get_reg (r1l);
 | |
| 	  if (bytes == 2)
 | |
| 	    return get_reg (r1);
 | |
| 	  break;
 | |
| 	case 2:
 | |
| 	  if (bytes == 2)
 | |
| 	    return get_reg (r2);
 | |
| 	  break;
 | |
| 	}
 | |
|     }
 | |
|   else
 | |
|     {
 | |
|       switch (argp)
 | |
| 	{
 | |
| 	case 1:
 | |
| 	  if (bytes == 1)
 | |
| 	    return get_reg (r0l);
 | |
| 	  if (bytes == 2)
 | |
| 	    return get_reg (r0);
 | |
| 	  break;
 | |
| 	}
 | |
|     }
 | |
|   if (bytes == 0)
 | |
|     bytes = 2;
 | |
|   switch (bytes)
 | |
|     {
 | |
|     case 1:
 | |
|       rv = mem_get_qi (get_reg (sp) + stackp);
 | |
|       if (A24)
 | |
| 	stackp++;
 | |
|       break;
 | |
|     case 2:
 | |
|       rv = mem_get_hi (get_reg (sp) + stackp);
 | |
|       break;
 | |
|     case 3:
 | |
|       rv = mem_get_psi (get_reg (sp) + stackp);
 | |
|       if (A24)
 | |
| 	stackp++;
 | |
|       break;
 | |
|     case 4:
 | |
|       rv = mem_get_si (get_reg (sp) + stackp);
 | |
|       break;
 | |
|     }
 | |
|   stackp += bytes;
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| static void
 | |
| read_target (char *buffer, int address, int count, int asciiz)
 | |
| {
 | |
|   char byte;
 | |
|   while (count > 0)
 | |
|     {
 | |
|       byte = mem_get_qi (address++);
 | |
|       *buffer++ = byte;
 | |
|       if (asciiz && (byte == 0))
 | |
| 	return;
 | |
|       count--;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| write_target (char *buffer, int address, int count, int asciiz)
 | |
| {
 | |
|   char byte;
 | |
|   while (count > 0)
 | |
|     {
 | |
|       byte = *buffer++;
 | |
|       mem_put_qi (address++, byte);
 | |
|       if (asciiz && (byte == 0))
 | |
| 	return;
 | |
|       count--;
 | |
|     }
 | |
| }
 | |
| 
 | |
| #define PTRSZ (A16 ? 2 : 3)
 | |
| 
 | |
| static char *callnames[] = {
 | |
|   "SYS_zero",
 | |
|   "SYS_exit",
 | |
|   "SYS_open",
 | |
|   "SYS_close",
 | |
|   "SYS_read",
 | |
|   "SYS_write",
 | |
|   "SYS_lseek",
 | |
|   "SYS_unlink",
 | |
|   "SYS_getpid",
 | |
|   "SYS_kill",
 | |
|   "SYS_fstat",
 | |
|   "SYS_sbrk",
 | |
|   "SYS_argvlen",
 | |
|   "SYS_argv",
 | |
|   "SYS_chdir",
 | |
|   "SYS_stat",
 | |
|   "SYS_chmod",
 | |
|   "SYS_utime",
 | |
|   "SYS_time",
 | |
|   "SYS_gettimeofday",
 | |
|   "SYS_times",
 | |
|   "SYS_link"
 | |
| };
 | |
| 
 | |
| void
 | |
| m32c_syscall (int id)
 | |
| {
 | |
|   static char buf[256];
 | |
|   int rv;
 | |
| 
 | |
|   argp = 0;
 | |
|   stackp = A16 ? 3 : 4;
 | |
|   if (trace)
 | |
|     printf ("\033[31m/* SYSCALL(%d) = %s */\033[0m\n", id, callnames[id]);
 | |
|   switch (id)
 | |
|     {
 | |
|     case SYS_exit:
 | |
|       {
 | |
| 	int ec = arg (2);
 | |
| 	if (verbose)
 | |
| 	  printf ("[exit %d]\n", ec);
 | |
| 	step_result = M32C_MAKE_EXITED (ec);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_open:
 | |
|       {
 | |
| 	int path = arg (PTRSZ);
 | |
| 	int oflags = arg (2);
 | |
| 	int cflags = arg (2);
 | |
| 
 | |
| 	read_target (buf, path, 256, 1);
 | |
| 	if (trace)
 | |
| 	  printf ("open(\"%s\",0x%x,%#o) = ", buf, oflags, cflags);
 | |
| 
 | |
| 	if (callbacks)
 | |
| 	  /* The callback vector ignores CFLAGS.  */
 | |
| 	  rv = callbacks->open (callbacks, buf, oflags);
 | |
| 	else
 | |
| 	  {
 | |
| 	    int h_oflags = 0;
 | |
| 
 | |
| 	    if (oflags & 0x0001)
 | |
| 	      h_oflags |= O_WRONLY;
 | |
| 	    if (oflags & 0x0002)
 | |
| 	      h_oflags |= O_RDWR;
 | |
| 	    if (oflags & 0x0200)
 | |
| 	      h_oflags |= O_CREAT;
 | |
| 	    if (oflags & 0x0008)
 | |
| 	      h_oflags |= O_APPEND;
 | |
| 	    if (oflags & 0x0400)
 | |
| 	      h_oflags |= O_TRUNC;
 | |
| 	    rv = open (buf, h_oflags, cflags);
 | |
| 	  }
 | |
| 	if (trace)
 | |
| 	  printf ("%d\n", rv);
 | |
| 	put_reg (r0, rv);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_close:
 | |
|       {
 | |
| 	int fd = arg (2);
 | |
| 
 | |
| 	if (callbacks)
 | |
| 	  rv = callbacks->close (callbacks, fd);
 | |
| 	else if (fd > 2)
 | |
| 	  rv = close (fd);
 | |
| 	else
 | |
| 	  rv = 0;
 | |
| 	if (trace)
 | |
| 	  printf ("close(%d) = %d\n", fd, rv);
 | |
| 	put_reg (r0, rv);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_read:
 | |
|       {
 | |
| 	int fd = arg (2);
 | |
| 	int addr = arg (PTRSZ);
 | |
| 	int count = arg (2);
 | |
| 
 | |
| 	if (count > sizeof (buf))
 | |
| 	  count = sizeof (buf);
 | |
| 	if (callbacks)
 | |
| 	  rv = callbacks->read (callbacks, fd, buf, count);
 | |
| 	else
 | |
| 	  rv = read (fd, buf, count);
 | |
| 	if (trace)
 | |
| 	  printf ("read(%d,%d) = %d\n", fd, count, rv);
 | |
| 	if (rv > 0)
 | |
| 	  write_target (buf, addr, rv, 0);
 | |
| 	put_reg (r0, rv);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_write:
 | |
|       {
 | |
| 	int fd = arg (2);
 | |
| 	int addr = arg (PTRSZ);
 | |
| 	int count = arg (2);
 | |
| 
 | |
| 	if (count > sizeof (buf))
 | |
| 	  count = sizeof (buf);
 | |
| 	if (trace)
 | |
| 	  printf ("write(%d,0x%x,%d)\n", fd, addr, count);
 | |
| 	read_target (buf, addr, count, 0);
 | |
| 	if (trace)
 | |
| 	  fflush (stdout);
 | |
| 	if (callbacks)
 | |
| 	  rv = callbacks->write (callbacks, fd, buf, count);
 | |
| 	else
 | |
| 	  rv = write (fd, buf, count);
 | |
| 	if (trace)
 | |
| 	  printf ("write(%d,%d) = %d\n", fd, count, rv);
 | |
| 	put_reg (r0, rv);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_getpid:
 | |
|       put_reg (r0, 42);
 | |
|       break;
 | |
| 
 | |
|     case SYS_gettimeofday:
 | |
|       {
 | |
| 	int tvaddr = arg (PTRSZ);
 | |
| 	struct timeval tv;
 | |
| 
 | |
| 	rv = gettimeofday (&tv, 0);
 | |
| 	if (trace)
 | |
| 	  printf ("gettimeofday: %ld sec %ld usec to 0x%x\n", tv.tv_sec,
 | |
| 		  tv.tv_usec, tvaddr);
 | |
| 	mem_put_si (tvaddr, tv.tv_sec);
 | |
| 	mem_put_si (tvaddr + 4, tv.tv_usec);
 | |
| 	put_reg (r0, rv);
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case SYS_kill:
 | |
|       {
 | |
| 	int pid = arg (2);
 | |
| 	int sig = arg (2);
 | |
| 	if (pid == 42)
 | |
| 	  {
 | |
| 	    if (verbose)
 | |
| 	      printf ("[signal %d]\n", sig);
 | |
| 	    step_result = M32C_MAKE_STOPPED (sig);
 | |
| 	  }
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     case 11:
 | |
|       {
 | |
| 	int heaptop_arg = arg (PTRSZ);
 | |
| 	if (trace)
 | |
| 	  printf ("sbrk: heap top set to %x\n", heaptop_arg);
 | |
| 	heaptop = heaptop_arg;
 | |
| 	if (heapbottom == 0)
 | |
| 	  heapbottom = heaptop_arg;
 | |
|       }
 | |
|       break;
 | |
| 
 | |
|     }
 | |
| }
 |