shim_parser.c 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275
  1. /* Copyright (C) 2014 Stony Brook University
  2. This file is part of Graphene Library OS.
  3. Graphene Library OS is free software: you can redistribute it and/or
  4. modify it under the terms of the GNU Lesser General Public License
  5. as published by the Free Software Foundation, either version 3 of the
  6. License, or (at your option) any later version.
  7. Graphene Library OS is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU Lesser General Public License for more details.
  11. You should have received a copy of the GNU Lesser General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>. */
  13. /*
  14. * shim_parser.c
  15. *
  16. * This file contains codes for parsing system call arguements for debug
  17. * purpose.
  18. */
  19. #include <shim_internal.h>
  20. #include <shim_tls.h>
  21. #include <shim_table.h>
  22. #include <shim_thread.h>
  23. #include <shim_utils.h>
  24. #include <pal.h>
  25. #include <pal_error.h>
  26. #include <errno.h>
  27. #include <asm/fcntl.h>
  28. #include <asm/unistd.h>
  29. #include <asm/prctl.h>
  30. #include <asm/mman.h>
  31. #include <asm/ioctls.h>
  32. #include <linux/stat.h>
  33. #include <linux/fcntl.h>
  34. #include <linux/sched.h>
  35. #include <linux/futex.h>
  36. #include <linux/wait.h>
  37. #include <linux/in.h>
  38. #include <linux/in6.h>
  39. #include <linux/un.h>
  40. static void parse_open_flags (const char *, va_list *);
  41. static void parse_open_mode (const char *, va_list *);
  42. static void parse_access_mode (const char *, va_list *);
  43. static void parse_clone_flags (const char *, va_list *);
  44. static void parse_mmap_prot (const char *, va_list *);
  45. static void parse_mmap_flags (const char *, va_list *);
  46. static void parse_exec_args (const char *, va_list *);
  47. static void parse_exec_envp (const char *, va_list *);
  48. static void parse_pipe_fds (const char *, va_list *);
  49. static void parse_signum (const char *, va_list *);
  50. static void parse_sigmask (const char *, va_list *);
  51. static void parse_sigprocmask_how (const char *, va_list *);
  52. static void parse_timespec (const char *, va_list *);
  53. static void parse_sockaddr (const char *, va_list *);
  54. static void parse_domain (const char *, va_list *);
  55. static void parse_socktype (const char *, va_list *);
  56. static void parse_futexop (const char *, va_list *);
  57. static void parse_ioctlop (const char *, va_list *);
  58. static void parse_fcntlop (const char *, va_list *);
  59. static void parse_seek (const char *, va_list *);
  60. static void parse_at_fdcwd (const char *, va_list *);
  61. static void parse_wait_option (const char *, va_list *);
  62. struct parser_table {
  63. int slow;
  64. int stop;
  65. void (*parser[6]) (const char *, va_list *);
  66. } syscall_parser_table[LIBOS_SYSCALL_BOUND] = {
  67. { .slow = 1, .parser = { NULL } }, /* read */
  68. { .slow = 1, .parser = { NULL } }, /* write */
  69. { .slow = 1, /* open */
  70. .parser = { NULL, &parse_open_flags, &parse_open_mode, } },
  71. { .slow = 0, .parser = { NULL } }, /* close */
  72. { .slow = 0, .parser = { NULL } }, /* stat */
  73. { .slow = 0, .parser = { NULL } }, /* fstat */
  74. { .slow = 0, .parser = { NULL } }, /* lstat */
  75. { .slow = 1, .parser = { NULL } }, /* poll */
  76. { .slow = 0, .parser = { NULL, NULL, &parse_seek } }, /* lseek */
  77. { .slow = 1, /* mmap */
  78. .parser = { NULL, NULL, &parse_mmap_prot, &parse_mmap_flags } },
  79. { .slow = 1, /* mprotect */
  80. .parser = { NULL, NULL, &parse_mmap_prot } },
  81. { .slow = 1, .parser = { NULL } }, /* munmap */
  82. { .slow = 0, .parser = { NULL } }, /* brk */
  83. { .slow = 0, .parser = { &parse_signum } }, /* rt_sigaction */
  84. { .slow = 0, /* rt_sigprocmask */
  85. .parser = { &parse_sigprocmask_how, &parse_sigmask, &parse_sigmask } },
  86. { .slow = 0, .parser = { NULL } }, /* rt_sigreturn */
  87. { .slow = 1, .parser = { NULL, &parse_ioctlop } }, /* ioctl */
  88. { .slow = 1, .parser = { NULL } }, /* pread64 */
  89. { .slow = 0, .parser = { NULL } }, /* pwrite64 */
  90. { .slow = 1, .parser = { NULL } }, /* readv */
  91. { .slow = 0, .parser = { NULL } }, /* writev */
  92. { .slow = 0, /* access */
  93. .parser = { NULL, &parse_access_mode } },
  94. { .slow = 0, /* pipe */
  95. .parser = { &parse_pipe_fds } },
  96. { .slow = 0, .parser = { NULL } }, /* select */
  97. { .slow = 0, .parser = { NULL } }, /* sched_yield */
  98. { .slow = 0, .parser = { NULL } }, /* mremap */
  99. { .slow = 0, .parser = { NULL } }, /* msync */
  100. { .slow = 0, .parser = { NULL } }, /* mincore */
  101. { .slow = 0, .parser = { NULL } }, /* madvise */
  102. { .slow = 0, .parser = { NULL } }, /* shmget */
  103. { .slow = 0, .parser = { NULL } }, /* shmat */
  104. { .slow = 0, .parser = { NULL } }, /* shmctl */
  105. { .slow = 0, .parser = { NULL } }, /* dup */
  106. { .slow = 0, .parser = { NULL } }, /* dup2 */
  107. { .slow = 1, .parser = { NULL } }, /* pause */
  108. { .slow = 1,
  109. .parser = { &parse_timespec } }, /* nanosleep */
  110. { .slow = 0, .parser = { NULL } }, /* getitimer */
  111. { .slow = 0, .parser = { NULL } }, /* alarm */
  112. { .slow = 0, .parser = { NULL } }, /* setitimer */
  113. { .slow = 0, .parser = { NULL } }, /* getpid */
  114. { .slow = 0, .parser = { NULL } }, /* sendfile */
  115. { .slow = 0, .parser = { &parse_domain, &parse_socktype } }, /* socket */
  116. { .slow = 1, .parser = { NULL, &parse_sockaddr } }, /* connect */
  117. { .slow = 1, .parser = { NULL } }, /* accept */
  118. { .slow = 0, .parser = { NULL } }, /* sendto */
  119. { .slow = 0, .parser = { NULL } }, /* recvfrom */
  120. { .slow = 0, .parser = { NULL } }, /* sendmsg */
  121. { .slow = 1, .parser = { NULL } }, /* recvmsg */
  122. { .slow = 0, .parser = { NULL } }, /* shutdown */
  123. { .slow = 0, .parser = { NULL } }, /* bind */
  124. { .slow = 0, .parser = { NULL } }, /* listen */
  125. { .slow = 0, .parser = { NULL } }, /* getsockname */
  126. { .slow = 0, .parser = { NULL } }, /* getpeername */
  127. { .slow = 0, .stop = 3, /* socketpair */
  128. .parser = { &parse_domain, &parse_socktype, NULL, &parse_pipe_fds } },
  129. { .slow = 0, .parser = { NULL } }, /* setsockopt */
  130. { .slow = 0, .parser = { NULL } }, /* getsockopt */
  131. { .slow = 1, .parser = { &parse_clone_flags } }, /* clone */
  132. { .slow = 1, .parser = { NULL } }, /* fork */
  133. { .slow = 1, .parser = { NULL } }, /* vfork */
  134. { .slow = 1, /* execve */
  135. .parser = { NULL, &parse_exec_args, &parse_exec_envp, } },
  136. { .slow = 0, .parser = { NULL } }, /* exit */
  137. { .slow = 1, .parser = { NULL, NULL,
  138. &parse_wait_option,
  139. NULL } }, /* wait4 */
  140. { .slow = 0, .parser = { NULL, &parse_signum, } }, /* kill */
  141. { .slow = 0, .parser = { NULL } }, /* uname */
  142. { .slow = 0, .parser = { NULL } }, /* semget */
  143. { .slow = 1, .parser = { NULL } }, /* semop */
  144. { .slow = 0, .parser = { NULL } }, /* semctl */
  145. { .slow = 0, .parser = { NULL } }, /* shmdt */
  146. { .slow = 1, .parser = { NULL } }, /* msgget */
  147. { .slow = 1, .parser = { NULL } }, /* msgsnd */
  148. { .slow = 1, .parser = { NULL } }, /* msgrcv */
  149. { .slow = 1, .parser = { NULL } }, /* msgctl */
  150. { .slow = 0, .parser = { NULL, &parse_fcntlop } }, /* fcntl */
  151. { .slow = 0, .parser = { NULL } }, /* flock */
  152. { .slow = 0, .parser = { NULL } }, /* fsync */
  153. { .slow = 0, .parser = { NULL } }, /* fdatasync */
  154. { .slow = 0, .parser = { NULL } }, /* truncate */
  155. { .slow = 0, .parser = { NULL } }, /* ftruncate */
  156. { .slow = 0, .parser = { NULL } }, /* getdents */
  157. { .slow = 0, .parser = { NULL } }, /* getcwd */
  158. { .slow = 0, .parser = { NULL } }, /* chdir */
  159. { .slow = 0, .parser = { NULL } }, /* fchdir */
  160. { .slow = 0, .parser = { NULL } }, /* rename */
  161. { .slow = 0, .parser = { NULL } }, /* mkdir */
  162. { .slow = 0, .parser = { NULL } }, /* rmdir */
  163. { .slow = 0, .parser = { NULL, &parse_open_mode } }, /* creat */
  164. { .slow = 0, .parser = { NULL } }, /* link */
  165. { .slow = 0, .parser = { NULL } }, /* unlink */
  166. { .slow = 0, .parser = { NULL } }, /* symlink */
  167. { .slow = 0, .parser = { NULL } }, /* readlink */
  168. { .slow = 0, .parser = { NULL } }, /* chmod */
  169. { .slow = 0, .parser = { NULL } }, /* fchmod */
  170. { .slow = 0, .parser = { NULL } }, /* chown */
  171. { .slow = 0, .parser = { NULL } }, /* fchown */
  172. { .slow = 0, .parser = { NULL } }, /* lchown */
  173. { .slow = 0, .parser = { NULL } }, /* umask */
  174. { .slow = 0, .parser = { NULL } }, /* gettimeofday */
  175. { .slow = 0, .parser = { NULL } }, /* getrlimit */
  176. { .slow = 0, .parser = { NULL } }, /* getrusage */
  177. { .slow = 0, .parser = { NULL } }, /* sysinfo */
  178. { .slow = 0, .parser = { NULL } }, /* times */
  179. { .slow = 0, .parser = { NULL } }, /* ptrace */
  180. { .slow = 0, .parser = { NULL } }, /* getuid */
  181. { .slow = 0, .parser = { NULL } }, /* syslog */
  182. { .slow = 0, .parser = { NULL } }, /* getgid */
  183. { .slow = 0, .parser = { NULL } }, /* setuid */
  184. { .slow = 0, .parser = { NULL } }, /* setgid */
  185. { .slow = 0, .parser = { NULL } }, /* geteuid */
  186. { .slow = 0, .parser = { NULL } }, /* getegid */
  187. { .slow = 0, .parser = { NULL } }, /* setpgid */
  188. { .slow = 0, .parser = { NULL } }, /* getppid */
  189. { .slow = 0, .parser = { NULL } }, /* getpgrp */
  190. { .slow = 0, .parser = { NULL } }, /* setsid */
  191. { .slow = 0, .parser = { NULL } }, /* setreuid */
  192. { .slow = 0, .parser = { NULL } }, /* setregid */
  193. { .slow = 0, .parser = { NULL } }, /* getgroups */
  194. { .slow = 0, .parser = { NULL } }, /* setgroups */
  195. { .slow = 0, .parser = { NULL } }, /* setresuid */
  196. { .slow = 0, .parser = { NULL } }, /* getresuid */
  197. { .slow = 0, .parser = { NULL } }, /* setresgid */
  198. { .slow = 0, .parser = { NULL } }, /* getresgid */
  199. { .slow = 0, .parser = { NULL } }, /* getpgid */
  200. { .slow = 0, .parser = { NULL } }, /* setfsuid */
  201. { .slow = 0, .parser = { NULL } }, /* setfsgid */
  202. { .slow = 0, .parser = { NULL } }, /* getsid */
  203. { .slow = 0, .parser = { NULL } }, /* capget */
  204. { .slow = 0, .parser = { NULL } }, /* capset */
  205. { .slow = 0, .parser = { NULL } }, /* rt_sigpending */
  206. { .slow = 0, .parser = { NULL } }, /* rt_sigtimedwait */
  207. { .slow = 0, .parser = { NULL } }, /* rt_sigqueueinfo */
  208. { .slow = 1, .parser = { NULL } }, /* rt_sigsuspend */
  209. { .slow = 0, .parser = { NULL } }, /* sigaltstack */
  210. { .slow = 0, .parser = { NULL } }, /* utime */
  211. { .slow = 0, .parser = { NULL } }, /* mknod */
  212. { .slow = 0, .parser = { NULL } }, /* uselib */
  213. { .slow = 0, .parser = { NULL } }, /* personality */
  214. { .slow = 0, .parser = { NULL } }, /* ustat */
  215. { .slow = 0, .parser = { NULL } }, /* statfs */
  216. { .slow = 0, .parser = { NULL } }, /* fstatfs */
  217. { .slow = 0, .parser = { NULL } }, /* sysfs */
  218. { .slow = 0, .parser = { NULL } }, /* getpriority */
  219. { .slow = 0, .parser = { NULL } }, /* setpriority */
  220. { .slow = 0, .parser = { NULL } }, /* sched_setparam */
  221. { .slow = 0, .parser = { NULL } }, /* sched_getparam */
  222. { .slow = 0, .parser = { NULL } }, /* sched_setscheduler */
  223. { .slow = 0, .parser = { NULL } }, /* sched_getscheduler */
  224. { .slow = 0, .parser = { NULL } }, /* sched_get_priority_max */
  225. { .slow = 0, .parser = { NULL } }, /* sched_get_priority_min */
  226. { .slow = 0, .parser = { NULL } }, /* sched_rr_get_interval */
  227. { .slow = 0, .parser = { NULL } }, /* mlock */
  228. { .slow = 0, .parser = { NULL } }, /* munlock */
  229. { .slow = 0, .parser = { NULL } }, /* mlockall */
  230. { .slow = 0, .parser = { NULL } }, /* munlockall */
  231. { .slow = 0, .parser = { NULL } }, /* vhangup */
  232. { .slow = 0, .parser = { NULL } }, /* modify_ldt */
  233. { .slow = 0, .parser = { NULL } }, /* pivot_root */
  234. { .slow = 0, .parser = { NULL } }, /* _sysctl */
  235. { .slow = 0, .parser = { NULL } }, /* prctl */
  236. { .slow = 0, .parser = { NULL } }, /* arch_prctl */
  237. { .slow = 0, .parser = { NULL } }, /* adjtimex */
  238. { .slow = 0, .parser = { NULL } }, /* setrlimit */
  239. { .slow = 0, .parser = { NULL } }, /* chroot */
  240. { .slow = 0, .parser = { NULL } }, /* sync */
  241. { .slow = 0, .parser = { NULL } }, /* acct */
  242. { .slow = 0, .parser = { NULL } }, /* settimeofday */
  243. { .slow = 0, .parser = { NULL } }, /* mount */
  244. { .slow = 0, .parser = { NULL } }, /* umount2 */
  245. { .slow = 0, .parser = { NULL } }, /* swapon */
  246. { .slow = 0, .parser = { NULL } }, /* swapoff */
  247. { .slow = 0, .parser = { NULL } }, /* reboot */
  248. { .slow = 0, .parser = { NULL } }, /* sethostname */
  249. { .slow = 0, .parser = { NULL } }, /* setdomainname */
  250. { .slow = 0, .parser = { NULL } }, /* iopl */
  251. { .slow = 0, .parser = { NULL } }, /* ioperm */
  252. { .slow = 0, .parser = { NULL } }, /* create_module */
  253. { .slow = 0, .parser = { NULL } }, /* init_module */
  254. { .slow = 0, .parser = { NULL } }, /* delete_module */
  255. { .slow = 0, .parser = { NULL } }, /* get_kernel_syms */
  256. { .slow = 0, .parser = { NULL } }, /* query_module */
  257. { .slow = 0, .parser = { NULL } }, /* quotactl */
  258. { .slow = 0, .parser = { NULL } }, /* nfsservctl */
  259. { .slow = 0, .parser = { NULL } }, /* getpmsg */
  260. { .slow = 0, .parser = { NULL } }, /* putpmsg */
  261. { .slow = 0, .parser = { NULL } }, /* afs_syscall */
  262. { .slow = 0, .parser = { NULL } }, /* tuxcall */
  263. { .slow = 0, .parser = { NULL } }, /* security */
  264. { .slow = 0, .parser = { NULL } }, /* gettid */
  265. { .slow = 0, .parser = { NULL } }, /* readahead */
  266. { .slow = 0, .parser = { NULL } }, /* setxattr */
  267. { .slow = 0, .parser = { NULL } }, /* lsetxattr */
  268. { .slow = 0, .parser = { NULL } }, /* fsetxattr */
  269. { .slow = 0, .parser = { NULL } }, /* getxattr */
  270. { .slow = 0, .parser = { NULL } }, /* lgetxattr */
  271. { .slow = 0, .parser = { NULL } }, /* fgetxattr */
  272. { .slow = 0, .parser = { NULL } }, /* listxattr */
  273. { .slow = 0, .parser = { NULL } }, /* llistxattr */
  274. { .slow = 0, .parser = { NULL } }, /* flistxattr */
  275. { .slow = 0, .parser = { NULL } }, /* removexattr */
  276. { .slow = 0, .parser = { NULL } }, /* lremovexattr */
  277. { .slow = 0, .parser = { NULL } }, /* fremovexattr */
  278. { .slow = 0, .parser = { NULL, &parse_signum } }, /* tkill */
  279. { .slow = 0, .parser = { NULL } }, /* time */
  280. { .slow = 1, .parser = { NULL, &parse_futexop } }, /* futex */
  281. { .slow = 0, .parser = { NULL } }, /* sched_setaffinity */
  282. { .slow = 0, .parser = { NULL } }, /* sched_getaffinity */
  283. { .slow = 0, .parser = { NULL } }, /* set_thread_area */
  284. { .slow = 0, .parser = { NULL } }, /* io_setup */
  285. { .slow = 0, .parser = { NULL } }, /* io_destroy */
  286. { .slow = 0, .parser = { NULL } }, /* io_getevents */
  287. { .slow = 0, .parser = { NULL } }, /* io_submit */
  288. { .slow = 0, .parser = { NULL } }, /* io_cancel */
  289. { .slow = 0, .parser = { NULL } }, /* get_thread_area */
  290. { .slow = 0, .parser = { NULL } }, /* lookup_dcookie */
  291. { .slow = 0, .parser = { NULL } }, /* epoll_create */
  292. { .slow = 0, .parser = { NULL } }, /* epoll_ctl_old */
  293. { .slow = 0, .parser = { NULL } }, /* epoll_wait_old */
  294. { .slow = 0, .parser = { NULL } }, /* remap_file_pages */
  295. { .slow = 0, .parser = { NULL } }, /* getdents64 */
  296. { .slow = 0, .parser = { NULL } }, /* set_tid_address */
  297. { .slow = 0, .parser = { NULL } }, /* restart_syscall */
  298. { .slow = 0, .parser = { NULL } }, /* semtimedop */
  299. { .slow = 0, .parser = { NULL } }, /* fadvise64 */
  300. { .slow = 0, .parser = { NULL } }, /* timer_create */
  301. { .slow = 0, .parser = { NULL } }, /* timer_settime */
  302. { .slow = 0, .parser = { NULL } }, /* timer_gettime */
  303. { .slow = 0, .parser = { NULL } }, /* timer_getoverrun */
  304. { .slow = 0, .parser = { NULL } }, /* timer_delete */
  305. { .slow = 0, .parser = { NULL } }, /* clock_settime */
  306. { .slow = 0, .parser = { NULL } }, /* clock_gettime */
  307. { .slow = 0, .parser = { NULL } }, /* clock_getres */
  308. { .slow = 0, .parser = { NULL } }, /* clock_nanosleep */
  309. { .slow = 0, .parser = { NULL } }, /* exit_group */
  310. { .slow = 1, .parser = { NULL } }, /* epoll_wait */
  311. { .slow = 0, .parser = { NULL } }, /* epoll_ctl */
  312. { .slow = 0,
  313. .parser = { NULL, NULL, &parse_signum } }, /* tgkill */
  314. { .slow = 0, .parser = { NULL } }, /* utimes */
  315. { .slow = 0, .parser = { NULL } }, /* vserver */
  316. { .slow = 0, .parser = { NULL } }, /* mbind */
  317. { .slow = 0, .parser = { NULL } }, /* set_mempolicy */
  318. { .slow = 0, .parser = { NULL } }, /* get_mempolicy */
  319. { .slow = 0, .parser = { NULL } }, /* mq_open */
  320. { .slow = 0, .parser = { NULL } }, /* mq_unlink */
  321. { .slow = 0, .parser = { NULL } }, /* mq_timedsend */
  322. { .slow = 0, .parser = { NULL } }, /* mq_timedreceive */
  323. { .slow = 0, .parser = { NULL } }, /* mq_notify */
  324. { .slow = 0, .parser = { NULL } }, /* mq_getsetattr */
  325. { .slow = 0, .parser = { NULL } }, /* kexec_load */
  326. { .slow = 1, .parser = { NULL } }, /* waitid */
  327. { .slow = 0, .parser = { NULL } }, /* add_key */
  328. { .slow = 0, .parser = { NULL } }, /* request_key */
  329. { .slow = 0, .parser = { NULL } }, /* keyctl */
  330. { .slow = 0, .parser = { NULL } }, /* ioprio_set */
  331. { .slow = 0, .parser = { NULL } }, /* ioprio_get */
  332. { .slow = 0, .parser = { NULL } }, /* inotify_init */
  333. { .slow = 0, .parser = { NULL } }, /* inotify_add_watch */
  334. { .slow = 0, .parser = { NULL } }, /* inotify_rm_watch */
  335. { .slow = 0, .parser = { NULL } }, /* migrate_pages */
  336. { .slow = 0, .parser = { &parse_at_fdcwd, NULL,
  337. &parse_open_flags,
  338. &parse_open_mode } }, /* openat */
  339. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* mkdirat */
  340. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* mknodat */
  341. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* fchownat */
  342. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* futimesat */
  343. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* newfstatat */
  344. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* unlinkat */
  345. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* renameat */
  346. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* linkat */
  347. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* symlinkat */
  348. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* readlinkat */
  349. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* fchmodat */
  350. { .slow = 0, .parser = { &parse_at_fdcwd, } }, /* faccessat */
  351. { .slow = 0, .parser = { NULL } }, /* pselect6 */
  352. { .slow = 1, .parser = { NULL } }, /* ppoll */
  353. { .slow = 0, .parser = { NULL } }, /* unshare */
  354. { .slow = 0, .parser = { NULL } }, /* set_robust_list */
  355. { .slow = 0, .parser = { NULL } }, /* get_robust_list */
  356. { .slow = 0, .parser = { NULL } }, /* splice */
  357. { .slow = 0, .parser = { NULL } }, /* tee */
  358. { .slow = 0, .parser = { NULL } }, /* sync_file_range */
  359. { .slow = 0, .parser = { NULL } }, /* vmsplice */
  360. { .slow = 0, .parser = { NULL } }, /* move_pages */
  361. { .slow = 0, .parser = { NULL } }, /* utimensat */
  362. { .slow = 1, .parser = { NULL } }, /* epoll_pwait */
  363. { .slow = 0, .parser = { NULL } }, /* signalfd */
  364. { .slow = 0, .parser = { NULL } }, /* timerfd_create */
  365. { .slow = 0, .parser = { NULL } }, /* eventfd */
  366. { .slow = 0, .parser = { NULL } }, /* fallocate */
  367. { .slow = 0, .parser = { NULL } }, /* timerfd_settime */
  368. { .slow = 0, .parser = { NULL } }, /* timerfd_gettime */
  369. { .slow = 1, .parser = { NULL } }, /* accept4 */
  370. { .slow = 0, .parser = { NULL } }, /* signalfd4 */
  371. { .slow = 0, .parser = { NULL } }, /* eventfd2 */
  372. { .slow = 0, .parser = { NULL } }, /* epoll_create1 */
  373. { .slow = 0, .parser = { NULL } }, /* dup3 */
  374. { .slow = 0, .parser = { NULL } }, /* pipe2 */
  375. { .slow = 0, .parser = { NULL } }, /* inotify_init1 */
  376. { .slow = 0, .parser = { NULL } }, /* preadv */
  377. { .slow = 0, .parser = { NULL } }, /* pwritev */
  378. { .slow = 0, .parser = { NULL } }, /* rt_tgsigqueueinfo */
  379. { .slow = 0, .parser = { NULL } }, /* perf_event_open */
  380. { .slow = 0, .parser = { NULL } }, /* recvmmsg */
  381. [LIBOS_SYSCALL_BASE] = { .slow = 0, .parser = { NULL } },
  382. { .slow = 1, .parser = { NULL } }, /* checkpoint */
  383. { .slow = 1, .parser = { NULL } }, /* restore */
  384. { .slow = 1, .parser = { NULL } }, /* sandbox_create */
  385. { .slow = 0, .parser = { NULL } }, /* sandbox_attach */
  386. { .slow = 0, .parser = { NULL } }, /* sandbox_current */
  387. { .slow = 1, .parser = { NULL } }, /* msgpersist */
  388. { .slow = 1, .parser = { NULL } }, /* benchmark_ipc */
  389. { .slow = 1, .parser = { NULL } }, /* send_rpc */
  390. { .slow = 1, .parser = { NULL } }, /* recv_rpc */
  391. };
  392. static inline int is_pointer (const char * type)
  393. {
  394. return type[strlen(type) - 1] == '*'
  395. || strcmp_static(type, "long")
  396. || strcmp_static(type, "unsigned long");
  397. }
  398. #define PRINTF(fmt, ...) \
  399. do { \
  400. debug_printf(fmt, __VA_ARGS__); \
  401. } while (0)
  402. #define PUTS(str) \
  403. do { \
  404. debug_puts(str); \
  405. } while (0)
  406. #define PUTCH(ch) \
  407. do { \
  408. debug_putch(ch); \
  409. } while (0)
  410. #define VPRINTF(fmt, ap) \
  411. do { \
  412. debug_vprintf(fmt, ap); \
  413. } while (0)
  414. static inline void parse_string_arg (va_list * ap)
  415. {
  416. va_list ap_test_arg;
  417. va_copy(ap_test_arg, *ap);
  418. const char* test_arg = va_arg(ap_test_arg, const char*);
  419. if (!test_user_string(test_arg)) {
  420. VPRINTF("\"%s\"", ap);
  421. } else {
  422. /* invalid memory region, print arg as ptr not string */
  423. VPRINTF("\"(invalid-addr %p)\"", ap);
  424. }
  425. va_end(ap_test_arg);
  426. }
  427. static inline void parse_pointer_arg (va_list * ap)
  428. {
  429. VPRINTF("%p", ap);
  430. }
  431. static inline void parse_integer_arg (va_list * ap)
  432. {
  433. VPRINTF("%d", ap);
  434. }
  435. static inline void parse_syscall_args (va_list * ap)
  436. {
  437. const char * arg_type = va_arg(*ap, const char *);
  438. if (strcmp_static(arg_type, "const char *"))
  439. parse_string_arg(ap);
  440. else if (is_pointer(arg_type))
  441. parse_pointer_arg(ap);
  442. else
  443. parse_integer_arg(ap);
  444. }
  445. static inline void skip_syscall_args (va_list * ap)
  446. {
  447. const char * arg_type = va_arg(*ap, const char *);
  448. if (strcmp_static(arg_type, "const char *"))
  449. va_arg(*ap, const char *);
  450. else if (is_pointer(arg_type))
  451. va_arg(*ap, void *);
  452. else
  453. va_arg(*ap, int);
  454. }
  455. void sysparser_printf (const char * fmt, ...)
  456. {
  457. va_list ap;
  458. va_start(ap, fmt);
  459. VPRINTF(fmt, &ap);
  460. va_end(ap);
  461. }
  462. void parse_syscall_before (int sysno, const char * name, int nr, ...)
  463. {
  464. if (!debug_handle)
  465. return;
  466. struct parser_table * parser = &syscall_parser_table[sysno];
  467. if (!parser->slow && !parser->stop)
  468. return;
  469. va_list ap;
  470. va_start(ap, nr);
  471. PRINTF("---- shim_%s(", name);
  472. for (int i = 0 ; i < nr ; i++) {
  473. if (parser->stop && parser->stop == i)
  474. goto dotdotdot;
  475. if (i)
  476. PUTCH(',');
  477. if (parser->parser[i]) {
  478. const char * type = va_arg(ap, const char *);
  479. (*parser->parser[i])(type, &ap);
  480. } else
  481. parse_syscall_args(&ap);
  482. }
  483. PUTCH(')');
  484. dotdotdot:
  485. PRINTF(" ... %s\n", name);
  486. va_end(ap);
  487. }
  488. void parse_syscall_after (int sysno, const char * name, int nr, ...)
  489. {
  490. if (!debug_handle)
  491. return;
  492. struct parser_table * parser = &syscall_parser_table[sysno];
  493. va_list ap;
  494. va_start(ap, nr);
  495. const char * ret_type = va_arg(ap, const char *);
  496. if (parser->slow || parser->stop)
  497. PRINTF("---- return from shim_%s(...", name);
  498. else
  499. PRINTF("---- shim_%s(", name);
  500. unsigned long ret_ptr;
  501. int ret_val;
  502. if (is_pointer(ret_type))
  503. ret_ptr = (unsigned long) va_arg(ap, void *);
  504. else
  505. ret_val = va_arg(ap, int);
  506. if (!parser->slow || parser->stop)
  507. for (int i = 0 ; i < nr ; i++) {
  508. if (parser->stop && i < parser->stop) {
  509. skip_syscall_args(&ap);
  510. continue;
  511. }
  512. if (i)
  513. PUTCH(',');
  514. if (parser->parser[i]) {
  515. const char * type = va_arg(ap, const char *);
  516. (*parser->parser[i])(type, &ap);
  517. } else
  518. parse_syscall_args(&ap);
  519. }
  520. if (is_pointer(ret_type)) {
  521. if (ret_ptr < -4095L)
  522. PRINTF(") = 0x%08lx\n", ret_ptr);
  523. else
  524. PRINTF(") = %ld\n", (long) ret_ptr);
  525. } else {
  526. if (ret_val >= 0)
  527. PRINTF(") = %d\n", ret_val);
  528. else
  529. PRINTF(") = %d\n", ret_val);
  530. }
  531. va_end (ap);
  532. }
  533. static void parse_open_flags (const char * type, va_list * ap)
  534. {
  535. int flags = va_arg(*ap, int);
  536. if (flags & O_WRONLY) {
  537. PUTS("O_WRONLY");
  538. flags &= ~O_WRONLY;
  539. } else if (flags & O_RDWR) {
  540. PUTS("O_RDWR");
  541. flags &= ~O_RDWR;
  542. } else
  543. PUTS("O_RDONLY");
  544. if (flags & O_APPEND) {
  545. PUTS("|O_APPEND");
  546. flags &= ~O_APPEND;
  547. }
  548. if (flags & O_CREAT) {
  549. PUTS("|O_CREAT");
  550. flags &= ~O_CREAT;
  551. }
  552. if (flags & O_TRUNC) {
  553. PUTS("|O_TRUNC");
  554. flags &= ~O_TRUNC;
  555. }
  556. if (flags & O_EXCL) {
  557. PUTS("|O_EXCL");
  558. flags &= ~O_EXCL;
  559. }
  560. if (flags)
  561. PRINTF("|%o", flags);
  562. }
  563. static void parse_open_mode (const char * type, va_list * ap)
  564. {
  565. VPRINTF("%04o", ap);
  566. }
  567. static void parse_access_mode (const char * type, va_list * ap)
  568. {
  569. int mode = va_arg(*ap, int);
  570. PUTS("F_OK");
  571. if (mode) {
  572. if (mode & R_OK)
  573. PUTS("|R_OK");
  574. if (mode & W_OK)
  575. PUTS("|W_OK");
  576. if (mode & X_OK)
  577. PUTS("|X_OK");
  578. }
  579. }
  580. static void parse_clone_flags (const char * type, va_list * ap)
  581. {
  582. int flags = va_arg(*ap, int);
  583. #define FLG(n) { "CLONE_" #n, CLONE_##n, }
  584. const struct {
  585. const char * name; int flag;
  586. } all_flags[] = {
  587. FLG(VM), FLG(FS), FLG(FILES), FLG(SIGHAND), FLG(PTRACE), FLG(VFORK),
  588. FLG(PARENT), FLG(THREAD), FLG(NEWNS), FLG(SYSVSEM), FLG(SETTLS),
  589. FLG(PARENT_SETTID), FLG(CHILD_CLEARTID), FLG(DETACHED), FLG(UNTRACED),
  590. FLG(CHILD_SETTID), FLG(NEWUTS), FLG(NEWIPC), FLG(NEWUSER),
  591. FLG(NEWPID), FLG(NEWNET), FLG(IO),
  592. };
  593. #undef FLG
  594. bool printed = false;
  595. for (int i = 0 ; i < sizeof(all_flags) / sizeof(all_flags[0]) ; i++)
  596. if (flags & all_flags[i].flag) {
  597. if (printed)
  598. PUTCH('|');
  599. else
  600. printed = true;
  601. PUTS(all_flags[i].name);
  602. flags &= ~all_flags[i].flag;
  603. }
  604. if (flags)
  605. PRINTF("|0x%x", flags);
  606. }
  607. static void parse_mmap_prot (const char * type, va_list * ap)
  608. {
  609. int prot = va_arg(*ap, int);
  610. int nflags = 0;
  611. if (prot == PROT_NONE) {
  612. PUTS("PROT_NONE");
  613. return;
  614. }
  615. if (prot & PROT_READ) {
  616. if (nflags++)
  617. PUTS("|");
  618. PUTS("PROT_READ");
  619. }
  620. if (prot & PROT_WRITE) {
  621. if (nflags++)
  622. PUTS("|");
  623. PUTS("PROT_WRITE");
  624. }
  625. if (prot & PROT_EXEC) {
  626. if (nflags++)
  627. PUTS("|");
  628. PUTS("PROT_EXEC");
  629. }
  630. }
  631. static void parse_mmap_flags (const char * type, va_list * ap)
  632. {
  633. int flags = va_arg(*ap, int);
  634. if (flags & MAP_SHARED) {
  635. PUTS("MAP_SHARED");
  636. flags &= ~MAP_SHARED;
  637. }
  638. if (flags & MAP_PRIVATE) {
  639. PUTS("MAP_PRIVATE");
  640. flags &= ~MAP_PRIVATE;
  641. }
  642. if (flags & MAP_ANONYMOUS) {
  643. PUTS("|MAP_ANON");
  644. flags &= ~MAP_ANONYMOUS;
  645. }
  646. if (flags & MAP_FILE) {
  647. PUTS("|MAP_FILE");
  648. flags &= ~MAP_FILE;
  649. }
  650. if (flags & MAP_FIXED) {
  651. PUTS("|MAP_FIXED");
  652. flags &= ~MAP_FIXED;
  653. }
  654. #ifdef CONFIG_MMAP_ALLOW_UNINITIALIZED
  655. if (flags & MAP_UNINITIALIZED) {
  656. PUTS("|MAP_UNINITIALIZED");
  657. flags &= ~MAP_UNINITIALIZED;
  658. }
  659. #endif
  660. if (flags)
  661. PRINTF("|0x%x", flags);
  662. }
  663. static void parse_exec_args (const char * type, va_list * ap)
  664. {
  665. const char ** args = va_arg(*ap, const char **);
  666. PUTS("[");
  667. for (; ; args++) {
  668. if (test_user_memory(args, sizeof(*args), false)) {
  669. PRINTF("(invalid-argv %p)", args);
  670. break;
  671. }
  672. if (*args == NULL)
  673. break;
  674. if (test_user_string(*args)) {
  675. PRINTF("(invalid-addr %p),", *args);
  676. continue;
  677. }
  678. PUTS(*args);
  679. PUTS(",");
  680. }
  681. PUTS("]");
  682. }
  683. static void parse_exec_envp (const char * type, va_list * ap)
  684. {
  685. const char ** envp = va_arg(*ap, const char **);
  686. if (!envp) {
  687. PUTS("NULL");
  688. return;
  689. }
  690. PUTS("[");
  691. int cnt = 0;
  692. for (; cnt < 2; cnt++, envp++) {
  693. if (test_user_memory(envp, sizeof(*envp), false)) {
  694. PRINTF("(invalid-envp %p)", envp);
  695. break;
  696. }
  697. if (*envp == NULL)
  698. break;
  699. if (test_user_string(*envp)) {
  700. PRINTF("(invalid-addr %p),", *envp);
  701. continue;
  702. }
  703. PUTS(*envp);
  704. PUTS(",");
  705. }
  706. if (cnt > 2)
  707. PRINTF("(%d more)", cnt);
  708. PUTS("]");
  709. }
  710. static void parse_pipe_fds (const char * type, va_list * ap)
  711. {
  712. int * fds = va_arg(*ap, int *);
  713. PRINTF("[%d, %d]", fds[0], fds[1]);
  714. }
  715. #define S(sig) #sig
  716. const char *const siglist[NUM_KNOWN_SIGS + 1] =
  717. {
  718. S(SIGUNUSED),
  719. S(SIGHUP),
  720. S(SIGINT),
  721. S(SIGQUIT),
  722. S(SIGILL),
  723. S(SIGTRAP),
  724. S(SIGABRT),
  725. S(SIGBUS),
  726. S(SIGFPE),
  727. S(SIGKILL),
  728. S(SIGUSR1),
  729. S(SIGSEGV),
  730. S(SIGUSR2),
  731. S(SIGPIPE),
  732. S(SIGALRM),
  733. S(SIGTERM),
  734. S(SIGSTKFLT),
  735. S(SIGCHLD),
  736. S(SIGCONT),
  737. S(SIGSTOP),
  738. S(SIGTSTP),
  739. S(SIGTTIN),
  740. S(SIGTTOU),
  741. S(SIGURG),
  742. S(SIGXCPU),
  743. S(SIGXFSZ),
  744. S(SIGVTALRM),
  745. S(SIGPROF),
  746. S(SIGWINCH),
  747. S(SIGIO),
  748. S(SIGPWR),
  749. S(SIGSYS),
  750. S(SIGRTMIN),
  751. };
  752. static void parse_signum (const char * type, va_list * ap)
  753. {
  754. int signum = va_arg(*ap, int);
  755. if (signum >= 0 && signum <= NUM_KNOWN_SIGS)
  756. PUTS(signal_name(signum));
  757. else
  758. PRINTF("[SIG %d]", signum);
  759. }
  760. static void parse_sigmask (const char * type, va_list * ap)
  761. {
  762. __sigset_t * sigset = va_arg(*ap, __sigset_t *);
  763. if (!sigset) {
  764. PUTS("NULL");
  765. return;
  766. }
  767. if (test_user_memory(sigset, sizeof(*sigset), false)) {
  768. PRINTF("(invalid-addr %p)", sigset);
  769. return;
  770. }
  771. PUTS("[");
  772. for (int signum = 1 ; signum <= sizeof(sigset) * 8 ; signum++)
  773. if (__sigismember(sigset, signum)) {
  774. PUTS(signal_name(signum));
  775. PUTS(",");
  776. }
  777. PUTS("]");
  778. }
  779. static void parse_sigprocmask_how (const char * type, va_list * ap)
  780. {
  781. int how = va_arg(*ap, int);
  782. switch (how) {
  783. case SIG_BLOCK:
  784. PUTS("BLOCK");
  785. break;
  786. case SIG_UNBLOCK:
  787. PUTS("UNBLOCK");
  788. break;
  789. case SIG_SETMASK:
  790. PUTS("SETMASK");
  791. break;
  792. default:
  793. PUTS("<unknown>");
  794. break;
  795. }
  796. }
  797. static void parse_timespec (const char * type, va_list * ap)
  798. {
  799. const struct timespec * tv = va_arg(*ap, const struct timespec *);
  800. if (!tv) {
  801. PUTS("NULL");
  802. return;
  803. }
  804. if (test_user_memory((void*)tv, sizeof(*tv), false)) {
  805. PRINTF("(invalid-addr %p)", tv);
  806. return;
  807. }
  808. PRINTF("[%ld,%ld]", tv->tv_sec, tv->tv_nsec);
  809. }
  810. static void parse_sockaddr (const char * type, va_list *ap)
  811. {
  812. const struct sockaddr *addr = va_arg(*ap, const struct sockaddr *);
  813. if (!addr) {
  814. PUTS("NULL");
  815. return;
  816. }
  817. if (test_user_memory((void*)addr, sizeof(*addr), false)) {
  818. PRINTF("(invalid-addr %p)", addr);
  819. return;
  820. }
  821. switch (addr->sa_family) {
  822. case AF_INET: {
  823. struct sockaddr_in * a = (void *) addr;
  824. unsigned char * ip = (void *) &a->sin_addr.s_addr;
  825. PRINTF("{family=INET,ip=%u.%u.%u.%u,port=htons(%u)}",
  826. ip[0], ip[1], ip[2], ip[3], __ntohs(a->sin_port));
  827. break;
  828. }
  829. case AF_INET6: {
  830. struct sockaddr_in6 * a = (void *) addr;
  831. unsigned short * ip = (void *) &a->sin6_addr.s6_addr;
  832. PRINTF("{family=INET,ip=[%x:%x:%x:%x:%x:%x:%x:%x],"
  833. "port=htons(%u)}",
  834. __ntohs(ip[0]), __ntohs(ip[1]),
  835. __ntohs(ip[2]), __ntohs(ip[3]),
  836. __ntohs(ip[4]), __ntohs(ip[5]),
  837. __ntohs(ip[6]), __ntohs(ip[7]),
  838. __ntohs(a->sin6_port));
  839. break;
  840. }
  841. case AF_UNIX: {
  842. struct sockaddr_un * a = (void *) addr;
  843. PRINTF("{family=UNIX,path=%s}", a->sun_path);
  844. break;
  845. }
  846. default:
  847. PUTS("UNKNOWN");
  848. break;
  849. }
  850. }
  851. static void parse_domain (const char * type, va_list * ap)
  852. {
  853. int domain = va_arg(*ap, int);
  854. #define PF_UNSPEC 0 /* Unspecified. */
  855. #define PF_INET 2 /* IP protocol family. */
  856. #define PF_AX25 3 /* Amateur Radio AX.25. */
  857. #define PF_IPX 4 /* Novell Internet Protocol. */
  858. #define PF_APPLETALK 5 /* Appletalk DDP. */
  859. #define PF_ATMPVC 8 /* ATM PVCs. */
  860. #define PF_X25 9 /* Reserved for X.25 project. */
  861. #define PF_INET6 10 /* IP version 6. */
  862. #define PF_NETLINK 16
  863. #define PF_PACKET 17 /* Packet family. */
  864. switch (domain) {
  865. case PF_UNSPEC:
  866. PUTS("UNSPEC");
  867. break;
  868. case PF_UNIX:
  869. PUTS("UNIX");
  870. break;
  871. case PF_INET:
  872. PUTS("INET");
  873. break;
  874. case PF_INET6:
  875. PUTS("INET6");
  876. break;
  877. case PF_IPX:
  878. PUTS("IPX");
  879. break;
  880. case PF_NETLINK:
  881. PUTS("NETLINK");
  882. break;
  883. case PF_X25:
  884. PUTS("X25");
  885. break;
  886. case PF_AX25:
  887. PUTS("AX25");
  888. break;
  889. case PF_ATMPVC:
  890. PUTS("ATMPVC");
  891. break;
  892. case PF_APPLETALK:
  893. PUTS("APPLETALK");
  894. break;
  895. case PF_PACKET:
  896. PUTS("PACKET");
  897. break;
  898. default:
  899. PUTS("UNKNOWN");
  900. break;
  901. }
  902. }
  903. static void parse_socktype (const char * type, va_list * ap)
  904. {
  905. int socktype = va_arg(*ap, int);
  906. if (socktype & SOCK_NONBLOCK) {
  907. socktype &= ~SOCK_NONBLOCK;
  908. PUTS("SOCK_NONBLOCK|");
  909. }
  910. if (socktype & SOCK_CLOEXEC) {
  911. socktype &= ~SOCK_CLOEXEC;
  912. PUTS("SOCK_CLOEXEC|");
  913. }
  914. #define SOCK_RAW 3 /* Raw protocol interface. */
  915. #define SOCK_RDM 4 /* Reliably-delivered messages. */
  916. #define SOCK_SEQPACKET 5 /* Sequenced, reliable, connection-based, */
  917. #define SOCK_DCCP 6 /* Datagram Congestion Control Protocol. */
  918. #define SOCK_PACKET 10 /* Linux specific way of getting packets */
  919. switch (socktype) {
  920. case SOCK_STREAM:
  921. PUTS("STREAM");
  922. break;
  923. case SOCK_DGRAM:
  924. PUTS("DGRAM");
  925. break;
  926. case SOCK_SEQPACKET:
  927. PUTS("SEQPACKET");
  928. break;
  929. case SOCK_RAW:
  930. PUTS("RAW");
  931. break;
  932. case SOCK_RDM:
  933. PUTS("RDM");
  934. break;
  935. case SOCK_PACKET:
  936. PUTS("PACKET");
  937. break;
  938. default:
  939. PUTS("UNKNOWN");
  940. break;
  941. }
  942. }
  943. static void parse_futexop (const char * type, va_list * ap)
  944. {
  945. int op = va_arg(*ap, int);
  946. #ifdef FUTEX_PRIVATE_FLAG
  947. if (op & FUTEX_PRIVATE_FLAG) {
  948. PUTS("FUTEX_PRIVATE|");
  949. op &= ~FUTEX_PRIVATE_FLAG;
  950. }
  951. #endif
  952. #ifdef FUTEX_CLOCK_REALTIME
  953. if (op & FUTEX_CLOCK_REALTIME) {
  954. PUTS("FUTEX_CLOCK_REALTIME|");
  955. op &= ~FUTEX_CLOCK_REALTIME;
  956. }
  957. #endif
  958. op &= FUTEX_CMD_MASK;
  959. switch (op) {
  960. case FUTEX_WAIT:
  961. PUTS("FUTEX_WAIT");
  962. break;
  963. case FUTEX_WAIT_BITSET:
  964. PUTS("FUTEX_WAIT_BITSET");
  965. break;
  966. case FUTEX_WAKE:
  967. PUTS("FUTEX_WAKE");
  968. break;
  969. case FUTEX_WAKE_BITSET:
  970. PUTS("FUTEX_WAKE_BITSET");
  971. break;
  972. case FUTEX_FD:
  973. PUTS("FUTEX_FD");
  974. break;
  975. case FUTEX_REQUEUE:
  976. PUTS("FUTEX_REQUEUE");
  977. break;
  978. case FUTEX_CMP_REQUEUE:
  979. PUTS("FUTEX_CMP_REQUEUE");
  980. break;
  981. case FUTEX_WAKE_OP:
  982. PUTS("FUTEX_WAKE_OP");
  983. break;
  984. default:
  985. PRINTF("OP %d", op);
  986. break;
  987. }
  988. }
  989. static void parse_fcntlop (const char * type, va_list * ap)
  990. {
  991. int op = va_arg(*ap, int);
  992. switch (op) {
  993. case F_DUPFD:
  994. PUTS("F_DUPFD");
  995. break;
  996. case F_GETFD:
  997. PUTS("F_GETFD");
  998. break;
  999. case F_SETFD:
  1000. PUTS("F_SETFD");
  1001. break;
  1002. case F_GETFL:
  1003. PUTS("F_GETFL");
  1004. break;
  1005. case F_SETFL:
  1006. PUTS("F_SETFL");
  1007. break;
  1008. case F_GETLK:
  1009. PUTS("F_GETLK");
  1010. break;
  1011. case F_SETLK:
  1012. PUTS("F_SETLK");
  1013. break;
  1014. case F_SETLKW:
  1015. PUTS("F_SETLKW");
  1016. break;
  1017. case F_SETOWN:
  1018. PUTS("F_SETOWN");
  1019. break;
  1020. case F_GETOWN:
  1021. PUTS("F_GETOWN");
  1022. break;
  1023. case F_SETSIG:
  1024. PUTS("F_SETSIG");
  1025. break;
  1026. case F_GETSIG:
  1027. PUTS("F_GETSIG");
  1028. break;
  1029. case F_GETLK64:
  1030. PUTS("F_GETLK64");
  1031. break;
  1032. case F_SETLK64:
  1033. PUTS("F_SETLK64");
  1034. break;
  1035. case F_SETLKW64:
  1036. PUTS("F_SETLKW64");
  1037. break;
  1038. case F_SETOWN_EX:
  1039. PUTS("F_SETOWN_EX");
  1040. break;
  1041. case F_GETOWN_EX:
  1042. PUTS("F_GETOWN_EX");
  1043. break;
  1044. case F_GETOWNER_UIDS:
  1045. PUTS("F_GETOWNER_UIDS");
  1046. break;
  1047. default:
  1048. PRINTF("OP %d", op);
  1049. break;
  1050. }
  1051. }
  1052. static void parse_ioctlop (const char * type, va_list * ap)
  1053. {
  1054. int op = va_arg(*ap, int);
  1055. if (op >= TCGETS && op <= TIOCVHANGUP) {
  1056. const char * opnames[] = {
  1057. "TCGETS", /* 0x5401 */ "TCSETS", /* 0x5402 */
  1058. "TCSETSW", /* 0x5403 */ "TCSETSF", /* 0x5404 */
  1059. "TCGETA", /* 0x5405 */ "TCSETA", /* 0x5406 */
  1060. "TCSETAW", /* 0x5407 */ "TCSETAF", /* 0x5408 */
  1061. "TCSBRK", /* 0x5409 */ "TCXONC", /* 0x540A */
  1062. "TCFLSH", /* 0x540B */ "TIOCEXCL", /* 0x540C */
  1063. "TIOCNXCL", /* 0x540D */ "TIOCSCTTY", /* 0x540E */
  1064. "TIOCGPGRP", /* 0x540F */ "TIOCSPGRP", /* 0x5410 */
  1065. "TIOCOUTQ", /* 0x5411 */ "TIOCSTI", /* 0x5412 */
  1066. "TIOCGWINSZ", /* 0x5413 */ "TIOCSWINSZ", /* 0x5414 */
  1067. "TIOCMGET", /* 0x5415 */ "TIOCMBIS", /* 0x5416 */
  1068. "TIOCMBIC", /* 0x5417 */ "TIOCMSET", /* 0x5418 */
  1069. "TIOCGSOFTCAR", /* 0x5419 */ "TIOCSSOFTCAR", /* 0x541A */
  1070. "FIONREAD", /* 0x541B */ "TIOCLINUX", /* 0x541C */
  1071. "TIOCCONS", /* 0x541D */ "TIOCGSERIAL", /* 0x541E */
  1072. "TIOCSSERIAL", /* 0x541F */ "TIOCPKT", /* 0x5420 */
  1073. "FIONBIO", /* 0x5421 */ "TIOCNOTTY", /* 0x5422 */
  1074. "TIOCSETD", /* 0x5423 */ "TIOCGETD", /* 0x5424 */
  1075. "TCSBRKP", /* 0x5425 */ "",
  1076. "TIOCSBRK", /* 0x5427 */ "TIOCCBRK", /* 0x5428 */
  1077. "TIOCGSID", /* 0x5429 */ "TCGETS2", /* 0x542A */
  1078. "TCSETS2", /* 0x542B */ "TCSETSW2", /* 0x542C */
  1079. "TCSETSF2", /* 0x542D */ "TIOCGRS485", /* 0x542E */
  1080. "TIOCSRS485", /* 0x542F */ "TIOCGPTN", /* 0x5430 */
  1081. "TIOCSPTLCK", /* 0x5431 */ "TCGETX", /* 0x5432 */
  1082. "TCSETX", /* 0x5433 */ "TCSETXF", /* 0x5434 */
  1083. "TCSETXW", /* 0x5435 */ "TIOCSIG", /* 0x5436 */
  1084. "TIOCVHANGUP", /* 0x5437 */
  1085. };
  1086. PUTS(opnames[op - TCGETS]);
  1087. return;
  1088. }
  1089. if (op >= FIONCLEX && op <= TIOCSERSETMULTI) {
  1090. const char * opnames[] = {
  1091. "FIONCLEX", /* 0x5450 */ "FIOCLEX", /* 0x5451 */
  1092. "FIOASYNC", /* 0x5452 */ "TIOCSERCONFIG", /* 0x5453 */
  1093. "TIOCSERGWILD", /* 0x5454 */ "TIOCSERSWILD", /* 0x5455 */
  1094. "TIOCGLCKTRMIOS", /* 0x5456 */ "TIOCSLCKTRMIOS", /* 0x5457 */
  1095. "TIOCSERGSTRUCT", /* 0x5458 */ "TIOCSERGETLSR", /* 0x5459 */
  1096. "TIOCSERGETMULTI", /* 0x545A */ "TIOCSERSETMULTI", /* 0x545B */
  1097. };
  1098. PUTS(opnames[op - FIONCLEX]);
  1099. return;
  1100. }
  1101. #define TIOCMIWAIT 0x545C /* wait for a change on serial input line(s) */
  1102. #define TIOCGICOUNT 0x545D /* read serial port __inline__ interrupt counts */
  1103. PRINTF("OP 0x%04x", op);
  1104. }
  1105. static void parse_seek (const char * type, va_list * ap)
  1106. {
  1107. int seek = va_arg(*ap, int);
  1108. switch(seek) {
  1109. case SEEK_CUR:
  1110. PUTS("SEEK_CUR");
  1111. break;
  1112. case SEEK_SET:
  1113. PUTS("SEEK_SET");
  1114. break;
  1115. case SEEK_END:
  1116. PUTS("SEEK_END");
  1117. break;
  1118. default:
  1119. PRINTF("%d", seek);
  1120. break;
  1121. }
  1122. }
  1123. static void parse_at_fdcwd (const char * type, va_list * ap)
  1124. {
  1125. int fd = va_arg(*ap, int);
  1126. switch(fd) {
  1127. case AT_FDCWD:
  1128. PUTS("AT_FDCWD");
  1129. break;
  1130. default:
  1131. PRINTF("%d", fd);
  1132. break;
  1133. }
  1134. }
  1135. static void parse_wait_option (const char * type, va_list * ap)
  1136. {
  1137. int option = va_arg(*ap, int);
  1138. if (option & WNOHANG)
  1139. PUTS("WNOHANG");
  1140. }