2018-03-12 10:39:13 +00:00
|
|
|
/*
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
Copyright (c) 2016-2018 Chung, Hyung-Hwan. All rights reserved.
|
|
|
|
|
|
|
|
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 WARRANTIES
|
|
|
|
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
|
2018-03-13 10:20:33 +00:00
|
|
|
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.
|
|
|
|
*/
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#include "hcl-s.h"
|
|
|
|
#include "hcl-opt.h"
|
|
|
|
#include "hcl-utl.h"
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <limits.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <locale.h>
|
|
|
|
#include <assert.h>
|
2018-03-12 10:39:13 +00:00
|
|
|
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#if defined(HAVE_TIME_H)
|
|
|
|
# include <time.h>
|
|
|
|
#endif
|
|
|
|
#if defined(HAVE_SYS_TIME_H)
|
|
|
|
# include <sys/time.h>
|
|
|
|
#endif
|
|
|
|
#if defined(HAVE_SIGNAL_H)
|
|
|
|
# include <signal.h>
|
|
|
|
#endif
|
|
|
|
#if defined(HAVE_SYS_MMAN_H)
|
|
|
|
# include <sys/mman.h>
|
|
|
|
#endif
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <netinet/in.h>
|
|
|
|
#include <arpa/inet.h>
|
2018-03-12 10:39:13 +00:00
|
|
|
|
|
|
|
/* ========================================================================= */
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
typedef struct server_xtn_t server_xtn_t;
|
|
|
|
struct xtn_t
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
int logfd;
|
|
|
|
unsigned int logmask;
|
|
|
|
int logfd_istty;
|
|
|
|
};
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
/* ========================================================================= */
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static hcl_server_t* g_server = HCL_NULL;
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
/* ========================================================================= */
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
typedef void (*signal_handler_t) (int, siginfo_t*, void*);
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static void handle_sigint (int sig, siginfo_t* siginfo, void* ctx)
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
if (g_server) hcl_server_stop (g_server);
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static void set_signal (int sig, signal_handler_t handler)
|
2018-03-12 16:45:42 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
struct sigaction sa;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
memset (&sa, 0, sizeof(sa));
|
|
|
|
/*sa.sa_handler = handler;*/
|
|
|
|
sa.sa_flags = SA_SIGINFO;
|
|
|
|
sa.sa_sigaction = handler;
|
|
|
|
sigemptyset (&sa.sa_mask);
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
sigaction (sig, &sa, NULL);
|
2018-03-12 16:45:42 +00:00
|
|
|
}
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static void set_signal_to_ignore (int sig)
|
2018-03-12 16:45:42 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
struct sigaction sa;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
memset (&sa, 0, sizeof(sa));
|
|
|
|
sa.sa_handler = SIG_IGN;
|
|
|
|
sa.sa_flags = 0;
|
|
|
|
sigemptyset (&sa.sa_mask);
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
sigaction (sig, &sa, NULL);
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static void set_signal_to_default (int sig)
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
struct sigaction sa;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
memset (&sa, 0, sizeof(sa));
|
|
|
|
sa.sa_handler = SIG_DFL;
|
|
|
|
sa.sa_flags = 0;
|
|
|
|
sigemptyset (&sa.sa_mask);
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
sigaction (sig, &sa, NULL);
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
/* ========================================================================= */
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
static int handle_logopt (hcl_server_t* server, const hcl_bch_t* str)
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_bch_t* xstr = (hcl_bch_t*)str;
|
|
|
|
hcl_bch_t* cm, * flt;
|
|
|
|
unsigned int logmask;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_server_getoption (server, HCL_SERVER_LOG_MASK, &logmask);
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
cm = strchr(xstr, ',');
|
|
|
|
if (cm)
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
/* i duplicate this string for open() below as open() doesn't
|
|
|
|
* accept a length-bounded string */
|
|
|
|
xstr = strdup (str);
|
|
|
|
if (!xstr)
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
fprintf (stderr, "ERROR: out of memory in duplicating %s\n", str);
|
2018-03-12 10:39:13 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
cm = strchr(xstr, ',');
|
|
|
|
*cm = '\0';
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
flt = cm + 1;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
cm = strchr(flt, ',');
|
|
|
|
if (cm) *cm = '\0';
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
if (hcl_compbcstr(flt, "app") == 0) logmask |= HCL_LOG_APP;
|
|
|
|
else if (hcl_compbcstr(flt, "compiler") == 0) logmask |= HCL_LOG_COMPILER;
|
|
|
|
else if (hcl_compbcstr(flt, "vm") == 0) logmask |= HCL_LOG_VM;
|
|
|
|
else if (hcl_compbcstr(flt, "mnemonic") == 0) logmask |= HCL_LOG_MNEMONIC;
|
|
|
|
else if (hcl_compbcstr(flt, "gc") == 0) logmask |= HCL_LOG_GC;
|
|
|
|
else if (hcl_compbcstr(flt, "ic") == 0) logmask |= HCL_LOG_IC;
|
|
|
|
else if (hcl_compbcstr(flt, "primitive") == 0) logmask |= HCL_LOG_PRIMITIVE;
|
|
|
|
|
|
|
|
else if (hcl_compbcstr(flt, "fatal") == 0) logmask |= HCL_LOG_FATAL;
|
|
|
|
else if (hcl_compbcstr(flt, "error") == 0) logmask |= HCL_LOG_ERROR;
|
|
|
|
else if (hcl_compbcstr(flt, "warn") == 0) logmask |= HCL_LOG_WARN;
|
|
|
|
else if (hcl_compbcstr(flt, "info") == 0) logmask |= HCL_LOG_INFO;
|
|
|
|
else if (hcl_compbcstr(flt, "debug") == 0) logmask |= HCL_LOG_DEBUG;
|
|
|
|
|
|
|
|
else if (hcl_compbcstr(flt, "fatal+") == 0) logmask |= HCL_LOG_FATAL;
|
|
|
|
else if (hcl_compbcstr(flt, "error+") == 0) logmask |= HCL_LOG_FATAL | HCL_LOG_ERROR;
|
|
|
|
else if (hcl_compbcstr(flt, "warn+") == 0) logmask |= HCL_LOG_FATAL | HCL_LOG_ERROR | HCL_LOG_WARN;
|
|
|
|
else if (hcl_compbcstr(flt, "info+") == 0) logmask |= HCL_LOG_FATAL | HCL_LOG_ERROR | HCL_LOG_WARN | HCL_LOG_INFO;
|
|
|
|
else if (hcl_compbcstr(flt, "debug+") == 0) logmask |= HCL_LOG_FATAL | HCL_LOG_ERROR | HCL_LOG_WARN | HCL_LOG_INFO | HCL_LOG_DEBUG;
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
else
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
fprintf (stderr, "ERROR: unknown log option value - %s\n", flt);
|
|
|
|
if (str != xstr) free (xstr);
|
|
|
|
return -1;
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
|
|
|
}
|
2018-03-13 10:20:33 +00:00
|
|
|
while (cm);
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
if (!(logmask & HCL_LOG_ALL_TYPES)) logmask |= HCL_LOG_ALL_TYPES; /* no types specified. force to all types */
|
|
|
|
if (!(logmask & HCL_LOG_ALL_LEVELS)) logmask |= HCL_LOG_ALL_LEVELS; /* no levels specified. force to all levels */
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
2018-03-13 10:20:33 +00:00
|
|
|
else
|
2018-03-13 02:02:25 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
logmask = HCL_LOG_ALL_LEVELS | HCL_LOG_ALL_TYPES;
|
2018-03-13 02:02:25 +00:00
|
|
|
}
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_server_setoption (server, HCL_SERVER_LOG_MASK, &logmask);
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
server->logfd = open(xstr, O_CREAT | O_WRONLY | O_APPEND , 0644);
|
|
|
|
if (server->logfd == -1)
|
2018-03-13 02:02:25 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
fprintf (stderr, "ERROR: cannot open a log file %s\n", xstr);
|
|
|
|
if (str != xstr) free (xstr);
|
|
|
|
return -1;
|
|
|
|
}
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#if defined(HAVE_ISATTY)
|
|
|
|
server->logfd_istty = isatty(server->logfd);
|
|
|
|
#endif
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
if (str != xstr) free (xstr);
|
|
|
|
return 0;
|
2018-03-12 16:45:42 +00:00
|
|
|
}
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
#if defined(HCL_BUILD_DEBUG)
|
|
|
|
static int parse_dbgopt (const char* str, unsigned int* dbgoptp)
|
2018-03-12 16:45:42 +00:00
|
|
|
{
|
2018-03-13 10:20:33 +00:00
|
|
|
const hcl_bch_t* cm, * flt;
|
|
|
|
hcl_oow_t len;
|
|
|
|
unsigned int dbgopt = 0;
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
cm = str - 1;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
flt = cm + 1;
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
cm = strchr(flt, ',');
|
|
|
|
len = cm? (cm - flt): strlen(flt);
|
|
|
|
if (strncasecmp(flt, "gc", len) == 0) dbgopt |= HCL_DEBUG_GC;
|
|
|
|
else if (strncasecmp (flt, "bigint", len) == 0) dbgopt |= HCL_DEBUG_BIGINT;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fprintf (stderr, "ERROR: unknown debug option value - %.*s\n", (int)len, flt);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (cm);
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
*dbgoptp = dbgopt;
|
|
|
|
return 0;
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|
2018-03-13 10:20:33 +00:00
|
|
|
#endif
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-12 16:45:42 +00:00
|
|
|
/* ========================================================================= */
|
|
|
|
|
2018-03-12 10:39:13 +00:00
|
|
|
#define MIN_MEMSIZE 512000ul
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
int main (int argc, char* argv[])
|
2018-03-12 10:39:13 +00:00
|
|
|
{
|
|
|
|
hcl_bci_t c;
|
|
|
|
static hcl_bopt_lng_t lopt[] =
|
|
|
|
{
|
|
|
|
{ ":log", 'l' },
|
|
|
|
{ ":memsize", 'm' },
|
|
|
|
{ "large-pages", '\0' },
|
2018-03-12 16:45:42 +00:00
|
|
|
#if defined(HCL_BUILD_DEBUG)
|
2018-03-12 10:39:13 +00:00
|
|
|
{ ":debug", '\0' }, /* NOTE: there is no short option for --debug */
|
2018-03-12 16:45:42 +00:00
|
|
|
#endif
|
2018-03-12 10:39:13 +00:00
|
|
|
{ HCL_NULL, '\0' }
|
|
|
|
};
|
|
|
|
static hcl_bopt_t opt =
|
|
|
|
{
|
|
|
|
"l:m:",
|
|
|
|
lopt
|
|
|
|
};
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_server_t* server;
|
|
|
|
server_xtn_t* xtn;
|
2018-03-12 16:45:42 +00:00
|
|
|
int n;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
|
|
|
const char* logopt = HCL_NULL;
|
|
|
|
hcl_oow_t memsize = MIN_MEMSIZE;
|
|
|
|
int large_pages = 0;
|
|
|
|
unsigned int dbgopt = 0;
|
2018-03-13 02:02:25 +00:00
|
|
|
unsigned int trait;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
{
|
|
|
|
print_usage:
|
|
|
|
fprintf (stderr, "Usage: %s bind-address:port\n", argv[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((c = hcl_getbopt (argc, argv, &opt)) != HCL_BCI_EOF)
|
|
|
|
{
|
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case 'l':
|
|
|
|
logopt = opt.arg;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'm':
|
|
|
|
memsize = strtoul(opt.arg, HCL_NULL, 0);
|
|
|
|
if (memsize <= MIN_MEMSIZE) memsize = MIN_MEMSIZE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '\0':
|
|
|
|
if (hcl_compbcstr(opt.lngopt, "large-pages") == 0)
|
|
|
|
{
|
|
|
|
large_pages = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#if defined(HCL_BUILD_DEBUG)
|
|
|
|
else if (hcl_compbcstr(opt.lngopt, "debug") == 0)
|
|
|
|
{
|
|
|
|
if (parse_dbgopt (opt.arg, &dbgopt) <= -1) return -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
goto print_usage;
|
|
|
|
|
|
|
|
case ':':
|
|
|
|
if (opt.lngopt)
|
|
|
|
fprintf (stderr, "bad argument for '%s'\n", opt.lngopt);
|
|
|
|
else
|
|
|
|
fprintf (stderr, "bad argument for '%c'\n", opt.opt);
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
default:
|
|
|
|
goto print_usage;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt.ind >= argc) goto print_usage;
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
server = hcl_server_open(HCL_SIZEOF(xtn_t), dbgopt);
|
2018-03-12 10:39:13 +00:00
|
|
|
if (!server)
|
|
|
|
{
|
|
|
|
fprintf (stderr, "cannot open server\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2018-03-13 10:20:33 +00:00
|
|
|
|
|
|
|
xtn = (server_xtn_t*)hcl_server_getxtn(server);
|
|
|
|
xtn->logfd = -1;
|
|
|
|
xtn->logfd_istty = 0;
|
|
|
|
|
2018-03-12 16:45:42 +00:00
|
|
|
if (handle_logopt(server, logopt) <= -1) goto oops;
|
2018-03-12 10:39:13 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_server_getoption (server, HCL_SERVER_TRAIT, &trait);
|
|
|
|
if (large_pages) trait |= HCL_SERVER_TRAIT_USE_LARGE_PAGES;
|
|
|
|
else trait &= HCL_SERVER_TRAIT_USE_LARGE_PAGES;
|
|
|
|
hcl_server_setoption (server, HCL_SERVER_TRAIT, &trait);
|
|
|
|
|
|
|
|
/*hcl_server_setoption (server, HCL_SERVER_WORKER_STACK_SIZE, ???);*/
|
|
|
|
hcl_server_setoption (server, HCL_SERVER_ACTOR_HEAP_SIZE, &memsize);
|
|
|
|
/*hcl_server_setoption (server, HCL_SERVER_ACTOR_DEBUG, &memsize);*/
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-12 10:39:13 +00:00
|
|
|
g_server = server;
|
|
|
|
set_signal (SIGINT, handle_sigint);
|
2018-03-12 16:45:42 +00:00
|
|
|
set_signal_to_ignore (SIGPIPE);
|
2018-03-13 02:02:25 +00:00
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
n = hcl_server_start(server, argv[opt.ind]);
|
2018-03-12 16:45:42 +00:00
|
|
|
|
2018-03-12 10:39:13 +00:00
|
|
|
set_signal_to_default (SIGINT);
|
2018-03-12 16:45:42 +00:00
|
|
|
set_signal_to_default (SIGPIPE);
|
2018-03-12 10:39:13 +00:00
|
|
|
g_server = NULL;
|
|
|
|
|
2018-03-13 10:20:33 +00:00
|
|
|
hcl_server_close (server);
|
2018-03-12 16:45:42 +00:00
|
|
|
return n;
|
2018-03-13 10:20:33 +00:00
|
|
|
|
2018-03-12 16:45:42 +00:00
|
|
|
oops:
|
2018-03-13 10:20:33 +00:00
|
|
|
if (server) hcl_server_close (server);
|
2018-03-12 16:45:42 +00:00
|
|
|
return -1;
|
2018-03-12 10:39:13 +00:00
|
|
|
}
|