lcg.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  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: Sascha Schumann <sascha@schumann.cx> |
  16. +----------------------------------------------------------------------+
  17. */
  18. /* $Id$ */
  19. #include "php.h"
  20. #include "php_lcg.h"
  21. #if HAVE_UNISTD_H
  22. #include <unistd.h>
  23. #endif
  24. #ifdef PHP_WIN32
  25. #include "win32/time.h"
  26. #else
  27. #include <sys/time.h>
  28. #endif
  29. #ifdef ZTS
  30. int lcg_globals_id;
  31. #else
  32. static php_lcg_globals lcg_globals;
  33. #endif
  34. #ifdef PHP_WIN32
  35. #include <process.h>
  36. #endif
  37. /*
  38. * combinedLCG() returns a pseudo random number in the range of (0, 1).
  39. * The function combines two CGs with periods of
  40. * 2^31 - 85 and 2^31 - 249. The period of this function
  41. * is equal to the product of both primes.
  42. */
  43. #define MODMULT(a, b, c, m, s) q = s/a;s=b*(s-a*q)-c*q;if(s<0)s+=m
  44. static void lcg_seed(TSRMLS_D);
  45. PHPAPI double php_combined_lcg(TSRMLS_D) /* {{{ */
  46. {
  47. php_int32 q;
  48. php_int32 z;
  49. if (!LCG(seeded)) {
  50. lcg_seed(TSRMLS_C);
  51. }
  52. MODMULT(53668, 40014, 12211, 2147483563L, LCG(s1));
  53. MODMULT(52774, 40692, 3791, 2147483399L, LCG(s2));
  54. z = LCG(s1) - LCG(s2);
  55. if (z < 1) {
  56. z += 2147483562;
  57. }
  58. return z * 4.656613e-10;
  59. }
  60. /* }}} */
  61. static void lcg_seed(TSRMLS_D) /* {{{ */
  62. {
  63. struct timeval tv;
  64. if (gettimeofday(&tv, NULL) == 0) {
  65. LCG(s1) = tv.tv_sec ^ (tv.tv_usec<<11);
  66. } else {
  67. LCG(s1) = 1;
  68. }
  69. #ifdef ZTS
  70. LCG(s2) = (long) tsrm_thread_id();
  71. #else
  72. LCG(s2) = (long) getpid();
  73. #endif
  74. /* Add entropy to s2 by calling gettimeofday() again */
  75. if (gettimeofday(&tv, NULL) == 0) {
  76. LCG(s2) ^= (tv.tv_usec<<11);
  77. }
  78. LCG(seeded) = 1;
  79. }
  80. /* }}} */
  81. static void lcg_init_globals(php_lcg_globals *lcg_globals_p TSRMLS_DC) /* {{{ */
  82. {
  83. LCG(seeded) = 0;
  84. }
  85. /* }}} */
  86. PHP_MINIT_FUNCTION(lcg) /* {{{ */
  87. {
  88. #ifdef ZTS
  89. ts_allocate_id(&lcg_globals_id, sizeof(php_lcg_globals), (ts_allocate_ctor) lcg_init_globals, NULL);
  90. #else
  91. lcg_init_globals(&lcg_globals);
  92. #endif
  93. return SUCCESS;
  94. }
  95. /* }}} */
  96. /* {{{ proto float lcg_value()
  97. Returns a value from the combined linear congruential generator */
  98. PHP_FUNCTION(lcg_value)
  99. {
  100. RETURN_DOUBLE(php_combined_lcg(TSRMLS_C));
  101. }
  102. /* }}} */
  103. /*
  104. * Local variables:
  105. * tab-width: 4
  106. * c-basic-offset: 4
  107. * End:
  108. * vim600: sw=4 ts=4 fdm=marker
  109. * vim<600: sw=4 ts=4
  110. */