2007-12-30 17:41:49 +01:00
|
|
|
/***************************************************************************
|
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
gbc.c
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-09-12 15:13:13 +02:00
|
|
|
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01: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, or (at your option)
|
|
|
|
any later version.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
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.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
|
|
MA 02110-1301, USA.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
2013-05-25 15:23:39 +02:00
|
|
|
#define __GBC_C
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
2019-05-01 11:59:26 +02:00
|
|
|
#include <sys/wait.h>
|
2007-12-30 17:41:49 +01:00
|
|
|
#include <sys/time.h>
|
|
|
|
#include <dirent.h>
|
2021-01-30 14:06:09 +01:00
|
|
|
#include <signal.h>
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
#include "gb_common.h"
|
|
|
|
#include "gb_error.h"
|
|
|
|
#include "gb_str.h"
|
|
|
|
#include "gb_file.h"
|
|
|
|
#include "gb_common_buffer.h"
|
2021-02-01 01:21:08 +01:00
|
|
|
#include "gb_system.h"
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
#include "gbc_compile.h"
|
|
|
|
|
|
|
|
#include "gb_reserved.h"
|
|
|
|
#include "gbc_read.h"
|
|
|
|
#include "gbc_form.h"
|
|
|
|
#include "gbc_trans.h"
|
|
|
|
#include "gbc_header.h"
|
|
|
|
#include "gbc_output.h"
|
|
|
|
|
2021-02-01 01:21:08 +01:00
|
|
|
#include "gb_system_temp.h"
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
typedef
|
|
|
|
void (*BACKGROUND_TASK)(const char *);
|
|
|
|
|
2021-05-07 19:20:30 +02:00
|
|
|
typedef
|
|
|
|
struct {
|
|
|
|
const char *option;
|
|
|
|
bool *variable;
|
|
|
|
}
|
|
|
|
COMPILER_FLAGS;
|
|
|
|
|
|
|
|
static bool main_debug = FALSE;
|
|
|
|
static bool main_exec = FALSE;
|
|
|
|
static bool main_compile_all = FALSE;
|
|
|
|
static bool main_trans = FALSE;
|
|
|
|
static bool main_warnings = FALSE;
|
|
|
|
static bool main_swap = FALSE;
|
|
|
|
|
|
|
|
static bool _opt_no_old_read_syntax = FALSE;
|
|
|
|
static bool _opt_check_prefix = FALSE;
|
|
|
|
static bool _opt_public_module = FALSE;
|
|
|
|
static bool _opt_public_control = FALSE;
|
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
static char *_convert_form = NULL;
|
|
|
|
|
2009-05-15 19:57:29 +02:00
|
|
|
#if HAVE_GETOPT_LONG
|
2021-01-30 14:06:09 +01:00
|
|
|
static struct option _long_options[] =
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-29 22:51:10 +02:00
|
|
|
{ "debug", 0, NULL, 'g' },
|
|
|
|
{ "version", 0, NULL, 'V' },
|
|
|
|
{ "help", 0, NULL, 'h' },
|
|
|
|
{ "license", 0, NULL, 'L' },
|
|
|
|
{ "verbose", 0, NULL, 'v' },
|
|
|
|
{ "translate", 0, NULL, 't' },
|
|
|
|
{ "swap", 0, NULL, 's' },
|
|
|
|
{ "all", 0, NULL, 'a' },
|
2009-09-20 19:32:12 +02:00
|
|
|
{ "translate-errors", 0, NULL, 'e' },
|
2021-05-07 19:20:30 +02:00
|
|
|
{ "warnings", 0, NULL, 'w' },
|
2022-02-05 18:41:36 +01:00
|
|
|
|
2021-05-07 19:20:30 +02:00
|
|
|
{ "jobs", 1, NULL, 'j' },
|
|
|
|
{ "root", 1, NULL, 'r' },
|
2021-05-21 16:48:59 +02:00
|
|
|
{ "default-namespace", 1, NULL, 'n' },
|
2022-02-05 18:41:36 +01:00
|
|
|
{ "form", 1, NULL, 'F' },
|
2021-05-07 19:20:30 +02:00
|
|
|
|
|
|
|
//{ "no-old-read-write-syntax", 0, &_opt_no_old_read_syntax, 1 },
|
|
|
|
//{ "check-prefix", 0, &_opt_check_prefix, 1},
|
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
{ 0 }
|
2007-12-30 17:41:49 +01:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2021-05-07 19:20:30 +02:00
|
|
|
static COMPILER_FLAGS _compiler_flags[] = {
|
|
|
|
{ "no-old-read-write-syntax", &_opt_no_old_read_syntax },
|
|
|
|
{ "check-prefix", &_opt_check_prefix },
|
|
|
|
{ "public-module", &_opt_public_module },
|
|
|
|
{ "public-control", &_opt_public_control },
|
|
|
|
{ NULL }
|
|
|
|
};
|
2020-02-22 15:57:46 +01:00
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
//static char *main_class_file = NULL;
|
|
|
|
|
|
|
|
static char **_files = NULL;
|
2020-02-22 15:57:46 +01:00
|
|
|
static bool make_test = FALSE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
static uint _ntask_max = 0;
|
|
|
|
static uint _ntask = 0;
|
|
|
|
static bool _child = FALSE;
|
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
static void print_version()
|
|
|
|
{
|
|
|
|
#ifdef TRUNK_VERSION
|
|
|
|
printf(VERSION " " TRUNK_VERSION "\n");
|
|
|
|
#else /* no TRUNK_VERSION */
|
|
|
|
printf(VERSION "\n");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static void print_title()
|
|
|
|
{
|
|
|
|
printf("\nGambas compiler version ");
|
|
|
|
print_version();
|
|
|
|
}
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
static void get_arguments(int argc, char **argv)
|
|
|
|
{
|
2010-08-29 22:51:10 +02:00
|
|
|
const char *dir;
|
|
|
|
int opt;
|
|
|
|
#if HAVE_GETOPT_LONG
|
|
|
|
int index = 0;
|
|
|
|
#endif
|
2021-05-07 19:20:30 +02:00
|
|
|
COMPILER_FLAGS *cf;
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
#if HAVE_GETOPT_LONG
|
2022-02-05 18:41:36 +01:00
|
|
|
opt = getopt_long(argc, argv, "gxvaVhj:Lwtser:f:n:F:", _long_options, &index);
|
2010-08-29 22:51:10 +02:00
|
|
|
#else
|
2022-02-05 18:41:36 +01:00
|
|
|
opt = getopt(argc, argv, "gxvaVhj:Lwtser:f:n:F:");
|
2010-08-29 22:51:10 +02:00
|
|
|
#endif
|
|
|
|
if (opt < 0) break;
|
|
|
|
|
|
|
|
switch (opt)
|
|
|
|
{
|
|
|
|
case 'V':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
print_version();
|
2010-08-29 22:51:10 +02:00
|
|
|
exit(0);
|
|
|
|
|
|
|
|
case 'g':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
main_debug = TRUE;
|
|
|
|
break;
|
|
|
|
|
2012-05-05 02:39:43 +02:00
|
|
|
case 'x':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2012-05-05 02:39:43 +02:00
|
|
|
main_exec = TRUE;
|
|
|
|
break;
|
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
case 'v':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2020-05-09 20:55:04 +02:00
|
|
|
COMP_verbose = TRUE;
|
2010-08-29 22:51:10 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
main_compile_all = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 't':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
main_trans = TRUE;
|
|
|
|
break;
|
|
|
|
|
2012-09-04 09:29:39 +02:00
|
|
|
case 'w':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2012-09-04 09:29:39 +02:00
|
|
|
main_warnings = TRUE;
|
|
|
|
break;
|
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
case 's':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
main_swap = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'r':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2013-01-05 00:24:37 +01:00
|
|
|
if (COMP_root)
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("option '-r' already specified.");
|
2010-08-29 22:51:10 +02:00
|
|
|
COMP_root = STR_copy(optarg);
|
|
|
|
break;
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2009-09-20 19:32:12 +02:00
|
|
|
case 'e':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2009-09-21 18:57:21 +02:00
|
|
|
ERROR_translate = TRUE;
|
2009-09-20 19:32:12 +02:00
|
|
|
break;
|
2018-06-01 03:50:42 +02:00
|
|
|
|
2010-02-21 19:15:16 +01:00
|
|
|
case 'L':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
print_title();
|
|
|
|
printf(COPYRIGHT);
|
2010-02-21 19:15:16 +01:00
|
|
|
exit(0);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
case 'h': case '?':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
print_title();
|
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
printf(
|
2012-08-21 03:03:52 +02:00
|
|
|
"\nCompile Gambas projects into architecture-independent bytecode.\n"
|
2022-02-05 18:41:36 +01:00
|
|
|
"\n gbc" GAMBAS_VERSION_STRING " [options] [<project directory>]\n"
|
|
|
|
"\nConvert a form file into Gambas code.\n"
|
|
|
|
"\n gbc" GAMBAS_VERSION_STRING " -F <form file>\n\n"
|
2010-08-29 22:51:10 +02:00
|
|
|
"Options:"
|
|
|
|
#if HAVE_GETOPT_LONG
|
2022-02-05 18:41:36 +01:00
|
|
|
"\n\n"
|
|
|
|
" -a --all compile all files\n"
|
|
|
|
" -e --translate-errors display translatable error messages\n"
|
|
|
|
" -F --form <form file> convert a form file into code and print it\n"
|
|
|
|
" -g --debug add debugging information\n"
|
|
|
|
" -h --help display this help\n"
|
|
|
|
" -j --jobs <count> number of background jobs (default: %d)\n"
|
|
|
|
" -L --license display license\n"
|
|
|
|
" -n --default-namespace <namespace> default namespace for exported classes\n"
|
|
|
|
" -r --root <directory> gives the gambas installation directory\n"
|
|
|
|
" -s --swap swap endianness\n"
|
|
|
|
" -t --translate output translation files and compile them if needed\n"
|
|
|
|
" -v --verbose verbose output\n"
|
|
|
|
" -V --version display version\n"
|
|
|
|
" -w --warnings display warnings\n"
|
|
|
|
" -x --exec executable mode (define the 'Exec' preprocessor constant and remove assertions)\n"
|
|
|
|
"\nCompiler flags:\n\n"
|
|
|
|
" -f check-prefix check the prefix of variables if warnings are enable\n"
|
|
|
|
" -f public-module module symbols are public by default\n"
|
|
|
|
" -f public-control form controls are public\n"
|
2010-08-29 22:51:10 +02:00
|
|
|
#else
|
2022-02-05 18:41:36 +01:00
|
|
|
" (no long options on this system)\n\n"
|
|
|
|
" -a compile all files\n"
|
|
|
|
" -e display translatable error messages\n"
|
|
|
|
" -F <form file> convert a form file into code and print it\n"
|
|
|
|
" -g add debugging information\n"
|
|
|
|
" -h display this help\n"
|
|
|
|
" -j <count> number of background jobs (default: %d)\n"
|
|
|
|
" -L display license\n"
|
|
|
|
" -n <namespace> default namespace for exported classes\n"
|
|
|
|
" -r <directory> gives the gambas installation directory\n"
|
|
|
|
" -s swap endianness\n"
|
|
|
|
" -t output translation files and compile them if needed\n"
|
|
|
|
" -v verbose output\n"
|
|
|
|
" -V display version\n"
|
|
|
|
" -w display warnings\n"
|
|
|
|
" -x executable mode (define the 'Exec' preprocessor constant and remove assertions)\n"
|
|
|
|
"\nCompiler flags:\n\n"
|
|
|
|
" -f check-prefix check the prefix of variables if warnings are enable\n"
|
|
|
|
" -f public-module module symbols are public by default\n"
|
|
|
|
" -f public-control form controls are public\n"
|
2010-08-29 22:51:10 +02:00
|
|
|
#endif
|
2021-02-01 01:21:08 +01:00
|
|
|
"\n",
|
|
|
|
SYSTEM_get_cpu_count());
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
exit(0);
|
2021-01-30 14:06:09 +01:00
|
|
|
|
|
|
|
case 'j':
|
2021-05-07 19:20:30 +02:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
_ntask_max = atoi(optarg);
|
2021-02-01 04:04:08 +01:00
|
|
|
if (_ntask_max < 0 || _ntask_max > 16)
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("Incorrect number of jobs.");
|
|
|
|
break;
|
2021-05-07 19:20:30 +02:00
|
|
|
|
|
|
|
case 'f':
|
|
|
|
|
|
|
|
cf = _compiler_flags;
|
|
|
|
while (cf->option)
|
|
|
|
{
|
|
|
|
if (strcmp(optarg, cf->option) == 0)
|
|
|
|
{
|
|
|
|
*cf->variable = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cf++;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2021-05-21 16:48:59 +02:00
|
|
|
|
|
|
|
case 'n':
|
|
|
|
|
|
|
|
if (COMP_default_namespace)
|
|
|
|
ERROR_fail("option '-n' already specified.");
|
|
|
|
COMP_default_namespace = STR_copy(optarg);
|
|
|
|
break;
|
2022-02-05 18:41:36 +01:00
|
|
|
|
|
|
|
case 'F':
|
|
|
|
|
|
|
|
if (_convert_form)
|
|
|
|
ERROR_fail("option '-F' already specified.");
|
|
|
|
_convert_form = STR_copy(optarg);
|
|
|
|
break;
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
default:
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (optind < (argc - 1))
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("too many arguments");
|
2010-08-29 22:51:10 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
if (_convert_form)
|
|
|
|
return;
|
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
if (optind < argc)
|
2010-10-31 23:05:29 +01:00
|
|
|
FILE_chdir(argv[optind]);
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
dir = FILE_get_current_dir();
|
|
|
|
if (!dir)
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("no current directory");
|
2010-08-29 22:51:10 +02:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
COMP_dir = STR_copy(dir);
|
|
|
|
COMP_project = STR_copy(FILE_cat(COMP_dir, ".project", NULL));
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
if (!FILE_exist(COMP_project))
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("project file not found: %s", COMP_project);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
static void compile_file(const char *file)
|
|
|
|
{
|
2021-04-03 12:45:19 +02:00
|
|
|
int i, lock;
|
2010-08-29 22:51:10 +02:00
|
|
|
time_t time_src, time_form, time_pot, time_output;
|
2012-03-18 14:14:17 +01:00
|
|
|
char *source;
|
2010-08-29 22:51:10 +02:00
|
|
|
|
2012-12-19 22:55:16 +01:00
|
|
|
COMPILE_begin(file, main_trans, main_debug);
|
2010-08-29 22:51:10 +02:00
|
|
|
|
|
|
|
if (!main_compile_all)
|
|
|
|
{
|
|
|
|
if (FILE_exist(JOB->output))
|
|
|
|
{
|
|
|
|
time_src = FILE_get_time(JOB->name);
|
|
|
|
time_output = FILE_get_time(JOB->output);
|
|
|
|
|
|
|
|
if (JOB->form)
|
|
|
|
time_form = FILE_get_time(JOB->form);
|
|
|
|
else
|
|
|
|
time_form = time_src;
|
|
|
|
|
|
|
|
if (main_trans)
|
|
|
|
time_pot = FILE_get_time(JOB->tname);
|
|
|
|
else
|
|
|
|
time_pot = time_src;
|
|
|
|
|
|
|
|
if (time_src <= time_output && time_src <= time_pot && time_form <= time_output)
|
|
|
|
goto _FIN;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
COMPILE_alloc();
|
|
|
|
|
2012-05-05 02:39:43 +02:00
|
|
|
JOB->exec = main_exec;
|
2012-09-04 09:29:39 +02:00
|
|
|
JOB->warnings = main_warnings;
|
2021-05-07 19:20:30 +02:00
|
|
|
JOB->check_prefix = main_warnings && _opt_check_prefix;
|
2010-08-29 22:51:10 +02:00
|
|
|
JOB->swap = main_swap;
|
2021-05-07 19:20:30 +02:00
|
|
|
JOB->public_module = _opt_public_module;
|
|
|
|
JOB->no_old_read_syntax = _opt_no_old_read_syntax;
|
2010-08-29 22:51:10 +02:00
|
|
|
//JOB->class_file = main_class_file;
|
|
|
|
|
2020-05-09 20:55:04 +02:00
|
|
|
if (COMP_verbose)
|
2010-08-29 22:51:10 +02:00
|
|
|
{
|
2021-01-30 14:06:09 +01:00
|
|
|
fputc('\n', stderr);
|
2010-08-29 22:51:10 +02:00
|
|
|
for (i = 1; i <= 9; i++)
|
2021-01-30 14:06:09 +01:00
|
|
|
fprintf(stderr, "--------");
|
|
|
|
fprintf(stderr, "\nCompiling %s...\n", FILE_get_name(JOB->name));
|
2008-09-22 01:22:07 +02:00
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-04-22 18:52:59 +02:00
|
|
|
JOB->first_line = 1;
|
|
|
|
|
|
|
|
if (JOB->form)
|
|
|
|
{
|
|
|
|
JOB->first_line = FORM_FIRST_LINE;
|
|
|
|
BUFFER_add(&JOB->source, "#Line " FORM_FIRST_LINE_STRING "\n", -1);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2012-03-18 14:14:17 +01:00
|
|
|
BUFFER_create(&source);
|
|
|
|
BUFFER_load_file(&source, JOB->form);
|
|
|
|
BUFFER_add(&source, "\n\0", 2);
|
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
FORM_set_target(&JOB->source);
|
|
|
|
|
2012-01-28 10:55:55 +01:00
|
|
|
switch (JOB->family->type)
|
|
|
|
{
|
|
|
|
case FORM_WEBPAGE:
|
2012-03-18 14:14:17 +01:00
|
|
|
FORM_webpage(source);
|
2012-01-28 10:55:55 +01:00
|
|
|
break;
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2012-01-28 10:55:55 +01:00
|
|
|
case FORM_NORMAL:
|
|
|
|
default:
|
2022-02-05 18:41:36 +01:00
|
|
|
FORM_do(source, FALSE, _opt_public_control);
|
2012-01-28 10:55:55 +01:00
|
|
|
break;
|
|
|
|
}
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2012-03-18 14:14:17 +01:00
|
|
|
BUFFER_delete(&source);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2011-04-22 18:52:59 +02:00
|
|
|
BUFFER_add(&JOB->source, "#Line 1\n", -1);
|
|
|
|
}
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2011-04-16 15:25:46 +02:00
|
|
|
COMPILE_load();
|
2012-03-02 23:38:50 +01:00
|
|
|
BUFFER_add(&JOB->source, "\n\0", 2);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2019-05-08 06:13:04 +02:00
|
|
|
JOB->step = JOB_STEP_READ;
|
2010-08-29 22:51:10 +02:00
|
|
|
READ_do();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
#ifdef DEBUG
|
|
|
|
TABLE_print(JOB->class->table, TRUE);
|
|
|
|
#endif
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2019-05-08 06:13:04 +02:00
|
|
|
JOB->step = JOB_STEP_CODE;
|
2010-08-29 22:51:10 +02:00
|
|
|
HEADER_do();
|
|
|
|
TRANS_code();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
#ifdef DEBUG
|
|
|
|
TABLE_print(JOB->class->string, FALSE);
|
|
|
|
#endif
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2019-05-08 06:13:04 +02:00
|
|
|
JOB->step = JOB_STEP_OUTPUT;
|
2010-08-29 22:51:10 +02:00
|
|
|
OUTPUT_do(main_swap);
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2021-04-03 12:45:19 +02:00
|
|
|
lock = COMPILE_lock_file(".gbc.lock");
|
2010-08-29 22:51:10 +02:00
|
|
|
CLASS_export();
|
2021-04-03 12:45:19 +02:00
|
|
|
COMPILE_unlock_file(lock);
|
2021-01-30 14:06:09 +01:00
|
|
|
|
|
|
|
COMPILE_free();
|
2019-01-19 01:03:57 +01:00
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
_FIN:
|
2010-08-29 22:51:10 +02:00
|
|
|
COMPILE_end();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
static void wait_for_task(void)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
pid_t pid;
|
|
|
|
|
|
|
|
if (COMP_verbose)
|
|
|
|
fprintf(stderr, "gbc" GAMBAS_VERSION_STRING ": wait for tasks...\n");
|
|
|
|
|
|
|
|
pid = wait(&status);
|
2021-02-04 04:04:23 +01:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
if (pid < 0)
|
2021-02-04 04:04:23 +01:00
|
|
|
ERROR_fail("wait() fails: &1", strerror(errno));
|
|
|
|
|
2021-01-30 15:25:08 +01:00
|
|
|
if (!WIFEXITED(status))
|
2021-02-04 04:04:23 +01:00
|
|
|
{
|
|
|
|
fprintf(stderr, "gbc" GAMBAS_VERSION_STRING ": the background job %d has crashed with signal %d", pid, WTERMSIG(status));
|
|
|
|
exit(status);
|
|
|
|
}
|
|
|
|
|
2021-01-30 15:25:08 +01:00
|
|
|
if (WEXITSTATUS(status))
|
2021-02-02 23:46:00 +01:00
|
|
|
{
|
|
|
|
while (_ntask > 0)
|
|
|
|
{
|
|
|
|
wait(&status);
|
|
|
|
_ntask--;
|
|
|
|
}
|
2021-01-30 15:25:08 +01:00
|
|
|
exit(1);
|
2021-02-02 23:46:00 +01:00
|
|
|
}
|
2021-01-30 14:06:09 +01:00
|
|
|
|
|
|
|
if (COMP_verbose)
|
|
|
|
fprintf(stderr, "gbc" GAMBAS_VERSION_STRING ": end task %d\n", pid);
|
|
|
|
_ntask--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void wait_for_all_task(void)
|
|
|
|
{
|
|
|
|
if (_ntask_max > 1)
|
|
|
|
{
|
|
|
|
while (_ntask > 0)
|
|
|
|
wait_for_task();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-04 04:04:23 +01:00
|
|
|
/*static void kill_tasks(void)
|
2021-01-30 14:06:09 +01:00
|
|
|
{
|
|
|
|
if (_ntask > 0)
|
|
|
|
{
|
|
|
|
if (COMP_verbose)
|
|
|
|
fprintf(stderr, "gbc" GAMBAS_VERSION_STRING ": kill pending tasks\n");
|
2021-02-04 04:04:23 +01:00
|
|
|
kill(-getpid(), SIGTERM);
|
2021-01-30 14:06:09 +01:00
|
|
|
}
|
2021-02-04 04:04:23 +01:00
|
|
|
}*/
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2021-02-04 04:04:23 +01:00
|
|
|
static void run_task(BACKGROUND_TASK func, const char *arg)
|
2021-01-30 14:06:09 +01:00
|
|
|
{
|
|
|
|
pid_t pid;
|
|
|
|
|
|
|
|
if (_ntask_max <= 1)
|
|
|
|
{
|
|
|
|
(*func)(arg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (_ntask >= _ntask_max)
|
|
|
|
wait_for_task();
|
|
|
|
|
|
|
|
pid = fork();
|
|
|
|
|
|
|
|
if (pid < 0)
|
|
|
|
THROW("Failed to run child process: &1", strerror(errno));
|
|
|
|
|
|
|
|
if (pid == 0)
|
|
|
|
{
|
|
|
|
_child = TRUE;
|
|
|
|
_ntask = 0;
|
|
|
|
|
|
|
|
if (setpgid(0, getppid()) < 0)
|
|
|
|
ERROR_fail("[%d] setpgid to %d failed: %s", getpid(), getppid(), strerror(errno));
|
|
|
|
|
|
|
|
(*func)(arg);
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (COMP_verbose)
|
2021-02-04 04:04:23 +01:00
|
|
|
fprintf(stderr, "gbc" GAMBAS_VERSION_STRING ": start task: [%d] '%s'\n", pid, arg);
|
2021-01-30 14:06:09 +01:00
|
|
|
_ntask++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
static int compare_path(char **a, char **b)
|
|
|
|
{
|
|
|
|
return strcmp(*a, *b);
|
|
|
|
}
|
|
|
|
|
2020-10-01 23:22:30 +02:00
|
|
|
/*static bool check_cvs_directory(const char *dir)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
char *buffer;
|
|
|
|
struct stat info;
|
|
|
|
|
|
|
|
len = strlen(dir);
|
|
|
|
buffer = alloca(len + 32);
|
|
|
|
strcpy(buffer, dir);
|
|
|
|
|
|
|
|
strcpy(&buffer[len], "Root");
|
|
|
|
if (stat(buffer, &info))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
strcpy(&buffer[len], "Entries");
|
|
|
|
if (stat(buffer, &info))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
strcpy(&buffer[len], "Repository");
|
|
|
|
if (stat(buffer, &info))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}*/
|
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
static void fill_files(const char *root, bool recursive)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-29 22:51:10 +02:00
|
|
|
DIR *dir;
|
|
|
|
char *path;
|
|
|
|
struct dirent *dirent;
|
|
|
|
char *file_name;
|
|
|
|
const char *file;
|
|
|
|
struct stat info;
|
|
|
|
const char *ext;
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
path = STR_copy(root);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
dir = opendir(path);
|
2007-12-30 17:41:49 +01:00
|
|
|
if (!dir)
|
2021-01-30 14:06:09 +01:00
|
|
|
ERROR_fail("cannot browse directory: %s", path);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
while ((dirent = readdir(dir)) != NULL)
|
|
|
|
{
|
|
|
|
file_name = dirent->d_name;
|
|
|
|
if (*file_name == '.')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
file = FILE_cat(path, file_name, NULL);
|
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
if (stat(file, &info))
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2019-01-19 01:03:57 +01:00
|
|
|
ERROR_warning("cannot stat file: %s", file);
|
2007-12-30 17:41:49 +01:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
if (S_ISDIR(info.st_mode))
|
|
|
|
{
|
|
|
|
if (recursive)
|
2020-10-01 23:22:30 +02:00
|
|
|
{
|
|
|
|
if (*file_name == 'C')
|
|
|
|
{
|
|
|
|
if (strcmp(file_name, "CVS") == 0) // && check_cvs_directory(file))
|
|
|
|
continue;
|
|
|
|
if (strcmp(file_name, "CVSROOT") == 0)
|
|
|
|
continue;
|
|
|
|
}
|
2009-07-01 02:00:47 +02:00
|
|
|
fill_files(file, TRUE);
|
2020-10-01 23:22:30 +02:00
|
|
|
}
|
2009-07-01 02:00:47 +02:00
|
|
|
}
|
|
|
|
else
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
|
|
|
ext = FILE_get_ext(file);
|
|
|
|
|
2011-10-16 21:24:01 +02:00
|
|
|
if ((strcmp(ext, "module") == 0)
|
|
|
|
|| (strcmp(ext, "class") == 0))
|
2009-07-03 22:15:26 +02:00
|
|
|
{
|
2010-08-29 22:51:10 +02:00
|
|
|
*((char **)ARRAY_add(&_files)) = STR_copy(file);
|
2009-07-03 22:15:26 +02:00
|
|
|
}
|
2020-02-22 15:57:46 +01:00
|
|
|
else if (strcmp(ext, "test") == 0)
|
|
|
|
{
|
|
|
|
*((char **)ARRAY_add(&_files)) = STR_copy(file);
|
|
|
|
make_test = TRUE;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
closedir(dir);
|
|
|
|
STR_free(path);
|
2009-07-01 02:00:47 +02:00
|
|
|
}
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
static void init_files(const char *first)
|
|
|
|
{
|
|
|
|
bool recursive;
|
2017-11-19 23:18:23 +01:00
|
|
|
const char *name;
|
2020-05-09 20:55:04 +02:00
|
|
|
const char *ext;
|
2017-11-19 23:18:23 +01:00
|
|
|
int i, n;
|
2020-05-09 20:55:04 +02:00
|
|
|
bool has_test;
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
ARRAY_create(&_files);
|
2009-07-01 02:00:47 +02:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
if (*first)
|
2010-10-31 23:05:29 +01:00
|
|
|
FILE_chdir(first);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2009-07-01 02:00:47 +02:00
|
|
|
recursive = chdir(".src") == 0;
|
|
|
|
fill_files(FILE_get_current_dir(), recursive);
|
2010-10-31 23:05:29 +01:00
|
|
|
if (recursive) FILE_chdir("..");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
// Sort paths
|
2017-11-19 23:18:23 +01:00
|
|
|
n = ARRAY_count(_files);
|
|
|
|
qsort(_files, n, sizeof(*_files), (int (*)(const void *, const void *))compare_path);
|
|
|
|
|
|
|
|
// Add the classes to the list of classes
|
2020-05-09 20:55:04 +02:00
|
|
|
has_test = FALSE;
|
2017-11-19 23:18:23 +01:00
|
|
|
for (i = 0; i < n; i++)
|
|
|
|
{
|
2020-05-09 20:55:04 +02:00
|
|
|
if (!has_test)
|
|
|
|
{
|
|
|
|
ext = FILE_get_ext(_files[i]);
|
|
|
|
if (strcmp(ext, "test") == 0)
|
|
|
|
{
|
|
|
|
has_test = TRUE;
|
|
|
|
COMPILE_add_component("gb.test");
|
|
|
|
}
|
|
|
|
}
|
2017-11-19 23:18:23 +01:00
|
|
|
name = FILE_get_basename(_files[i]);
|
|
|
|
COMPILE_add_class(name, strlen(name));
|
|
|
|
}
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2009-07-03 22:15:26 +02:00
|
|
|
// End the list of classes
|
2010-05-22 20:02:34 +02:00
|
|
|
COMPILE_end_class();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void exit_files(void)
|
|
|
|
{
|
|
|
|
int i;
|
2020-02-22 15:57:46 +01:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
for (i = 0; i < ARRAY_count(_files); i++)
|
|
|
|
STR_free(_files[i]);
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
ARRAY_delete(&_files);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void compile_lang(const char *file_po)
|
|
|
|
{
|
2021-11-08 01:43:27 +01:00
|
|
|
const char *file_log;
|
|
|
|
char *file_mo;
|
2021-01-30 14:06:09 +01:00
|
|
|
time_t time_po, time_mo;
|
|
|
|
char *cmd;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
time_po = FILE_get_time(file_po);
|
|
|
|
|
|
|
|
if (time_po == ((time_t)-1))
|
|
|
|
return;
|
2021-11-08 01:43:27 +01:00
|
|
|
|
|
|
|
file_mo = (char *)FILE_set_ext(file_po, "mo");
|
2021-01-30 14:06:09 +01:00
|
|
|
|
|
|
|
if (!main_compile_all)
|
|
|
|
{
|
|
|
|
time_mo = FILE_get_time(file_mo);
|
|
|
|
if (time_mo >= time_po)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-08 01:43:27 +01:00
|
|
|
file_mo = STR_copy(file_mo);
|
|
|
|
file_log = FILE_set_ext(file_po, "log");
|
|
|
|
unlink(file_log);
|
2021-01-30 14:06:09 +01:00
|
|
|
unlink(file_mo);
|
|
|
|
|
|
|
|
// Shell "msgfmt -o " & Shell$(sPath) & " " & Shell(sTrans) Wait
|
|
|
|
if (COMP_verbose)
|
|
|
|
{
|
|
|
|
cmd = STR_print("msgfmt -o %s %s 2>&1", file_mo, file_po);
|
|
|
|
fprintf(stderr, "running: %s\n", cmd);
|
2020-02-22 15:57:46 +01:00
|
|
|
}
|
|
|
|
else
|
2021-11-08 01:43:27 +01:00
|
|
|
cmd = STR_print("msgfmt -o %s %s > %s 2>&1", file_mo, file_po, file_log);
|
2020-02-22 15:57:46 +01:00
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
ret = system(cmd);
|
|
|
|
|
|
|
|
if (!WIFEXITED(ret) || WEXITSTATUS(ret))
|
|
|
|
ERROR_warning("unable to compile translation file with 'msgfmt': %s", file_po);
|
|
|
|
|
2021-11-08 01:43:27 +01:00
|
|
|
if (FILE_get_size(file_log) == 0)
|
|
|
|
unlink(file_log);
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
STR_free(cmd);
|
2021-11-08 01:43:27 +01:00
|
|
|
STR_free(file_mo);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
static void compile_all_lang(void)
|
2019-01-19 01:03:57 +01:00
|
|
|
{
|
|
|
|
DIR *dir;
|
|
|
|
char *path;
|
|
|
|
struct dirent *dirent;
|
|
|
|
char *file_name;
|
|
|
|
int i;
|
|
|
|
char c;
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
path = STR_copy(FILE_cat(COMP_dir, ".lang", NULL));
|
2019-01-19 01:03:57 +01:00
|
|
|
FILE_chdir(path);
|
|
|
|
|
|
|
|
dir = opendir(".");
|
|
|
|
if (!dir)
|
|
|
|
{
|
|
|
|
ERROR_warning("cannot browse directory: %s", path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((dirent = readdir(dir)) != NULL)
|
|
|
|
{
|
|
|
|
file_name = dirent->d_name;
|
|
|
|
if (*file_name == '.')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (strcmp(FILE_get_ext(file_name), "po"))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (i = 0; i < strlen(file_name); i++)
|
|
|
|
{
|
|
|
|
c = file_name[i];
|
|
|
|
if (c == '.')
|
|
|
|
break;
|
|
|
|
if (!isalnum(c))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-01-30 14:06:09 +01:00
|
|
|
run_task(compile_lang, file_name);
|
2019-01-19 01:03:57 +01:00
|
|
|
}
|
2021-01-30 14:06:09 +01:00
|
|
|
|
|
|
|
wait_for_all_task();
|
2019-01-19 01:03:57 +01:00
|
|
|
|
|
|
|
closedir(dir);
|
|
|
|
STR_free(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
int i;
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
MEMORY_init();
|
|
|
|
COMMON_init();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
TRY
|
|
|
|
{
|
2011-09-28 23:10:18 +02:00
|
|
|
get_arguments(argc, argv);
|
2021-01-30 15:25:08 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
if (_convert_form)
|
2021-01-30 15:25:08 +01:00
|
|
|
{
|
2022-02-05 18:41:36 +01:00
|
|
|
FORM_convert(_convert_form);
|
|
|
|
STR_free(_convert_form);
|
2021-01-30 15:25:08 +01:00
|
|
|
}
|
2022-02-05 18:41:36 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (_ntask_max == 0)
|
|
|
|
_ntask_max = SYSTEM_get_cpu_count() + 1;
|
|
|
|
|
|
|
|
if (_ntask_max >= 2)
|
|
|
|
{
|
|
|
|
if (setpgid(0, 0))
|
|
|
|
ERROR_fail("setpgid() fails: %s", strerror(errno));
|
|
|
|
}
|
2011-09-28 23:10:18 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
COMPILE_init();
|
2022-12-11 16:44:46 +01:00
|
|
|
COMP_do_not_lock = FALSE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
// Remove information files if we are compiling everything
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
if (main_compile_all)
|
|
|
|
{
|
|
|
|
if (COMP_verbose)
|
|
|
|
fputs("Removing .info and .list files", stderr);
|
|
|
|
FILE_chdir(COMP_dir);
|
|
|
|
FILE_unlink(".info");
|
|
|
|
FILE_unlink(".list");
|
|
|
|
}
|
2008-09-22 01:22:07 +02:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
init_files(COMP_dir);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
for (i = 0; i < ARRAY_count(_files); i++)
|
|
|
|
run_task(compile_file, _files[i]);
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
wait_for_all_task();
|
|
|
|
|
|
|
|
COMPILE_remove_lock(".gbc.lock");
|
|
|
|
COMPILE_remove_lock(".gbc.stderr");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
exit_files();
|
|
|
|
|
|
|
|
if (main_trans)
|
|
|
|
compile_all_lang();
|
|
|
|
|
2022-07-22 20:40:20 +02:00
|
|
|
COMPILE_exit(_ntask_max == 1);
|
2022-02-05 18:41:36 +01:00
|
|
|
FILE_exit();
|
2016-02-01 03:18:49 +01:00
|
|
|
|
2022-02-05 18:41:36 +01:00
|
|
|
puts("OK");
|
|
|
|
}
|
2010-08-29 22:51:10 +02:00
|
|
|
}
|
|
|
|
CATCH
|
|
|
|
{
|
2021-02-04 04:04:23 +01:00
|
|
|
//wait_for_all_task();
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2010-08-29 22:51:10 +02:00
|
|
|
fflush(NULL);
|
2021-01-30 14:06:09 +01:00
|
|
|
|
2012-09-04 09:29:39 +02:00
|
|
|
COMPILE_print(MSG_ERROR, -1, NULL);
|
2010-08-29 22:51:10 +02:00
|
|
|
ERROR_print();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
END_TRY
|
|
|
|
|
|
|
|
return 0;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|