am 82ec6850: am 6c0e373d: am ec432608: am 0c24effd: Merge "Add the libcutils localtime_tz and mktime_t extensions to bionic."
				
					
				
			* commit '82ec685054bd14e596fcb0661c7f5c398743b4b3': Add the libcutils localtime_tz and mktime_t extensions to bionic.
This commit is contained in:
		| @@ -49,7 +49,12 @@ struct strftime_locale { | |||||||
|     const char *  date_fmt; |     const char *  date_fmt; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Note: you should consider these extensions deprecated and use managed code or icu4c instead. | ||||||
|  |  */ | ||||||
| extern size_t strftime_tz(char* s, size_t max, const char* format, const struct tm* tm, const struct strftime_locale* lc); | extern size_t strftime_tz(char* s, size_t max, const char* format, const struct tm* tm, const struct strftime_locale* lc); | ||||||
|  | extern time_t mktime_tz(struct tm* const tmp, char const* tz); | ||||||
|  | extern void localtime_tz(const time_t* const timep, struct tm* tmp, const char* tz); | ||||||
|  |  | ||||||
| #endif /* _BIONIC_STRFTIME_TZ_DECLARED */ | #endif /* _BIONIC_STRFTIME_TZ_DECLARED */ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -217,9 +217,9 @@ static const char * getoffset P((const char * strp, long * offsetp)); | |||||||
| static const char * getrule P((const char * strp, struct rule * rulep)); | static const char * getrule P((const char * strp, struct rule * rulep)); | ||||||
| static void     gmtload P((struct state * sp)); | static void     gmtload P((struct state * sp)); | ||||||
| static struct tm *  gmtsub P((const time_t * timep, long offset, | static struct tm *  gmtsub P((const time_t * timep, long offset, | ||||||
|                 struct tm * tmp)); |                 struct tm * tmp, const struct state * sp)); // android-changed: added sp. | ||||||
| static struct tm *  localsub P((const time_t * timep, long offset, | static struct tm *  localsub P((const time_t * timep, long offset, | ||||||
|                 struct tm * tmp)); |                 struct tm * tmp, const struct state * sp)); // android-changed: added sp. | ||||||
| static int      increment_overflow P((int * number, int delta)); | static int      increment_overflow P((int * number, int delta)); | ||||||
| static int      leaps_thru_end_of P((int y)); | static int      leaps_thru_end_of P((int y)); | ||||||
| static int      long_increment_overflow P((long * number, int delta)); | static int      long_increment_overflow P((long * number, int delta)); | ||||||
| @@ -230,16 +230,16 @@ static int      normalize_overflow P((int * tensptr, int * unitsptr, | |||||||
| static void     settzname P((void)); | static void     settzname P((void)); | ||||||
| static time_t       time1 P((struct tm * tmp, | static time_t       time1 P((struct tm * tmp, | ||||||
|                 struct tm * (*funcp) P((const time_t *, |                 struct tm * (*funcp) P((const time_t *, | ||||||
|                 long, struct tm *)), |                 long, struct tm *, const struct state *)), // android-changed: added state*. | ||||||
|                 long offset)); |                 long offset, const struct state * sp)); // android-changed: added sp. | ||||||
| static time_t       time2 P((struct tm *tmp, | static time_t       time2 P((struct tm *tmp, | ||||||
|                 struct tm * (*funcp) P((const time_t *, |                 struct tm * (*funcp) P((const time_t *, | ||||||
|                 long, struct tm*)), |                 long, struct tm*, const struct state *)), // android-changed: added state*. | ||||||
|                 long offset, int * okayp)); |                 long offset, int * okayp, const struct state * sp)); // android-changed: added sp. | ||||||
| static time_t       time2sub P((struct tm *tmp, | static time_t       time2sub P((struct tm *tmp, | ||||||
|                 struct tm * (*funcp) P((const time_t *, |                 struct tm * (*funcp) P((const time_t *, | ||||||
|                 long, struct tm*)), |                 long, struct tm*, const struct state *)), // android-changed: added state*. | ||||||
|                 long offset, int * okayp, int do_norm_secs)); |                 long offset, int * okayp, int do_norm_secs, const struct state * sp)); // android-change: added sp. | ||||||
| static struct tm *  timesub P((const time_t * timep, long offset, | static struct tm *  timesub P((const time_t * timep, long offset, | ||||||
|                 const struct state * sp, struct tm * tmp)); |                 const struct state * sp, struct tm * tmp)); | ||||||
| static int      tmcomp P((const struct tm * atmp, | static int      tmcomp P((const struct tm * atmp, | ||||||
| @@ -1331,21 +1331,25 @@ tzset P((void)) | |||||||
|  |  | ||||||
| /*ARGSUSED*/ | /*ARGSUSED*/ | ||||||
| static struct tm * | static struct tm * | ||||||
| localsub(timep, offset, tmp) | localsub(timep, offset, tmp, sp) // android-changed: added sp. | ||||||
| const time_t * const    timep; | const time_t * const    timep; | ||||||
| const long      offset; | const long      offset; | ||||||
| struct tm * const   tmp; | struct tm * const   tmp; | ||||||
|  | const struct state * sp; // android-added: added sp. | ||||||
| { | { | ||||||
|     register struct state *     sp; |  | ||||||
|     register const struct ttinfo *  ttisp; |     register const struct ttinfo *  ttisp; | ||||||
|     register int            i; |     register int            i; | ||||||
|     register struct tm *        result; |     register struct tm *        result; | ||||||
|     const time_t            t = *timep; |     const time_t            t = *timep; | ||||||
|  |  | ||||||
|  |     // BEGIN android-changed: support user-supplied sp. | ||||||
|  |     if (sp == NULL) { | ||||||
|         sp = lclptr; |         sp = lclptr; | ||||||
|  |     } | ||||||
|  |     // END android-changed | ||||||
| #ifdef ALL_STATE | #ifdef ALL_STATE | ||||||
|     if (sp == NULL) |     if (sp == NULL) | ||||||
|         return gmtsub(timep, offset, tmp); |         return gmtsub(timep, offset, tmp, sp); // android-changed: added sp. | ||||||
| #endif /* defined ALL_STATE */ | #endif /* defined ALL_STATE */ | ||||||
|     if ((sp->goback && t < sp->ats[0]) || |     if ((sp->goback && t < sp->ats[0]) || | ||||||
|         (sp->goahead && t > sp->ats[sp->timecnt - 1])) { |         (sp->goahead && t > sp->ats[sp->timecnt - 1])) { | ||||||
| @@ -1372,7 +1376,7 @@ struct tm * const   tmp; | |||||||
|             if (newt < sp->ats[0] || |             if (newt < sp->ats[0] || | ||||||
|                 newt > sp->ats[sp->timecnt - 1]) |                 newt > sp->ats[sp->timecnt - 1]) | ||||||
|                     return NULL;    /* "cannot happen" */ |                     return NULL;    /* "cannot happen" */ | ||||||
|             result = localsub(&newt, offset, tmp); |             result = localsub(&newt, offset, tmp, sp); // android-changed: added sp. | ||||||
|             if (result == tmp) { |             if (result == tmp) { | ||||||
|                 register time_t newy; |                 register time_t newy; | ||||||
|  |  | ||||||
| @@ -1442,7 +1446,7 @@ struct tm *     tmp; | |||||||
|  |  | ||||||
|     _tzLock(); |     _tzLock(); | ||||||
|     tzset_locked(); |     tzset_locked(); | ||||||
|     result = localsub(timep, 0L, tmp); |     result = localsub(timep, 0L, tmp, NULL); // android-changed: extra parameter. | ||||||
|     _tzUnlock(); |     _tzUnlock(); | ||||||
|  |  | ||||||
|     return result; |     return result; | ||||||
| @@ -1453,13 +1457,16 @@ struct tm *     tmp; | |||||||
| */ | */ | ||||||
|  |  | ||||||
| static struct tm * | static struct tm * | ||||||
| gmtsub(timep, offset, tmp) | gmtsub(timep, offset, tmp, sp) // android-changed: added sp. | ||||||
| const time_t * const    timep; | const time_t * const    timep; | ||||||
| const long      offset; | const long      offset; | ||||||
| struct tm * const   tmp; | struct tm * const   tmp; | ||||||
|  | const struct state * sp; // android-changed: added sp. | ||||||
| { | { | ||||||
|     register struct tm *    result; |     register struct tm *    result; | ||||||
|  |  | ||||||
|  |     (void) sp; // android-added: unused. | ||||||
|  |  | ||||||
|     if (!gmt_is_set) { |     if (!gmt_is_set) { | ||||||
|         gmt_is_set = TRUE; |         gmt_is_set = TRUE; | ||||||
| #ifdef ALL_STATE | #ifdef ALL_STATE | ||||||
| @@ -1510,7 +1517,7 @@ struct tm *     tmp; | |||||||
|     struct tm*  result; |     struct tm*  result; | ||||||
|  |  | ||||||
|     _tzLock(); |     _tzLock(); | ||||||
|     result = gmtsub(timep, 0L, tmp); |     result = gmtsub(timep, 0L, tmp, NULL); // android-changed: extra parameter. | ||||||
|     _tzUnlock(); |     _tzUnlock(); | ||||||
|  |  | ||||||
|     return result; |     return result; | ||||||
| @@ -1523,7 +1530,7 @@ offtime(timep, offset) | |||||||
| const time_t * const    timep; | const time_t * const    timep; | ||||||
| const long      offset; | const long      offset; | ||||||
| { | { | ||||||
|     return gmtsub(timep, offset, &tmGlobal); |     return gmtsub(timep, offset, &tmGlobal, NULL); // android-changed: extra parameter. | ||||||
| } | } | ||||||
| #endif /* 0 */ | #endif /* 0 */ | ||||||
| #endif /* defined STD_INSPIRED */ | #endif /* defined STD_INSPIRED */ | ||||||
| @@ -1796,14 +1803,14 @@ register const struct tm * const btmp; | |||||||
| } | } | ||||||
|  |  | ||||||
| static time_t | static time_t | ||||||
| time2sub(tmp, funcp, offset, okayp, do_norm_secs) | time2sub(tmp, funcp, offset, okayp, do_norm_secs, sp) // android-changed: added sp | ||||||
| struct tm * const   tmp; | struct tm * const   tmp; | ||||||
| struct tm * (* const    funcp) P((const time_t*, long, struct tm*)); | struct tm * (* const    funcp) P((const time_t*, long, struct tm*, const struct state*)); // android-changed: added state* | ||||||
| const long      offset; | const long      offset; | ||||||
| int * const     okayp; | int * const     okayp; | ||||||
| const int       do_norm_secs; | const int       do_norm_secs; | ||||||
|  | const struct state * sp; // android-changed: added sp | ||||||
| { | { | ||||||
|     register const struct state *   sp; |  | ||||||
|     register int            dir; |     register int            dir; | ||||||
|     register int            i, j; |     register int            i, j; | ||||||
|     register int            saved_seconds; |     register int            saved_seconds; | ||||||
| @@ -1905,7 +1912,7 @@ const int       do_norm_secs; | |||||||
|             t = lo; |             t = lo; | ||||||
|         else if (t > hi) |         else if (t > hi) | ||||||
|             t = hi; |             t = hi; | ||||||
|         if ((*funcp)(&t, offset, &mytm) == NULL) { |         if ((*funcp)(&t, offset, &mytm, sp) == NULL) { // android-changed: added sp. | ||||||
|             /* |             /* | ||||||
|             ** Assume that t is too extreme to be represented in |             ** Assume that t is too extreme to be represented in | ||||||
|             ** a struct tm; arrange things so that it is less |             ** a struct tm; arrange things so that it is less | ||||||
| @@ -1943,9 +1950,13 @@ const int       do_norm_secs; | |||||||
|         /* |         /* | ||||||
|         ** The (void *) casts are the benefit of SunOS 3.3 on Sun 2's. |         ** The (void *) casts are the benefit of SunOS 3.3 on Sun 2's. | ||||||
|         */ |         */ | ||||||
|  |         // BEGIN android-changed: support user-supplied sp | ||||||
|  |         if (sp == NULL) { | ||||||
|             sp = (const struct state *) |             sp = (const struct state *) | ||||||
|                 (((void *) funcp == (void *) localsub) ? |                 (((void *) funcp == (void *) localsub) ? | ||||||
|                 lclptr : gmtptr); |                 lclptr : gmtptr); | ||||||
|  |         } | ||||||
|  |         // END android-changed | ||||||
| #ifdef ALL_STATE | #ifdef ALL_STATE | ||||||
|         if (sp == NULL) |         if (sp == NULL) | ||||||
|             return WRONG; |             return WRONG; | ||||||
| @@ -1958,7 +1969,7 @@ const int       do_norm_secs; | |||||||
|                     continue; |                     continue; | ||||||
|                 newt = t + sp->ttis[j].tt_gmtoff - |                 newt = t + sp->ttis[j].tt_gmtoff - | ||||||
|                     sp->ttis[i].tt_gmtoff; |                     sp->ttis[i].tt_gmtoff; | ||||||
|                 if ((*funcp)(&newt, offset, &mytm) == NULL) |                 if ((*funcp)(&newt, offset, &mytm, sp) == NULL) // android-changed: added sp. | ||||||
|                     continue; |                     continue; | ||||||
|                 if (tmcomp(&mytm, &yourtm) != 0) |                 if (tmcomp(&mytm, &yourtm) != 0) | ||||||
|                     continue; |                     continue; | ||||||
| @@ -1978,17 +1989,19 @@ label: | |||||||
|     if ((newt < t) != (saved_seconds < 0)) |     if ((newt < t) != (saved_seconds < 0)) | ||||||
|         return WRONG; |         return WRONG; | ||||||
|     t = newt; |     t = newt; | ||||||
|     if ((*funcp)(&t, offset, tmp)) |     if ((*funcp)(&t, offset, tmp, sp)) // android-changed: added sp. | ||||||
|         *okayp = TRUE; |         *okayp = TRUE; | ||||||
|     return t; |     return t; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // BEGIN android-changed: added sp. | ||||||
| static time_t | static time_t | ||||||
| time2(tmp, funcp, offset, okayp) | time2(tmp, funcp, offset, okayp, sp) | ||||||
| struct tm * const   tmp; | struct tm * const   tmp; | ||||||
| struct tm * (* const    funcp) P((const time_t*, long, struct tm*)); | struct tm * (* const    funcp) P((const time_t*, long, struct tm*, const struct state*)); | ||||||
| const long      offset; | const long      offset; | ||||||
| int * const     okayp; | int * const     okayp; | ||||||
|  | const struct state * sp; | ||||||
| { | { | ||||||
|     time_t  t; |     time_t  t; | ||||||
|  |  | ||||||
| @@ -1997,18 +2010,19 @@ int * const     okayp; | |||||||
|     ** (in case tm_sec contains a value associated with a leap second). |     ** (in case tm_sec contains a value associated with a leap second). | ||||||
|     ** If that fails, try with normalization of seconds. |     ** If that fails, try with normalization of seconds. | ||||||
|     */ |     */ | ||||||
|     t = time2sub(tmp, funcp, offset, okayp, FALSE); |     t = time2sub(tmp, funcp, offset, okayp, FALSE, sp); | ||||||
|     return *okayp ? t : time2sub(tmp, funcp, offset, okayp, TRUE); |     return *okayp ? t : time2sub(tmp, funcp, offset, okayp, TRUE, sp); | ||||||
| } | } | ||||||
|  | // END android-changed | ||||||
|  |  | ||||||
| static time_t | static time_t | ||||||
| time1(tmp, funcp, offset) | time1(tmp, funcp, offset, sp) // android-changed: added sp. | ||||||
| struct tm * const   tmp; | struct tm * const   tmp; | ||||||
| struct tm * (* const    funcp) P((const time_t *, long, struct tm *)); | struct tm * (* const    funcp) P((const time_t *, long, struct tm *, const struct state *)); | ||||||
| const long      offset; | const long      offset; | ||||||
|  | const struct state * sp; // android-changed: added sp. | ||||||
| { | { | ||||||
|     register time_t         t; |     register time_t         t; | ||||||
|     register const struct state *   sp; |  | ||||||
|     register int            samei, otheri; |     register int            samei, otheri; | ||||||
|     register int            sameind, otherind; |     register int            sameind, otherind; | ||||||
|     register int            i; |     register int            i; | ||||||
| @@ -2019,7 +2033,7 @@ const long      offset; | |||||||
|  |  | ||||||
|     if (tmp->tm_isdst > 1) |     if (tmp->tm_isdst > 1) | ||||||
|         tmp->tm_isdst = 1; |         tmp->tm_isdst = 1; | ||||||
|     t = time2(tmp, funcp, offset, &okay); |     t = time2(tmp, funcp, offset, &okay, sp); // android-changed: added sp. | ||||||
| #ifdef PCTS | #ifdef PCTS | ||||||
|     /* |     /* | ||||||
|     ** PCTS code courtesy Grant Sullivan. |     ** PCTS code courtesy Grant Sullivan. | ||||||
| @@ -2042,8 +2056,12 @@ const long      offset; | |||||||
|     /* |     /* | ||||||
|     ** The (void *) casts are the benefit of SunOS 3.3 on Sun 2's. |     ** The (void *) casts are the benefit of SunOS 3.3 on Sun 2's. | ||||||
|     */ |     */ | ||||||
|  |     // BEGIN android-changed: support user-supplied sp. | ||||||
|  |     if (sp == NULL) { | ||||||
|         sp = (const struct state *) (((void *) funcp == (void *) localsub) ? |         sp = (const struct state *) (((void *) funcp == (void *) localsub) ? | ||||||
|             lclptr : gmtptr); |             lclptr : gmtptr); | ||||||
|  |     } | ||||||
|  |     // BEGIN android-changed | ||||||
| #ifdef ALL_STATE | #ifdef ALL_STATE | ||||||
|     if (sp == NULL) |     if (sp == NULL) | ||||||
|         return WRONG; |         return WRONG; | ||||||
| @@ -2067,7 +2085,7 @@ const long      offset; | |||||||
|             tmp->tm_sec += sp->ttis[otheri].tt_gmtoff - |             tmp->tm_sec += sp->ttis[otheri].tt_gmtoff - | ||||||
|                     sp->ttis[samei].tt_gmtoff; |                     sp->ttis[samei].tt_gmtoff; | ||||||
|             tmp->tm_isdst = !tmp->tm_isdst; |             tmp->tm_isdst = !tmp->tm_isdst; | ||||||
|             t = time2(tmp, funcp, offset, &okay); |             t = time2(tmp, funcp, offset, &okay, sp); // android-changed: added sp. | ||||||
|             if (okay) |             if (okay) | ||||||
|                 return t; |                 return t; | ||||||
|             tmp->tm_sec -= sp->ttis[otheri].tt_gmtoff - |             tmp->tm_sec -= sp->ttis[otheri].tt_gmtoff - | ||||||
| @@ -2085,11 +2103,40 @@ struct tm * const   tmp; | |||||||
|     time_t  result; |     time_t  result; | ||||||
|     _tzLock(); |     _tzLock(); | ||||||
|     tzset_locked(); |     tzset_locked(); | ||||||
|     result = time1(tmp, localsub, 0L); |     result = time1(tmp, localsub, 0L, NULL); // android-changed: extra parameter. | ||||||
|     _tzUnlock(); |     _tzUnlock(); | ||||||
|     return result; |     return result; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // BEGIN android-added | ||||||
|  | time_t | ||||||
|  | mktime_tz(tmp, tz) | ||||||
|  | struct tm * const tmp; | ||||||
|  | char const * tz; | ||||||
|  | { | ||||||
|  |     struct state st; | ||||||
|  |     if (tzload(tz, &st, TRUE) != 0) { | ||||||
|  |         // TODO: not sure what's best here, but for now, we fall back to gmt. | ||||||
|  |         gmtload(&st); | ||||||
|  |     } | ||||||
|  |     return time1(tmp, localsub, 0L, &st); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void | ||||||
|  | localtime_tz(timep, tmp, tz) | ||||||
|  | const time_t * const timep; | ||||||
|  | struct tm * tmp; | ||||||
|  | const char* tz; | ||||||
|  | { | ||||||
|  |     struct state st; | ||||||
|  |     if (tzload(tz, &st, TRUE) != 0) { | ||||||
|  |         // TODO: not sure what's best here, but for now, we fall back to gmt. | ||||||
|  |         gmtload(&st); | ||||||
|  |     } | ||||||
|  |     localsub(timep, 0L, tmp, &st); | ||||||
|  | } | ||||||
|  | // END android-added | ||||||
|  |  | ||||||
| #ifdef STD_INSPIRED | #ifdef STD_INSPIRED | ||||||
|  |  | ||||||
| time_t | time_t | ||||||
| @@ -2108,7 +2155,7 @@ struct tm * const   tmp; | |||||||
|  |  | ||||||
|     tmp->tm_isdst = 0; |     tmp->tm_isdst = 0; | ||||||
|     _tzLock(); |     _tzLock(); | ||||||
|     result = time1(tmp, gmtsub, 0L); |     result = time1(tmp, gmtsub, 0L, NULL); // android-changed: extra parameter. | ||||||
|     _tzUnlock(); |     _tzUnlock(); | ||||||
|  |  | ||||||
|     return result; |     return result; | ||||||
| @@ -2124,7 +2171,7 @@ const long      offset; | |||||||
|  |  | ||||||
|     tmp->tm_isdst = 0; |     tmp->tm_isdst = 0; | ||||||
|     _tzLock(); |     _tzLock(); | ||||||
|     result = time1(tmp, gmtsub, offset); |     result = time1(tmp, gmtsub, offset, NULL); // android-changed: extra parameter. | ||||||
|     _tzUnlock(); |     _tzUnlock(); | ||||||
|  |  | ||||||
|     return result; |     return result; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Elliott Hughes
					Elliott Hughes