Fix whitespace snafu in tc-riscv.c
[binutils-gdb.git] / sim / common / syscall.c
blob96033db84c3b3bee1d74039382982a40f381985c
1 /* Remote target system call support.
2 Copyright 1997-2023 Free Software Foundation, Inc.
3 Contributed by Cygnus Solutions.
5 This file is part of GDB.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20 /* This interface isn't intended to be specific to any particular kind
21 of remote (hardware, simulator, whatever). As such, support for it
22 (e.g. sim/common/callback.c) should *not* live in the simulator source
23 tree, nor should it live in the gdb source tree. K&R C must be
24 supported. */
26 /* This must come before any other includes. */
27 #include "defs.h"
29 #include <errno.h>
30 #include <fcntl.h>
31 #include <stdarg.h>
32 #include <stdio.h>
33 #include <stdlib.h>
34 #include <string.h>
35 #include <time.h>
36 #include <unistd.h>
37 #include <sys/stat.h>
38 #include <sys/types.h>
40 #include "ansidecl.h"
41 #include "libiberty.h"
43 #include "sim/callback.h"
45 #ifndef ENOSYS
46 #define ENOSYS EINVAL
47 #endif
48 #ifndef ENAMETOOLONG
49 #define ENAMETOOLONG EINVAL
50 #endif
52 /* Maximum length of a path name. */
53 #ifndef MAX_PATH_LEN
54 #define MAX_PATH_LEN 1024
55 #endif
57 /* When doing file read/writes, do this many bytes at a time. */
58 #define FILE_XFR_SIZE 4096
60 /* FIXME: for now, need to consider target word size. */
61 #define TWORD long
62 #define TADDR unsigned long
64 /* Path to be prepended to syscalls with absolute paths, and to be
65 chdir:ed at startup, if not empty. */
66 char *simulator_sysroot = "";
68 /* Utility of cb_syscall to fetch a path name or other string from the target.
69 The result is 0 for success or a host errno value. */
71 int
72 cb_get_string (host_callback *cb, CB_SYSCALL *sc, char *buf, int buflen,
73 TADDR addr)
75 char *p, *pend;
77 for (p = buf, pend = buf + buflen; p < pend; ++p, ++addr)
79 /* No, it isn't expected that this would cause one transaction with
80 the remote target for each byte. The target could send the
81 path name along with the syscall request, and cache the file
82 name somewhere (or otherwise tweak this as desired). */
83 unsigned int count = (*sc->read_mem) (cb, sc, addr, p, 1);
85 if (count != 1)
86 return EINVAL;
87 if (*p == 0)
88 break;
90 if (p == pend)
91 return ENAMETOOLONG;
92 return 0;
95 /* Utility of cb_syscall to fetch a path name.
96 The buffer is malloc'd and the address is stored in BUFP.
97 The result is that of get_string, but prepended with
98 simulator_sysroot if the string starts with '/'.
99 If an error occurs, no buffer is left malloc'd. */
101 static int
102 get_path (host_callback *cb, CB_SYSCALL *sc, TADDR addr, char **bufp)
104 char *buf = xmalloc (MAX_PATH_LEN);
105 int result;
106 int sysroot_len = strlen (simulator_sysroot);
108 result = cb_get_string (cb, sc, buf, MAX_PATH_LEN - sysroot_len, addr);
109 if (result == 0)
111 /* Prepend absolute paths with simulator_sysroot. Relative paths
112 are supposed to be relative to a chdir within that path, but at
113 this point unknown where. */
114 if (simulator_sysroot[0] != '\0' && *buf == '/')
116 /* Considering expected rareness of syscalls with absolute
117 file paths (compared to relative file paths and insn
118 execution), it does not seem worthwhile to rearrange things
119 to get rid of the string moves here; we'd need at least an
120 extra call to check the initial '/' in the path. */
121 memmove (buf + sysroot_len, buf, sysroot_len);
122 memcpy (buf, simulator_sysroot, sysroot_len);
125 *bufp = buf;
127 else
128 free (buf);
129 return result;
132 /* Perform a system call on behalf of the target. */
134 CB_RC
135 cb_syscall (host_callback *cb, CB_SYSCALL *sc)
137 TWORD result = 0, errcode = 0;
139 if (sc->magic != CB_SYSCALL_MAGIC)
140 abort ();
142 switch (cb_target_to_host_syscall (cb, sc->func))
144 case CB_SYS_argc:
145 result = countargv (cb->argv);
146 break;
148 case CB_SYS_argnlen:
150 if (sc->arg1 >= 0 && sc->arg1 < countargv (cb->argv))
151 result = strlen (cb->argv[sc->arg1]);
152 else
154 result = -1;
155 errcode = EINVAL;
158 break;
160 case CB_SYS_argn:
162 if (sc->arg1 >= 0 && sc->arg1 < countargv (cb->argv))
164 const char *argn = cb->argv[sc->arg1];
165 int len = strlen (argn);
166 int written = sc->write_mem (cb, sc, sc->arg2, argn, len + 1);
168 if (written == len + 1)
169 result = sc->arg2;
170 else
172 result = -1;
173 errcode = EINVAL;
176 else
178 result = -1;
179 errcode = EINVAL;
182 break;
184 case CB_SYS_argvlen :
186 /* Compute how much space is required to store the argv,envp
187 strings so that the program can allocate the space and then
188 call SYS_argv to fetch the values. */
189 int argc, envc, arglen, envlen;
190 char **argv = cb->argv;
191 char **envp = cb->envp;
193 argc = arglen = 0;
194 if (argv)
196 for ( ; argv[argc]; ++argc)
197 arglen += strlen (argv[argc]) + 1;
199 envc = envlen = 0;
200 if (envp)
202 for ( ; envp[envc]; ++envc)
203 envlen += strlen (envp[envc]) + 1;
205 result = arglen + 1 + envlen + 1;
206 break;
209 case CB_SYS_argv :
211 /* Pointer to target's buffer. */
212 TADDR tbuf = sc->arg1;
213 /* Buffer size. */
214 int bufsize = sc->arg2;
215 int written = 0;
216 /* Q is the target address of where all the strings go. */
217 TADDR q;
218 int i, argc, envc, len, ret;
219 char **argv = cb->argv;
220 char **envp = cb->envp;
222 result = -1;
224 argc = 0;
225 if (argv)
227 for ( ; argv[argc]; ++argc)
229 len = strlen (argv[argc]) + 1;
230 if (written + len > bufsize)
231 goto efault;
233 ret = (*sc->write_mem) (cb, sc, tbuf + written, argv[argc],
234 len);
235 if (ret != len)
236 goto einval;
238 written += ret;
241 /* Double NUL bytes indicates end of strings. */
242 if (written >= bufsize)
243 goto efault;
244 if ((*sc->write_mem) (cb, sc, tbuf + written, "", 1) != 1)
245 goto einval;
246 ++written;
248 envc = 0;
249 if (envp)
251 for ( ; envp[envc]; ++envc)
253 len = strlen (envp[envc]) + 1;
254 if (written + len > bufsize)
255 goto efault;
257 ret = (*sc->write_mem) (cb, sc, tbuf + written, envp[envc],
258 len);
259 if (ret != len)
260 goto einval;
261 written += ret;
264 /* Double NUL bytes indicates end of strings. */
265 if (written >= bufsize)
266 goto efault;
267 if ((*sc->write_mem) (cb, sc, tbuf + written, "", 1) != 1)
268 goto einval;
270 result = argc;
271 sc->result2 = envc;
272 break;
274 efault:
275 errcode = EFAULT;
276 goto FinishSyscall;
278 einval:
279 errcode = EINVAL;
280 goto FinishSyscall;
283 case CB_SYS_exit :
284 /* Caller must catch and handle; see sim_syscall as an example. */
285 break;
287 case CB_SYS_open :
289 char *path;
291 errcode = get_path (cb, sc, sc->arg1, &path);
292 if (errcode != 0)
294 result = -1;
295 goto FinishSyscall;
297 result = (*cb->open) (cb, path, sc->arg2 /*, sc->arg3*/);
298 free (path);
299 if (result < 0)
300 goto ErrorFinish;
302 break;
304 case CB_SYS_close :
305 result = (*cb->close) (cb, sc->arg1);
306 if (result < 0)
307 goto ErrorFinish;
308 break;
310 case CB_SYS_read :
312 /* ??? Perfect handling of error conditions may require only one
313 call to cb->read. One can't assume all the data is
314 contiguously stored in host memory so that would require
315 malloc'ing/free'ing the space. Maybe later. */
316 char buf[FILE_XFR_SIZE];
317 int fd = sc->arg1;
318 TADDR addr = sc->arg2;
319 size_t count = sc->arg3;
320 size_t bytes_read = 0;
321 int bytes_written;
323 while (count > 0)
325 if (cb_is_stdin (cb, fd))
326 result = (int) (*cb->read_stdin) (cb, buf,
327 (count < FILE_XFR_SIZE
328 ? count : FILE_XFR_SIZE));
329 else
330 result = (int) (*cb->read) (cb, fd, buf,
331 (count < FILE_XFR_SIZE
332 ? count : FILE_XFR_SIZE));
333 if (result == -1)
334 goto ErrorFinish;
335 if (result == 0) /* EOF */
336 break;
337 bytes_written = (*sc->write_mem) (cb, sc, addr, buf, result);
338 if (bytes_written != result)
340 result = -1;
341 errcode = EINVAL;
342 goto FinishSyscall;
344 bytes_read += result;
345 count -= result;
346 addr += result;
347 /* If this is a short read, don't go back for more */
348 if (result != FILE_XFR_SIZE)
349 break;
351 result = bytes_read;
353 break;
355 case CB_SYS_write :
357 /* ??? Perfect handling of error conditions may require only one
358 call to cb->write. One can't assume all the data is
359 contiguously stored in host memory so that would require
360 malloc'ing/free'ing the space. Maybe later. */
361 char buf[FILE_XFR_SIZE];
362 int fd = sc->arg1;
363 TADDR addr = sc->arg2;
364 size_t count = sc->arg3;
365 int bytes_read;
366 size_t bytes_written = 0;
368 while (count > 0)
370 int bytes_to_read = count < FILE_XFR_SIZE ? count : FILE_XFR_SIZE;
371 bytes_read = (*sc->read_mem) (cb, sc, addr, buf, bytes_to_read);
372 if (bytes_read != bytes_to_read)
374 result = -1;
375 errcode = EINVAL;
376 goto FinishSyscall;
378 if (cb_is_stdout (cb, fd))
380 result = (int) (*cb->write_stdout) (cb, buf, bytes_read);
381 (*cb->flush_stdout) (cb);
383 else if (cb_is_stderr (cb, fd))
385 result = (int) (*cb->write_stderr) (cb, buf, bytes_read);
386 (*cb->flush_stderr) (cb);
388 else
389 result = (int) (*cb->write) (cb, fd, buf, bytes_read);
390 if (result == -1)
391 goto ErrorFinish;
392 bytes_written += result;
393 count -= result;
394 addr += result;
396 result = bytes_written;
398 break;
400 case CB_SYS_lseek :
402 int fd = sc->arg1;
403 unsigned long offset = sc->arg2;
404 int whence = sc->arg3;
406 result = (*cb->lseek) (cb, fd, offset, whence);
407 if (result < 0)
408 goto ErrorFinish;
410 break;
412 case CB_SYS_unlink :
414 char *path;
416 errcode = get_path (cb, sc, sc->arg1, &path);
417 if (errcode != 0)
419 result = -1;
420 goto FinishSyscall;
422 result = (*cb->unlink) (cb, path);
423 free (path);
424 if (result < 0)
425 goto ErrorFinish;
427 break;
429 case CB_SYS_truncate :
431 char *path;
432 long len = sc->arg2;
434 errcode = get_path (cb, sc, sc->arg1, &path);
435 if (errcode != 0)
437 result = -1;
438 errcode = EFAULT;
439 goto FinishSyscall;
441 result = (*cb->truncate) (cb, path, len);
442 free (path);
443 if (result < 0)
444 goto ErrorFinish;
446 break;
448 case CB_SYS_ftruncate :
450 int fd = sc->arg1;
451 long len = sc->arg2;
453 result = (*cb->ftruncate) (cb, fd, len);
454 if (result < 0)
455 goto ErrorFinish;
457 break;
459 case CB_SYS_rename :
461 char *path1, *path2;
463 errcode = get_path (cb, sc, sc->arg1, &path1);
464 if (errcode != 0)
466 result = -1;
467 errcode = EFAULT;
468 goto FinishSyscall;
470 errcode = get_path (cb, sc, sc->arg2, &path2);
471 if (errcode != 0)
473 result = -1;
474 errcode = EFAULT;
475 free (path1);
476 goto FinishSyscall;
478 result = (*cb->rename) (cb, path1, path2);
479 free (path1);
480 free (path2);
481 if (result < 0)
482 goto ErrorFinish;
484 break;
486 case CB_SYS_stat :
488 char *path,*buf;
489 int buflen;
490 struct stat statbuf;
491 TADDR addr = sc->arg2;
493 errcode = get_path (cb, sc, sc->arg1, &path);
494 if (errcode != 0)
496 result = -1;
497 goto FinishSyscall;
499 result = (*cb->to_stat) (cb, path, &statbuf);
500 free (path);
501 if (result < 0)
502 goto ErrorFinish;
503 buflen = cb_host_to_target_stat (cb, NULL, NULL);
504 buf = xmalloc (buflen);
505 if (cb_host_to_target_stat (cb, &statbuf, buf) != buflen)
507 /* The translation failed. This is due to an internal
508 host program error, not the target's fault. */
509 free (buf);
510 errcode = ENOSYS;
511 result = -1;
512 goto FinishSyscall;
514 if ((*sc->write_mem) (cb, sc, addr, buf, buflen) != buflen)
516 free (buf);
517 errcode = EINVAL;
518 result = -1;
519 goto FinishSyscall;
521 free (buf);
523 break;
525 case CB_SYS_fstat :
527 char *buf;
528 int buflen;
529 struct stat statbuf;
530 TADDR addr = sc->arg2;
532 result = (*cb->to_fstat) (cb, sc->arg1, &statbuf);
533 if (result < 0)
534 goto ErrorFinish;
535 buflen = cb_host_to_target_stat (cb, NULL, NULL);
536 buf = xmalloc (buflen);
537 if (cb_host_to_target_stat (cb, &statbuf, buf) != buflen)
539 /* The translation failed. This is due to an internal
540 host program error, not the target's fault. */
541 free (buf);
542 errcode = ENOSYS;
543 result = -1;
544 goto FinishSyscall;
546 if ((*sc->write_mem) (cb, sc, addr, buf, buflen) != buflen)
548 free (buf);
549 errcode = EINVAL;
550 result = -1;
551 goto FinishSyscall;
553 free (buf);
555 break;
557 case CB_SYS_lstat :
559 char *path, *buf;
560 int buflen;
561 struct stat statbuf;
562 TADDR addr = sc->arg2;
564 errcode = get_path (cb, sc, sc->arg1, &path);
565 if (errcode != 0)
567 result = -1;
568 goto FinishSyscall;
570 result = (*cb->to_lstat) (cb, path, &statbuf);
571 free (path);
572 if (result < 0)
573 goto ErrorFinish;
575 buflen = cb_host_to_target_stat (cb, NULL, NULL);
576 buf = xmalloc (buflen);
577 if (cb_host_to_target_stat (cb, &statbuf, buf) != buflen)
579 /* The translation failed. This is due to an internal
580 host program error, not the target's fault.
581 Unfortunately, it's hard to test this case, so there's no
582 test-case for this execution path. */
583 free (buf);
584 errcode = ENOSYS;
585 result = -1;
586 goto FinishSyscall;
589 if ((*sc->write_mem) (cb, sc, addr, buf, buflen) != buflen)
591 free (buf);
592 errcode = EINVAL;
593 result = -1;
594 goto FinishSyscall;
597 free (buf);
599 break;
601 case CB_SYS_pipe :
603 int p[2];
604 char *target_p = xcalloc (1, cb->target_sizeof_int * 2);
606 result = (*cb->pipe) (cb, p);
607 if (result != 0)
608 goto ErrorFinish;
610 cb_store_target_endian (cb, target_p, cb->target_sizeof_int, p[0]);
611 cb_store_target_endian (cb, target_p + cb->target_sizeof_int,
612 cb->target_sizeof_int, p[1]);
613 if ((*sc->write_mem) (cb, sc, sc->arg1, target_p,
614 cb->target_sizeof_int * 2)
615 != cb->target_sizeof_int * 2)
617 /* Close the pipe fd:s. */
618 (*cb->close) (cb, p[0]);
619 (*cb->close) (cb, p[1]);
620 errcode = EFAULT;
621 result = -1;
624 free (target_p);
626 break;
628 case CB_SYS_getpid:
629 /* POSIX says getpid always succeeds. */
630 result = (*cb->getpid) (cb);
631 break;
633 case CB_SYS_kill:
634 /* If killing self, leave it to the caller to process so it can send the
635 signal to the engine. */
636 if (sc->arg1 == (*cb->getpid) (cb))
638 result = -1;
639 errcode = ENOSYS;
641 else
643 int signum = cb_target_to_host_signal (cb, sc->arg2);
645 result = (*cb->kill) (cb, sc->arg1, signum);
646 cb->last_errno = errno;
647 goto ErrorFinish;
649 break;
651 case CB_SYS_time :
653 /* FIXME: May wish to change CB_SYS_time to something else.
654 We might also want gettimeofday or times, but if system calls
655 can be built on others, we can keep the number we have to support
656 here down. */
657 time_t t = (*cb->time) (cb);
658 result = t;
659 /* It is up to target code to process the argument to time(). */
661 break;
663 case CB_SYS_chdir :
664 case CB_SYS_chmod :
665 case CB_SYS_utime :
666 /* fall through for now */
668 default :
669 result = -1;
670 errcode = ENOSYS;
671 break;
674 FinishSyscall:
675 sc->result = result;
676 if (errcode == 0)
677 sc->errcode = 0;
678 else
679 sc->errcode = cb_host_to_target_errno (cb, errcode);
680 return CB_RC_OK;
682 ErrorFinish:
683 sc->result = result;
684 sc->errcode = (*cb->get_errno) (cb);
685 return CB_RC_OK;