|
|
@ -1,5 +1,5 @@ |
|
|
|
/*
|
|
|
|
** $Id: loslib.c,v 1.63 2016/02/09 12:16:11 roberto Exp roberto $ |
|
|
|
** $Id: loslib.c,v 1.64 2016/04/18 13:06:55 roberto Exp $ |
|
|
|
** Standard Operating System library |
|
|
|
** See Copyright Notice in lua.h |
|
|
|
*/ |
|
|
@ -30,16 +30,16 @@ |
|
|
|
*/ |
|
|
|
#if !defined(LUA_STRFTIMEOPTIONS) /* { */ |
|
|
|
|
|
|
|
/* options for ANSI C 89 */ |
|
|
|
/* options for ANSI C 89 (only 1-char options) */ |
|
|
|
#define L_STRFTIMEC89 "aAbBcdHIjmMpSUwWxXyYZ%" |
|
|
|
|
|
|
|
/* options for ISO C 99 and POSIX */ |
|
|
|
#define L_STRFTIMEC99 "aAbBcCdDeFgGhHIjmMnprRStTuUVwWxXyYzZ%" \ |
|
|
|
"||" "EcECExEXEyEY" "OdOeOHOIOmOMOSOuOUOVOwOWOy" |
|
|
|
"||" "EcECExEXEyEY" "OdOeOHOIOmOMOSOuOUOVOwOWOy" /* two-char options */ |
|
|
|
|
|
|
|
/* options for Windows */ |
|
|
|
#define L_STRFTIMEWIN "aAbBcdHIjmMpSUwWxXyYzZ%" \ |
|
|
|
"||" "#c#x#d#H#I#j#m#M#S#U#w#W#y#Y" |
|
|
|
"||" "#c#x#d#H#I#j#m#M#S#U#w#W#y#Y" /* two-char options */ |
|
|
|
|
|
|
|
#if defined(LUA_USE_WINDOWS) |
|
|
|
#define LUA_STRFTIMEOPTIONS L_STRFTIMEWIN |
|
|
@ -257,12 +257,13 @@ static int getfield (lua_State *L, const char *key, int d, int delta) { |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
static const char *checkoption (lua_State *L, const char *conv, char *buff) { |
|
|
|
const char *option; |
|
|
|
int oplen = 1; |
|
|
|
for (option = LUA_STRFTIMEOPTIONS; *option != '\0'; option += oplen) { |
|
|
|
static const char *checkoption (lua_State *L, const char *conv, |
|
|
|
ptrdiff_t convlen, char *buff) { |
|
|
|
const char *option = LUA_STRFTIMEOPTIONS; |
|
|
|
int oplen = 1; /* length of options being checked */ |
|
|
|
for (; *option != '\0' && oplen <= convlen; option += oplen) { |
|
|
|
if (*option == '|') /* next block? */ |
|
|
|
oplen++; /* next length */ |
|
|
|
oplen++; /* will check options with next length (+1) */ |
|
|
|
else if (memcmp(conv, option, oplen) == 0) { /* match? */ |
|
|
|
memcpy(buff, conv, oplen); /* copy valid option to buffer */ |
|
|
|
buff[oplen] = '\0'; |
|
|
@ -280,8 +281,10 @@ static const char *checkoption (lua_State *L, const char *conv, char *buff) { |
|
|
|
|
|
|
|
|
|
|
|
static int os_date (lua_State *L) { |
|
|
|
const char *s = luaL_optstring(L, 1, "%c"); |
|
|
|
size_t slen; |
|
|
|
const char *s = luaL_optlstring(L, 1, "%c", &slen); |
|
|
|
time_t t = luaL_opt(L, l_checktime, 2, time(NULL)); |
|
|
|
const char *se = s + slen; /* 's' end */ |
|
|
|
struct tm tmr, *stm; |
|
|
|
if (*s == '!') { /* UTC? */ |
|
|
|
stm = l_gmtime(&t, &tmr); |
|
|
@ -300,13 +303,14 @@ static int os_date (lua_State *L) { |
|
|
|
luaL_Buffer b; |
|
|
|
cc[0] = '%'; |
|
|
|
luaL_buffinit(L, &b); |
|
|
|
while (*s) { |
|
|
|
while (s < se) { |
|
|
|
if (*s != '%') /* not a conversion specifier? */ |
|
|
|
luaL_addchar(&b, *s++); |
|
|
|
else { |
|
|
|
size_t reslen; |
|
|
|
char *buff = luaL_prepbuffsize(&b, SIZETIMEFMT); |
|
|
|
s = checkoption(L, s + 1, cc + 1); /* copy specifier to 'cc' */ |
|
|
|
s++; /* skip '%' */ |
|
|
|
s = checkoption(L, s, se - s, cc + 1); /* copy specifier to 'cc' */ |
|
|
|
reslen = strftime(buff, SIZETIMEFMT, cc, stm); |
|
|
|
luaL_addsize(&b, reslen); |
|
|
|
} |
|
|
|