
Путь: Toys/POSIX, команды версии: Ver.4 Ver.9 date Комментарии в файле date.c :
Исходный текст в файле date.c #define FOR_date
#include "toys.h"
GLOBALS(
char *s, *r, *I, *D, *d;
unsigned nano;
)
// Handles any leading `TZ="blah" ` in the input string.
static void parse_date(char *str, time_t *t)
{
char *new_tz = NULL, *old_tz, *s = str;
if (!strncmp(str, "TZ=\"", 4)) {
// Extract the time zone and skip any whitespace.
new_tz = str+4;
if (!(str = strchr(new_tz, '"'))) xvali_date(0, s);
*str++ = 0;
while (isspace(*str)) str++;
// Switch $TZ.
old_tz = getenv("TZ");
setenv("TZ", new_tz, 1);
tzset();
}
time(t);
xparsedate(str, t, &TT.nano, 1);
if (new_tz) {
if (old_tz) setenv("TZ", old_tz, 1);
else unsetenv("TZ");
}
}
// Print strftime plus %N and %:z escape(s). Note: modifies fmt in those cases.
static void puts_time(char *fmt, struct tm *tm)
{
char *s, *snap, *out;
for (s = fmt;;s++) {
long n = 0;
// Find next %N/%:z or end of format string.
if (*(snap = s)) {
if (*s != '%') continue;
if (*++s == 'N') n = 9;
else if (isdigit(*s) && s[1] == 'N') n = *s++-'0';
else if (*s == ':' && s[1] == 'z') s++, n++;
else continue;
}
// Only modify input string if needed (default format is constant string).
if (*s) *snap = 0;
// Do we have any regular work for strftime to do?
out = toybuf;
if (*fmt) {
if (!strftime(out, sizeof(toybuf)-12, fmt, tm))
perror_exit("bad format '%s'", fmt);
out += strlen(out);
}
// Do we have any custom formatting to append to that?
if (*s == 'N') {
sprintf(out, "%09u", TT.nano);
out[n] = 0;
} else if (*s == 'z') {
strftime(out, 10, "%z", tm);
memmove(out+4, out+3, strlen(out+3)+1);
out[3] = ':';
}
xputsn(toybuf);
if (!*s || !*(fmt = s+1)) break;
}
xputc('\n');
}
void date_main(void)
{
char *setdate = *toys.optargs, *format_string = "%a %b %e %H:%M:%S %Z %Y",
*tz = NULL;
time_t t;
if (FLAG(I)) {
char *iso_formats[] = {"%F","%FT%H%:z","%FT%R%:z","%FT%T%:z","%FT%T,%N%:z"};
int i = stridx("dhmsn", (TT.I && *TT.I) ? *TT.I : 'd');
if (i<0) help_exit("bad -I: %s", TT.I);
format_string = xstrdup(iso_formats[i]);
}
if (FLAG(u)) {
tz = getenv("TZ");
setenv("TZ", "UTC", 1);
tzset();
}
if (TT.d) {
if (TT.D) {
struct tm tm = {};
char *s = strptime(TT.d, TT.D+(*TT.D=='+'), &tm);
t = (s && !*s) ? xvali_date(&tm, s) : xvali_date(0, TT.d);
} else parse_date(TT.d, &t);
} else {
struct timespec ts;
struct stat st;
if (TT.r) {
xstat(TT.r, &st);
ts = st.st_mtim;
} else clock_gettime(CLOCK_REALTIME, &ts);
t = ts.tv_sec;
TT.nano = ts.tv_nsec;
}
if (FLAG(s)) {
if (setdate) help_exit("can't set two dates at once");
setdate = TT.s;
}
// Fall through if no arguments
if (!setdate);
// Display the date?
else if (*setdate == '+') {
format_string = toys.optargs[0]+1;
setdate = toys.optargs[1];
// Set the date
} else if (setdate) {
struct timeval tv;
parse_date(setdate, &t);
tv.tv_sec = t;
tv.tv_usec = TT.nano/1000;
if (settimeofday(&tv, NULL) < 0) perror_msg("cannot set date");
}
puts_time(format_string, localtime(&t));
if (FLAG(u)) {
if (tz) setenv("TZ", tz, 1);
else unsetenv("TZ");
tzset();
}
if (CFG_TOYBOX_FREE && FLAG(I)) free(format_string);
} |
![]() |