]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blob - gdb/remote-fileio.c
update copyright year range in GDB files
[thirdparty/binutils-gdb.git] / gdb / remote-fileio.c
1 /* Remote File-I/O communications
2
3 Copyright (C) 2003-2017 Free Software Foundation, Inc.
4
5 This file is part of GDB.
6
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.
11
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.
16
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/>. */
19
20 /* See the GDB User Guide for details of the GDB remote protocol. */
21
22 #include "defs.h"
23 #include "gdbcmd.h"
24 #include "remote.h"
25 #include "gdb_wait.h"
26 #include <sys/stat.h>
27 #include "remote-fileio.h"
28 #include "event-loop.h"
29 #include "target.h"
30 #include "filenames.h"
31 #include "filestuff.h"
32
33 #include <fcntl.h>
34 #include "gdb_sys_time.h"
35 #ifdef __CYGWIN__
36 #include <sys/cygwin.h> /* For cygwin_conv_path. */
37 #endif
38 #include <signal.h>
39
40 static struct {
41 int *fd_map;
42 int fd_map_size;
43 } remote_fio_data;
44
45 #define FIO_FD_INVALID -1
46 #define FIO_FD_CONSOLE_IN -2
47 #define FIO_FD_CONSOLE_OUT -3
48
49 static int remote_fio_system_call_allowed = 0;
50
51 static int
52 remote_fileio_init_fd_map (void)
53 {
54 int i;
55
56 if (!remote_fio_data.fd_map)
57 {
58 remote_fio_data.fd_map = XNEWVEC (int, 10);
59 remote_fio_data.fd_map_size = 10;
60 remote_fio_data.fd_map[0] = FIO_FD_CONSOLE_IN;
61 remote_fio_data.fd_map[1] = FIO_FD_CONSOLE_OUT;
62 remote_fio_data.fd_map[2] = FIO_FD_CONSOLE_OUT;
63 for (i = 3; i < 10; ++i)
64 remote_fio_data.fd_map[i] = FIO_FD_INVALID;
65 }
66 return 3;
67 }
68
69 static int
70 remote_fileio_resize_fd_map (void)
71 {
72 int i = remote_fio_data.fd_map_size;
73
74 if (!remote_fio_data.fd_map)
75 return remote_fileio_init_fd_map ();
76 remote_fio_data.fd_map_size += 10;
77 remote_fio_data.fd_map =
78 (int *) xrealloc (remote_fio_data.fd_map,
79 remote_fio_data.fd_map_size * sizeof (int));
80 for (; i < remote_fio_data.fd_map_size; i++)
81 remote_fio_data.fd_map[i] = FIO_FD_INVALID;
82 return remote_fio_data.fd_map_size - 10;
83 }
84
85 static int
86 remote_fileio_next_free_fd (void)
87 {
88 int i;
89
90 for (i = 0; i < remote_fio_data.fd_map_size; ++i)
91 if (remote_fio_data.fd_map[i] == FIO_FD_INVALID)
92 return i;
93 return remote_fileio_resize_fd_map ();
94 }
95
96 static int
97 remote_fileio_fd_to_targetfd (int fd)
98 {
99 int target_fd = remote_fileio_next_free_fd ();
100
101 remote_fio_data.fd_map[target_fd] = fd;
102 return target_fd;
103 }
104
105 static int
106 remote_fileio_map_fd (int target_fd)
107 {
108 remote_fileio_init_fd_map ();
109 if (target_fd < 0 || target_fd >= remote_fio_data.fd_map_size)
110 return FIO_FD_INVALID;
111 return remote_fio_data.fd_map[target_fd];
112 }
113
114 static void
115 remote_fileio_close_target_fd (int target_fd)
116 {
117 remote_fileio_init_fd_map ();
118 if (target_fd >= 0 && target_fd < remote_fio_data.fd_map_size)
119 remote_fio_data.fd_map[target_fd] = FIO_FD_INVALID;
120 }
121
122 static int
123 remote_fileio_oflags_to_host (long flags)
124 {
125 int hflags = 0;
126
127 if (flags & FILEIO_O_CREAT)
128 hflags |= O_CREAT;
129 if (flags & FILEIO_O_EXCL)
130 hflags |= O_EXCL;
131 if (flags & FILEIO_O_TRUNC)
132 hflags |= O_TRUNC;
133 if (flags & FILEIO_O_APPEND)
134 hflags |= O_APPEND;
135 if (flags & FILEIO_O_RDONLY)
136 hflags |= O_RDONLY;
137 if (flags & FILEIO_O_WRONLY)
138 hflags |= O_WRONLY;
139 if (flags & FILEIO_O_RDWR)
140 hflags |= O_RDWR;
141 /* On systems supporting binary and text mode, always open files in
142 binary mode. */
143 #ifdef O_BINARY
144 hflags |= O_BINARY;
145 #endif
146 return hflags;
147 }
148
149 static mode_t
150 remote_fileio_mode_to_host (long mode, int open_call)
151 {
152 mode_t hmode = 0;
153
154 if (!open_call)
155 {
156 if (mode & FILEIO_S_IFREG)
157 hmode |= S_IFREG;
158 if (mode & FILEIO_S_IFDIR)
159 hmode |= S_IFDIR;
160 if (mode & FILEIO_S_IFCHR)
161 hmode |= S_IFCHR;
162 }
163 if (mode & FILEIO_S_IRUSR)
164 hmode |= S_IRUSR;
165 if (mode & FILEIO_S_IWUSR)
166 hmode |= S_IWUSR;
167 if (mode & FILEIO_S_IXUSR)
168 hmode |= S_IXUSR;
169 #ifdef S_IRGRP
170 if (mode & FILEIO_S_IRGRP)
171 hmode |= S_IRGRP;
172 #endif
173 #ifdef S_IWGRP
174 if (mode & FILEIO_S_IWGRP)
175 hmode |= S_IWGRP;
176 #endif
177 #ifdef S_IXGRP
178 if (mode & FILEIO_S_IXGRP)
179 hmode |= S_IXGRP;
180 #endif
181 if (mode & FILEIO_S_IROTH)
182 hmode |= S_IROTH;
183 #ifdef S_IWOTH
184 if (mode & FILEIO_S_IWOTH)
185 hmode |= S_IWOTH;
186 #endif
187 #ifdef S_IXOTH
188 if (mode & FILEIO_S_IXOTH)
189 hmode |= S_IXOTH;
190 #endif
191 return hmode;
192 }
193
194 static int
195 remote_fileio_seek_flag_to_host (long num, int *flag)
196 {
197 if (!flag)
198 return 0;
199 switch (num)
200 {
201 case FILEIO_SEEK_SET:
202 *flag = SEEK_SET;
203 break;
204 case FILEIO_SEEK_CUR:
205 *flag = SEEK_CUR;
206 break;
207 case FILEIO_SEEK_END:
208 *flag = SEEK_END;
209 break;
210 default:
211 return -1;
212 }
213 return 0;
214 }
215
216 static int
217 remote_fileio_extract_long (char **buf, LONGEST *retlong)
218 {
219 char *c;
220 int sign = 1;
221
222 if (!buf || !*buf || !**buf || !retlong)
223 return -1;
224 c = strchr (*buf, ',');
225 if (c)
226 *c++ = '\0';
227 else
228 c = strchr (*buf, '\0');
229 while (strchr ("+-", **buf))
230 {
231 if (**buf == '-')
232 sign = -sign;
233 ++*buf;
234 }
235 for (*retlong = 0; **buf; ++*buf)
236 {
237 *retlong <<= 4;
238 if (**buf >= '0' && **buf <= '9')
239 *retlong += **buf - '0';
240 else if (**buf >= 'a' && **buf <= 'f')
241 *retlong += **buf - 'a' + 10;
242 else if (**buf >= 'A' && **buf <= 'F')
243 *retlong += **buf - 'A' + 10;
244 else
245 return -1;
246 }
247 *retlong *= sign;
248 *buf = c;
249 return 0;
250 }
251
252 static int
253 remote_fileio_extract_int (char **buf, long *retint)
254 {
255 int ret;
256 LONGEST retlong;
257
258 if (!retint)
259 return -1;
260 ret = remote_fileio_extract_long (buf, &retlong);
261 if (!ret)
262 *retint = (long) retlong;
263 return ret;
264 }
265
266 static int
267 remote_fileio_extract_ptr_w_len (char **buf, CORE_ADDR *ptrval, int *length)
268 {
269 char *c;
270 LONGEST retlong;
271
272 if (!buf || !*buf || !**buf || !ptrval || !length)
273 return -1;
274 c = strchr (*buf, '/');
275 if (!c)
276 return -1;
277 *c++ = '\0';
278 if (remote_fileio_extract_long (buf, &retlong))
279 return -1;
280 *ptrval = (CORE_ADDR) retlong;
281 *buf = c;
282 if (remote_fileio_extract_long (buf, &retlong))
283 return -1;
284 *length = (int) retlong;
285 return 0;
286 }
287
288 static void
289 remote_fileio_to_fio_long (LONGEST num, fio_long_t fnum)
290 {
291 host_to_bigendian (num, (char *) fnum, 8);
292 }
293
294 static void
295 remote_fileio_to_fio_timeval (struct timeval *tv, struct fio_timeval *ftv)
296 {
297 host_to_fileio_time (tv->tv_sec, ftv->ftv_sec);
298 remote_fileio_to_fio_long (tv->tv_usec, ftv->ftv_usec);
299 }
300
301 /* The quit handler originally installed. */
302 static quit_handler_ftype *remote_fileio_o_quit_handler;
303
304 /* What to do on a QUIT call while handling a file I/O request. We
305 throw a quit exception, which is caught by remote_fileio_request
306 and translated to an EINTR reply back to the target. */
307
308 static void
309 remote_fileio_quit_handler (void)
310 {
311 if (check_quit_flag ())
312 quit ();
313 }
314
315 static void
316 remote_fileio_reply (int retcode, int error)
317 {
318 char buf[32];
319 int ctrl_c = check_quit_flag ();
320
321 strcpy (buf, "F");
322 if (retcode < 0)
323 {
324 strcat (buf, "-");
325 retcode = -retcode;
326 }
327 sprintf (buf + strlen (buf), "%x", retcode);
328 if (error || ctrl_c)
329 {
330 if (error && ctrl_c)
331 error = FILEIO_EINTR;
332 if (error < 0)
333 {
334 strcat (buf, "-");
335 error = -error;
336 }
337 sprintf (buf + strlen (buf), ",%x", error);
338 if (ctrl_c)
339 strcat (buf, ",C");
340 }
341 quit_handler = remote_fileio_o_quit_handler;
342 putpkt (buf);
343 }
344
345 static void
346 remote_fileio_ioerror (void)
347 {
348 remote_fileio_reply (-1, FILEIO_EIO);
349 }
350
351 static void
352 remote_fileio_badfd (void)
353 {
354 remote_fileio_reply (-1, FILEIO_EBADF);
355 }
356
357 static void
358 remote_fileio_return_errno (int retcode)
359 {
360 remote_fileio_reply (retcode, retcode < 0
361 ? host_to_fileio_error (errno) : 0);
362 }
363
364 static void
365 remote_fileio_return_success (int retcode)
366 {
367 remote_fileio_reply (retcode, 0);
368 }
369
370 static void
371 remote_fileio_func_open (char *buf)
372 {
373 CORE_ADDR ptrval;
374 int length;
375 long num;
376 int flags, fd;
377 mode_t mode;
378 char *pathname;
379 struct stat st;
380
381 /* 1. Parameter: Ptr to pathname / length incl. trailing zero. */
382 if (remote_fileio_extract_ptr_w_len (&buf, &ptrval, &length))
383 {
384 remote_fileio_ioerror ();
385 return;
386 }
387 /* 2. Parameter: open flags */
388 if (remote_fileio_extract_int (&buf, &num))
389 {
390 remote_fileio_ioerror ();
391 return;
392 }
393 flags = remote_fileio_oflags_to_host (num);
394 /* 3. Parameter: open mode */
395 if (remote_fileio_extract_int (&buf, &num))
396 {
397 remote_fileio_ioerror ();
398 return;
399 }
400 mode = remote_fileio_mode_to_host (num, 1);
401
402 /* Request pathname. */
403 pathname = (char *) alloca (length);
404 if (target_read_memory (ptrval, (gdb_byte *) pathname, length) != 0)
405 {
406 remote_fileio_ioerror ();
407 return;
408 }
409
410 /* Check if pathname exists and is not a regular file or directory. If so,
411 return an appropriate error code. Same for trying to open directories
412 for writing. */
413 if (!stat (pathname, &st))
414 {
415 if (!S_ISREG (st.st_mode) && !S_ISDIR (st.st_mode))
416 {
417 remote_fileio_reply (-1, FILEIO_ENODEV);
418 return;
419 }
420 if (S_ISDIR (st.st_mode)
421 && ((flags & O_WRONLY) == O_WRONLY || (flags & O_RDWR) == O_RDWR))
422 {
423 remote_fileio_reply (-1, FILEIO_EISDIR);
424 return;
425 }
426 }
427
428 fd = gdb_open_cloexec (pathname, flags, mode);
429 if (fd < 0)
430 {
431 remote_fileio_return_errno (-1);
432 return;
433 }
434
435 fd = remote_fileio_fd_to_targetfd (fd);
436 remote_fileio_return_success (fd);
437 }
438
439 static void
440 remote_fileio_func_close (char *buf)
441 {
442 long num;
443 int fd;
444
445 /* Parameter: file descriptor */
446 if (remote_fileio_extract_int (&buf, &num))
447 {
448 remote_fileio_ioerror ();
449 return;
450 }
451 fd = remote_fileio_map_fd ((int) num);
452 if (fd == FIO_FD_INVALID)
453 {
454 remote_fileio_badfd ();
455 return;
456 }
457
458 if (fd != FIO_FD_CONSOLE_IN && fd != FIO_FD_CONSOLE_OUT && close (fd))
459 remote_fileio_return_errno (-1);
460 remote_fileio_close_target_fd ((int) num);
461 remote_fileio_return_success (0);
462 }
463
464 static void
465 remote_fileio_func_read (char *buf)
466 {
467 long target_fd, num;
468 LONGEST lnum;
469 CORE_ADDR ptrval;
470 int fd, ret;
471 gdb_byte *buffer;
472 size_t length;
473 off_t old_offset, new_offset;
474
475 /* 1. Parameter: file descriptor */
476 if (remote_fileio_extract_int (&buf, &target_fd))
477 {
478 remote_fileio_ioerror ();
479 return;
480 }
481 fd = remote_fileio_map_fd ((int) target_fd);
482 if (fd == FIO_FD_INVALID)
483 {
484 remote_fileio_badfd ();
485 return;
486 }
487 /* 2. Parameter: buffer pointer */
488 if (remote_fileio_extract_long (&buf, &lnum))
489 {
490 remote_fileio_ioerror ();
491 return;
492 }
493 ptrval = (CORE_ADDR) lnum;
494 /* 3. Parameter: buffer length */
495 if (remote_fileio_extract_int (&buf, &num))
496 {
497 remote_fileio_ioerror ();
498 return;
499 }
500 length = (size_t) num;
501
502 switch (fd)
503 {
504 case FIO_FD_CONSOLE_OUT:
505 remote_fileio_badfd ();
506 return;
507 case FIO_FD_CONSOLE_IN:
508 {
509 static char *remaining_buf = NULL;
510 static int remaining_length = 0;
511
512 buffer = (gdb_byte *) xmalloc (16384);
513 if (remaining_buf)
514 {
515 if (remaining_length > length)
516 {
517 memcpy (buffer, remaining_buf, length);
518 memmove (remaining_buf, remaining_buf + length,
519 remaining_length - length);
520 remaining_length -= length;
521 ret = length;
522 }
523 else
524 {
525 memcpy (buffer, remaining_buf, remaining_length);
526 xfree (remaining_buf);
527 remaining_buf = NULL;
528 ret = remaining_length;
529 }
530 }
531 else
532 {
533 /* Windows (at least XP and Server 2003) has difficulty
534 with large reads from consoles. If a handle is
535 backed by a real console device, overly large reads
536 from the handle will fail and set errno == ENOMEM.
537 On a Windows Server 2003 system where I tested,
538 reading 26608 bytes from the console was OK, but
539 anything above 26609 bytes would fail. The limit has
540 been observed to vary on different systems. So, we
541 limit this read to something smaller than that - by a
542 safe margin, in case the limit depends on system
543 resources or version. */
544 ret = ui_file_read (gdb_stdtargin, (char *) buffer, 16383);
545 if (ret > 0 && (size_t)ret > length)
546 {
547 remaining_buf = (char *) xmalloc (ret - length);
548 remaining_length = ret - length;
549 memcpy (remaining_buf, buffer + length, remaining_length);
550 ret = length;
551 }
552 }
553 }
554 break;
555 default:
556 buffer = (gdb_byte *) xmalloc (length);
557 /* POSIX defines EINTR behaviour of read in a weird way. It's allowed
558 for read() to return -1 even if "some" bytes have been read. It
559 has been corrected in SUSv2 but that doesn't help us much...
560 Therefore a complete solution must check how many bytes have been
561 read on EINTR to return a more reliable value to the target */
562 old_offset = lseek (fd, 0, SEEK_CUR);
563 ret = read (fd, buffer, length);
564 if (ret < 0 && errno == EINTR)
565 {
566 new_offset = lseek (fd, 0, SEEK_CUR);
567 /* If some data has been read, return the number of bytes read.
568 The Ctrl-C flag is set in remote_fileio_reply() anyway. */
569 if (old_offset != new_offset)
570 ret = new_offset - old_offset;
571 }
572 break;
573 }
574
575 if (ret > 0)
576 {
577 errno = target_write_memory (ptrval, buffer, ret);
578 if (errno != 0)
579 ret = -1;
580 }
581
582 if (ret < 0)
583 remote_fileio_return_errno (-1);
584 else
585 remote_fileio_return_success (ret);
586
587 xfree (buffer);
588 }
589
590 static void
591 remote_fileio_func_write (char *buf)
592 {
593 long target_fd, num;
594 LONGEST lnum;
595 CORE_ADDR ptrval;
596 int fd, ret;
597 gdb_byte *buffer;
598 size_t length;
599
600 /* 1. Parameter: file descriptor */
601 if (remote_fileio_extract_int (&buf, &target_fd))
602 {
603 remote_fileio_ioerror ();
604 return;
605 }
606 fd = remote_fileio_map_fd ((int) target_fd);
607 if (fd == FIO_FD_INVALID)
608 {
609 remote_fileio_badfd ();
610 return;
611 }
612 /* 2. Parameter: buffer pointer */
613 if (remote_fileio_extract_long (&buf, &lnum))
614 {
615 remote_fileio_ioerror ();
616 return;
617 }
618 ptrval = (CORE_ADDR) lnum;
619 /* 3. Parameter: buffer length */
620 if (remote_fileio_extract_int (&buf, &num))
621 {
622 remote_fileio_ioerror ();
623 return;
624 }
625 length = (size_t) num;
626
627 buffer = (gdb_byte *) xmalloc (length);
628 if (target_read_memory (ptrval, buffer, length) != 0)
629 {
630 xfree (buffer);
631 remote_fileio_ioerror ();
632 return;
633 }
634
635 switch (fd)
636 {
637 case FIO_FD_CONSOLE_IN:
638 remote_fileio_badfd ();
639 xfree (buffer);
640 return;
641 case FIO_FD_CONSOLE_OUT:
642 ui_file_write (target_fd == 1 ? gdb_stdtarg : gdb_stdtargerr,
643 (char *) buffer, length);
644 gdb_flush (target_fd == 1 ? gdb_stdtarg : gdb_stdtargerr);
645 ret = length;
646 break;
647 default:
648 ret = write (fd, buffer, length);
649 if (ret < 0 && errno == EACCES)
650 errno = EBADF; /* Cygwin returns EACCESS when writing to a
651 R/O file. */
652 break;
653 }
654
655 if (ret < 0)
656 remote_fileio_return_errno (-1);
657 else
658 remote_fileio_return_success (ret);
659
660 xfree (buffer);
661 }
662
663 static void
664 remote_fileio_func_lseek (char *buf)
665 {
666 long num;
667 LONGEST lnum;
668 int fd, flag;
669 off_t offset, ret;
670
671 /* 1. Parameter: file descriptor */
672 if (remote_fileio_extract_int (&buf, &num))
673 {
674 remote_fileio_ioerror ();
675 return;
676 }
677 fd = remote_fileio_map_fd ((int) num);
678 if (fd == FIO_FD_INVALID)
679 {
680 remote_fileio_badfd ();
681 return;
682 }
683 else if (fd == FIO_FD_CONSOLE_IN || fd == FIO_FD_CONSOLE_OUT)
684 {
685 remote_fileio_reply (-1, FILEIO_ESPIPE);
686 return;
687 }
688
689 /* 2. Parameter: offset */
690 if (remote_fileio_extract_long (&buf, &lnum))
691 {
692 remote_fileio_ioerror ();
693 return;
694 }
695 offset = (off_t) lnum;
696 /* 3. Parameter: flag */
697 if (remote_fileio_extract_int (&buf, &num))
698 {
699 remote_fileio_ioerror ();
700 return;
701 }
702 if (remote_fileio_seek_flag_to_host (num, &flag))
703 {
704 remote_fileio_reply (-1, FILEIO_EINVAL);
705 return;
706 }
707
708 ret = lseek (fd, offset, flag);
709
710 if (ret == (off_t) -1)
711 remote_fileio_return_errno (-1);
712 else
713 remote_fileio_return_success (ret);
714 }
715
716 static void
717 remote_fileio_func_rename (char *buf)
718 {
719 CORE_ADDR old_ptr, new_ptr;
720 int old_len, new_len;
721 char *oldpath, *newpath;
722 int ret, of, nf;
723 struct stat ost, nst;
724
725 /* 1. Parameter: Ptr to oldpath / length incl. trailing zero */
726 if (remote_fileio_extract_ptr_w_len (&buf, &old_ptr, &old_len))
727 {
728 remote_fileio_ioerror ();
729 return;
730 }
731
732 /* 2. Parameter: Ptr to newpath / length incl. trailing zero */
733 if (remote_fileio_extract_ptr_w_len (&buf, &new_ptr, &new_len))
734 {
735 remote_fileio_ioerror ();
736 return;
737 }
738
739 /* Request oldpath using 'm' packet */
740 oldpath = (char *) alloca (old_len);
741 if (target_read_memory (old_ptr, (gdb_byte *) oldpath, old_len) != 0)
742 {
743 remote_fileio_ioerror ();
744 return;
745 }
746
747 /* Request newpath using 'm' packet */
748 newpath = (char *) alloca (new_len);
749 if (target_read_memory (new_ptr, (gdb_byte *) newpath, new_len) != 0)
750 {
751 remote_fileio_ioerror ();
752 return;
753 }
754
755 /* Only operate on regular files and directories. */
756 of = stat (oldpath, &ost);
757 nf = stat (newpath, &nst);
758 if ((!of && !S_ISREG (ost.st_mode) && !S_ISDIR (ost.st_mode))
759 || (!nf && !S_ISREG (nst.st_mode) && !S_ISDIR (nst.st_mode)))
760 {
761 remote_fileio_reply (-1, FILEIO_EACCES);
762 return;
763 }
764
765 ret = rename (oldpath, newpath);
766
767 if (ret == -1)
768 {
769 /* Special case: newpath is a non-empty directory. Some systems
770 return ENOTEMPTY, some return EEXIST. We coerce that to be
771 always EEXIST. */
772 if (errno == ENOTEMPTY)
773 errno = EEXIST;
774 #ifdef __CYGWIN__
775 /* Workaround some Cygwin problems with correct errnos. */
776 if (errno == EACCES)
777 {
778 if (!of && !nf && S_ISDIR (nst.st_mode))
779 {
780 if (S_ISREG (ost.st_mode))
781 errno = EISDIR;
782 else
783 {
784 char oldfullpath[PATH_MAX];
785 char newfullpath[PATH_MAX];
786 int len;
787
788 cygwin_conv_path (CCP_WIN_A_TO_POSIX, oldpath, oldfullpath,
789 PATH_MAX);
790 cygwin_conv_path (CCP_WIN_A_TO_POSIX, newpath, newfullpath,
791 PATH_MAX);
792 len = strlen (oldfullpath);
793 if (IS_DIR_SEPARATOR (newfullpath[len])
794 && !filename_ncmp (oldfullpath, newfullpath, len))
795 errno = EINVAL;
796 else
797 errno = EEXIST;
798 }
799 }
800 }
801 #endif
802
803 remote_fileio_return_errno (-1);
804 }
805 else
806 remote_fileio_return_success (ret);
807 }
808
809 static void
810 remote_fileio_func_unlink (char *buf)
811 {
812 CORE_ADDR ptrval;
813 int length;
814 char *pathname;
815 int ret;
816 struct stat st;
817
818 /* Parameter: Ptr to pathname / length incl. trailing zero */
819 if (remote_fileio_extract_ptr_w_len (&buf, &ptrval, &length))
820 {
821 remote_fileio_ioerror ();
822 return;
823 }
824 /* Request pathname using 'm' packet */
825 pathname = (char *) alloca (length);
826 if (target_read_memory (ptrval, (gdb_byte *) pathname, length) != 0)
827 {
828 remote_fileio_ioerror ();
829 return;
830 }
831
832 /* Only operate on regular files (and directories, which allows to return
833 the correct return code). */
834 if (!stat (pathname, &st) && !S_ISREG (st.st_mode) && !S_ISDIR (st.st_mode))
835 {
836 remote_fileio_reply (-1, FILEIO_ENODEV);
837 return;
838 }
839
840 ret = unlink (pathname);
841
842 if (ret == -1)
843 remote_fileio_return_errno (-1);
844 else
845 remote_fileio_return_success (ret);
846 }
847
848 static void
849 remote_fileio_func_stat (char *buf)
850 {
851 CORE_ADDR statptr, nameptr;
852 int ret, namelength;
853 char *pathname;
854 LONGEST lnum;
855 struct stat st;
856 struct fio_stat fst;
857
858 /* 1. Parameter: Ptr to pathname / length incl. trailing zero */
859 if (remote_fileio_extract_ptr_w_len (&buf, &nameptr, &namelength))
860 {
861 remote_fileio_ioerror ();
862 return;
863 }
864
865 /* 2. Parameter: Ptr to struct stat */
866 if (remote_fileio_extract_long (&buf, &lnum))
867 {
868 remote_fileio_ioerror ();
869 return;
870 }
871 statptr = (CORE_ADDR) lnum;
872
873 /* Request pathname using 'm' packet */
874 pathname = (char *) alloca (namelength);
875 if (target_read_memory (nameptr, (gdb_byte *) pathname, namelength) != 0)
876 {
877 remote_fileio_ioerror ();
878 return;
879 }
880
881 ret = stat (pathname, &st);
882
883 if (ret == -1)
884 {
885 remote_fileio_return_errno (-1);
886 return;
887 }
888 /* Only operate on regular files and directories. */
889 if (!ret && !S_ISREG (st.st_mode) && !S_ISDIR (st.st_mode))
890 {
891 remote_fileio_reply (-1, FILEIO_EACCES);
892 return;
893 }
894 if (statptr)
895 {
896 host_to_fileio_stat (&st, &fst);
897 host_to_fileio_uint (0, fst.fst_dev);
898
899 errno = target_write_memory (statptr, (gdb_byte *) &fst, sizeof fst);
900 if (errno != 0)
901 {
902 remote_fileio_return_errno (-1);
903 return;
904 }
905 }
906 remote_fileio_return_success (ret);
907 }
908
909 static void
910 remote_fileio_func_fstat (char *buf)
911 {
912 CORE_ADDR ptrval;
913 int fd, ret;
914 long target_fd;
915 LONGEST lnum;
916 struct stat st;
917 struct fio_stat fst;
918 struct timeval tv;
919
920 /* 1. Parameter: file descriptor */
921 if (remote_fileio_extract_int (&buf, &target_fd))
922 {
923 remote_fileio_ioerror ();
924 return;
925 }
926 fd = remote_fileio_map_fd ((int) target_fd);
927 if (fd == FIO_FD_INVALID)
928 {
929 remote_fileio_badfd ();
930 return;
931 }
932 /* 2. Parameter: Ptr to struct stat */
933 if (remote_fileio_extract_long (&buf, &lnum))
934 {
935 remote_fileio_ioerror ();
936 return;
937 }
938 ptrval = (CORE_ADDR) lnum;
939
940 if (fd == FIO_FD_CONSOLE_IN || fd == FIO_FD_CONSOLE_OUT)
941 {
942 host_to_fileio_uint (1, fst.fst_dev);
943 memset (&st, 0, sizeof (st));
944 st.st_mode = S_IFCHR | (fd == FIO_FD_CONSOLE_IN ? S_IRUSR : S_IWUSR);
945 st.st_nlink = 1;
946 #ifdef HAVE_GETUID
947 st.st_uid = getuid ();
948 #endif
949 #ifdef HAVE_GETGID
950 st.st_gid = getgid ();
951 #endif
952 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
953 st.st_blksize = 512;
954 #endif
955 #if HAVE_STRUCT_STAT_ST_BLOCKS
956 st.st_blocks = 0;
957 #endif
958 if (!gettimeofday (&tv, NULL))
959 st.st_atime = st.st_mtime = st.st_ctime = tv.tv_sec;
960 else
961 st.st_atime = st.st_mtime = st.st_ctime = (time_t) 0;
962 ret = 0;
963 }
964 else
965 ret = fstat (fd, &st);
966
967 if (ret == -1)
968 {
969 remote_fileio_return_errno (-1);
970 return;
971 }
972 if (ptrval)
973 {
974 host_to_fileio_stat (&st, &fst);
975
976 errno = target_write_memory (ptrval, (gdb_byte *) &fst, sizeof fst);
977 if (errno != 0)
978 {
979 remote_fileio_return_errno (-1);
980 return;
981 }
982 }
983 remote_fileio_return_success (ret);
984 }
985
986 static void
987 remote_fileio_func_gettimeofday (char *buf)
988 {
989 LONGEST lnum;
990 CORE_ADDR ptrval;
991 int ret;
992 struct timeval tv;
993 struct fio_timeval ftv;
994
995 /* 1. Parameter: struct timeval pointer */
996 if (remote_fileio_extract_long (&buf, &lnum))
997 {
998 remote_fileio_ioerror ();
999 return;
1000 }
1001 ptrval = (CORE_ADDR) lnum;
1002 /* 2. Parameter: some pointer value... */
1003 if (remote_fileio_extract_long (&buf, &lnum))
1004 {
1005 remote_fileio_ioerror ();
1006 return;
1007 }
1008 /* ...which has to be NULL. */
1009 if (lnum)
1010 {
1011 remote_fileio_reply (-1, FILEIO_EINVAL);
1012 return;
1013 }
1014
1015 ret = gettimeofday (&tv, NULL);
1016
1017 if (ret == -1)
1018 {
1019 remote_fileio_return_errno (-1);
1020 return;
1021 }
1022
1023 if (ptrval)
1024 {
1025 remote_fileio_to_fio_timeval (&tv, &ftv);
1026
1027 errno = target_write_memory (ptrval, (gdb_byte *) &ftv, sizeof ftv);
1028 if (errno != 0)
1029 {
1030 remote_fileio_return_errno (-1);
1031 return;
1032 }
1033 }
1034 remote_fileio_return_success (ret);
1035 }
1036
1037 static void
1038 remote_fileio_func_isatty (char *buf)
1039 {
1040 long target_fd;
1041 int fd;
1042
1043 /* Parameter: file descriptor */
1044 if (remote_fileio_extract_int (&buf, &target_fd))
1045 {
1046 remote_fileio_ioerror ();
1047 return;
1048 }
1049 fd = remote_fileio_map_fd ((int) target_fd);
1050 remote_fileio_return_success (fd == FIO_FD_CONSOLE_IN ||
1051 fd == FIO_FD_CONSOLE_OUT ? 1 : 0);
1052 }
1053
1054 static void
1055 remote_fileio_func_system (char *buf)
1056 {
1057 CORE_ADDR ptrval;
1058 int ret, length;
1059 char *cmdline = NULL;
1060
1061 /* Parameter: Ptr to commandline / length incl. trailing zero */
1062 if (remote_fileio_extract_ptr_w_len (&buf, &ptrval, &length))
1063 {
1064 remote_fileio_ioerror ();
1065 return;
1066 }
1067
1068 if (length)
1069 {
1070 /* Request commandline using 'm' packet */
1071 cmdline = (char *) alloca (length);
1072 if (target_read_memory (ptrval, (gdb_byte *) cmdline, length) != 0)
1073 {
1074 remote_fileio_ioerror ();
1075 return;
1076 }
1077 }
1078
1079 /* Check if system(3) has been explicitely allowed using the
1080 `set remote system-call-allowed 1' command. If length is 0,
1081 indicating a NULL parameter to the system call, return zero to
1082 indicate a shell is not available. Otherwise fail with EPERM. */
1083 if (!remote_fio_system_call_allowed)
1084 {
1085 if (!length)
1086 remote_fileio_return_success (0);
1087 else
1088 remote_fileio_reply (-1, FILEIO_EPERM);
1089 return;
1090 }
1091
1092 ret = system (cmdline);
1093
1094 if (!length)
1095 remote_fileio_return_success (ret);
1096 else if (ret == -1)
1097 remote_fileio_return_errno (-1);
1098 else
1099 remote_fileio_return_success (WEXITSTATUS (ret));
1100 }
1101
1102 static struct {
1103 char *name;
1104 void (*func)(char *);
1105 } remote_fio_func_map[] = {
1106 { "open", remote_fileio_func_open },
1107 { "close", remote_fileio_func_close },
1108 { "read", remote_fileio_func_read },
1109 { "write", remote_fileio_func_write },
1110 { "lseek", remote_fileio_func_lseek },
1111 { "rename", remote_fileio_func_rename },
1112 { "unlink", remote_fileio_func_unlink },
1113 { "stat", remote_fileio_func_stat },
1114 { "fstat", remote_fileio_func_fstat },
1115 { "gettimeofday", remote_fileio_func_gettimeofday },
1116 { "isatty", remote_fileio_func_isatty },
1117 { "system", remote_fileio_func_system },
1118 { NULL, NULL }
1119 };
1120
1121 static int
1122 do_remote_fileio_request (struct ui_out *uiout, void *buf_arg)
1123 {
1124 char *buf = (char *) buf_arg;
1125 char *c;
1126 int idx;
1127
1128 quit_handler = remote_fileio_quit_handler;
1129
1130 c = strchr (++buf, ',');
1131 if (c)
1132 *c++ = '\0';
1133 else
1134 c = strchr (buf, '\0');
1135 for (idx = 0; remote_fio_func_map[idx].name; ++idx)
1136 if (!strcmp (remote_fio_func_map[idx].name, buf))
1137 break;
1138 if (!remote_fio_func_map[idx].name) /* ERROR: No such function. */
1139 return RETURN_ERROR;
1140 remote_fio_func_map[idx].func (c);
1141 return 0;
1142 }
1143
1144 /* Close any open descriptors, and reinitialize the file mapping. */
1145
1146 void
1147 remote_fileio_reset (void)
1148 {
1149 int ix;
1150
1151 for (ix = 0; ix != remote_fio_data.fd_map_size; ix++)
1152 {
1153 int fd = remote_fio_data.fd_map[ix];
1154
1155 if (fd >= 0)
1156 close (fd);
1157 }
1158 if (remote_fio_data.fd_map)
1159 {
1160 xfree (remote_fio_data.fd_map);
1161 remote_fio_data.fd_map = NULL;
1162 remote_fio_data.fd_map_size = 0;
1163 }
1164 }
1165
1166 /* Handle a file I/O request. BUF points to the packet containing the
1167 request. CTRLC_PENDING_P should be nonzero if the target has not
1168 acknowledged the Ctrl-C sent asynchronously earlier. */
1169
1170 void
1171 remote_fileio_request (char *buf, int ctrlc_pending_p)
1172 {
1173 int ex;
1174
1175 /* Save the previous quit handler, so we can restore it. No need
1176 for a cleanup since we catch all exceptions below. Note that the
1177 quit handler is also restored by remote_fileio_reply just before
1178 pushing a packet. */
1179 remote_fileio_o_quit_handler = quit_handler;
1180
1181 if (ctrlc_pending_p)
1182 {
1183 /* If the target hasn't responded to the Ctrl-C sent
1184 asynchronously earlier, take this opportunity to send the
1185 Ctrl-C synchronously. */
1186 set_quit_flag ();
1187 remote_fileio_reply (-1, FILEIO_EINTR);
1188 }
1189 else
1190 {
1191 ex = catch_exceptions (current_uiout,
1192 do_remote_fileio_request, (void *)buf,
1193 RETURN_MASK_ALL);
1194 switch (ex)
1195 {
1196 case RETURN_ERROR:
1197 remote_fileio_reply (-1, FILEIO_ENOSYS);
1198 break;
1199 case RETURN_QUIT:
1200 remote_fileio_reply (-1, FILEIO_EINTR);
1201 break;
1202 default:
1203 break;
1204 }
1205 }
1206
1207 quit_handler = remote_fileio_o_quit_handler;
1208 }
1209 \f
1210
1211 /* Unpack an fio_uint_t. */
1212
1213 static unsigned int
1214 remote_fileio_to_host_uint (fio_uint_t fnum)
1215 {
1216 return extract_unsigned_integer ((gdb_byte *) fnum, 4,
1217 BFD_ENDIAN_BIG);
1218 }
1219
1220 /* Unpack an fio_ulong_t. */
1221
1222 static ULONGEST
1223 remote_fileio_to_host_ulong (fio_ulong_t fnum)
1224 {
1225 return extract_unsigned_integer ((gdb_byte *) fnum, 8,
1226 BFD_ENDIAN_BIG);
1227 }
1228
1229 /* Unpack an fio_mode_t. */
1230
1231 static mode_t
1232 remote_fileio_to_host_mode (fio_mode_t fnum)
1233 {
1234 return remote_fileio_mode_to_host (remote_fileio_to_host_uint (fnum),
1235 0);
1236 }
1237
1238 /* Unpack an fio_time_t. */
1239
1240 static time_t
1241 remote_fileio_to_host_time (fio_time_t fnum)
1242 {
1243 return remote_fileio_to_host_uint (fnum);
1244 }
1245
1246
1247 /* See remote-fileio.h. */
1248
1249 void
1250 remote_fileio_to_host_stat (struct fio_stat *fst, struct stat *st)
1251 {
1252 memset (st, 0, sizeof (struct stat));
1253
1254 st->st_dev = remote_fileio_to_host_uint (fst->fst_dev);
1255 st->st_ino = remote_fileio_to_host_uint (fst->fst_ino);
1256 st->st_mode = remote_fileio_to_host_mode (fst->fst_mode);
1257 st->st_nlink = remote_fileio_to_host_uint (fst->fst_nlink);
1258 st->st_uid = remote_fileio_to_host_uint (fst->fst_uid);
1259 st->st_gid = remote_fileio_to_host_uint (fst->fst_gid);
1260 st->st_rdev = remote_fileio_to_host_uint (fst->fst_rdev);
1261 st->st_size = remote_fileio_to_host_ulong (fst->fst_size);
1262 #ifdef HAVE_STRUCT_STAT_ST_BLKSIZE
1263 st->st_blksize = remote_fileio_to_host_ulong (fst->fst_blksize);
1264 #endif
1265 #if HAVE_STRUCT_STAT_ST_BLOCKS
1266 st->st_blocks = remote_fileio_to_host_ulong (fst->fst_blocks);
1267 #endif
1268 st->st_atime = remote_fileio_to_host_time (fst->fst_atime);
1269 st->st_mtime = remote_fileio_to_host_time (fst->fst_mtime);
1270 st->st_ctime = remote_fileio_to_host_time (fst->fst_ctime);
1271 }
1272 \f
1273
1274 static void
1275 set_system_call_allowed (char *args, int from_tty)
1276 {
1277 if (args)
1278 {
1279 char *arg_end;
1280 int val = strtoul (args, &arg_end, 10);
1281
1282 if (*args && *arg_end == '\0')
1283 {
1284 remote_fio_system_call_allowed = !!val;
1285 return;
1286 }
1287 }
1288 error (_("Illegal argument for \"set remote system-call-allowed\" command"));
1289 }
1290
1291 static void
1292 show_system_call_allowed (char *args, int from_tty)
1293 {
1294 if (args)
1295 error (_("Garbage after \"show remote "
1296 "system-call-allowed\" command: `%s'"), args);
1297 printf_unfiltered ("Calling host system(3) call from target is %sallowed\n",
1298 remote_fio_system_call_allowed ? "" : "not ");
1299 }
1300
1301 void
1302 initialize_remote_fileio (struct cmd_list_element *remote_set_cmdlist,
1303 struct cmd_list_element *remote_show_cmdlist)
1304 {
1305 add_cmd ("system-call-allowed", no_class,
1306 set_system_call_allowed,
1307 _("Set if the host system(3) call is allowed for the target."),
1308 &remote_set_cmdlist);
1309 add_cmd ("system-call-allowed", no_class,
1310 show_system_call_allowed,
1311 _("Show if the host system(3) call is allowed for the target."),
1312 &remote_show_cmdlist);
1313 }