2005-06-15 05:32:29 +08:00
/*
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| PHP Version 5 |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2005-08-03 22:08:58 +08:00
| Copyright ( c ) 1997 - 2005 The PHP Group |
2005-06-15 05:32:29 +08:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
| This source file is subject to version 3.0 of the PHP license , |
| that is bundled with this package in the file LICENSE , and is |
| available through the world - wide - web at the following url : |
| http : //www.php.net/license/3_0.txt. |
| If you did not receive a copy of the PHP license and are unable to |
| obtain it through the world - wide - web , please send a note to |
| license @ php . net so we can mail you a copy immediately . |
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
2005-06-17 22:54:00 +08:00
| Authors : Derick Rethans < derick @ derickrethans . nl > |
2005-06-15 05:32:29 +08:00
+ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +
*/
/* $Id$ */
# include "php.h"
# include "php_streams.h"
# include "php_main.h"
# include "php_globals.h"
# include "php_ini.h"
# include "ext/standard/info.h"
# include "php_date.h"
2005-06-17 01:12:41 +08:00
# include "lib/timelib.h"
2005-06-15 05:32:29 +08:00
# include <time.h>
2005-07-05 05:27:26 +08:00
/* {{{ Function table */
2005-06-15 05:32:29 +08:00
function_entry date_functions [ ] = {
2005-07-04 04:45:08 +08:00
PHP_FE ( strtotime , NULL )
2005-07-01 05:38:06 +08:00
PHP_FE ( date , NULL )
PHP_FE ( gmdate , NULL )
2005-07-03 22:27:31 +08:00
PHP_FE ( mktime , NULL )
2005-07-04 07:30:52 +08:00
PHP_FE ( gmmktime , NULL )
2005-07-04 03:14:55 +08:00
PHP_FE ( checkdate , NULL )
2005-07-04 04:45:08 +08:00
2005-07-04 07:30:52 +08:00
# ifdef HAVE_STRFTIME
2005-07-04 03:14:55 +08:00
PHP_FE ( strftime , NULL )
2005-07-04 07:30:52 +08:00
PHP_FE ( gmstrftime , NULL )
2005-07-04 04:45:08 +08:00
# endif
PHP_FE ( time , NULL )
PHP_FE ( localtime , NULL )
PHP_FE ( getdate , NULL )
2005-07-20 16:31:02 +08:00
# ifdef EXPERIMENTAL_DATE_SUPPORT
/* Advanced Interface */
PHP_FE ( date_create , NULL )
PHP_FE ( date_format , NULL )
PHP_FE ( date_modify , NULL )
PHP_FE ( date_timezone_get , NULL )
PHP_FE ( date_timezone_set , NULL )
PHP_FE ( date_offset_get , NULL )
PHP_FE ( timezone_open , NULL )
PHP_FE ( timezone_name_get , NULL )
PHP_FE ( timezone_offset_get , NULL )
PHP_FE ( timezone_transistions_get , NULL )
PHP_FE ( timezone_identifiers_list , NULL )
PHP_FE ( timezone_abbreviations_list , NULL )
# endif
/* Options and Configuration */
2005-07-05 05:27:26 +08:00
PHP_FE ( date_default_timezone_set , NULL )
PHP_FE ( date_default_timezone_get , NULL )
2005-06-15 05:32:29 +08:00
{ NULL , NULL , NULL }
} ;
2005-07-20 16:31:02 +08:00
# ifdef EXPERIMENTAL_DATE_SUPPORT
function_entry date_funcs_date [ ] = {
ZEND_NAMED_FE ( format , ZEND_FN ( date_format ) , NULL )
ZEND_NAMED_FE ( modify , ZEND_FN ( date_modify ) , NULL )
ZEND_NAMED_FE ( getTimezone , ZEND_FN ( date_timezone_get ) , NULL )
ZEND_NAMED_FE ( setTimezone , ZEND_FN ( date_timezone_set ) , NULL )
ZEND_NAMED_FE ( getOffset , ZEND_FN ( date_offset_get ) , NULL )
{ NULL , NULL , NULL }
} ;
function_entry date_funcs_timezone [ ] = {
ZEND_NAMED_FE ( getName , ZEND_FN ( timezone_name_get ) , NULL )
ZEND_NAMED_FE ( getOffset , ZEND_FN ( timezone_offset_get ) , NULL )
ZEND_NAMED_FE ( getTransistions , ZEND_FN ( timezone_transistions_get ) , NULL )
ZEND_MALIAS ( timezone , listAbbreviations , abbreviations_list , NULL , ZEND_ACC_PUBLIC | ZEND_ACC_STATIC )
ZEND_MALIAS ( timezone , listIdentifiers , identifiers_list , NULL , ZEND_ACC_PUBLIC | ZEND_ACC_STATIC )
{ NULL , NULL , NULL }
} ;
2005-08-10 05:10:22 +08:00
static void date_register_classes ( TSRMLS_D ) ;
# define DATE_REGISTER_CLASSES date_register_classes(TSRMLS_C)
2005-07-20 16:31:02 +08:00
# else
# define DATE_REGISTER_CLASSES /* */
# endif
2005-08-10 05:10:22 +08:00
static char * guess_timezone ( TSRMLS_D ) ;
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-06-19 04:23:19 +08:00
ZEND_DECLARE_MODULE_GLOBALS ( date )
2005-07-05 05:27:26 +08:00
/* {{{ INI Settings */
2005-06-15 05:32:29 +08:00
PHP_INI_BEGIN ( )
2005-07-03 05:19:25 +08:00
STD_PHP_INI_ENTRY ( " date.timezone " , " " , PHP_INI_ALL , OnUpdateString , default_timezone , zend_date_globals , date_globals )
2005-06-15 05:32:29 +08:00
PHP_INI_END ( )
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-20 16:31:02 +08:00
# ifdef EXPERIMENTAL_DATE_SUPPORT
typedef struct _php_date_obj php_date_obj ;
typedef struct _php_timezone_obj php_timezone_obj ;
struct _php_date_obj {
zend_object std ;
timelib_time * time ;
} ;
struct _php_timezone_obj {
zend_object std ;
timelib_tzinfo * tz ;
} ;
zend_class_entry * date_ce_date , * date_ce_timezone ;
static zend_object_handlers date_object_handlers_date ;
static zend_object_handlers date_object_handlers_timezone ;
# define DATE_SET_CONTEXT \
zval * object ; \
object = getThis ( ) ; \
# define DATE_FETCH_OBJECT \
php_date_obj * obj ; \
DATE_SET_CONTEXT ; \
if ( object ) { \
if ( ZEND_NUM_ARGS ( ) ) { \
WRONG_PARAM_COUNT ; \
} \
} else { \
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , NULL , " O " , & object , date_ce_date ) = = FAILURE ) { \
RETURN_FALSE ; \
} \
} \
obj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ; \
static zend_object_value date_object_new_date ( zend_class_entry * class_type TSRMLS_DC ) ;
static zend_object_value date_object_new_timezone ( zend_class_entry * class_type TSRMLS_DC ) ;
static void date_object_free_storage_date ( void * object TSRMLS_DC ) ;
static void date_object_free_storage_timezone ( void * object TSRMLS_DC ) ;
# endif
2005-07-05 05:27:26 +08:00
/* {{{ Module struct */
2005-06-15 05:32:29 +08:00
zend_module_entry date_module_entry = {
STANDARD_MODULE_HEADER ,
" date " , /* extension name */
date_functions , /* function list */
PHP_MINIT ( date ) , /* process startup */
PHP_MSHUTDOWN ( date ) , /* process shutdown */
2005-07-03 05:19:25 +08:00
PHP_RINIT ( date ) , /* request startup */
PHP_RSHUTDOWN ( date ) , /* request shutdown */
2005-06-15 05:32:29 +08:00
PHP_MINFO ( date ) , /* extension info */
PHP_VERSION , /* extension version */
STANDARD_MODULE_PROPERTIES
} ;
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-06-19 04:23:19 +08:00
/* {{{ php_date_init_globals */
static void php_date_init_globals ( zend_date_globals * date_globals )
{
date_globals - > default_timezone = NULL ;
2005-07-03 05:19:25 +08:00
date_globals - > timezone = NULL ;
2005-06-19 04:23:19 +08:00
}
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ PHP_RINIT_FUNCTION */
2005-07-03 05:19:25 +08:00
PHP_RINIT_FUNCTION ( date )
{
if ( DATEG ( timezone ) ) {
efree ( DATEG ( timezone ) ) ;
}
DATEG ( timezone ) = NULL ;
return SUCCESS ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-03 05:19:25 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ PHP_RSHUTDOWN_FUNCTION */
2005-07-03 05:19:25 +08:00
PHP_RSHUTDOWN_FUNCTION ( date )
{
if ( DATEG ( timezone ) ) {
efree ( DATEG ( timezone ) ) ;
}
DATEG ( timezone ) = NULL ;
return SUCCESS ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-20 16:31:02 +08:00
# define DATE_FORMAT_ISO8601 "Y-m-d\\TH:i:sO"
# define DATE_FORMAT_RFC1036 "l, d-M-y H:i:s T"
# define DATE_FORMAT_RFC1123 "D, d M Y H:i:s T"
# define DATE_FORMAT_RFC2822 "D, d M Y H:i:s O"
2005-07-05 05:27:26 +08:00
/* {{{ PHP_MINIT_FUNCTION */
2005-06-15 05:32:29 +08:00
PHP_MINIT_FUNCTION ( date )
{
2005-06-19 04:23:19 +08:00
ZEND_INIT_MODULE_GLOBALS ( date , php_date_init_globals , NULL ) ;
2005-06-15 05:32:29 +08:00
REGISTER_INI_ENTRIES ( ) ;
2005-07-20 16:31:02 +08:00
DATE_REGISTER_CLASSES ;
REGISTER_STRING_CONSTANT ( " DATE_ATOM " , DATE_FORMAT_ISO8601 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_COOKIE " , DATE_FORMAT_RFC1123 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_ISO8601 " , DATE_FORMAT_ISO8601 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RFC822 " , DATE_FORMAT_RFC1123 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RFC850 " , DATE_FORMAT_RFC1036 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RFC1036 " , DATE_FORMAT_RFC1036 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RFC1123 " , DATE_FORMAT_RFC1123 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RFC2822 " , DATE_FORMAT_RFC2822 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_RSS " , DATE_FORMAT_RFC1123 , CONST_CS | CONST_PERSISTENT ) ;
REGISTER_STRING_CONSTANT ( " DATE_W3C " , DATE_FORMAT_ISO8601 , CONST_CS | CONST_PERSISTENT ) ;
2005-06-15 05:32:29 +08:00
return SUCCESS ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ PHP_MSHUTDOWN_FUNCTION */
2005-06-15 05:32:29 +08:00
PHP_MSHUTDOWN_FUNCTION ( date )
{
UNREGISTER_INI_ENTRIES ( ) ;
return SUCCESS ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ PHP_MINFO_FUNCTION */
2005-06-15 05:32:29 +08:00
PHP_MINFO_FUNCTION ( date )
{
php_info_print_table_start ( ) ;
php_info_print_table_row ( 2 , " date/time support " , " enabled " ) ;
2005-08-10 05:10:22 +08:00
php_info_print_table_row ( 2 , " Default timezone " , guess_timezone ( TSRMLS_C ) ) ;
2005-06-15 05:32:29 +08:00
php_info_print_table_end ( ) ;
2005-08-10 05:10:22 +08:00
DISPLAY_INI_ENTRIES ( ) ;
2005-06-15 05:32:29 +08:00
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-15 05:32:29 +08:00
2005-07-01 05:38:06 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ Helper functions */
2005-07-01 05:38:06 +08:00
static char * guess_timezone ( TSRMLS_D )
{
char * env ;
2005-07-03 05:19:25 +08:00
/* Checking configure timezone */
if ( DATEG ( timezone ) & & ( strlen ( DATEG ( timezone ) ) > 0 ) ) {
return DATEG ( timezone ) ;
}
/* Check environment variable */
2005-07-01 05:38:06 +08:00
env = getenv ( " TZ " ) ;
2005-07-03 05:19:25 +08:00
if ( env & & * env ) {
2005-07-01 05:38:06 +08:00
return env ;
}
2005-07-03 05:19:25 +08:00
/* Check config setting for default timezone */
if ( DATEG ( default_timezone ) & & ( strlen ( DATEG ( default_timezone ) ) > 0 ) ) {
2005-07-01 05:38:06 +08:00
return DATEG ( default_timezone ) ;
}
2005-07-03 23:01:29 +08:00
# if HAVE_TM_ZONE
/* Try to guess timezone from system information */
{
struct tm * ta , tmbuf ;
time_t the_time ;
char * tzid ;
the_time = time ( NULL ) ;
ta = php_localtime_r ( & the_time , & tmbuf ) ;
tzid = timelib_timezone_id_from_abbr ( ta - > tm_zone ) ;
if ( ! tzid ) {
tzid = " UTC " ;
}
2005-07-20 16:31:02 +08:00
php_error_docref ( NULL TSRMLS_CC , E_STRICT , " It is not safe to rely on the systems timezone settings, please use the date.timezone setting, the TZ environment variable or the date_default_timezone_set() function. We use '%s' for '%s' instead. " , tzid , ta - > tm_zone ) ;
2005-07-03 23:01:29 +08:00
return tzid ;
}
# endif
2005-07-03 05:19:25 +08:00
/* Fallback to UTC */
return " UTC " ;
}
static timelib_tzinfo * get_timezone_info ( TSRMLS_D )
{
char * tz ;
timelib_tzinfo * tzi ;
tz = guess_timezone ( TSRMLS_C ) ;
tzi = timelib_parse_tzfile ( tz ) ;
if ( ! tzi ) {
2005-07-13 15:04:29 +08:00
php_error_docref ( NULL TSRMLS_CC , E_NOTICE , " Timezone setting (date.timezone) or TZ environment variable contains an unknown timezone. " ) ;
2005-07-03 05:19:25 +08:00
tzi = timelib_parse_tzfile ( " UTC " ) ;
if ( ! tzi ) {
php_error_docref ( NULL TSRMLS_CC , E_ERROR , " Timezone database is corrupt - this should *never* happen! " ) ;
}
}
return tzi ;
2005-07-01 05:38:06 +08:00
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-01 05:38:06 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ date() and gmdate() data */
2005-07-01 05:38:06 +08:00
# include "ext/standard/php_smart_str.h"
static char * mon_full_names [ ] = {
" January " , " February " , " March " , " April " ,
" May " , " June " , " July " , " August " ,
" September " , " October " , " November " , " December "
} ;
static char * mon_short_names [ ] = {
" Jan " , " Feb " , " Mar " , " Apr " , " May " , " Jun " , " Jul " , " Aug " , " Sep " , " Oct " , " Nov " , " Dec "
} ;
static char * day_full_names [ ] = {
" Sunday " , " Monday " , " Tuesday " , " Wednesday " , " Thursday " , " Friday " , " Saturday "
} ;
static char * day_short_names [ ] = {
" Sun " , " Mon " , " Tue " , " Wed " , " Thu " , " Fri " , " Sat "
} ;
2005-08-25 17:47:28 +08:00
static char * english_suffix ( timelib_sll number )
2005-07-01 05:38:06 +08:00
{
if ( number > = 10 & & number < = 19 ) {
return " th " ;
} else {
switch ( number % 10 ) {
case 1 : return " st " ;
case 2 : return " nd " ;
case 3 : return " rd " ;
}
}
return " th " ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-01 05:38:06 +08:00
2005-07-08 20:26:30 +08:00
/* {{{ date_format - (gm)date helper */
static char * date_format ( char * format , int format_len , timelib_time * t , int localtime )
2005-07-01 05:38:06 +08:00
{
smart_str string = { 0 } ;
int i ;
char buffer [ 33 ] ;
timelib_time_offset * offset ;
timelib_sll isoweek , isoyear ;
2005-08-09 00:49:30 +08:00
if ( ! format_len ) {
return estrdup ( " " ) ;
}
2005-07-01 05:38:06 +08:00
if ( localtime ) {
2005-07-20 16:31:02 +08:00
if ( t - > zone_type = = TIMELIB_ZONETYPE_ABBR ) {
offset = timelib_time_offset_ctor ( ) ;
offset - > offset = ( t - > z - ( t - > dst * 60 ) ) * - 60 ;
offset - > leap_secs = 0 ;
offset - > is_dst = t - > dst ;
offset - > abbr = strdup ( t - > tz_abbr ) ;
} else if ( t - > zone_type = = TIMELIB_ZONETYPE_OFFSET ) {
offset = timelib_time_offset_ctor ( ) ;
offset - > offset = ( t - > z - ( t - > dst * 60 ) ) * - 60 ;
offset - > leap_secs = 0 ;
offset - > is_dst = t - > dst ;
offset - > abbr = malloc ( 9 ) ; /* GMT<4D> xxxx\0 */
snprintf ( offset - > abbr , 9 , " GMT%c%02d%02d " ,
localtime ? ( ( offset - > offset < 0 ) ? ' - ' : ' + ' ) : ' + ' ,
localtime ? abs ( offset - > offset / 3600 ) : 0 ,
localtime ? abs ( ( offset - > offset % 3600 ) / 60 ) : 0 ) ;
} else {
offset = timelib_get_time_zone_info ( t - > sse , t - > tz_info ) ;
}
2005-07-01 05:38:06 +08:00
}
buffer [ 32 ] = ' \0 ' ;
timelib_isoweek_from_date ( t - > y , t - > m , t - > d , & isoweek , & isoyear ) ;
for ( i = 0 ; i < format_len ; i + + ) {
switch ( format [ i ] ) {
/* day */
2005-07-01 06:47:39 +08:00
case ' d ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > d ) ; break ;
case ' D ' : snprintf ( buffer , 32 , " %s " , day_short_names [ timelib_day_of_week ( t - > y , t - > m , t - > d ) ] ) ; break ;
case ' j ' : snprintf ( buffer , 32 , " %d " , ( int ) t - > d ) ; break ;
case ' l ' : snprintf ( buffer , 32 , " %s " , day_full_names [ timelib_day_of_week ( t - > y , t - > m , t - > d ) ] ) ; break ;
case ' S ' : snprintf ( buffer , 32 , " %s " , english_suffix ( t - > d ) ) ; break ;
case ' w ' : snprintf ( buffer , 32 , " %d " , ( int ) timelib_day_of_week ( t - > y , t - > m , t - > d ) ) ; break ;
2005-09-02 17:33:08 +08:00
case ' N ' : snprintf ( buffer , 32 , " %d " , ( int ) timelib_iso_day_of_week ( t - > y , t - > m , t - > d ) ) ; break ;
2005-07-01 06:47:39 +08:00
case ' z ' : snprintf ( buffer , 32 , " %d " , ( int ) timelib_day_of_year ( t - > y , t - > m , t - > d ) ) ; break ;
2005-07-01 05:38:06 +08:00
/* week */
2005-08-30 17:17:09 +08:00
case ' W ' : snprintf ( buffer , 32 , " %02d " , ( int ) isoweek ) ; break ; /* iso weeknr */
2005-07-01 06:47:39 +08:00
case ' o ' : snprintf ( buffer , 32 , " %d " , ( int ) isoyear ) ; break ; /* iso year */
2005-07-01 05:38:06 +08:00
/* month */
2005-07-01 06:47:39 +08:00
case ' F ' : snprintf ( buffer , 32 , " %s " , mon_full_names [ t - > m - 1 ] ) ; break ;
case ' m ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > m ) ; break ;
case ' M ' : snprintf ( buffer , 32 , " %s " , mon_short_names [ t - > m - 1 ] ) ; break ;
case ' n ' : snprintf ( buffer , 32 , " %d " , ( int ) t - > m ) ; break ;
case ' t ' : snprintf ( buffer , 32 , " %d " , ( int ) timelib_days_in_month ( t - > y , t - > m ) ) ; break ;
2005-07-01 05:38:06 +08:00
/* year */
2005-07-01 06:47:39 +08:00
case ' L ' : snprintf ( buffer , 32 , " %d " , timelib_is_leap ( ( int ) t - > y ) ) ; break ;
case ' y ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > y % 100 ) ; break ;
case ' Y ' : snprintf ( buffer , 32 , " %04d " , ( int ) t - > y ) ; break ;
2005-07-01 05:38:06 +08:00
/* time */
2005-07-01 06:47:39 +08:00
case ' a ' : snprintf ( buffer , 32 , " %s " , t - > h > = 12 ? " pm " : " am " ) ; break ;
case ' A ' : snprintf ( buffer , 32 , " %s " , t - > h > = 12 ? " PM " : " AM " ) ; break ;
case ' B ' : snprintf ( buffer , 32 , " [B unimplemented] " ) ; break ;
case ' g ' : snprintf ( buffer , 32 , " %d " , ( t - > h % 12 ) ? ( int ) t - > h % 12 : 12 ) ; break ;
case ' G ' : snprintf ( buffer , 32 , " %d " , ( int ) t - > h ) ; break ;
case ' h ' : snprintf ( buffer , 32 , " %02d " , ( t - > h % 12 ) ? ( int ) t - > h % 12 : 12 ) ; break ;
case ' H ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > h ) ; break ;
case ' i ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > i ) ; break ;
case ' s ' : snprintf ( buffer , 32 , " %02d " , ( int ) t - > s ) ; break ;
2005-07-01 05:38:06 +08:00
/* timezone */
2005-07-01 06:47:39 +08:00
case ' I ' : snprintf ( buffer , 32 , " %d " , localtime ? offset - > is_dst : 0 ) ; break ;
case ' O ' : snprintf ( buffer , 32 , " %c%02d%02d " ,
2005-07-01 05:38:06 +08:00
localtime ? ( ( offset - > offset < 0 ) ? ' - ' : ' + ' ) : ' + ' ,
localtime ? abs ( offset - > offset / 3600 ) : 0 ,
localtime ? abs ( ( offset - > offset % 3600 ) / 60 ) : 0
) ;
break ;
2005-07-01 06:47:39 +08:00
case ' T ' : snprintf ( buffer , 32 , " %s " , localtime ? offset - > abbr : " GMT " ) ; break ;
case ' e ' : snprintf ( buffer , 32 , " %s " , localtime ? t - > tz_info - > name : " UTC " ) ; break ;
case ' Z ' : snprintf ( buffer , 32 , " %d " , localtime ? offset - > offset : 0 ) ; break ;
2005-07-01 05:38:06 +08:00
/* full date/time */
2005-07-01 06:47:39 +08:00
case ' c ' : snprintf ( buffer , 32 , " %04d-%02d-%02dT%02d:%02d:%02d%c%02d:%02d " ,
2005-07-01 05:38:06 +08:00
( int ) t - > y , ( int ) t - > m , ( int ) t - > d ,
( int ) t - > h , ( int ) t - > i , ( int ) t - > s ,
localtime ? ( ( offset - > offset < 0 ) ? ' - ' : ' + ' ) : ' + ' ,
localtime ? abs ( offset - > offset / 3600 ) : 0 ,
localtime ? abs ( ( offset - > offset % 3600 ) / 60 ) : 0
) ;
break ;
2005-07-01 06:47:39 +08:00
case ' r ' : snprintf ( buffer , 32 , " %3s, %02d %3s %04d %02d:%02d:%02d %c%02d%02d " ,
2005-07-01 05:38:06 +08:00
day_short_names [ timelib_day_of_week ( t - > y , t - > m , t - > d ) ] ,
( int ) t - > d , mon_short_names [ t - > m - 1 ] ,
( int ) t - > y , ( int ) t - > h , ( int ) t - > i , ( int ) t - > s ,
localtime ? ( ( offset - > offset < 0 ) ? ' - ' : ' + ' ) : ' + ' ,
localtime ? abs ( offset - > offset / 3600 ) : 0 ,
localtime ? abs ( ( offset - > offset % 3600 ) / 60 ) : 0
) ;
break ;
2005-07-01 16:59:57 +08:00
case ' U ' : snprintf ( buffer , 32 , " %lld " , ( timelib_sll ) t - > sse ) ; break ;
2005-07-01 05:38:06 +08:00
case ' \\ ' : if ( i < format_len ) i + + ; buffer [ 0 ] = format [ i ] ; buffer [ 1 ] = ' \0 ' ; break ;
default : buffer [ 0 ] = format [ i ] ; buffer [ 1 ] = ' \0 ' ;
}
smart_str_appends ( & string , buffer ) ;
buffer [ 0 ] = ' \0 ' ;
}
smart_str_0 ( & string ) ;
2005-07-01 07:33:37 +08:00
if ( localtime ) {
timelib_time_offset_dtor ( offset ) ;
}
2005-07-01 05:38:06 +08:00
return string . c ;
}
static void php_date ( INTERNAL_FUNCTION_PARAMETERS , int localtime )
{
char * format ;
int format_len ;
2005-08-25 17:47:28 +08:00
time_t ts = time ( NULL ) ;
2005-07-01 05:38:06 +08:00
char * string ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " s|l " , & format , & format_len , & ts ) = = FAILURE ) {
RETURN_FALSE ;
}
2005-07-09 02:16:46 +08:00
string = php_format_date ( format , format_len , ts , localtime TSRMLS_CC ) ;
2005-07-08 20:26:30 +08:00
RETVAL_STRING ( string , 0 ) ;
}
/* }}} */
2005-08-25 17:47:28 +08:00
PHPAPI char * php_format_date ( char * format , int format_len , time_t ts , int localtime TSRMLS_DC ) /* { { { */
2005-07-08 20:26:30 +08:00
{
timelib_time * t ;
timelib_tzinfo * tzi ;
char * string ;
2005-07-01 06:44:28 +08:00
t = timelib_time_ctor ( ) ;
2005-07-01 05:38:06 +08:00
if ( localtime ) {
2005-07-03 05:19:25 +08:00
tzi = get_timezone_info ( TSRMLS_C ) ;
2005-07-01 06:44:28 +08:00
timelib_unixtime2local ( t , ts , tzi ) ;
2005-07-01 05:38:06 +08:00
} else {
tzi = NULL ;
timelib_unixtime2gmt ( t , ts ) ;
}
2005-07-08 20:26:30 +08:00
string = date_format ( format , format_len , t , localtime ) ;
2005-07-01 05:38:06 +08:00
2005-07-06 02:30:30 +08:00
if ( localtime ) {
timelib_tzinfo_dtor ( tzi ) ;
}
2005-07-08 20:26:30 +08:00
2005-07-01 05:38:06 +08:00
timelib_time_dtor ( t ) ;
2005-07-08 20:26:30 +08:00
return string ;
2005-07-01 05:38:06 +08:00
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-01 05:38:06 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ proto string date(string format [, long timestamp])
Format a local date / time */
2005-07-01 05:38:06 +08:00
PHP_FUNCTION ( date )
{
php_date ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 1 ) ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-01 05:38:06 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ proto string gmdate(string format [, long timestamp])
Format a GMT date / time */
2005-07-01 05:38:06 +08:00
PHP_FUNCTION ( gmdate )
{
php_date ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 0 ) ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-01 05:38:06 +08:00
2005-07-05 05:27:26 +08:00
/* {{{ php_parse_date: Backwards compability function */
2005-06-20 06:15:27 +08:00
signed long php_parse_date ( char * string , signed long * now )
{
timelib_time * parsed_time ;
2005-07-04 05:38:54 +08:00
int error1 , error2 ;
2005-06-20 06:15:27 +08:00
signed long retval ;
2005-07-04 05:38:54 +08:00
parsed_time = timelib_strtotime ( string , & error1 ) ;
2005-06-20 06:15:27 +08:00
timelib_update_ts ( parsed_time , NULL ) ;
2005-07-04 05:38:54 +08:00
retval = timelib_date_to_int ( parsed_time , & error2 ) ;
2005-06-20 06:15:27 +08:00
timelib_time_dtor ( parsed_time ) ;
2005-07-04 05:38:54 +08:00
if ( error1 | | error2 ) {
2005-06-20 06:15:27 +08:00
return - 1 ;
}
return retval ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-06-20 06:15:27 +08:00
2005-06-15 05:32:29 +08:00
/* {{{ proto int strtotime(string time, int now)
Convert string representation of date and time to a timestamp */
PHP_FUNCTION ( strtotime )
{
char * times , * initial_ts ;
2005-07-04 05:38:54 +08:00
int time_len , error1 , error2 ;
2005-06-15 05:32:29 +08:00
long preset_ts , ts ;
timelib_time * t , * now ;
timelib_tzinfo * tzi ;
2005-07-03 05:19:25 +08:00
tzi = get_timezone_info ( TSRMLS_C ) ;
2005-06-15 05:32:29 +08:00
if ( zend_parse_parameters_ex ( ZEND_PARSE_PARAMS_QUIET , ZEND_NUM_ARGS ( ) TSRMLS_CC , " sl " , & times , & time_len , & preset_ts ) ! = FAILURE ) {
/* We have an initial timestamp */
now = timelib_time_ctor ( ) ;
initial_ts = emalloc ( 25 ) ;
snprintf ( initial_ts , 24 , " @%lu " , preset_ts ) ;
2005-07-04 05:38:54 +08:00
t = timelib_strtotime ( initial_ts , & error1 ) ; /* we ignore the error here, as this should never fail */
2005-06-15 05:32:29 +08:00
timelib_update_ts ( t , tzi ) ;
2005-06-17 22:54:00 +08:00
timelib_unixtime2local ( now , t - > sse , tzi ) ;
2005-06-15 05:32:29 +08:00
timelib_time_dtor ( t ) ;
efree ( initial_ts ) ;
} else if ( zend_parse_parameters_ex ( ZEND_PARSE_PARAMS_QUIET , ZEND_NUM_ARGS ( ) TSRMLS_CC , " s " , & times , & time_len ) ! = FAILURE ) {
/* We have no initial timestamp */
now = timelib_time_ctor ( ) ;
2005-06-15 08:11:29 +08:00
timelib_unixtime2local ( now , ( timelib_sll ) time ( NULL ) , tzi ) ;
2005-06-15 05:32:29 +08:00
} else {
2005-06-15 15:23:27 +08:00
timelib_tzinfo_dtor ( tzi ) ;
2005-06-15 05:32:29 +08:00
RETURN_FALSE ;
}
2005-07-04 05:38:54 +08:00
t = timelib_strtotime ( times , & error1 ) ;
2005-06-15 05:32:29 +08:00
timelib_fill_holes ( t , now , 0 ) ;
timelib_update_ts ( t , tzi ) ;
2005-07-04 05:38:54 +08:00
ts = timelib_date_to_int ( t , & error2 ) ;
2005-06-15 07:40:57 +08:00
/* if tz_info is not a copy, avoid double free */
2005-07-06 02:30:30 +08:00
if ( now - > tz_info ! = tzi ) {
timelib_tzinfo_dtor ( now - > tz_info ) ;
2005-06-15 07:40:57 +08:00
}
2005-07-06 02:30:30 +08:00
if ( t - > tz_info ! = tzi ) {
timelib_tzinfo_dtor ( t - > tz_info ) ;
2005-06-16 07:30:20 +08:00
}
2005-06-15 07:40:57 +08:00
2005-07-06 02:30:30 +08:00
timelib_tzinfo_dtor ( tzi ) ;
2005-06-15 07:40:57 +08:00
timelib_time_dtor ( now ) ;
timelib_time_dtor ( t ) ;
2005-07-04 05:38:54 +08:00
if ( error1 | | error2 ) {
2005-06-15 05:32:29 +08:00
RETURN_FALSE ;
} else {
RETURN_LONG ( ts ) ;
}
}
/* }}} */
2005-07-05 05:27:26 +08:00
/* {{{ php_mktime - (gm)mktime helper */
2005-07-04 21:21:39 +08:00
PHPAPI void php_mktime ( INTERNAL_FUNCTION_PARAMETERS , int gmt )
2005-07-03 22:27:31 +08:00
{
2005-07-04 21:22:27 +08:00
long hou , min , sec , mon , day , yea , dst = - 1 ;
2005-07-03 22:27:31 +08:00
timelib_time * now ;
timelib_tzinfo * tzi ;
long ts , adjust_seconds = 0 ;
int error ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " |lllllll " , & hou , & min , & sec , & mon , & day , & yea , & dst ) = = FAILURE ) {
RETURN_FALSE ;
}
/* Initialize structure with current time */
now = timelib_time_ctor ( ) ;
if ( gmt ) {
timelib_unixtime2gmt ( now , ( timelib_sll ) time ( NULL ) ) ;
} else {
tzi = get_timezone_info ( TSRMLS_C ) ;
timelib_unixtime2local ( now , ( timelib_sll ) time ( NULL ) , tzi ) ;
}
/* Fill in the new data */
switch ( ZEND_NUM_ARGS ( ) ) {
case 7 :
/* break intentionally missing */
case 6 :
2005-07-21 18:09:40 +08:00
if ( yea > = 0 & & yea < 70 ) {
yea + = 2000 ;
} else if ( yea > = 70 & & yea < = 100 ) {
yea + = 1900 ;
}
2005-07-03 22:27:31 +08:00
now - > y = yea ;
/* break intentionally missing again */
case 5 :
now - > d = day ;
/* break missing intentionally here too */
case 4 :
now - > m = mon ;
/* and here */
case 3 :
now - > s = sec ;
/* yup, this break isn't here on purpose too */
case 2 :
now - > i = min ;
/* last intentionally missing break */
case 1 :
now - > h = hou ;
break ;
default :
php_error_docref ( NULL TSRMLS_CC , E_STRICT , " You should be using the time() function instead. " ) ;
}
/* Update the timestamp */
if ( gmt ) {
timelib_update_ts ( now , NULL ) ;
} else {
timelib_update_ts ( now , tzi ) ;
}
/* Support for the deprecated is_dst parameter */
if ( dst ! = - 1 ) {
php_error_docref ( NULL TSRMLS_CC , E_STRICT , " The is_dst parameter is deprecated. " ) ;
if ( gmt ) {
/* GMT never uses DST */
if ( dst = = 1 ) {
adjust_seconds = - 3600 ;
}
} else {
/* Figure out is_dst for current TS */
timelib_time_offset * tmp_offset ;
tmp_offset = timelib_get_time_zone_info ( now - > sse , tzi ) ;
if ( dst = = 1 & & tmp_offset - > is_dst = = 0 ) {
adjust_seconds = - 3600 ;
}
if ( dst = = 0 & & tmp_offset - > is_dst = = 1 ) {
adjust_seconds = + 3600 ;
}
2005-07-03 22:36:59 +08:00
timelib_time_offset_dtor ( tmp_offset ) ;
2005-07-03 22:27:31 +08:00
}
}
/* Clean up and return */
ts = timelib_date_to_int ( now , & error ) ;
ts + = adjust_seconds ;
timelib_time_dtor ( now ) ;
2005-07-06 02:30:30 +08:00
if ( ! gmt ) {
timelib_tzinfo_dtor ( tzi ) ;
}
2005-07-03 22:27:31 +08:00
if ( error ) {
RETURN_FALSE ;
} else {
RETURN_LONG ( ts ) ;
}
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-03 22:27:31 +08:00
/* {{{ proto int mktime(int hour, int min, int sec, int mon, int day, int year)
Get UNIX timestamp for a date */
PHP_FUNCTION ( mktime )
{
php_mktime ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 0 ) ;
}
/* }}} */
/* {{{ proto int gmmktime(int hour, int min, int sec, int mon, int day, int year)
Get UNIX timestamp for a GMT date */
PHP_FUNCTION ( gmmktime )
{
php_mktime ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 1 ) ;
}
/* }}} */
2005-07-05 05:27:26 +08:00
2005-07-04 03:14:55 +08:00
/* {{{ proto bool checkdate(int month, int day, int year)
Returns true ( 1 ) if it is a valid date in gregorian calendar */
PHP_FUNCTION ( checkdate )
{
long m , d , y ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " lll " , & m , & d , & y ) = = FAILURE ) {
RETURN_FALSE ;
}
if ( y < 1 | | y > 32767 | | m < 1 | | m > 12 | | d < 1 | | d > timelib_days_in_month ( y , m ) ) {
RETURN_FALSE ;
}
RETURN_TRUE ; /* True : This month, day, year arguments are valid */
}
/* }}} */
2005-07-04 07:30:52 +08:00
# ifdef HAVE_STRFTIME
2005-07-05 05:27:26 +08:00
/* {{{ php_strftime - (gm)strftime helper */
2005-07-04 03:14:55 +08:00
PHPAPI void php_strftime ( INTERNAL_FUNCTION_PARAMETERS , int gmt )
{
char * format , * buf ;
int format_len ;
long timestamp ;
struct tm ta ;
int max_reallocs = 5 ;
size_t buf_len = 64 , real_len ;
timelib_time * ts ;
timelib_tzinfo * tzi ;
timelib_time_offset * offset ;
timestamp = ( long ) time ( NULL ) ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " s|l " , & format , & format_len , & timestamp ) = = FAILURE ) {
RETURN_FALSE ;
}
if ( format_len = = 0 ) {
RETURN_FALSE ;
}
ts = timelib_time_ctor ( ) ;
if ( gmt ) {
tzi = NULL ;
timelib_unixtime2gmt ( ts , ( timelib_sll ) timestamp ) ;
} else {
tzi = get_timezone_info ( TSRMLS_C ) ;
timelib_unixtime2local ( ts , ( timelib_sll ) timestamp , tzi ) ;
}
ta . tm_sec = ts - > s ;
ta . tm_min = ts - > i ;
ta . tm_hour = ts - > h ;
ta . tm_mday = ts - > d ;
ta . tm_mon = ts - > m - 1 ;
ta . tm_year = ts - > y - 1900 ;
ta . tm_wday = timelib_day_of_week ( ts - > y , ts - > m , ts - > d ) ;
ta . tm_yday = timelib_day_of_year ( ts - > y , ts - > m , ts - > d ) ;
if ( gmt ) {
ta . tm_isdst = 0 ;
# if HAVE_TM_GMTOFF
ta . tm_gmtoff = 0 ;
# endif
# if HAVE_TM_ZONE
ta . tm_zone = " GMT " ;
# endif
} else {
offset = timelib_get_time_zone_info ( timestamp , tzi ) ;
ta . tm_isdst = offset - > is_dst ;
# if HAVE_TM_GMTOFF
ta . tm_gmtoff = offset - > offset ;
# endif
# if HAVE_TM_ZONE
ta . tm_zone = offset - > abbr ;
# endif
}
2005-07-08 18:23:33 +08:00
if ( ! gmt ) {
timelib_tzinfo_dtor ( tzi ) ;
}
2005-07-04 03:14:55 +08:00
buf = ( char * ) emalloc ( buf_len ) ;
while ( ( real_len = strftime ( buf , buf_len , format , & ta ) ) = = buf_len | | real_len = = 0 ) {
buf_len * = 2 ;
buf = ( char * ) erealloc ( buf , buf_len ) ;
if ( ! - - max_reallocs ) {
break ;
}
}
timelib_time_dtor ( ts ) ;
2005-07-04 22:32:50 +08:00
if ( ! gmt ) {
2005-07-04 03:14:55 +08:00
timelib_time_offset_dtor ( offset ) ;
}
if ( real_len & & real_len ! = buf_len ) {
buf = ( char * ) erealloc ( buf , real_len + 1 ) ;
RETURN_STRINGL ( buf , real_len , 0 ) ;
}
efree ( buf ) ;
RETURN_FALSE ;
}
/* }}} */
/* {{{ proto string strftime(string format [, int timestamp])
Format a local time / date according to locale settings */
PHP_FUNCTION ( strftime )
{
2005-07-05 05:27:26 +08:00
php_strftime ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 0 ) ;
2005-07-04 03:14:55 +08:00
}
/* }}} */
/* {{{ proto string gmstrftime(string format [, int timestamp])
Format a GMT / UCT time / date according to locale settings */
PHP_FUNCTION ( gmstrftime )
{
2005-07-05 05:27:26 +08:00
php_strftime ( INTERNAL_FUNCTION_PARAM_PASSTHRU , 1 ) ;
2005-07-04 03:14:55 +08:00
}
/* }}} */
# endif
2005-07-04 04:45:08 +08:00
/* {{{ proto int time(void)
Return current UNIX timestamp */
PHP_FUNCTION ( time )
{
RETURN_LONG ( ( long ) time ( NULL ) ) ;
}
/* }}} */
/* {{{ proto array localtime([int timestamp [, bool associative_array]])
Returns the results of the C system call localtime as an associative array if the associative_array argument is set to 1 other wise it is a regular array */
PHP_FUNCTION ( localtime )
{
long timestamp = ( long ) time ( NULL ) ;
int associative = 0 ;
timelib_tzinfo * tzi ;
timelib_time * ts ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " |lb " , & timestamp , & associative ) = = FAILURE ) {
RETURN_FALSE ;
}
tzi = get_timezone_info ( TSRMLS_C ) ;
ts = timelib_time_ctor ( ) ;
timelib_unixtime2local ( ts , ( timelib_sll ) timestamp , tzi ) ;
array_init ( return_value ) ;
if ( associative ) {
add_assoc_long ( return_value , " tm_sec " , ts - > s ) ;
add_assoc_long ( return_value , " tm_min " , ts - > i ) ;
add_assoc_long ( return_value , " tm_hour " , ts - > h ) ;
add_assoc_long ( return_value , " tm_mday " , ts - > d ) ;
add_assoc_long ( return_value , " tm_mon " , ts - > m - 1 ) ;
add_assoc_long ( return_value , " tm_year " , ts - > y - 1900 ) ;
add_assoc_long ( return_value , " tm_wday " , timelib_day_of_week ( ts - > y , ts - > m , ts - > d ) ) ;
add_assoc_long ( return_value , " tm_yday " , timelib_day_of_year ( ts - > y , ts - > m , ts - > d ) ) ;
add_assoc_long ( return_value , " tm_isdst " , ts - > dst ) ;
} else {
add_next_index_long ( return_value , ts - > s ) ;
add_next_index_long ( return_value , ts - > i ) ;
add_next_index_long ( return_value , ts - > h ) ;
add_next_index_long ( return_value , ts - > d ) ;
add_next_index_long ( return_value , ts - > m - 1 ) ;
add_next_index_long ( return_value , ts - > y - 1900 ) ;
add_next_index_long ( return_value , timelib_day_of_week ( ts - > y , ts - > m , ts - > d ) ) ;
add_next_index_long ( return_value , timelib_day_of_year ( ts - > y , ts - > m , ts - > d ) ) ;
add_next_index_long ( return_value , ts - > dst ) ;
}
2005-07-06 02:30:30 +08:00
timelib_tzinfo_dtor ( tzi ) ;
2005-07-04 04:45:08 +08:00
timelib_time_dtor ( ts ) ;
}
/* }}} */
/* {{{ proto array getdate([int timestamp])
Get date / time information */
PHP_FUNCTION ( getdate )
{
long timestamp = ( long ) time ( NULL ) ;
timelib_tzinfo * tzi ;
timelib_time * ts ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " |l " , & timestamp ) = = FAILURE ) {
RETURN_FALSE ;
}
tzi = get_timezone_info ( TSRMLS_C ) ;
ts = timelib_time_ctor ( ) ;
timelib_unixtime2local ( ts , ( timelib_sll ) timestamp , tzi ) ;
array_init ( return_value ) ;
add_assoc_long ( return_value , " seconds " , ts - > s ) ;
add_assoc_long ( return_value , " minutes " , ts - > i ) ;
add_assoc_long ( return_value , " hours " , ts - > h ) ;
add_assoc_long ( return_value , " mday " , ts - > d ) ;
add_assoc_long ( return_value , " wday " , timelib_day_of_week ( ts - > y , ts - > m , ts - > d ) ) ;
add_assoc_long ( return_value , " mon " , ts - > m ) ;
add_assoc_long ( return_value , " year " , ts - > y ) ;
add_assoc_long ( return_value , " yday " , timelib_day_of_year ( ts - > y , ts - > m , ts - > d ) ) ;
add_assoc_string ( return_value , " weekday " , day_full_names [ timelib_day_of_week ( ts - > y , ts - > m , ts - > d ) ] , 1 ) ;
add_assoc_string ( return_value , " month " , mon_full_names [ ts - > m - 1 ] , 1 ) ;
add_index_long ( return_value , 0 , timestamp ) ;
2005-07-06 02:30:30 +08:00
timelib_tzinfo_dtor ( tzi ) ;
2005-07-04 04:45:08 +08:00
timelib_time_dtor ( ts ) ;
}
/* }}} */
2005-07-20 16:31:02 +08:00
# ifdef EXPERIMENTAL_DATE_SUPPORT
static zend_object_value date_object_new_date ( zend_class_entry * class_type TSRMLS_DC )
{
php_date_obj * intern ;
zend_object_value retval ;
intern = emalloc ( sizeof ( php_date_obj ) ) ;
memset ( intern , 0 , sizeof ( php_date_obj ) ) ;
intern - > std . ce = class_type ;
retval . handle = zend_objects_store_put ( intern , NULL , ( zend_objects_free_object_storage_t ) date_object_free_storage_date , NULL TSRMLS_CC ) ;
retval . handlers = & date_object_handlers_date ;
return retval ;
}
static zend_object_value date_object_new_timezone ( zend_class_entry * class_type TSRMLS_DC )
{
php_timezone_obj * intern ;
zend_object_value retval ;
intern = emalloc ( sizeof ( php_timezone_obj ) ) ;
memset ( intern , 0 , sizeof ( php_timezone_obj ) ) ;
intern - > std . ce = class_type ;
retval . handle = zend_objects_store_put ( intern , NULL , ( zend_objects_free_object_storage_t ) date_object_free_storage_timezone , NULL TSRMLS_CC ) ;
retval . handlers = & date_object_handlers_timezone ;
return retval ;
}
static void date_object_free_storage_date ( void * object TSRMLS_DC )
{
php_date_obj * intern = ( php_date_obj * ) object ;
if ( intern - > time - > tz_info ) {
timelib_tzinfo_dtor ( intern - > time - > tz_info ) ;
}
timelib_time_dtor ( intern - > time ) ;
efree ( object ) ;
}
static void date_object_free_storage_timezone ( void * object TSRMLS_DC )
{
php_timezone_obj * intern = ( php_timezone_obj * ) object ;
timelib_tzinfo_dtor ( intern - > tz ) ;
efree ( object ) ;
}
2005-08-10 05:10:22 +08:00
static void date_register_classes ( TSRMLS_D )
2005-07-20 16:31:02 +08:00
{
zend_class_entry ce_date , ce_timezone ;
INIT_CLASS_ENTRY ( ce_date , " date " , date_funcs_date ) ;
ce_date . create_object = date_object_new_date ;
date_ce_date = zend_register_internal_class_ex ( & ce_date , NULL , NULL TSRMLS_CC ) ;
memcpy ( & date_object_handlers_date , zend_get_std_object_handlers ( ) , sizeof ( zend_object_handlers ) ) ;
date_object_handlers_date . clone_obj = NULL ;
INIT_CLASS_ENTRY ( ce_timezone , " timezone " , date_funcs_timezone ) ;
ce_timezone . create_object = date_object_new_timezone ;
date_ce_timezone = zend_register_internal_class_ex ( & ce_timezone , NULL , NULL TSRMLS_CC ) ;
memcpy ( & date_object_handlers_timezone , zend_get_std_object_handlers ( ) , sizeof ( zend_object_handlers ) ) ;
date_object_handlers_timezone . clone_obj = NULL ;
}
/* Advanced Interface */
static zval * date_instanciate ( zend_class_entry * pce , zval * object TSRMLS_DC )
{
if ( ! object ) {
ALLOC_ZVAL ( object ) ;
}
Z_TYPE_P ( object ) = IS_OBJECT ;
object_init_ex ( object , pce ) ;
object - > refcount = 1 ;
object - > is_ref = 1 ;
return object ;
}
PHP_FUNCTION ( date_create )
{
php_date_obj * dateobj ;
zval * timezone_object = NULL ;
int error ;
timelib_time * now ;
timelib_tzinfo * tzi ;
char * time_str ;
int time_str_len = 0 ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " |sO " , & time_str , & time_str_len , & timezone_object , date_ce_timezone ) = = FAILURE ) {
RETURN_FALSE ;
}
date_instanciate ( date_ce_date , return_value TSRMLS_CC ) ;
dateobj = ( php_date_obj * ) zend_object_store_get_object ( return_value TSRMLS_CC ) ;
dateobj - > time = timelib_strtotime ( time_str_len ? time_str : " now " , & error ) ;
if ( timezone_object ) {
php_timezone_obj * tzobj ;
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( timezone_object TSRMLS_CC ) ;
tzi = timelib_tzinfo_clone ( tzobj - > tz ) ;
} else if ( dateobj - > time - > tz_info ) {
tzi = timelib_tzinfo_clone ( dateobj - > time - > tz_info ) ;
} else {
tzi = get_timezone_info ( TSRMLS_C ) ;
}
now = timelib_time_ctor ( ) ;
timelib_unixtime2local ( now , ( timelib_sll ) time ( NULL ) , tzi ) ;
timelib_fill_holes ( dateobj - > time , now , 0 ) ;
timelib_update_ts ( dateobj - > time , tzi ) ;
if ( now - > tz_info ! = tzi ) {
timelib_tzinfo_dtor ( now - > tz_info ) ;
}
timelib_tzinfo_dtor ( now - > tz_info ) ;
timelib_time_dtor ( now ) ;
}
PHP_FUNCTION ( date_format )
{
zval * object ;
php_date_obj * dateobj ;
char * format ;
int format_len ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " Os " , & object , date_ce_date , & format , & format_len ) = = FAILURE ) {
RETURN_FALSE ;
}
dateobj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
RETURN_STRING ( date_format ( format , format_len , dateobj - > time , dateobj - > time - > is_localtime ) , 0 ) ;
}
PHP_FUNCTION ( date_modify )
{
zval * object ;
php_date_obj * dateobj ;
char * modify ;
int modify_len ;
int error ;
timelib_time * tmp_time ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " Os " , & object , date_ce_date , & modify , & modify_len ) = = FAILURE ) {
RETURN_FALSE ;
}
dateobj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
tmp_time = timelib_strtotime ( modify , & error ) ;
dateobj - > time - > relative . y = tmp_time - > relative . y ;
dateobj - > time - > relative . m = tmp_time - > relative . m ;
dateobj - > time - > relative . d = tmp_time - > relative . d ;
dateobj - > time - > relative . h = tmp_time - > relative . h ;
dateobj - > time - > relative . i = tmp_time - > relative . i ;
dateobj - > time - > relative . s = tmp_time - > relative . s ;
dateobj - > time - > relative . weekday = tmp_time - > relative . weekday ;
dateobj - > time - > have_relative = tmp_time - > have_relative ;
dateobj - > time - > have_weekday_relative = tmp_time - > have_weekday_relative ;
dateobj - > time - > sse_uptodate = 0 ;
timelib_time_dtor ( tmp_time ) ;
timelib_update_ts ( dateobj - > time , NULL ) ;
timelib_update_from_sse ( dateobj - > time ) ;
}
PHP_FUNCTION ( date_timezone_get )
{
zval * object ;
php_date_obj * dateobj ;
php_timezone_obj * tzobj ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " O " , & object , date_ce_date ) = = FAILURE ) {
RETURN_FALSE ;
}
dateobj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
if ( dateobj - > time - > is_localtime & & dateobj - > time - > tz_info ) {
date_instanciate ( date_ce_timezone , return_value TSRMLS_CC ) ;
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( return_value TSRMLS_CC ) ;
tzobj - > tz = timelib_tzinfo_clone ( dateobj - > time - > tz_info ) ;
} else {
RETURN_FALSE ;
}
}
PHP_FUNCTION ( date_timezone_set )
{
zval * object ;
zval * timezone_object ;
php_date_obj * dateobj ;
php_timezone_obj * tzobj ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " O|O " , & object , date_ce_date , & timezone_object , date_ce_timezone ) = = FAILURE ) {
RETURN_FALSE ;
}
dateobj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( timezone_object TSRMLS_CC ) ;
if ( dateobj - > time - > tz_info ) {
timelib_tzinfo_dtor ( dateobj - > time - > tz_info ) ;
}
timelib_set_timezone ( dateobj - > time , timelib_tzinfo_clone ( tzobj - > tz ) ) ;
timelib_unixtime2local ( dateobj - > time , dateobj - > time - > sse , dateobj - > time - > tz_info ) ;
}
2005-07-05 05:27:26 +08:00
2005-07-20 16:31:02 +08:00
PHP_FUNCTION ( date_offset_get )
{
zval * object ;
php_date_obj * dateobj ;
timelib_time_offset * offset ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " O " , & object , date_ce_date ) = = FAILURE ) {
RETURN_FALSE ;
}
dateobj = ( php_date_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
if ( dateobj - > time - > is_localtime & & dateobj - > time - > tz_info ) {
offset = timelib_get_time_zone_info ( dateobj - > time - > sse , dateobj - > time - > tz_info ) ;
RETVAL_LONG ( offset - > offset ) ;
timelib_time_offset_dtor ( offset ) ;
return ;
} else {
RETURN_LONG ( 0 ) ;
}
}
PHP_FUNCTION ( timezone_open )
{
php_timezone_obj * tzobj ;
char * tz ;
int tz_len ;
timelib_tzinfo * tzi = NULL ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " s " , & tz , & tz_len ) = = FAILURE ) {
RETURN_FALSE ;
}
/* Try finding the tz information as "Timezone Abbreviation" */
if ( ! tzi ) {
char * tzid ;
tzid = timelib_timezone_id_from_abbr ( tz ) ;
if ( tzid ) {
tzi = timelib_parse_tzfile ( tzid ) ;
}
}
/* Try finding the tz information as "Timezone Identifier" */
if ( ! tzi ) {
tzi = timelib_parse_tzfile ( tz ) ;
}
/* If we find it we instantiate the object otherwise, well, we don't and return false */
if ( tzi ) {
date_instanciate ( date_ce_timezone , return_value TSRMLS_CC ) ;
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( return_value TSRMLS_CC ) ;
tzobj - > tz = tzi ;
} else {
RETURN_FALSE ;
}
}
PHP_FUNCTION ( timezone_name_get )
{
zval * object ;
php_timezone_obj * tzobj ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " O " , & object , date_ce_timezone ) = = FAILURE ) {
RETURN_FALSE ;
}
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
RETURN_STRING ( tzobj - > tz - > name , 1 ) ;
}
PHP_FUNCTION ( timezone_offset_get )
{
zval * object , * dateobject ;
php_timezone_obj * tzobj ;
php_date_obj * dateobj ;
timelib_time_offset * offset ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " OO " , & object , date_ce_timezone , & dateobject , date_ce_date ) = = FAILURE ) {
RETURN_FALSE ;
}
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
dateobj = ( php_date_obj * ) zend_object_store_get_object ( dateobject TSRMLS_CC ) ;
offset = timelib_get_time_zone_info ( dateobj - > time - > sse , tzobj - > tz ) ;
RETVAL_LONG ( offset - > offset ) ;
timelib_time_offset_dtor ( offset ) ;
}
PHP_FUNCTION ( timezone_transistions_get )
{
zval * object , * element ;
php_timezone_obj * tzobj ;
int i ;
if ( zend_parse_method_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , getThis ( ) , " O " , & object , date_ce_timezone ) = = FAILURE ) {
RETURN_FALSE ;
}
tzobj = ( php_timezone_obj * ) zend_object_store_get_object ( object TSRMLS_CC ) ;
array_init ( return_value ) ;
for ( i = 0 ; i < tzobj - > tz - > timecnt ; + + i ) {
MAKE_STD_ZVAL ( element ) ;
array_init ( element ) ;
add_assoc_long ( element , " ts " , tzobj - > tz - > trans [ i ] ) ;
add_assoc_string ( element , " time " , php_format_date ( DATE_FORMAT_ISO8601 , 13 , tzobj - > tz - > trans [ i ] , 0 TSRMLS_CC ) , 0 ) ;
add_assoc_long ( element , " offset " , tzobj - > tz - > type [ tzobj - > tz - > trans_idx [ i ] ] . offset ) ;
add_assoc_bool ( element , " isdst " , tzobj - > tz - > type [ tzobj - > tz - > trans_idx [ i ] ] . isdst ) ;
add_assoc_string ( element , " abbr " , & tzobj - > tz - > timezone_abbr [ tzobj - > tz - > type [ tzobj - > tz - > trans_idx [ i ] ] . abbr_idx ] , 1 ) ;
add_next_index_zval ( return_value , element ) ;
}
}
PHP_FUNCTION ( timezone_identifiers_list )
{
timelib_tzdb_index_entry * table ;
int i , item_count ;
table = timelib_timezone_identifiers_list ( & item_count ) ;
array_init ( return_value ) ;
for ( i = 0 ; i < item_count ; + + i ) {
add_next_index_string ( return_value , table [ i ] . id , 1 ) ;
} ;
}
PHP_FUNCTION ( timezone_abbreviations_list )
{
timelib_tz_lookup_table * table , * entry ;
zval * element ;
table = timelib_timezone_abbreviations_list ( ) ;
array_init ( return_value ) ;
entry = table ;
do {
MAKE_STD_ZVAL ( element ) ;
array_init ( element ) ;
add_assoc_bool ( element , " dst " , entry - > type ) ;
add_assoc_long ( element , " offset " , - entry - > value * 60 ) ;
if ( entry - > full_tz_name ) {
add_assoc_string ( element , " timezone_id " , entry - > full_tz_name , 1 ) ;
} else {
add_assoc_null ( element , " timezone_id " ) ;
}
add_assoc_zval ( return_value , entry - > name , element ) ;
entry + + ;
} while ( entry - > name ) ;
}
# endif
/* {{{ proto bool date_default_timezone_set(string timezone_identifier)
2005-07-05 05:27:26 +08:00
Sets the default timezone used by all date / time functions in a script */
PHP_FUNCTION ( date_default_timezone_set )
2005-07-03 05:19:25 +08:00
{
char * zone ;
int zone_len ;
if ( zend_parse_parameters ( ZEND_NUM_ARGS ( ) TSRMLS_CC , " s " , & zone , & zone_len ) = = FAILURE ) {
RETURN_FALSE ;
}
if ( DATEG ( timezone ) ) {
efree ( DATEG ( timezone ) ) ;
2005-07-05 02:13:15 +08:00
DATEG ( timezone ) = NULL ;
2005-07-03 05:19:25 +08:00
}
2005-07-04 02:56:57 +08:00
DATEG ( timezone ) = estrndup ( zone , zone_len ) ;
2005-07-03 05:19:25 +08:00
RETURN_TRUE ;
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-03 05:19:25 +08:00
2005-07-20 16:31:02 +08:00
/* {{{ proto string date_default_timezone_get()
2005-07-05 05:27:26 +08:00
Gets the default timezone used by all date / time functions in a script */
PHP_FUNCTION ( date_default_timezone_get )
2005-07-03 05:19:25 +08:00
{
2005-07-20 16:31:02 +08:00
timelib_tzinfo * default_tz ;
default_tz = get_timezone_info ( TSRMLS_C ) ;
RETVAL_STRING ( default_tz - > name , 1 ) ;
timelib_tzinfo_dtor ( default_tz ) ;
2005-07-03 05:19:25 +08:00
}
2005-07-05 05:27:26 +08:00
/* }}} */
2005-07-03 05:19:25 +08:00
2005-06-15 05:32:29 +08:00
/*
* Local variables :
* tab - width : 4
* c - basic - offset : 4
* End :
* vim600 : fdm = marker
* vim : noet sw = 4 ts = 4
*/