gambas-source-code/main/gbx/gbx_subr_misc.c
Benoît Minisini f532a37249 [CONFIGURATION]
* NEW: WARNING! The bytecode has changed, so every project needs to be
  recompiled.

[WEB SITE MAKER]
* NEW: Some more translations.

[INTERPRETER]
* NEW: Assign() is a new function. It is like Eval(), except that the
  expression can be an assignment.

[GB.EVAL]
* NEW: Implementation of the new Assign() function.
* NEW: The Expression.Compile() method takes a new optional boolean 
  argument that indicates if we are compiling an assignment or not.

[GB.NET.CURL]
* BUG: A asynchronous transfer that ends do not stop the other ones 
  anymore.


git-svn-id: svn://localhost/gambas/trunk@1986 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2009-05-26 22:34:39 +00:00

373 lines
6.4 KiB
C

/***************************************************************************
subr_misc.c
Miscellaneous subroutine
(c) 2000-2007 Benoit Minisini <gambas@users.sourceforge.net>
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 1, 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 the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
***************************************************************************/
#include <sys/wait.h>
#include <time.h>
#include "gb_common.h"
#include "gbx_subr.h"
#include "gbx_class.h"
#include "gambas.h"
#include "gbx_eval.h"
#include "gbx_date.h"
#include "gbx_archive.h"
#include "gbx_api.h"
#include "gbx_c_collection.h"
#include "gbx_c_process.h"
#include "gbx_debug.h"
#include "gbx_watch.h"
#include "gbx_math.h"
static EVAL_INTERFACE EVAL;
static CCOLLECTION *eval_env;
static void init_eval()
{
static bool init = FALSE;
if (init)
return;
COMPONENT_load(COMPONENT_create("gb.eval"));
LIBRARY_get_interface_by_name("gb.eval", EVAL_INTERFACE_VERSION, &EVAL);
init = TRUE;
}
void SUBR_error(void)
{
SP->type = T_BOOLEAN;
SP->_boolean.value = EXEC_got_error ? -1 : 0;
SP++;
}
void SUBR_wait(void)
{
struct timespec rem;
double wait;
double stop, time;
SUBR_ENTER();
if (NPARAM == 0)
{
HOOK_DEFAULT(wait, WATCH_wait)(0);
}
else
{
wait = SUBR_get_float(PARAM);
DATE_timer(&stop, FALSE);
stop += wait;
for(;;)
{
HOOK_DEFAULT(wait, WATCH_wait)((int)(wait * 1000 + 0.5));
if (DATE_timer(&time, FALSE))
break;
wait = stop - time;
if (wait <= 0.0)
break;
rem.tv_sec = 0;
rem.tv_nsec = 1000000;
nanosleep(&rem, &rem);
}
}
SUBR_LEAVE_VOID();
}
void SUBR_sleep(void)
{
double wait;
struct timespec rem;
SUBR_ENTER_PARAM(1);
wait = SUBR_get_float(PARAM);
rem.tv_sec = (time_t)(int)wait;
rem.tv_nsec = (int)(frac(wait) * 1E9);
while (nanosleep(&rem, &rem) < 0);
SUBR_LEAVE();
}
void SUBR_exec(void)
{
void *cmd;
bool wait;
int mode;
CPROCESS *process;
bool ret;
bool shell;
char *name;
CARRAY *env;
SUBR_ENTER_PARAM(4);
shell = (EXEC_code & 0x1F) != 0;
if (shell)
cmd = (void *)SUBR_get_string(PARAM);
else
{
VALUE_conv(PARAM, (TYPE)CLASS_StringArray);
cmd = (void *)(PARAM->_object.object);
}
if (!cmd)
THROW(E_ARG);
if (VALUE_is_null(&PARAM[1]))
env = NULL;
else
{
VALUE_conv(&PARAM[1], (TYPE)CLASS_StringArray);
env = (PARAM[1]._object.object);
}
VALUE_conv(&PARAM[2], T_INTEGER);
mode = PARAM[2]._integer.value;
wait = mode & PM_WAIT;
name = SUBR_get_string(&PARAM[3]);
ret = !PCODE_is_void(EXEC_code);
if (shell)
mode |= PM_SHELL;
//STRING_ref(name); ## This should not be needed
process = CPROCESS_create(mode, cmd, name, env);
//STRING_unref(&name);
if (wait)
{
OBJECT_REF(process, "subr_exec");
CPROCESS_wait_for(process);
if (!ret)
{
OBJECT_UNREF(process, "subr_exec");
}
else if (!process->to_string)
{
OBJECT_UNREF_KEEP(process, "subr_exec");
}
}
if (ret)
{
if (process->to_string)
{
RETURN->type = T_STRING;
RETURN->_string.addr = process->result;
RETURN->_string.start = 0;
RETURN->_string.len = STRING_length(process->result);
STRING_unref_keep(&process->result);
process->result = NULL;
OBJECT_UNREF(process, "subr_exec");
}
else
{
RETURN->_object.class = CLASS_Process;
RETURN->_object.object = process;
}
SUBR_LEAVE();
}
else
{
SUBR_LEAVE_VOID();
}
}
static bool get_value(const char *sym, int len, GB_VARIANT *value)
{
if (eval_env)
if (!GB_CollectionGet(eval_env, sym, len, value))
return FALSE;
value->type = GB_T_NULL;
return TRUE;
}
void EVAL_string(char *expr)
{
int len;
EXPRESSION *eval;
init_eval();
len = strlen(expr);
eval_env = NULL;
EVAL.New((void **)(void *)&eval, expr, len);
if (EVAL.Compile(eval, FALSE))
goto _ERROR;
if (!EVAL.Run(eval, get_value))
goto _ERROR;
goto _FREE;
_ERROR:
EVAL.Free((void **)(void *)&eval);
PROPAGATE();
_FREE:
EVAL.Free((void **)(void *)&eval);
VALUE_to_string(RETURN, &expr, &len);
STREAM_write(CSTREAM_stream(CFILE_out), expr, len);
STREAM_write_eol(CSTREAM_stream(CFILE_out));
STREAM_flush(CSTREAM_stream(CFILE_out));
}
void SUBR_eval(void)
{
char *expr;
int len;
EXPRESSION *eval;
SUBR_ENTER();
init_eval();
SUBR_get_string_len(PARAM, &expr, &len);
if (NPARAM == 2)
{
VALUE_conv(&PARAM[1], (TYPE)CLASS_Collection);
eval_env = (CCOLLECTION *)(PARAM[1]._object.object);
}
else
eval_env = NULL;
EVAL.New((void **)(void *)&eval, expr, len);
if (EVAL.Compile(eval, (EXEC_code >> 8) == CODE_ASSIGN))
goto _ERROR;
if (!EVAL.Run(eval, get_value))
goto _ERROR;
goto _FREE;
_ERROR:
EVAL.Free((void **)(void *)&eval);
PROPAGATE();
_FREE:
EVAL.Free((void **)(void *)&eval);
SUBR_LEAVE();
}
static TYPE conv_type(TYPE type)
{
if (type <= T_BYTE)
type = T_BYTE;
/*else if (type <= T_INTEGER)
;*/
else if (type == T_CSTRING || type == T_NULL)
type = T_STRING;
return type;
}
void SUBR_array(void)
{
TYPE type, type2;
int i;
GB_ARRAY array;
SUBR_ENTER();
type = conv_type(PARAM[0].type);
if (NPARAM >= 2)
{
type2 = conv_type(PARAM[1].type);
if (type != type2)
{
if (TYPE_is_object(type) && TYPE_is_object(type2))
type = T_OBJECT;
else
type = T_VARIANT;
}
}
GB_ArrayNew(&array, type, NPARAM);
for (i = 0; i < NPARAM; i++)
{
VALUE_conv(&PARAM[i], type);
GB_Store(type, (GB_VALUE *)&PARAM[i], GB_ArrayGet(array, i));
}
RETURN->_object.class = OBJECT_class(array); //CLASS_Array;
RETURN->_object.object = array;
SUBR_LEAVE();
}
void SUBR_collection(void)
{
int i;
GB_COLLECTION col;
char *key;
int len;
VALUE *vkey, *vval;
SUBR_ENTER();
GB_CollectionNew(&col, GB_COMP_BINARY);
for (i = 0; i < NPARAM; i += 2)
{
vkey = &PARAM[i];
vval = vkey + 1;
SUBR_get_string_len(vkey, &key, &len);
VALUE_conv(vval, T_VARIANT);
GB_CollectionSet(col, key, len, (GB_VARIANT *)vval);
}
RETURN->_object.class = OBJECT_class(col); //CLASS_Array;
RETURN->_object.object = col;
SUBR_LEAVE();
}