gregoriancalendar_methods.cpp 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | This source file is subject to version 3.01 of the PHP license, |
  6. | that is bundled with this package in the file LICENSE, and is |
  7. | available through the world-wide-web at the following url: |
  8. | http://www.php.net/license/3_01.txt |
  9. | If you did not receive a copy of the PHP license and are unable to |
  10. | obtain it through the world-wide-web, please send a note to |
  11. | license@php.net so we can mail you a copy immediately. |
  12. +----------------------------------------------------------------------+
  13. | Authors: Gustavo Lopes <cataphract@php.net> |
  14. +----------------------------------------------------------------------+
  15. */
  16. #ifdef HAVE_CONFIG_H
  17. #include "config.h"
  18. #endif
  19. #include "../intl_cppshims.h"
  20. #include <unicode/locid.h>
  21. #include <unicode/calendar.h>
  22. #include <unicode/gregocal.h>
  23. #include <unicode/ustring.h>
  24. extern "C" {
  25. #include "../php_intl.h"
  26. #include "../intl_common.h"
  27. #define USE_TIMEZONE_POINTER 1
  28. #include "../timezone/timezone_class.h"
  29. #define USE_CALENDAR_POINTER 1
  30. #include "calendar_class.h"
  31. #include <ext/date/php_date.h>
  32. #include "zend_exceptions.h"
  33. }
  34. using icu::GregorianCalendar;
  35. using icu::Locale;
  36. using icu::UnicodeString;
  37. using icu::StringPiece;
  38. static inline GregorianCalendar *fetch_greg(Calendar_object *co) {
  39. return (GregorianCalendar*)co->ucal;
  40. }
  41. static void _php_intlgregcal_constructor_body(
  42. INTERNAL_FUNCTION_PARAMETERS, zend_bool is_constructor)
  43. {
  44. zval *tz_object = NULL;
  45. zval args_a[6] = {0},
  46. *args = &args_a[0];
  47. char *locale = NULL;
  48. size_t locale_len;
  49. zend_long largs[6];
  50. UErrorCode status = U_ZERO_ERROR;
  51. int variant;
  52. int zpp_flags = is_constructor ? ZEND_PARSE_PARAMS_THROW : 0;
  53. intl_error_reset(NULL);
  54. // parameter number validation / variant determination
  55. if (ZEND_NUM_ARGS() > 6 ||
  56. zend_get_parameters_array_ex(ZEND_NUM_ARGS(), args) == FAILURE) {
  57. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  58. "intlgregcal_create_instance: too many arguments", 0);
  59. if (!is_constructor) {
  60. zval_ptr_dtor(return_value);
  61. RETVAL_NULL();
  62. }
  63. return;
  64. }
  65. for (variant = ZEND_NUM_ARGS();
  66. variant > 0 && Z_TYPE(args[variant - 1]) == IS_NULL;
  67. variant--) {}
  68. if (variant == 4) {
  69. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  70. "intlgregcal_create_instance: no variant with 4 arguments "
  71. "(excluding trailing NULLs)", 0);
  72. if (!is_constructor) {
  73. zval_ptr_dtor(return_value);
  74. RETVAL_NULL();
  75. }
  76. return;
  77. }
  78. // argument parsing
  79. if (variant <= 2) {
  80. if (zend_parse_parameters_ex(zpp_flags, MIN(ZEND_NUM_ARGS(), 2),
  81. "|z!s!", &tz_object, &locale, &locale_len) == FAILURE) {
  82. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  83. "intlgregcal_create_instance: bad arguments", 0);
  84. if (!is_constructor) {
  85. zval_ptr_dtor(return_value);
  86. RETVAL_NULL();
  87. }
  88. return;
  89. }
  90. }
  91. if (variant > 2 && zend_parse_parameters_ex(zpp_flags, ZEND_NUM_ARGS(),
  92. "lll|lll", &largs[0], &largs[1], &largs[2], &largs[3], &largs[4],
  93. &largs[5]) == FAILURE) {
  94. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  95. "intlgregcal_create_instance: bad arguments", 0);
  96. if (!is_constructor) {
  97. zval_ptr_dtor(return_value);
  98. RETVAL_NULL();
  99. }
  100. return;
  101. }
  102. // instantion of ICU object
  103. GregorianCalendar *gcal = NULL;
  104. if (variant <= 2) {
  105. // From timezone and locale (0 to 2 arguments)
  106. TimeZone *tz = timezone_process_timezone_argument(tz_object, NULL,
  107. "intlgregcal_create_instance");
  108. if (tz == NULL) {
  109. if (!EG(exception)) {
  110. zend_throw_exception(IntlException_ce_ptr, "Constructor failed", 0);
  111. }
  112. if (!is_constructor) {
  113. zval_ptr_dtor(return_value);
  114. RETVAL_NULL();
  115. }
  116. return;
  117. }
  118. if (!locale) {
  119. locale = const_cast<char*>(intl_locale_get_default());
  120. }
  121. gcal = new GregorianCalendar(tz, Locale::createFromName(locale),
  122. status);
  123. if (U_FAILURE(status)) {
  124. intl_error_set(NULL, status, "intlgregcal_create_instance: error "
  125. "creating ICU GregorianCalendar from time zone and locale", 0);
  126. if (gcal) {
  127. delete gcal;
  128. }
  129. delete tz;
  130. if (!is_constructor) {
  131. zval_ptr_dtor(return_value);
  132. RETVAL_NULL();
  133. }
  134. return;
  135. }
  136. } else {
  137. // From date/time (3, 5 or 6 arguments)
  138. for (int i = 0; i < variant; i++) {
  139. if (largs[i] < INT32_MIN || largs[i] > INT32_MAX) {
  140. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  141. "intlgregcal_create_instance: at least one of the arguments"
  142. " has an absolute value that is too large", 0);
  143. if (!is_constructor) {
  144. zval_ptr_dtor(return_value);
  145. RETVAL_NULL();
  146. }
  147. return;
  148. }
  149. }
  150. if (variant == 3) {
  151. gcal = new GregorianCalendar((int32_t)largs[0], (int32_t)largs[1],
  152. (int32_t)largs[2], status);
  153. } else if (variant == 5) {
  154. gcal = new GregorianCalendar((int32_t)largs[0], (int32_t)largs[1],
  155. (int32_t)largs[2], (int32_t)largs[3], (int32_t)largs[4], status);
  156. } else if (variant == 6) {
  157. gcal = new GregorianCalendar((int32_t)largs[0], (int32_t)largs[1],
  158. (int32_t)largs[2], (int32_t)largs[3], (int32_t)largs[4], (int32_t)largs[5],
  159. status);
  160. }
  161. if (U_FAILURE(status)) {
  162. intl_error_set(NULL, status, "intlgregcal_create_instance: error "
  163. "creating ICU GregorianCalendar from date", 0);
  164. if (gcal) {
  165. delete gcal;
  166. }
  167. if (!is_constructor) {
  168. zval_ptr_dtor(return_value);
  169. RETVAL_NULL();
  170. }
  171. return;
  172. }
  173. timelib_tzinfo *tzinfo = get_timezone_info();
  174. #if U_ICU_VERSION_MAJOR_NUM * 10 + U_ICU_VERSION_MINOR_NUM >= 42
  175. UnicodeString tzstr = UnicodeString::fromUTF8(StringPiece(tzinfo->name));
  176. #else
  177. UnicodeString tzstr = UnicodeString(tzinfo->name,
  178. strlen(tzinfo->name), US_INV);
  179. #endif
  180. if (tzstr.isBogus()) {
  181. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  182. "intlgregcal_create_instance: could not create UTF-8 string "
  183. "from PHP's default timezone name (see date_default_timezone_get())",
  184. 0);
  185. delete gcal;
  186. if (!is_constructor) {
  187. zval_ptr_dtor(return_value);
  188. RETVAL_NULL();
  189. }
  190. return;
  191. }
  192. TimeZone *tz = TimeZone::createTimeZone(tzstr);
  193. gcal->adoptTimeZone(tz);
  194. }
  195. Calendar_object *co = Z_INTL_CALENDAR_P(return_value);
  196. co->ucal = gcal;
  197. }
  198. U_CFUNC PHP_FUNCTION(intlgregcal_create_instance)
  199. {
  200. intl_error_reset(NULL);
  201. object_init_ex(return_value, GregorianCalendar_ce_ptr);
  202. _php_intlgregcal_constructor_body(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);
  203. }
  204. U_CFUNC PHP_METHOD(IntlGregorianCalendar, __construct)
  205. {
  206. zend_error_handling error_handling;
  207. zend_replace_error_handling(EH_THROW, IntlException_ce_ptr, &error_handling);
  208. return_value = getThis();
  209. _php_intlgregcal_constructor_body(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);
  210. zend_restore_error_handling(&error_handling);
  211. }
  212. U_CFUNC PHP_FUNCTION(intlgregcal_set_gregorian_change)
  213. {
  214. double date;
  215. CALENDAR_METHOD_INIT_VARS;
  216. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(),
  217. "Od", &object, GregorianCalendar_ce_ptr, &date) == FAILURE) {
  218. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  219. "intlgregcal_set_gregorian_change: bad arguments", 0);
  220. RETURN_FALSE;
  221. }
  222. CALENDAR_METHOD_FETCH_OBJECT;
  223. fetch_greg(co)->setGregorianChange(date, CALENDAR_ERROR_CODE(co));
  224. INTL_METHOD_CHECK_STATUS(co, "intlgregcal_set_gregorian_change: error "
  225. "calling ICU method");
  226. RETURN_TRUE;
  227. }
  228. U_CFUNC PHP_FUNCTION(intlgregcal_get_gregorian_change)
  229. {
  230. CALENDAR_METHOD_INIT_VARS;
  231. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(),
  232. "O", &object, GregorianCalendar_ce_ptr) == FAILURE) {
  233. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  234. "intlgregcal_get_gregorian_change: bad arguments", 0);
  235. RETURN_FALSE;
  236. }
  237. CALENDAR_METHOD_FETCH_OBJECT;
  238. RETURN_DOUBLE((double)fetch_greg(co)->getGregorianChange());
  239. }
  240. U_CFUNC PHP_FUNCTION(intlgregcal_is_leap_year)
  241. {
  242. zend_long year;
  243. CALENDAR_METHOD_INIT_VARS;
  244. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(),
  245. "Ol", &object, GregorianCalendar_ce_ptr, &year) == FAILURE) {
  246. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  247. "intlgregcal_is_leap_year: bad arguments", 0);
  248. RETURN_FALSE;
  249. }
  250. if (year < INT32_MIN || year > INT32_MAX) {
  251. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  252. "intlgregcal_is_leap_year: year out of bounds", 0);
  253. RETURN_FALSE;
  254. }
  255. CALENDAR_METHOD_FETCH_OBJECT;
  256. RETURN_BOOL((int)fetch_greg(co)->isLeapYear((int32_t)year));
  257. }