File: syscall_restart.c

package info (click to toggle)
libpulp 0.3.16-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 1,976 kB
  • sloc: ansic: 11,792; python: 1,216; sh: 881; makefile: 871; cpp: 582; asm: 387
file content (81 lines) | stat: -rw-r--r-- 2,918 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
/*
 *  libpulp - User-space Livepatching Library
 *
 *  Copyright (C) 2020-2021 SUSE Software Solutions GmbH
 *
 *  This file is part of libpulp.
 *
 *  libpulp is free software; you can redistribute it and/or
 *  modify it under the terms of the GNU Lesser General Public
 *  License as published by the Free Software Foundation; either
 *  version 2.1 of the License, or (at your option) any later version.
 *
 *  libpulp 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
 *  Lesser General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with libpulp.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <errno.h>
#include <stdio.h>
#include <unistd.h>

#include <libparameters.h>

int
main(void)
{
  char buffer[128] = "X";
  ssize_t ret;

  /* Signal readiness. */
  printf("Waiting for input.\n");

  /*
   * When a thread that is in the middle of a syscall receives a signal,
   * the syscall gets interrupted. On some occasions, such as when there
   * are no handlers registered for the signal, or when the SA_RESTART
   * flag is in effect (see sigaction(2)), the kernel arranges an
   * automatic restarting of the syscall to happen when the thread is
   * rescheduled, and it does so by subtracting one instruction from the
   * program counter.
   *
   * Notice, however, that the subtracting happens only immediately
   * before the rescheduling, and it can't be easily detected from a
   * tracer, such as gdb or libpulp's tools. The tracer sees a program
   * counter that points to the next instruction after the syscall,
   * i.e.: it doesn't see the subtraction.
   *
   * Now, since libpulp hijacks the target process, modifies the
   * program counter of a selected thread, and asks the kernel to resume
   * execution; the subtraction mechanism could be consumed, and the
   * actual restarting of the syscall could be lost. If that happens,
   * programs that do not handle syscall interruption would break.
   * Besides, even if the change in behavior would not be catastrophic,
   * it would make live patching less transparent.
   *
   * The following program intentionally exits in error if the read
   * syscall is not restarted.
   */
  errno = 0;
  ret = read(STDIN_FILENO, buffer, 1);

  if (ret < 0) {
    if (errno == EINTR) {
      printf("read syscall interrupted but not restarted.\n");
    }
    printf("read syscall error code: %d", errno);
    return 1; /* Libpulp should restart syscalls on its own. */
  }
  if (ret == 0) {
    printf("End of file.\n");
    return 1; /* The test driver is not expected to send EOF. */
  }

  int_params(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
  float_params(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
  return 0; /* At least one byte was read. */
}