reentrancy.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 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: Sascha Schumann <sascha@schumann.cx> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #include <sys/types.h>
  19. #include <string.h>
  20. #include <errno.h>
  21. #ifdef HAVE_DIRENT_H
  22. #include <dirent.h>
  23. #endif
  24. #include "php_reentrancy.h"
  25. #include "ext/standard/php_rand.h" /* for PHP_RAND_MAX */
  26. enum {
  27. LOCALTIME_R,
  28. CTIME_R,
  29. ASCTIME_R,
  30. GMTIME_R,
  31. READDIR_R,
  32. NUMBER_OF_LOCKS
  33. };
  34. #if defined(PHP_NEED_REENTRANCY)
  35. #include <TSRM.h>
  36. static MUTEX_T reentrant_locks[NUMBER_OF_LOCKS];
  37. #define local_lock(x) tsrm_mutex_lock(reentrant_locks[x])
  38. #define local_unlock(x) tsrm_mutex_unlock(reentrant_locks[x])
  39. #else
  40. #define local_lock(x)
  41. #define local_unlock(x)
  42. #endif
  43. #if defined(PHP_IRIX_TIME_R)
  44. #define HAVE_CTIME_R 1
  45. #define HAVE_ASCTIME_R 1
  46. PHPAPI char *php_ctime_r(const time_t *clock, char *buf)
  47. {
  48. if (ctime_r(clock, buf) == buf)
  49. return (buf);
  50. return (NULL);
  51. }
  52. PHPAPI char *php_asctime_r(const struct tm *tm, char *buf)
  53. {
  54. if (asctime_r(tm, buf) == buf)
  55. return (buf);
  56. return (NULL);
  57. }
  58. #endif
  59. #if defined(PHP_HPUX_TIME_R)
  60. #define HAVE_LOCALTIME_R 1
  61. #define HAVE_CTIME_R 1
  62. #define HAVE_ASCTIME_R 1
  63. #define HAVE_GMTIME_R 1
  64. PHPAPI struct tm *php_localtime_r(const time_t *const timep, struct tm *p_tm)
  65. {
  66. if (localtime_r(timep, p_tm) == 0)
  67. return (p_tm);
  68. return (NULL);
  69. }
  70. PHPAPI char *php_ctime_r(const time_t *clock, char *buf)
  71. {
  72. if (ctime_r(clock, buf, 26) != -1)
  73. return (buf);
  74. return (NULL);
  75. }
  76. PHPAPI char *php_asctime_r(const struct tm *tm, char *buf)
  77. {
  78. if (asctime_r(tm, buf, 26) != -1)
  79. return (buf);
  80. return (NULL);
  81. }
  82. PHPAPI struct tm *php_gmtime_r(const time_t *const timep, struct tm *p_tm)
  83. {
  84. if (gmtime_r(timep, p_tm) == 0)
  85. return (p_tm);
  86. return (NULL);
  87. }
  88. #endif
  89. #if !defined(HAVE_POSIX_READDIR_R)
  90. PHPAPI int php_readdir_r(DIR *dirp, struct dirent *entry,
  91. struct dirent **result)
  92. {
  93. #if defined(HAVE_OLD_READDIR_R)
  94. int ret = 0;
  95. /* We cannot rely on the return value of readdir_r
  96. as it differs between various platforms
  97. (HPUX returns 0 on success whereas Solaris returns non-zero)
  98. */
  99. entry->d_name[0] = '\0';
  100. readdir_r(dirp, entry);
  101. if (entry->d_name[0] == '\0') {
  102. *result = NULL;
  103. ret = errno;
  104. } else {
  105. *result = entry;
  106. }
  107. return ret;
  108. #else
  109. struct dirent *ptr;
  110. int ret = 0;
  111. local_lock(READDIR_R);
  112. errno = 0;
  113. ptr = readdir(dirp);
  114. if (!ptr && errno != 0)
  115. ret = errno;
  116. if (ptr)
  117. memcpy(entry, ptr, sizeof(*ptr));
  118. *result = ptr;
  119. local_unlock(READDIR_R);
  120. return ret;
  121. #endif
  122. }
  123. #endif
  124. #if !defined(HAVE_LOCALTIME_R) && defined(HAVE_LOCALTIME)
  125. PHPAPI struct tm *php_localtime_r(const time_t *const timep, struct tm *p_tm)
  126. {
  127. struct tm *tmp;
  128. local_lock(LOCALTIME_R);
  129. tmp = localtime(timep);
  130. if (tmp) {
  131. memcpy(p_tm, tmp, sizeof(struct tm));
  132. tmp = p_tm;
  133. }
  134. local_unlock(LOCALTIME_R);
  135. return tmp;
  136. }
  137. #endif
  138. #if !defined(HAVE_CTIME_R) && defined(HAVE_CTIME)
  139. PHPAPI char *php_ctime_r(const time_t *clock, char *buf)
  140. {
  141. char *tmp;
  142. local_lock(CTIME_R);
  143. tmp = ctime(clock);
  144. if (tmp) {
  145. strcpy(buf, tmp);
  146. tmp = buf;
  147. }
  148. local_unlock(CTIME_R);
  149. return tmp;
  150. }
  151. #endif
  152. #if !defined(HAVE_ASCTIME_R) && defined(HAVE_ASCTIME)
  153. PHPAPI char *php_asctime_r(const struct tm *tm, char *buf)
  154. {
  155. char *tmp;
  156. local_lock(ASCTIME_R);
  157. tmp = asctime(tm);
  158. if (tmp) {
  159. strcpy(buf, tmp);
  160. tmp = buf;
  161. }
  162. local_unlock(ASCTIME_R);
  163. return tmp;
  164. }
  165. #endif
  166. #if !defined(HAVE_GMTIME_R) && defined(HAVE_GMTIME)
  167. PHPAPI struct tm *php_gmtime_r(const time_t *const timep, struct tm *p_tm)
  168. {
  169. struct tm *tmp;
  170. local_lock(GMTIME_R);
  171. tmp = gmtime(timep);
  172. if (tmp) {
  173. memcpy(p_tm, tmp, sizeof(struct tm));
  174. tmp = p_tm;
  175. }
  176. local_unlock(GMTIME_R);
  177. return tmp;
  178. }
  179. #endif
  180. #if defined(PHP_NEED_REENTRANCY)
  181. void reentrancy_startup(void)
  182. {
  183. int i;
  184. for (i = 0; i < NUMBER_OF_LOCKS; i++) {
  185. reentrant_locks[i] = tsrm_mutex_alloc();
  186. }
  187. }
  188. void reentrancy_shutdown(void)
  189. {
  190. int i;
  191. for (i = 0; i < NUMBER_OF_LOCKS; i++) {
  192. tsrm_mutex_free(reentrant_locks[i]);
  193. }
  194. }
  195. #endif
  196. #ifndef HAVE_RAND_R
  197. /*-
  198. * Copyright (c) 1990, 1993
  199. * The Regents of the University of California. All rights reserved.
  200. *
  201. * Redistribution and use in source and binary forms, with or without
  202. * modification, are permitted provided that the following conditions
  203. * are met:
  204. * 1. Redistributions of source code must retain the above copyright
  205. * notice, this list of conditions and the following disclaimer.
  206. * 2. Redistributions in binary form must reproduce the above copyright
  207. * notice, this list of conditions and the following disclaimer in the
  208. * documentation and/or other materials provided with the distribution.
  209. * 3. All advertising materials mentioning features or use of this software
  210. * must display the following acknowledgement:
  211. * This product includes software developed by the University of
  212. * California, Berkeley and its contributors.
  213. * 4. Neither the name of the University nor the names of its contributors
  214. * may be used to endorse or promote products derived from this software
  215. * without specific prior written permission.
  216. *
  217. * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  218. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  219. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  220. * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  221. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  222. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  223. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  224. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  225. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  226. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  227. * SUCH DAMAGE.
  228. *
  229. * Posix rand_r function added May 1999 by Wes Peters <wes@softweyr.com>.
  230. */
  231. #include <sys/types.h>
  232. #include <stdlib.h>
  233. static int
  234. do_rand(unsigned long *ctx)
  235. {
  236. return ((*ctx = *ctx * 1103515245 + 12345) % ((u_long)PHP_RAND_MAX + 1));
  237. }
  238. PHPAPI int
  239. php_rand_r(unsigned int *ctx)
  240. {
  241. u_long val = (u_long) *ctx;
  242. *ctx = do_rand(&val);
  243. return (int) *ctx;
  244. }
  245. #endif
  246. #ifndef HAVE_STRTOK_R
  247. /*
  248. * Copyright (c) 1998 Softweyr LLC. All rights reserved.
  249. *
  250. * strtok_r, from Berkeley strtok
  251. * Oct 13, 1998 by Wes Peters <wes@softweyr.com>
  252. *
  253. * Copyright (c) 1988, 1993
  254. * The Regents of the University of California. All rights reserved.
  255. *
  256. * Redistribution and use in source and binary forms, with or without
  257. * modification, are permitted provided that the following conditions
  258. * are met:
  259. *
  260. * 1. Redistributions of source code must retain the above copyright
  261. * notices, this list of conditions and the following disclaimer.
  262. *
  263. * 2. Redistributions in binary form must reproduce the above copyright
  264. * notices, this list of conditions and the following disclaimer in the
  265. * documentation and/or other materials provided with the distribution.
  266. *
  267. * 3. All advertising materials mentioning features or use of this software
  268. * must display the following acknowledgement:
  269. *
  270. * This product includes software developed by Softweyr LLC, the
  271. * University of California, Berkeley, and its contributors.
  272. *
  273. * 4. Neither the name of the University nor the names of its contributors
  274. * may be used to endorse or promote products derived from this software
  275. * without specific prior written permission.
  276. *
  277. * THIS SOFTWARE IS PROVIDED BY SOFTWEYR LLC, THE REGENTS AND CONTRIBUTORS
  278. * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  279. * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
  280. * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL SOFTWEYR LLC, THE
  281. * REGENTS, OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  282. * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
  283. * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  284. * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  285. * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  286. * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  287. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  288. */
  289. #include <stddef.h>
  290. PHPAPI char *
  291. php_strtok_r(char *s, const char *delim, char **last)
  292. {
  293. char *spanp;
  294. int c, sc;
  295. char *tok;
  296. if (s == NULL && (s = *last) == NULL)
  297. {
  298. return NULL;
  299. }
  300. /*
  301. * Skip (span) leading delimiters (s += strspn(s, delim), sort of).
  302. */
  303. cont:
  304. c = *s++;
  305. for (spanp = (char *)delim; (sc = *spanp++) != 0; )
  306. {
  307. if (c == sc)
  308. {
  309. goto cont;
  310. }
  311. }
  312. if (c == 0) /* no non-delimiter characters */
  313. {
  314. *last = NULL;
  315. return NULL;
  316. }
  317. tok = s - 1;
  318. /*
  319. * Scan token (scan for delimiters: s += strcspn(s, delim), sort of).
  320. * Note that delim must have one NUL; we stop if we see that, too.
  321. */
  322. for (;;)
  323. {
  324. c = *s++;
  325. spanp = (char *)delim;
  326. do
  327. {
  328. if ((sc = *spanp++) == c)
  329. {
  330. if (c == 0)
  331. {
  332. s = NULL;
  333. }
  334. else
  335. {
  336. char *w = s - 1;
  337. *w = '\0';
  338. }
  339. *last = s;
  340. return tok;
  341. }
  342. }
  343. while (sc != 0);
  344. }
  345. /* NOTREACHED */
  346. }
  347. #endif
  348. /*
  349. * Local variables:
  350. * tab-width: 4
  351. * c-basic-offset: 4
  352. * End:
  353. * vim600: sw=4 ts=4 fdm=marker
  354. * vim<600: sw=4 ts=4
  355. */