timezone_methods.cpp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 5 |
  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/timezone.h>
  22. #include <unicode/ustring.h>
  23. #include "intl_convertcpp.h"
  24. #include "../common/common_date.h"
  25. extern "C" {
  26. #include "../php_intl.h"
  27. #define USE_TIMEZONE_POINTER 1
  28. #include "timezone_class.h"
  29. #include "intl_convert.h"
  30. #include <zend_exceptions.h>
  31. #include <ext/date/php_date.h>
  32. }
  33. #include "common/common_enum.h"
  34. U_CFUNC PHP_METHOD(IntlTimeZone, __construct)
  35. {
  36. zend_throw_exception( NULL,
  37. "An object of this type cannot be created with the new operator",
  38. 0 TSRMLS_CC );
  39. }
  40. U_CFUNC PHP_FUNCTION(intltz_create_time_zone)
  41. {
  42. char *str_id;
  43. int str_id_len;
  44. intl_error_reset(NULL TSRMLS_CC);
  45. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s",
  46. &str_id, &str_id_len) == FAILURE) {
  47. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  48. "intltz_create_time_zone: bad arguments", 0 TSRMLS_CC);
  49. RETURN_NULL();
  50. }
  51. UErrorCode status = UErrorCode();
  52. UnicodeString id = UnicodeString();
  53. if (intl_stringFromChar(id, str_id, str_id_len, &status) == FAILURE) {
  54. intl_error_set(NULL, status,
  55. "intltz_create_time_zone: could not convert time zone id to UTF-16", 0 TSRMLS_CC);
  56. RETURN_NULL();
  57. }
  58. //guaranteed non-null; GMT if timezone cannot be understood
  59. TimeZone *tz = TimeZone::createTimeZone(id);
  60. timezone_object_construct(tz, return_value, 1 TSRMLS_CC);
  61. }
  62. U_CFUNC PHP_FUNCTION(intltz_from_date_time_zone)
  63. {
  64. zval *zv_timezone;
  65. TimeZone *tz;
  66. php_timezone_obj *tzobj;
  67. intl_error_reset(NULL TSRMLS_CC);
  68. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O",
  69. &zv_timezone, php_date_get_timezone_ce()) == FAILURE) {
  70. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  71. "intltz_from_date_time_zone: bad arguments", 0 TSRMLS_CC);
  72. RETURN_NULL();
  73. }
  74. tzobj = (php_timezone_obj *)zend_objects_get_address(zv_timezone TSRMLS_CC);
  75. if (!tzobj->initialized) {
  76. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  77. "intltz_from_date_time_zone: DateTimeZone object is unconstructed",
  78. 0 TSRMLS_CC);
  79. RETURN_NULL();
  80. }
  81. tz = timezone_convert_datetimezone(tzobj->type, tzobj, FALSE, NULL,
  82. "intltz_from_date_time_zone" TSRMLS_CC);
  83. if (tz == NULL) {
  84. RETURN_NULL();
  85. }
  86. timezone_object_construct(tz, return_value, 1 TSRMLS_CC);
  87. }
  88. U_CFUNC PHP_FUNCTION(intltz_create_default)
  89. {
  90. intl_error_reset(NULL TSRMLS_CC);
  91. if (zend_parse_parameters_none() == FAILURE) {
  92. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  93. "intltz_create_default: bad arguments", 0 TSRMLS_CC);
  94. RETURN_NULL();
  95. }
  96. TimeZone *tz = TimeZone::createDefault();
  97. timezone_object_construct(tz, return_value, 1 TSRMLS_CC);
  98. }
  99. U_CFUNC PHP_FUNCTION(intltz_get_gmt)
  100. {
  101. intl_error_reset(NULL TSRMLS_CC);
  102. if (zend_parse_parameters_none() == FAILURE) {
  103. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  104. "intltz_get_gmt: bad arguments", 0 TSRMLS_CC);
  105. RETURN_NULL();
  106. }
  107. timezone_object_construct(TimeZone::getGMT(), return_value, 0 TSRMLS_CC);
  108. }
  109. #if U_ICU_VERSION_MAJOR_NUM >= 49
  110. U_CFUNC PHP_FUNCTION(intltz_get_unknown)
  111. {
  112. intl_error_reset(NULL TSRMLS_CC);
  113. if (zend_parse_parameters_none() == FAILURE) {
  114. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  115. "intltz_get_unknown: bad arguments", 0 TSRMLS_CC);
  116. RETURN_NULL();
  117. }
  118. timezone_object_construct(&TimeZone::getUnknown(), return_value, 0 TSRMLS_CC);
  119. }
  120. #endif
  121. U_CFUNC PHP_FUNCTION(intltz_create_enumeration)
  122. {
  123. zval **arg = NULL;
  124. StringEnumeration *se = NULL;
  125. intl_error_reset(NULL TSRMLS_CC);
  126. /* double indirection to have the zend engine destroy the new zval that
  127. * results from separation */
  128. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "|Z", &arg) == FAILURE) {
  129. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  130. "intltz_create_enumeration: bad arguments", 0 TSRMLS_CC);
  131. RETURN_FALSE;
  132. }
  133. if (arg == NULL || Z_TYPE_PP(arg) == IS_NULL) {
  134. se = TimeZone::createEnumeration();
  135. } else if (Z_TYPE_PP(arg) == IS_LONG) {
  136. int_offset:
  137. if (Z_LVAL_PP(arg) < (long)INT32_MIN ||
  138. Z_LVAL_PP(arg) > (long)INT32_MAX) {
  139. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  140. "intltz_create_enumeration: value is out of range", 0 TSRMLS_CC);
  141. RETURN_FALSE;
  142. } else {
  143. se = TimeZone::createEnumeration((int32_t) Z_LVAL_PP(arg));
  144. }
  145. } else if (Z_TYPE_PP(arg) == IS_DOUBLE) {
  146. double_offset:
  147. convert_to_long_ex(arg);
  148. goto int_offset;
  149. } else if (Z_TYPE_PP(arg) == IS_OBJECT || Z_TYPE_PP(arg) == IS_STRING) {
  150. long lval;
  151. double dval;
  152. convert_to_string_ex(arg);
  153. switch (is_numeric_string(Z_STRVAL_PP(arg), Z_STRLEN_PP(arg), &lval, &dval, 0)) {
  154. case IS_DOUBLE:
  155. SEPARATE_ZVAL(arg);
  156. zval_dtor(*arg);
  157. Z_TYPE_PP(arg) = IS_DOUBLE;
  158. Z_DVAL_PP(arg) = dval;
  159. goto double_offset;
  160. case IS_LONG:
  161. SEPARATE_ZVAL(arg);
  162. zval_dtor(*arg);
  163. Z_TYPE_PP(arg) = IS_LONG;
  164. Z_LVAL_PP(arg) = lval;
  165. goto int_offset;
  166. }
  167. /* else call string version */
  168. se = TimeZone::createEnumeration(Z_STRVAL_PP(arg));
  169. } else {
  170. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  171. "intltz_create_enumeration: invalid argument type", 0 TSRMLS_CC);
  172. RETURN_FALSE;
  173. }
  174. if (se) {
  175. IntlIterator_from_StringEnumeration(se, return_value TSRMLS_CC);
  176. } else {
  177. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  178. "intltz_create_enumeration: error obtaining enumeration", 0 TSRMLS_CC);
  179. RETVAL_FALSE;
  180. }
  181. }
  182. U_CFUNC PHP_FUNCTION(intltz_count_equivalent_ids)
  183. {
  184. char *str_id;
  185. int str_id_len;
  186. intl_error_reset(NULL TSRMLS_CC);
  187. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s",
  188. &str_id, &str_id_len) == FAILURE) {
  189. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  190. "intltz_count_equivalent_ids: bad arguments", 0 TSRMLS_CC);
  191. RETURN_FALSE;
  192. }
  193. UErrorCode status = UErrorCode();
  194. UnicodeString id = UnicodeString();
  195. if (intl_stringFromChar(id, str_id, str_id_len, &status) == FAILURE) {
  196. intl_error_set(NULL, status,
  197. "intltz_count_equivalent_ids: could not convert time zone id to UTF-16", 0 TSRMLS_CC);
  198. RETURN_FALSE;
  199. }
  200. int32_t result = TimeZone::countEquivalentIDs(id);
  201. RETURN_LONG((long)result);
  202. }
  203. #if U_ICU_VERSION_MAJOR_NUM * 10 + U_ICU_VERSION_MINOR_NUM >= 48
  204. U_CFUNC PHP_FUNCTION(intltz_create_time_zone_id_enumeration)
  205. {
  206. long zoneType,
  207. offset_arg;
  208. char *region = NULL;
  209. int region_len = 0;
  210. int32_t offset,
  211. *offsetp = NULL;
  212. int arg3isnull = 0;
  213. intl_error_reset(NULL TSRMLS_CC);
  214. /* must come before zpp because zpp would convert the arg in the stack to 0 */
  215. if (ZEND_NUM_ARGS() == 3) {
  216. zval **dummy, **zvoffset;
  217. arg3isnull = zend_get_parameters_ex(3, &dummy, &dummy, &zvoffset)
  218. != FAILURE && Z_TYPE_PP(zvoffset) == IS_NULL;
  219. }
  220. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "l|s!l",
  221. &zoneType, &region, &region_len, &offset_arg) == FAILURE) {
  222. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  223. "intltz_create_time_zone_id_enumeration: bad arguments", 0 TSRMLS_CC);
  224. RETURN_FALSE;
  225. }
  226. if (zoneType != UCAL_ZONE_TYPE_ANY && zoneType != UCAL_ZONE_TYPE_CANONICAL
  227. && zoneType != UCAL_ZONE_TYPE_CANONICAL_LOCATION) {
  228. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  229. "intltz_create_time_zone_id_enumeration: bad zone type", 0 TSRMLS_CC);
  230. RETURN_FALSE;
  231. }
  232. if (ZEND_NUM_ARGS() == 3) {
  233. if (offset_arg < (long)INT32_MIN || offset_arg > (long)INT32_MAX) {
  234. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  235. "intltz_create_time_zone_id_enumeration: offset out of bounds", 0 TSRMLS_CC);
  236. RETURN_FALSE;
  237. }
  238. if (!arg3isnull) {
  239. offset = (int32_t)offset_arg;
  240. offsetp = &offset;
  241. } //else leave offsetp NULL
  242. }
  243. StringEnumeration *se;
  244. UErrorCode uec = UErrorCode();
  245. se = TimeZone::createTimeZoneIDEnumeration((USystemTimeZoneType)zoneType,
  246. region, offsetp, uec);
  247. INTL_CHECK_STATUS(uec, "intltz_create_time_zone_id_enumeration: "
  248. "Error obtaining time zone id enumeration")
  249. IntlIterator_from_StringEnumeration(se, return_value TSRMLS_CC);
  250. }
  251. #endif
  252. U_CFUNC PHP_FUNCTION(intltz_get_canonical_id)
  253. {
  254. char *str_id;
  255. int str_id_len;
  256. zval *is_systemid = NULL;
  257. intl_error_reset(NULL TSRMLS_CC);
  258. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s|z",
  259. &str_id, &str_id_len, &is_systemid) == FAILURE) {
  260. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  261. "intltz_get_canonical_id: bad arguments", 0 TSRMLS_CC);
  262. RETURN_FALSE;
  263. }
  264. UErrorCode status = UErrorCode();
  265. UnicodeString id;
  266. if (intl_stringFromChar(id, str_id, str_id_len, &status) == FAILURE) {
  267. intl_error_set(NULL, status,
  268. "intltz_get_canonical_id: could not convert time zone id to UTF-16", 0 TSRMLS_CC);
  269. RETURN_FALSE;
  270. }
  271. UnicodeString result;
  272. UBool isSystemID;
  273. TimeZone::getCanonicalID(id, result, isSystemID, status);
  274. INTL_CHECK_STATUS(status, "intltz_get_canonical_id: error obtaining canonical ID");
  275. intl_convert_utf16_to_utf8(&Z_STRVAL_P(return_value), &Z_STRLEN_P(return_value),
  276. result.getBuffer(), result.length(), &status);
  277. INTL_CHECK_STATUS(status,
  278. "intltz_get_canonical_id: could not convert time zone id to UTF-16");
  279. Z_TYPE_P(return_value) = IS_STRING;
  280. if (is_systemid) { /* by-ref argument passed */
  281. zval_dtor(is_systemid);
  282. ZVAL_BOOL(is_systemid, isSystemID);
  283. }
  284. }
  285. #if U_ICU_VERSION_MAJOR_NUM * 10 + U_ICU_VERSION_MINOR_NUM >= 48
  286. U_CFUNC PHP_FUNCTION(intltz_get_region)
  287. {
  288. char *str_id;
  289. int str_id_len;
  290. char outbuf[3];
  291. intl_error_reset(NULL TSRMLS_CC);
  292. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s",
  293. &str_id, &str_id_len) == FAILURE) {
  294. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  295. "intltz_get_region: bad arguments", 0 TSRMLS_CC);
  296. RETURN_FALSE;
  297. }
  298. UErrorCode status = UErrorCode();
  299. UnicodeString id;
  300. if (intl_stringFromChar(id, str_id, str_id_len, &status) == FAILURE) {
  301. intl_error_set(NULL, status,
  302. "intltz_get_region: could not convert time zone id to UTF-16", 0 TSRMLS_CC);
  303. RETURN_FALSE;
  304. }
  305. int32_t region_len = TimeZone::getRegion(id, outbuf, sizeof(outbuf), status);
  306. INTL_CHECK_STATUS(status, "intltz_get_region: Error obtaining region");
  307. RETURN_STRINGL(outbuf, region_len, 1);
  308. }
  309. #endif
  310. U_CFUNC PHP_FUNCTION(intltz_get_tz_data_version)
  311. {
  312. intl_error_reset(NULL TSRMLS_CC);
  313. if (zend_parse_parameters_none() == FAILURE) {
  314. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  315. "intltz_get_tz_data_version: bad arguments", 0 TSRMLS_CC);
  316. RETURN_FALSE;
  317. }
  318. UErrorCode status = UErrorCode();
  319. const char *res = TimeZone::getTZDataVersion(status);
  320. INTL_CHECK_STATUS(status, "intltz_get_tz_data_version: "
  321. "Error obtaining time zone data version");
  322. RETURN_STRING(res, 1);
  323. }
  324. U_CFUNC PHP_FUNCTION(intltz_get_equivalent_id)
  325. {
  326. char *str_id;
  327. int str_id_len;
  328. long index;
  329. intl_error_reset(NULL TSRMLS_CC);
  330. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "sl",
  331. &str_id, &str_id_len, &index) == FAILURE ||
  332. index < (long)INT32_MIN || index > (long)INT32_MAX) {
  333. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  334. "intltz_get_equivalent_id: bad arguments", 0 TSRMLS_CC);
  335. RETURN_FALSE;
  336. }
  337. UErrorCode status = UErrorCode();
  338. UnicodeString id;
  339. if (intl_stringFromChar(id, str_id, str_id_len, &status) == FAILURE) {
  340. intl_error_set(NULL, status,
  341. "intltz_get_equivalent_id: could not convert time zone id to UTF-16", 0 TSRMLS_CC);
  342. RETURN_FALSE;
  343. }
  344. const UnicodeString result = TimeZone::getEquivalentID(id, (int32_t)index);
  345. intl_convert_utf16_to_utf8(&Z_STRVAL_P(return_value), &Z_STRLEN_P(return_value),
  346. result.getBuffer(), result.length(), &status);
  347. INTL_CHECK_STATUS(status, "intltz_get_equivalent_id: "
  348. "could not convert resulting time zone id to UTF-16");
  349. Z_TYPE_P(return_value) = IS_STRING;
  350. }
  351. U_CFUNC PHP_FUNCTION(intltz_get_id)
  352. {
  353. TIMEZONE_METHOD_INIT_VARS;
  354. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O",
  355. &object, TimeZone_ce_ptr) == FAILURE) {
  356. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  357. "intltz_get_id: bad arguments", 0 TSRMLS_CC);
  358. RETURN_FALSE;
  359. }
  360. TIMEZONE_METHOD_FETCH_OBJECT;
  361. UnicodeString id_us;
  362. to->utimezone->getID(id_us);
  363. char *id = NULL;
  364. int id_len = 0;
  365. intl_convert_utf16_to_utf8(&id, &id_len,
  366. id_us.getBuffer(), id_us.length(), TIMEZONE_ERROR_CODE_P(to));
  367. INTL_METHOD_CHECK_STATUS(to, "intltz_get_id: Could not convert id to UTF-8");
  368. RETURN_STRINGL(id, id_len, 0);
  369. }
  370. U_CFUNC PHP_FUNCTION(intltz_use_daylight_time)
  371. {
  372. TIMEZONE_METHOD_INIT_VARS;
  373. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O",
  374. &object, TimeZone_ce_ptr) == FAILURE) {
  375. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  376. "intltz_use_daylight_time: bad arguments", 0 TSRMLS_CC);
  377. RETURN_FALSE;
  378. }
  379. TIMEZONE_METHOD_FETCH_OBJECT;
  380. RETURN_BOOL(to->utimezone->useDaylightTime());
  381. }
  382. U_CFUNC PHP_FUNCTION(intltz_get_offset)
  383. {
  384. UDate date;
  385. zend_bool local;
  386. zval *rawOffsetArg,
  387. *dstOffsetArg;
  388. int32_t rawOffset,
  389. dstOffset;
  390. TIMEZONE_METHOD_INIT_VARS;
  391. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  392. "Odbzz", &object, TimeZone_ce_ptr, &date, &local, &rawOffsetArg,
  393. &dstOffsetArg) == FAILURE) {
  394. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  395. "intltz_get_offset: bad arguments", 0 TSRMLS_CC);
  396. RETURN_FALSE;
  397. }
  398. TIMEZONE_METHOD_FETCH_OBJECT;
  399. to->utimezone->getOffset(date, (UBool) local, rawOffset, dstOffset,
  400. TIMEZONE_ERROR_CODE(to));
  401. INTL_METHOD_CHECK_STATUS(to, "intltz_get_offset: error obtaining offset");
  402. zval_dtor(rawOffsetArg);
  403. ZVAL_LONG(rawOffsetArg, rawOffset);
  404. zval_dtor(dstOffsetArg);
  405. ZVAL_LONG(dstOffsetArg, dstOffset);
  406. RETURN_TRUE;
  407. }
  408. U_CFUNC PHP_FUNCTION(intltz_get_raw_offset)
  409. {
  410. TIMEZONE_METHOD_INIT_VARS;
  411. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  412. "O", &object, TimeZone_ce_ptr) == FAILURE) {
  413. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  414. "intltz_get_raw_offset: bad arguments", 0 TSRMLS_CC);
  415. RETURN_FALSE;
  416. }
  417. TIMEZONE_METHOD_FETCH_OBJECT;
  418. RETURN_LONG(to->utimezone->getRawOffset());
  419. }
  420. U_CFUNC PHP_FUNCTION(intltz_has_same_rules)
  421. {
  422. zval *other_object;
  423. TimeZone_object *other_to;
  424. TIMEZONE_METHOD_INIT_VARS;
  425. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  426. "OO", &object, TimeZone_ce_ptr, &other_object, TimeZone_ce_ptr)
  427. == FAILURE) {
  428. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  429. "intltz_has_same_rules: bad arguments", 0 TSRMLS_CC);
  430. RETURN_FALSE;
  431. }
  432. TIMEZONE_METHOD_FETCH_OBJECT;
  433. other_to = (TimeZone_object *) zend_object_store_get_object(other_object TSRMLS_CC);
  434. if (other_to->utimezone == NULL) {
  435. intl_errors_set(&to->err, U_ILLEGAL_ARGUMENT_ERROR,
  436. "intltz_has_same_rules: The second IntlTimeZone is unconstructed", 0 TSRMLS_CC);
  437. RETURN_FALSE;
  438. }
  439. RETURN_BOOL(to->utimezone->hasSameRules(*other_to->utimezone));
  440. }
  441. static const TimeZone::EDisplayType display_types[] = {
  442. TimeZone::SHORT, TimeZone::LONG,
  443. #if U_ICU_VERSION_MAJOR_NUM * 10 + U_ICU_VERSION_MINOR_NUM >= 44
  444. TimeZone::SHORT_GENERIC, TimeZone::LONG_GENERIC,
  445. TimeZone::SHORT_GMT, TimeZone::LONG_GMT,
  446. TimeZone::SHORT_COMMONLY_USED, TimeZone::GENERIC_LOCATION
  447. #endif
  448. };
  449. U_CFUNC PHP_FUNCTION(intltz_get_display_name)
  450. {
  451. zend_bool daylight = 0;
  452. long display_type = TimeZone::LONG;
  453. const char *locale_str = NULL;
  454. int dummy = 0;
  455. TIMEZONE_METHOD_INIT_VARS;
  456. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  457. "O|bls!", &object, TimeZone_ce_ptr, &daylight, &display_type,
  458. &locale_str, &dummy) == FAILURE) {
  459. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  460. "intltz_get_display_name: bad arguments", 0 TSRMLS_CC);
  461. RETURN_FALSE;
  462. }
  463. bool found = false;
  464. for (int i = 0; !found && i < sizeof(display_types)/sizeof(*display_types); i++) {
  465. if (display_types[i] == display_type)
  466. found = true;
  467. }
  468. if (!found) {
  469. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  470. "intltz_get_display_name: wrong display type", 0 TSRMLS_CC);
  471. RETURN_FALSE;
  472. }
  473. if (!locale_str) {
  474. locale_str = intl_locale_get_default(TSRMLS_C);
  475. }
  476. TIMEZONE_METHOD_FETCH_OBJECT;
  477. UnicodeString result;
  478. to->utimezone->getDisplayName((UBool)daylight, (TimeZone::EDisplayType)display_type,
  479. Locale::createFromName(locale_str), result);
  480. intl_convert_utf16_to_utf8(&Z_STRVAL_P(return_value), &Z_STRLEN_P(return_value),
  481. result.getBuffer(), result.length(), TIMEZONE_ERROR_CODE_P(to));
  482. INTL_METHOD_CHECK_STATUS(to, "intltz_get_display_name: "
  483. "could not convert resulting time zone id to UTF-16");
  484. Z_TYPE_P(return_value) = IS_STRING;
  485. }
  486. U_CFUNC PHP_FUNCTION(intltz_get_dst_savings)
  487. {
  488. TIMEZONE_METHOD_INIT_VARS;
  489. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  490. "O", &object, TimeZone_ce_ptr) == FAILURE) {
  491. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  492. "intltz_get_dst_savings: bad arguments", 0 TSRMLS_CC);
  493. RETURN_FALSE;
  494. }
  495. TIMEZONE_METHOD_FETCH_OBJECT;
  496. RETURN_LONG((long)to->utimezone->getDSTSavings());
  497. }
  498. U_CFUNC PHP_FUNCTION(intltz_to_date_time_zone)
  499. {
  500. TIMEZONE_METHOD_INIT_VARS;
  501. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(),
  502. "O", &object, TimeZone_ce_ptr) == FAILURE) {
  503. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  504. "intltz_to_date_time_zone: bad arguments", 0 TSRMLS_CC);
  505. RETURN_FALSE;
  506. }
  507. TIMEZONE_METHOD_FETCH_OBJECT;
  508. zval *ret = timezone_convert_to_datetimezone(to->utimezone,
  509. &TIMEZONE_ERROR(to), "intltz_to_date_time_zone" TSRMLS_CC);
  510. if (ret) {
  511. RETURN_ZVAL(ret, 1, 1);
  512. } else {
  513. RETURN_FALSE;
  514. }
  515. }
  516. U_CFUNC PHP_FUNCTION(intltz_get_error_code)
  517. {
  518. TIMEZONE_METHOD_INIT_VARS
  519. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O",
  520. &object, TimeZone_ce_ptr) == FAILURE) {
  521. intl_error_set(NULL, U_ILLEGAL_ARGUMENT_ERROR,
  522. "intltz_get_error_code: bad arguments", 0 TSRMLS_CC);
  523. RETURN_FALSE;
  524. }
  525. /* Fetch the object (without resetting its last error code ). */
  526. to = (TimeZone_object*)zend_object_store_get_object(object TSRMLS_CC);
  527. if (to == NULL)
  528. RETURN_FALSE;
  529. RETURN_LONG((long)TIMEZONE_ERROR_CODE(to));
  530. }
  531. U_CFUNC PHP_FUNCTION(intltz_get_error_message)
  532. {
  533. const char* message = NULL;
  534. TIMEZONE_METHOD_INIT_VARS
  535. if (zend_parse_method_parameters(ZEND_NUM_ARGS() TSRMLS_CC, getThis(), "O",
  536. &object, TimeZone_ce_ptr) == FAILURE) {
  537. intl_error_set( NULL, U_ILLEGAL_ARGUMENT_ERROR,
  538. "intltz_get_error_message: bad arguments", 0 TSRMLS_CC );
  539. RETURN_FALSE;
  540. }
  541. /* Fetch the object (without resetting its last error code ). */
  542. to = (TimeZone_object*)zend_object_store_get_object(object TSRMLS_CC);
  543. if (to == NULL)
  544. RETURN_FALSE;
  545. /* Return last error message. */
  546. message = intl_error_get_message(TIMEZONE_ERROR_P(to) TSRMLS_CC);
  547. RETURN_STRING(message, 0);
  548. }