2019-01-24 18:03:27 +00:00
|
|
|
/*
|
|
|
|
* $Id$
|
|
|
|
*
|
2020-02-20 15:35:16 +00:00
|
|
|
Copyright (c) 2016-2020 Chung, Hyung-Hwan. All rights reserved.
|
2019-01-24 18:03:27 +00:00
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions
|
|
|
|
are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
|
|
|
|
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WAfRRANTIES
|
|
|
|
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
2020-04-30 02:28:09 +00:00
|
|
|
#include "sys-prv.h"
|
2019-01-24 18:03:27 +00:00
|
|
|
|
2019-01-27 02:09:22 +00:00
|
|
|
#if defined(_WIN32)
|
|
|
|
# include <windows.h>
|
|
|
|
# include <time.h>
|
|
|
|
# include <errno.h>
|
|
|
|
# include <io.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
|
|
|
|
#elif defined(__OS2__)
|
|
|
|
# define INCL_DOSDATETIME
|
|
|
|
# include <os2.h>
|
|
|
|
# include <time.h>
|
|
|
|
# include <errno.h>
|
|
|
|
# include <io.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
|
|
|
|
#elif defined(__DOS__)
|
|
|
|
# include <dos.h>
|
|
|
|
# include <time.h>
|
|
|
|
# include <errno.h>
|
|
|
|
# include <io.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
|
|
|
|
#elif defined(macintosh)
|
|
|
|
# include <Types.h>
|
|
|
|
# include <OSUtils.h>
|
|
|
|
# include <Timer.h>
|
|
|
|
# include <MacErrors.h>
|
|
|
|
|
|
|
|
/* TODO: a lot to do */
|
|
|
|
|
|
|
|
#elif defined(vms) || defined(__vms)
|
|
|
|
# define __NEW_STARLET 1
|
|
|
|
# include <starlet.h> /* (SYS$...) */
|
|
|
|
# include <ssdef.h> /* (SS$...) */
|
|
|
|
# include <lib$routines.h> /* (lib$...) */
|
|
|
|
|
|
|
|
/* TODO: a lot to do */
|
|
|
|
|
|
|
|
#else
|
|
|
|
# include <sys/types.h>
|
|
|
|
# include <unistd.h>
|
|
|
|
# include <fcntl.h>
|
|
|
|
# include <errno.h>
|
|
|
|
|
|
|
|
# if defined(HAVE_TIME_H)
|
|
|
|
# include <time.h>
|
|
|
|
# endif
|
|
|
|
# if defined(HAVE_SYS_TIME_H)
|
|
|
|
# include <sys/time.h>
|
|
|
|
# endif
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2019-01-30 10:18:58 +00:00
|
|
|
#include <stdio.h> /* for sprintf */
|
|
|
|
|
|
|
|
|
2019-01-27 02:09:22 +00:00
|
|
|
enum logfd_flag_t
|
|
|
|
{
|
|
|
|
LOGFD_TTY = (1 << 0),
|
|
|
|
LOGFD_OPENED_HERE = (1 << 1)
|
|
|
|
};
|
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
static int write_all (mio_t* mio, int fd, mio_bitmask_t mask, const mio_bch_t* ptr, mio_oow_t len)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mio->option.log_writer) return mio->option.log_writer(mio, mask, ptr, len);
|
|
|
|
|
|
|
|
if (MIO_UNLIKELY(fd <= -1)) return 0; /* MIO_FEATURE_LOG_WRITER is off but mio->option.log_write is not set */
|
|
|
|
|
2019-01-24 18:03:27 +00:00
|
|
|
while (len > 0)
|
|
|
|
{
|
|
|
|
mio_ooi_t wr;
|
|
|
|
|
|
|
|
wr = write(fd, ptr, len);
|
|
|
|
|
|
|
|
if (wr <= -1)
|
|
|
|
{
|
|
|
|
#if defined(EAGAIN) && defined(EWOULDBLOCK) && (EAGAIN == EWOULDBLOCK)
|
|
|
|
if (errno == EAGAIN) continue;
|
|
|
|
#else
|
|
|
|
#if defined(EAGAIN)
|
|
|
|
if (errno == EAGAIN) continue;
|
|
|
|
#elif defined(EWOULDBLOCK)
|
|
|
|
if (errno == EWOULDBLOCK) continue;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(EINTR)
|
|
|
|
/* TODO: would this interfere with non-blocking nature of this VM? */
|
|
|
|
if (errno == EINTR) continue;
|
|
|
|
#endif
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
ptr += wr;
|
|
|
|
len -= wr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
static int write_log (mio_t* mio, int fd, mio_bitmask_t mask, const mio_bch_t* ptr, mio_oow_t len)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2019-01-29 16:57:16 +00:00
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
2019-01-24 18:03:27 +00:00
|
|
|
|
|
|
|
while (len > 0)
|
|
|
|
{
|
2019-01-28 08:13:06 +00:00
|
|
|
if (log->out.len > 0)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
|
|
|
mio_oow_t rcapa, cplen;
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
rcapa = MIO_COUNTOF(log->out.buf) - log->out.len;
|
2019-01-24 18:03:27 +00:00
|
|
|
cplen = (len >= rcapa)? rcapa: len;
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
MIO_MEMCPY (&log->out.buf[log->out.len], ptr, cplen);
|
|
|
|
log->out.len += cplen;
|
2019-01-24 18:03:27 +00:00
|
|
|
ptr += cplen;
|
|
|
|
len -= cplen;
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
if (log->out.len >= MIO_COUNTOF(log->out.buf))
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
|
|
|
int n;
|
2021-07-19 04:57:30 +00:00
|
|
|
n = write_all(mio, fd, mask, log->out.buf, log->out.len);
|
2019-01-28 08:13:06 +00:00
|
|
|
log->out.len = 0;
|
2019-01-24 18:03:27 +00:00
|
|
|
if (n <= -1) return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mio_oow_t rcapa;
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
rcapa = MIO_COUNTOF(log->out.buf);
|
2019-01-24 18:03:27 +00:00
|
|
|
if (len >= rcapa)
|
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (write_all(mio, fd, mask, ptr, rcapa) <= -1) return -1;
|
2019-01-24 18:03:27 +00:00
|
|
|
ptr += rcapa;
|
|
|
|
len -= rcapa;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-01-28 08:13:06 +00:00
|
|
|
MIO_MEMCPY (log->out.buf, ptr, len);
|
|
|
|
log->out.len += len;
|
2019-01-24 18:03:27 +00:00
|
|
|
ptr += len;
|
|
|
|
len -= len;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
static void flush_log (mio_t* mio, int fd, mio_bitmask_t mask)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2019-01-29 16:57:16 +00:00
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
2019-01-28 08:13:06 +00:00
|
|
|
if (log->out.len > 0)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
write_all (mio, fd, mask, log->out.buf, log->out.len);
|
2019-01-28 08:13:06 +00:00
|
|
|
log->out.len = 0;
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void mio_sys_writelog (mio_t* mio, mio_bitmask_t mask, const mio_ooch_t* msg, mio_oow_t len)
|
|
|
|
{
|
2019-01-29 16:57:16 +00:00
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
2019-01-24 18:03:27 +00:00
|
|
|
mio_bch_t buf[256];
|
|
|
|
mio_oow_t ucslen, bcslen, msgidx;
|
2021-07-19 04:57:30 +00:00
|
|
|
int n, logfd = -1;
|
2019-01-24 18:03:27 +00:00
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mio->_features & MIO_FEATURE_LOG_WRITER)
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mask & MIO_LOG_STDERR)
|
|
|
|
{
|
|
|
|
logfd = 2;
|
|
|
|
}
|
2019-01-24 18:03:27 +00:00
|
|
|
else
|
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mask & MIO_LOG_STDOUT) logfd = 1;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
logfd = log->fd;
|
|
|
|
if (logfd <= -1) return;
|
|
|
|
}
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: beautify the log message.
|
|
|
|
* do classification based on mask. */
|
|
|
|
if (!(mask & (MIO_LOG_STDOUT | MIO_LOG_STDERR)))
|
|
|
|
{
|
|
|
|
time_t now;
|
|
|
|
char ts[32];
|
|
|
|
size_t tslen;
|
|
|
|
struct tm tm, *tmp;
|
|
|
|
|
|
|
|
now = time(MIO_NULL);
|
|
|
|
#if defined(_WIN32)
|
|
|
|
tmp = localtime(&now);
|
|
|
|
tslen = strftime(ts, sizeof(ts), "%Y-%m-%d %H:%M:%S %z ", tmp);
|
|
|
|
if (tslen == 0)
|
|
|
|
{
|
|
|
|
tslen = sprintf(ts, "%04d-%02d-%02d %02d:%02d:%02d ", tmp->tm_year + 1900, tmp->tm_mon + 1, tmp->tm_mday, tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
|
|
|
|
}
|
|
|
|
#elif defined(__OS2__)
|
|
|
|
#if defined(__WATCOMC__)
|
|
|
|
tmp = _localtime(&now, &tm);
|
|
|
|
#else
|
|
|
|
tmp = localtime(&now);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__BORLANDC__)
|
|
|
|
/* the borland compiler doesn't handle %z properly - it showed 00 all the time */
|
|
|
|
tslen = strftime(ts, sizeof(ts), "%Y-%m-%d %H:%M:%S %Z ", tmp);
|
|
|
|
#else
|
|
|
|
tslen = strftime(ts, sizeof(ts), "%Y-%m-%d %H:%M:%S %z ", tmp);
|
|
|
|
#endif
|
|
|
|
if (tslen == 0)
|
|
|
|
{
|
|
|
|
tslen = sprintf(ts, "%04d-%02d-%02d %02d:%02d:%02d ", tmp->tm_year + 1900, tmp->tm_mon + 1, tmp->tm_mday, tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif defined(__DOS__)
|
|
|
|
tmp = localtime(&now);
|
|
|
|
/* since i know that %z/%Z is not available in strftime, i switch to sprintf immediately */
|
|
|
|
tslen = sprintf(ts, "%04d-%02d-%02d %02d:%02d:%02d ", tmp->tm_year + 1900, tmp->tm_mon + 1, tmp->tm_mday, tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
|
|
|
|
#else
|
|
|
|
#if defined(HAVE_LOCALTIME_R)
|
|
|
|
tmp = localtime_r(&now, &tm);
|
|
|
|
#else
|
|
|
|
tmp = localtime(&now);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(HAVE_STRFTIME_SMALL_Z)
|
|
|
|
tslen = strftime(ts, sizeof(ts), "%Y-%m-%d %H:%M:%S %z ", tmp);
|
|
|
|
#else
|
|
|
|
tslen = strftime(ts, sizeof(ts), "%Y-%m-%d %H:%M:%S %Z ", tmp);
|
|
|
|
#endif
|
|
|
|
if (tslen == 0)
|
|
|
|
{
|
|
|
|
tslen = sprintf(ts, "%04d-%02d-%02d %02d:%02d:%02d ", tmp->tm_year + 1900, tmp->tm_mon + 1, tmp->tm_mday, tmp->tm_hour, tmp->tm_min, tmp->tm_sec);
|
|
|
|
}
|
|
|
|
#endif
|
2021-07-19 04:57:30 +00:00
|
|
|
write_log (mio, logfd, mask, ts, tslen);
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
if (logfd == log->fd && (log->fd_flag & LOGFD_TTY))
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mask & MIO_LOG_FATAL) write_log (mio, logfd, mask, "\x1B[1;31m", 7);
|
|
|
|
else if (mask & MIO_LOG_ERROR) write_log (mio, logfd, mask, "\x1B[1;32m", 7);
|
|
|
|
else if (mask & MIO_LOG_WARN) write_log (mio, logfd, mask, "\x1B[1;33m", 7);
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#if defined(MIO_OOCH_IS_UCH)
|
|
|
|
msgidx = 0;
|
|
|
|
while (len > 0)
|
|
|
|
{
|
|
|
|
ucslen = len;
|
|
|
|
bcslen = MIO_COUNTOF(buf);
|
|
|
|
|
|
|
|
n = mio_convootobchars(mio, &msg[msgidx], &ucslen, buf, &bcslen);
|
|
|
|
if (n == 0 || n == -2)
|
|
|
|
{
|
|
|
|
/* n = 0:
|
|
|
|
* converted all successfully
|
|
|
|
* n == -2:
|
|
|
|
* buffer not sufficient. not all got converted yet.
|
|
|
|
* write what have been converted this round. */
|
|
|
|
|
|
|
|
MIO_ASSERT (mio, ucslen > 0); /* if this fails, the buffer size must be increased */
|
|
|
|
|
|
|
|
/* attempt to write all converted characters */
|
2021-07-19 04:57:30 +00:00
|
|
|
if (write_log(mio, logfd, mask, buf, bcslen) <= -1) break;
|
2019-01-24 18:03:27 +00:00
|
|
|
|
|
|
|
if (n == 0) break;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
msgidx += ucslen;
|
|
|
|
len -= ucslen;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (n <= -1)
|
|
|
|
{
|
|
|
|
/* conversion error */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2021-07-19 04:57:30 +00:00
|
|
|
write_log (mio, logfd, mask, msg, len);
|
2019-01-24 18:03:27 +00:00
|
|
|
#endif
|
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
if (logfd == log->fd && (log->fd_flag & LOGFD_TTY))
|
2019-01-24 18:03:27 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mask & (MIO_LOG_FATAL | MIO_LOG_ERROR | MIO_LOG_WARN)) write_log (mio, logfd, mask, "\x1B[0m", 4);
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
flush_log (mio, logfd, mask);
|
2019-01-24 18:03:27 +00:00
|
|
|
}
|
2019-01-27 02:09:22 +00:00
|
|
|
|
2019-01-28 08:13:06 +00:00
|
|
|
int mio_sys_initlog (mio_t* mio)
|
2019-01-27 02:09:22 +00:00
|
|
|
{
|
2019-01-29 16:57:16 +00:00
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
2019-01-28 08:13:06 +00:00
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mio->_features & MIO_FEATURE_LOG_WRITER)
|
2019-01-27 02:09:22 +00:00
|
|
|
{
|
2021-07-19 09:27:44 +00:00
|
|
|
log->fd = 2;
|
|
|
|
log->fd_flag = 0;
|
2019-01-27 02:09:22 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
log->fd = -1;
|
2019-01-27 02:09:22 +00:00
|
|
|
}
|
|
|
|
|
2021-07-17 15:17:07 +00:00
|
|
|
pthread_mutex_init (&log->mtx, MIO_NULL);
|
2019-01-28 08:13:06 +00:00
|
|
|
return 0;
|
2019-01-27 02:09:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void mio_sys_finilog (mio_t* mio)
|
|
|
|
{
|
2019-01-29 16:57:16 +00:00
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
2019-01-28 08:13:06 +00:00
|
|
|
|
2021-07-17 15:17:07 +00:00
|
|
|
pthread_mutex_destroy (&log->mtx);
|
|
|
|
|
2021-07-19 04:57:30 +00:00
|
|
|
if (mio->_features & MIO_FEATURE_LOG_WRITER)
|
2019-01-28 08:13:06 +00:00
|
|
|
{
|
2021-07-19 04:57:30 +00:00
|
|
|
if ((log->fd_flag & LOGFD_OPENED_HERE) && log->fd >= 0)
|
|
|
|
{
|
|
|
|
close (log->fd);
|
|
|
|
log->fd = -1;
|
|
|
|
log->fd_flag = 0;
|
|
|
|
}
|
2019-01-28 08:13:06 +00:00
|
|
|
}
|
2019-01-27 02:09:22 +00:00
|
|
|
}
|
2021-07-17 15:17:07 +00:00
|
|
|
|
2021-07-19 09:27:44 +00:00
|
|
|
void mio_sys_resetlog (mio_t* mio)
|
|
|
|
{
|
|
|
|
mio_sys_log_t* log = &mio->sysdep->log;
|
|
|
|
|
|
|
|
if (mio->_features & MIO_FEATURE_LOG_WRITER)
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
int fd_flag = 0;
|
|
|
|
int kill_fd = -1;
|
|
|
|
|
|
|
|
fd = open(mio->option.log_target_b, O_CREAT | O_WRONLY | O_APPEND , 0644);
|
|
|
|
if (fd == -1)
|
|
|
|
{
|
|
|
|
fd = 2;
|
|
|
|
fd_flag = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fd_flag |= LOGFD_OPENED_HERE;
|
|
|
|
#if defined(HAVE_ISATTY)
|
|
|
|
if (isatty(fd)) fd_flag |= LOGFD_TTY;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((log->fd_flag & LOGFD_OPENED_HERE) && log->fd >= 0) kill_fd = log->fd;
|
|
|
|
|
|
|
|
/* swap - risk of race condition depite this late swap */
|
|
|
|
log->fd = fd;
|
|
|
|
log->fd_flag = fd_flag;
|
|
|
|
|
|
|
|
if (kill_fd >= 0) close (kill_fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-17 15:17:07 +00:00
|
|
|
void mio_sys_locklog (mio_t* mio)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock (&mio->sysdep->log.mtx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void mio_sys_unlocklog (mio_t* mio)
|
|
|
|
{
|
|
|
|
pthread_mutex_unlock (&mio->sysdep->log.mtx);
|
|
|
|
}
|