posix.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2016 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Kristian Koehntopp <kris@koehntopp.de> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include <unistd.h>
  24. #include "ext/standard/info.h"
  25. #include "ext/standard/php_string.h"
  26. #include "php_posix.h"
  27. #if HAVE_POSIX
  28. #ifdef HAVE_SYS_TIME_H
  29. #include <sys/time.h>
  30. #endif
  31. #include <sys/resource.h>
  32. #if defined(_GNU_SOURCE) && !defined(__USE_GNU)
  33. # define __USE_GNU
  34. #endif
  35. #include <sys/utsname.h>
  36. #include <sys/types.h>
  37. #include <sys/stat.h>
  38. #include <signal.h>
  39. #include <sys/times.h>
  40. #include <errno.h>
  41. #include <grp.h>
  42. #include <pwd.h>
  43. #if HAVE_SYS_MKDEV_H
  44. # include <sys/mkdev.h>
  45. #endif
  46. ZEND_DECLARE_MODULE_GLOBALS(posix)
  47. static PHP_MINFO_FUNCTION(posix);
  48. /* {{{ arginfo */
  49. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_kill, 0, 0, 2)
  50. ZEND_ARG_INFO(0, pid)
  51. ZEND_ARG_INFO(0, sig)
  52. ZEND_END_ARG_INFO()
  53. ZEND_BEGIN_ARG_INFO(arginfo_posix_getpid, 0)
  54. ZEND_END_ARG_INFO()
  55. ZEND_BEGIN_ARG_INFO(arginfo_posix_getppid, 0)
  56. ZEND_END_ARG_INFO()
  57. ZEND_BEGIN_ARG_INFO(arginfo_posix_getuid, 0)
  58. ZEND_END_ARG_INFO()
  59. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_setuid, 0, 0, 1)
  60. ZEND_ARG_INFO(0, uid)
  61. ZEND_END_ARG_INFO()
  62. ZEND_BEGIN_ARG_INFO(arginfo_posix_geteuid, 0)
  63. ZEND_END_ARG_INFO()
  64. #ifdef HAVE_SETEUID
  65. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_seteuid, 0, 0, 1)
  66. ZEND_ARG_INFO(0, uid)
  67. ZEND_END_ARG_INFO()
  68. #endif
  69. ZEND_BEGIN_ARG_INFO(arginfo_posix_getgid, 0)
  70. ZEND_END_ARG_INFO()
  71. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_setgid, 0, 0, 1)
  72. ZEND_ARG_INFO(0, gid)
  73. ZEND_END_ARG_INFO()
  74. ZEND_BEGIN_ARG_INFO(arginfo_posix_getegid, 0)
  75. ZEND_END_ARG_INFO()
  76. #ifdef HAVE_SETEGID
  77. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_setegid, 0, 0, 1)
  78. ZEND_ARG_INFO(0, gid)
  79. ZEND_END_ARG_INFO()
  80. #endif
  81. #ifdef HAVE_GETGROUPS
  82. ZEND_BEGIN_ARG_INFO(arginfo_posix_getgroups, 0)
  83. ZEND_END_ARG_INFO()
  84. #endif
  85. #ifdef HAVE_GETLOGIN
  86. ZEND_BEGIN_ARG_INFO(arginfo_posix_getlogin, 0)
  87. ZEND_END_ARG_INFO()
  88. #endif
  89. ZEND_BEGIN_ARG_INFO(arginfo_posix_getpgrp, 0)
  90. ZEND_END_ARG_INFO()
  91. #ifdef HAVE_SETSID
  92. ZEND_BEGIN_ARG_INFO(arginfo_posix_setsid, 0)
  93. ZEND_END_ARG_INFO()
  94. #endif
  95. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_setpgid, 0, 0, 2)
  96. ZEND_ARG_INFO(0, pid)
  97. ZEND_ARG_INFO(0, pgid)
  98. ZEND_END_ARG_INFO()
  99. #ifdef HAVE_GETPGID
  100. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getpgid, 0, 0, 1)
  101. ZEND_ARG_INFO(0, pid)
  102. ZEND_END_ARG_INFO()
  103. #endif
  104. #ifdef HAVE_GETSID
  105. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getsid, 0, 0, 1)
  106. ZEND_ARG_INFO(0, pid)
  107. ZEND_END_ARG_INFO()
  108. #endif
  109. ZEND_BEGIN_ARG_INFO(arginfo_posix_uname, 0)
  110. ZEND_END_ARG_INFO()
  111. ZEND_BEGIN_ARG_INFO(arginfo_posix_times, 0)
  112. ZEND_END_ARG_INFO()
  113. #ifdef HAVE_CTERMID
  114. ZEND_BEGIN_ARG_INFO(arginfo_posix_ctermid, 0)
  115. ZEND_END_ARG_INFO()
  116. #endif
  117. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_ttyname, 0, 0, 1)
  118. ZEND_ARG_INFO(0, fd)
  119. ZEND_END_ARG_INFO()
  120. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_isatty, 0, 0, 1)
  121. ZEND_ARG_INFO(0, fd)
  122. ZEND_END_ARG_INFO()
  123. ZEND_BEGIN_ARG_INFO(arginfo_posix_getcwd, 0)
  124. ZEND_END_ARG_INFO()
  125. #ifdef HAVE_MKFIFO
  126. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_mkfifo, 0, 0, 2)
  127. ZEND_ARG_INFO(0, pathname)
  128. ZEND_ARG_INFO(0, mode)
  129. ZEND_END_ARG_INFO()
  130. #endif
  131. #ifdef HAVE_MKNOD
  132. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_mknod, 0, 0, 2)
  133. ZEND_ARG_INFO(0, pathname)
  134. ZEND_ARG_INFO(0, mode)
  135. ZEND_ARG_INFO(0, major)
  136. ZEND_ARG_INFO(0, minor)
  137. ZEND_END_ARG_INFO()
  138. #endif
  139. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_access, 0, 0, 1)
  140. ZEND_ARG_INFO(0, file)
  141. ZEND_ARG_INFO(0, mode)
  142. ZEND_END_ARG_INFO()
  143. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getgrnam, 0, 0, 1)
  144. ZEND_ARG_INFO(0, name)
  145. ZEND_END_ARG_INFO()
  146. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getgrgid, 0, 0, 1)
  147. ZEND_ARG_INFO(0, gid)
  148. ZEND_END_ARG_INFO()
  149. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getpwnam, 0, 0, 1)
  150. ZEND_ARG_INFO(0, username)
  151. ZEND_END_ARG_INFO()
  152. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_getpwuid, 0, 0, 1)
  153. ZEND_ARG_INFO(0, uid)
  154. ZEND_END_ARG_INFO()
  155. #ifdef HAVE_GETRLIMIT
  156. ZEND_BEGIN_ARG_INFO(arginfo_posix_getrlimit, 0)
  157. ZEND_END_ARG_INFO()
  158. #endif
  159. ZEND_BEGIN_ARG_INFO(arginfo_posix_get_last_error, 0)
  160. ZEND_END_ARG_INFO()
  161. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_strerror, 0, 0, 1)
  162. ZEND_ARG_INFO(0, errno)
  163. ZEND_END_ARG_INFO()
  164. #ifdef HAVE_INITGROUPS
  165. ZEND_BEGIN_ARG_INFO_EX(arginfo_posix_initgroups, 0, 0, 2)
  166. ZEND_ARG_INFO(0, name)
  167. ZEND_ARG_INFO(0, base_group_id)
  168. ZEND_END_ARG_INFO()
  169. #endif
  170. /* }}} */
  171. /* {{{ posix_functions[]
  172. */
  173. const zend_function_entry posix_functions[] = {
  174. /* POSIX.1, 3.3 */
  175. PHP_FE(posix_kill, arginfo_posix_kill)
  176. /* POSIX.1, 4.1 */
  177. PHP_FE(posix_getpid, arginfo_posix_getpid)
  178. PHP_FE(posix_getppid, arginfo_posix_getppid)
  179. /* POSIX.1, 4.2 */
  180. PHP_FE(posix_getuid, arginfo_posix_getuid)
  181. PHP_FE(posix_setuid, arginfo_posix_setuid)
  182. PHP_FE(posix_geteuid, arginfo_posix_geteuid)
  183. #ifdef HAVE_SETEUID
  184. PHP_FE(posix_seteuid, arginfo_posix_seteuid)
  185. #endif
  186. PHP_FE(posix_getgid, arginfo_posix_getgid)
  187. PHP_FE(posix_setgid, arginfo_posix_setgid)
  188. PHP_FE(posix_getegid, arginfo_posix_getegid)
  189. #ifdef HAVE_SETEGID
  190. PHP_FE(posix_setegid, arginfo_posix_setegid)
  191. #endif
  192. #ifdef HAVE_GETGROUPS
  193. PHP_FE(posix_getgroups, arginfo_posix_getgroups)
  194. #endif
  195. #ifdef HAVE_GETLOGIN
  196. PHP_FE(posix_getlogin, arginfo_posix_getlogin)
  197. #endif
  198. /* POSIX.1, 4.3 */
  199. PHP_FE(posix_getpgrp, arginfo_posix_getpgrp)
  200. #ifdef HAVE_SETSID
  201. PHP_FE(posix_setsid, arginfo_posix_setsid)
  202. #endif
  203. PHP_FE(posix_setpgid, arginfo_posix_setpgid)
  204. /* Non-Posix functions which are common */
  205. #ifdef HAVE_GETPGID
  206. PHP_FE(posix_getpgid, arginfo_posix_getpgid)
  207. #endif /* HAVE_GETPGID */
  208. #ifdef HAVE_GETSID
  209. PHP_FE(posix_getsid, arginfo_posix_getsid)
  210. #endif /* HAVE_GETSID */
  211. /* POSIX.1, 4.4 */
  212. PHP_FE(posix_uname, arginfo_posix_uname)
  213. /* POSIX.1, 4.5 */
  214. PHP_FE(posix_times, arginfo_posix_times)
  215. /* POSIX.1, 4.7 */
  216. #ifdef HAVE_CTERMID
  217. PHP_FE(posix_ctermid, arginfo_posix_ctermid)
  218. #endif
  219. PHP_FE(posix_ttyname, arginfo_posix_ttyname)
  220. PHP_FE(posix_isatty, arginfo_posix_isatty)
  221. /* POSIX.1, 5.2 */
  222. PHP_FE(posix_getcwd, arginfo_posix_getcwd)
  223. /* POSIX.1, 5.4 */
  224. #ifdef HAVE_MKFIFO
  225. PHP_FE(posix_mkfifo, arginfo_posix_mkfifo)
  226. #endif
  227. #ifdef HAVE_MKNOD
  228. PHP_FE(posix_mknod, arginfo_posix_mknod)
  229. #endif
  230. /* POSIX.1, 5.6 */
  231. PHP_FE(posix_access, arginfo_posix_access)
  232. /* POSIX.1, 9.2 */
  233. PHP_FE(posix_getgrnam, arginfo_posix_getgrnam)
  234. PHP_FE(posix_getgrgid, arginfo_posix_getgrgid)
  235. PHP_FE(posix_getpwnam, arginfo_posix_getpwnam)
  236. PHP_FE(posix_getpwuid, arginfo_posix_getpwuid)
  237. #ifdef HAVE_GETRLIMIT
  238. PHP_FE(posix_getrlimit, arginfo_posix_getrlimit)
  239. #endif
  240. PHP_FE(posix_get_last_error, arginfo_posix_get_last_error)
  241. PHP_FALIAS(posix_errno, posix_get_last_error, arginfo_posix_get_last_error)
  242. PHP_FE(posix_strerror, arginfo_posix_strerror)
  243. #ifdef HAVE_INITGROUPS
  244. PHP_FE(posix_initgroups, arginfo_posix_initgroups)
  245. #endif
  246. PHP_FE_END
  247. };
  248. /* }}} */
  249. /* {{{ PHP_MINFO_FUNCTION
  250. */
  251. static PHP_MINFO_FUNCTION(posix)
  252. {
  253. php_info_print_table_start();
  254. php_info_print_table_row(2, "Revision", "$Id$");
  255. php_info_print_table_end();
  256. }
  257. /* }}} */
  258. static PHP_GINIT_FUNCTION(posix) /* {{{ */
  259. {
  260. posix_globals->last_error = 0;
  261. }
  262. /* }}} */
  263. /* {{{ PHP_MINIT_FUNCTION(posix)
  264. */
  265. static PHP_MINIT_FUNCTION(posix)
  266. {
  267. REGISTER_LONG_CONSTANT("POSIX_F_OK", F_OK, CONST_CS | CONST_PERSISTENT);
  268. REGISTER_LONG_CONSTANT("POSIX_X_OK", X_OK, CONST_CS | CONST_PERSISTENT);
  269. REGISTER_LONG_CONSTANT("POSIX_W_OK", W_OK, CONST_CS | CONST_PERSISTENT);
  270. REGISTER_LONG_CONSTANT("POSIX_R_OK", R_OK, CONST_CS | CONST_PERSISTENT);
  271. #ifdef S_IFREG
  272. REGISTER_LONG_CONSTANT("POSIX_S_IFREG", S_IFREG, CONST_CS | CONST_PERSISTENT);
  273. #endif
  274. #ifdef S_IFCHR
  275. REGISTER_LONG_CONSTANT("POSIX_S_IFCHR", S_IFCHR, CONST_CS | CONST_PERSISTENT);
  276. #endif
  277. #ifdef S_IFBLK
  278. REGISTER_LONG_CONSTANT("POSIX_S_IFBLK", S_IFBLK, CONST_CS | CONST_PERSISTENT);
  279. #endif
  280. #ifdef S_IFIFO
  281. REGISTER_LONG_CONSTANT("POSIX_S_IFIFO", S_IFIFO, CONST_CS | CONST_PERSISTENT);
  282. #endif
  283. #ifdef S_IFSOCK
  284. REGISTER_LONG_CONSTANT("POSIX_S_IFSOCK", S_IFSOCK, CONST_CS | CONST_PERSISTENT);
  285. #endif
  286. return SUCCESS;
  287. }
  288. /* }}} */
  289. /* {{{ posix_module_entry
  290. */
  291. zend_module_entry posix_module_entry = {
  292. STANDARD_MODULE_HEADER,
  293. "posix",
  294. posix_functions,
  295. PHP_MINIT(posix),
  296. NULL,
  297. NULL,
  298. NULL,
  299. PHP_MINFO(posix),
  300. NO_VERSION_YET,
  301. PHP_MODULE_GLOBALS(posix),
  302. PHP_GINIT(posix),
  303. NULL,
  304. NULL,
  305. STANDARD_MODULE_PROPERTIES_EX
  306. };
  307. /* }}} */
  308. #ifdef COMPILE_DL_POSIX
  309. ZEND_GET_MODULE(posix)
  310. #endif
  311. #define PHP_POSIX_NO_ARGS if (zend_parse_parameters_none() == FAILURE) return;
  312. #define PHP_POSIX_RETURN_LONG_FUNC(func_name) \
  313. PHP_POSIX_NO_ARGS \
  314. RETURN_LONG(func_name());
  315. #define PHP_POSIX_SINGLE_ARG_FUNC(func_name) \
  316. long val; \
  317. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &val) == FAILURE) RETURN_FALSE; \
  318. if (func_name(val) < 0) { \
  319. POSIX_G(last_error) = errno; \
  320. RETURN_FALSE; \
  321. } \
  322. RETURN_TRUE;
  323. /* {{{ proto bool posix_kill(int pid, int sig)
  324. Send a signal to a process (POSIX.1, 3.3.2) */
  325. PHP_FUNCTION(posix_kill)
  326. {
  327. long pid, sig;
  328. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ll", &pid, &sig) == FAILURE) {
  329. RETURN_FALSE;
  330. }
  331. if (kill(pid, sig) < 0) {
  332. POSIX_G(last_error) = errno;
  333. RETURN_FALSE;
  334. }
  335. RETURN_TRUE;
  336. }
  337. /* }}} */
  338. /* {{{ proto int posix_getpid(void)
  339. Get the current process id (POSIX.1, 4.1.1) */
  340. PHP_FUNCTION(posix_getpid)
  341. {
  342. PHP_POSIX_RETURN_LONG_FUNC(getpid);
  343. }
  344. /* }}} */
  345. /* {{{ proto int posix_getppid(void)
  346. Get the parent process id (POSIX.1, 4.1.1) */
  347. PHP_FUNCTION(posix_getppid)
  348. {
  349. PHP_POSIX_RETURN_LONG_FUNC(getppid);
  350. }
  351. /* }}} */
  352. /* {{{ proto int posix_getuid(void)
  353. Get the current user id (POSIX.1, 4.2.1) */
  354. PHP_FUNCTION(posix_getuid)
  355. {
  356. PHP_POSIX_RETURN_LONG_FUNC(getuid);
  357. }
  358. /* }}} */
  359. /* {{{ proto int posix_getgid(void)
  360. Get the current group id (POSIX.1, 4.2.1) */
  361. PHP_FUNCTION(posix_getgid)
  362. {
  363. PHP_POSIX_RETURN_LONG_FUNC(getgid);
  364. }
  365. /* }}} */
  366. /* {{{ proto int posix_geteuid(void)
  367. Get the current effective user id (POSIX.1, 4.2.1) */
  368. PHP_FUNCTION(posix_geteuid)
  369. {
  370. PHP_POSIX_RETURN_LONG_FUNC(geteuid);
  371. }
  372. /* }}} */
  373. /* {{{ proto int posix_getegid(void)
  374. Get the current effective group id (POSIX.1, 4.2.1) */
  375. PHP_FUNCTION(posix_getegid)
  376. {
  377. PHP_POSIX_RETURN_LONG_FUNC(getegid);
  378. }
  379. /* }}} */
  380. /* {{{ proto bool posix_setuid(long uid)
  381. Set user id (POSIX.1, 4.2.2) */
  382. PHP_FUNCTION(posix_setuid)
  383. {
  384. PHP_POSIX_SINGLE_ARG_FUNC(setuid);
  385. }
  386. /* }}} */
  387. /* {{{ proto bool posix_setgid(int uid)
  388. Set group id (POSIX.1, 4.2.2) */
  389. PHP_FUNCTION(posix_setgid)
  390. {
  391. PHP_POSIX_SINGLE_ARG_FUNC(setgid);
  392. }
  393. /* }}} */
  394. /* {{{ proto bool posix_seteuid(long uid)
  395. Set effective user id */
  396. #ifdef HAVE_SETEUID
  397. PHP_FUNCTION(posix_seteuid)
  398. {
  399. PHP_POSIX_SINGLE_ARG_FUNC(seteuid);
  400. }
  401. #endif
  402. /* }}} */
  403. /* {{{ proto bool posix_setegid(long uid)
  404. Set effective group id */
  405. #ifdef HAVE_SETEGID
  406. PHP_FUNCTION(posix_setegid)
  407. {
  408. PHP_POSIX_SINGLE_ARG_FUNC(setegid);
  409. }
  410. #endif
  411. /* }}} */
  412. /* {{{ proto array posix_getgroups(void)
  413. Get supplementary group id's (POSIX.1, 4.2.3) */
  414. #ifdef HAVE_GETGROUPS
  415. PHP_FUNCTION(posix_getgroups)
  416. {
  417. gid_t gidlist[NGROUPS_MAX];
  418. int result;
  419. int i;
  420. PHP_POSIX_NO_ARGS;
  421. if ((result = getgroups(NGROUPS_MAX, gidlist)) < 0) {
  422. POSIX_G(last_error) = errno;
  423. RETURN_FALSE;
  424. }
  425. array_init(return_value);
  426. for (i=0; i<result; i++) {
  427. add_next_index_long(return_value, gidlist[i]);
  428. }
  429. }
  430. #endif
  431. /* }}} */
  432. /* {{{ proto string posix_getlogin(void)
  433. Get user name (POSIX.1, 4.2.4) */
  434. #ifdef HAVE_GETLOGIN
  435. PHP_FUNCTION(posix_getlogin)
  436. {
  437. char *p;
  438. PHP_POSIX_NO_ARGS;
  439. if (NULL == (p = getlogin())) {
  440. POSIX_G(last_error) = errno;
  441. RETURN_FALSE;
  442. }
  443. RETURN_STRING(p, 1);
  444. }
  445. #endif
  446. /* }}} */
  447. /* {{{ proto int posix_getpgrp(void)
  448. Get current process group id (POSIX.1, 4.3.1) */
  449. PHP_FUNCTION(posix_getpgrp)
  450. {
  451. PHP_POSIX_RETURN_LONG_FUNC(getpgrp);
  452. }
  453. /* }}} */
  454. /* {{{ proto int posix_setsid(void)
  455. Create session and set process group id (POSIX.1, 4.3.2) */
  456. #ifdef HAVE_SETSID
  457. PHP_FUNCTION(posix_setsid)
  458. {
  459. PHP_POSIX_RETURN_LONG_FUNC(setsid);
  460. }
  461. #endif
  462. /* }}} */
  463. /* {{{ proto bool posix_setpgid(int pid, int pgid)
  464. Set process group id for job control (POSIX.1, 4.3.3) */
  465. PHP_FUNCTION(posix_setpgid)
  466. {
  467. long pid, pgid;
  468. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ll", &pid, &pgid) == FAILURE) {
  469. RETURN_FALSE;
  470. }
  471. if (setpgid(pid, pgid) < 0) {
  472. POSIX_G(last_error) = errno;
  473. RETURN_FALSE;
  474. }
  475. RETURN_TRUE;
  476. }
  477. /* }}} */
  478. /* {{{ proto int posix_getpgid(void)
  479. Get the process group id of the specified process (This is not a POSIX function, but a SVR4ism, so we compile conditionally) */
  480. #ifdef HAVE_GETPGID
  481. PHP_FUNCTION(posix_getpgid)
  482. {
  483. long val;
  484. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &val) == FAILURE) {
  485. RETURN_FALSE;
  486. }
  487. if ((val = getpgid(val)) < 0) {
  488. POSIX_G(last_error) = errno;
  489. RETURN_FALSE;
  490. }
  491. RETURN_LONG(val);
  492. }
  493. #endif
  494. /* }}} */
  495. /* {{{ proto int posix_getsid(void)
  496. Get process group id of session leader (This is not a POSIX function, but a SVR4ism, so be compile conditionally) */
  497. #ifdef HAVE_GETSID
  498. PHP_FUNCTION(posix_getsid)
  499. {
  500. long val;
  501. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &val) == FAILURE) {
  502. RETURN_FALSE;
  503. }
  504. if ((val = getsid(val)) < 0) {
  505. POSIX_G(last_error) = errno;
  506. RETURN_FALSE;
  507. }
  508. RETURN_LONG(val);
  509. }
  510. #endif
  511. /* }}} */
  512. /* {{{ proto array posix_uname(void)
  513. Get system name (POSIX.1, 4.4.1) */
  514. PHP_FUNCTION(posix_uname)
  515. {
  516. struct utsname u;
  517. PHP_POSIX_NO_ARGS;
  518. if (uname(&u) < 0) {
  519. POSIX_G(last_error) = errno;
  520. RETURN_FALSE;
  521. }
  522. array_init(return_value);
  523. add_assoc_string(return_value, "sysname", u.sysname, 1);
  524. add_assoc_string(return_value, "nodename", u.nodename, 1);
  525. add_assoc_string(return_value, "release", u.release, 1);
  526. add_assoc_string(return_value, "version", u.version, 1);
  527. add_assoc_string(return_value, "machine", u.machine, 1);
  528. #if defined(_GNU_SOURCE) && !defined(DARWIN) && defined(HAVE_UTSNAME_DOMAINNAME)
  529. add_assoc_string(return_value, "domainname", u.domainname, 1);
  530. #endif
  531. }
  532. /* }}} */
  533. /* POSIX.1, 4.5.1 time() - Get System Time
  534. already covered by PHP
  535. */
  536. /* {{{ proto array posix_times(void)
  537. Get process times (POSIX.1, 4.5.2) */
  538. PHP_FUNCTION(posix_times)
  539. {
  540. struct tms t;
  541. clock_t ticks;
  542. PHP_POSIX_NO_ARGS;
  543. if ((ticks = times(&t)) == -1) {
  544. POSIX_G(last_error) = errno;
  545. RETURN_FALSE;
  546. }
  547. array_init(return_value);
  548. add_assoc_long(return_value, "ticks", ticks); /* clock ticks */
  549. add_assoc_long(return_value, "utime", t.tms_utime); /* user time */
  550. add_assoc_long(return_value, "stime", t.tms_stime); /* system time */
  551. add_assoc_long(return_value, "cutime", t.tms_cutime); /* user time of children */
  552. add_assoc_long(return_value, "cstime", t.tms_cstime); /* system time of children */
  553. }
  554. /* }}} */
  555. /* POSIX.1, 4.6.1 getenv() - Environment Access
  556. already covered by PHP
  557. */
  558. /* {{{ proto string posix_ctermid(void)
  559. Generate terminal path name (POSIX.1, 4.7.1) */
  560. #ifdef HAVE_CTERMID
  561. PHP_FUNCTION(posix_ctermid)
  562. {
  563. char buffer[L_ctermid];
  564. PHP_POSIX_NO_ARGS;
  565. if (NULL == ctermid(buffer)) {
  566. /* Found no documentation how the defined behaviour is when this
  567. * function fails
  568. */
  569. POSIX_G(last_error) = errno;
  570. RETURN_FALSE;
  571. }
  572. RETURN_STRING(buffer, 1);
  573. }
  574. #endif
  575. /* }}} */
  576. /* Checks if the provides resource is a stream and if it provides a file descriptor */
  577. static int php_posix_stream_get_fd(zval *zfp, int *fd TSRMLS_DC) /* {{{ */
  578. {
  579. php_stream *stream;
  580. php_stream_from_zval_no_verify(stream, &zfp);
  581. if (stream == NULL) {
  582. php_error_docref(NULL TSRMLS_CC, E_WARNING, "expects argument 1 to be a valid stream resource");
  583. return 0;
  584. }
  585. if (php_stream_can_cast(stream, PHP_STREAM_AS_FD_FOR_SELECT) == SUCCESS) {
  586. php_stream_cast(stream, PHP_STREAM_AS_FD_FOR_SELECT, (void*)fd, 0);
  587. } else if (php_stream_can_cast(stream, PHP_STREAM_AS_FD) == SUCCESS) {
  588. php_stream_cast(stream, PHP_STREAM_AS_FD, (void*)fd, 0);
  589. } else {
  590. php_error_docref(NULL TSRMLS_CC, E_WARNING, "could not use stream of type '%s'",
  591. stream->ops->label);
  592. return 0;
  593. }
  594. return 1;
  595. }
  596. /* }}} */
  597. /* {{{ proto string posix_ttyname(int fd)
  598. Determine terminal device name (POSIX.1, 4.7.2) */
  599. PHP_FUNCTION(posix_ttyname)
  600. {
  601. zval **z_fd;
  602. char *p;
  603. int fd;
  604. #if defined(ZTS) && defined(HAVE_TTYNAME_R) && defined(_SC_TTY_NAME_MAX)
  605. long buflen;
  606. #endif
  607. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "Z", &z_fd) == FAILURE) {
  608. RETURN_FALSE;
  609. }
  610. switch (Z_TYPE_PP(z_fd)) {
  611. case IS_RESOURCE:
  612. if (!php_posix_stream_get_fd(*z_fd, &fd TSRMLS_CC)) {
  613. RETURN_FALSE;
  614. }
  615. break;
  616. default:
  617. convert_to_long_ex(z_fd);
  618. fd = Z_LVAL_PP(z_fd);
  619. }
  620. #if defined(ZTS) && defined(HAVE_TTYNAME_R) && defined(_SC_TTY_NAME_MAX)
  621. buflen = sysconf(_SC_TTY_NAME_MAX);
  622. if (buflen < 1) {
  623. RETURN_FALSE;
  624. }
  625. p = emalloc(buflen);
  626. if (ttyname_r(fd, p, buflen)) {
  627. POSIX_G(last_error) = errno;
  628. efree(p);
  629. RETURN_FALSE;
  630. }
  631. RETURN_STRING(p, 0);
  632. #else
  633. if (NULL == (p = ttyname(fd))) {
  634. POSIX_G(last_error) = errno;
  635. RETURN_FALSE;
  636. }
  637. #endif
  638. RETURN_STRING(p, 1);
  639. }
  640. /* }}} */
  641. /* {{{ proto bool posix_isatty(int fd)
  642. Determine if filedesc is a tty (POSIX.1, 4.7.1) */
  643. PHP_FUNCTION(posix_isatty)
  644. {
  645. zval **z_fd;
  646. int fd;
  647. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "Z", &z_fd) == FAILURE) {
  648. RETURN_FALSE;
  649. }
  650. switch (Z_TYPE_PP(z_fd)) {
  651. case IS_RESOURCE:
  652. if (!php_posix_stream_get_fd(*z_fd, &fd TSRMLS_CC)) {
  653. RETURN_FALSE;
  654. }
  655. break;
  656. default:
  657. convert_to_long_ex(z_fd);
  658. fd = Z_LVAL_PP(z_fd);
  659. }
  660. if (isatty(fd)) {
  661. RETURN_TRUE;
  662. } else {
  663. RETURN_FALSE;
  664. }
  665. }
  666. /* }}} */
  667. /*
  668. POSIX.1, 4.8.1 sysconf() - TODO
  669. POSIX.1, 5.7.1 pathconf(), fpathconf() - TODO
  670. POSIX.1, 5.1.2 opendir(), readdir(), rewinddir(), closedir()
  671. POSIX.1, 5.2.1 chdir()
  672. already supported by PHP
  673. */
  674. /* {{{ proto string posix_getcwd(void)
  675. Get working directory pathname (POSIX.1, 5.2.2) */
  676. PHP_FUNCTION(posix_getcwd)
  677. {
  678. char buffer[MAXPATHLEN];
  679. char *p;
  680. PHP_POSIX_NO_ARGS;
  681. p = VCWD_GETCWD(buffer, MAXPATHLEN);
  682. if (!p) {
  683. POSIX_G(last_error) = errno;
  684. RETURN_FALSE;
  685. }
  686. RETURN_STRING(buffer, 1);
  687. }
  688. /* }}} */
  689. /*
  690. POSIX.1, 5.3.x open(), creat(), umask()
  691. POSIX.1, 5.4.1 link()
  692. already supported by PHP.
  693. */
  694. /* {{{ proto bool posix_mkfifo(string pathname, int mode)
  695. Make a FIFO special file (POSIX.1, 5.4.2) */
  696. #ifdef HAVE_MKFIFO
  697. PHP_FUNCTION(posix_mkfifo)
  698. {
  699. char *path;
  700. int path_len;
  701. long mode;
  702. int result;
  703. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "pl", &path, &path_len, &mode) == FAILURE) {
  704. RETURN_FALSE;
  705. }
  706. if (php_check_open_basedir_ex(path, 0 TSRMLS_CC)) {
  707. RETURN_FALSE;
  708. }
  709. result = mkfifo(path, mode);
  710. if (result < 0) {
  711. POSIX_G(last_error) = errno;
  712. RETURN_FALSE;
  713. }
  714. RETURN_TRUE;
  715. }
  716. #endif
  717. /* }}} */
  718. /* {{{ proto bool posix_mknod(string pathname, int mode [, int major [, int minor]])
  719. Make a special or ordinary file (POSIX.1) */
  720. #ifdef HAVE_MKNOD
  721. PHP_FUNCTION(posix_mknod)
  722. {
  723. char *path;
  724. int path_len;
  725. long mode;
  726. long major = 0, minor = 0;
  727. int result;
  728. dev_t php_dev;
  729. php_dev = 0;
  730. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "pl|ll", &path, &path_len,
  731. &mode, &major, &minor) == FAILURE) {
  732. RETURN_FALSE;
  733. }
  734. if (php_check_open_basedir_ex(path, 0 TSRMLS_CC)) {
  735. RETURN_FALSE;
  736. }
  737. if ((mode & S_IFCHR) || (mode & S_IFBLK)) {
  738. if (ZEND_NUM_ARGS() == 2) {
  739. php_error_docref(NULL TSRMLS_CC, E_WARNING, "For S_IFCHR and S_IFBLK you need to pass a major device kernel identifier");
  740. RETURN_FALSE;
  741. }
  742. if (major == 0) {
  743. php_error_docref(NULL TSRMLS_CC, E_WARNING,
  744. "Expects argument 3 to be non-zero for POSIX_S_IFCHR and POSIX_S_IFBLK");
  745. RETURN_FALSE;
  746. } else {
  747. #if defined(HAVE_MAKEDEV) || defined(makedev)
  748. php_dev = makedev(major, minor);
  749. #else
  750. php_error_docref(NULL TSRMLS_CC, E_WARNING, "Cannot create a block or character device, creating a normal file instead");
  751. #endif
  752. }
  753. }
  754. result = mknod(path, mode, php_dev);
  755. if (result < 0) {
  756. POSIX_G(last_error) = errno;
  757. RETURN_FALSE;
  758. }
  759. RETURN_TRUE;
  760. }
  761. #endif
  762. /* }}} */
  763. /* Takes a pointer to posix group and a pointer to an already initialized ZVAL
  764. * array container and fills the array with the posix group member data. */
  765. int php_posix_group_to_array(struct group *g, zval *array_group) /* {{{ */
  766. {
  767. zval *array_members;
  768. int count;
  769. if (NULL == g)
  770. return 0;
  771. if (array_group == NULL || Z_TYPE_P(array_group) != IS_ARRAY)
  772. return 0;
  773. MAKE_STD_ZVAL(array_members);
  774. array_init(array_members);
  775. add_assoc_string(array_group, "name", g->gr_name, 1);
  776. add_assoc_string(array_group, "passwd", g->gr_passwd, 1);
  777. for (count=0; g->gr_mem[count] != NULL; count++) {
  778. add_next_index_string(array_members, g->gr_mem[count], 1);
  779. }
  780. zend_hash_update(Z_ARRVAL_P(array_group), "members", sizeof("members"), (void*)&array_members, sizeof(zval*), NULL);
  781. add_assoc_long(array_group, "gid", g->gr_gid);
  782. return 1;
  783. }
  784. /* }}} */
  785. /*
  786. POSIX.1, 5.5.1 unlink()
  787. POSIX.1, 5.5.2 rmdir()
  788. POSIX.1, 5.5.3 rename()
  789. POSIX.1, 5.6.x stat(), chmod(), utime() already supported by PHP.
  790. */
  791. /* {{{ proto bool posix_access(string file [, int mode])
  792. Determine accessibility of a file (POSIX.1 5.6.3) */
  793. PHP_FUNCTION(posix_access)
  794. {
  795. long mode = 0;
  796. int filename_len, ret;
  797. char *filename, *path;
  798. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "p|l", &filename, &filename_len, &mode) == FAILURE) {
  799. RETURN_FALSE;
  800. }
  801. path = expand_filepath(filename, NULL TSRMLS_CC);
  802. if (!path) {
  803. POSIX_G(last_error) = EIO;
  804. RETURN_FALSE;
  805. }
  806. if (php_check_open_basedir_ex(path, 0 TSRMLS_CC)) {
  807. efree(path);
  808. POSIX_G(last_error) = EPERM;
  809. RETURN_FALSE;
  810. }
  811. ret = access(path, mode);
  812. efree(path);
  813. if (ret) {
  814. POSIX_G(last_error) = errno;
  815. RETURN_FALSE;
  816. }
  817. RETURN_TRUE;
  818. }
  819. /* }}} */
  820. /*
  821. POSIX.1, 6.x most I/O functions already supported by PHP.
  822. POSIX.1, 7.x tty functions, TODO
  823. POSIX.1, 8.x interactions with other C language functions
  824. POSIX.1, 9.x system database access
  825. */
  826. /* {{{ proto array posix_getgrnam(string groupname)
  827. Group database access (POSIX.1, 9.2.1) */
  828. PHP_FUNCTION(posix_getgrnam)
  829. {
  830. char *name;
  831. struct group *g;
  832. int name_len;
  833. #if defined(ZTS) && defined(HAVE_GETGRNAM_R) && defined(_SC_GETGR_R_SIZE_MAX)
  834. struct group gbuf;
  835. long buflen;
  836. char *buf;
  837. #endif
  838. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &name, &name_len) == FAILURE) {
  839. RETURN_FALSE;
  840. }
  841. #if defined(ZTS) && defined(HAVE_GETGRNAM_R) && defined(_SC_GETGR_R_SIZE_MAX)
  842. buflen = sysconf(_SC_GETGR_R_SIZE_MAX);
  843. if (buflen < 1) {
  844. RETURN_FALSE;
  845. }
  846. buf = emalloc(buflen);
  847. g = &gbuf;
  848. if (getgrnam_r(name, g, buf, buflen, &g) || g == NULL) {
  849. POSIX_G(last_error) = errno;
  850. efree(buf);
  851. RETURN_FALSE;
  852. }
  853. #else
  854. if (NULL == (g = getgrnam(name))) {
  855. POSIX_G(last_error) = errno;
  856. RETURN_FALSE;
  857. }
  858. #endif
  859. array_init(return_value);
  860. if (!php_posix_group_to_array(g, return_value)) {
  861. zval_dtor(return_value);
  862. php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to convert posix group to array");
  863. RETVAL_FALSE;
  864. }
  865. #if defined(ZTS) && defined(HAVE_GETGRNAM_R) && defined(_SC_GETGR_R_SIZE_MAX)
  866. efree(buf);
  867. #endif
  868. }
  869. /* }}} */
  870. /* {{{ proto array posix_getgrgid(long gid)
  871. Group database access (POSIX.1, 9.2.1) */
  872. PHP_FUNCTION(posix_getgrgid)
  873. {
  874. long gid;
  875. #if defined(ZTS) && defined(HAVE_GETGRGID_R) && defined(_SC_GETGR_R_SIZE_MAX)
  876. int ret;
  877. struct group _g;
  878. struct group *retgrptr = NULL;
  879. long grbuflen;
  880. char *grbuf;
  881. #endif
  882. struct group *g;
  883. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &gid) == FAILURE) {
  884. RETURN_FALSE;
  885. }
  886. #if defined(ZTS) && defined(HAVE_GETGRGID_R) && defined(_SC_GETGR_R_SIZE_MAX)
  887. grbuflen = sysconf(_SC_GETGR_R_SIZE_MAX);
  888. if (grbuflen < 1) {
  889. RETURN_FALSE;
  890. }
  891. grbuf = emalloc(grbuflen);
  892. ret = getgrgid_r(gid, &_g, grbuf, grbuflen, &retgrptr);
  893. if (ret || retgrptr == NULL) {
  894. POSIX_G(last_error) = ret;
  895. efree(grbuf);
  896. RETURN_FALSE;
  897. }
  898. g = &_g;
  899. #else
  900. if (NULL == (g = getgrgid(gid))) {
  901. POSIX_G(last_error) = errno;
  902. RETURN_FALSE;
  903. }
  904. #endif
  905. array_init(return_value);
  906. if (!php_posix_group_to_array(g, return_value)) {
  907. zval_dtor(return_value);
  908. php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to convert posix group struct to array");
  909. RETVAL_FALSE;
  910. }
  911. #if defined(ZTS) && defined(HAVE_GETGRGID_R) && defined(_SC_GETGR_R_SIZE_MAX)
  912. efree(grbuf);
  913. #endif
  914. }
  915. /* }}} */
  916. int php_posix_passwd_to_array(struct passwd *pw, zval *return_value) /* {{{ */
  917. {
  918. if (NULL == pw)
  919. return 0;
  920. if (NULL == return_value || Z_TYPE_P(return_value) != IS_ARRAY)
  921. return 0;
  922. add_assoc_string(return_value, "name", pw->pw_name, 1);
  923. add_assoc_string(return_value, "passwd", pw->pw_passwd, 1);
  924. add_assoc_long (return_value, "uid", pw->pw_uid);
  925. add_assoc_long (return_value, "gid", pw->pw_gid);
  926. add_assoc_string(return_value, "gecos", pw->pw_gecos, 1);
  927. add_assoc_string(return_value, "dir", pw->pw_dir, 1);
  928. add_assoc_string(return_value, "shell", pw->pw_shell, 1);
  929. return 1;
  930. }
  931. /* }}} */
  932. /* {{{ proto array posix_getpwnam(string groupname)
  933. User database access (POSIX.1, 9.2.2) */
  934. PHP_FUNCTION(posix_getpwnam)
  935. {
  936. struct passwd *pw;
  937. char *name;
  938. int name_len;
  939. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWNAM_R)
  940. struct passwd pwbuf;
  941. long buflen;
  942. char *buf;
  943. #endif
  944. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &name, &name_len) == FAILURE) {
  945. RETURN_FALSE;
  946. }
  947. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWNAM_R)
  948. buflen = sysconf(_SC_GETPW_R_SIZE_MAX);
  949. if (buflen < 1) {
  950. RETURN_FALSE;
  951. }
  952. buf = emalloc(buflen);
  953. pw = &pwbuf;
  954. if (getpwnam_r(name, pw, buf, buflen, &pw) || pw == NULL) {
  955. efree(buf);
  956. POSIX_G(last_error) = errno;
  957. RETURN_FALSE;
  958. }
  959. #else
  960. if (NULL == (pw = getpwnam(name))) {
  961. POSIX_G(last_error) = errno;
  962. RETURN_FALSE;
  963. }
  964. #endif
  965. array_init(return_value);
  966. if (!php_posix_passwd_to_array(pw, return_value)) {
  967. zval_dtor(return_value);
  968. php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to convert posix passwd struct to array");
  969. RETVAL_FALSE;
  970. }
  971. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWNAM_R)
  972. efree(buf);
  973. #endif
  974. }
  975. /* }}} */
  976. /* {{{ proto array posix_getpwuid(long uid)
  977. User database access (POSIX.1, 9.2.2) */
  978. PHP_FUNCTION(posix_getpwuid)
  979. {
  980. long uid;
  981. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWUID_R)
  982. struct passwd _pw;
  983. struct passwd *retpwptr = NULL;
  984. long pwbuflen;
  985. char *pwbuf;
  986. int ret;
  987. #endif
  988. struct passwd *pw;
  989. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &uid) == FAILURE) {
  990. RETURN_FALSE;
  991. }
  992. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWUID_R)
  993. pwbuflen = sysconf(_SC_GETPW_R_SIZE_MAX);
  994. if (pwbuflen < 1) {
  995. RETURN_FALSE;
  996. }
  997. pwbuf = emalloc(pwbuflen);
  998. ret = getpwuid_r(uid, &_pw, pwbuf, pwbuflen, &retpwptr);
  999. if (ret || retpwptr == NULL) {
  1000. POSIX_G(last_error) = ret;
  1001. efree(pwbuf);
  1002. RETURN_FALSE;
  1003. }
  1004. pw = &_pw;
  1005. #else
  1006. if (NULL == (pw = getpwuid(uid))) {
  1007. POSIX_G(last_error) = errno;
  1008. RETURN_FALSE;
  1009. }
  1010. #endif
  1011. array_init(return_value);
  1012. if (!php_posix_passwd_to_array(pw, return_value)) {
  1013. zval_dtor(return_value);
  1014. php_error_docref(NULL TSRMLS_CC, E_WARNING, "unable to convert posix passwd struct to array");
  1015. RETVAL_FALSE;
  1016. }
  1017. #if defined(ZTS) && defined(_SC_GETPW_R_SIZE_MAX) && defined(HAVE_GETPWUID_R)
  1018. efree(pwbuf);
  1019. #endif
  1020. }
  1021. /* }}} */
  1022. #ifdef HAVE_GETRLIMIT
  1023. #define UNLIMITED_STRING "unlimited"
  1024. /* {{{ posix_addlimit
  1025. */
  1026. static int posix_addlimit(int limit, char *name, zval *return_value TSRMLS_DC) {
  1027. int result;
  1028. struct rlimit rl;
  1029. char hard[80];
  1030. char soft[80];
  1031. snprintf(hard, 80, "hard %s", name);
  1032. snprintf(soft, 80, "soft %s", name);
  1033. result = getrlimit(limit, &rl);
  1034. if (result < 0) {
  1035. POSIX_G(last_error) = errno;
  1036. return FAILURE;
  1037. }
  1038. if (rl.rlim_cur == RLIM_INFINITY) {
  1039. add_assoc_stringl(return_value, soft, UNLIMITED_STRING, sizeof(UNLIMITED_STRING)-1, 1);
  1040. } else {
  1041. add_assoc_long(return_value, soft, rl.rlim_cur);
  1042. }
  1043. if (rl.rlim_max == RLIM_INFINITY) {
  1044. add_assoc_stringl(return_value, hard, UNLIMITED_STRING, sizeof(UNLIMITED_STRING)-1, 1);
  1045. } else {
  1046. add_assoc_long(return_value, hard, rl.rlim_max);
  1047. }
  1048. return SUCCESS;
  1049. }
  1050. /* }}} */
  1051. /* {{{ limits[]
  1052. */
  1053. struct limitlist {
  1054. int limit;
  1055. char *name;
  1056. } limits[] = {
  1057. #ifdef RLIMIT_CORE
  1058. { RLIMIT_CORE, "core" },
  1059. #endif
  1060. #ifdef RLIMIT_DATA
  1061. { RLIMIT_DATA, "data" },
  1062. #endif
  1063. #ifdef RLIMIT_STACK
  1064. { RLIMIT_STACK, "stack" },
  1065. #endif
  1066. #ifdef RLIMIT_VMEM
  1067. { RLIMIT_VMEM, "virtualmem" },
  1068. #endif
  1069. #ifdef RLIMIT_AS
  1070. { RLIMIT_AS, "totalmem" },
  1071. #endif
  1072. #ifdef RLIMIT_RSS
  1073. { RLIMIT_RSS, "rss" },
  1074. #endif
  1075. #ifdef RLIMIT_NPROC
  1076. { RLIMIT_NPROC, "maxproc" },
  1077. #endif
  1078. #ifdef RLIMIT_MEMLOCK
  1079. { RLIMIT_MEMLOCK, "memlock" },
  1080. #endif
  1081. #ifdef RLIMIT_CPU
  1082. { RLIMIT_CPU, "cpu" },
  1083. #endif
  1084. #ifdef RLIMIT_FSIZE
  1085. { RLIMIT_FSIZE, "filesize" },
  1086. #endif
  1087. #ifdef RLIMIT_NOFILE
  1088. { RLIMIT_NOFILE, "openfiles" },
  1089. #endif
  1090. #ifdef RLIMIT_OFILE
  1091. { RLIMIT_OFILE, "openfiles" },
  1092. #endif
  1093. { 0, NULL }
  1094. };
  1095. /* }}} */
  1096. /* {{{ proto array posix_getrlimit(void)
  1097. Get system resource consumption limits (This is not a POSIX function, but a BSDism and a SVR4ism. We compile conditionally) */
  1098. PHP_FUNCTION(posix_getrlimit)
  1099. {
  1100. struct limitlist *l = NULL;
  1101. PHP_POSIX_NO_ARGS;
  1102. array_init(return_value);
  1103. for (l=limits; l->name; l++) {
  1104. if (posix_addlimit(l->limit, l->name, return_value TSRMLS_CC) == FAILURE) {
  1105. zval_dtor(return_value);
  1106. RETURN_FALSE;
  1107. }
  1108. }
  1109. }
  1110. /* }}} */
  1111. #endif /* HAVE_GETRLIMIT */
  1112. /* {{{ proto int posix_get_last_error(void)
  1113. Retrieve the error number set by the last posix function which failed. */
  1114. PHP_FUNCTION(posix_get_last_error)
  1115. {
  1116. PHP_POSIX_NO_ARGS;
  1117. RETURN_LONG(POSIX_G(last_error));
  1118. }
  1119. /* }}} */
  1120. /* {{{ proto string posix_strerror(int errno)
  1121. Retrieve the system error message associated with the given errno. */
  1122. PHP_FUNCTION(posix_strerror)
  1123. {
  1124. long error;
  1125. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l", &error) == FAILURE) {
  1126. RETURN_FALSE;
  1127. }
  1128. RETURN_STRING(strerror(error), 1);
  1129. }
  1130. /* }}} */
  1131. #endif
  1132. #ifdef HAVE_INITGROUPS
  1133. /* {{{ proto bool posix_initgroups(string name, int base_group_id)
  1134. Calculate the group access list for the user specified in name. */
  1135. PHP_FUNCTION(posix_initgroups)
  1136. {
  1137. long basegid;
  1138. char *name;
  1139. int name_len;
  1140. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sl", &name, &name_len, &basegid) == FAILURE) {
  1141. RETURN_FALSE;
  1142. }
  1143. if (name_len == 0) {
  1144. RETURN_FALSE;
  1145. }
  1146. RETURN_BOOL(!initgroups((const char *)name, basegid));
  1147. }
  1148. /* }}} */
  1149. #endif
  1150. /*
  1151. * Local variables:
  1152. * tab-width: 4
  1153. * c-basic-offset: 4
  1154. * End:
  1155. * vim600: sw=4 ts=4 fdm=marker
  1156. * vim<600: sw=4 ts=4
  1157. */