2001-09-28 07:09:38 +00:00
|
|
|
/*
|
|
|
|
#FILENAME#
|
|
|
|
|
2020-03-20 15:01:36 +00:00
|
|
|
Qwaq
|
2001-09-28 07:09:38 +00:00
|
|
|
|
2020-03-20 15:01:36 +00:00
|
|
|
Copyright (C) 2001 Bill Currie
|
2001-09-28 07:09:38 +00:00
|
|
|
|
2020-03-20 15:01:36 +00:00
|
|
|
Author: Bill Currie <bill@taniwha.org>
|
|
|
|
Date: 2001/06/01
|
2001-09-28 07:09:38 +00:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU General Public License
|
|
|
|
as published by the Free Software Foundation; either version 2
|
|
|
|
of the License, or (at your option) any later version.
|
|
|
|
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to:
|
|
|
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
59 Temple Place - Suite 330
|
|
|
|
Boston, MA 02111-1307, USA
|
|
|
|
|
|
|
|
*/
|
2003-01-14 20:18:29 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2001-06-01 21:57:59 +00:00
|
|
|
#include <stdlib.h>
|
2020-03-08 13:19:37 +00:00
|
|
|
#include <stdio.h>
|
2020-03-20 06:47:30 +00:00
|
|
|
#include <string.h>
|
2020-03-20 15:01:36 +00:00
|
|
|
#include <pthread.h>
|
2020-03-08 13:19:37 +00:00
|
|
|
#include <errno.h>
|
2020-03-20 06:47:30 +00:00
|
|
|
#include <getopt.h>
|
2001-06-01 21:57:59 +00:00
|
|
|
|
2020-02-26 13:10:59 +00:00
|
|
|
#include "QF/cbuf.h"
|
|
|
|
#include "QF/cmd.h"
|
|
|
|
#include "QF/cvar.h"
|
|
|
|
#include "QF/gib.h"
|
|
|
|
#include "QF/idparse.h"
|
2020-03-22 04:47:44 +00:00
|
|
|
#include "QF/keys.h"
|
2020-02-26 13:10:59 +00:00
|
|
|
#include "QF/progs.h"
|
|
|
|
#include "QF/qargs.h"
|
|
|
|
#include "QF/quakefs.h"
|
2004-01-16 08:02:31 +00:00
|
|
|
#include "QF/ruamoko.h"
|
2020-02-26 13:10:59 +00:00
|
|
|
#include "QF/sys.h"
|
2002-11-12 02:30:08 +00:00
|
|
|
#include "QF/va.h"
|
2020-02-26 13:10:59 +00:00
|
|
|
#include "QF/zone.h"
|
2001-06-01 21:57:59 +00:00
|
|
|
|
2003-01-06 18:28:13 +00:00
|
|
|
#include "qwaq.h"
|
|
|
|
|
2001-06-05 23:53:55 +00:00
|
|
|
#define MAX_EDICTS 1024
|
|
|
|
|
2020-02-26 13:10:59 +00:00
|
|
|
cbuf_t *qwaq_cbuf;
|
|
|
|
|
2020-03-20 06:47:30 +00:00
|
|
|
const char *this_program;
|
|
|
|
|
|
|
|
enum {
|
|
|
|
start_opts = 255,
|
|
|
|
OPT_QARGS,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct option const long_options[] = {
|
|
|
|
{"qargs", no_argument, 0, OPT_QARGS},
|
|
|
|
{NULL, 0, NULL, 0},
|
|
|
|
};
|
|
|
|
|
|
|
|
static const char *short_options =
|
|
|
|
"-" // magic option parsing mode doohicky (must come first)
|
|
|
|
;
|
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
struct DARRAY_TYPE(qwaq_thread_t *) thread_data;
|
2020-03-20 06:47:30 +00:00
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
static QFile *
|
|
|
|
open_file (const char *path, int *len)
|
|
|
|
{
|
|
|
|
QFile *file = Qopen (path, "rbz");
|
2020-03-22 04:49:42 +00:00
|
|
|
char errbuff[1024];
|
2001-06-01 21:57:59 +00:00
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
if (!file) {
|
2020-03-22 04:49:42 +00:00
|
|
|
strerror_r(errno, errbuff, sizeof (errbuff));
|
|
|
|
Sys_Printf ("%s\n", errbuff);
|
2002-11-12 02:30:08 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*len = Qfilesize (file);
|
|
|
|
return file;
|
|
|
|
}
|
2001-06-06 20:05:08 +00:00
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
static void *
|
2020-02-21 12:17:28 +00:00
|
|
|
load_file (progs_t *pr, const char *name, off_t *_size)
|
2001-06-01 21:57:59 +00:00
|
|
|
{
|
2002-11-12 02:30:08 +00:00
|
|
|
QFile *file;
|
|
|
|
int size;
|
2004-11-02 05:12:00 +00:00
|
|
|
char *sym;
|
2002-11-12 02:30:08 +00:00
|
|
|
|
|
|
|
file = open_file (name, &size);
|
|
|
|
if (!file) {
|
|
|
|
file = open_file (va ("%s.gz", name), &size);
|
|
|
|
if (!file) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2004-11-02 05:12:00 +00:00
|
|
|
sym = malloc (size + 1);
|
|
|
|
sym[size] = 0;
|
2002-11-12 02:30:08 +00:00
|
|
|
Qread (file, sym, size);
|
2020-02-21 12:17:28 +00:00
|
|
|
*_size = size;
|
2002-11-12 02:30:08 +00:00
|
|
|
return sym;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
|
|
|
allocate_progs_mem (progs_t *pr, int size)
|
|
|
|
{
|
|
|
|
return malloc (size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_progs_mem (progs_t *pr, void *mem)
|
|
|
|
{
|
|
|
|
free (mem);
|
|
|
|
}
|
2001-06-01 21:57:59 +00:00
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
static void
|
|
|
|
init_qf (void)
|
|
|
|
{
|
2020-02-26 13:10:59 +00:00
|
|
|
qwaq_cbuf = Cbuf_New (&id_interp);
|
|
|
|
|
2011-09-11 05:56:47 +00:00
|
|
|
Sys_Init ();
|
2020-02-26 13:10:59 +00:00
|
|
|
COM_ParseConfig ();
|
|
|
|
|
|
|
|
//Cvar_Set (developer, "1");
|
2001-06-01 21:57:59 +00:00
|
|
|
|
2020-02-26 13:10:59 +00:00
|
|
|
Memory_Init (malloc (8 * 1024 * 1024), 8 * 1024 * 1024);
|
2007-06-09 09:43:21 +00:00
|
|
|
|
2007-06-09 13:44:06 +00:00
|
|
|
Cvar_Get ("pr_debug", "2", 0, 0, 0);
|
2001-12-12 08:39:47 +00:00
|
|
|
Cvar_Get ("pr_boundscheck", "0", 0, 0, 0);
|
2002-11-12 02:30:08 +00:00
|
|
|
|
2020-03-20 01:13:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static progs_t *
|
|
|
|
create_progs (void)
|
|
|
|
{
|
|
|
|
progs_t *pr = calloc (1, sizeof (*pr));
|
|
|
|
|
|
|
|
pr->load_file = load_file;
|
|
|
|
pr->allocate_progs_mem = allocate_progs_mem;
|
|
|
|
pr->free_progs_mem = free_progs_mem;
|
|
|
|
pr->no_exec_limit = 1;
|
2001-07-14 23:52:56 +00:00
|
|
|
|
2001-06-01 21:57:59 +00:00
|
|
|
PR_Init_Cvars ();
|
2020-03-20 01:13:37 +00:00
|
|
|
PR_Init (pr);
|
|
|
|
RUA_Init (pr, 0);
|
2020-03-22 11:51:55 +00:00
|
|
|
Key_Progs_Init (pr); // FIXME not all threads
|
|
|
|
PR_Cmds_Init (pr); // FIXME not all threads
|
|
|
|
BI_Init (pr); // FIXME not all threads
|
|
|
|
QWAQ_EditBuffer_Init (pr); // FIXME not all threads
|
2020-03-20 01:13:37 +00:00
|
|
|
|
|
|
|
return pr;
|
2002-11-12 02:30:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2020-03-20 01:13:37 +00:00
|
|
|
load_progs (progs_t *pr, const char *name)
|
2002-11-12 02:30:08 +00:00
|
|
|
{
|
|
|
|
QFile *file;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
file = open_file (name, &size);
|
|
|
|
if (!file) {
|
|
|
|
return 0;
|
2001-06-01 21:57:59 +00:00
|
|
|
}
|
2020-03-20 01:13:37 +00:00
|
|
|
pr->progs_name = name;
|
|
|
|
pr->max_edicts = 1;
|
|
|
|
pr->zone_size = 1024*1024;
|
|
|
|
PR_LoadProgsFile (pr, file, size);
|
2002-11-12 02:30:08 +00:00
|
|
|
Qclose (file);
|
2020-03-20 01:13:37 +00:00
|
|
|
if (!PR_RunLoadFuncs (pr))
|
|
|
|
PR_Error (pr, "unable to load %s", pr->progs_name);
|
2002-11-12 02:30:08 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-03-20 07:59:59 +00:00
|
|
|
static void
|
|
|
|
spawn_progs (qwaq_thread_t *thread)
|
|
|
|
{
|
|
|
|
dfunction_t *dfunc;
|
2020-03-20 08:30:26 +00:00
|
|
|
const char *name = 0;
|
2020-03-20 07:59:59 +00:00
|
|
|
string_t *pr_argv;
|
|
|
|
int pr_argc = 1, i;
|
|
|
|
progs_t *pr;
|
|
|
|
|
|
|
|
thread->main_func = 0;
|
|
|
|
pr = thread->pr = create_progs ();
|
|
|
|
if (thread->args.size) {
|
|
|
|
name = thread->args.a[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!load_progs (pr, name)) {
|
|
|
|
Sys_Error ("couldn't load %s", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
//pr->pr_trace = 1;
|
|
|
|
//pr->pr_trace_depth = -1;
|
|
|
|
|
|
|
|
PR_PushFrame (pr);
|
|
|
|
if (thread->args.size > 2) {
|
|
|
|
pr_argc = thread->args.size - 1;
|
|
|
|
}
|
|
|
|
pr_argv = PR_Zone_Malloc (pr, (pr_argc + 1) * 4);
|
|
|
|
pr_argv[0] = PR_SetTempString (pr, name);
|
|
|
|
for (i = 1; i < pr_argc; i++)
|
|
|
|
pr_argv[i] = PR_SetTempString (pr, thread->args.a[1 + i]);
|
|
|
|
pr_argv[i] = 0;
|
|
|
|
|
|
|
|
if ((dfunc = PR_FindFunction (pr, ".main"))
|
|
|
|
|| (dfunc = PR_FindFunction (pr, "main"))) {
|
|
|
|
thread->main_func = dfunc - pr->pr_functions;
|
|
|
|
} else {
|
|
|
|
PR_Undefined (pr, "function", "main");
|
|
|
|
}
|
|
|
|
PR_RESET_PARAMS (pr);
|
|
|
|
P_INT (pr, 0) = pr_argc;
|
|
|
|
P_POINTER (pr, 1) = PR_SetPointer (pr, pr_argv);
|
|
|
|
pr->pr_argc = 2;
|
|
|
|
}
|
|
|
|
|
2020-03-20 15:01:36 +00:00
|
|
|
static void *
|
|
|
|
run_progs (void *data)
|
2020-03-20 07:59:59 +00:00
|
|
|
{
|
2020-03-20 15:01:36 +00:00
|
|
|
__auto_type thread = (qwaq_thread_t *) data;
|
|
|
|
|
2020-03-20 07:59:59 +00:00
|
|
|
PR_ExecuteProgram (thread->pr, thread->main_func);
|
|
|
|
PR_PopFrame (thread->pr);
|
2020-03-20 15:01:36 +00:00
|
|
|
thread->return_code = R_INT (thread->pr);
|
|
|
|
return thread;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
start_progs_thread (qwaq_thread_t *thread)
|
|
|
|
{
|
|
|
|
pthread_create (&thread->thread_id, 0, run_progs, thread);
|
|
|
|
}
|
|
|
|
|
|
|
|
qwaq_thread_t *
|
2020-03-20 15:14:33 +00:00
|
|
|
create_thread (void *(*thread_func) (qwaq_thread_t *), void *data)
|
2020-03-20 15:01:36 +00:00
|
|
|
{
|
|
|
|
qwaq_thread_t *thread = calloc (1, sizeof (*thread));
|
|
|
|
|
2020-03-20 15:14:33 +00:00
|
|
|
thread->data = data;
|
2020-03-20 15:01:36 +00:00
|
|
|
DARRAY_APPEND (&thread_data, thread);
|
2020-03-20 15:14:33 +00:00
|
|
|
pthread_create (&thread->thread_id, 0,
|
|
|
|
(void*(*)(void*))thread_func, thread);
|
2020-03-20 15:01:36 +00:00
|
|
|
return thread;
|
2020-03-20 07:59:59 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 06:47:30 +00:00
|
|
|
static void
|
|
|
|
usage (int status)
|
|
|
|
{
|
|
|
|
printf ("%s - QuakeForge runtime\n", this_program);
|
|
|
|
printf ("sorry, no help yet\n");
|
|
|
|
exit (status);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_argset (int argc, char **argv)
|
|
|
|
{
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *thread = calloc (1, sizeof (*thread));
|
|
|
|
DARRAY_INIT (&thread->args, 8);
|
2020-03-20 06:47:30 +00:00
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
DARRAY_APPEND (&thread->args, 0);
|
2020-03-20 06:47:30 +00:00
|
|
|
while (optind < argc && strcmp (argv[optind], "--")) {
|
2020-03-20 14:17:06 +00:00
|
|
|
DARRAY_APPEND (&thread->args, argv[optind++]);
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
|
|
|
if (optind < argc) {
|
|
|
|
optind++;
|
|
|
|
}
|
2020-03-20 12:16:56 +00:00
|
|
|
DARRAY_APPEND (&thread_data, thread);
|
|
|
|
return thread_data.size - 1;
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
parse_args (int argc, char **argv)
|
|
|
|
{
|
|
|
|
int c;
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *main_thread = calloc (1, sizeof (*main_thread));
|
2020-03-20 06:47:30 +00:00
|
|
|
int qargs_ind = -1;
|
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
DARRAY_INIT (&main_thread->args, 8);
|
2020-03-20 06:47:30 +00:00
|
|
|
|
|
|
|
while ((c = getopt_long (argc, argv,
|
|
|
|
short_options, long_options, 0)) != -1) {
|
|
|
|
switch (c) {
|
|
|
|
case 1:
|
2020-03-20 14:17:06 +00:00
|
|
|
DARRAY_APPEND (&main_thread->args, argv[optind - 1]);
|
2020-03-20 06:47:30 +00:00
|
|
|
break;
|
|
|
|
case OPT_QARGS:
|
|
|
|
if (qargs_ind < 0) {
|
|
|
|
qargs_ind = parse_argset (argc, argv);
|
2020-03-20 14:17:06 +00:00
|
|
|
thread_data.a[qargs_ind]->args.a[0] = "--qargs";
|
2020-03-20 06:47:30 +00:00
|
|
|
goto done;
|
|
|
|
} else {
|
|
|
|
printf ("more than one set of qargs given");
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
usage (1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
done:
|
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
free (thread_data.a[0]->args.a);
|
|
|
|
free (thread_data.a[0]);
|
2020-03-20 12:16:56 +00:00
|
|
|
thread_data.a[0] = main_thread;
|
2020-03-20 06:47:30 +00:00
|
|
|
|
|
|
|
while (optind < argc) {
|
|
|
|
parse_argset (argc, argv);
|
|
|
|
}
|
|
|
|
return qargs_ind;
|
|
|
|
}
|
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
int
|
2020-03-20 06:47:30 +00:00
|
|
|
main (int argc, char **argv)
|
2002-11-12 02:30:08 +00:00
|
|
|
{
|
2020-03-20 07:59:59 +00:00
|
|
|
int qargs_ind = -1;
|
2020-03-20 08:30:26 +00:00
|
|
|
int main_ind = -1;
|
|
|
|
int ret = 0;
|
|
|
|
size_t num_sys = 1;
|
2002-11-12 02:30:08 +00:00
|
|
|
|
2020-03-20 06:47:30 +00:00
|
|
|
this_program = argv[0];
|
|
|
|
|
2020-03-20 12:16:56 +00:00
|
|
|
DARRAY_INIT (&thread_data, 4);
|
2020-03-20 06:47:30 +00:00
|
|
|
for (optind = 1; optind < argc; ) {
|
|
|
|
parse_argset (argc, argv);
|
|
|
|
}
|
2020-03-20 12:16:56 +00:00
|
|
|
if (thread_data.size) {
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *thread = thread_data.a[0];
|
2020-03-20 06:47:30 +00:00
|
|
|
// the first arg is initialized to null, but this is for getopt, so
|
|
|
|
// set to main program name
|
|
|
|
thread->args.a[0] = this_program;
|
|
|
|
optind = 0;
|
|
|
|
qargs_ind = parse_args (thread->args.size, (char **) thread->args.a);
|
|
|
|
} else {
|
|
|
|
// create a blank main thread set
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *thread = calloc (1, sizeof (*thread));
|
|
|
|
DARRAY_INIT (&thread->args, 4);
|
2020-03-20 12:16:56 +00:00
|
|
|
DARRAY_APPEND (&thread_data, thread);
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (qargs_ind >= 0) {
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *qargs = thread_data.a[qargs_ind];
|
2020-03-20 06:47:30 +00:00
|
|
|
// the first arg is initialized to --qargs, so
|
|
|
|
// set to main program name for now
|
2020-03-20 07:59:59 +00:00
|
|
|
qargs->args.a[0] = this_program;
|
|
|
|
COM_InitArgv (qargs->args.size, qargs->args.a);
|
2020-03-20 08:30:26 +00:00
|
|
|
num_sys++;
|
2020-03-20 07:59:59 +00:00
|
|
|
} else {
|
|
|
|
qwaq_thread_t qargs = {};
|
|
|
|
DARRAY_INIT (&qargs.args, 2);
|
|
|
|
DARRAY_APPEND (&qargs.args, this_program);
|
|
|
|
COM_InitArgv (qargs.args.size, qargs.args.a);
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
|
|
|
|
2002-11-12 02:30:08 +00:00
|
|
|
init_qf ();
|
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
if (thread_data.a[0]->args.size < 1) {
|
|
|
|
DARRAY_APPEND (&thread_data.a[0]->args, "qwaq-app.dat");
|
2020-03-20 08:30:26 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 14:17:06 +00:00
|
|
|
while (thread_data.size < thread_data.a[0]->args.size + num_sys) {
|
|
|
|
qwaq_thread_t *thread = calloc (1, sizeof (*thread));
|
|
|
|
DARRAY_INIT (&thread->args, 4);
|
|
|
|
DARRAY_APPEND (&thread->args, 0);
|
2020-03-20 12:16:56 +00:00
|
|
|
DARRAY_APPEND (&thread_data, thread);
|
2020-03-20 08:30:26 +00:00
|
|
|
}
|
|
|
|
|
2020-03-20 12:16:56 +00:00
|
|
|
for (size_t i = 1, thread_ind = 0; i < thread_data.size; i++) {
|
2020-03-20 14:17:06 +00:00
|
|
|
qwaq_thread_t *thread = thread_data.a[i];
|
2020-03-20 07:59:59 +00:00
|
|
|
if (thread->args.size && thread->args.a[0]
|
|
|
|
&& strcmp (thread->args.a[0], "--qargs")) {
|
|
|
|
// skip the args set that's passed to qargs
|
|
|
|
continue;
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
2020-03-20 14:17:06 +00:00
|
|
|
if (thread_ind < thread_data.a[0]->args.size) {
|
|
|
|
thread->args.a[0] = thread_data.a[0]->args.a[thread_ind++];
|
2020-03-20 08:30:26 +00:00
|
|
|
} else {
|
|
|
|
printf ("ignoring extra arg sets\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (main_ind < 0) {
|
|
|
|
main_ind = i;
|
|
|
|
}
|
2020-03-20 07:59:59 +00:00
|
|
|
spawn_progs (thread);
|
2020-03-20 06:47:30 +00:00
|
|
|
}
|
2020-03-20 08:30:26 +00:00
|
|
|
if (main_ind >= 0) {
|
2020-03-20 15:01:36 +00:00
|
|
|
// threads might start new threads before the end is reached
|
|
|
|
size_t count = thread_data.size;
|
|
|
|
for (size_t i = 0; i < count; i++) {
|
|
|
|
if (thread_data.a[i]->pr && thread_data.a[i]->main_func) {
|
|
|
|
start_progs_thread (thread_data.a[main_ind]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_join (thread_data.a[main_ind]->thread_id, 0);
|
2020-03-20 08:30:26 +00:00
|
|
|
}
|
2020-03-20 15:01:36 +00:00
|
|
|
|
2020-02-26 16:18:38 +00:00
|
|
|
Sys_Shutdown ();
|
2020-03-20 07:59:59 +00:00
|
|
|
return ret;
|
2001-06-01 21:57:59 +00:00
|
|
|
}
|