shim_parser.c 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976
  1. /* -*- mode:c; c-file-style:"k&r"; c-basic-offset: 4; tab-width:4; indent-tabs-mode:nil; mode:auto-fill; fill-column:78; -*- */
  2. /* vim: set ts=4 sw=4 et tw=78 fo=cqt wm=0: */
  3. /* Copyright (C) 2014 OSCAR lab, Stony Brook University
  4. This file is part of Graphene Library OS.
  5. Graphene Library OS is free software: you can redistribute it and/or
  6. modify it under the terms of the GNU General Public License
  7. as published by the Free Software Foundation, either version 3 of the
  8. License, or (at your option) any later version.
  9. Graphene Library OS is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with this program. If not, see <http://www.gnu.org/licenses/>. */
  15. /*
  16. * shim_parser.c
  17. *
  18. * This file contains codes for parsing system call arguements for debug
  19. * purpose.
  20. */
  21. #include <shim_internal.h>
  22. #include <shim_tls.h>
  23. #include <shim_table.h>
  24. #include <shim_thread.h>
  25. #include <shim_utils.h>
  26. #include <pal.h>
  27. #include <pal_error.h>
  28. #include <asm/unistd.h>
  29. #include <asm/prctl.h>
  30. #include <asm/mman.h>
  31. #include <asm/ioctls.h>
  32. #include <linux/futex.h>
  33. #include <linux/wait.h>
  34. #include <errno.h>
  35. #include <fcntl.h>
  36. static void parse_open_flags (const char *, va_list);
  37. static void parse_open_mode (const char *, va_list);
  38. static void parse_access_mode (const char *, va_list);
  39. static void parse_mmap_prot (const char *, va_list);
  40. static void parse_mmap_flags (const char *, va_list);
  41. static void parse_exec_args (const char *, va_list);
  42. static void parse_exec_envp (const char *, va_list);
  43. static void parse_pipe_fds (const char *, va_list);
  44. static void parse_signum (const char *, va_list);
  45. static void parse_sigmask (const char *, va_list);
  46. static void parse_sigprocmask_how (const char *, va_list);
  47. static void parse_timespec (const char *, va_list);
  48. static void parse_sockaddr (const char *, va_list);
  49. static void parse_futexop (const char *, va_list);
  50. static void parse_ioctlop (const char *, va_list);
  51. static void parse_seek (const char *, va_list);
  52. static void parse_at_fdcwd (const char *, va_list);
  53. static void parse_wait_option (const char *, va_list);
  54. struct parser_table {
  55. int slow;
  56. int stop;
  57. void (*parser[6]) (const char *, va_list);
  58. } syscall_parser_table[SHIM_NSYSCALLS] = {
  59. { .slow = 1, .parser = { NULL } }, /* read */
  60. { .slow = 1, .parser = { NULL } }, /* write */
  61. { .slow = 1, /* open */
  62. .parser = { NULL, &parse_open_flags, &parse_open_mode, } },
  63. { .slow = 0, .parser = { NULL } }, /* close */
  64. { .slow = 0, .parser = { NULL } }, /* stat */
  65. { .slow = 0, .parser = { NULL } }, /* fstat */
  66. { .slow = 0, .parser = { NULL } }, /* lstat */
  67. { .slow = 1, .parser = { NULL } }, /* poll */
  68. { .slow = 0, .parser = { NULL, NULL, &parse_seek } }, /* lseek */
  69. { .slow = 0, /* mmap */
  70. .parser = { NULL, NULL, &parse_mmap_prot, &parse_mmap_flags } },
  71. { .slow = 0, /* mprotect */
  72. .parser = { NULL, NULL, &parse_mmap_prot } },
  73. { .slow = 0, .parser = { NULL } }, /* munmap */
  74. { .slow = 0, .parser = { NULL } }, /* brk */
  75. { .slow = 0, .parser = { &parse_signum } }, /* rt_sigaction */
  76. { .slow = 0, /* rt_sigprocmask */
  77. .parser = { &parse_sigprocmask_how, &parse_sigmask, &parse_sigmask } },
  78. { .slow = 0, .parser = { NULL } }, /* rt_sigreturn */
  79. { .slow = 1, .parser = { NULL, &parse_ioctlop } }, /* ioctl */
  80. { .slow = 1, .parser = { NULL } }, /* pread64 */
  81. { .slow = 0, .parser = { NULL } }, /* pwrite64 */
  82. { .slow = 1, .parser = { NULL } }, /* readv */
  83. { .slow = 0, .parser = { NULL } }, /* writev */
  84. { .slow = 0, /* access */
  85. .parser = { NULL, &parse_access_mode } },
  86. { .slow = 0, /* pipe */
  87. .parser = { &parse_pipe_fds } },
  88. { .slow = 0, .parser = { NULL } }, /* select */
  89. { .slow = 0, .parser = { NULL } }, /* sched_yield */
  90. { .slow = 0, .parser = { NULL } }, /* mremap */
  91. { .slow = 0, .parser = { NULL } }, /* msync */
  92. { .slow = 0, .parser = { NULL } }, /* mincore */
  93. { .slow = 0, .parser = { NULL } }, /* madvise */
  94. { .slow = 0, .parser = { NULL } }, /* shmget */
  95. { .slow = 0, .parser = { NULL } }, /* shmat */
  96. { .slow = 0, .parser = { NULL } }, /* shmctl */
  97. { .slow = 0, .parser = { NULL } }, /* dup */
  98. { .slow = 0, .parser = { NULL } }, /* dup2 */
  99. { .slow = 0, .parser = { NULL } }, /* pause */
  100. { .slow = 1,
  101. .parser = { &parse_timespec } }, /* nanosleep */
  102. { .slow = 0, .parser = { NULL } }, /* getitimer */
  103. { .slow = 0, .parser = { NULL } }, /* alarm */
  104. { .slow = 0, .parser = { NULL } }, /* setitimer */
  105. { .slow = 0, .parser = { NULL } }, /* getpid */
  106. { .slow = 0, .parser = { NULL } }, /* sendfile */
  107. { .slow = 0, .parser = { NULL } }, /* socket */
  108. { .slow = 1, .parser = { NULL, &parse_sockaddr } }, /* connect */
  109. { .slow = 1, .parser = { NULL } }, /* accept */
  110. { .slow = 0, .parser = { NULL } }, /* sendto */
  111. { .slow = 0, .parser = { NULL } }, /* recvfrom */
  112. { .slow = 0, .parser = { NULL } }, /* sendmsg */
  113. { .slow = 1, .parser = { NULL } }, /* recvmsg */
  114. { .slow = 0, .parser = { NULL } }, /* shutdown */
  115. { .slow = 0, .parser = { NULL } }, /* bind */
  116. { .slow = 0, .parser = { NULL } }, /* listen */
  117. { .slow = 0, .parser = { NULL } }, /* getsockname */
  118. { .slow = 0, .parser = { NULL } }, /* getpeername */
  119. { .slow = 0, .stop = 3, /* socketpair */
  120. .parser = { NULL, NULL, NULL, &parse_pipe_fds } },
  121. { .slow = 0, .parser = { NULL } }, /* setsockopt */
  122. { .slow = 0, .parser = { NULL } }, /* getsockopt */
  123. { .slow = 1, .parser = { NULL } }, /* clone */
  124. { .slow = 1, .parser = { NULL } }, /* fork */
  125. { .slow = 1, .parser = { NULL } }, /* vfork */
  126. { .slow = 1, /* execve */
  127. .parser = { NULL, &parse_exec_args, &parse_exec_envp, } },
  128. { .slow = 0, .parser = { NULL } }, /* exit */
  129. { .slow = 1, .parser = { NULL, NULL,
  130. &parse_wait_option,
  131. NULL } }, /* wait4 */
  132. { .slow = 0, .parser = { NULL, &parse_signum, } }, /* kill */
  133. { .slow = 0, .parser = { NULL } }, /* uname */
  134. { .slow = 0, .parser = { NULL } }, /* semget */
  135. { .slow = 1, .parser = { NULL } }, /* semop */
  136. { .slow = 0, .parser = { NULL } }, /* semctl */
  137. { .slow = 0, .parser = { NULL } }, /* shmdt */
  138. { .slow = 1, .parser = { NULL } }, /* msgget */
  139. { .slow = 1, .parser = { NULL } }, /* msgsnd */
  140. { .slow = 1, .parser = { NULL } }, /* msgrcv */
  141. { .slow = 1, .parser = { NULL } }, /* msgctl */
  142. { .slow = 0, .parser = { NULL } }, /* fcntl */
  143. { .slow = 0, .parser = { NULL } }, /* flock */
  144. { .slow = 0, .parser = { NULL } }, /* fsync */
  145. { .slow = 0, .parser = { NULL } }, /* fdatasync */
  146. { .slow = 0, .parser = { NULL } }, /* truncate */
  147. { .slow = 0, .parser = { NULL } }, /* ftruncate */
  148. { .slow = 0, .parser = { NULL } }, /* getdents */
  149. { .slow = 0, .parser = { NULL } }, /* getcwd */
  150. { .slow = 0, .parser = { NULL } }, /* chdir */
  151. { .slow = 0, .parser = { NULL } }, /* fchdir */
  152. { .slow = 0, .parser = { NULL } }, /* rename */
  153. { .slow = 0, .parser = { NULL } }, /* mkdir */
  154. { .slow = 0, .parser = { NULL } }, /* rmdir */
  155. { .slow = 0, .parser = { NULL, &parse_open_mode } }, /* creat */
  156. { .slow = 0, .parser = { NULL } }, /* link */
  157. { .slow = 0, .parser = { NULL } }, /* unlink */
  158. { .slow = 0, .parser = { NULL } }, /* symlink */
  159. { .slow = 0, .parser = { NULL } }, /* readlink */
  160. { .slow = 0, .parser = { NULL } }, /* chmod */
  161. { .slow = 0, .parser = { NULL } }, /* fchmod */
  162. { .slow = 0, .parser = { NULL } }, /* chown */
  163. { .slow = 0, .parser = { NULL } }, /* fchown */
  164. { .slow = 0, .parser = { NULL } }, /* lchown */
  165. { .slow = 0, .parser = { NULL } }, /* umask */
  166. { .slow = 0, .parser = { NULL } }, /* gettimeofday */
  167. { .slow = 0, .parser = { NULL } }, /* getrlimit */
  168. { .slow = 0, .parser = { NULL } }, /* getrusage */
  169. { .slow = 0, .parser = { NULL } }, /* sysinfo */
  170. { .slow = 0, .parser = { NULL } }, /* times */
  171. { .slow = 0, .parser = { NULL } }, /* ptrace */
  172. { .slow = 0, .parser = { NULL } }, /* getuid */
  173. { .slow = 0, .parser = { NULL } }, /* syslog */
  174. { .slow = 0, .parser = { NULL } }, /* getgid */
  175. { .slow = 0, .parser = { NULL } }, /* setuid */
  176. { .slow = 0, .parser = { NULL } }, /* setgid */
  177. { .slow = 0, .parser = { NULL } }, /* geteuid */
  178. { .slow = 0, .parser = { NULL } }, /* getegid */
  179. { .slow = 0, .parser = { NULL } }, /* setpgid */
  180. { .slow = 0, .parser = { NULL } }, /* getppid */
  181. { .slow = 0, .parser = { NULL } }, /* getpgrp */
  182. { .slow = 0, .parser = { NULL } }, /* setsid */
  183. { .slow = 0, .parser = { NULL } }, /* setreuid */
  184. { .slow = 0, .parser = { NULL } }, /* setregid */
  185. { .slow = 0, .parser = { NULL } }, /* getgroups */
  186. { .slow = 0, .parser = { NULL } }, /* setgroups */
  187. { .slow = 0, .parser = { NULL } }, /* setresuid */
  188. { .slow = 0, .parser = { NULL } }, /* getresuid */
  189. { .slow = 0, .parser = { NULL } }, /* setresgid */
  190. { .slow = 0, .parser = { NULL } }, /* getresgid */
  191. { .slow = 0, .parser = { NULL } }, /* getpgid */
  192. { .slow = 0, .parser = { NULL } }, /* setfsuid */
  193. { .slow = 0, .parser = { NULL } }, /* setfsgid */
  194. { .slow = 0, .parser = { NULL } }, /* getsid */
  195. { .slow = 0, .parser = { NULL } }, /* capget */
  196. { .slow = 0, .parser = { NULL } }, /* capset */
  197. { .slow = 0, .parser = { NULL } }, /* rt_sigpending */
  198. { .slow = 0, .parser = { NULL } }, /* rt_sigtimedwait */
  199. { .slow = 0, .parser = { NULL } }, /* rt_sigqueueinfo */
  200. { .slow = 0, .parser = { NULL } }, /* rt_sigsuspend */
  201. { .slow = 0, .parser = { NULL } }, /* sigaltstack */
  202. { .slow = 0, .parser = { NULL } }, /* utime */
  203. { .slow = 0, .parser = { NULL } }, /* mknod */
  204. { .slow = 0, .parser = { NULL } }, /* uselib */
  205. { .slow = 0, .parser = { NULL } }, /* personality */
  206. { .slow = 0, .parser = { NULL } }, /* ustat */
  207. { .slow = 0, .parser = { NULL } }, /* statfs */
  208. { .slow = 0, .parser = { NULL } }, /* fstatfs */
  209. { .slow = 0, .parser = { NULL } }, /* sysfs */
  210. { .slow = 0, .parser = { NULL } }, /* getpriority */
  211. { .slow = 0, .parser = { NULL } }, /* setpriority */
  212. { .slow = 0, .parser = { NULL } }, /* sched_setparam */
  213. { .slow = 0, .parser = { NULL } }, /* sched_getparam */
  214. { .slow = 0, .parser = { NULL } }, /* sched_setscheduler */
  215. { .slow = 0, .parser = { NULL } }, /* sched_getscheduler */
  216. { .slow = 0, .parser = { NULL } }, /* sched_get_priority_max */
  217. { .slow = 0, .parser = { NULL } }, /* sched_get_priority_min */
  218. { .slow = 0, .parser = { NULL } }, /* sched_rr_get_interval */
  219. { .slow = 0, .parser = { NULL } }, /* mlock */
  220. { .slow = 0, .parser = { NULL } }, /* munlock */
  221. { .slow = 0, .parser = { NULL } }, /* mlockall */
  222. { .slow = 0, .parser = { NULL } }, /* munlockall */
  223. { .slow = 0, .parser = { NULL } }, /* vhangup */
  224. { .slow = 0, .parser = { NULL } }, /* modify_ldt */
  225. { .slow = 0, .parser = { NULL } }, /* pivot_root */
  226. { .slow = 0, .parser = { NULL } }, /* _sysctl */
  227. { .slow = 0, .parser = { NULL } }, /* prctl */
  228. { .slow = 0, .parser = { NULL } }, /* arch_prctl */
  229. { .slow = 0, .parser = { NULL } }, /* adjtimex */
  230. { .slow = 0, .parser = { NULL } }, /* setrlimit */
  231. { .slow = 0, .parser = { NULL } }, /* chroot */
  232. { .slow = 0, .parser = { NULL } }, /* sync */
  233. { .slow = 0, .parser = { NULL } }, /* acct */
  234. { .slow = 0, .parser = { NULL } }, /* settimeofday */
  235. { .slow = 0, .parser = { NULL } }, /* mount */
  236. { .slow = 0, .parser = { NULL } }, /* umount2 */
  237. { .slow = 0, .parser = { NULL } }, /* swapon */
  238. { .slow = 0, .parser = { NULL } }, /* swapoff */
  239. { .slow = 0, .parser = { NULL } }, /* reboot */
  240. { .slow = 0, .parser = { NULL } }, /* sethostname */
  241. { .slow = 0, .parser = { NULL } }, /* setdomainname */
  242. { .slow = 0, .parser = { NULL } }, /* iopl */
  243. { .slow = 0, .parser = { NULL } }, /* ioperm */
  244. { .slow = 0, .parser = { NULL } }, /* create_module */
  245. { .slow = 0, .parser = { NULL } }, /* init_module */
  246. { .slow = 0, .parser = { NULL } }, /* delete_module */
  247. { .slow = 0, .parser = { NULL } }, /* get_kernel_syms */
  248. { .slow = 0, .parser = { NULL } }, /* query_module */
  249. { .slow = 0, .parser = { NULL } }, /* quotactl */
  250. { .slow = 0, .parser = { NULL } }, /* nfsservctl */
  251. { .slow = 0, .parser = { NULL } }, /* getpmsg */
  252. { .slow = 0, .parser = { NULL } }, /* putpmsg */
  253. { .slow = 0, .parser = { NULL } }, /* afs_syscall */
  254. { .slow = 0, .parser = { NULL } }, /* tuxcall */
  255. { .slow = 0, .parser = { NULL } }, /* security */
  256. { .slow = 0, .parser = { NULL } }, /* gettid */
  257. { .slow = 0, .parser = { NULL } }, /* readahead */
  258. { .slow = 0, .parser = { NULL } }, /* setxattr */
  259. { .slow = 0, .parser = { NULL } }, /* lsetxattr */
  260. { .slow = 0, .parser = { NULL } }, /* fsetxattr */
  261. { .slow = 0, .parser = { NULL } }, /* getxattr */
  262. { .slow = 0, .parser = { NULL } }, /* lgetxattr */
  263. { .slow = 0, .parser = { NULL } }, /* fgetxattr */
  264. { .slow = 0, .parser = { NULL } }, /* listxattr */
  265. { .slow = 0, .parser = { NULL } }, /* llistxattr */
  266. { .slow = 0, .parser = { NULL } }, /* flistxattr */
  267. { .slow = 0, .parser = { NULL } }, /* removexattr */
  268. { .slow = 0, .parser = { NULL } }, /* lremovexattr */
  269. { .slow = 0, .parser = { NULL } }, /* fremovexattr */
  270. { .slow = 0, .parser = { NULL, &parse_signum } }, /* tkill */
  271. { .slow = 0, .parser = { NULL } }, /* time */
  272. { .slow = 1, .parser = { NULL, &parse_futexop } }, /* futex */
  273. { .slow = 0, .parser = { NULL } }, /* sched_setaffinity */
  274. { .slow = 0, .parser = { NULL } }, /* sched_getaffinity */
  275. { .slow = 0, .parser = { NULL } }, /* set_thread_area */
  276. { .slow = 0, .parser = { NULL } }, /* io_setup */
  277. { .slow = 0, .parser = { NULL } }, /* io_destroy */
  278. { .slow = 0, .parser = { NULL } }, /* io_getevents */
  279. { .slow = 0, .parser = { NULL } }, /* io_submit */
  280. { .slow = 0, .parser = { NULL } }, /* io_cancel */
  281. { .slow = 0, .parser = { NULL } }, /* get_thread_area */
  282. { .slow = 0, .parser = { NULL } }, /* lookup_dcookie */
  283. { .slow = 0, .parser = { NULL } }, /* epoll_create */
  284. { .slow = 0, .parser = { NULL } }, /* epoll_ctl_old */
  285. { .slow = 0, .parser = { NULL } }, /* epoll_wait_old */
  286. { .slow = 0, .parser = { NULL } }, /* remap_file_pages */
  287. { .slow = 0, .parser = { NULL } }, /* getdents64 */
  288. { .slow = 0, .parser = { NULL } }, /* set_tid_address */
  289. { .slow = 0, .parser = { NULL } }, /* restart_syscall */
  290. { .slow = 0, .parser = { NULL } }, /* semtimedop */
  291. { .slow = 0, .parser = { NULL } }, /* fadvise64 */
  292. { .slow = 0, .parser = { NULL } }, /* timer_create */
  293. { .slow = 0, .parser = { NULL } }, /* timer_settime */
  294. { .slow = 0, .parser = { NULL } }, /* timer_gettime */
  295. { .slow = 0, .parser = { NULL } }, /* timer_getoverrun */
  296. { .slow = 0, .parser = { NULL } }, /* timer_delete */
  297. { .slow = 0, .parser = { NULL } }, /* clock_settime */
  298. { .slow = 0, .parser = { NULL } }, /* clock_gettime */
  299. { .slow = 0, .parser = { NULL } }, /* clock_getres */
  300. { .slow = 0, .parser = { NULL } }, /* clock_nanosleep */
  301. { .slow = 0, .parser = { NULL } }, /* exit_group */
  302. { .slow = 1, .parser = { NULL } }, /* epoll_wait */
  303. { .slow = 0, .parser = { NULL } }, /* epoll_ctl */
  304. { .slow = 0,
  305. .parser = { NULL, NULL, &parse_signum } }, /* tgkill */
  306. { .slow = 0, .parser = { NULL } }, /* utimes */
  307. { .slow = 0, .parser = { NULL } }, /* vserver */
  308. { .slow = 0, .parser = { NULL } }, /* mbind */
  309. { .slow = 0, .parser = { NULL } }, /* set_mempolicy */
  310. { .slow = 0, .parser = { NULL } }, /* get_mempolicy */
  311. { .slow = 0, .parser = { NULL } }, /* mq_open */
  312. { .slow = 0, .parser = { NULL } }, /* mq_unlink */
  313. { .slow = 0, .parser = { NULL } }, /* mq_timedsend */
  314. { .slow = 0, .parser = { NULL } }, /* mq_timedreceive */
  315. { .slow = 0, .parser = { NULL } }, /* mq_notify */
  316. { .slow = 0, .parser = { NULL } }, /* mq_getsetattr */
  317. { .slow = 0, .parser = { NULL } }, /* kexec_load */
  318. { .slow = 1, .parser = { NULL } }, /* waitid */
  319. { .slow = 0, .parser = { NULL } }, /* add_key */
  320. { .slow = 0, .parser = { NULL } }, /* request_key */
  321. { .slow = 0, .parser = { NULL } }, /* keyctl */
  322. { .slow = 0, .parser = { NULL } }, /* ioprio_set */
  323. { .slow = 0, .parser = { NULL } }, /* ioprio_get */
  324. { .slow = 0, .parser = { NULL } }, /* inotify_init */
  325. { .slow = 0, .parser = { NULL } }, /* inotify_add_watch */
  326. { .slow = 0, .parser = { NULL } }, /* inotify_rm_watch */
  327. { .slow = 0, .parser = { NULL } }, /* migrate_pages */
  328. { .slow = 0, .parser = { &parse_at_fdcwd, NULL,
  329. &parse_open_flags,
  330. &parse_open_mode } }, /* openat */
  331. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* mkdirat */
  332. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* mknodat */
  333. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* fchownat */
  334. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* futimesat */
  335. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* newfstatat */
  336. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* unlinkat */
  337. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* renameat */
  338. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* linkat */
  339. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* symlinkat */
  340. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* readlinkat */
  341. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* fchmodat */
  342. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* faccessat */
  343. { .slow = 0, .parser = { NULL } }, /* pselect6 */
  344. { .slow = 1, .parser = { NULL } }, /* ppoll */
  345. { .slow = 0, .parser = { NULL } }, /* unshare */
  346. { .slow = 0, .parser = { NULL } }, /* set_robust_list */
  347. { .slow = 0, .parser = { NULL } }, /* get_robust_list */
  348. { .slow = 0, .parser = { NULL } }, /* splice */
  349. { .slow = 0, .parser = { NULL } }, /* tee */
  350. { .slow = 0, .parser = { NULL } }, /* sync_file_range */
  351. { .slow = 0, .parser = { NULL } }, /* vmsplice */
  352. { .slow = 0, .parser = { NULL } }, /* move_pages */
  353. { .slow = 0, .parser = { NULL } }, /* utimensat */
  354. { .slow = 1, .parser = { NULL } }, /* epoll_pwait */
  355. { .slow = 0, .parser = { NULL } }, /* signalfd */
  356. { .slow = 0, .parser = { NULL } }, /* timerfd_create */
  357. { .slow = 0, .parser = { NULL } }, /* eventfd */
  358. { .slow = 0, .parser = { NULL } }, /* fallocate */
  359. { .slow = 0, .parser = { NULL } }, /* timerfd_settime */
  360. { .slow = 0, .parser = { NULL } }, /* timerfd_gettime */
  361. { .slow = 1, .parser = { NULL } }, /* accept4 */
  362. { .slow = 0, .parser = { NULL } }, /* signalfd4 */
  363. { .slow = 0, .parser = { NULL } }, /* eventfd2 */
  364. { .slow = 0, .parser = { NULL } }, /* epoll_create1 */
  365. { .slow = 0, .parser = { NULL } }, /* dup3 */
  366. { .slow = 0, .parser = { NULL } }, /* pipe2 */
  367. { .slow = 0, .parser = { NULL } }, /* inotify_init1 */
  368. { .slow = 0, .parser = { NULL } }, /* preadv */
  369. { .slow = 0, .parser = { NULL } }, /* pwritev */
  370. { .slow = 0, .parser = { NULL } }, /* rt_tgsigqueueinfo */
  371. { .slow = 0, .parser = { NULL } }, /* perf_event_open */
  372. { .slow = 0, .parser = { NULL } }, /* recvmmsg */
  373. { .slow = 0, .parser = { NULL } },
  374. { .slow = 1, .parser = { NULL } }, /* checkpoint */
  375. { .slow = 1, .parser = { NULL } }, /* restore */
  376. { .slow = 1, .parser = { NULL } }, /* sandbox_create */
  377. { .slow = 0, .parser = { NULL } }, /* sandbox_attach */
  378. { .slow = 0, .parser = { NULL } }, /* sandbox_current */
  379. { .slow = 1, .parser = { NULL } }, /* msgpersist */
  380. { .slow = 1, .parser = { NULL } }, /* benchmark_ipc */
  381. { .slow = 1, .parser = { NULL } }, /* send_rpc */
  382. { .slow = 1, .parser = { NULL } }, /* recv_rpc */
  383. };
  384. static inline int is_pointer (const char * type)
  385. {
  386. return type[strlen(type) - 1] == '*' || !memcmp(type, "long", 5);
  387. }
  388. #define PRINTF(fmt, ...) \
  389. do { \
  390. debug_printf((fmt), __VA_ARGS__); \
  391. } while (0)
  392. #define PUTS(str) \
  393. do { \
  394. debug_puts((str)); \
  395. } while (0)
  396. #define PUTCH(ch) \
  397. do { \
  398. debug_putch((ch)); \
  399. } while (0)
  400. #define VPRINTF(fmt, ap) \
  401. do { \
  402. debug_vprintf((fmt), (ap)); \
  403. } while (0)
  404. static inline void parse_string_arg (va_list ap)
  405. {
  406. VPRINTF("\"%s\"", ap);
  407. }
  408. static inline void parse_pointer_arg (va_list ap)
  409. {
  410. VPRINTF("%p", ap);
  411. }
  412. static inline void parse_integer_arg (va_list ap)
  413. {
  414. VPRINTF("%d", ap);
  415. }
  416. static inline void parse_syscall_args (va_list ap)
  417. {
  418. const char * arg_type = va_arg(ap, const char *);
  419. if (memcmp(arg_type, "const char *", 13) == 0)
  420. parse_string_arg(ap);
  421. else if (is_pointer(arg_type))
  422. parse_pointer_arg(ap);
  423. else
  424. parse_integer_arg(ap);
  425. }
  426. static inline void skip_syscall_args (va_list ap)
  427. {
  428. const char * arg_type = va_arg (ap, const char *);
  429. if (memcmp(arg_type, "const char *", 13) == 0)
  430. va_arg(ap, const char *);
  431. else if (is_pointer(arg_type))
  432. va_arg(ap, void *);
  433. else
  434. va_arg(ap, int);
  435. }
  436. void sysparser_printf (const char * fmt, ...)
  437. {
  438. va_list ap;
  439. va_start(ap, fmt);
  440. VPRINTF(fmt, ap);
  441. va_end(ap);
  442. }
  443. void parse_syscall_before (int sysno, const char * name, int nr, ...)
  444. {
  445. if (!debug_handle)
  446. return;
  447. struct parser_table * parser = &syscall_parser_table[sysno];
  448. if (!parser->slow && !parser->stop)
  449. return;
  450. va_list ap;
  451. va_start(ap, nr);
  452. PRINTF("---- shim_%s(", name);
  453. for (int i = 0 ; i < nr ; i++) {
  454. if (parser->stop && parser->stop == i)
  455. goto dotdotdot;
  456. if (i)
  457. PUTCH(',');
  458. if (parser->parser[i]) {
  459. const char * type = va_arg(ap, const char *);
  460. (*parser->parser[i])(type, ap);
  461. } else
  462. parse_syscall_args(ap);
  463. }
  464. PUTCH(')');
  465. dotdotdot:
  466. PRINTF(" ...\n", name);
  467. va_end(ap);
  468. }
  469. void parse_syscall_after (int sysno, const char * name, int nr, ...)
  470. {
  471. if (!debug_handle)
  472. return;
  473. struct parser_table * parser = &syscall_parser_table[sysno];
  474. va_list ap;
  475. va_start (ap, nr);
  476. const char * ret_type = va_arg(ap, const char *);
  477. if (parser->slow || parser->stop)
  478. PRINTF("---- return from shim_%s(...", name);
  479. else
  480. PRINTF("---- shim_%s(", name);
  481. unsigned long ret_ptr;
  482. int ret_val;
  483. if (is_pointer(ret_type))
  484. ret_ptr = (unsigned long) va_arg(ap, void *);
  485. else
  486. ret_val = va_arg(ap, int);
  487. if (!parser->slow || parser->stop)
  488. for (int i = 0 ; i < nr ; i++) {
  489. if (parser->stop && i < parser->stop) {
  490. skip_syscall_args (ap);
  491. continue;
  492. }
  493. if (i)
  494. PUTCH(',');
  495. if (parser->parser[i]) {
  496. const char * type = va_arg(ap, const char *);
  497. (*parser->parser[i])(type, ap);
  498. } else
  499. parse_syscall_args(ap);
  500. }
  501. if (is_pointer(ret_type)) {
  502. if (ret_ptr < -4095L)
  503. PRINTF(") = %p\n", ret_ptr);
  504. else
  505. PRINTF(") = %ld (%e)\n", (long) ret_ptr, -ret_ptr);
  506. } else {
  507. if (ret_val >= 0)
  508. PRINTF(") = %d\n", ret_val);
  509. else
  510. PRINTF(") = %d (%e)\n", ret_val, ret_val);
  511. }
  512. va_end (ap);
  513. }
  514. static void parse_open_flags (const char * type, va_list ap)
  515. {
  516. int flags = va_arg (ap, int);
  517. if (flags & O_WRONLY) {
  518. PUTS("O_WRONLY");
  519. flags &= ~O_WRONLY;
  520. } else if (flags & O_RDWR) {
  521. PUTS("O_RDWR");
  522. flags &= ~O_RDWR;
  523. } else
  524. PUTS("O_RDONLY");
  525. if (flags & O_APPEND) {
  526. PUTS("|O_APPEND");
  527. flags &= ~O_APPEND;
  528. }
  529. if (flags & O_CREAT) {
  530. PUTS("|O_CREAT");
  531. flags &= ~O_CREAT;
  532. }
  533. if (flags & O_TRUNC) {
  534. PUTS("|O_TRUNC");
  535. flags &= ~O_TRUNC;
  536. }
  537. if (flags & O_ASYNC) {
  538. PUTS("|O_ASYNC");
  539. flags &= ~O_ASYNC;
  540. }
  541. if (flags & O_EXCL) {
  542. PUTS("|O_EXCL");
  543. flags &= ~O_EXCL;
  544. }
  545. if (flags)
  546. PRINTF("|%o", flags);
  547. }
  548. static void parse_open_mode (const char * type, va_list ap)
  549. {
  550. VPRINTF("%04o", ap);
  551. }
  552. static void parse_access_mode (const char * type, va_list ap)
  553. {
  554. int mode = va_arg (ap, int);
  555. PUTS("F_OK");
  556. if (mode) {
  557. if (mode & R_OK)
  558. PUTS("|R_OK");
  559. if (mode & W_OK)
  560. PUTS("|W_OK");
  561. if (mode & X_OK)
  562. PUTS("|X_OK");
  563. }
  564. }
  565. static void parse_mmap_prot (const char * type, va_list ap)
  566. {
  567. int prot = va_arg (ap, int);
  568. int nflags = 0;
  569. if (prot == PROT_NONE) {
  570. PUTS("PROT_NONE");
  571. return;
  572. }
  573. if (prot & PROT_READ) {
  574. if (nflags++)
  575. PUTS("|");
  576. PUTS("PROT_READ");
  577. }
  578. if (prot & PROT_WRITE) {
  579. if (nflags++)
  580. PUTS("|");
  581. PUTS("PROT_WRITE");
  582. }
  583. if (prot & PROT_EXEC) {
  584. if (nflags++)
  585. PUTS("|");
  586. PUTS("PROT_EXEC");
  587. }
  588. }
  589. static void parse_mmap_flags (const char * type, va_list ap)
  590. {
  591. int flags = va_arg (ap, int);
  592. if (flags & MAP_SHARED) {
  593. PUTS("MAP_SHARED");
  594. flags &= ~MAP_SHARED;
  595. }
  596. else
  597. PUTS("MAP_PRIVATE");
  598. if (flags & MAP_ANONYMOUS) {
  599. PUTS("|MAP_ANON");
  600. flags &= ~MAP_ANONYMOUS;
  601. }
  602. else
  603. PUTS("|MAP_FILE");
  604. if (flags & MAP_FIXED) {
  605. PUTS("|MAP_FIXED");
  606. flags &= ~MAP_FIXED;
  607. }
  608. if (flags)
  609. PRINTF("|%o", flags);
  610. }
  611. static void parse_exec_args (const char * type, va_list ap)
  612. {
  613. const char ** args = va_arg (ap, const char **);
  614. PUTS("[");
  615. for (; *args ; args++) {
  616. PUTS(*args);
  617. PUTS(",");
  618. }
  619. PUTS("]");
  620. }
  621. static void parse_exec_envp (const char * type, va_list ap)
  622. {
  623. const char ** envp = va_arg (ap, const char **);
  624. if (!envp) {
  625. PUTS("NULL");
  626. return;
  627. }
  628. int cnt = 0;
  629. PUTS("[");
  630. for (; *envp ; envp++)
  631. if (cnt++ < 2) {
  632. PUTS(*envp);
  633. PUTS(",");
  634. }
  635. if (cnt > 2)
  636. PRINTF("(%d more)", cnt);
  637. PUTS("]");
  638. }
  639. static void parse_pipe_fds (const char * type, va_list ap)
  640. {
  641. int * fds = va_arg (ap, int *);
  642. PRINTF("[%d, %d]", fds[0], fds[1]);
  643. }
  644. #define S(sig) #sig
  645. const char *const siglist[NUM_KNOWN_SIGS + 1] =
  646. { NULL,
  647. S(SIGHUP),
  648. S(SIGINT),
  649. S(SIGQUIT),
  650. S(SIGILL),
  651. S(SIGTRAP),
  652. S(SIGABRT),
  653. S(SIGBUS),
  654. S(SIGFPE),
  655. S(SIGKILL),
  656. S(SIGUSR1),
  657. S(SIGSEGV),
  658. S(SIGUSR2),
  659. S(SIGPIPE),
  660. S(SIGALRM),
  661. S(SIGTERM),
  662. S(SIGSTKFLT),
  663. S(SIGCHLD),
  664. S(SIGCONT),
  665. S(SIGSTOP),
  666. S(SIGTSTP),
  667. S(SIGTTIN),
  668. S(SIGTTOU), };
  669. static void parse_signum (const char * type, va_list ap)
  670. {
  671. unsigned int signum = va_arg (ap, unsigned int);
  672. if (signum > 0 && signum <= NUM_KNOWN_SIGS)
  673. PUTS(siglist[signum]);
  674. else
  675. PRINTF("Signal %u", signum);
  676. }
  677. static void parse_sigmask (const char * type, va_list ap)
  678. {
  679. sigset_t * sigset = va_arg (ap, sigset_t *);
  680. if (!sigset) {
  681. PUTS("NULL");
  682. return;
  683. }
  684. PUTS("[");
  685. for (int signum = 1 ; signum <= sizeof(sigset) * 8 ; signum++)
  686. if (__sigismember(sigset, signum)) {
  687. if (signum <= NUM_KNOWN_SIGS) {
  688. PUTS(siglist[signum]);
  689. PUTS(",");
  690. } else
  691. PRINTF("Signal %u,", signum);
  692. }
  693. PUTS("]");
  694. }
  695. static void parse_sigprocmask_how (const char * type, va_list ap)
  696. {
  697. int how = va_arg (ap, int);
  698. switch (how) {
  699. case SIG_BLOCK:
  700. PUTS("BLOCK");
  701. break;
  702. case SIG_UNBLOCK:
  703. PUTS("UNBLOCK");
  704. break;
  705. case SIG_SETMASK:
  706. PUTS("SETMASK");
  707. break;
  708. default:
  709. PUTS("<unknown>");
  710. break;
  711. }
  712. }
  713. static void parse_timespec (const char * type, va_list ap)
  714. {
  715. const struct timespec *tv = va_arg (ap, const struct timespec *);
  716. if (!tv) {
  717. PUTS("NULL");
  718. return;
  719. }
  720. PRINTF("[%ld,%lld]", tv->tv_sec, tv->tv_nsec);
  721. }
  722. static void parse_sockaddr (const char * type, va_list ap)
  723. {
  724. const struct sockaddr *addr = va_arg (ap, const struct sockaddr *);
  725. if (!addr) {
  726. PUTS("NULL");
  727. return;
  728. }
  729. switch (addr->sa_family) {
  730. case AF_INET: {
  731. struct sockaddr_in * a = (void *) addr;
  732. unsigned char * ip = (void *) &a->sin_addr.s_addr;
  733. PRINTF("{family=INET,ip=%u.%u.%u.%u,port=htons(%u)}",
  734. ip[0], ip[1], ip[2], ip[3], __ntohs(a->sin_port));
  735. break;
  736. }
  737. case AF_INET6: {
  738. struct sockaddr_in6 * a = (void *) addr;
  739. unsigned short * ip = (void *) &a->sin6_addr.s6_addr;
  740. PRINTF("{family=INET,ip=[%x:%x:%x:%x:%x:%x:%x:%x],"
  741. "port=htons(%u)}",
  742. ip[0], ip[1], ip[2], ip[3], ip[4], ip[5], ip[6],
  743. ip[7], __ntohs(a->sin6_port));
  744. break;
  745. }
  746. case AF_UNIX: {
  747. struct sockaddr_un * a = (void *) addr;
  748. PRINTF("{family=UNIX,path=%s}", a->sun_path);
  749. break;
  750. }
  751. default:
  752. PUTS("UNKNOWN");
  753. break;
  754. }
  755. }
  756. static void parse_futexop (const char * type, va_list ap)
  757. {
  758. int op = va_arg (ap, int);
  759. #ifdef FUTEX_PRIVATE_FLAG
  760. if (op & FUTEX_PRIVATE_FLAG) {
  761. PUTS("FUTEX_PRIVATE|");
  762. op &= ~FUTEX_PRIVATE_FLAG;
  763. }
  764. #endif
  765. #ifdef FUTEX_CLOCK_REALTIME
  766. if (op & FUTEX_CLOCK_REALTIME) {
  767. PUTS("FUTEX_CLOCK_REALTIME|");
  768. op &= ~FUTEX_CLOCK_REALTIME;
  769. }
  770. #endif
  771. op &= FUTEX_CMD_MASK;
  772. switch (op) {
  773. case FUTEX_WAIT:
  774. PUTS("FUTEX_WAIT");
  775. break;
  776. case FUTEX_WAKE:
  777. PUTS("FUTEX_WAKE");
  778. break;
  779. case FUTEX_FD:
  780. PUTS("FUTEX_FD");
  781. break;
  782. case FUTEX_REQUEUE:
  783. PUTS("FUTEX_REQUEUE");
  784. break;
  785. case FUTEX_CMP_REQUEUE:
  786. PUTS("FUTEX_CMP_REQUEUE");
  787. break;
  788. case FUTEX_WAKE_OP:
  789. PUTS("FUTEX_WAKE");
  790. break;
  791. default:
  792. PRINTF("OP %d", op);
  793. break;
  794. }
  795. }
  796. static void parse_ioctlop (const char * type, va_list ap)
  797. {
  798. int op = va_arg (ap, int);
  799. if (op >= TCGETS && op <= TIOCVHANGUP) {
  800. const char * opnames[] = {
  801. "TCGETS", /* 0x5401 */ "TCSETS", /* 0x5402 */
  802. "TCSETSW", /* 0x5403 */ "TCSETSF", /* 0x5404 */
  803. "TCGETA", /* 0x5405 */ "TCSETA", /* 0x5406 */
  804. "TCSETAW", /* 0x5407 */ "TCSETAF", /* 0x5408 */
  805. "TCSBRK", /* 0x5409 */ "TCXONC", /* 0x540A */
  806. "TCFLSH", /* 0x540B */ "TIOCEXCL", /* 0x540C */
  807. "TIOCNXCL", /* 0x540D */ "TIOCSCTTY", /* 0x540E */
  808. "TIOCGPGRP", /* 0x540F */ "TIOCSPGRP", /* 0x5410 */
  809. "TIOCOUTQ", /* 0x5411 */ "TIOCSTI", /* 0x5412 */
  810. "TIOCGWINSZ", /* 0x5413 */ "TIOCSWINSZ", /* 0x5414 */
  811. "TIOCMGET", /* 0x5415 */ "TIOCMBIS", /* 0x5416 */
  812. "TIOCMBIC", /* 0x5417 */ "TIOCMSET", /* 0x5418 */
  813. "TIOCGSOFTCAR", /* 0x5419 */ "TIOCSSOFTCAR", /* 0x541A */
  814. "FIONREAD", /* 0x541B */ "TIOCLINUX", /* 0x541C */
  815. "TIOCCONS", /* 0x541D */ "TIOCGSERIAL", /* 0x541E */
  816. "TIOCSSERIAL", /* 0x541F */ "TIOCPKT", /* 0x5420 */
  817. "FIONBIO", /* 0x5421 */ "TIOCNOTTY", /* 0x5422 */
  818. "TIOCSETD", /* 0x5423 */ "TIOCGETD", /* 0x5424 */
  819. "TCSBRKP", /* 0x5425 */ "",
  820. "TIOCSBRK", /* 0x5427 */ "TIOCCBRK", /* 0x5428 */
  821. "TIOCGSID", /* 0x5429 */ "TCGETS2", /* 0x542A */
  822. "TCSETS2", /* 0x542B */ "TCSETSW2", /* 0x542C */
  823. "TCSETSF2", /* 0x542D */ "TIOCGRS485", /* 0x542E */
  824. "TIOCSRS485", /* 0x542F */ "TIOCGPTN" /* 0x5430 */
  825. "TIOCSPTLCK", /* 0x5431 */ "TCGETX", /* 0x5432 */
  826. "TCSETX", /* 0x5433 */ "TCSETXF", /* 0x5434 */
  827. "TCSETXW", /* 0x5435 */ "TIOCSIG", /* 0x5436 */
  828. "TIOCVHANGUP", /* 0x5437 */
  829. };
  830. PUTS(opnames[op - TCGETS]);
  831. return;
  832. }
  833. PRINTF("OP 0x%04u", op);
  834. }
  835. static void parse_seek (const char * type, va_list ap)
  836. {
  837. int seek = va_arg (ap, int);
  838. switch(seek) {
  839. case SEEK_CUR:
  840. PUTS("SEEK_CUR");
  841. break;
  842. case SEEK_SET:
  843. PUTS("SEEK_SET");
  844. break;
  845. case SEEK_END:
  846. PUTS("SEEK_END");
  847. break;
  848. default:
  849. PRINTF("%d", seek);
  850. break;
  851. }
  852. }
  853. static void parse_at_fdcwd (const char * type, va_list ap)
  854. {
  855. int fd = va_arg (ap, int);
  856. switch(fd) {
  857. case AT_FDCWD:
  858. PUTS("AT_FDCWD");
  859. break;
  860. default:
  861. PRINTF("%d", fd);
  862. break;
  863. }
  864. }
  865. static void parse_wait_option (const char * type, va_list ap)
  866. {
  867. int option = va_arg (ap, int);
  868. if (option & WNOHANG)
  869. PUTS("WNOHANG");
  870. }