mirror of
https://github.com/php/php-src.git
synced 2026-03-24 16:22:37 +01:00
390 lines
11 KiB
C
390 lines
11 KiB
C
/*
|
|
* The MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2015-2019 Derick Rethans
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
* THE SOFTWARE.
|
|
*/
|
|
|
|
#include "timelib.h"
|
|
#include "timelib_private.h"
|
|
#include <math.h>
|
|
|
|
static void swap_times(timelib_time **one, timelib_time **two, timelib_rel_time *rt)
|
|
{
|
|
timelib_time *swp;
|
|
|
|
swp = *two;
|
|
*two = *one;
|
|
*one = swp;
|
|
rt->invert = 1;
|
|
}
|
|
|
|
static void sort_old_to_new(timelib_time **one, timelib_time **two, timelib_rel_time *rt)
|
|
{
|
|
/* Check whether date/times need to be inverted. If both times are
|
|
* TIMELIB_ZONETYPE_ID times with the same TZID, we use the y-s + us fields. */
|
|
if (
|
|
(*one)->zone_type == TIMELIB_ZONETYPE_ID &&
|
|
(*two)->zone_type == TIMELIB_ZONETYPE_ID &&
|
|
(strcmp((*one)->tz_info->name, (*two)->tz_info->name) == 0)
|
|
) {
|
|
if (
|
|
((*one)->y > (*two)->y) ||
|
|
((*one)->y == (*two)->y && (*one)->m > (*two)->m) ||
|
|
((*one)->y == (*two)->y && (*one)->m == (*two)->m && (*one)->d > (*two)->d) ||
|
|
((*one)->y == (*two)->y && (*one)->m == (*two)->m && (*one)->d == (*two)->d && (*one)->h > (*two)->h) ||
|
|
((*one)->y == (*two)->y && (*one)->m == (*two)->m && (*one)->d == (*two)->d && (*one)->h == (*two)->h && (*one)->i > (*two)->i) ||
|
|
((*one)->y == (*two)->y && (*one)->m == (*two)->m && (*one)->d == (*two)->d && (*one)->h == (*two)->h && (*one)->i == (*two)->i && (*one)->s > (*two)->s) ||
|
|
((*one)->y == (*two)->y && (*one)->m == (*two)->m && (*one)->d == (*two)->d && (*one)->h == (*two)->h && (*one)->i == (*two)->i && (*one)->s == (*two)->s && (*one)->us > (*two)->us)
|
|
) {
|
|
swap_times(one, two, rt);
|
|
}
|
|
return;
|
|
}
|
|
|
|
/* Fall back to using the SSE instead to rearrange */
|
|
if (
|
|
((*one)->sse > (*two)->sse) ||
|
|
((*one)->sse == (*two)->sse && (*one)->us > (*two)->us)
|
|
) {
|
|
swap_times(one, two, rt);
|
|
}
|
|
}
|
|
|
|
static timelib_rel_time *timelib_diff_with_tzid(timelib_time *one, timelib_time *two)
|
|
{
|
|
timelib_rel_time *rt;
|
|
timelib_sll dst_corr = 0, dst_h_corr = 0, dst_m_corr = 0;
|
|
int32_t trans_offset;
|
|
timelib_sll trans_transition_time;
|
|
|
|
rt = timelib_rel_time_ctor();
|
|
rt->invert = 0;
|
|
|
|
sort_old_to_new(&one, &two, rt);
|
|
|
|
/* Calculate correction for UTC offset changes between first and second SSE */
|
|
dst_corr = two->z - one->z;
|
|
dst_h_corr = dst_corr / 3600;
|
|
dst_m_corr = (dst_corr % 3600) / 60;
|
|
|
|
rt->y = two->y - one->y;
|
|
rt->m = two->m - one->m;
|
|
rt->d = two->d - one->d;
|
|
rt->h = two->h - one->h;
|
|
rt->i = two->i - one->i;
|
|
rt->s = two->s - one->s;
|
|
rt->us = two->us - one->us;
|
|
|
|
rt->days = timelib_diff_days(one, two);
|
|
|
|
/* Fall Back: Cater for transition period, where rt->invert is 0, but there are negative numbers */
|
|
if (two->sse < one->sse) {
|
|
timelib_sll flipped = llabs((rt->i * 60) + (rt->s) - dst_corr);
|
|
rt->h = flipped / SECS_PER_HOUR;
|
|
rt->i = (flipped - rt->h * SECS_PER_HOUR) / 60;
|
|
rt->s = flipped % 60;
|
|
|
|
rt->invert = 1 - rt->invert;
|
|
}
|
|
|
|
timelib_do_rel_normalize(rt->invert ? one : two, rt);
|
|
|
|
if (one->dst == 1 && two->dst == 0) { /* Fall Back */
|
|
if (two->tz_info) {
|
|
if ((two->sse - one->sse + dst_corr) < SECS_PER_DAY) {
|
|
rt->h -= dst_h_corr;
|
|
rt->i -= dst_m_corr;
|
|
}
|
|
}
|
|
} else if (one->dst == 0 && two->dst == 1) { /* Spring Forward */
|
|
if (two->tz_info) {
|
|
int success = timelib_get_time_zone_offset_info(two->sse, two->tz_info, &trans_offset, &trans_transition_time, NULL);
|
|
|
|
if (
|
|
success &&
|
|
!((one->sse + SECS_PER_DAY > trans_transition_time) && (one->sse + SECS_PER_DAY <= (trans_transition_time + dst_corr))) &&
|
|
two->sse >= trans_transition_time &&
|
|
((two->sse - one->sse + dst_corr) % SECS_PER_DAY) > (two->sse - trans_transition_time)
|
|
) {
|
|
rt->h -= dst_h_corr;
|
|
rt->i -= dst_m_corr;
|
|
}
|
|
}
|
|
} else if (two->sse - one->sse >= SECS_PER_DAY) {
|
|
/* Check whether we're in the period to the next transition time */
|
|
if (timelib_get_time_zone_offset_info(two->sse - two->z, two->tz_info, &trans_offset, &trans_transition_time, NULL)) {
|
|
dst_corr = one->z - trans_offset;
|
|
|
|
if (two->sse >= trans_transition_time - dst_corr && two->sse < trans_transition_time) {
|
|
rt->d--;
|
|
rt->h = 24;
|
|
}
|
|
}
|
|
}
|
|
|
|
return rt;
|
|
}
|
|
|
|
timelib_rel_time *timelib_diff(timelib_time *one, timelib_time *two)
|
|
{
|
|
timelib_rel_time *rt;
|
|
|
|
if (one->zone_type == TIMELIB_ZONETYPE_ID && two->zone_type == TIMELIB_ZONETYPE_ID && strcmp(one->tz_info->name, two->tz_info->name) == 0) {
|
|
return timelib_diff_with_tzid(one, two);
|
|
}
|
|
|
|
rt = timelib_rel_time_ctor();
|
|
rt->invert = 0;
|
|
|
|
sort_old_to_new(&one, &two, rt);
|
|
|
|
rt->y = two->y - one->y;
|
|
rt->m = two->m - one->m;
|
|
rt->d = two->d - one->d;
|
|
rt->h = two->h - one->h;
|
|
if (one->zone_type != TIMELIB_ZONETYPE_ID) {
|
|
rt->h = rt->h + one->dst;
|
|
}
|
|
if (two->zone_type != TIMELIB_ZONETYPE_ID) {
|
|
rt->h = rt->h - two->dst;
|
|
}
|
|
rt->i = two->i - one->i;
|
|
rt->s = two->s - one->s - two->z + one->z;
|
|
rt->us = two->us - one->us;
|
|
|
|
rt->days = timelib_diff_days(one, two);
|
|
|
|
timelib_do_rel_normalize(rt->invert ? one : two, rt);
|
|
|
|
return rt;
|
|
}
|
|
|
|
|
|
int timelib_diff_days(timelib_time *one, timelib_time *two)
|
|
{
|
|
int days = 0;
|
|
|
|
if (timelib_same_timezone(one, two)) {
|
|
timelib_time *earliest, *latest;
|
|
double earliest_time, latest_time;
|
|
|
|
if (timelib_time_compare(one, two) < 0) {
|
|
earliest = one;
|
|
latest = two;
|
|
} else {
|
|
earliest = two;
|
|
latest = one;
|
|
}
|
|
timelib_hmsf_to_decimal_hour(earliest->h, earliest->i, earliest->s, earliest->us, &earliest_time);
|
|
timelib_hmsf_to_decimal_hour(latest->h, latest->i, latest->s, latest->us, &latest_time);
|
|
|
|
days = llabs(timelib_epoch_days_from_time(one) - timelib_epoch_days_from_time(two));
|
|
if (latest_time < earliest_time && days > 0) {
|
|
days--;
|
|
}
|
|
} else {
|
|
days = fabs(floor(one->sse - two->sse) / 86400);
|
|
}
|
|
|
|
return days;
|
|
}
|
|
|
|
|
|
timelib_time *timelib_add(timelib_time *old_time, timelib_rel_time *interval)
|
|
{
|
|
int bias = 1;
|
|
timelib_time *t = timelib_time_clone(old_time);
|
|
|
|
if (interval->have_weekday_relative || interval->have_special_relative) {
|
|
memcpy(&t->relative, interval, sizeof(timelib_rel_time));
|
|
} else {
|
|
if (interval->invert) {
|
|
bias = -1;
|
|
}
|
|
memset(&t->relative, 0, sizeof(timelib_rel_time));
|
|
t->relative.y = interval->y * bias;
|
|
t->relative.m = interval->m * bias;
|
|
t->relative.d = interval->d * bias;
|
|
t->relative.h = interval->h * bias;
|
|
t->relative.i = interval->i * bias;
|
|
t->relative.s = interval->s * bias;
|
|
t->relative.us = interval->us * bias;
|
|
}
|
|
t->have_relative = 1;
|
|
t->sse_uptodate = 0;
|
|
|
|
timelib_update_ts(t, NULL);
|
|
|
|
timelib_update_from_sse(t);
|
|
t->have_relative = 0;
|
|
|
|
return t;
|
|
}
|
|
|
|
timelib_time *timelib_sub(timelib_time *old_time, timelib_rel_time *interval)
|
|
{
|
|
int bias = 1;
|
|
timelib_time *t = timelib_time_clone(old_time);
|
|
|
|
if (interval->invert) {
|
|
bias = -1;
|
|
}
|
|
|
|
memset(&t->relative, 0, sizeof(timelib_rel_time));
|
|
t->relative.y = 0 - (interval->y * bias);
|
|
t->relative.m = 0 - (interval->m * bias);
|
|
t->relative.d = 0 - (interval->d * bias);
|
|
t->relative.h = 0 - (interval->h * bias);
|
|
t->relative.i = 0 - (interval->i * bias);
|
|
t->relative.s = 0 - (interval->s * bias);
|
|
t->relative.us = 0 - (interval->us * bias);
|
|
t->have_relative = 1;
|
|
t->sse_uptodate = 0;
|
|
|
|
timelib_update_ts(t, NULL);
|
|
|
|
timelib_update_from_sse(t);
|
|
|
|
t->have_relative = 0;
|
|
|
|
return t;
|
|
}
|
|
|
|
static void do_range_limit(timelib_sll start, timelib_sll end, timelib_sll adj, timelib_sll *a, timelib_sll *b)
|
|
{
|
|
if (*a < start) {
|
|
*b -= (start - *a - 1) / adj + 1;
|
|
*a += adj * ((start - *a - 1) / adj + 1);
|
|
}
|
|
if (*a >= end) {
|
|
*b += *a / adj;
|
|
*a -= adj * (*a / adj);
|
|
}
|
|
}
|
|
|
|
|
|
timelib_time *timelib_add_wall(timelib_time *old_time, timelib_rel_time *interval)
|
|
{
|
|
int bias = 1;
|
|
timelib_time *t = timelib_time_clone(old_time);
|
|
|
|
t->have_relative = 1;
|
|
t->sse_uptodate = 0;
|
|
|
|
if (interval->have_weekday_relative || interval->have_special_relative) {
|
|
memcpy(&t->relative, interval, sizeof(timelib_rel_time));
|
|
|
|
timelib_update_ts(t, NULL);
|
|
|
|
timelib_update_from_sse(t);
|
|
} else {
|
|
if (interval->invert) {
|
|
bias = -1;
|
|
}
|
|
memset(&t->relative, 0, sizeof(timelib_rel_time));
|
|
t->relative.y = interval->y * bias;
|
|
t->relative.m = interval->m * bias;
|
|
t->relative.d = interval->d * bias;
|
|
|
|
if (t->relative.y || t->relative.m || t->relative.d) {
|
|
timelib_update_ts(t, NULL);
|
|
}
|
|
|
|
if (interval->us == 0) {
|
|
t->sse += bias * timelib_hms_to_seconds(interval->h, interval->i, interval->s);
|
|
timelib_update_from_sse(t);
|
|
} else {
|
|
timelib_rel_time *temp_interval = timelib_rel_time_clone(interval);
|
|
|
|
do_range_limit(0, 1000000, 1000000, &temp_interval->us, &temp_interval->s);
|
|
t->sse += bias * timelib_hms_to_seconds(temp_interval->h, temp_interval->i, temp_interval->s);
|
|
timelib_update_from_sse(t);
|
|
t->us += temp_interval->us * bias;
|
|
|
|
timelib_do_normalize(t);
|
|
timelib_update_ts(t, NULL);
|
|
|
|
timelib_rel_time_dtor(temp_interval);
|
|
}
|
|
timelib_do_normalize(t);
|
|
}
|
|
|
|
if (t->zone_type == TIMELIB_ZONETYPE_ID) {
|
|
timelib_set_timezone(t, t->tz_info);
|
|
}
|
|
t->have_relative = 0;
|
|
|
|
return t;
|
|
}
|
|
|
|
timelib_time *timelib_sub_wall(timelib_time *old_time, timelib_rel_time *interval)
|
|
{
|
|
int bias = 1;
|
|
timelib_time *t = timelib_time_clone(old_time);
|
|
|
|
t->have_relative = 1;
|
|
t->sse_uptodate = 0;
|
|
|
|
if (interval->have_weekday_relative || interval->have_special_relative) {
|
|
memcpy(&t->relative, interval, sizeof(timelib_rel_time));
|
|
|
|
timelib_update_ts(t, NULL);
|
|
timelib_update_from_sse(t);
|
|
} else {
|
|
if (interval->invert) {
|
|
bias = -1;
|
|
}
|
|
memset(&t->relative, 0, sizeof(timelib_rel_time));
|
|
t->relative.y = 0 - (interval->y * bias);
|
|
t->relative.m = 0 - (interval->m * bias);
|
|
t->relative.d = 0 - (interval->d * bias);
|
|
|
|
if (t->relative.y || t->relative.m || t->relative.d) {
|
|
timelib_update_ts(t, NULL);
|
|
}
|
|
|
|
if (interval->us == 0) {
|
|
t->sse -= bias * timelib_hms_to_seconds(interval->h, interval->i, interval->s);
|
|
timelib_update_from_sse(t);
|
|
} else {
|
|
timelib_rel_time *temp_interval = timelib_rel_time_clone(interval);
|
|
|
|
do_range_limit(0, 1000000, 1000000, &temp_interval->us, &temp_interval->s);
|
|
t->sse -= bias * timelib_hms_to_seconds(temp_interval->h, temp_interval->i, temp_interval->s);
|
|
timelib_update_from_sse(t);
|
|
t->us -= temp_interval->us * bias;
|
|
|
|
timelib_do_normalize(t);
|
|
timelib_update_ts(t, NULL);
|
|
|
|
timelib_rel_time_dtor(temp_interval);
|
|
}
|
|
timelib_do_normalize(t);
|
|
}
|
|
|
|
if (t->zone_type == TIMELIB_ZONETYPE_ID) {
|
|
timelib_set_timezone(t, t->tz_info);
|
|
}
|
|
t->have_relative = 0;
|
|
|
|
return t;
|
|
}
|