2000-02-25 10:15:17 +00:00
|
|
|
// This may look like C code, but it is really -*- C++ -*-
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// The Goldware Library
|
|
|
|
// Copyright (C) 1990-1999 Odinn Sorensen
|
|
|
|
// Copyright (C) 1999-2000 Alexander S. Aganichev
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Library General Public
|
|
|
|
// License as published by the Free Software Foundation; either
|
|
|
|
// version 2 of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Library General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Library General Public
|
|
|
|
// License along with this program; if not, write to the Free
|
|
|
|
// Software Foundation, Inc., 59 Temple Place, Suite 330, Boston,
|
|
|
|
// MA 02111-1307, USA
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// $Id$
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// Time utility functions.
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include <cstdarg>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <gmemall.h>
|
|
|
|
#include <gstrall.h>
|
|
|
|
#include <gtimall.h>
|
|
|
|
#include <gutlmisc.h>
|
|
|
|
#include <gutlmtsk.h>
|
|
|
|
|
|
|
|
#ifdef __OS2__
|
|
|
|
#define INCL_BASE
|
|
|
|
#include <os2.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __WIN32__
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
const char* gmonths[] = {
|
|
|
|
"ERR",
|
|
|
|
"Jan", "Feb", "Mar",
|
|
|
|
"Apr", "May", "Jun",
|
|
|
|
"Jul", "Aug", "Sep",
|
|
|
|
"Oct", "Nov", "Dec",
|
|
|
|
"ERR"
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// Returns current timezone offset based on TZ environment variable.
|
|
|
|
|
2006-04-21 19:57:30 +00:00
|
|
|
int tzoffset()
|
|
|
|
{
|
|
|
|
time32_t t1 = gtime(NULL);
|
|
|
|
struct tm tp; ggmtime(&tp, &t1);
|
|
|
|
tp.tm_isdst = -1;
|
|
|
|
time32_t t2 = gmktime(&tp);
|
2002-06-06 07:12:34 +00:00
|
|
|
int dt = (int)(t1 - t2);
|
|
|
|
return (dt / 3600) * 100 + (dt % 3600) / 60;
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
char* __gampm[2] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"AM", (char *)"PM"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* __gsweekday[7] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"Sun", (char *)"Mon", (char *)"Tue", (char *)"Wed", (char *)"Thu", (char *)"Fri", (char *)"Sat"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* __glweekday[7] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"Sunday", (char *)"Monday", (char *)"Tuesday", (char *)"Wednesday", (char *)"Thursday", (char *)"Friday", (char *)"Saturday"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* __gsmonth[12] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"Jan", (char *)"Feb", (char *)"Mar", (char *)"Apr", (char *)"May", (char *)"Jun",
|
|
|
|
(char *)"Jul", (char *)"Aug", (char *)"Sep", (char *)"Oct", (char *)"Nov", (char *)"Dec"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* __glmonth[12] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"January", (char *)"February", (char *)"March", (char *)"April", (char *)"May", (char *)"June",
|
|
|
|
(char *)"July", (char *)"August", (char *)"September", (char *)"October", (char *)"November", (char *)"December"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* gampm[2] = {
|
2018-07-12 21:36:37 +03:00
|
|
|
(char *)"AM", (char *)"PM"
|
2000-02-25 10:15:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
char* gsweekday[7] = {
|
|
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
char* glweekday[7] = {
|
|
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
char* gsmonth[12] = {
|
|
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
char* glmonth[12] = {
|
|
|
|
NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// written 6 september 1989 by jim nutt
|
|
|
|
// released into the public domain by jim nutt
|
|
|
|
// modified 21-Oct-89 by Rob Duff
|
|
|
|
|
|
|
|
static char buf[26];
|
|
|
|
static char format[] = "%?";
|
|
|
|
|
2011-04-10 06:26:02 +00:00
|
|
|
static int gpow[5] = { 1, 10, 100, 1000, 10000 };
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// static void strfmt(char *str, char *fmt);
|
|
|
|
// simple sprintf for strftime
|
|
|
|
// each format descriptor is of the form %n
|
|
|
|
// where n goes from zero to four
|
|
|
|
// 0 -- string %s
|
|
|
|
// 1..4 -- int %?.?d
|
|
|
|
|
|
|
|
static void strfmt(char *str, const char *fmt, ...) {
|
|
|
|
|
|
|
|
int ival, ilen;
|
|
|
|
char *sval;
|
|
|
|
va_list vp;
|
|
|
|
|
|
|
|
va_start(vp, fmt);
|
|
|
|
while(*fmt) {
|
|
|
|
if(*fmt++ == '%') {
|
|
|
|
ilen = *fmt++ - '0';
|
|
|
|
if(ilen == 0) // zero means string arg
|
|
|
|
{
|
|
|
|
sval = va_arg(vp, char*);
|
|
|
|
while(*sval)
|
|
|
|
*str++ = *sval++;
|
|
|
|
}
|
|
|
|
else { // always leading zeros
|
|
|
|
if(ilen == ('-' - '0')) {
|
|
|
|
ilen = *fmt++ - '0';
|
|
|
|
ival = va_arg(vp, int);
|
|
|
|
bool padding = true;
|
|
|
|
while(ilen) {
|
2011-04-10 06:26:02 +00:00
|
|
|
ival %= gpow[ilen];
|
|
|
|
int cval = ival / gpow[ilen-1];
|
2001-03-07 05:46:58 +00:00
|
|
|
if(cval)
|
2000-02-25 10:15:17 +00:00
|
|
|
padding = false;
|
|
|
|
if(--ilen and padding)
|
2001-03-07 05:46:58 +00:00
|
|
|
cval = ' ' - '0';
|
|
|
|
*str++ = (char)('0' + cval);
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ival = va_arg(vp, int);
|
|
|
|
while(ilen) {
|
2011-04-10 06:26:02 +00:00
|
|
|
ival %= gpow[ilen--];
|
|
|
|
*str++ = (char)('0' + ival / gpow[ilen]);
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else *str++ = fmt[-1];
|
|
|
|
}
|
|
|
|
*str = '\0';
|
|
|
|
va_end(vp);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// In differ to ANSI C strftime this function supports reloading of
|
|
|
|
// field names
|
2011-02-21 13:09:11 +00:00
|
|
|
// Return NULL if output truncated, otherwize return s.
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
#define setvar(internal,external,field) \
|
|
|
|
char **internal = external; \
|
|
|
|
if(internal[field] == NULL) internal = __##external;
|
|
|
|
|
|
|
|
char *strftimei(char *s, size_t maxs, const char *f, const struct tm *t)
|
|
|
|
{
|
|
|
|
char *p, *q, *r;
|
|
|
|
|
|
|
|
setvar(aday, gsweekday, t->tm_wday);
|
|
|
|
setvar(day, glweekday, t->tm_wday);
|
|
|
|
setvar(amonth, gsmonth, t->tm_mon);
|
|
|
|
setvar(month, glmonth, t->tm_mon);
|
|
|
|
|
|
|
|
p = s;
|
|
|
|
q = s + maxs - 1;
|
|
|
|
while((*f != NUL)) {
|
|
|
|
if(*f++ == '%') {
|
|
|
|
r = buf;
|
|
|
|
switch(*f++) {
|
|
|
|
|
|
|
|
case '%':
|
|
|
|
format[1] = NUL;
|
|
|
|
r = format;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a':
|
|
|
|
r = aday[t->tm_wday];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'A':
|
|
|
|
r = day[t->tm_wday];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'b':
|
|
|
|
r = amonth[t->tm_mon];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'B':
|
|
|
|
r = month[t->tm_mon];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'C':
|
|
|
|
strfmt(r, "%0 %0 %-2 %2:%2:%2 %4",
|
|
|
|
aday[t->tm_wday], amonth[t->tm_mon],
|
|
|
|
t->tm_mday,t->tm_hour, t->tm_min,
|
|
|
|
t->tm_sec, t->tm_year+1900);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'e':
|
|
|
|
strfmt(r, "%-2", t->tm_mday);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'E':
|
|
|
|
sprintf(r, "%u", (uint)t->tm_mday);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'p':
|
|
|
|
r = gampm[(t->tm_hour>11)?1:0];
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
format[1] = f[-1];
|
|
|
|
if(strftime(r, 26, format, t) == 0) {
|
|
|
|
buf[0] = '%'; // reconstruct the format
|
|
|
|
buf[1] = f[-1];
|
|
|
|
buf[2] = '\0';
|
|
|
|
if(buf[1] == 0)
|
|
|
|
f--; // back up if at end of string
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
while(*r) {
|
|
|
|
if(p == q) {
|
|
|
|
*q = '\0';
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*p++ = *r++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if(p == q) {
|
|
|
|
*q = '\0';
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*p++ = f[-1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*p = '\0';
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
// Convert string-month to integer
|
|
|
|
|
|
|
|
int str2mon(const char* ptr) {
|
|
|
|
|
|
|
|
int mon;
|
|
|
|
|
2005-10-31 22:18:26 +00:00
|
|
|
switch(g_toupper(*ptr)) {
|
2000-02-25 10:15:17 +00:00
|
|
|
case 'A':
|
2005-10-31 22:18:26 +00:00
|
|
|
if(g_toupper(ptr[1]) == 'P')
|
2000-02-25 10:15:17 +00:00
|
|
|
mon = 4;
|
|
|
|
else
|
|
|
|
mon = 8;
|
|
|
|
break;
|
|
|
|
case 'D':
|
|
|
|
mon = 12;
|
|
|
|
break;
|
|
|
|
case 'F':
|
|
|
|
mon = 2;
|
|
|
|
break;
|
|
|
|
case 'J':
|
2005-10-31 22:18:26 +00:00
|
|
|
if(g_toupper(ptr[1]) == 'A')
|
2000-02-25 10:15:17 +00:00
|
|
|
mon = 1;
|
2005-10-31 22:18:26 +00:00
|
|
|
else if(g_toupper(ptr[2]) == 'L')
|
2000-02-25 10:15:17 +00:00
|
|
|
mon = 7;
|
|
|
|
else
|
|
|
|
mon = 6;
|
|
|
|
break;
|
|
|
|
case 'M':
|
2005-10-31 22:18:26 +00:00
|
|
|
if(g_toupper(ptr[2]) == 'R')
|
2000-02-25 10:15:17 +00:00
|
|
|
mon = 3;
|
|
|
|
else
|
|
|
|
mon = 5;
|
|
|
|
break;
|
|
|
|
case 'N':
|
|
|
|
mon = 11;
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
mon = 10;
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
mon = 9;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
mon = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mon;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
2005-10-20 21:10:42 +00:00
|
|
|
time32_t FTimeToTime(FTime* __ftime, struct tm* __tm) {
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
struct tm _tm;
|
2005-10-20 21:41:32 +00:00
|
|
|
uint32_t _time = 0;
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
if(__tm == NULL)
|
|
|
|
__tm = &_tm;
|
|
|
|
|
|
|
|
// Only try to convert a valid date
|
|
|
|
if(__ftime->ft_year >= 5) { // FidoNet standards didn't exist before 1985
|
|
|
|
if((__ftime->ft_day >= 1) and (__ftime->ft_day <= 31)) {
|
|
|
|
if((__ftime->ft_month >= 1) and (__ftime->ft_month <= 12)) {
|
|
|
|
|
|
|
|
__tm->tm_year = __ftime->ft_year + 80;
|
|
|
|
__tm->tm_mon = __ftime->ft_month - 1;
|
|
|
|
__tm->tm_mday = __ftime->ft_day;
|
|
|
|
__tm->tm_hour = __ftime->ft_hour;
|
|
|
|
__tm->tm_min = __ftime->ft_min;
|
|
|
|
__tm->tm_sec = __ftime->ft_tsec * 2;
|
|
|
|
__tm->tm_isdst = -1;
|
|
|
|
|
2006-04-21 19:57:30 +00:00
|
|
|
time32_t a = gmktime(__tm);
|
|
|
|
struct tm tp; ggmtime(&tp, &a);
|
|
|
|
tp.tm_isdst = -1;
|
|
|
|
time32_t b = gmktime(&tp);
|
2000-02-25 10:15:17 +00:00
|
|
|
_time = a + a - b;
|
|
|
|
|
2005-10-20 21:41:32 +00:00
|
|
|
if(_time == (uint32_t)0xFFFFFFFFL)
|
2000-02-25 10:15:17 +00:00
|
|
|
_time = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return _time;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
2005-10-20 21:10:42 +00:00
|
|
|
FTime TimeToFTime(time32_t __time) {
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
FTime _ft;
|
|
|
|
memset(&_ft, 0, sizeof(FTime));
|
|
|
|
|
2006-04-21 19:57:30 +00:00
|
|
|
if (__time)
|
|
|
|
{
|
|
|
|
struct tm _tmp; ggmtime(&_tmp, &__time);
|
|
|
|
_ft.ft_year = (word)(_tmp.tm_year - 80);
|
|
|
|
_ft.ft_month = (word)(_tmp.tm_mon + 1);
|
|
|
|
_ft.ft_day = (word)(_tmp.tm_mday);
|
|
|
|
_ft.ft_hour = (word)(_tmp.tm_hour);
|
|
|
|
_ft.ft_min = (word)(_tmp.tm_min);
|
|
|
|
_ft.ft_tsec = (word)(_tmp.tm_sec / 2);
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return _ft;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
2005-10-20 21:10:42 +00:00
|
|
|
time32_t FidoTimeToUnix(char* ptr) {
|
2000-02-25 10:15:17 +00:00
|
|
|
|
|
|
|
bool date_ok = false;
|
|
|
|
int year=0, month=0, day=0;
|
|
|
|
int hour=0, minute=0, second=0;
|
|
|
|
|
|
|
|
ptr = strskip_wht(ptr);
|
|
|
|
if(not isdigit(*ptr)) {
|
|
|
|
// Skip past weekday string (SEA format)
|
|
|
|
ptr = strskip_wht(strskip_txt(ptr));
|
|
|
|
}
|
|
|
|
if(*ptr) {
|
|
|
|
if(isdigit(*ptr)) {
|
|
|
|
day = atoi(ptr);
|
|
|
|
ptr = strskip_wht(strskip_txt(ptr));
|
2005-10-31 22:18:26 +00:00
|
|
|
if(g_isalpha(*ptr)) {
|
2000-02-25 10:15:17 +00:00
|
|
|
month = str2mon(ptr);
|
|
|
|
if(month) {
|
|
|
|
ptr = strskip_wht(strskip_txt(ptr));
|
|
|
|
if(isdigit(*ptr)) {
|
|
|
|
year = atoi(ptr);
|
|
|
|
ptr = strskip_wht(strskip_txt(ptr));
|
|
|
|
if(isdigit(*ptr)) {
|
|
|
|
hour = atoi(ptr);
|
|
|
|
ptr = strskip_digits(ptr);
|
|
|
|
if(*ptr and isdigit(ptr[1])) {
|
|
|
|
minute = atoi(++ptr);
|
|
|
|
date_ok = true;
|
|
|
|
// The seconds part is only in the FTS-1 format
|
|
|
|
ptr = strskip_digits(ptr);
|
|
|
|
if(*ptr and isdigit(ptr[1])) {
|
|
|
|
second = atoi(++ptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert datetime to UNIX timestamp
|
|
|
|
if(date_ok) {
|
|
|
|
struct tm t;
|
|
|
|
t.tm_year = (year < 80) ? (year+100) : year;
|
|
|
|
t.tm_mon = month - 1;
|
|
|
|
t.tm_mday = day;
|
|
|
|
t.tm_hour = hour;
|
|
|
|
t.tm_min = minute;
|
|
|
|
t.tm_sec = second;
|
|
|
|
t.tm_isdst = -1;
|
2006-04-21 19:57:30 +00:00
|
|
|
time32_t a = gmktime(&t);
|
|
|
|
struct tm tp; ggmtime(&tp, &a);
|
|
|
|
tp.tm_isdst = -1;
|
|
|
|
time32_t b = gmktime(&tp);
|
2000-02-25 10:15:17 +00:00
|
|
|
return a + a - b;
|
|
|
|
}
|
2005-10-20 21:41:32 +00:00
|
|
|
return (uint32_t)-1;
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
2006-04-21 19:57:30 +00:00
|
|
|
char* TimeToStr(char* buf, time32_t t)
|
|
|
|
{
|
|
|
|
struct tm tm; ggmtime(&tm, &t);
|
|
|
|
return strftimei(buf, 20, "%Y-%m-%d %H:%M:%S", &tm);
|
2000-02-25 10:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
char* FTimeToStr(char* buf, FTime& t) {
|
|
|
|
|
|
|
|
sprintf(buf, "%04u-%02u-%02u %02u:%02u:%02u",
|
|
|
|
t.ft_year+1980, t.ft_month, t.ft_day,
|
|
|
|
t.ft_hour, t.ft_min, t.ft_tsec*2
|
|
|
|
);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
const char* gfiletime::c_str(char* buf) {
|
|
|
|
|
|
|
|
sprintf(buf, "%04u-%02u-%02u %02u:%02u:%02u",
|
|
|
|
ft_year+1980, ft_month, ft_day,
|
|
|
|
ft_hour, ft_min, ft_tsec*2
|
|
|
|
);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|
|
|
|
|
|
|
|
const char* gopustime::c_str(char* buf) {
|
|
|
|
|
|
|
|
sprintf(buf, "%04u-%02u-%02u %02u:%02u:%02u",
|
|
|
|
ft_year+1980, ft_month, ft_day,
|
|
|
|
ft_hour, ft_min, ft_tsec*2
|
|
|
|
);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ------------------------------------------------------------------
|