1 /* asctime and asctime_r a la POSIX and ISO C, except pad years before 1000.  */
2 
3 /*
4 ** This file is in the public domain, so clarified as of
5 ** 1996-06-05 by Arthur David Olson.
6 */
7 
8 /*
9 ** Avoid the temptation to punt entirely to strftime;
10 ** the output of strftime is supposed to be locale specific
11 ** whereas the output of asctime is supposed to be constant.
12 */
13 
14 /*LINTLIBRARY*/
15 
16 #include "private.h"
17 #include <stdio.h>
18 
19 /*
20 ** All years associated with 32-bit time_t values are exactly four digits long;
21 ** some years associated with 64-bit time_t values are not.
22 ** Vintage programs are coded for years that are always four digits long
23 ** and may assume that the newline always lands in the same place.
24 ** For years that are less than four digits, we pad the output with
25 ** leading zeroes to get the newline in the traditional place.
26 ** The -4 ensures that we get four characters of output even if
27 ** we call a strftime variant that produces fewer characters for some years.
28 ** The ISO C and POSIX standards prohibit padding the year,
29 ** but many implementations pad anyway; most likely the standards are buggy.
30 */
31 static char const ASCTIME_FMT[] = "%s %s%3d %.2d:%.2d:%.2d %-4s\n";
32 /*
33 ** For years that are more than four digits we put extra spaces before the year
34 ** so that code trying to overwrite the newline won't end up overwriting
35 ** a digit within a year and truncating the year (operating on the assumption
36 ** that no output is better than wrong output).
37 */
38 static char const ASCTIME_FMT_B[] = "%s %s%3d %.2d:%.2d:%.2d     %s\n";
39 
40 enum { STD_ASCTIME_BUF_SIZE = 26 };
41 /*
42 ** Big enough for something such as
43 ** ??? ???-2147483648 -2147483648:-2147483648:-2147483648     -2147483648\n
44 ** (two three-character abbreviations, five strings denoting integers,
45 ** seven explicit spaces, two explicit colons, a newline,
46 ** and a trailing NUL byte).
47 ** The values above are for systems where an int is 32 bits and are provided
48 ** as an example; the size expression below is a bound for the system at
49 ** hand.
50 */
51 static char buf_asctime[2*3 + 5*INT_STRLEN_MAXIMUM(int) + 7 + 2 + 1 + 1];
52 
53 /* A similar buffer for ctime.
54    C89 requires that they be the same buffer.
55    This requirement was removed in C99, so support it only if requested,
56    as support is more likely to lead to bugs in badly written programs.  */
57 #if SUPPORT_C89
58 # define buf_ctime buf_asctime
59 #else
60 static char buf_ctime[sizeof buf_asctime];
61 #endif
62 
63 char *
asctime_r(struct tm const * restrict timeptr,char * restrict buf)64 asctime_r(struct tm const *restrict timeptr, char *restrict buf)
65 {
66 	static const char	wday_name[][4] = {
67 		"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"
68 	};
69 	static const char	mon_name[][4] = {
70 		"Jan", "Feb", "Mar", "Apr", "May", "Jun",
71 		"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"
72 	};
73 	register const char *	wn;
74 	register const char *	mn;
75 	char			year[INT_STRLEN_MAXIMUM(int) + 2];
76 	char result[sizeof buf_asctime];
77 
78 	if (timeptr == NULL) {
79 		errno = EINVAL;
80 		return strcpy(buf, "??? ??? ?? ??:??:?? ????\n");
81 	}
82 	if (timeptr->tm_wday < 0 || timeptr->tm_wday >= DAYSPERWEEK)
83 		wn = "???";
84 	else	wn = wday_name[timeptr->tm_wday];
85 	if (timeptr->tm_mon < 0 || timeptr->tm_mon >= MONSPERYEAR)
86 		mn = "???";
87 	else	mn = mon_name[timeptr->tm_mon];
88 	/*
89 	** Use strftime's %Y to generate the year, to avoid overflow problems
90 	** when computing timeptr->tm_year + TM_YEAR_BASE.
91 	** Assume that strftime is unaffected by other out-of-range members
92 	** (e.g., timeptr->tm_mday) when processing "%Y".
93 	*/
94 	strftime(year, sizeof year, "%Y", timeptr);
95 	/*
96 	** We avoid using snprintf since it's not available on all systems.
97 	*/
98 	snprintf(result, sizeof(result), /* Android change: use snprintf. */
99 		((strlen(year) <= 4) ? ASCTIME_FMT : ASCTIME_FMT_B),
100 		wn, mn,
101 		timeptr->tm_mday, timeptr->tm_hour,
102 		timeptr->tm_min, timeptr->tm_sec,
103 		year);
104 	if (strlen(result) < STD_ASCTIME_BUF_SIZE
105 	    || buf == buf_ctime || buf == buf_asctime)
106 		return strcpy(buf, result);
107 	else {
108 		errno = EOVERFLOW;
109 		return NULL;
110 	}
111 }
112 
113 char *
asctime(register const struct tm * timeptr)114 asctime(register const struct tm *timeptr)
115 {
116 	return asctime_r(timeptr, buf_asctime);
117 }
118 
119 char *
ctime_r(const time_t * timep,char * buf)120 ctime_r(const time_t *timep, char *buf)
121 {
122   struct tm mytm;
123   struct tm *tmp = localtime_r(timep, &mytm);
124   return tmp ? asctime_r(tmp, buf) : NULL;
125 }
126 
127 char *
ctime(const time_t * timep)128 ctime(const time_t *timep)
129 {
130   return ctime_r(timep, buf_ctime);
131 }
132