winamp/Src/ns-eel2/nseel-compiler.c
2024-09-24 14:54:57 +02:00

5700 lines
187 KiB
C

/*
Expression Evaluator Library (NS-EEL) v2
Copyright (C) 2004-2013 Cockos Incorporated
Copyright (C) 1999-2003 Nullsoft, Inc.
nseel-compiler.c
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
#include "ns-eel-int.h"
#include "denormal.h"
#include "wdlcstring.h"
#include <string.h>
#include <math.h>
#include <stdio.h>
#include <ctype.h>
#if !defined(EEL_TARGET_PORTABLE) && !defined(_WIN32)
#include <sys/mman.h>
#include <stdint.h>
#include <unistd.h>
#endif
#include "glue_x86.h"
#ifdef _WIN64
#include "glue_x86_64.h"
#endif // _WIN64
#define NSEEL_VARS_MALLOC_CHUNKSIZE 8
//#define LOG_OPT
//#define EEL_PRINT_FAILS
//#define EEL_VALIDATE_WORKTABLE_USE
//#define EEL_VALIDATE_FSTUBS
#ifdef EEL_PRINT_FAILS
#ifdef _WIN32
#define RET_MINUS1_FAIL(x) { OutputDebugString(x); return -1; }
#else
#define RET_MINUS1_FAIL(x) { printf("%s\n",x); return -1; }
#endif
#else
#define RET_MINUS1_FAIL(x) return -1;
#endif
#ifdef EEL_DUMP_OPS
FILE *g_eel_dump_fp, *g_eel_dump_fp2;
#endif
#ifdef EEL_VALIDATE_WORKTABLE_USE
#define MIN_COMPUTABLE_SIZE 0
#define COMPUTABLE_EXTRA_SPACE 64 // safety buffer, if EEL_VALIDATE_WORKTABLE_USE set, used for magic-value-checking
#else
#define MIN_COMPUTABLE_SIZE 32 // always use at least this big of a temp storage table (and reset the temp ptr when it goes past this boundary)
#define COMPUTABLE_EXTRA_SPACE 16 // safety buffer, if EEL_VALIDATE_WORKTABLE_USE set, used for magic-value-checking
#endif
/*
P1 is rightmost parameter
P2 is second rightmost, if any
P3 is third rightmost, if any
registers on x86 are (RAX etc on x86-64)
P1(ret) EAX
P2 EDI
P3 ECX
WTP RSI
x86_64: r12 is a pointer to ram_state.blocks
x86_64: r13 is a pointer to closenessfactor
registers on PPC are:
P1(ret) r3
P2 r14
P3 r15
WTP r16 (r17 has the original value)
r13 is a pointer to ram_state.blocks
ppc uses f31 and f30 and others for certain constants
*/
#ifdef EEL_TARGET_PORTABLE
#define EEL_DOESNT_NEED_EXEC_PERMS
#include "glue_port.h"
#elif defined(__ppc__)
#include "glue_ppc.h"
#elif defined(__aarch64__)
#include "glue_aarch64.h"
#elif defined(__arm__) || (defined (_M_ARM) && _M_ARM == 7)
#include "glue_arm.h"
#elif defined(_WIN64) || defined(__LP64__)
#include "glue_x86_64.h"
#else
#include "glue_x86.h"
#endif
#ifndef GLUE_INVSQRT_NEEDREPL
#define GLUE_INVSQRT_NEEDREPL 0
#endif
// used by //#eel-no-optimize:xxx, in ctx->optimizeDisableFlags
#define OPTFLAG_NO_OPTIMIZE 1
#define OPTFLAG_NO_FPSTACK 2
#define OPTFLAG_NO_INLINEFUNC 4
#define OPTFLAG_FULL_DENORMAL_CHECKS 8 // if set, denormals/NaN are always filtered on assign
#define OPTFLAG_NO_DENORMAL_CHECKS 16 // if set and FULL not set, denormals/NaN are never filtered on assign
#define DENORMAL_CLEARING_THRESHOLD 1.0e-50 // when adding/subtracting a constant, assume if it's greater than this, it will clear denormal (the actual value is probably 10^-290...)
#define MAX_SUB_NAMESPACES 32
typedef struct
{
const char *namespacePathToThis;
const char *subParmInfo[MAX_SUB_NAMESPACES];
} namespaceInformation;
static int nseel_evallib_stats[5]; // source bytes, static code bytes, call code bytes, data bytes, segments
int *NSEEL_getstats()
{
return nseel_evallib_stats;
}
static int findLineNumber(const char *exp, int byteoffs)
{
int lc=0;
while (byteoffs-->0 && *exp) if (*exp++ =='\n') lc++;
return lc;
}
static int nseel_vms_referencing_globallist_cnt;
nseel_globalVarItem *nseel_globalreg_list;
static EEL_F *get_global_var(compileContext *ctx, const char *gv, int addIfNotPresent);
static void *__newBlock(llBlock **start,int size, int wantMprotect);
#define OPCODE_IS_TRIVIAL(x) ((x)->opcodeType <= OPCODETYPE_VARPTRPTR)
enum {
OPCODETYPE_DIRECTVALUE=0,
OPCODETYPE_DIRECTVALUE_TEMPSTRING, // like directvalue, but will generate a new tempstring value on generate
OPCODETYPE_VALUE_FROM_NAMESPACENAME, // this.* or namespace.* are encoded this way
OPCODETYPE_VARPTR,
OPCODETYPE_VARPTRPTR,
OPCODETYPE_FUNC1,
OPCODETYPE_FUNC2,
OPCODETYPE_FUNC3,
OPCODETYPE_FUNCX,
OPCODETYPE_MOREPARAMS,
OPCODETYPE_INVALID,
};
struct opcodeRec
{
int opcodeType;
int fntype;
void *fn;
union {
struct opcodeRec *parms[3];
struct {
double directValue;
EEL_F *valuePtr; // if direct value, valuePtr can be cached
} dv;
} parms;
int namespaceidx;
// OPCODETYPE_VALUE_FROM_NAMESPACENAME (relname is either empty or blah)
// OPCODETYPE_VARPTR if it represents a global variable, will be nonempty
// OPCODETYPE_FUNC* with fntype=FUNCTYPE_EELFUNC
const char *relname;
};
static void *newTmpBlock(compileContext *ctx, int size)
{
const int align = 8;
const int a1=align-1;
char *p=(char*)__newBlock(&ctx->tmpblocks_head,size+a1, 0);
return p+((align-(((INT_PTR)p)&a1))&a1);
}
static void *__newBlock_align(compileContext *ctx, int size, int align, int isForCode)
{
const int a1=align-1;
char *p=(char*)__newBlock(
(
isForCode < 0 ? (isForCode == -2 ? &ctx->pblocks : &ctx->tmpblocks_head) :
isForCode > 0 ? &ctx->blocks_head :
&ctx->blocks_head_data) ,size+a1, isForCode>0);
return p+((align-(((INT_PTR)p)&a1))&a1);
}
static opcodeRec *newOpCode(compileContext *ctx, const char *str, int opType)
{
const size_t strszfull = str ? strlen(str) : 0;
const size_t str_sz = wdl_min(NSEEL_MAX_VARIABLE_NAMELEN, strszfull);
opcodeRec *rec = (opcodeRec*)__newBlock_align(ctx,
(int) (sizeof(opcodeRec) + (str_sz>0 ? str_sz+1 : 0)),
8, ctx->isSharedFunctions ? 0 : -1);
if (rec)
{
memset(rec,0,sizeof(*rec));
rec->opcodeType = opType;
if (str_sz > 0)
{
char *p = (char *)(rec+1);
memcpy(p,str,str_sz);
p[str_sz]=0;
rec->relname = p;
}
else
{
rec->relname = "";
}
}
return rec;
}
#define newCodeBlock(x,a) __newBlock_align(ctx,x,a,1)
#define newDataBlock(x,a) __newBlock_align(ctx,x,a,0)
#define newCtxDataBlock(x,a) __newBlock_align(ctx,x,a,-2)
static void freeBlocks(llBlock **start);
static int __growbuf_resize(eel_growbuf *buf, int newsize)
{
if (newsize<0)
{
free(buf->ptr);
buf->ptr=NULL;
buf->alloc=buf->size=0;
return 0;
}
if (newsize > buf->alloc)
{
const int newalloc = newsize + 4096 + newsize/2;
void *newptr = realloc(buf->ptr,newalloc);
if (!newptr)
{
newptr = malloc(newalloc);
if (!newptr) return 1;
if (buf->ptr && buf->size) memcpy(newptr,buf->ptr,buf->size);
free(buf->ptr);
buf->ptr=newptr;
}
else
buf->ptr = newptr;
buf->alloc=newalloc;
}
buf->size = newsize;
return 0;
}
#ifndef DECL_ASMFUNC
#define DECL_ASMFUNC(x) \
void nseel_asm_##x(void); \
void nseel_asm_##x##_end(void);
void _asm_megabuf(void);
void _asm_megabuf_end(void);
void _asm_gmegabuf(void);
void _asm_gmegabuf_end(void);
#endif
DECL_ASMFUNC(booltofp)
DECL_ASMFUNC(fptobool)
DECL_ASMFUNC(fptobool_rev)
DECL_ASMFUNC(sin)
DECL_ASMFUNC(cos)
DECL_ASMFUNC(tan)
DECL_ASMFUNC(1pdd)
DECL_ASMFUNC(2pdd)
DECL_ASMFUNC(2pdds)
DECL_ASMFUNC(1pp)
DECL_ASMFUNC(2pp)
DECL_ASMFUNC(sqr)
DECL_ASMFUNC(sqrt)
DECL_ASMFUNC(log)
DECL_ASMFUNC(log10)
DECL_ASMFUNC(abs)
DECL_ASMFUNC(min)
DECL_ASMFUNC(max)
DECL_ASMFUNC(min_fp)
DECL_ASMFUNC(max_fp)
DECL_ASMFUNC(sig)
DECL_ASMFUNC(sign)
DECL_ASMFUNC(band)
DECL_ASMFUNC(bor)
DECL_ASMFUNC(bnot)
DECL_ASMFUNC(bnotnot)
DECL_ASMFUNC(if)
DECL_ASMFUNC(fcall)
DECL_ASMFUNC(repeat)
DECL_ASMFUNC(repeatwhile)
DECL_ASMFUNC(equal)
DECL_ASMFUNC(equal_exact)
DECL_ASMFUNC(notequal_exact)
DECL_ASMFUNC(notequal)
DECL_ASMFUNC(below)
DECL_ASMFUNC(above)
DECL_ASMFUNC(beloweq)
DECL_ASMFUNC(aboveeq)
DECL_ASMFUNC(assign)
DECL_ASMFUNC(assign_fromfp)
DECL_ASMFUNC(assign_fast)
DECL_ASMFUNC(assign_fast_fromfp)
DECL_ASMFUNC(add)
DECL_ASMFUNC(sub)
DECL_ASMFUNC(add_op)
DECL_ASMFUNC(sub_op)
DECL_ASMFUNC(add_op_fast)
DECL_ASMFUNC(sub_op_fast)
DECL_ASMFUNC(mul)
DECL_ASMFUNC(div)
DECL_ASMFUNC(mul_op)
DECL_ASMFUNC(div_op)
DECL_ASMFUNC(mul_op_fast)
DECL_ASMFUNC(div_op_fast)
DECL_ASMFUNC(mod)
DECL_ASMFUNC(shl)
DECL_ASMFUNC(shr)
DECL_ASMFUNC(mod_op)
DECL_ASMFUNC(or)
DECL_ASMFUNC(or0)
DECL_ASMFUNC(xor)
DECL_ASMFUNC(xor_op)
DECL_ASMFUNC(and)
DECL_ASMFUNC(or_op)
DECL_ASMFUNC(and_op)
DECL_ASMFUNC(uplus)
DECL_ASMFUNC(uminus)
DECL_ASMFUNC(invsqrt)
DECL_ASMFUNC(dbg_getstackptr)
#ifdef NSEEL_EEL1_COMPAT_MODE
DECL_ASMFUNC(exec2)
#endif
DECL_ASMFUNC(stack_push)
DECL_ASMFUNC(stack_pop)
DECL_ASMFUNC(stack_pop_fast) // just returns value, doesn't mod param
DECL_ASMFUNC(stack_peek)
DECL_ASMFUNC(stack_peek_int)
DECL_ASMFUNC(stack_peek_top)
DECL_ASMFUNC(stack_exch)
static void *NSEEL_PProc_GRAM(void *data, int data_size, compileContext *ctx)
{
if (data_size>0) data=EEL_GLUE_set_immediate(data, (INT_PTR)ctx->gram_blocks);
return data;
}
static void *NSEEL_PProc_Stack(void *data, int data_size, compileContext *ctx)
{
codeHandleType *ch=ctx->tmpCodeHandle;
if (data_size>0)
{
UINT_PTR m1=(UINT_PTR)(NSEEL_STACK_SIZE * sizeof(EEL_F) - 1);
UINT_PTR stackptr = ((UINT_PTR) (&ch->stack));
ch->want_stack=1;
if (!ch->stack) ch->stack = newDataBlock(NSEEL_STACK_SIZE*sizeof(EEL_F),NSEEL_STACK_SIZE*sizeof(EEL_F));
data=EEL_GLUE_set_immediate(data, stackptr);
data=EEL_GLUE_set_immediate(data, m1); // and
data=EEL_GLUE_set_immediate(data, ((UINT_PTR)ch->stack&~m1)); //or
}
return data;
}
static void *NSEEL_PProc_Stack_PeekInt(void *data, int data_size, compileContext *ctx, INT_PTR offs)
{
codeHandleType *ch=ctx->tmpCodeHandle;
if (data_size>0)
{
UINT_PTR m1=(UINT_PTR)(NSEEL_STACK_SIZE * sizeof(EEL_F) - 1);
UINT_PTR stackptr = ((UINT_PTR) (&ch->stack));
ch->want_stack=1;
if (!ch->stack) ch->stack = newDataBlock(NSEEL_STACK_SIZE*sizeof(EEL_F),NSEEL_STACK_SIZE*sizeof(EEL_F));
data=EEL_GLUE_set_immediate(data, stackptr);
data=EEL_GLUE_set_immediate(data, offs);
data=EEL_GLUE_set_immediate(data, m1); // and
data=EEL_GLUE_set_immediate(data, ((UINT_PTR)ch->stack&~m1)); //or
}
return data;
}
static void *NSEEL_PProc_Stack_PeekTop(void *data, int data_size, compileContext *ctx)
{
codeHandleType *ch=ctx->tmpCodeHandle;
if (data_size>0)
{
UINT_PTR stackptr = ((UINT_PTR) (&ch->stack));
ch->want_stack=1;
if (!ch->stack) ch->stack = newDataBlock(NSEEL_STACK_SIZE*sizeof(EEL_F),NSEEL_STACK_SIZE*sizeof(EEL_F));
data=EEL_GLUE_set_immediate(data, stackptr);
}
return data;
}
#if defined(_MSC_VER) && _MSC_VER >= 1400
//static double __floor(double a) { return floor(a); }
//static double __ceil(double a) { return ceil(a); }
#define floor __floor
#define ceil __ceil
#endif
#ifdef NSEEL_EEL1_COMPAT_MODE
static double eel1band(double a, double b)
{
return (fabs(a)>NSEEL_CLOSEFACTOR && fabs(b) > NSEEL_CLOSEFACTOR) ? 1.0 : 0.0;
}
static double eel1bor(double a, double b)
{
return (fabs(a)>NSEEL_CLOSEFACTOR || fabs(b) > NSEEL_CLOSEFACTOR) ? 1.0 : 0.0;
}
static double eel1sigmoid(double x, double constraint)
{
double t = (1+exp(-x * (constraint)));
return fabs(t)>NSEEL_CLOSEFACTOR ? 1.0/t : 0;
}
#endif
#define FUNCTIONTYPE_PARAMETERCOUNTMASK 0xff
#define BIF_NPARAMS_MASK 0x7ffff00
#define BIF_RETURNSONSTACK 0x0000100
#define BIF_LASTPARMONSTACK 0x0000200
#define BIF_RETURNSBOOL 0x0000400
#define BIF_LASTPARM_ASBOOL 0x0000800
// 0x00?0000 -- taken by FP stack flags
#define BIF_TAKES_VARPARM 0x0400000
#define BIF_TAKES_VARPARM_EX 0x0C00000 // this is like varparm but check count exactly
#define BIF_WONTMAKEDENORMAL 0x0100000
#define BIF_CLEARDENORMAL 0x0200000
#if defined(GLUE_HAS_FXCH) && GLUE_MAX_FPSTACK_SIZE > 0
#define BIF_SECONDLASTPARMST 0x0001000 // use with BIF_LASTPARMONSTACK only (last two parameters get passed on fp stack)
#define BIF_LAZYPARMORDERING 0x0002000 // allow optimizer to avoid fxch when using BIF_TWOPARMSONFPSTACK_LAZY etc
#define BIF_REVERSEFPORDER 0x0004000 // force a fxch (reverse order of last two parameters on fp stack, used by comparison functions)
#ifndef BIF_FPSTACKUSE
#define BIF_FPSTACKUSE(x) (((x)>=0&&(x)<8) ? ((7-(x))<<16):0)
#endif
#ifndef BIF_GETFPSTACKUSE
#define BIF_GETFPSTACKUSE(x) (7 - (((x)>>16)&7))
#endif
#else
// do not support fp stack use unless GLUE_HAS_FXCH and GLUE_MAX_FPSTACK_SIZE>0
#define BIF_SECONDLASTPARMST 0
#define BIF_LAZYPARMORDERING 0
#define BIF_REVERSEFPORDER 0
#define BIF_FPSTACKUSE(x) 0
#define BIF_GETFPSTACKUSE(x) 0
#endif
#define BIF_TWOPARMSONFPSTACK (BIF_SECONDLASTPARMST|BIF_LASTPARMONSTACK)
#define BIF_TWOPARMSONFPSTACK_LAZY (BIF_LAZYPARMORDERING|BIF_SECONDLASTPARMST|BIF_LASTPARMONSTACK)
#ifndef GLUE_HAS_NATIVE_TRIGSQRTLOG
static double sqrt_fabs(double a) { return sqrt(fabs(a)); }
#endif
EEL_F NSEEL_CGEN_CALL nseel_int_rand(EEL_F f);
#define FNPTR_HAS_CONDITIONAL_EXEC(op) \
(op->fntype == FN_LOGICAL_AND || \
op->fntype == FN_LOGICAL_OR || \
op->fntype == FN_IF_ELSE || \
op->fntype == FN_WHILE || \
op->fntype == FN_LOOP)
static functionType fnTable1[] = {
#ifndef GLUE_HAS_NATIVE_TRIGSQRTLOG
{ "sin", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_WONTMAKEDENORMAL, {&sin} },
{ "cos", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL, {&cos} },
{ "tan", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&tan} },
{ "sqrt", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_WONTMAKEDENORMAL, {&sqrt_fabs}, },
{ "log", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&log} },
{ "log10", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&log10} },
#else
{ "sin", nseel_asm_sin,nseel_asm_sin_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_WONTMAKEDENORMAL|BIF_FPSTACKUSE(1) },
{ "cos", nseel_asm_cos,nseel_asm_cos_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL|BIF_FPSTACKUSE(1) },
{ "tan", nseel_asm_tan,nseel_asm_tan_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1) },
{ "sqrt", nseel_asm_sqrt,nseel_asm_sqrt_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1)|BIF_WONTMAKEDENORMAL },
{ "log", nseel_asm_log,nseel_asm_log_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(3), },
{ "log10", nseel_asm_log10,nseel_asm_log10_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(3), },
#endif
{ "asin", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&asin}, },
{ "acos", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&acos}, },
{ "atan", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&atan}, },
{ "atan2", nseel_asm_2pdd,nseel_asm_2pdd_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK, {&atan2}, },
{ "exp", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&exp}, },
{ "abs", nseel_asm_abs,nseel_asm_abs_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(0)|BIF_WONTMAKEDENORMAL },
{ "sqr", nseel_asm_sqr,nseel_asm_sqr_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1) },
{ "min", nseel_asm_min,nseel_asm_min_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_FPSTACKUSE(3)|BIF_WONTMAKEDENORMAL },
{ "max", nseel_asm_max,nseel_asm_max_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_FPSTACKUSE(3)|BIF_WONTMAKEDENORMAL },
{ "sign", nseel_asm_sign,nseel_asm_sign_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL, },
{ "rand", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL, {&nseel_int_rand}, },
//{ "floor", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL, {&floor} },
//{ "ceil", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL, {&ceil} },
{ "invsqrt", nseel_asm_invsqrt,nseel_asm_invsqrt_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(3), {GLUE_INVSQRT_NEEDREPL} },
{ "__dbg_getstackptr", nseel_asm_dbg_getstackptr,nseel_asm_dbg_getstackptr_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1), },
#ifdef NSEEL_EEL1_COMPAT_MODE
{ "sigmoid", nseel_asm_2pdd,nseel_asm_2pdd_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK, {&eel1sigmoid}, },
// these differ from _and/_or, they always evaluate both...
{ "band", nseel_asm_2pdd,nseel_asm_2pdd_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_CLEARDENORMAL , {&eel1band}, },
{ "bor", nseel_asm_2pdd,nseel_asm_2pdd_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_CLEARDENORMAL , {&eel1bor}, },
{"exec2",nseel_asm_exec2,nseel_asm_exec2_end,2|NSEEL_NPARAMS_FLAG_CONST|BIF_WONTMAKEDENORMAL},
{"exec3",nseel_asm_exec2,nseel_asm_exec2_end,3|NSEEL_NPARAMS_FLAG_CONST|BIF_WONTMAKEDENORMAL},
#endif // end EEL1 compat
{"freembuf",_asm_generic1parm,_asm_generic1parm_end,1,{&__NSEEL_RAM_MemFree},NSEEL_PProc_RAM},
{"memcpy",_asm_generic3parm,_asm_generic3parm_end,3,{&__NSEEL_RAM_MemCpy},NSEEL_PProc_RAM},
{"memset",_asm_generic3parm,_asm_generic3parm_end,3,{&__NSEEL_RAM_MemSet},NSEEL_PProc_RAM},
{"__memtop",_asm_generic1parm,_asm_generic1parm_end,1,{&__NSEEL_RAM_MemTop},NSEEL_PProc_RAM},
{"mem_set_values",_asm_generic2parm_retd,_asm_generic2parm_retd_end,2|BIF_TAKES_VARPARM|BIF_RETURNSONSTACK,{&__NSEEL_RAM_Mem_SetValues},NSEEL_PProc_RAM},
{"mem_get_values",_asm_generic2parm_retd,_asm_generic2parm_retd_end,2|BIF_TAKES_VARPARM|BIF_RETURNSONSTACK,{&__NSEEL_RAM_Mem_GetValues},NSEEL_PProc_RAM},
{"stack_push",nseel_asm_stack_push,nseel_asm_stack_push_end,1|BIF_FPSTACKUSE(0),{0,},NSEEL_PProc_Stack},
{"stack_pop",nseel_asm_stack_pop,nseel_asm_stack_pop_end,1|BIF_FPSTACKUSE(1),{0,},NSEEL_PProc_Stack},
{"stack_peek",nseel_asm_stack_peek,nseel_asm_stack_peek_end,1|NSEEL_NPARAMS_FLAG_CONST|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(0),{0,},NSEEL_PProc_Stack},
{"stack_exch",nseel_asm_stack_exch,nseel_asm_stack_exch_end,1|BIF_FPSTACKUSE(1), {0,},NSEEL_PProc_Stack_PeekTop},
};
static eel_function_table default_user_funcs;
static int functable_lowerbound(functionType *list, int list_sz, const char *name, int *ismatch)
{
int a = 0, c = list_sz;
while (a != c)
{
const int b = (a+c)/2;
const int cmp = stricmp(name,list[b].name);
if (cmp > 0) a = b+1;
else if (cmp < 0) c = b;
else
{
*ismatch = 1;
return b;
}
}
*ismatch = 0;
return a;
}
static int funcTypeCmp(const void *a, const void *b) { return stricmp(((functionType*)a)->name,((functionType*)b)->name); }
functionType *nseel_getFunctionByName(compileContext *ctx, const char *name, int *mchk)
{
eel_function_table *tab = ctx && ctx->registered_func_tab ? ctx->registered_func_tab : &default_user_funcs;
static char sorted;
const int fn1size = (int) (sizeof(fnTable1)/sizeof(fnTable1[0]));
int idx,match;
if (!sorted)
{
NSEEL_HOSTSTUB_EnterMutex();
if (!sorted) qsort(fnTable1,fn1size,sizeof(fnTable1[0]),funcTypeCmp);
sorted=1;
NSEEL_HOSTSTUB_LeaveMutex();
}
idx=functable_lowerbound(fnTable1,fn1size,name,&match);
if (match) return fnTable1+idx;
if ((!ctx || !(ctx->current_compile_flags&NSEEL_CODE_COMPILE_FLAG_ONLY_BUILTIN_FUNCTIONS)) && tab->list)
{
idx=functable_lowerbound(tab->list,tab->list_size,name,&match);
if (match)
{
if (mchk)
{
while (idx>0 && !stricmp(tab->list[idx-1].name,name)) idx--;
*mchk = tab->list_size - 1 - idx;
}
return tab->list + idx;
}
}
return NULL;
}
int NSEEL_init() // returns 0 on success
{
#ifdef EEL_VALIDATE_FSTUBS
int a;
for (a=0;a < sizeof(fnTable1)/sizeof(fnTable1[0]);a++)
{
char *code_startaddr = (char*)fnTable1[a].afunc;
char *endp = (char *)fnTable1[a].func_e;
// validate
int sz=0;
char *f=(char *)GLUE_realAddress(code_startaddr,endp,&sz);
if (f+sz > endp)
{
#ifdef _WIN32
OutputDebugString("bad eel function stub\n");
#else
printf("bad eel function stub\n");
#endif
*(char *)NULL = 0;
}
}
#ifdef _WIN32
OutputDebugString("eel function stub (builtin) validation complete\n");
#else
printf("eel function stub (builtin) validation complete\n");
#endif
#endif
NSEEL_quit();
return 0;
}
void NSEEL_quit()
{
free(default_user_funcs.list);
default_user_funcs.list = NULL;
default_user_funcs.list_size = 0;
}
void NSEEL_addfunc_varparm_ex(const char *name, int min_np, int want_exact, NSEEL_PPPROC pproc, EEL_F (NSEEL_CGEN_CALL *fptr)(void *, INT_PTR, EEL_F **), eel_function_table *destination)
{
const int sz = (int) ((char *)_asm_generic2parm_retd_end-(char *)_asm_generic2parm_retd);
NSEEL_addfunctionex2(name,min_np|(want_exact?BIF_TAKES_VARPARM_EX:BIF_TAKES_VARPARM),(char *)_asm_generic2parm_retd,sz,pproc,fptr,NULL,destination);
}
void NSEEL_addfunc_ret_type(const char *name, int np, int ret_type, NSEEL_PPPROC pproc, void *fptr, eel_function_table *destination) // ret_type=-1 for bool, 1 for value, 0 for ptr
{
char *stub=NULL;
int stubsz=0;
#define DOSTUB(np) { \
stub = (ret_type == 1 ? (char*)_asm_generic##np##parm_retd : (char*)_asm_generic##np##parm); \
stubsz = (int) ((ret_type == 1 ? (char*)_asm_generic##np##parm_retd_end : (char *)_asm_generic##np##parm_end) - stub); \
}
if (np == 1) DOSTUB(1)
else if (np == 2) DOSTUB(2)
else if (np == 3) DOSTUB(3)
#undef DOSTUB
if (stub) NSEEL_addfunctionex2(name,np|(ret_type == -1 ? BIF_RETURNSBOOL:0), stub, stubsz, pproc,fptr,NULL,destination);
}
void NSEEL_addfunctionex2(const char *name, int nparms, char *code_startaddr, int code_len, NSEEL_PPPROC pproc, void *fptr, void *fptr2, eel_function_table *destination)
{
const int list_size_chunk = 128;
functionType *r;
if (!destination) destination = &default_user_funcs;
if (!destination->list || !(destination->list_size & (list_size_chunk-1)))
{
void *nv = realloc(destination->list, (destination->list_size + list_size_chunk)*sizeof(functionType));
if (!nv) return;
destination->list = (functionType *)nv;
}
if (destination->list)
{
int match,idx;
idx=functable_lowerbound(destination->list,destination->list_size,name,&match);
#ifdef EEL_VALIDATE_FSTUBS
{
char *endp = code_startaddr+code_len;
// validate
int sz=0;
char *f=(char *)GLUE_realAddress(code_startaddr,endp,&sz);
if (f+sz > endp)
{
#ifdef _WIN32
OutputDebugString("bad eel function stub\n");
#else
printf("bad eel function stub\n");
#endif
*(char *)NULL = 0;
}
#ifdef _WIN32
OutputDebugString(name);
OutputDebugString(" - validated eel function stub\n");
#else
printf("eel function stub validation complete for %s\n",name);
#endif
}
#endif
r = destination->list + idx;
if (idx < destination->list_size)
memmove(r + 1, r, (destination->list_size - idx) * sizeof(functionType));
destination->list_size++;
memset(r, 0, sizeof(functionType));
if (!(nparms & BIF_RETURNSBOOL))
{
if (code_startaddr == (void *)&_asm_generic1parm_retd ||
code_startaddr == (void *)&_asm_generic2parm_retd ||
code_startaddr == (void *)&_asm_generic3parm_retd)
{
nparms |= BIF_RETURNSONSTACK;
}
}
r->nParams = nparms;
r->name = name;
r->afunc = code_startaddr;
r->func_e = code_startaddr + code_len;
r->pProc = pproc;
r->replptrs[0] = fptr;
r->replptrs[1] = fptr2;
}
}
//---------------------------------------------------------------------------------------------------------------
static void freeBlocks(llBlock **start)
{
llBlock *s=*start;
*start=0;
while (s)
{
llBlock *llB = s->next;
free(s);
s=llB;
}
}
//---------------------------------------------------------------------------------------------------------------
static void *__newBlock(llBlock **start, int size, int wantMprotect)
{
#if !defined(EEL_DOESNT_NEED_EXEC_PERMS) && defined(_WIN32)
DWORD ov;
UINT_PTR offs,eoffs;
#endif
llBlock *llb;
int alloc_size;
if (*start && (LLB_DSIZE - (*start)->sizeused) >= size)
{
void *t=(*start)->block+(*start)->sizeused;
(*start)->sizeused+=(size+7)&~7;
return t;
}
alloc_size=sizeof(llBlock);
if ((int)size > LLB_DSIZE) alloc_size += size - LLB_DSIZE;
llb = (llBlock *)malloc(alloc_size); // grab bigger block if absolutely necessary (heh)
if (!llb) return NULL;
#ifndef EEL_DOESNT_NEED_EXEC_PERMS
if (wantMprotect)
{
#ifdef _WIN32
offs=((UINT_PTR)llb)&~4095;
eoffs=((UINT_PTR)llb + alloc_size + 4095)&~4095;
VirtualProtect((LPVOID)offs,eoffs-offs,PAGE_EXECUTE_READWRITE,&ov);
// MessageBox(NULL,"vprotecting, yay\n","a",0);
#else
{
static int pagesize = 0;
if (!pagesize)
{
pagesize=sysconf(_SC_PAGESIZE);
if (!pagesize) pagesize=4096;
}
uintptr_t offs,eoffs;
offs=((uintptr_t)llb)&~(pagesize-1);
eoffs=((uintptr_t)llb + alloc_size + pagesize-1)&~(pagesize-1);
mprotect((void*)offs,eoffs-offs,PROT_WRITE|PROT_READ|PROT_EXEC);
}
#endif
}
#endif
llb->sizeused=(size+7)&~7;
llb->next = *start;
*start = llb;
return llb->block;
}
//---------------------------------------------------------------------------------------------------------------
opcodeRec *nseel_createCompiledValue(compileContext *ctx, EEL_F value)
{
opcodeRec *r=newOpCode(ctx,NULL,OPCODETYPE_DIRECTVALUE);
if (r)
{
r->parms.dv.directValue = value;
}
return r;
}
opcodeRec *nseel_createCompiledValuePtr(compileContext *ctx, EEL_F *addrValue, const char *namestr)
{
opcodeRec *r=newOpCode(ctx,namestr,OPCODETYPE_VARPTR);
if (!r) return 0;
r->parms.dv.valuePtr=addrValue;
return r;
}
static int validate_varname_for_function(compileContext *ctx, const char *name)
{
if (!ctx->function_curName || !ctx->function_globalFlag) return 1;
if (ctx->function_localTable_Size[2] > 0 && ctx->function_localTable_Names[2])
{
char * const * const namelist = ctx->function_localTable_Names[2];
const int namelist_sz = ctx->function_localTable_Size[2];
int i;
const size_t name_len = strlen(name);
for (i=0;i<namelist_sz;i++)
{
const char *nmchk=namelist[i];
const size_t l = strlen(nmchk);
if (l > 1 && nmchk[l-1] == '*')
{
if (name_len >= l && !strnicmp(nmchk,name,l-1) && name[l-1]=='.') return 1;
}
else
{
if (name_len == l && !stricmp(nmchk,name)) return 1;
}
}
}
return 0;
}
opcodeRec *nseel_resolve_named_symbol(compileContext *ctx, opcodeRec *rec, int parmcnt, int *errOut)
{
const int isFunctionMode = parmcnt >= 0;
int rel_prefix_len=0;
int rel_prefix_idx=-2;
int i;
char match_parmcnt[4]={-1,-1,-1,-1}; // [3] is guess
unsigned char match_parmcnt_pos=0;
char *sname = (char *)rec->relname;
int is_string_prefix = parmcnt < 0 && sname[0] == '#';
const char *prevent_function_calls = NULL;
if (errOut) *errOut = 0;
if (sname) sname += is_string_prefix;
if (rec->opcodeType != OPCODETYPE_VARPTR || !sname || !sname[0]) return NULL;
if (!isFunctionMode && !is_string_prefix && !strnicmp(sname,"reg",3) && isdigit(sname[3]) && isdigit(sname[4]) && !sname[5])
{
EEL_F *a=get_global_var(ctx,sname,1);
if (a)
{
rec->parms.dv.valuePtr = a;
sname[0]=0; // for dump_ops compat really, but this shouldn't be needed anyway
}
return rec;
}
if (ctx->function_curName)
{
if (!strnicmp(sname,"this.",5))
{
rel_prefix_len=5;
rel_prefix_idx=-1;
}
else if (!stricmp(sname,"this"))
{
rel_prefix_len=4;
rel_prefix_idx=-1;
}
// scan for parameters/local variables before user functions
if (rel_prefix_idx < -1 &&
ctx->function_localTable_Size[0] > 0 &&
ctx->function_localTable_Names[0] &&
ctx->function_localTable_ValuePtrs)
{
char * const * const namelist = ctx->function_localTable_Names[0];
const int namelist_sz = ctx->function_localTable_Size[0];
for (i=0; i < namelist_sz; i++)
{
const char *p = namelist[i];
if (p)
{
if (!isFunctionMode && !is_string_prefix && !strnicmp(p,sname,NSEEL_MAX_VARIABLE_NAMELEN))
{
rec->opcodeType = OPCODETYPE_VARPTRPTR;
rec->parms.dv.valuePtr=(EEL_F *)(ctx->function_localTable_ValuePtrs+i);
rec->parms.dv.directValue=0.0;
return rec;
}
else
{
const size_t plen = strlen(p);
if (plen > 1 && p[plen-1] == '*' && !strnicmp(p,sname,plen-1) && ((sname[plen-1] == '.'&&sname[plen]) || !sname[plen-1]))
{
rel_prefix_len=(int) (sname[plen-1] ? plen : plen-1);
rel_prefix_idx=i;
break;
}
}
}
}
}
// if instance name set, translate sname or sname.* into "this.sname.*"
if (rel_prefix_idx < -1 &&
ctx->function_localTable_Size[1] > 0 &&
ctx->function_localTable_Names[1])
{
char * const * const namelist = ctx->function_localTable_Names[1];
const int namelist_sz = ctx->function_localTable_Size[1];
const char *full_sname = rec->relname; // include # in checks
for (i=0; i < namelist_sz; i++)
{
const char *p = namelist[i];
if (p && *p)
{
const size_t tl = strlen(p);
if (!strnicmp(p,full_sname,tl) && (full_sname[tl] == 0 || full_sname[tl] == '.'))
{
rel_prefix_len=0; // treat as though this. prefixes is present
rel_prefix_idx=-1;
break;
}
}
}
}
if (rel_prefix_idx >= -1)
{
ctx->function_usesNamespaces=1;
}
} // ctx->function_curName
if (!isFunctionMode)
{
// instance variables
if (rel_prefix_idx >= -1)
{
rec->opcodeType = OPCODETYPE_VALUE_FROM_NAMESPACENAME;
rec->namespaceidx = rel_prefix_idx;
if (rel_prefix_len > 0)
{
if (is_string_prefix) sname[-1] = '#';
memmove(sname, sname+rel_prefix_len, strlen(sname + rel_prefix_len) + 1);
}
}
else
{
// no namespace index, so it must be a global
if (!validate_varname_for_function(ctx,rec->relname))
{
if (errOut) *errOut = 1;
if (ctx->last_error_string[0]) lstrcatn(ctx->last_error_string, ", ", sizeof(ctx->last_error_string));
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"global '%s' inaccessible",rec->relname);
return NULL;
}
}
return rec;
}
if (ctx->func_check)
prevent_function_calls = ctx->func_check(sname,ctx->func_check_user);
////////// function mode
// first off, while() and loop() are special and can't be overridden
//
if (parmcnt == 1 && !stricmp("while",sname) && !prevent_function_calls)
{
rec->opcodeType = OPCODETYPE_FUNC1;
rec->fntype = FN_WHILE;
return rec;
}
if (parmcnt == 2 && !stricmp("loop",sname) && !prevent_function_calls)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_LOOP;
return rec;
}
//
// resolve user function names before builtin functions -- this allows the user to override default functions
if (!(ctx->current_compile_flags & NSEEL_CODE_COMPILE_FLAG_ONLY_BUILTIN_FUNCTIONS))
{
_codeHandleFunctionRec *best=NULL;
size_t bestlen=0;
const char * const ourcall = sname+rel_prefix_len;
const size_t ourcall_len = strlen(ourcall);
int pass;
for (pass=0;pass<2;pass++)
{
_codeHandleFunctionRec *fr = pass ? ctx->functions_common : ctx->functions_local;
// sname is [namespace.[ns.]]function, find best match of function that matches the right end
while (fr)
{
int this_np = fr->num_params;
const char *thisfunc = fr->fname;
const size_t thisfunc_len = strlen(thisfunc);
if (this_np < 1) this_np=1;
if (thisfunc_len == ourcall_len && !stricmp(thisfunc,ourcall))
{
if (this_np == parmcnt)
{
bestlen = thisfunc_len;
best = fr;
break; // found exact match, finished
}
else
{
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = fr->num_params;
}
}
if (thisfunc_len > bestlen && thisfunc_len < ourcall_len && ourcall[ourcall_len - thisfunc_len - 1] == '.' && !stricmp(thisfunc,ourcall + ourcall_len - thisfunc_len))
{
if (this_np == parmcnt)
{
bestlen = thisfunc_len;
best = fr;
}
else
if (match_parmcnt[3]<0) match_parmcnt[3]=fr->num_params;
}
fr=fr->next;
}
if (fr) break; // found exact match, finished
}
if (best)
{
switch (parmcnt)
{
case 0:
case 1: rec->opcodeType = OPCODETYPE_FUNC1; break;
case 2: rec->opcodeType = OPCODETYPE_FUNC2; break;
case 3: rec->opcodeType = OPCODETYPE_FUNC3; break;
default: rec->opcodeType = OPCODETYPE_FUNCX; break;
}
if (ourcall != rec->relname) memmove((char *)rec->relname, ourcall, strlen(ourcall)+1);
if (ctx->function_curName && rel_prefix_idx<0)
{
// if no namespace specified, and this.commonprefix.func() called, remove common prefixes and set prefixidx to be this
const char *p=ctx->function_curName;
if (*p) p++;
while (*p && *p != '.') p++;
if (*p && p[1]) // we have a dot!
{
while (p[1]) p++; // go to last char of string, which doesn't allow possible trailing dot to be checked
while (--p > ctx->function_curName) // do not check possible leading dot
{
if (*p == '.')
{
const size_t cmplen = p+1-ctx->function_curName;
if (!strnicmp(rec->relname,ctx->function_curName,cmplen) && rec->relname[cmplen])
{
const char *src=rec->relname + cmplen;
memmove((char *)rec->relname, src, strlen(src)+1);
rel_prefix_idx=-1;
ctx->function_usesNamespaces=1;
break;
}
}
}
}
}
if (ctx->function_curName && rel_prefix_idx < -1 &&
strchr(rec->relname,'.') && !validate_varname_for_function(ctx,rec->relname))
{
if (errOut) *errOut = 1;
if (ctx->last_error_string[0]) lstrcatn(ctx->last_error_string, ", ", sizeof(ctx->last_error_string));
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"namespaced function '%s' inaccessible",rec->relname);
return NULL;
}
rec->namespaceidx = rel_prefix_idx;
rec->fntype = FUNCTYPE_EELFUNC;
rec->fn = best;
return rec;
}
}
if (prevent_function_calls)
{
if (ctx->last_error_string[0]) lstrcatn(ctx->last_error_string, ", ", sizeof(ctx->last_error_string));
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"'%.30s': %s",sname, prevent_function_calls);
if (errOut) *errOut = 0;
return NULL;
}
#ifdef NSEEL_EEL1_COMPAT_MODE
if (!stricmp(sname,"assign"))
{
if (parmcnt == 2)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_ASSIGN;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 2;
}
else if (!stricmp(sname,"if"))
{
if (parmcnt == 3)
{
rec->opcodeType = OPCODETYPE_FUNC3;
rec->fntype = FN_IF_ELSE;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 3;
}
else if (!stricmp(sname,"equal"))
{
if (parmcnt == 2)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_EQ;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 2;
}
else if (!stricmp(sname,"below"))
{
if (parmcnt == 2)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_LT;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 2;
}
else if (!stricmp(sname,"above"))
{
if (parmcnt == 2)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_GT;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 2;
}
else if (!stricmp(sname,"bnot"))
{
if (parmcnt == 1)
{
rec->opcodeType = OPCODETYPE_FUNC1;
rec->fntype = FN_NOT;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 1;
}
else if (!stricmp(sname,"megabuf"))
{
if (parmcnt == 1)
{
rec->opcodeType = OPCODETYPE_FUNC1;
rec->fntype = FN_MEMORY;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 1;
}
else if (!stricmp(sname,"gmegabuf"))
{
if (parmcnt == 1)
{
rec->opcodeType = OPCODETYPE_FUNC1;
rec->fntype = FN_GMEMORY;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 1;
}
else
#endif
// convert legacy pow() to FN_POW
if (!stricmp("pow",sname))
{
if (parmcnt == 2)
{
rec->opcodeType = OPCODETYPE_FUNC2;
rec->fntype = FN_POW;
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = 2;
}
else if (!stricmp("__denormal_likely",sname) || !stricmp("__denormal_unlikely",sname))
{
if (parmcnt == 1)
{
rec->opcodeType = OPCODETYPE_FUNC1;
rec->fntype = !stricmp("__denormal_likely",sname) ? FN_DENORMAL_LIKELY : FN_DENORMAL_UNLIKELY;
return rec;
}
}
{
int chkamt=0;
functionType *f=nseel_getFunctionByName(ctx,sname,&chkamt);
if (f) while (chkamt-->=0)
{
const int pc_needed=(f->nParams&FUNCTIONTYPE_PARAMETERCOUNTMASK);
if ((f->nParams&BIF_TAKES_VARPARM_EX)==BIF_TAKES_VARPARM ? (parmcnt >= pc_needed) : (parmcnt == pc_needed))
{
rec->fntype = FUNCTYPE_FUNCTIONTYPEREC;
rec->fn = (void *)f;
switch (parmcnt)
{
case 0:
case 1: rec->opcodeType = OPCODETYPE_FUNC1; break;
case 2: rec->opcodeType = OPCODETYPE_FUNC2; break;
case 3: rec->opcodeType = OPCODETYPE_FUNC3; break;
default: rec->opcodeType = OPCODETYPE_FUNCX; break;
}
return rec;
}
if (match_parmcnt_pos < 3) match_parmcnt[match_parmcnt_pos++] = (f->nParams&FUNCTIONTYPE_PARAMETERCOUNTMASK);
f++;
if (stricmp(f->name,sname)) break;
}
}
if (ctx->last_error_string[0]) lstrcatn(ctx->last_error_string, ", ", sizeof(ctx->last_error_string));
if (match_parmcnt[3] >= 0)
{
if (match_parmcnt_pos<3) match_parmcnt[match_parmcnt_pos] = match_parmcnt[3];
match_parmcnt_pos++;
}
if (!match_parmcnt_pos)
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"'%.30s' undefined",sname);
else
{
int x;
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"'%.30s' needs ",sname);
for (x = 0; x < match_parmcnt_pos; x++)
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"%s%d",x==0?"" : x == match_parmcnt_pos-1?" or ":",",match_parmcnt[x]);
lstrcatn(ctx->last_error_string," parms",sizeof(ctx->last_error_string));
}
if (errOut) *errOut = match_parmcnt_pos > 0 ? parmcnt<match_parmcnt[0]?2:(match_parmcnt[0] < 2 ? 4:1) : 0;
return NULL;
}
opcodeRec *nseel_setCompiledFunctionCallParameters(compileContext *ctx, opcodeRec *fn, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3, opcodeRec *postCode, int *errOut)
{
opcodeRec *r;
int np=0,x;
if (!fn || fn->opcodeType != OPCODETYPE_VARPTR || !fn->relname || !fn->relname[0])
{
return NULL;
}
fn->parms.parms[0] = code1;
fn->parms.parms[1] = code2;
fn->parms.parms[2] = code3;
for (x=0;x<3;x++)
{
opcodeRec *prni=fn->parms.parms[x];
while (prni && np < NSEEL_MAX_EELFUNC_PARAMETERS)
{
const int isMP = prni->opcodeType == OPCODETYPE_MOREPARAMS;
np++;
if (!isMP) break;
prni = prni->parms.parms[1];
}
}
r = nseel_resolve_named_symbol(ctx, fn, np<1 ? 1 : np ,errOut);
if (postCode && r)
{
if (code1 && r->opcodeType == OPCODETYPE_FUNC1 && r->fntype == FN_WHILE)
{
// change while(x) (postcode) to be
// while ((x) ? (postcode;1) : 0);
r->parms.parms[0] =
nseel_createIfElse(ctx,r->parms.parms[0],
nseel_createSimpleCompiledFunction(ctx,FN_JOIN_STATEMENTS,2,postCode,nseel_createCompiledValue(ctx,1.0f)),
NULL); // NULL defaults to 0.0
}
else
{
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"syntax error following function");
*errOut = -1;
return NULL;
}
}
return r;
}
struct eelStringSegmentRec *nseel_createStringSegmentRec(compileContext *ctx, const char *str, int len)
{
struct eelStringSegmentRec *r = newTmpBlock(ctx,sizeof(struct eelStringSegmentRec));
if (r)
{
r->_next=0;
r->str_start=str;
r->str_len = len;
}
return r;
}
opcodeRec *nseel_eelMakeOpcodeFromStringSegments(compileContext *ctx, struct eelStringSegmentRec *rec)
{
if (ctx && ctx->onString)
{
return nseel_createCompiledValue(ctx, ctx->onString(ctx->caller_this,rec));
}
return NULL;
}
opcodeRec *nseel_createMoreParametersOpcode(compileContext *ctx, opcodeRec *code1, opcodeRec *code2)
{
opcodeRec *r=code1 && code2 ? newOpCode(ctx,NULL,OPCODETYPE_MOREPARAMS) : NULL;
if (r)
{
r->parms.parms[0] = code1;
r->parms.parms[1] = code2;
}
return r;
}
opcodeRec *nseel_createIfElse(compileContext *ctx, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3)
{
opcodeRec *r=code1 ? newOpCode(ctx,NULL,OPCODETYPE_FUNC3) : NULL;
if (r)
{
if (!code2) code2 = nseel_createCompiledValue(ctx,0.0);
if (!code3) code3 = nseel_createCompiledValue(ctx,0.0);
if (!code2||!code3) return NULL;
r->fntype = FN_IF_ELSE;
r->parms.parms[0] = code1;
r->parms.parms[1] = code2;
r->parms.parms[2] = code3;
}
return r;
}
opcodeRec *nseel_createMemoryAccess(compileContext *ctx, opcodeRec *code1, opcodeRec *code2)
{
if (code1 && code1->opcodeType == OPCODETYPE_VARPTR && !stricmp(code1->relname,"gmem"))
{
return nseel_createSimpleCompiledFunction(ctx, FN_GMEMORY,1,code2?code2:nseel_createCompiledValue(ctx,0.0),0);
}
if (code2 && (code2->opcodeType != OPCODETYPE_DIRECTVALUE || code2->parms.dv.directValue != 0.0))
{
code1 = nseel_createSimpleCompiledFunction(ctx,FN_ADD,2,code1,code2);
}
return nseel_createSimpleCompiledFunction(ctx, FN_MEMORY,1,code1,0);
}
opcodeRec *nseel_createSimpleCompiledFunction(compileContext *ctx, int fn, int np, opcodeRec *code1, opcodeRec *code2)
{
opcodeRec *r=code1 && (np<2 || code2) ? newOpCode(ctx,NULL,np>=2 ? OPCODETYPE_FUNC2:OPCODETYPE_FUNC1) : NULL;
if (r)
{
r->fntype = fn;
r->parms.parms[0] = code1;
r->parms.parms[1] = code2;
if (fn == FN_JOIN_STATEMENTS)
{
r->fn = r; // for joins, fn is temporarily used for _tail pointers
if (code1 && code1->opcodeType == OPCODETYPE_FUNC2 && code1->fntype == fn)
{
opcodeRec *t = (opcodeRec *)code1->fn;
// keep joins in the form of dosomething->morestuff.
// in this instance, code1 is previous stuff to do, code2 is new stuff to do
r->parms.parms[0] = t->parms.parms[1];
code1->fn = (t->parms.parms[1] = r);
return code1;
}
}
}
return r;
}
// these are bitmasks; on request you can tell what is supported, and compileOpcodes will return one of them
#define RETURNVALUE_IGNORE 0 // ignore return value
#define RETURNVALUE_NORMAL 1 // pointer
#define RETURNVALUE_FPSTACK 2
#define RETURNVALUE_BOOL 4 // P1 is nonzero if true
#define RETURNVALUE_BOOL_REVERSED 8 // P1 is zero if true
#define RETURNVALUE_CACHEABLE 16 // only to be used when (at least) RETURNVALUE_NORMAL is set
static int compileOpcodes(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int bufOut_len, int *computTable, const namespaceInformation *namespacePathToThis,
int supportedReturnValues, int *rvType, int *fpStackUsage, int *canHaveDenormalOutput);
static unsigned char *compileCodeBlockWithRet(compileContext *ctx, opcodeRec *rec, int *computTableSize, const namespaceInformation *namespacePathToThis,
int supportedReturnValues, int *rvType, int *fpStackUse, int *canHaveDenormalOutput);
_codeHandleFunctionRec *eel_createFunctionNamespacedInstance(compileContext *ctx, _codeHandleFunctionRec *fr, const char *nameptr)
{
size_t n;
_codeHandleFunctionRec *subfr =
fr->isCommonFunction ?
ctx->isSharedFunctions ? newDataBlock(sizeof(_codeHandleFunctionRec),8) :
newCtxDataBlock(sizeof(_codeHandleFunctionRec),8) : // if common function, but derived version is in non-common context, set ownership to VM rather than us
newTmpBlock(ctx,sizeof(_codeHandleFunctionRec));
if (!subfr) return 0;
// fr points to functionname()'s rec, nameptr to blah.functionname()
*subfr = *fr;
n = strlen(nameptr);
if (n > sizeof(subfr->fname)-1) n=sizeof(subfr->fname)-1;
memcpy(subfr->fname,nameptr,n);
subfr->fname[n]=0;
subfr->next = NULL;
subfr->startptr=0; // make sure this code gets recompiled (with correct member ptrs) for this instance!
subfr->startptr_size=-1;
// subfr->derivedCopies already points to the right place
fr->derivedCopies = subfr;
return subfr;
}
static void combineNamespaceFields(char *nm, const namespaceInformation *namespaceInfo, const char *relname, int thisctx) // nm must be NSEEL_MAX_VARIABLE_NAMELEN+1 bytes
{
const char *prefix = namespaceInfo ?
thisctx<0 ? (thisctx == -1 ? namespaceInfo->namespacePathToThis : NULL) : (thisctx < MAX_SUB_NAMESPACES ? namespaceInfo->subParmInfo[thisctx] : NULL)
: NULL;
int lfp = 0, lrn=relname ? (int)strlen(relname) : 0;
if (prefix) while (prefix[lfp] && prefix[lfp] != ':' && lfp < NSEEL_MAX_VARIABLE_NAMELEN) lfp++;
if (!relname) relname = "";
while (*relname == '.') // if relname begins with ., then remove a chunk of context from prefix
{
relname++;
while (lfp>0 && prefix[lfp-1] != '.') lfp--;
if (lfp>0) lfp--;
}
if (lfp > NSEEL_MAX_VARIABLE_NAMELEN-3) lfp=NSEEL_MAX_VARIABLE_NAMELEN-3;
if (lfp>0) memcpy(nm,prefix,lfp);
if (lrn > NSEEL_MAX_VARIABLE_NAMELEN - lfp - (lfp>0)) lrn=NSEEL_MAX_VARIABLE_NAMELEN - lfp - (lfp>0);
if (lrn > 0)
{
if (lfp>0) nm[lfp++] = '.';
memcpy(nm+lfp,relname,lrn);
lfp+=lrn;
}
nm[lfp++]=0;
}
//---------------------------------------------------------------------------------------------------------------
static void *nseel_getBuiltinFunctionAddress(compileContext *ctx,
int fntype, void *fn,
NSEEL_PPPROC *pProc, void ***replList,
void **endP, int *abiInfo, int preferredReturnValues, const EEL_F *hasConstParm1, const EEL_F *hasConstParm2)
{
const EEL_F *firstConstParm = hasConstParm1 ? hasConstParm1 : hasConstParm2;
static void *pow_replptrs[4]={&pow,};
switch (fntype)
{
#define RF(x) *endP = nseel_asm_##x##_end; return (void*)nseel_asm_##x
case FN_MUL_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(mul_op);
case FN_DIV_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(div_op);
case FN_OR_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(or_op);
case FN_XOR_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(xor_op);
case FN_AND_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(and_op);
case FN_MOD_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(mod_op);
case FN_ADD_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(add_op);
case FN_SUB_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(sub_op);
case FN_POW_OP:
*abiInfo=BIF_LASTPARMONSTACK|BIF_CLEARDENORMAL;
*replList = pow_replptrs;
RF(2pdds);
case FN_POW:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK;//BIF_FPSTACKUSE(2) might be safe, need to look at pow()'s implementation, but safer bet is to disallow fp stack caching for this expression
*replList = pow_replptrs;
RF(2pdd);
case FN_ADD:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2);
// for x +- non-denormal-constant, we can set BIF_CLEARDENORMAL
if (firstConstParm && fabs(*firstConstParm) > DENORMAL_CLEARING_THRESHOLD) *abiInfo |= BIF_CLEARDENORMAL;
RF(add);
case FN_SUB:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_FPSTACKUSE(2);
// for x +- non-denormal-constant, we can set BIF_CLEARDENORMAL
if (firstConstParm && fabs(*firstConstParm) > DENORMAL_CLEARING_THRESHOLD) *abiInfo |= BIF_CLEARDENORMAL;
RF(sub);
case FN_MULTIPLY:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2);
// for x*constant-greater-than-eq-1, we can set BIF_WONTMAKEDENORMAL
if (firstConstParm && fabs(*firstConstParm) >= 1.0) *abiInfo |= BIF_WONTMAKEDENORMAL;
RF(mul);
case FN_DIVIDE:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_FPSTACKUSE(2);
// for x/constant-less-than-eq-1, we can set BIF_WONTMAKEDENORMAL
if (firstConstParm && fabs(*firstConstParm) <= 1.0) *abiInfo |= BIF_WONTMAKEDENORMAL;
RF(div);
case FN_MOD:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_FPSTACKUSE(1)|BIF_CLEARDENORMAL;
RF(mod);
case FN_ASSIGN:
*abiInfo = BIF_FPSTACKUSE(1)|BIF_CLEARDENORMAL;
RF(assign);
case FN_AND: *abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL; RF(and);
case FN_OR: *abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL; RF(or);
case FN_XOR:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(xor);
case FN_SHR:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(shr);
case FN_SHL:
*abiInfo = BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK|BIF_FPSTACKUSE(2)|BIF_CLEARDENORMAL;
RF(shl);
#ifndef EEL_TARGET_PORTABLE
case FN_NOTNOT: *abiInfo = BIF_LASTPARM_ASBOOL|BIF_RETURNSBOOL|BIF_FPSTACKUSE(1); RF(uplus);
#else
case FN_NOTNOT: *abiInfo = BIF_LASTPARM_ASBOOL|BIF_RETURNSBOOL|BIF_FPSTACKUSE(1); RF(bnotnot);
#endif
case FN_UMINUS: *abiInfo = BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_WONTMAKEDENORMAL; RF(uminus);
case FN_NOT: *abiInfo = BIF_LASTPARM_ASBOOL|BIF_RETURNSBOOL|BIF_FPSTACKUSE(1); RF(bnot);
case FN_EQ:
*abiInfo = BIF_TWOPARMSONFPSTACK_LAZY|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(equal);
case FN_EQ_EXACT:
*abiInfo=BIF_TWOPARMSONFPSTACK_LAZY|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(equal_exact);
case FN_NE:
*abiInfo=BIF_TWOPARMSONFPSTACK_LAZY|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(notequal);
case FN_NE_EXACT:
*abiInfo=BIF_TWOPARMSONFPSTACK_LAZY|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(notequal_exact);
case FN_LOGICAL_AND:
*abiInfo = BIF_RETURNSBOOL;
RF(band);
case FN_LOGICAL_OR:
*abiInfo = BIF_RETURNSBOOL;
RF(bor);
#ifdef GLUE_HAS_FXCH
case FN_GT:
*abiInfo = BIF_TWOPARMSONFPSTACK|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(above);
case FN_GTE:
*abiInfo = BIF_TWOPARMSONFPSTACK|BIF_RETURNSBOOL|BIF_REVERSEFPORDER|BIF_FPSTACKUSE(2);
RF(beloweq);
case FN_LT:
*abiInfo = BIF_TWOPARMSONFPSTACK|BIF_RETURNSBOOL|BIF_REVERSEFPORDER|BIF_FPSTACKUSE(2);
RF(above);
case FN_LTE:
*abiInfo = BIF_TWOPARMSONFPSTACK|BIF_RETURNSBOOL|BIF_FPSTACKUSE(2);
RF(beloweq);
#else
case FN_GT:
*abiInfo = BIF_RETURNSBOOL|BIF_LASTPARMONSTACK;
RF(above);
case FN_GTE:
*abiInfo = BIF_RETURNSBOOL|BIF_LASTPARMONSTACK;
RF(aboveeq);
case FN_LT:
*abiInfo = BIF_RETURNSBOOL|BIF_LASTPARMONSTACK;
RF(below);
case FN_LTE:
*abiInfo = BIF_RETURNSBOOL|BIF_LASTPARMONSTACK;
RF(beloweq);
#endif
#undef RF
#define RF(x) *endP = _asm_##x##_end; return (void*)_asm_##x
case FN_MEMORY:
{
static void *replptrs[4]={&__NSEEL_RAMAlloc,};
*replList = replptrs;
*abiInfo = BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1)|BIF_CLEARDENORMAL;
#ifdef GLUE_MEM_NEEDS_PPROC
*pProc = NSEEL_PProc_RAM;
#endif
RF(megabuf);
}
break;
case FN_GMEMORY:
{
static void *replptrs[4]={&__NSEEL_RAMAllocGMEM,};
*replList = replptrs;
*abiInfo=BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1)|BIF_CLEARDENORMAL;
*pProc=NSEEL_PProc_GRAM;
RF(gmegabuf);
}
break;
#undef RF
case FUNCTYPE_FUNCTIONTYPEREC:
if (fn)
{
functionType *p=(functionType *)fn;
// if prefers fpstack or bool, or ignoring value, then use fp-stack versions
if ((preferredReturnValues&(RETURNVALUE_BOOL|RETURNVALUE_FPSTACK)) || !preferredReturnValues)
{
static functionType min2={ "min", nseel_asm_min_fp,nseel_asm_min_fp_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2)|BIF_WONTMAKEDENORMAL };
static functionType max2={ "max", nseel_asm_max_fp,nseel_asm_max_fp_end, 2|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_TWOPARMSONFPSTACK_LAZY|BIF_FPSTACKUSE(2)|BIF_WONTMAKEDENORMAL };
if (p->afunc == (void*)nseel_asm_min) p = &min2;
else if (p->afunc == (void*)nseel_asm_max) p = &max2;
}
*replList=p->replptrs;
*pProc=p->pProc;
*endP = p->func_e;
*abiInfo = p->nParams & BIF_NPARAMS_MASK;
if (firstConstParm)
{
const char *name=p->name;
if (!strcmp(name,"min") && *firstConstParm < -1.0e-10) *abiInfo |= BIF_CLEARDENORMAL;
else if (!strcmp(name,"max") && *firstConstParm > 1.0e-10) *abiInfo |= BIF_CLEARDENORMAL;
}
return p->afunc;
}
break;
}
return 0;
}
static void *nseel_getEELFunctionAddress(compileContext *ctx,
opcodeRec *op,
int *customFuncParmSize, int *customFuncLocalStorageSize,
EEL_F ***customFuncLocalStorage, int *computTableTop,
void **endP, int *isRaw, int wantCodeGenerated,
const namespaceInformation *namespacePathToThis, int *rvMode, int *fpStackUse, int *canHaveDenormalOutput,
opcodeRec **ordered_parmptrs, int num_ordered_parmptrs
) // if wantCodeGenerated is false, can return bogus pointers in raw mode
{
_codeHandleFunctionRec *fn = (_codeHandleFunctionRec*)op->fn;
namespaceInformation local_namespace={NULL};
char prefix_buf[NSEEL_MAX_VARIABLE_NAMELEN+1], nm[NSEEL_MAX_FUNCSIG_NAME+1];
if (!fn) return NULL;
// op->relname ptr is [whatever.]funcname
if (fn->parameterAsNamespaceMask || fn->usesNamespaces)
{
if (wantCodeGenerated)
{
char *p = prefix_buf;
combineNamespaceFields(nm,namespacePathToThis,op->relname,op->namespaceidx);
lstrcpyn_safe(prefix_buf,nm,sizeof(prefix_buf));
local_namespace.namespacePathToThis = prefix_buf;
// nm is full path of function, prefix_buf will be the path not including function name (unless function name only)
while (*p) p++;
while (p >= prefix_buf && *p != '.') p--;
if (p > prefix_buf) *p=0;
}
if (fn->parameterAsNamespaceMask)
{
int x;
for(x=0;x<MAX_SUB_NAMESPACES && x < fn->num_params;x++)
{
if (fn->parameterAsNamespaceMask & (((unsigned int)1)<<x))
{
if (wantCodeGenerated)
{
const char *rn=NULL;
char tmp[NSEEL_MAX_VARIABLE_NAMELEN+1];
if (x < num_ordered_parmptrs && ordered_parmptrs[x])
{
if (ordered_parmptrs[x]->opcodeType == OPCODETYPE_VARPTR)
{
rn=ordered_parmptrs[x]->relname;
}
else if (ordered_parmptrs[x]->opcodeType == OPCODETYPE_VALUE_FROM_NAMESPACENAME)
{
const char *p=ordered_parmptrs[x]->relname;
if (*p == '#') p++;
combineNamespaceFields(tmp,namespacePathToThis,p,ordered_parmptrs[x]->namespaceidx);
rn = tmp;
}
}
if (!rn)
{
// todo: figure out how to give correct line number/offset (ugh)
snprintf(ctx->last_error_string,sizeof(ctx->last_error_string),"parameter %d to %.120s() must be namespace",x+1,fn->fname);
return NULL;
}
lstrcatn(nm,":",sizeof(nm));
local_namespace.subParmInfo[x] = nm+strlen(nm);
lstrcatn(nm,rn,sizeof(nm));
}
ordered_parmptrs[x] = NULL; // prevent caller from bothering generating parameters
}
}
}
if (wantCodeGenerated)
{
_codeHandleFunctionRec *fr = fn;
// find namespace-adjusted function (if generating code, otherwise assume size is the same)
fn = 0; // if this gets re-set, it will be the new function
while (fr && !fn)
{
if (!stricmp(fr->fname,nm)) fn = fr;
fr=fr->derivedCopies;
}
if (!fn) // generate copy of function
{
fn = eel_createFunctionNamespacedInstance(ctx,(_codeHandleFunctionRec*)op->fn,nm);
}
}
}
if (!fn) return NULL;
if (!fn->startptr && fn->opcodes && fn->startptr_size != 0)
{
int sz = fn->startptr_size;
if (sz < 0)
{
fn->tmpspace_req=0;
fn->rvMode = RETURNVALUE_IGNORE;
fn->canHaveDenormalOutput=0;
sz = compileOpcodes(ctx,fn->opcodes,NULL,128*1024*1024,&fn->tmpspace_req,
wantCodeGenerated ? &local_namespace : NULL,RETURNVALUE_NORMAL|RETURNVALUE_FPSTACK,
&fn->rvMode,&fn->fpStackUsage,&fn->canHaveDenormalOutput);
if (sz<0) return NULL;
fn->startptr_size = sz;
}
if (!wantCodeGenerated)
{
// don't compile anything for now, just give stats
if (computTableTop) *computTableTop += fn->tmpspace_req;
*customFuncParmSize = fn->num_params;
*customFuncLocalStorage = fn->localstorage;
*customFuncLocalStorageSize = fn->localstorage_size;
*rvMode = fn->rvMode;
*fpStackUse = fn->fpStackUsage;
if (canHaveDenormalOutput) *canHaveDenormalOutput=fn->canHaveDenormalOutput;
if (sz <= NSEEL_MAX_FUNCTION_SIZE_FOR_INLINE && !(ctx->optimizeDisableFlags&OPTFLAG_NO_INLINEFUNC))
{
*isRaw = 1;
*endP = ((char *)1) + sz;
return (char *)1;
}
*endP = (void*)nseel_asm_fcall_end;
return (void*)nseel_asm_fcall;
}
if (sz <= NSEEL_MAX_FUNCTION_SIZE_FOR_INLINE && !(ctx->optimizeDisableFlags&OPTFLAG_NO_INLINEFUNC))
{
void *p=newTmpBlock(ctx,sz);
fn->tmpspace_req=0;
if (p)
{
fn->canHaveDenormalOutput=0;
if (fn->isCommonFunction) ctx->isGeneratingCommonFunction++;
sz=compileOpcodes(ctx,fn->opcodes,(unsigned char*)p,sz,&fn->tmpspace_req,&local_namespace,RETURNVALUE_NORMAL|RETURNVALUE_FPSTACK,&fn->rvMode,&fn->fpStackUsage,&fn->canHaveDenormalOutput);
if (fn->isCommonFunction) ctx->isGeneratingCommonFunction--;
// recompile function with native context pointers
if (sz>0)
{
fn->startptr_size=sz;
fn->startptr=p;
}
}
}
else
{
unsigned char *codeCall;
fn->tmpspace_req=0;
fn->fpStackUsage=0;
fn->canHaveDenormalOutput=0;
if (fn->isCommonFunction) ctx->isGeneratingCommonFunction++;
codeCall=compileCodeBlockWithRet(ctx,fn->opcodes,&fn->tmpspace_req,&local_namespace,RETURNVALUE_NORMAL|RETURNVALUE_FPSTACK,&fn->rvMode,&fn->fpStackUsage,&fn->canHaveDenormalOutput);
if (fn->isCommonFunction) ctx->isGeneratingCommonFunction--;
if (codeCall)
{
void *f=GLUE_realAddress(nseel_asm_fcall,nseel_asm_fcall_end,&sz);
fn->startptr = newTmpBlock(ctx,sz);
if (fn->startptr)
{
memcpy(fn->startptr,f,sz);
EEL_GLUE_set_immediate(fn->startptr,(INT_PTR)codeCall);
fn->startptr_size = sz;
}
}
}
}
if (fn->startptr)
{
if (computTableTop) *computTableTop += fn->tmpspace_req;
*customFuncParmSize = fn->num_params;
*customFuncLocalStorage = fn->localstorage;
*customFuncLocalStorageSize = fn->localstorage_size;
*rvMode = fn->rvMode;
*fpStackUse = fn->fpStackUsage;
if (canHaveDenormalOutput) *canHaveDenormalOutput= fn->canHaveDenormalOutput;
*endP = (char*)fn->startptr + fn->startptr_size;
*isRaw=1;
return fn->startptr;
}
return 0;
}
// returns true if does something (other than calculating and throwing away a value)
static char optimizeOpcodes(compileContext *ctx, opcodeRec *op, int needsResult)
{
opcodeRec *lastJoinOp=NULL;
char retv, retv_parm[3], joined_retv=0;
while (op && op->opcodeType == OPCODETYPE_FUNC2 && op->fntype == FN_JOIN_STATEMENTS)
{
if (!optimizeOpcodes(ctx,op->parms.parms[0], 0) || OPCODE_IS_TRIVIAL(op->parms.parms[0]))
{
// direct value, can skip ourselves
memcpy(op,op->parms.parms[1],sizeof(*op));
}
else
{
joined_retv |= 1;
lastJoinOp = op;
op = op->parms.parms[1];
}
}
goto start_over;
#define RESTART_DIRECTVALUE(X) { op->parms.dv.directValue = (X); goto start_over_directvalue; }
start_over_directvalue:
op->opcodeType = OPCODETYPE_DIRECTVALUE;
op->parms.dv.valuePtr=NULL;
start_over: // when an opcode changed substantially in optimization, goto here to reprocess it
retv = retv_parm[0]=retv_parm[1]=retv_parm[2]=0;
if (!op || // should never really happen
OPCODE_IS_TRIVIAL(op) || // should happen often (vars)
op->opcodeType < 0 || op->opcodeType >= OPCODETYPE_INVALID // should never happen (assert would be appropriate heh)
) return joined_retv;
if (!needsResult)
{
if (op->fntype == FUNCTYPE_EELFUNC)
{
needsResult=1; // assume eel functions are non-const for now
}
else if (op->fntype == FUNCTYPE_FUNCTIONTYPEREC)
{
functionType *pfn = (functionType *)op->fn;
if (!pfn || !(pfn->nParams&NSEEL_NPARAMS_FLAG_CONST)) needsResult=1;
}
else if (op->fntype >= FN_NONCONST_BEGIN && op->fntype < FUNCTYPE_SIMPLEMAX)
{
needsResult=1;
}
}
if (op->opcodeType>=OPCODETYPE_FUNC2) retv_parm[1] = optimizeOpcodes(ctx,op->parms.parms[1], needsResult);
if (op->opcodeType>=OPCODETYPE_FUNC3) retv_parm[2] = optimizeOpcodes(ctx,op->parms.parms[2], needsResult);
retv_parm[0] = optimizeOpcodes(ctx,op->parms.parms[0], needsResult ||
(FNPTR_HAS_CONDITIONAL_EXEC(op) && (retv_parm[1] || retv_parm[2] || op->opcodeType <= OPCODETYPE_FUNC1)) );
if (op->opcodeType != OPCODETYPE_MOREPARAMS)
{
if (op->fntype >= 0 && op->fntype < FUNCTYPE_SIMPLEMAX)
{
if (op->opcodeType == OPCODETYPE_FUNC1) // within FUNCTYPE_SIMPLE
{
if (op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE)
{
switch (op->fntype)
{
case FN_NOTNOT: RESTART_DIRECTVALUE(fabs(op->parms.parms[0]->parms.dv.directValue)>=NSEEL_CLOSEFACTOR ? 1.0 : 0.0);
case FN_NOT: RESTART_DIRECTVALUE(fabs(op->parms.parms[0]->parms.dv.directValue)>=NSEEL_CLOSEFACTOR ? 0.0 : 1.0);
case FN_UMINUS: RESTART_DIRECTVALUE(- op->parms.parms[0]->parms.dv.directValue);
}
}
else if (op->fntype == FN_NOT || op->fntype == FN_NOTNOT)
{
if (op->parms.parms[0]->opcodeType == OPCODETYPE_FUNC1)
{
switch (op->parms.parms[0]->fntype)
{
case FN_UMINUS:
case FN_NOTNOT: // ignore any NOTNOTs UMINUS or UPLUS, they would have no effect anyway
op->parms.parms[0] = op->parms.parms[0]->parms.parms[0];
goto start_over;
case FN_NOT:
op->fntype = op->fntype==FN_NOT ? FN_NOTNOT : FN_NOT; // switch between FN_NOT and FN_NOTNOT
op->parms.parms[0] = op->parms.parms[0]->parms.parms[0];
goto start_over;
}
}
else if (op->parms.parms[0]->opcodeType == OPCODETYPE_FUNC2)
{
int repl_type = -1;
switch (op->parms.parms[0]->fntype)
{
case FN_EQ: repl_type = FN_NE; break;
case FN_NE: repl_type = FN_EQ; break;
case FN_EQ_EXACT: repl_type = FN_NE_EXACT; break;
case FN_NE_EXACT: repl_type = FN_EQ_EXACT; break;
case FN_LT: repl_type = FN_GTE; break;
case FN_LTE: repl_type = FN_GT; break;
case FN_GT: repl_type = FN_LTE; break;
case FN_GTE: repl_type = FN_LT; break;
}
if (repl_type != -1)
{
const int oldtype = op->fntype;
memcpy(op,op->parms.parms[0],sizeof(*op));
if (oldtype == FN_NOT) op->fntype = repl_type;
goto start_over;
}
}
}
}
else if (op->opcodeType == OPCODETYPE_FUNC2) // within FUNCTYPE_SIMPLE
{
const int dv0 = op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE;
const int dv1 = op->parms.parms[1]->opcodeType == OPCODETYPE_DIRECTVALUE;
if (dv0 && dv1)
{
int reval = -1;
switch (op->fntype)
{
case FN_MOD:
{
int a = (int) op->parms.parms[1]->parms.dv.directValue;
if (a)
{
a = (int) op->parms.parms[0]->parms.dv.directValue % a;
if (a<0) a=-a;
}
RESTART_DIRECTVALUE((EEL_F)a);
}
break;
case FN_SHL: RESTART_DIRECTVALUE(((int)op->parms.parms[0]->parms.dv.directValue) << ((int)op->parms.parms[1]->parms.dv.directValue));
case FN_SHR: RESTART_DIRECTVALUE(((int)op->parms.parms[0]->parms.dv.directValue) >> ((int)op->parms.parms[1]->parms.dv.directValue));
case FN_POW: RESTART_DIRECTVALUE(pow(op->parms.parms[0]->parms.dv.directValue, op->parms.parms[1]->parms.dv.directValue));
case FN_DIVIDE: RESTART_DIRECTVALUE(op->parms.parms[0]->parms.dv.directValue / op->parms.parms[1]->parms.dv.directValue);
case FN_MULTIPLY: RESTART_DIRECTVALUE(op->parms.parms[0]->parms.dv.directValue * op->parms.parms[1]->parms.dv.directValue);
case FN_ADD: RESTART_DIRECTVALUE(op->parms.parms[0]->parms.dv.directValue + op->parms.parms[1]->parms.dv.directValue);
case FN_SUB: RESTART_DIRECTVALUE(op->parms.parms[0]->parms.dv.directValue - op->parms.parms[1]->parms.dv.directValue);
case FN_AND: RESTART_DIRECTVALUE((double) (((WDL_INT64)op->parms.parms[0]->parms.dv.directValue) & ((WDL_INT64)op->parms.parms[1]->parms.dv.directValue)));
case FN_OR: RESTART_DIRECTVALUE((double) (((WDL_INT64)op->parms.parms[0]->parms.dv.directValue) | ((WDL_INT64)op->parms.parms[1]->parms.dv.directValue)));
case FN_XOR: RESTART_DIRECTVALUE((double) (((WDL_INT64)op->parms.parms[0]->parms.dv.directValue) ^ ((WDL_INT64)op->parms.parms[1]->parms.dv.directValue)));
case FN_EQ: reval = fabs(op->parms.parms[0]->parms.dv.directValue - op->parms.parms[1]->parms.dv.directValue) < NSEEL_CLOSEFACTOR; break;
case FN_NE: reval = fabs(op->parms.parms[0]->parms.dv.directValue - op->parms.parms[1]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR; break;
case FN_EQ_EXACT: reval = op->parms.parms[0]->parms.dv.directValue == op->parms.parms[1]->parms.dv.directValue; break;
case FN_NE_EXACT: reval = op->parms.parms[0]->parms.dv.directValue != op->parms.parms[1]->parms.dv.directValue; break;
case FN_LT: reval = op->parms.parms[0]->parms.dv.directValue < op->parms.parms[1]->parms.dv.directValue; break;
case FN_LTE: reval = op->parms.parms[0]->parms.dv.directValue <= op->parms.parms[1]->parms.dv.directValue; break;
case FN_GT: reval = op->parms.parms[0]->parms.dv.directValue > op->parms.parms[1]->parms.dv.directValue; break;
case FN_GTE: reval = op->parms.parms[0]->parms.dv.directValue >= op->parms.parms[1]->parms.dv.directValue; break;
case FN_LOGICAL_AND: reval = fabs(op->parms.parms[0]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR && fabs(op->parms.parms[1]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR; break;
case FN_LOGICAL_OR: reval = fabs(op->parms.parms[0]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR || fabs(op->parms.parms[1]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR; break;
}
if (reval >= 0) RESTART_DIRECTVALUE((EEL_F) reval);
}
else if (dv0 || dv1)
{
double dvalue = op->parms.parms[!dv0]->parms.dv.directValue;
switch (op->fntype)
{
case FN_OR:
case FN_XOR:
if (!(WDL_INT64)dvalue)
{
// replace with or0
static functionType fr={"or0",nseel_asm_or0, nseel_asm_or0_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_LASTPARMONSTACK|BIF_RETURNSONSTACK|BIF_CLEARDENORMAL, {0}, NULL};
op->opcodeType = OPCODETYPE_FUNC1;
op->fntype = FUNCTYPE_FUNCTIONTYPEREC;
op->fn = &fr;
if (dv0) op->parms.parms[0] = op->parms.parms[1];
goto start_over;
}
break;
case FN_SUB:
if (dv0)
{
if (dvalue == 0.0)
{
op->opcodeType = OPCODETYPE_FUNC1;
op->fntype = FN_UMINUS;
op->parms.parms[0] = op->parms.parms[1];
goto start_over;
}
break;
}
// fall through, if dv1 we can remove +0.0
case FN_ADD:
if (dvalue == 0.0)
{
memcpy(op,op->parms.parms[!!dv0],sizeof(*op));
goto start_over;
}
break;
case FN_AND:
if ((WDL_INT64)dvalue) break;
dvalue = 0.0; // treat x&0 as x*0, which optimizes to 0
// fall through
case FN_MULTIPLY:
if (dvalue == 0.0) // remove multiply by 0.0 (using 0.0 direct value as replacement), unless the nonzero side did something
{
if (!retv_parm[!!dv0])
{
memcpy(op,op->parms.parms[!dv0],sizeof(*op)); // set to 0 if other action wouldn't do anything
goto start_over;
}
else
{
// this is 0.0 * oldexpressionthatmustbeprocessed or oldexpressionthatmustbeprocessed*0.0
op->fntype = FN_JOIN_STATEMENTS;
if (dv0) // 0.0*oldexpression, reverse the order so that 0 is returned
{
// set to (oldexpression;0)
opcodeRec *tmp = op->parms.parms[1];
op->parms.parms[1] = op->parms.parms[0];
op->parms.parms[0] = tmp;
}
goto start_over;
}
}
else if (dvalue == 1.0) // remove multiply by 1.0 (using non-1.0 value as replacement)
{
memcpy(op,op->parms.parms[!!dv0],sizeof(*op));
goto start_over;
}
break;
case FN_POW:
if (dv1)
{
// x^0 = 1
if (fabs(dvalue) < 1e-30)
{
RESTART_DIRECTVALUE(1.0);
}
// x^1 = x
if (fabs(dvalue-1.0) < 1e-30)
{
memcpy(op,op->parms.parms[0],sizeof(*op));
goto start_over;
}
}
else if (dv0)
{
// pow(constant, x) = exp((x) * ln(constant)), if constant>0
// opcodeRec *parm0 = op->parms.parms[0];
if (dvalue > 0.0)
{
static functionType expcpy={ "exp", nseel_asm_1pdd,nseel_asm_1pdd_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK, {&exp}, };
// 1^x = 1
if (fabs(dvalue-1.0) < 1e-30)
{
RESTART_DIRECTVALUE(1.0);
}
dvalue=log(dvalue);
if (fabs(dvalue-1.0) < 1e-9)
{
// caller wanted e^x
op->parms.parms[0]=op->parms.parms[1];
}
else
{
// it would be nice to replace 10^x with exp(log(10)*x) or 2^x with exp(log(2),x), but
// doing so breaks rounding. we could maybe only allow 10^x, which is used for dB conversion,
// but for now we should just force the programmer do it exp(log(10)*x) themselves.
break;
/*
parm0->opcodeType = OPCODETYPE_FUNC2;
parm0->fntype = FN_MULTIPLY;
parm0->parms.parms[0] = nseel_createCompiledValue(ctx,dvalue);
parm0->parms.parms[1] = op->parms.parms[1];
*/
}
op->opcodeType = OPCODETYPE_FUNC1;
op->fntype = FUNCTYPE_FUNCTIONTYPEREC;
op->fn = &expcpy;
goto start_over;
}
}
break;
case FN_MOD:
if (dv1)
{
const int a = (int) dvalue;
if (!a)
{
RESTART_DIRECTVALUE(0.0);
}
}
break;
case FN_DIVIDE:
if (dv1)
{
if (dvalue == 1.0) // remove divide by 1.0 (using non-1.0 value as replacement)
{
memcpy(op,op->parms.parms[!!dv0],sizeof(*op));
goto start_over;
}
else
{
// change to a multiply
if (dvalue == 0.0)
{
op->fntype = FN_MULTIPLY;
goto start_over;
}
else
{
double d = 1.0/dvalue;
WDL_DenormalDoubleAccess *p = (WDL_DenormalDoubleAccess*)&d;
// allow conversion to multiply if reciprocal is exact
// we could also just look to see if the last few digits of the mantissa were 0, which would probably be good
// enough, but if the user really wants it they should do * (1/x) instead to force precalculation of reciprocal.
if (!p->w.lw && !(p->w.hw & 0xfffff))
{
op->fntype = FN_MULTIPLY;
op->parms.parms[1]->parms.dv.directValue = d;
op->parms.parms[1]->parms.dv.valuePtr=NULL;
goto start_over;
}
}
}
}
else if (dvalue == 0.0)
{
if (!retv_parm[!!dv0])
{
// if 0/x set to always 0.
// this is 0.0 / (oldexpression that can be eliminated)
memcpy(op,op->parms.parms[!dv0],sizeof(*op)); // set to 0 if other action wouldn't do anything
}
else
{
opcodeRec *tmp;
// this is 0.0 / oldexpressionthatmustbeprocessed
op->fntype = FN_JOIN_STATEMENTS;
tmp = op->parms.parms[1];
op->parms.parms[1] = op->parms.parms[0];
op->parms.parms[0] = tmp;
// set to (oldexpression;0)
}
goto start_over;
}
break;
case FN_EQ:
if (dvalue == 0.0)
{
// convert x == 0.0 to !x
op->opcodeType=OPCODETYPE_FUNC1;
op->fntype = FN_NOT;
if (dv0) op->parms.parms[0]=op->parms.parms[1];
goto start_over;
}
break;
case FN_NE:
if (dvalue == 0.0)
{
// convert x != 0.0 to !!
op->opcodeType=OPCODETYPE_FUNC1;
op->fntype = FN_NOTNOT;
if (dv0) op->parms.parms[0]=op->parms.parms[1];
goto start_over;
}
break;
case FN_LOGICAL_AND:
if (dv0)
{
// dvalue && expr
if (fabs(dvalue) < NSEEL_CLOSEFACTOR)
{
// 0 && expr, replace with 0
RESTART_DIRECTVALUE(0.0);
}
else
{
// 1 && expr, replace with 0 != expr
op->fntype = FN_NE;
op->parms.parms[0]->parms.dv.valuePtr=NULL;
op->parms.parms[0]->parms.dv.directValue = 0.0;
}
}
else
{
// expr && dvalue
if (fabs(dvalue) < NSEEL_CLOSEFACTOR)
{
// expr && 0
if (!retv_parm[0])
{
// expr has no consequence, drop it
RESTART_DIRECTVALUE(0.0);
}
else
{
// replace with (expr; 0)
op->fntype = FN_JOIN_STATEMENTS;
op->parms.parms[1]->parms.dv.valuePtr=NULL;
op->parms.parms[1]->parms.dv.directValue = 0.0;
}
}
else
{
// expr && 1, replace with expr != 0
op->fntype = FN_NE;
op->parms.parms[1]->parms.dv.valuePtr=NULL;
op->parms.parms[1]->parms.dv.directValue = 0.0;
}
}
goto start_over;
case FN_LOGICAL_OR:
if (dv0)
{
// dvalue || expr
if (fabs(dvalue) >= NSEEL_CLOSEFACTOR)
{
// 1 || expr, replace with 1
RESTART_DIRECTVALUE(1.0);
}
else
{
// 0 || expr, replace with 0 != expr
op->fntype = FN_NE;
op->parms.parms[0]->parms.dv.valuePtr=NULL;
op->parms.parms[0]->parms.dv.directValue = 0.0;
}
}
else
{
// expr || dvalue
if (fabs(dvalue) >= NSEEL_CLOSEFACTOR)
{
// expr || 1
if (!retv_parm[0])
{
// expr has no consequence, drop it and return 1
RESTART_DIRECTVALUE(1.0);
}
else
{
// replace with (expr; 1)
op->fntype = FN_JOIN_STATEMENTS;
op->parms.parms[1]->parms.dv.valuePtr=NULL;
op->parms.parms[1]->parms.dv.directValue = 1.0;
}
}
else
{
// expr || 0, replace with expr != 0
op->fntype = FN_NE;
op->parms.parms[1]->parms.dv.valuePtr=NULL;
op->parms.parms[1]->parms.dv.directValue = 0.0;
}
}
goto start_over;
}
} // dv0 || dv1
// general optimization of two parameters
switch (op->fntype)
{
case FN_MULTIPLY:
{
opcodeRec *first_parm = op->parms.parms[0],*second_parm = op->parms.parms[1];
if (second_parm->opcodeType == first_parm->opcodeType)
{
switch(first_parm->opcodeType)
{
case OPCODETYPE_VALUE_FROM_NAMESPACENAME:
if (first_parm->namespaceidx != second_parm->namespaceidx) break;
// fall through
case OPCODETYPE_VARPTR:
if (first_parm->relname && second_parm->relname && !stricmp(second_parm->relname,first_parm->relname)) second_parm=NULL;
break;
case OPCODETYPE_VARPTRPTR:
if (first_parm->parms.dv.valuePtr && first_parm->parms.dv.valuePtr==second_parm->parms.dv.valuePtr) second_parm=NULL;
break;
}
if (!second_parm) // switch from x*x to sqr(x)
{
static functionType sqrcpy={ "sqr", nseel_asm_sqr,nseel_asm_sqr_end, 1|NSEEL_NPARAMS_FLAG_CONST|BIF_RETURNSONSTACK|BIF_LASTPARMONSTACK|BIF_FPSTACKUSE(1) };
op->opcodeType = OPCODETYPE_FUNC1;
op->fntype = FUNCTYPE_FUNCTIONTYPEREC;
op->fn = &sqrcpy;
goto start_over;
}
}
}
break;
case FN_POW:
{
opcodeRec *first_parm = op->parms.parms[0];
if (first_parm->opcodeType == op->opcodeType && first_parm->fntype == FN_POW)
{
// since first_parm is a pow too, we can multiply the exponents.
// set our base to be the base of the inner pow
op->parms.parms[0] = first_parm->parms.parms[0];
// make the old extra pow be a multiply of the exponents
first_parm->fntype = FN_MULTIPLY;
first_parm->parms.parms[0] = op->parms.parms[1];
// put that as the exponent
op->parms.parms[1] = first_parm;
goto start_over;
}
}
break;
case FN_LOGICAL_AND:
case FN_LOGICAL_OR:
if (op->parms.parms[0]->fntype == FN_NOTNOT)
{
// remove notnot, unnecessary for input to &&/|| operators
op->parms.parms[0] = op->parms.parms[0]->parms.parms[0];
goto start_over;
}
if (op->parms.parms[1]->fntype == FN_NOTNOT)
{
// remove notnot, unnecessary for input to &&/|| operators
op->parms.parms[1] = op->parms.parms[1]->parms.parms[0];
goto start_over;
}
break;
}
}
else if (op->opcodeType==OPCODETYPE_FUNC3) // within FUNCTYPE_SIMPLE
{
if (op->fntype == FN_IF_ELSE)
{
if (op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE)
{
int s = fabs(op->parms.parms[0]->parms.dv.directValue) >= NSEEL_CLOSEFACTOR;
memcpy(op,op->parms.parms[s ? 1 : 2],sizeof(opcodeRec));
goto start_over;
}
if (op->parms.parms[0]->opcodeType == OPCODETYPE_FUNC1)
{
if (op->parms.parms[0]->fntype == FN_NOTNOT)
{
// remove notnot, unnecessary for input to ? operator
op->parms.parms[0] = op->parms.parms[0]->parms.parms[0];
goto start_over;
}
}
}
}
if (op->fntype >= FN_NONCONST_BEGIN && op->fntype < FUNCTYPE_SIMPLEMAX) retv|=1;
// FUNCTYPE_SIMPLE
}
else if (op->fntype == FUNCTYPE_FUNCTIONTYPEREC && op->fn)
{
/*
probably worth doing reduction on:
_divop (constant change to multiply)
_and
_or
abs
maybe:
min
max
also, optimize should (recursively or maybe iteratively?) search transitive functions (mul/div) for more constant reduction possibilities
*/
functionType *pfn = (functionType *)op->fn;
if (!(pfn->nParams&NSEEL_NPARAMS_FLAG_CONST)) retv|=1;
if (op->opcodeType==OPCODETYPE_FUNC1) // within FUNCTYPE_FUNCTIONTYPEREC
{
if (op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE)
{
int suc=1;
EEL_F v = op->parms.parms[0]->parms.dv.directValue;
#define DOF(x) if (!strcmp(pfn->name,#x)) v = x(v); else
#define DOF2(x,y) if (!strcmp(pfn->name,#x)) v = x(y); else
DOF(sin)
DOF(cos)
DOF(tan)
DOF(asin)
DOF(acos)
DOF(atan)
DOF2(sqrt, fabs(v))
DOF(exp)
DOF(log)
DOF(log10)
/* else */ suc=0;
#undef DOF
#undef DOF2
if (suc)
{
RESTART_DIRECTVALUE(v);
}
}
}
else if (op->opcodeType==OPCODETYPE_FUNC2) // within FUNCTYPE_FUNCTIONTYPEREC
{
const int dv0=op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE;
const int dv1=op->parms.parms[1]->opcodeType == OPCODETYPE_DIRECTVALUE;
if (dv0 && dv1)
{
if (!strcmp(pfn->name,"atan2"))
{
RESTART_DIRECTVALUE(atan2(op->parms.parms[0]->parms.dv.directValue, op->parms.parms[1]->parms.dv.directValue));
}
}
}
// FUNCTYPE_FUNCTIONTYPEREC
}
else
{
// unknown or eel func, assume non-const
retv |= 1;
}
}
// if we need results, or our function has effects itself, then finish
if (retv || needsResult)
{
return retv || joined_retv || retv_parm[0] || retv_parm[1] || retv_parm[2];
}
// we don't need results here, and our function is const, which means we can remove it
{
int cnt=0, idx1=0, idx2=0, x;
for (x=0;x<3;x++) if (retv_parm[x]) { if (!cnt++) idx1=x; else idx2=x; }
if (!cnt) // none of the parameters do anything, remove this opcode
{
if (lastJoinOp)
{
// replace previous join with its first linked opcode, removing this opcode completely
memcpy(lastJoinOp,lastJoinOp->parms.parms[0],sizeof(*lastJoinOp));
}
else if (op->opcodeType != OPCODETYPE_DIRECTVALUE)
{
// allow caller to easily detect this as trivial and remove it
op->opcodeType = OPCODETYPE_DIRECTVALUE;
op->parms.dv.valuePtr=NULL;
op->parms.dv.directValue=0.0;
}
// return joined_retv below
}
else
{
// if parameters are non-const, and we're a conditional, preserve our function
if (FNPTR_HAS_CONDITIONAL_EXEC(op)) return 1;
// otherwise, condense into either the non-const statement, or a join
if (cnt==1)
{
memcpy(op,op->parms.parms[idx1],sizeof(*op));
}
else if (cnt == 2)
{
op->opcodeType = OPCODETYPE_FUNC2;
op->fntype = FN_JOIN_STATEMENTS;
op->fn = op;
op->parms.parms[0] = op->parms.parms[idx1];
op->parms.parms[1] = op->parms.parms[idx2];
op->parms.parms[2] = NULL;
}
else
{
// todo need to create a new opcodeRec here, for now just leave as is
// (non-conditional const 3 parameter functions are rare anyway)
}
return 1;
}
}
return joined_retv;
}
static int generateValueToReg(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int whichReg, const namespaceInformation *functionPrefix, int allowCache)
{
EEL_F *b=NULL;
if (op->opcodeType==OPCODETYPE_VALUE_FROM_NAMESPACENAME)
{
char nm[NSEEL_MAX_VARIABLE_NAMELEN+1];
const char *p = op->relname;
combineNamespaceFields(nm,functionPrefix,p+(*p == '#'),op->namespaceidx);
if (!nm[0]) return -1;
if (*p == '#')
{
if (ctx->isGeneratingCommonFunction)
b = newCtxDataBlock(sizeof(EEL_F),sizeof(EEL_F));
else
b = newDataBlock(sizeof(EEL_F),sizeof(EEL_F));
if (!b) RET_MINUS1_FAIL("error creating storage for str")
if (!ctx->onNamedString) return -1; // should never happen, will not generate OPCODETYPE_VALUE_FROM_NAMESPACENAME with # prefix if !onNamedString
*b = ctx->onNamedString(ctx->caller_this,nm);
}
else
{
b = nseel_int_register_var(ctx,nm,0,NULL);
if (!b) RET_MINUS1_FAIL("error registering var")
}
}
else
{
if (op->opcodeType != OPCODETYPE_DIRECTVALUE) allowCache=0;
if (op->opcodeType==OPCODETYPE_DIRECTVALUE_TEMPSTRING && ctx->onNamedString)
{
op->parms.dv.directValue = ctx->onNamedString(ctx->caller_this,"");
op->parms.dv.valuePtr = NULL;
}
b=op->parms.dv.valuePtr;
if (!b && op->opcodeType == OPCODETYPE_VARPTR && op->relname && op->relname[0])
{
op->parms.dv.valuePtr = b = nseel_int_register_var(ctx,op->relname,0,NULL);
}
if (b && op->opcodeType == OPCODETYPE_VARPTRPTR) b = *(EEL_F **)b;
if (!b && allowCache)
{
int n=50; // only scan last X items
opcodeRec *r = ctx->directValueCache;
while (r && n--)
{
if (r->parms.dv.directValue == op->parms.dv.directValue && (b=r->parms.dv.valuePtr)) break;
r=(opcodeRec*)r->fn;
}
}
if (!b)
{
ctx->l_stats[3]++;
if (ctx->isGeneratingCommonFunction)
b = newCtxDataBlock(sizeof(EEL_F),sizeof(EEL_F));
else
b = newDataBlock(sizeof(EEL_F),sizeof(EEL_F));
if (!b) RET_MINUS1_FAIL("error allocating data block")
if (op->opcodeType != OPCODETYPE_VARPTRPTR) op->parms.dv.valuePtr = b;
#if EEL_F_SIZE == 8
*b = denormal_filter_double2(op->parms.dv.directValue);
#else
*b = denormal_filter_float2(op->parms.dv.directValue);
#endif
if (allowCache)
{
op->fn = ctx->directValueCache;
ctx->directValueCache = op;
}
}
}
GLUE_MOV_PX_DIRECTVALUE_GEN(bufOut,(INT_PTR)b,whichReg);
return GLUE_MOV_PX_DIRECTVALUE_SIZE;
}
unsigned char *compileCodeBlockWithRet(compileContext *ctx, opcodeRec *rec, int *computTableSize, const namespaceInformation *namespacePathToThis,
int supportedReturnValues, int *rvType, int *fpStackUsage, int *canHaveDenormalOutput)
{
unsigned char *p, *newblock2;
// generate code call
int funcsz=compileOpcodes(ctx,rec,NULL,1024*1024*128,NULL,namespacePathToThis,supportedReturnValues, rvType,fpStackUsage, NULL);
if (funcsz<0) return NULL;
p = newblock2 = newCodeBlock(funcsz+ sizeof(GLUE_RET)+GLUE_FUNC_ENTER_SIZE+GLUE_FUNC_LEAVE_SIZE,32);
if (!newblock2) return NULL;
#if GLUE_FUNC_ENTER_SIZE > 0
memcpy(p,&GLUE_FUNC_ENTER,GLUE_FUNC_ENTER_SIZE);
p += GLUE_FUNC_ENTER_SIZE;
#endif
*fpStackUsage=0;
funcsz=compileOpcodes(ctx,rec,p, funcsz, computTableSize,namespacePathToThis,supportedReturnValues, rvType,fpStackUsage, canHaveDenormalOutput);
if (funcsz<0) return NULL;
p+=funcsz;
#if GLUE_FUNC_LEAVE_SIZE > 0
memcpy(p,&GLUE_FUNC_LEAVE,GLUE_FUNC_LEAVE_SIZE);
p+=GLUE_FUNC_LEAVE_SIZE;
#endif
memcpy(p,&GLUE_RET,sizeof(GLUE_RET)); p+=sizeof(GLUE_RET);
#if defined(__arm__) || defined(__aarch64__)
__clear_cache(newblock2,p);
#endif
ctx->l_stats[2]+=funcsz+2;
return newblock2;
}
static int compileNativeFunctionCall(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int bufOut_len, int *computTableSize, const namespaceInformation *namespacePathToThis,
int *rvMode, int *fpStackUsage, int preferredReturnValues, int *canHaveDenormalOutput)
{
// builtin function generation
int func_size=0;
int cfunc_abiinfo=0;
int local_fpstack_use=0; // how many items we have pushed onto the fp stack
int parm_size=0;
int restore_stack_amt=0;
void *func_e=NULL;
NSEEL_PPPROC preProc=0;
void **repl=NULL;
int n_params= 1 + op->opcodeType - OPCODETYPE_FUNC1;
const int parm0_dv = op->parms.parms[0]->opcodeType == OPCODETYPE_DIRECTVALUE;
const int parm1_dv = n_params > 1 && op->parms.parms[1]->opcodeType == OPCODETYPE_DIRECTVALUE;
void *func = nseel_getBuiltinFunctionAddress(ctx, op->fntype, op->fn, &preProc,&repl,&func_e,&cfunc_abiinfo,preferredReturnValues,
parm0_dv ? &op->parms.parms[0]->parms.dv.directValue : NULL,
parm1_dv ? &op->parms.parms[1]->parms.dv.directValue : NULL
);
if (!func) RET_MINUS1_FAIL("error getting funcaddr")
*fpStackUsage=BIF_GETFPSTACKUSE(cfunc_abiinfo);
*rvMode = RETURNVALUE_NORMAL;
if (cfunc_abiinfo & BIF_TAKES_VARPARM)
{
#if defined(__arm__) || (defined (_M_ARM) && _M_ARM == 7)
const int max_params=16384; // arm uses up to two instructions, should be good for at leaast 64k (16384*4)
#elif defined(__ppc__)
const int max_params=4096; // 32kb max offset addressing for stack, so 4096*4 = 16384, should be safe
#elif defined(__aarch64__)
const int max_params=32768;
#else
const int max_params=32768; // sanity check, the stack is free to grow on x86/x86-64
#endif
int x;
// this mode is less efficient in that it creates a list of pointers on the stack to pass to the function
// but it is more flexible and works for >3 parameters.
if (op->opcodeType == OPCODETYPE_FUNCX)
{
n_params=0;
for (x=0;x<3;x++)
{
opcodeRec *prni=op->parms.parms[x];
while (prni)
{
const int isMP = prni->opcodeType == OPCODETYPE_MOREPARAMS;
n_params++;
if (!isMP||n_params>=max_params) break;
prni = prni->parms.parms[1];
}
}
}
restore_stack_amt = (sizeof(void *) * n_params + 15)&~15;
if (restore_stack_amt)
{
int offs = restore_stack_amt;
while (offs > 0)
{
int amt = offs;
if (amt > 4096) amt=4096;
if (bufOut_len < parm_size+GLUE_MOVE_STACK_SIZE) RET_MINUS1_FAIL("insufficient size for varparm")
if (bufOut) GLUE_MOVE_STACK(bufOut+parm_size, - amt);
parm_size += GLUE_MOVE_STACK_SIZE;
offs -= amt;
if (offs>0) // make sure this page is in memory
{
if (bufOut_len < parm_size+GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(0))
RET_MINUS1_FAIL("insufficient size for varparm stackchk")
if (bufOut) GLUE_STORE_P1_TO_STACK_AT_OFFS(bufOut+parm_size,0);
parm_size += GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(0);
}
}
}
if (op->opcodeType == OPCODETYPE_FUNCX)
{
n_params=0;
for (x=0;x<3;x++)
{
opcodeRec *prni=op->parms.parms[x];
while (prni)
{
const int isMP = prni->opcodeType == OPCODETYPE_MOREPARAMS;
opcodeRec *r = isMP ? prni->parms.parms[0] : prni;
if (r)
{
int canHaveDenorm=0;
int rvt=RETURNVALUE_NORMAL;
int subfpstackuse=0, use_offs;
int lsz = compileOpcodes(ctx,r,bufOut ? bufOut + parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, rvt,&rvt, &subfpstackuse, &canHaveDenorm);
if (canHaveDenorm && canHaveDenormalOutput) *canHaveDenormalOutput = 1;
if (lsz<0) RET_MINUS1_FAIL("call coc for varparmX failed")
if (rvt != RETURNVALUE_NORMAL) RET_MINUS1_FAIL("call coc for varparmX gave bad type back");
parm_size += lsz;
use_offs = n_params*(int) sizeof(void *);
if (bufOut_len < parm_size+GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(use_offs))
RET_MINUS1_FAIL("call coc for varparmX size");
if (bufOut) GLUE_STORE_P1_TO_STACK_AT_OFFS(bufOut + parm_size, use_offs);
parm_size+=GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(use_offs);
if (subfpstackuse+local_fpstack_use > *fpStackUsage) *fpStackUsage = subfpstackuse+local_fpstack_use;
}
else RET_MINUS1_FAIL("zero parameter varparmX")
n_params++;
if (!isMP||n_params>=max_params) break;
prni = prni->parms.parms[1];
}
}
}
else for (x=0;x<n_params;x++)
{
opcodeRec *r = op->parms.parms[x];
if (r)
{
int canHaveDenorm=0;
int subfpstackuse=0;
int rvt=RETURNVALUE_NORMAL;
int use_offs;
int lsz = compileOpcodes(ctx,r,bufOut ? bufOut + parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, rvt,&rvt, &subfpstackuse, &canHaveDenorm);
if (canHaveDenorm && canHaveDenormalOutput) *canHaveDenormalOutput = 1;
if (lsz<0) RET_MINUS1_FAIL("call coc for varparm123 failed")
if (rvt != RETURNVALUE_NORMAL) RET_MINUS1_FAIL("call coc for varparm123 gave bad type back");
parm_size += lsz;
use_offs = x*(int)sizeof(void *);
if (bufOut_len < parm_size+GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(use_offs))
RET_MINUS1_FAIL("call coc for varparm123 size");
if (bufOut) GLUE_STORE_P1_TO_STACK_AT_OFFS(bufOut + parm_size, use_offs);
parm_size+=GLUE_STORE_P1_TO_STACK_AT_OFFS_SIZE(use_offs);
if (subfpstackuse+local_fpstack_use > *fpStackUsage) *fpStackUsage = subfpstackuse+local_fpstack_use;
}
else RET_MINUS1_FAIL("zero parameter for varparm123");
}
if (bufOut_len < parm_size+GLUE_MOV_PX_DIRECTVALUE_SIZE+GLUE_MOVE_PX_STACKPTR_SIZE) RET_MINUS1_FAIL("insufficient size for varparm p1")
if (bufOut) GLUE_MOV_PX_DIRECTVALUE_GEN(bufOut+parm_size, (INT_PTR)n_params,1);
parm_size+=GLUE_MOV_PX_DIRECTVALUE_SIZE;
if (bufOut) GLUE_MOVE_PX_STACKPTR_GEN(bufOut+parm_size, 0);
parm_size+=GLUE_MOVE_PX_STACKPTR_SIZE;
}
else // not varparm
{
int pn;
#ifdef GLUE_HAS_FXCH
int need_fxch=0;
#endif
int last_nt_parm=-1, last_nt_parm_type=-1;
if (op->opcodeType == OPCODETYPE_FUNCX)
{
// this is not yet supported (calling conventions will need to be sorted, among other things)
RET_MINUS1_FAIL("funcx for native functions requires BIF_TAKES_VARPARM or BIF_TAKES_VARPARM_EX")
}
if (parm0_dv)
{
if (func == nseel_asm_stack_pop)
{
func = GLUE_realAddress(nseel_asm_stack_pop_fast,nseel_asm_stack_pop_fast_end,&func_size);
if (!func || bufOut_len < func_size) RET_MINUS1_FAIL(func?"failed on popfast size":"failed on popfast addr")
if (bufOut)
{
memcpy(bufOut,func,func_size);
NSEEL_PProc_Stack(bufOut,func_size,ctx);
}
return func_size;
}
else if (func == nseel_asm_stack_peek)
{
int f = (int) op->parms.parms[0]->parms.dv.directValue;
if (!f)
{
func = GLUE_realAddress(nseel_asm_stack_peek_top,nseel_asm_stack_peek_top_end,&func_size);
if (!func || bufOut_len < func_size) RET_MINUS1_FAIL(func?"failed on peek size":"failed on peek addr")
if (bufOut)
{
memcpy(bufOut,func,func_size);
NSEEL_PProc_Stack_PeekTop(bufOut,func_size,ctx);
}
return func_size;
}
else
{
func = GLUE_realAddress(nseel_asm_stack_peek_int,nseel_asm_stack_peek_int_end,&func_size);
if (!func || bufOut_len < func_size) RET_MINUS1_FAIL(func?"failed on peekint size":"failed on peekint addr")
if (bufOut)
{
memcpy(bufOut,func,func_size);
NSEEL_PProc_Stack_PeekInt(bufOut,func_size,ctx,f*sizeof(EEL_F));
}
return func_size;
}
}
}
// end of built-in function specific special casing
// first pass, calculate any non-trivial parameters
for (pn=0; pn < n_params; pn++)
{
if (!OPCODE_IS_TRIVIAL(op->parms.parms[pn]))
{
int canHaveDenorm=0;
int subfpstackuse=0;
int lsz=0;
int rvt=RETURNVALUE_NORMAL;
int may_need_fppush=-1;
if (last_nt_parm>=0)
{
if (last_nt_parm_type==RETURNVALUE_FPSTACK)
{
may_need_fppush= parm_size;
}
else
{
// push last result
if (bufOut_len < parm_size + (int)sizeof(GLUE_PUSH_P1)) RET_MINUS1_FAIL("failed on size, pushp1")
if (bufOut) memcpy(bufOut + parm_size, &GLUE_PUSH_P1, sizeof(GLUE_PUSH_P1));
parm_size += sizeof(GLUE_PUSH_P1);
}
}
if (func == nseel_asm_bnot) rvt=RETURNVALUE_BOOL_REVERSED|RETURNVALUE_BOOL;
else if (pn == n_params - 1)
{
if (cfunc_abiinfo&BIF_LASTPARMONSTACK) rvt=RETURNVALUE_FPSTACK;
else if (cfunc_abiinfo&BIF_LASTPARM_ASBOOL) rvt=RETURNVALUE_BOOL;
else if (func == nseel_asm_assign) rvt=RETURNVALUE_FPSTACK|RETURNVALUE_NORMAL;
}
else if (pn == n_params -2 && (cfunc_abiinfo&BIF_SECONDLASTPARMST))
{
rvt=RETURNVALUE_FPSTACK;
}
lsz = compileOpcodes(ctx,op->parms.parms[pn],bufOut ? bufOut + parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, rvt,&rvt, &subfpstackuse, &canHaveDenorm);
if (lsz<0) RET_MINUS1_FAIL("call coc failed")
if (func == nseel_asm_bnot && rvt==RETURNVALUE_BOOL_REVERSED)
{
// remove bnot, compileOpcodes() used fptobool_rev
#ifndef EEL_TARGET_PORTABLE
func = nseel_asm_uplus;
func_e = nseel_asm_uplus_end;
#else
func = nseel_asm_bnotnot;
func_e = nseel_asm_bnotnot_end;
#endif
rvt = RETURNVALUE_BOOL;
}
if (canHaveDenorm && canHaveDenormalOutput) *canHaveDenormalOutput = 1;
parm_size += lsz;
if (may_need_fppush>=0)
{
if (local_fpstack_use+subfpstackuse >= (GLUE_MAX_FPSTACK_SIZE-1) || (ctx->optimizeDisableFlags&OPTFLAG_NO_FPSTACK))
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_POP_FPSTACK_TOSTACK))
RET_MINUS1_FAIL("failed on size, popfpstacktostack")
if (bufOut)
{
memmove(bufOut + may_need_fppush + sizeof(GLUE_POP_FPSTACK_TOSTACK), bufOut + may_need_fppush, parm_size - may_need_fppush);
memcpy(bufOut + may_need_fppush, &GLUE_POP_FPSTACK_TOSTACK, sizeof(GLUE_POP_FPSTACK_TOSTACK));
}
parm_size += sizeof(GLUE_POP_FPSTACK_TOSTACK);
}
else
{
local_fpstack_use++;
}
}
if (subfpstackuse+local_fpstack_use > *fpStackUsage) *fpStackUsage = subfpstackuse+local_fpstack_use;
last_nt_parm = pn;
last_nt_parm_type = rvt;
if (pn == n_params - 1 && func == nseel_asm_assign)
{
if (!(ctx->optimizeDisableFlags & OPTFLAG_FULL_DENORMAL_CHECKS) &&
(!canHaveDenorm || (ctx->optimizeDisableFlags & OPTFLAG_NO_DENORMAL_CHECKS)))
{
if (rvt == RETURNVALUE_FPSTACK)
{
cfunc_abiinfo |= BIF_LASTPARMONSTACK;
func = nseel_asm_assign_fast_fromfp;
func_e = nseel_asm_assign_fast_fromfp_end;
}
else
{
func = nseel_asm_assign_fast;
func_e = nseel_asm_assign_fast_end;
}
}
else
{
if (rvt == RETURNVALUE_FPSTACK)
{
cfunc_abiinfo |= BIF_LASTPARMONSTACK;
func = nseel_asm_assign_fromfp;
func_e = nseel_asm_assign_fromfp_end;
}
}
}
}
}
pn = last_nt_parm;
if (pn >= 0) // if the last thing executed doesn't go to the last parameter, move it there
{
if ((cfunc_abiinfo&BIF_SECONDLASTPARMST) && pn == n_params-2)
{
// do nothing, things are in the right place
}
else if (pn != n_params-1)
{
// generate mov p1->pX
if (bufOut_len < parm_size + GLUE_SET_PX_FROM_P1_SIZE) RET_MINUS1_FAIL("size, pxfromp1")
if (bufOut) GLUE_SET_PX_FROM_P1(bufOut + parm_size,n_params - 1 - pn);
parm_size += GLUE_SET_PX_FROM_P1_SIZE;
}
}
// pop any pushed parameters
while (--pn >= 0)
{
if (!OPCODE_IS_TRIVIAL(op->parms.parms[pn]))
{
if ((cfunc_abiinfo&BIF_SECONDLASTPARMST) && pn == n_params-2)
{
if (!local_fpstack_use)
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_POP_STACK_TO_FPSTACK)) RET_MINUS1_FAIL("size, popstacktofpstack 2")
if (bufOut) memcpy(bufOut+parm_size,GLUE_POP_STACK_TO_FPSTACK,sizeof(GLUE_POP_STACK_TO_FPSTACK));
parm_size += sizeof(GLUE_POP_STACK_TO_FPSTACK);
#ifdef GLUE_HAS_FXCH
need_fxch = 1;
#endif
}
else
{
local_fpstack_use--;
}
}
else
{
if (bufOut_len < parm_size + GLUE_POP_PX_SIZE) RET_MINUS1_FAIL("size, poppx")
if (bufOut) GLUE_POP_PX(bufOut + parm_size,n_params - 1 - pn);
parm_size += GLUE_POP_PX_SIZE;
}
}
}
// finally, set trivial pointers
for (pn=0; pn < n_params; pn++)
{
if (OPCODE_IS_TRIVIAL(op->parms.parms[pn]))
{
if (pn == n_params-2 && (cfunc_abiinfo&(BIF_SECONDLASTPARMST))) // second to last parameter
{
int a = compileOpcodes(ctx,op->parms.parms[pn],bufOut ? bufOut+parm_size : NULL,bufOut_len - parm_size,computTableSize,namespacePathToThis,
RETURNVALUE_FPSTACK,NULL,NULL,canHaveDenormalOutput);
if (a<0) RET_MINUS1_FAIL("coc call here 2")
parm_size+=a;
#ifdef GLUE_HAS_FXCH
need_fxch = 1;
#endif
}
else if (pn == n_params-1) // last parameter, but we should call compileOpcodes to get it in the right format (compileOpcodes can optimize that process if it needs to)
{
int rvt=0, a;
int wantFpStack = func == nseel_asm_assign;
#ifdef GLUE_PREFER_NONFP_DV_ASSIGNS // x86-64, and maybe others, prefer to avoid the fp stack for a simple copy
if (wantFpStack &&
(op->parms.parms[pn]->opcodeType != OPCODETYPE_DIRECTVALUE ||
(op->parms.parms[pn]->parms.dv.directValue != 1.0 && op->parms.parms[pn]->parms.dv.directValue != 0.0)))
{
wantFpStack=-1; // cacheable but non-FP stack
}
#endif
a = compileOpcodes(ctx,op->parms.parms[pn],bufOut ? bufOut+parm_size : NULL,bufOut_len - parm_size,computTableSize,namespacePathToThis,
func == nseel_asm_bnot ? (RETURNVALUE_BOOL_REVERSED|RETURNVALUE_BOOL) :
(cfunc_abiinfo & BIF_LASTPARMONSTACK) ? RETURNVALUE_FPSTACK :
(cfunc_abiinfo & BIF_LASTPARM_ASBOOL) ? RETURNVALUE_BOOL :
wantFpStack < 0 ? (RETURNVALUE_CACHEABLE|RETURNVALUE_NORMAL) :
wantFpStack ? (RETURNVALUE_FPSTACK|RETURNVALUE_NORMAL) :
RETURNVALUE_NORMAL,
&rvt, NULL,canHaveDenormalOutput);
if (a<0) RET_MINUS1_FAIL("coc call here 3")
if (func == nseel_asm_bnot && rvt == RETURNVALUE_BOOL_REVERSED)
{
// remove bnot, compileOpcodes() used fptobool_rev
#ifndef EEL_TARGET_PORTABLE
func = nseel_asm_uplus;
func_e = nseel_asm_uplus_end;
#else
func = nseel_asm_bnotnot;
func_e = nseel_asm_bnotnot_end;
#endif
rvt = RETURNVALUE_BOOL;
}
parm_size+=a;
#ifdef GLUE_HAS_FXCH
need_fxch = 0;
#endif
if (func == nseel_asm_assign)
{
if (rvt == RETURNVALUE_FPSTACK)
{
if (!(ctx->optimizeDisableFlags & OPTFLAG_FULL_DENORMAL_CHECKS))
{
func = nseel_asm_assign_fast_fromfp;
func_e = nseel_asm_assign_fast_fromfp_end;
}
else
{
func = nseel_asm_assign_fromfp;
func_e = nseel_asm_assign_fromfp_end;
}
}
else if (!(ctx->optimizeDisableFlags & OPTFLAG_FULL_DENORMAL_CHECKS))
{
// assigning a value (from a variable or other non-computer), can use a fast assign (no denormal/result checking)
func = nseel_asm_assign_fast;
func_e = nseel_asm_assign_fast_end;
}
}
}
else
{
if (bufOut_len < parm_size + GLUE_MOV_PX_DIRECTVALUE_SIZE) RET_MINUS1_FAIL("size, pxdvsz")
if (bufOut)
{
if (generateValueToReg(ctx,op->parms.parms[pn],bufOut + parm_size,n_params - 1 - pn,namespacePathToThis, 0/*nocaching, function gets pointer*/)<0) RET_MINUS1_FAIL("gvtr")
}
parm_size += GLUE_MOV_PX_DIRECTVALUE_SIZE;
}
}
}
#ifdef GLUE_HAS_FXCH
if ((cfunc_abiinfo&(BIF_SECONDLASTPARMST)) && !(cfunc_abiinfo&(BIF_LAZYPARMORDERING))&&
((!!need_fxch)^!!(cfunc_abiinfo&BIF_REVERSEFPORDER))
)
{
// emit fxch
if (bufOut_len < sizeof(GLUE_FXCH)) RET_MINUS1_FAIL("len,fxch")
if (bufOut)
{
memcpy(bufOut+parm_size,GLUE_FXCH,sizeof(GLUE_FXCH));
}
parm_size+=sizeof(GLUE_FXCH);
}
#endif
if (!*canHaveDenormalOutput)
{
// if add_op or sub_op, and constant non-denormal input, safe to omit denormal checks
if (func == (void*)nseel_asm_add_op && parm1_dv && fabs(op->parms.parms[1]->parms.dv.directValue) >= DENORMAL_CLEARING_THRESHOLD)
{
func = nseel_asm_add_op_fast;
func_e = nseel_asm_add_op_fast_end;
}
else if (func == (void*)nseel_asm_sub_op && parm1_dv && fabs(op->parms.parms[1]->parms.dv.directValue) >= DENORMAL_CLEARING_THRESHOLD)
{
func = nseel_asm_sub_op_fast;
func_e = nseel_asm_sub_op_fast_end;
}
// or if mul/div by a fixed value of >= or <= 1.0
else if (func == (void *)nseel_asm_mul_op && parm1_dv && fabs(op->parms.parms[1]->parms.dv.directValue) >= 1.0)
{
func = nseel_asm_mul_op_fast;
func_e = nseel_asm_mul_op_fast_end;
}
else if (func == (void *)nseel_asm_div_op && parm1_dv && fabs(op->parms.parms[1]->parms.dv.directValue) <= 1.0)
{
func = nseel_asm_div_op_fast;
func_e = nseel_asm_div_op_fast_end;
}
}
} // not varparm
if (cfunc_abiinfo & (BIF_CLEARDENORMAL | BIF_RETURNSBOOL) ) *canHaveDenormalOutput=0;
else if (!(cfunc_abiinfo & BIF_WONTMAKEDENORMAL)) *canHaveDenormalOutput=1;
func = GLUE_realAddress(func,func_e,&func_size);
if (!func) RET_MINUS1_FAIL("failrealladdrfunc")
if (bufOut_len < parm_size + func_size) RET_MINUS1_FAIL("funcsz")
if (bufOut)
{
unsigned char *p=bufOut + parm_size;
memcpy(p, func, func_size);
if (preProc) p=preProc(p,func_size,ctx);
if (repl)
{
if (repl[0]) p=EEL_GLUE_set_immediate(p,(INT_PTR)repl[0]);
if (repl[1]) p=EEL_GLUE_set_immediate(p,(INT_PTR)repl[1]);
if (repl[2]) p=EEL_GLUE_set_immediate(p,(INT_PTR)repl[2]);
if (repl[3]) p=EEL_GLUE_set_immediate(p,(INT_PTR)repl[3]);
}
}
if (restore_stack_amt)
{
int rem = restore_stack_amt;
while (rem > 0)
{
int amt = rem;
if (amt > 4096) amt=4096;
rem -= amt;
if (bufOut_len < parm_size + func_size + GLUE_MOVE_STACK_SIZE) RET_MINUS1_FAIL("insufficient size for varparm")
if (bufOut) GLUE_MOVE_STACK(bufOut + parm_size + func_size, amt);
parm_size += GLUE_MOVE_STACK_SIZE;
}
}
if (cfunc_abiinfo&BIF_RETURNSONSTACK) *rvMode = RETURNVALUE_FPSTACK;
else if (cfunc_abiinfo&BIF_RETURNSBOOL) *rvMode=RETURNVALUE_BOOL;
return parm_size + func_size;
}
static int compileEelFunctionCall(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int bufOut_len, int *computTableSize, const namespaceInformation *namespacePathToThis,
int *rvMode, int *fpStackUse, int *canHaveDenormalOutput)
{
int func_size=0, parm_size=0;
int pn;
int last_nt_parm=-1,last_nt_parm_mode=0;
void *func_e=NULL;
int n_params;
opcodeRec *parmptrs[NSEEL_MAX_EELFUNC_PARAMETERS];
int cfp_numparams=-1;
int cfp_statesize=0;
EEL_F **cfp_ptrs=NULL;
int func_raw=0;
int do_parms;
int x;
void *func;
for (x=0; x < 3; x ++) parmptrs[x] = op->parms.parms[x];
if (op->opcodeType == OPCODETYPE_FUNCX)
{
n_params=0;
for (x=0;x<3;x++)
{
opcodeRec *prni=op->parms.parms[x];
while (prni && n_params < NSEEL_MAX_EELFUNC_PARAMETERS)
{
const int isMP = prni->opcodeType == OPCODETYPE_MOREPARAMS;
parmptrs[n_params++] = isMP ? prni->parms.parms[0] : prni;
if (!isMP) break;
prni = prni->parms.parms[1];
}
}
}
else
{
n_params = 1 + op->opcodeType - OPCODETYPE_FUNC1;
}
*fpStackUse = 0;
func = nseel_getEELFunctionAddress(ctx, op,
&cfp_numparams,&cfp_statesize,&cfp_ptrs,
computTableSize,
&func_e, &func_raw,
!!bufOut,namespacePathToThis,rvMode,fpStackUse,canHaveDenormalOutput, parmptrs, n_params);
if (func_raw) func_size = (int) ((char*)func_e - (char*)func);
else if (func) func = GLUE_realAddress(func,func_e,&func_size);
if (!func) RET_MINUS1_FAIL("eelfuncaddr")
*fpStackUse += 1;
if (cfp_numparams>0 && n_params != cfp_numparams)
{
RET_MINUS1_FAIL("eelfuncnp")
}
// user defined function
do_parms = cfp_numparams>0 && cfp_ptrs && cfp_statesize>0;
// if function local/parameter state is zero, we need to allocate storage for it
if (cfp_statesize>0 && cfp_ptrs && !cfp_ptrs[0])
{
EEL_F *pstate = newDataBlock(sizeof(EEL_F)*cfp_statesize,8);
if (!pstate) RET_MINUS1_FAIL("eelfuncdb")
for (pn=0;pn<cfp_statesize;pn++)
{
pstate[pn]=0;
cfp_ptrs[pn] = pstate + pn;
}
}
// first process parameters that are non-trivial
for (pn=0; pn < n_params; pn++)
{
int needDenorm=0;
int lsz,sUse=0;
if (!parmptrs[pn] || OPCODE_IS_TRIVIAL(parmptrs[pn])) continue; // skip and process after
if (last_nt_parm >= 0 && do_parms)
{
if (last_nt_parm_mode == RETURNVALUE_FPSTACK)
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_POP_FPSTACK_TOSTACK)) RET_MINUS1_FAIL("eelfunc_size popfpstacktostack")
if (bufOut) memcpy(bufOut + parm_size,GLUE_POP_FPSTACK_TOSTACK,sizeof(GLUE_POP_FPSTACK_TOSTACK));
parm_size+=sizeof(GLUE_POP_FPSTACK_TOSTACK);
}
else
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_PUSH_P1PTR_AS_VALUE)) RET_MINUS1_FAIL("eelfunc_size pushp1ptrasval")
// push
if (bufOut) memcpy(bufOut + parm_size,&GLUE_PUSH_P1PTR_AS_VALUE,sizeof(GLUE_PUSH_P1PTR_AS_VALUE));
parm_size+=sizeof(GLUE_PUSH_P1PTR_AS_VALUE);
}
}
last_nt_parm_mode=0;
lsz = compileOpcodes(ctx,parmptrs[pn],bufOut ? bufOut + parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis,
do_parms ? (RETURNVALUE_FPSTACK|RETURNVALUE_NORMAL) : RETURNVALUE_IGNORE,&last_nt_parm_mode,&sUse, &needDenorm);
// todo: if needDenorm, denorm convert when copying parameter
if (lsz<0) RET_MINUS1_FAIL("eelfunc, coc fail")
if (last_nt_parm_mode == RETURNVALUE_FPSTACK) sUse++;
if (sUse > *fpStackUse) *fpStackUse=sUse;
parm_size += lsz;
last_nt_parm = pn;
}
// pop non-trivial results into place
if (last_nt_parm >=0 && do_parms)
{
while (--pn >= 0)
{
if (!parmptrs[pn] || OPCODE_IS_TRIVIAL(parmptrs[pn])) continue; // skip and process after
if (pn == last_nt_parm)
{
if (last_nt_parm_mode == RETURNVALUE_FPSTACK)
{
// pop to memory directly
const int cpsize = GLUE_POP_FPSTACK_TO_PTR(NULL,NULL);
if (bufOut_len < parm_size + cpsize) RET_MINUS1_FAIL("eelfunc size popfpstacktoptr")
if (bufOut) GLUE_POP_FPSTACK_TO_PTR((unsigned char *)bufOut + parm_size,cfp_ptrs[pn]);
parm_size += cpsize;
}
else
{
// copy direct p1ptr to mem
const int cpsize = GLUE_COPY_VALUE_AT_P1_TO_PTR(NULL,NULL);
if (bufOut_len < parm_size + cpsize) RET_MINUS1_FAIL("eelfunc size copyvalueatp1toptr")
if (bufOut) GLUE_COPY_VALUE_AT_P1_TO_PTR((unsigned char *)bufOut + parm_size,cfp_ptrs[pn]);
parm_size += cpsize;
}
}
else
{
const int popsize = GLUE_POP_VALUE_TO_ADDR(NULL,NULL);
if (bufOut_len < parm_size + popsize) RET_MINUS1_FAIL("eelfunc size pop value to addr")
if (bufOut) GLUE_POP_VALUE_TO_ADDR((unsigned char *)bufOut + parm_size,cfp_ptrs[pn]);
parm_size+=popsize;
}
}
}
// finally, set any trivial parameters
if (do_parms)
{
const int cpsize = GLUE_MOV_PX_DIRECTVALUE_SIZE + GLUE_COPY_VALUE_AT_P1_TO_PTR(NULL,NULL);
for (pn=0; pn < n_params; pn++)
{
if (!parmptrs[pn] || !OPCODE_IS_TRIVIAL(parmptrs[pn])) continue; // set trivial values, we already set nontrivials
if (bufOut_len < parm_size + cpsize) RET_MINUS1_FAIL("eelfunc size trivial set")
if (bufOut)
{
if (generateValueToReg(ctx,parmptrs[pn],bufOut + parm_size,0,namespacePathToThis, 1)<0) RET_MINUS1_FAIL("eelfunc gvr fail")
GLUE_COPY_VALUE_AT_P1_TO_PTR(bufOut + parm_size + GLUE_MOV_PX_DIRECTVALUE_SIZE,cfp_ptrs[pn]);
}
parm_size += cpsize;
}
}
if (bufOut_len < parm_size + func_size) RET_MINUS1_FAIL("eelfunc size combined")
if (bufOut) memcpy(bufOut + parm_size, func, func_size);
return parm_size + func_size;
// end of EEL function generation
}
#ifdef DUMP_OPS_DURING_COMPILE
void dumpOp(compileContext *ctx, opcodeRec *op, int start);
#endif
#ifdef EEL_DUMP_OPS
void dumpOpcodeTree(compileContext *ctx, FILE *fp, opcodeRec *op, int indent_amt)
{
const char *fname="";
fprintf(fp,"%*sOP TYPE %d", indent_amt, "",
op->opcodeType==OPCODETYPE_DIRECTVALUE_TEMPSTRING ? 10000 : // remap around OPCODETYPE_DIRECTVALUE_TEMPSTRING
op->opcodeType > OPCODETYPE_DIRECTVALUE_TEMPSTRING ? op->opcodeType - 1 :
op->opcodeType);
if ((op->opcodeType == OPCODETYPE_FUNC1 ||
op->opcodeType == OPCODETYPE_FUNC2 ||
op->opcodeType == OPCODETYPE_FUNC3 ||
op->opcodeType == OPCODETYPE_FUNCX))
{
if (op->fntype == FUNCTYPE_FUNCTIONTYPEREC)
{
functionType *fn_ptr = (functionType *)op->fn;
fname = fn_ptr->name;
}
else if (op->fntype == FUNCTYPE_EELFUNC)
{
fname = op->relname;
}
if (!fname) fname ="";
}
switch (op->opcodeType)
{
case OPCODETYPE_DIRECTVALUE:
fprintf(fp," DV=%f\r\n",op->parms.dv.directValue);
break;
case OPCODETYPE_VALUE_FROM_NAMESPACENAME: // this.* or namespace.* are encoded this way
fprintf(fp," NSN=%s(%d)\r\n",op->relname?op->relname : "(null)",op->namespaceidx);
break;
case OPCODETYPE_VARPTR:
{
const char *nm = op->relname;
if (!nm || !*nm)
{
int wb;
for (wb = 0; wb < ctx->varTable_numBlocks; wb ++)
{
char **plist=ctx->varTable_Names[wb];
if (!plist) break;
if (op->parms.dv.valuePtr >= ctx->varTable_Values[wb] && op->parms.dv.valuePtr < ctx->varTable_Values[wb] + NSEEL_VARS_PER_BLOCK)
{
nm = plist[op->parms.dv.valuePtr - ctx->varTable_Values[wb]];
break;
}
}
}
fprintf(fp," VP=%s\r\n", nm?nm : "(null)");
}
break;
case OPCODETYPE_VARPTRPTR:
fprintf(fp, " VPP?\r\n");
break;
case OPCODETYPE_FUNC1:
if (op->fntype == FN_NOT)
fprintf(fp," FUNC1 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_not");
else if (op->fntype == FN_NOTNOT)
fprintf(fp," FUNC1 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_notnot");
else if (op->fntype == FN_MEMORY)
fprintf(fp," FUNC1 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_mem");
else if (op->fntype == FN_GMEMORY)
fprintf(fp," FUNC1 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_gmem");
else if (op->fntype == FN_WHILE)
fprintf(fp," FUNC1 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "while");
else
fprintf(fp," FUNC1 %d %s {\r\n",op->fntype, fname);
if (op->parms.parms[0])
dumpOpcodeTree(ctx,fp,op->parms.parms[0],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
fprintf(fp,"%*s}\r\n", indent_amt, "");
break;
case OPCODETYPE_MOREPARAMS:
case OPCODETYPE_FUNC2:
if (op->opcodeType == OPCODETYPE_MOREPARAMS)
fprintf(fp," MOREPARAMS {\r\n");
else
{
if (op->fntype == FN_POW)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "pow");
else if (op->fntype == FN_MOD)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_mod");
else if (op->fntype == FN_XOR)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_xor");
else if (op->fntype == FN_SHL)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_shl");
else if (op->fntype == FN_SHR)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_shr");
else if (op->fntype == FN_LT)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_below");
else if (op->fntype == FN_GT)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_above");
else if (op->fntype == FN_LTE)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_beleq");
else if (op->fntype == FN_GTE)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_aboeq");
else if (op->fntype == FN_EQ)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_equal");
else if (op->fntype == FN_NE)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_noteq");
else if (op->fntype == FN_EQ_EXACT)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_equal_exact");
else if (op->fntype == FN_NE_EXACT)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_noteq_exact");
else if (op->fntype == FN_LOGICAL_AND)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_and");
else if (op->fntype == FN_LOGICAL_OR)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_or");
else if (op->fntype == FN_ASSIGN)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_set");
else if (op->fntype == FN_ADD_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_addop");
else if (op->fntype == FN_SUB_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_subop");
else if (op->fntype == FN_MUL_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_mulop");
else if (op->fntype == FN_DIV_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_divop");
else if (op->fntype == FN_OR_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_orop");
else if (op->fntype == FN_AND_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_andop");
else if (op->fntype == FN_XOR_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_xorop");
else if (op->fntype == FN_MOD_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_modop");
else if (op->fntype == FN_POW_OP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_powop");
else if (op->fntype == FN_LOOP)
fprintf(fp," FUNC2 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "loop");
else
fprintf(fp," FUNC2 %d %s {\r\n",op->fntype, fname);
}
if (op->parms.parms[0])
dumpOpcodeTree(ctx,fp,op->parms.parms[0],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
if (op->parms.parms[1])
dumpOpcodeTree(ctx,fp,op->parms.parms[1],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
fprintf(fp,"%*s}\r\n", indent_amt, "");
break;
case OPCODETYPE_FUNCX:
case OPCODETYPE_FUNC3:
if (op->opcodeType == OPCODETYPE_FUNCX)
fprintf(fp," FUNCX %d %s {\r\n",op->fntype, fname);
else if (op->fntype == FN_IF_ELSE)
fprintf(fp," FUNC3 %d %s {\r\n",FUNCTYPE_FUNCTIONTYPEREC, "_if");
else
fprintf(fp," FUNC3 %d %s {\r\n",op->fntype, fname);
if (op->parms.parms[0])
dumpOpcodeTree(ctx,fp,op->parms.parms[0],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
if (op->parms.parms[1])
dumpOpcodeTree(ctx,fp,op->parms.parms[1],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
if (op->parms.parms[2])
dumpOpcodeTree(ctx,fp,op->parms.parms[2],indent_amt+2);
else
fprintf(fp,"%*sINVALID PARM\r\n",indent_amt+2,"");
fprintf(fp,"%*s}\r\n", indent_amt, "");
break;
}
}
#endif
#ifdef GLUE_MAX_JMPSIZE
#define CHECK_SIZE_FORJMP(x,y) if ((x)<0 || (x)>=GLUE_MAX_JMPSIZE) goto y;
#define RET_MINUS1_FAIL_FALLBACK(err,j) goto j;
#else
#define CHECK_SIZE_FORJMP(x,y)
#define RET_MINUS1_FAIL_FALLBACK(err,j) RET_MINUS1_FAIL(err)
#endif
static int compileOpcodesInternal(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int bufOut_len, int *computTableSize, const namespaceInformation *namespacePathToThis, int *calledRvType, int preferredReturnValues, int *fpStackUse, int *canHaveDenormalOutput)
{
int rv_offset=0, denormal_force=-1;
if (!op) RET_MINUS1_FAIL("coi !op")
*fpStackUse=0;
for (;;)
{
// special case: statement delimiting means we can process the left side into place, and iteratively do the second parameter without recursing
// also we don't need to save/restore anything to the stack (which the normal 2 parameter function processing does)
if (op->opcodeType == OPCODETYPE_FUNC2 && op->fntype == FN_JOIN_STATEMENTS)
{
int fUse1;
int parm_size = compileOpcodes(ctx,op->parms.parms[0],bufOut,bufOut_len, computTableSize, namespacePathToThis, RETURNVALUE_IGNORE, NULL,&fUse1,NULL);
if (parm_size < 0) RET_MINUS1_FAIL("coc join fail")
op = op->parms.parms[1];
if (!op) RET_MINUS1_FAIL("join got to null")
if (fUse1>*fpStackUse) *fpStackUse=fUse1;
if (bufOut) bufOut += parm_size;
bufOut_len -= parm_size;
rv_offset += parm_size;
#ifdef DUMP_OPS_DURING_COMPILE
if (op->opcodeType != OPCODETYPE_FUNC2 || op->fntype != FN_JOIN_STATEMENTS) dumpOp(ctx,op,0);
#endif
denormal_force=-1;
}
// special case: __denormal_likely(), __denormal_unlikely()
else if (op->opcodeType == OPCODETYPE_FUNC1 && (op->fntype == FN_DENORMAL_LIKELY || op->fntype == FN_DENORMAL_UNLIKELY))
{
denormal_force = op->fntype == FN_DENORMAL_LIKELY;
op = op->parms.parms[0];
}
else
{
break;
}
}
if (denormal_force >= 0 && canHaveDenormalOutput)
{
*canHaveDenormalOutput = denormal_force;
canHaveDenormalOutput = &denormal_force; // prevent it from being changed by functions below
}
// special case: BAND/BOR
if (op->opcodeType == OPCODETYPE_FUNC2 && (op->fntype == FN_LOGICAL_AND || op->fntype == FN_LOGICAL_OR))
{
int fUse1=0;
int parm_size;
#ifdef GLUE_MAX_JMPSIZE
int parm_size_pre;
#endif
int retType=RETURNVALUE_IGNORE;
if (preferredReturnValues != RETURNVALUE_IGNORE) retType = RETURNVALUE_BOOL;
*calledRvType = retType;
parm_size = compileOpcodes(ctx,op->parms.parms[0],bufOut,bufOut_len, computTableSize, namespacePathToThis, RETURNVALUE_BOOL, NULL, &fUse1, NULL);
if (parm_size < 0) RET_MINUS1_FAIL("loop band/bor coc fail")
if (fUse1 > *fpStackUse) *fpStackUse=fUse1;
#ifdef GLUE_MAX_JMPSIZE
parm_size_pre=parm_size;
#endif
{
int sz2, fUse2=0;
unsigned char *destbuf;
const int testsz=op->fntype == FN_LOGICAL_OR ? sizeof(GLUE_JMP_IF_P1_NZ) : sizeof(GLUE_JMP_IF_P1_Z);
if (bufOut_len < parm_size+testsz) RET_MINUS1_FAIL_FALLBACK("band/bor size fail",doNonInlinedAndOr_)
if (bufOut) memcpy(bufOut+parm_size,op->fntype == FN_LOGICAL_OR ? GLUE_JMP_IF_P1_NZ : GLUE_JMP_IF_P1_Z,testsz);
parm_size += testsz;
destbuf = bufOut + parm_size;
sz2= compileOpcodes(ctx,op->parms.parms[1],bufOut?bufOut+parm_size:NULL,bufOut_len-parm_size, computTableSize, namespacePathToThis, retType, NULL,&fUse2, NULL);
CHECK_SIZE_FORJMP(sz2,doNonInlinedAndOr_)
if (sz2<0) RET_MINUS1_FAIL("band/bor coc fail")
parm_size+=sz2;
if (bufOut) GLUE_JMP_SET_OFFSET(destbuf, (bufOut + parm_size) - destbuf);
if (fUse2 > *fpStackUse) *fpStackUse=fUse2;
return rv_offset + parm_size;
}
#ifdef GLUE_MAX_JMPSIZE
if (0)
{
void *stub;
int stubsize;
unsigned char *newblock2, *p;
// encode as function call
doNonInlinedAndOr_:
parm_size = parm_size_pre;
if (op->fntype == FN_LOGICAL_AND)
{
stub = GLUE_realAddress(nseel_asm_band,nseel_asm_band_end,&stubsize);
}
else
{
stub = GLUE_realAddress(nseel_asm_bor,nseel_asm_bor_end,&stubsize);
}
if (bufOut_len < parm_size + stubsize) RET_MINUS1_FAIL("band/bor len fail")
if (bufOut)
{
int fUse2=0;
newblock2 = compileCodeBlockWithRet(ctx,op->parms.parms[1],computTableSize,namespacePathToThis, retType, NULL, &fUse2, NULL);
if (!newblock2) RET_MINUS1_FAIL("band/bor ccbwr fail")
if (fUse2 > *fpStackUse) *fpStackUse=fUse2;
p = bufOut + parm_size;
memcpy(p, stub, stubsize);
p=EEL_GLUE_set_immediate(p,(INT_PTR)newblock2);
}
return rv_offset + parm_size + stubsize;
}
#endif
}
if (op->opcodeType == OPCODETYPE_FUNC3 && op->fntype == FN_IF_ELSE) // special case: IF
{
int fUse1=0;
#ifdef GLUE_MAX_JMPSIZE
int parm_size_pre;
#endif
int use_rv = RETURNVALUE_IGNORE;
int rvMode=0;
int parm_size = compileOpcodes(ctx,op->parms.parms[0],bufOut,bufOut_len, computTableSize, namespacePathToThis, RETURNVALUE_BOOL|RETURNVALUE_BOOL_REVERSED, &rvMode,&fUse1, NULL);
if (parm_size < 0) RET_MINUS1_FAIL("if coc fail")
if (fUse1 > *fpStackUse) *fpStackUse=fUse1;
if (preferredReturnValues & RETURNVALUE_NORMAL) use_rv=RETURNVALUE_NORMAL;
else if (preferredReturnValues & RETURNVALUE_FPSTACK) use_rv=RETURNVALUE_FPSTACK;
else if (preferredReturnValues & RETURNVALUE_BOOL) use_rv=RETURNVALUE_BOOL;
*calledRvType = use_rv;
#ifdef GLUE_MAX_JMPSIZE
parm_size_pre = parm_size;
#endif
{
int csz,hasSecondHalf;
if (rvMode & RETURNVALUE_BOOL_REVERSED)
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_JMP_IF_P1_NZ)) RET_MINUS1_FAIL_FALLBACK("if size fail",doNonInlineIf_)
if (bufOut) memcpy(bufOut+parm_size,GLUE_JMP_IF_P1_NZ,sizeof(GLUE_JMP_IF_P1_NZ));
parm_size += sizeof(GLUE_JMP_IF_P1_NZ);
}
else
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_JMP_IF_P1_Z)) RET_MINUS1_FAIL_FALLBACK("if size fail",doNonInlineIf_)
if (bufOut) memcpy(bufOut+parm_size,GLUE_JMP_IF_P1_Z,sizeof(GLUE_JMP_IF_P1_Z));
parm_size += sizeof(GLUE_JMP_IF_P1_Z);
}
csz=compileOpcodes(ctx,op->parms.parms[1],bufOut ? bufOut+parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, use_rv, NULL,&fUse1, canHaveDenormalOutput);
if (fUse1 > *fpStackUse) *fpStackUse=fUse1;
hasSecondHalf = preferredReturnValues || !OPCODE_IS_TRIVIAL(op->parms.parms[2]);
CHECK_SIZE_FORJMP(csz,doNonInlineIf_)
if (csz<0) RET_MINUS1_FAIL("if coc fial")
if (bufOut) GLUE_JMP_SET_OFFSET(bufOut + parm_size, csz + (hasSecondHalf?sizeof(GLUE_JMP_NC):0));
parm_size+=csz;
if (hasSecondHalf)
{
if (bufOut_len < parm_size + (int)sizeof(GLUE_JMP_NC)) RET_MINUS1_FAIL_FALLBACK("if len fail",doNonInlineIf_)
if (bufOut) memcpy(bufOut+parm_size,GLUE_JMP_NC,sizeof(GLUE_JMP_NC));
parm_size+=sizeof(GLUE_JMP_NC);
csz=compileOpcodes(ctx,op->parms.parms[2],bufOut ? bufOut+parm_size : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, use_rv, NULL, &fUse1, canHaveDenormalOutput);
CHECK_SIZE_FORJMP(csz,doNonInlineIf_)
if (csz<0) RET_MINUS1_FAIL("if coc 2 fail")
// update jump address
if (bufOut) GLUE_JMP_SET_OFFSET(bufOut + parm_size,csz);
parm_size+=csz;
if (fUse1 > *fpStackUse) *fpStackUse=fUse1;
}
return rv_offset + parm_size;
}
#ifdef GLUE_MAX_JMPSIZE
if (0)
{
unsigned char *newblock2,*newblock3,*ptr;
void *stub;
int stubsize;
doNonInlineIf_:
parm_size = parm_size_pre;
stub = GLUE_realAddress(nseel_asm_if,nseel_asm_if_end,&stubsize);
if (!stub || bufOut_len < parm_size + stubsize) RET_MINUS1_FAIL(stub ? "if sz fail" : "if addr fail")
if (bufOut)
{
int fUse2=0;
newblock2 = compileCodeBlockWithRet(ctx,op->parms.parms[1],computTableSize,namespacePathToThis, use_rv, NULL,&fUse2, canHaveDenormalOutput);
if (fUse2 > *fpStackUse) *fpStackUse=fUse2;
newblock3 = compileCodeBlockWithRet(ctx,op->parms.parms[2],computTableSize,namespacePathToThis, use_rv, NULL,&fUse2, canHaveDenormalOutput);
if (fUse2 > *fpStackUse) *fpStackUse=fUse2;
if (!newblock2 || !newblock3) RET_MINUS1_FAIL("if subblock gen fail")
ptr = bufOut + parm_size;
memcpy(ptr, stub, stubsize);
ptr=EEL_GLUE_set_immediate(ptr,(INT_PTR)newblock2);
EEL_GLUE_set_immediate(ptr,(INT_PTR)newblock3);
}
return rv_offset + parm_size + stubsize;
}
#endif
}
{
// special case: while
if (op->opcodeType == OPCODETYPE_FUNC1 && op->fntype == FN_WHILE)
{
*calledRvType = RETURNVALUE_BOOL;
#ifndef GLUE_INLINE_LOOPS
// todo: PPC looping support when loop length is small enough
{
unsigned char *pwr=bufOut;
unsigned char *newblock2;
int stubsz;
void *stubfunc = GLUE_realAddress(nseel_asm_repeatwhile,nseel_asm_repeatwhile_end,&stubsz);
if (!stubfunc || bufOut_len < stubsz) RET_MINUS1_FAIL(stubfunc ? "repeatwhile size fail" :"repeatwhile addr fail")
if (bufOut)
{
newblock2=compileCodeBlockWithRet(ctx,op->parms.parms[0],computTableSize,namespacePathToThis, RETURNVALUE_BOOL, NULL, fpStackUse, NULL);
if (!newblock2) RET_MINUS1_FAIL("repeatwhile ccbwr fail")
memcpy(pwr,stubfunc,stubsz);
pwr=EEL_GLUE_set_immediate(pwr,(INT_PTR)newblock2);
}
return rv_offset+stubsz;
}
#else
{
#ifndef GLUE_WHILE_END_NOJUMP
unsigned char *jzoutpt;
#endif
unsigned char *looppt;
int parm_size=0,subsz;
if (bufOut_len < parm_size + (int)(GLUE_WHILE_SETUP_SIZE + sizeof(GLUE_WHILE_BEGIN))) RET_MINUS1_FAIL("while size fail 1")
if (bufOut) memcpy(bufOut + parm_size,GLUE_WHILE_SETUP,GLUE_WHILE_SETUP_SIZE);
parm_size+=GLUE_WHILE_SETUP_SIZE;
looppt = bufOut + parm_size;
if (bufOut) memcpy(bufOut + parm_size,GLUE_WHILE_BEGIN,sizeof(GLUE_WHILE_BEGIN));
parm_size+=sizeof(GLUE_WHILE_BEGIN);
subsz = compileOpcodes(ctx,op->parms.parms[0],bufOut ? (bufOut + parm_size) : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, RETURNVALUE_BOOL, NULL,fpStackUse, NULL);
if (subsz<0) RET_MINUS1_FAIL("while coc fail")
if (bufOut_len < parm_size + (int)(sizeof(GLUE_WHILE_END) + sizeof(GLUE_WHILE_CHECK_RV))) RET_MINUS1_FAIL("which size fial 2")
parm_size+=subsz;
if (bufOut) memcpy(bufOut + parm_size, GLUE_WHILE_END, sizeof(GLUE_WHILE_END));
parm_size+=sizeof(GLUE_WHILE_END);
#ifndef GLUE_WHILE_END_NOJUMP
jzoutpt = bufOut + parm_size;
#endif
if (bufOut) memcpy(bufOut + parm_size, GLUE_WHILE_CHECK_RV, sizeof(GLUE_WHILE_CHECK_RV));
parm_size+=sizeof(GLUE_WHILE_CHECK_RV);
if (bufOut)
{
GLUE_JMP_SET_OFFSET(bufOut + parm_size,(looppt - (bufOut+parm_size)) );
#ifndef GLUE_WHILE_END_NOJUMP
GLUE_JMP_SET_OFFSET(jzoutpt, (bufOut + parm_size) - jzoutpt);
#endif
}
return rv_offset+parm_size;
}
#endif
}
// special case: loop
if (op->opcodeType == OPCODETYPE_FUNC2 && op->fntype == FN_LOOP)
{
int fUse1;
int parm_size = compileOpcodes(ctx,op->parms.parms[0],bufOut,bufOut_len, computTableSize, namespacePathToThis, RETURNVALUE_FPSTACK, NULL,&fUse1, NULL);
if (parm_size < 0) RET_MINUS1_FAIL("loop coc fail")
*calledRvType = RETURNVALUE_BOOL;
if (fUse1 > *fpStackUse) *fpStackUse=fUse1;
#ifndef GLUE_INLINE_LOOPS
// todo: PPC looping support when loop length is small enough
{
void *stub;
int stubsize;
unsigned char *newblock2, *p;
stub = GLUE_realAddress(nseel_asm_repeat,nseel_asm_repeat_end,&stubsize);
if (bufOut_len < parm_size + stubsize) RET_MINUS1_FAIL("loop size fail")
if (bufOut)
{
newblock2 = compileCodeBlockWithRet(ctx,op->parms.parms[1],computTableSize,namespacePathToThis, RETURNVALUE_IGNORE, NULL,fpStackUse, NULL);
p = bufOut + parm_size;
memcpy(p, stub, stubsize);
p=EEL_GLUE_set_immediate(p,(INT_PTR)newblock2);
}
return rv_offset + parm_size + stubsize;
}
#else
{
int subsz;
int fUse2=0;
unsigned char *skipptr1,*loopdest;
if (bufOut_len < parm_size + (int)(sizeof(GLUE_LOOP_LOADCNT) + GLUE_LOOP_CLAMPCNT_SIZE + GLUE_LOOP_BEGIN_SIZE)) RET_MINUS1_FAIL("loop size fail")
// store, convert to int, compare against 1, if less than, skip to end
if (bufOut) memcpy(bufOut+parm_size,GLUE_LOOP_LOADCNT,sizeof(GLUE_LOOP_LOADCNT));
parm_size += sizeof(GLUE_LOOP_LOADCNT);
skipptr1 = bufOut+parm_size;
// compare aginst max loop length, jump to loop start if not above it
if (bufOut) memcpy(bufOut+parm_size,GLUE_LOOP_CLAMPCNT,GLUE_LOOP_CLAMPCNT_SIZE);
parm_size += GLUE_LOOP_CLAMPCNT_SIZE;
// loop code:
loopdest = bufOut + parm_size;
if (bufOut) memcpy(bufOut+parm_size,GLUE_LOOP_BEGIN,GLUE_LOOP_BEGIN_SIZE);
parm_size += GLUE_LOOP_BEGIN_SIZE;
subsz = compileOpcodes(ctx,op->parms.parms[1],bufOut ? (bufOut + parm_size) : NULL,bufOut_len - parm_size, computTableSize, namespacePathToThis, RETURNVALUE_IGNORE, NULL, &fUse2, NULL);
if (subsz<0) RET_MINUS1_FAIL("loop coc fail")
if (fUse2 > *fpStackUse) *fpStackUse=fUse2;
parm_size += subsz;
if (bufOut_len < parm_size + (int)sizeof(GLUE_LOOP_END)) RET_MINUS1_FAIL("loop size fail 2")
if (bufOut) memcpy(bufOut+parm_size,GLUE_LOOP_END,sizeof(GLUE_LOOP_END));
parm_size += sizeof(GLUE_LOOP_END);
if (bufOut)
{
GLUE_JMP_SET_OFFSET(bufOut + parm_size,loopdest - (bufOut+parm_size));
GLUE_JMP_SET_OFFSET(skipptr1, (bufOut+parm_size) - skipptr1);
}
return rv_offset + parm_size;
}
#endif
}
}
switch (op->opcodeType)
{
case OPCODETYPE_DIRECTVALUE:
if (preferredReturnValues == RETURNVALUE_BOOL)
{
int w = fabs(op->parms.dv.directValue) >= NSEEL_CLOSEFACTOR;
int wsz=(w?sizeof(GLUE_SET_P1_NZ):sizeof(GLUE_SET_P1_Z));
*calledRvType = RETURNVALUE_BOOL;
if (bufOut_len < wsz) RET_MINUS1_FAIL("direct bool size fail3")
if (bufOut) memcpy(bufOut,w?GLUE_SET_P1_NZ:GLUE_SET_P1_Z,wsz);
return rv_offset+wsz;
}
else if (preferredReturnValues & RETURNVALUE_FPSTACK)
{
#ifdef GLUE_HAS_FLDZ
if (op->parms.dv.directValue == 0.0)
{
*fpStackUse = 1;
*calledRvType = RETURNVALUE_FPSTACK;
if (bufOut_len < sizeof(GLUE_FLDZ)) RET_MINUS1_FAIL("direct fp fail 1")
if (bufOut) memcpy(bufOut,GLUE_FLDZ,sizeof(GLUE_FLDZ));
return rv_offset+sizeof(GLUE_FLDZ);
}
#endif
#ifdef GLUE_HAS_FLD1
if (op->parms.dv.directValue == 1.0)
{
*fpStackUse = 1;
*calledRvType = RETURNVALUE_FPSTACK;
if (bufOut_len < sizeof(GLUE_FLD1)) RET_MINUS1_FAIL("direct fp fail 1")
if (bufOut) memcpy(bufOut,GLUE_FLD1,sizeof(GLUE_FLD1));
return rv_offset+sizeof(GLUE_FLD1);
}
#endif
}
// fall through
case OPCODETYPE_DIRECTVALUE_TEMPSTRING:
case OPCODETYPE_VALUE_FROM_NAMESPACENAME:
case OPCODETYPE_VARPTR:
case OPCODETYPE_VARPTRPTR:
#ifdef GLUE_MOV_PX_DIRECTVALUE_TOSTACK_SIZE
if (OPCODE_IS_TRIVIAL(op))
{
if (preferredReturnValues & RETURNVALUE_FPSTACK)
{
*fpStackUse = 1;
if (bufOut_len < GLUE_MOV_PX_DIRECTVALUE_TOSTACK_SIZE) RET_MINUS1_FAIL("direct fp fail 2")
if (bufOut)
{
if (generateValueToReg(ctx,op,bufOut,-1,namespacePathToThis, 1 /*allow caching*/)<0) RET_MINUS1_FAIL("direct fp fail gvr")
}
*calledRvType = RETURNVALUE_FPSTACK;
return rv_offset+GLUE_MOV_PX_DIRECTVALUE_TOSTACK_SIZE;
}
}
#endif
if (bufOut_len < GLUE_MOV_PX_DIRECTVALUE_SIZE)
{
RET_MINUS1_FAIL("direct value fail 1")
}
if (bufOut)
{
if (generateValueToReg(ctx,op,bufOut,0,namespacePathToThis,
(preferredReturnValues&(RETURNVALUE_FPSTACK|RETURNVALUE_CACHEABLE))!=0)<0)
{
RET_MINUS1_FAIL("direct value gvr fail3")
}
}
return rv_offset + GLUE_MOV_PX_DIRECTVALUE_SIZE;
case OPCODETYPE_FUNCX:
case OPCODETYPE_FUNC1:
case OPCODETYPE_FUNC2:
case OPCODETYPE_FUNC3:
if (op->fntype == FUNCTYPE_EELFUNC)
{
int a;
a = compileEelFunctionCall(ctx,op,bufOut,bufOut_len,computTableSize,namespacePathToThis, calledRvType,fpStackUse,canHaveDenormalOutput);
if (a<0) return a;
rv_offset += a;
}
else
{
int a;
a = compileNativeFunctionCall(ctx,op,bufOut,bufOut_len,computTableSize,namespacePathToThis, calledRvType,fpStackUse,preferredReturnValues,canHaveDenormalOutput);
if (a<0)return a;
rv_offset += a;
}
return rv_offset;
}
RET_MINUS1_FAIL("default opcode fail")
}
#ifdef DUMP_OPS_DURING_COMPILE
FILE *g_debugfp;
int g_debugfp_indent;
int g_debugfp_histsz=0;
void dumpOp(compileContext *ctx, opcodeRec *op, int start)
{
if (start>=0)
{
if (g_debugfp)
{
static opcodeRec **hist;
int x;
int hit=0;
if (!hist) hist = (opcodeRec**) calloc(1024,1024*sizeof(opcodeRec*));
for(x=0;x<g_debugfp_histsz;x++)
{
if (hist[x] == op) { hit=1; break; }
}
if (x ==g_debugfp_histsz && g_debugfp_histsz<1024*1024) hist[g_debugfp_histsz++] = op;
if (!start)
{
g_debugfp_indent-=2;
fprintf(g_debugfp,"%*s}(join)\n",g_debugfp_indent," ");
}
if (g_debugfp_indent>=100) *(char *)1=0;
fprintf(g_debugfp,"%*s{ %p : %d%s: ",g_debugfp_indent," ",op,op->opcodeType, hit ? " -- DUPLICATE" : "");
switch (op->opcodeType)
{
case OPCODETYPE_DIRECTVALUE:
fprintf(g_debugfp,"dv %f",op->parms.dv.directValue);
break;
case OPCODETYPE_VARPTR:
if (op->relname && op->relname[0])
{
fprintf(g_debugfp,"var %s",op->relname);
}
else
{
int wb;
for (wb = 0; wb < ctx->varTable_numBlocks; wb ++)
{
char **plist=ctx->varTable_Names[wb];
if (!plist) break;
if (op->parms.dv.valuePtr >= ctx->varTable_Values[wb] && op->parms.dv.valuePtr < ctx->varTable_Values[wb] + NSEEL_VARS_PER_BLOCK)
{
fprintf(g_debugfp,"var %s",plist[op->parms.dv.valuePtr - ctx->varTable_Values[wb]]);
break;
}
}
}
break;
case OPCODETYPE_FUNC1:
case OPCODETYPE_FUNC2:
case OPCODETYPE_FUNC3:
case OPCODETYPE_FUNCX:
if (op->fntype == FUNCTYPE_FUNCTIONTYPEREC)
{
functionType *p=(functionType*)op->fn;
fprintf(g_debugfp,"func %d: %s",p->nParams&0xff,p->name);
}
else
fprintf(g_debugfp,"sf %d",op->fntype);
break;
}
fprintf(g_debugfp,"\n");
g_debugfp_indent+=2;
}
}
else
{
if (g_debugfp)
{
g_debugfp_indent-=2;
fprintf(g_debugfp,"%*s}%p\n",g_debugfp_indent," ",op);
}
}
}
#endif
int compileOpcodes(compileContext *ctx, opcodeRec *op, unsigned char *bufOut, int bufOut_len, int *computTableSize, const namespaceInformation *namespacePathToThis,
int supportedReturnValues, int *rvType, int *fpStackUse, int *canHaveDenormalOutput)
{
int code_returns=RETURNVALUE_NORMAL;
int fpsu=0;
int codesz;
int denorm=0;
#ifdef DUMP_OPS_DURING_COMPILE
dumpOp(ctx,op,1);
#endif
codesz = compileOpcodesInternal(ctx,op,bufOut,bufOut_len,computTableSize,namespacePathToThis,&code_returns, supportedReturnValues,&fpsu,&denorm);
if (denorm && canHaveDenormalOutput) *canHaveDenormalOutput=1;
#ifdef DUMP_OPS_DURING_COMPILE
dumpOp(ctx,op,-1);
#endif
#ifdef EEL_DUMP_OPS
// dump opcode trees for verification, after optimizing
if (g_eel_dump_fp2)
{
fprintf(g_eel_dump_fp2,"-- compileOpcodes generated %d bytes of code!\r\n",codesz);
}
#endif
if (codesz < 0) return codesz;
/*
{
char buf[512];
sprintf(buf,"opcode %d %d (%s): fpu use: %d\n",op->opcodeType,op->fntype,
op->opcodeType >= OPCODETYPE_FUNC1 && op->fntype == FUNCTYPE_FUNCTIONTYPEREC ? (
((functionType *)op->fn)->name
) : "",
fpsu);
OutputDebugString(buf);
}
*/
if (fpStackUse) *fpStackUse=fpsu;
if (bufOut) bufOut += codesz;
bufOut_len -= codesz;
if (code_returns == RETURNVALUE_BOOL && !(supportedReturnValues & RETURNVALUE_BOOL) && supportedReturnValues)
{
int stubsize;
void *stub = GLUE_realAddress(nseel_asm_booltofp,nseel_asm_booltofp_end,&stubsize);
if (!stub || bufOut_len < stubsize) RET_MINUS1_FAIL(stub?"booltofp size":"booltfp addr")
if (bufOut)
{
memcpy(bufOut,stub,stubsize);
bufOut += stubsize;
}
codesz+=stubsize;
bufOut_len -= stubsize;
code_returns = RETURNVALUE_FPSTACK;
}
// default processing of code_returns to meet return value requirements
if (supportedReturnValues & code_returns)
{
if (rvType) *rvType = code_returns;
return codesz;
}
if (rvType) *rvType = RETURNVALUE_IGNORE;
if (code_returns == RETURNVALUE_NORMAL)
{
if (supportedReturnValues & (RETURNVALUE_FPSTACK|RETURNVALUE_BOOL))
{
if (bufOut_len < GLUE_PUSH_VAL_AT_PX_TO_FPSTACK_SIZE) RET_MINUS1_FAIL("pushvalatpxtofpstack,size")
if (bufOut)
{
GLUE_PUSH_VAL_AT_PX_TO_FPSTACK(bufOut,0); // always fld qword [eax] but we might change that later
bufOut += GLUE_PUSH_VAL_AT_PX_TO_FPSTACK_SIZE;
}
codesz += GLUE_PUSH_VAL_AT_PX_TO_FPSTACK_SIZE;
bufOut_len -= GLUE_PUSH_VAL_AT_PX_TO_FPSTACK_SIZE;
if (supportedReturnValues & RETURNVALUE_BOOL)
{
code_returns = RETURNVALUE_FPSTACK;
}
else
{
if (rvType) *rvType = RETURNVALUE_FPSTACK;
}
}
}
if (code_returns == RETURNVALUE_FPSTACK)
{
if (supportedReturnValues & (RETURNVALUE_BOOL|RETURNVALUE_BOOL_REVERSED))
{
int stubsize;
void *stub;
if (supportedReturnValues & RETURNVALUE_BOOL_REVERSED)
{
if (rvType) *rvType = RETURNVALUE_BOOL_REVERSED;
stub = GLUE_realAddress(nseel_asm_fptobool_rev,nseel_asm_fptobool_rev_end,&stubsize);
}
else
{
if (rvType) *rvType = RETURNVALUE_BOOL;
stub = GLUE_realAddress(nseel_asm_fptobool,nseel_asm_fptobool_end,&stubsize);
}
if (!stub || bufOut_len < stubsize) RET_MINUS1_FAIL(stub?"fptobool size":"fptobool addr")
if (bufOut)
{
memcpy(bufOut,stub,stubsize);
bufOut += stubsize;
}
codesz+=stubsize;
bufOut_len -= stubsize;
}
else if (supportedReturnValues & RETURNVALUE_NORMAL)
{
if (computTableSize) (*computTableSize) ++;
if (bufOut_len < GLUE_POP_FPSTACK_TO_WTP_TO_PX_SIZE) RET_MINUS1_FAIL("popfpstacktowtptopxsize")
// generate fp-pop to temp space
if (bufOut) GLUE_POP_FPSTACK_TO_WTP_TO_PX(bufOut,0);
codesz+=GLUE_POP_FPSTACK_TO_WTP_TO_PX_SIZE;
if (rvType) *rvType = RETURNVALUE_NORMAL;
}
else
{
// toss return value that will be ignored
if (bufOut_len < GLUE_POP_FPSTACK_SIZE) RET_MINUS1_FAIL("popfpstack size")
if (bufOut) memcpy(bufOut,GLUE_POP_FPSTACK,GLUE_POP_FPSTACK_SIZE);
codesz+=GLUE_POP_FPSTACK_SIZE;
}
}
return codesz;
}
#if 0
static void movestringover(char *str, int amount)
{
char tmp[1024+8];
int l=(int)strlen(str);
l=wdl_min(1024-amount-1,l);
memcpy(tmp,str,l+1);
while (l >= 0 && tmp[l]!='\n') l--;
l++;
tmp[l]=0;//ensure we null terminate
memcpy(str+amount,tmp,l+1);
}
#endif
//------------------------------------------------------------------------------
NSEEL_CODEHANDLE NSEEL_code_compile(NSEEL_VMCTX _ctx, const char *_expression, int lineoffs)
{
return NSEEL_code_compile_ex(_ctx,_expression,lineoffs,0);
}
typedef struct topLevelCodeSegmentRec {
struct topLevelCodeSegmentRec *_next;
void *code;
int codesz;
int tmptable_use;
} topLevelCodeSegmentRec;
NSEEL_CODEHANDLE NSEEL_code_compile_ex(NSEEL_VMCTX _ctx, const char *_expression, int lineoffs, int compile_flags)
{
compileContext *ctx = (compileContext *)_ctx;
const char *endptr;
const char *_expression_end;
codeHandleType *handle;
topLevelCodeSegmentRec *startpts_tail=NULL;
topLevelCodeSegmentRec *startpts=NULL;
_codeHandleFunctionRec *oldCommonFunctionList;
int curtabptr_sz=0;
void *curtabptr=NULL;
int had_err=0;
if (!ctx) return 0;
ctx->directValueCache=0;
ctx->optimizeDisableFlags=0;
ctx->gotEndOfInput=0;
ctx->current_compile_flags = compile_flags;
if (compile_flags & NSEEL_CODE_COMPILE_FLAG_COMMONFUNCS_RESET)
{
ctx->functions_common=NULL; // reset common function list
}
else
{
// reset common compiled function code, forcing a recompile if shared
_codeHandleFunctionRec *a = ctx->functions_common;
while (a)
{
_codeHandleFunctionRec *b = a->derivedCopies;
if (a->localstorage)
{
// force local storage actual values to be reallocated if used again
memset(a->localstorage,0,sizeof(EEL_F *) * a->localstorage_size);
}
a->startptr = NULL; // force this copy to be recompiled
a->startptr_size = -1;
while (b)
{
b->startptr = NULL; // force derived copies to get recompiled
b->startptr_size = -1;
// no need to reset b->localstorage, since it points to a->localstorage
b=b->derivedCopies;
}
a=a->next;
}
}
ctx->last_error_string[0]=0;
if (!_expression || !*_expression) return 0;
_expression_end = _expression + strlen(_expression);
oldCommonFunctionList = ctx->functions_common;
ctx->isGeneratingCommonFunction=0;
ctx->isSharedFunctions = !!(compile_flags & NSEEL_CODE_COMPILE_FLAG_COMMONFUNCS);
ctx->functions_local = NULL;
freeBlocks(&ctx->tmpblocks_head); // free blocks
freeBlocks(&ctx->blocks_head); // free blocks
freeBlocks(&ctx->blocks_head_data); // free blocks
memset(ctx->l_stats,0,sizeof(ctx->l_stats));
handle = (codeHandleType*)newDataBlock(sizeof(codeHandleType),8);
if (!handle)
{
return 0;
}
memset(handle,0,sizeof(codeHandleType));
ctx->l_stats[0] += (int)(_expression_end - _expression);
ctx->tmpCodeHandle = handle;
endptr=_expression;
while (*endptr)
{
int computTableTop = 0;
int startptr_size=0;
void *startptr=NULL;
opcodeRec *start_opcode=NULL;
const char *expr=endptr;
int function_numparms=0;
char is_fname[NSEEL_MAX_VARIABLE_NAMELEN+1];
is_fname[0]=0;
memset(ctx->function_localTable_Size,0,sizeof(ctx->function_localTable_Size));
memset(ctx->function_localTable_Names,0,sizeof(ctx->function_localTable_Names));
ctx->function_localTable_ValuePtrs=0;
ctx->function_usesNamespaces=0;
ctx->function_curName=NULL;
ctx->function_globalFlag=0;
ctx->errVar=0;
// single out top level segment
{
int had_something = 0, pcnt=0, pcnt2=0;
int state=0;
for (;;)
{
int l;
const char *p=nseel_simple_tokenizer(&endptr,_expression_end,&l,&state);
if (!p)
{
if (pcnt || pcnt2) ctx->gotEndOfInput|=4;
break;
}
if (*p == ';')
{
if (had_something && !pcnt && !pcnt2) break;
}
else if (*p == '/' && l > 1 && (p[1] == '/' || p[1] == '*'))
{
if (l > 19 && !strnicmp(p,"//#eel-no-optimize:",19))
ctx->optimizeDisableFlags = atoi(p+19);
}
else
{
if (!had_something)
{
expr = p;
had_something = 1;
}
if (*p == '(') pcnt++;
else if (*p == ')') { if (--pcnt<0) pcnt=0; }
else if (*p == '[') pcnt2++;
else if (*p == ']') { if (--pcnt2<0) pcnt2=0; }
}
}
if (!*expr || !had_something) break;
}
// parse
{
int tmplen,funcname_len;
const char *p = expr;
const char *tok1 = nseel_simple_tokenizer(&p,endptr,&tmplen,NULL);
const char *funcname = nseel_simple_tokenizer(&p,endptr,&funcname_len,NULL);
if (tok1 && funcname && tmplen == 8 && !strnicmp(tok1,"function",8) && (isalpha(funcname[0]) || funcname[0] == '_'))
{
int had_parms_locals=0;
if (funcname_len > sizeof(is_fname)-1) funcname_len=sizeof(is_fname)-1;
memcpy(is_fname, funcname, funcname_len);
is_fname[funcname_len]=0;
ctx->function_curName = is_fname; // only assigned for the duration of the loop, cleared later //-V507
while (NULL != (tok1 = nseel_simple_tokenizer(&p,endptr,&tmplen,NULL)))
{
int is_parms = 0, localTableContext = 0;
int maxcnt=0;
const char *sp_save;
if (tok1[0] == '(')
{
if (had_parms_locals)
{
expr = p-1; // begin compilation at this code!
break;
}
is_parms = 1;
}
else
{
if (tmplen == 5 && !strnicmp(tok1,"local",tmplen)) localTableContext=0;
else if (tmplen == 6 && !strnicmp(tok1,"static",tmplen)) localTableContext=0;
else if (tmplen == 8 && !strnicmp(tok1,"instance",tmplen)) localTableContext=1;
else if ((tmplen == 7 && !strnicmp(tok1,"globals",tmplen)) ||
(tmplen == 6 && !strnicmp(tok1,"global",tmplen)))
{
ctx->function_globalFlag = 1;
localTableContext=2;
}
else break; // unknown token!
tok1 = nseel_simple_tokenizer(&p,endptr,&tmplen,NULL);
if (!tok1 || tok1[0] != '(') break;
}
had_parms_locals = 1;
sp_save=p;
while (NULL != (tok1 = nseel_simple_tokenizer(&p,endptr,&tmplen,NULL)))
{
if (tok1[0] == ')') break;
if (*tok1 == '#' && localTableContext!=1 && localTableContext!=2)
{
ctx->errVar = (int) (tok1 - _expression);
lstrcpyn_safe(ctx->last_error_string,"#string can only be in instance() or globals()",sizeof(ctx->last_error_string));
goto had_error;
}
if (isalpha(*tok1) || *tok1 == '_' || *tok1 == '#')
{
maxcnt++;
if (p < endptr && *p == '*')
{
if (!is_parms && localTableContext!=2)
{
ctx->errVar = (int) (p - _expression);
lstrcpyn_safe(ctx->last_error_string,"namespace* can only be used in parameters or globals()",sizeof(ctx->last_error_string));
goto had_error;
}
p++;
}
}
else if (*tok1 != ',')
{
ctx->errVar = (int)(tok1 - _expression);
lstrcpyn_safe(ctx->last_error_string,"unknown character in function parameters",sizeof(ctx->last_error_string));
goto had_error;
}
}
if (tok1 && maxcnt > 0)
{
char **ot = ctx->function_localTable_Names[localTableContext];
const int osz = ctx->function_localTable_Size[localTableContext];
maxcnt += osz;
ctx->function_localTable_Names[localTableContext] = (char **)newTmpBlock(ctx,sizeof(char *) * maxcnt);
if (ctx->function_localTable_Names[localTableContext])
{
int i=osz;
if (osz && ot) memcpy(ctx->function_localTable_Names[localTableContext],ot,sizeof(char *) * osz);
p=sp_save;
while (NULL != (tok1 = nseel_simple_tokenizer(&p,endptr,&tmplen,NULL)))
{
if (tok1[0] == ')') break;
if (isalpha(*tok1) || *tok1 == '_' || *tok1 == '#')
{
char *newstr;
int l = tmplen;
if (*p == '*') // xyz* for namespace
{
p++;
l++;
}
if (l > NSEEL_MAX_VARIABLE_NAMELEN) l = NSEEL_MAX_VARIABLE_NAMELEN;
newstr = newTmpBlock(ctx,l+1);
if (newstr)
{
memcpy(newstr,tok1,l);
newstr[l]=0;
ctx->function_localTable_Names[localTableContext][i++] = newstr;
}
}
}
ctx->function_localTable_Size[localTableContext]=i;
if (is_parms) function_numparms = i;
}
}
}
}
}
if (ctx->function_localTable_Size[0]>0)
{
ctx->function_localTable_ValuePtrs =
ctx->isSharedFunctions ? newDataBlock(ctx->function_localTable_Size[0] * sizeof(EEL_F *),8) :
newTmpBlock(ctx,ctx->function_localTable_Size[0] * sizeof(EEL_F *));
if (!ctx->function_localTable_ValuePtrs)
{
ctx->function_localTable_Size[0]=0;
function_numparms=0;
}
else
{
memset(ctx->function_localTable_ValuePtrs,0,sizeof(EEL_F *) * ctx->function_localTable_Size[0]); // force values to be allocated
}
}
{
int nseelparse(compileContext* context);
void nseelrestart (void *input_file ,void *yyscanner );
ctx->rdbuf_start = _expression;
#ifdef NSEEL_SUPER_MINIMAL_LEXER
ctx->rdbuf = expr;
ctx->rdbuf_end = endptr;
if (!nseelparse(ctx) && !ctx->errVar)
{
start_opcode = ctx->result;
}
#else
nseelrestart(NULL,ctx->scanner);
ctx->rdbuf = expr;
ctx->rdbuf_end = endptr;
if (!nseelparse(ctx) && !ctx->errVar)
{
start_opcode = ctx->result;
}
if (ctx->errVar)
{
const char *p=expr;
ctx->errVar += expr-_expression;
}
#endif
ctx->rdbuf = NULL;
}
if (start_opcode)
{
int rvMode=0, fUse=0;
#ifdef LOG_OPT
char buf[512];
int sd=0;
sprintf(buf,"pre opt sz=%d (tsackDepth=%d)\n",compileOpcodes(ctx,start_opcode,NULL,1024*1024*256,NULL, NULL,RETURNVALUE_IGNORE,NULL,&sd,NULL),sd);
#ifdef _WIN32
OutputDebugString(buf);
#else
printf("%s\n",buf);
#endif
#endif
#ifdef EEL_DUMP_OPS
// dump opcode trees for verification, before optimizing
if (g_eel_dump_fp)
{
fprintf(g_eel_dump_fp,"-- opcode chunk --\r\n");
dumpOpcodeTree(ctx,g_eel_dump_fp,start_opcode,2);
}
#endif
if (!(ctx->optimizeDisableFlags&OPTFLAG_NO_OPTIMIZE)) optimizeOpcodes(ctx,start_opcode,is_fname[0] ? 1 : 0);
#ifdef LOG_OPT
sprintf(buf,"post opt sz=%d, stack depth=%d\n",compileOpcodes(ctx,start_opcode,NULL,1024*1024*256,NULL,NULL, RETURNVALUE_IGNORE,NULL,&sd,NULL),sd);
#ifdef _WIN32
OutputDebugString(buf);
#else
printf("%s\n",buf);
#endif
#endif
#ifdef EEL_DUMP_OPS
// dump opcode trees for verification, after optimizing
if (g_eel_dump_fp2)
{
fprintf(g_eel_dump_fp2,"-- POST-OPTIMIZED opcode chunk --\r\n");
dumpOpcodeTree(ctx,g_eel_dump_fp2,start_opcode,2);
}
#endif
if (is_fname[0])
{
_codeHandleFunctionRec *fr = ctx->isSharedFunctions ? newDataBlock(sizeof(_codeHandleFunctionRec),8) :
newTmpBlock(ctx,sizeof(_codeHandleFunctionRec));
if (fr)
{
memset(fr,0,sizeof(_codeHandleFunctionRec));
fr->startptr_size = -1;
fr->opcodes = start_opcode;
if (ctx->function_localTable_Size[0] > 0 && ctx->function_localTable_ValuePtrs)
{
if (ctx->function_localTable_Names[0])
{
int i;
for(i=0;i<function_numparms;i++)
{
const char *nptr = ctx->function_localTable_Names[0][i];
if (nptr && *nptr && nptr[strlen(nptr)-1] == '*')
{
fr->parameterAsNamespaceMask |= ((unsigned int)1)<<i;
}
}
}
fr->num_params=function_numparms;
fr->localstorage = ctx->function_localTable_ValuePtrs;
fr->localstorage_size = ctx->function_localTable_Size[0];
}
fr->usesNamespaces = ctx->function_usesNamespaces;
fr->isCommonFunction = ctx->isSharedFunctions;
lstrcpyn_safe(fr->fname,is_fname,sizeof(fr->fname));
if (ctx->isSharedFunctions)
{
fr->next = ctx->functions_common;
ctx->functions_common = fr;
}
else
{
fr->next = ctx->functions_local;
ctx->functions_local = fr;
}
}
continue;
}
#ifdef DUMP_OPS_DURING_COMPILE
g_debugfp_indent=0;
g_debugfp_histsz=0;
g_debugfp = fopen("C:/temp/foo.txt","w");
#endif
startptr_size = compileOpcodes(ctx,start_opcode,NULL,1024*1024*256,NULL, NULL,
is_fname[0] ? (RETURNVALUE_NORMAL|RETURNVALUE_FPSTACK) : RETURNVALUE_IGNORE, &rvMode, &fUse, NULL); // if not a function, force return value as address (avoid having to pop it ourselves
// if a function, allow the code to decide how return values are generated
#ifdef DUMP_OPS_DURING_COMPILE
if (g_debugfp) fclose(g_debugfp);
g_debugfp=0;
#endif
if (!startptr_size) continue; // optimized away
if (startptr_size>0)
{
startptr = newTmpBlock(ctx,startptr_size);
if (startptr)
{
startptr_size=compileOpcodes(ctx,start_opcode,(unsigned char*)startptr,startptr_size,&computTableTop, NULL, RETURNVALUE_IGNORE, NULL,NULL, NULL);
if (startptr_size<=0) startptr = NULL;
}
}
}
if (!startptr)
{
had_error:
#ifdef NSEEL_EEL1_COMPAT_MODE
continue;
#else
//if (!ctx->last_error_string[0])
{
int byteoffs = ctx->errVar;
int linenumber;
char cur_err[sizeof(ctx->last_error_string)];
lstrcpyn_safe(cur_err,ctx->last_error_string,sizeof(cur_err));
if (cur_err[0]) lstrcatn(cur_err,": ",sizeof(cur_err));
else lstrcpyn_safe(cur_err,"syntax error: ",sizeof(cur_err));
if (_expression + byteoffs >= _expression_end)
{
if (ctx->gotEndOfInput&4) byteoffs = (int)(expr-_expression);
else byteoffs=(int)(_expression_end-_expression);
}
if (byteoffs < 0) byteoffs=0;
linenumber=findLineNumber(_expression,byteoffs)+1;
if (ctx->gotEndOfInput&4)
{
snprintf(ctx->last_error_string,sizeof(ctx->last_error_string),"%d: %smissing ) or ]",linenumber+lineoffs,cur_err);
}
else
{
const char *p = _expression + byteoffs;
int x=0, right_amt_nospace=0, left_amt_nospace=0;
while (x < 32 && p-x > _expression && p[-x] != '\r' && p[-x] != '\n')
{
if (!isspace(p[-x])) left_amt_nospace=x;
x++;
}
x=0;
while (x < 60 && p[x] && p[x] != '\r' && p[x] != '\n')
{
if (!isspace(p[x])) right_amt_nospace=x;
x++;
}
if (right_amt_nospace<1) right_amt_nospace=1;
// display left_amt >>>> right_amt_nospace
if (left_amt_nospace > 0)
snprintf(ctx->last_error_string,sizeof(ctx->last_error_string),"%d: %s'%.*s <!> %.*s'",linenumber+lineoffs,cur_err,
left_amt_nospace,p-left_amt_nospace,
right_amt_nospace,p);
else
snprintf(ctx->last_error_string,sizeof(ctx->last_error_string),"%d: %s'%.*s'",linenumber+lineoffs,cur_err,right_amt_nospace,p);
}
}
startpts=NULL;
startpts_tail=NULL;
had_err=1;
break;
#endif
}
if (!is_fname[0]) // redundant check (if is_fname[0] is set and we succeeded, it should continue)
// but we'll be on the safe side
{
topLevelCodeSegmentRec *p = newTmpBlock(ctx,sizeof(topLevelCodeSegmentRec));
p->_next=0;
p->code = startptr;
p->codesz = startptr_size;
p->tmptable_use = computTableTop;
if (!startpts_tail) startpts_tail=startpts=p;
else
{
startpts_tail->_next=p;
startpts_tail=p;
}
if (curtabptr_sz < computTableTop)
{
curtabptr_sz=computTableTop;
}
}
}
memset(ctx->function_localTable_Size,0,sizeof(ctx->function_localTable_Size));
memset(ctx->function_localTable_Names,0,sizeof(ctx->function_localTable_Names));
ctx->function_localTable_ValuePtrs=0;
ctx->function_usesNamespaces=0;
ctx->function_curName=NULL;
ctx->function_globalFlag=0;
ctx->tmpCodeHandle = NULL;
if (handle->want_stack)
{
if (!handle->stack) startpts=NULL;
}
if (startpts)
{
curtabptr_sz += 2; // many functions use the worktable for temporary storage of up to 2 EEL_F's
handle->workTable_size = curtabptr_sz;
handle->workTable = curtabptr = newDataBlock((curtabptr_sz+MIN_COMPUTABLE_SIZE + COMPUTABLE_EXTRA_SPACE) * sizeof(EEL_F),32);
#ifdef EEL_VALIDATE_WORKTABLE_USE
if (curtabptr) memset(curtabptr,0x3a,(curtabptr_sz+MIN_COMPUTABLE_SIZE + COMPUTABLE_EXTRA_SPACE) * sizeof(EEL_F));
#endif
if (!curtabptr) startpts=NULL;
}
if (startpts || (!had_err && (compile_flags & NSEEL_CODE_COMPILE_FLAG_COMMONFUNCS)))
{
unsigned char *writeptr;
topLevelCodeSegmentRec *p=startpts;
int size=sizeof(GLUE_RET)+GLUE_FUNC_ENTER_SIZE+GLUE_FUNC_LEAVE_SIZE; // for ret at end :)
int wtpos=0;
// now we build one big code segment out of our list of them, inserting a mov esi, computable before each item as necessary
while (p)
{
if (wtpos <= 0)
{
wtpos=MIN_COMPUTABLE_SIZE;
size += GLUE_RESET_WTP(NULL,0);
}
size+=p->codesz;
wtpos -= p->tmptable_use;
p=p->_next;
}
handle->code = newCodeBlock(size,32);
if (handle->code)
{
writeptr=(unsigned char *)handle->code;
#if GLUE_FUNC_ENTER_SIZE > 0
memcpy(writeptr,&GLUE_FUNC_ENTER,GLUE_FUNC_ENTER_SIZE);
writeptr += GLUE_FUNC_ENTER_SIZE;
#endif
p=startpts;
wtpos=0;
while (p)
{
if (wtpos <= 0)
{
wtpos=MIN_COMPUTABLE_SIZE;
writeptr+=GLUE_RESET_WTP(writeptr,curtabptr);
}
memcpy(writeptr,(char*)p->code,p->codesz);
writeptr += p->codesz;
wtpos -= p->tmptable_use;
p=p->_next;
}
#if GLUE_FUNC_LEAVE_SIZE > 0
memcpy(writeptr,&GLUE_FUNC_LEAVE,GLUE_FUNC_LEAVE_SIZE);
writeptr += GLUE_FUNC_LEAVE_SIZE;
#endif
memcpy(writeptr,&GLUE_RET,sizeof(GLUE_RET)); writeptr += sizeof(GLUE_RET);
ctx->l_stats[1]=size;
handle->code_size = (int) (writeptr - (unsigned char *)handle->code);
#if defined(__arm__) || defined(__aarch64__)
__clear_cache(handle->code,writeptr);
#endif
}
handle->blocks = ctx->blocks_head;
handle->blocks_data = ctx->blocks_head_data;
ctx->blocks_head=0;
ctx->blocks_head_data=0;
}
else
{
// failed compiling, or failed calloc()
handle=NULL; // return NULL (after resetting blocks_head)
}
ctx->directValueCache=0;
ctx->functions_local = NULL;
ctx->isGeneratingCommonFunction=0;
ctx->isSharedFunctions=0;
freeBlocks(&ctx->tmpblocks_head); // free blocks
freeBlocks(&ctx->blocks_head); // free blocks of code (will be nonzero only on error)
freeBlocks(&ctx->blocks_head_data); // free blocks of data (will be nonzero only on error)
if (handle)
{
handle->compile_flags = compile_flags;
handle->ramPtr = ctx->ram_state.blocks;
memcpy(handle->code_stats,ctx->l_stats,sizeof(ctx->l_stats));
nseel_evallib_stats[0]+=ctx->l_stats[0];
nseel_evallib_stats[1]+=ctx->l_stats[1];
nseel_evallib_stats[2]+=ctx->l_stats[2];
nseel_evallib_stats[3]+=ctx->l_stats[3];
nseel_evallib_stats[4]++;
}
else
{
ctx->functions_common = oldCommonFunctionList; // failed compiling, remove any added common functions from the list
// remove any derived copies of functions due to error, since we may have added some that have been freed
while (oldCommonFunctionList)
{
oldCommonFunctionList->derivedCopies=NULL;
oldCommonFunctionList=oldCommonFunctionList->next;
}
}
memset(ctx->l_stats,0,sizeof(ctx->l_stats));
return (NSEEL_CODEHANDLE)handle;
}
//------------------------------------------------------------------------------
void NSEEL_code_execute(NSEEL_CODEHANDLE code)
{
#ifndef GLUE_TABPTR_IGNORED
INT_PTR tabptr;
#endif
INT_PTR codeptr;
codeHandleType *h = (codeHandleType *)code;
if (!h || !h->code) return;
codeptr = (INT_PTR) h->code;
#if 0
{
unsigned int *p=(unsigned int *)codeptr;
while (*p != GLUE_RET[0])
{
printf("instr:%04X:%04X\n",*p>>16,*p&0xffff);
p++;
}
}
#endif
#ifndef GLUE_TABPTR_IGNORED
tabptr=(INT_PTR)h->workTable;
#endif
//printf("calling code!\n");
GLUE_CALL_CODE(tabptr,codeptr,(INT_PTR)h->ramPtr);
}
int NSEEL_code_geterror_flag(NSEEL_VMCTX ctx)
{
compileContext *c=(compileContext *)ctx;
if (c) return (c->gotEndOfInput ? 1 : 0);
return 0;
}
char *NSEEL_code_getcodeerror(NSEEL_VMCTX ctx)
{
compileContext *c=(compileContext *)ctx;
if (ctx && c->last_error_string[0]) return c->last_error_string;
return 0;
}
//------------------------------------------------------------------------------
void NSEEL_code_free(NSEEL_CODEHANDLE code)
{
codeHandleType *h = (codeHandleType *)code;
if (h != NULL)
{
#ifdef EEL_VALIDATE_WORKTABLE_USE
if (h->workTable)
{
char *p = ((char*)h->workTable) + h->workTable_size*sizeof(EEL_F);
int x;
for(x=COMPUTABLE_EXTRA_SPACE*sizeof(EEL_F) - 1;x >= 0; x --)
if (p[x] != 0x3a)
{
char buf[512];
snprintf(buf,sizeof(buf),"worktable overrun at byte %d (wts=%d), value = %f\n",x,h->workTable_size, *(EEL_F*)(p+(x&~(sizeof(EEL_F)-1))));
#ifdef _WIN32
OutputDebugString(buf);
#else
printf("%s",buf);
#endif
break;
}
}
#endif
nseel_evallib_stats[0]-=h->code_stats[0];
nseel_evallib_stats[1]-=h->code_stats[1];
nseel_evallib_stats[2]-=h->code_stats[2];
nseel_evallib_stats[3]-=h->code_stats[3];
nseel_evallib_stats[4]--;
#if defined(__ppc__) && defined(__APPLE__)
{
FILE *fp = fopen("/var/db/receipts/com.apple.pkg.Rosetta.plist","r");
if (fp)
{
fclose(fp);
// on PPC, but rosetta installed, do not free h->blocks, as rosetta won't detect changes to these pages
}
else
{
freeBlocks(&h->blocks);
}
}
#else
freeBlocks(&h->blocks);
#endif
freeBlocks(&h->blocks_data);
}
}
//------------------------------------------------------------------------------
NSEEL_VMCTX NSEEL_VM_alloc() // return a handle
{
compileContext *ctx=calloc(1,sizeof(compileContext));
#ifdef NSEEL_SUPER_MINIMAL_LEXER
if (ctx) ctx->scanner = ctx;
#else
if (ctx)
{
int nseellex_init(void ** ptr_yy_globals);
void nseelset_extra(void *user_defined , void *yyscanner);
if (nseellex_init(&ctx->scanner))
{
free(ctx);
return NULL;
}
nseelset_extra(ctx,ctx->scanner);
}
#endif
if (ctx)
{
ctx->ram_state.maxblocks = NSEEL_RAM_BLOCKS_DEFAULTMAX;
ctx->ram_state.closefact = NSEEL_CLOSEFACTOR;
}
return ctx;
}
int NSEEL_VM_setramsize(NSEEL_VMCTX _ctx, int maxent)
{
compileContext *ctx = (compileContext *)_ctx;
if (!ctx) return 0;
if (maxent > 0)
{
maxent = (maxent + NSEEL_RAM_ITEMSPERBLOCK - 1)/NSEEL_RAM_ITEMSPERBLOCK;
if (maxent > NSEEL_RAM_BLOCKS) maxent = NSEEL_RAM_BLOCKS;
ctx->ram_state.maxblocks = maxent;
}
return ctx->ram_state.maxblocks * NSEEL_RAM_ITEMSPERBLOCK;
}
void NSEEL_VM_SetFunctionValidator(NSEEL_VMCTX _ctx, const char * (*validateFunc)(const char *fn_name, void *user), void *user)
{
if (_ctx)
{
compileContext *ctx = (compileContext *)_ctx;
ctx->func_check = validateFunc;
ctx->func_check_user = user;
}
}
void NSEEL_VM_SetFunctionTable(NSEEL_VMCTX _ctx, eel_function_table *tab)
{
if (_ctx)
{
compileContext *ctx = (compileContext *)_ctx;
ctx->registered_func_tab = tab;
}
}
void NSEEL_VM_free(NSEEL_VMCTX _ctx) // free when done with a VM and ALL of its code have been freed, as well
{
if (_ctx)
{
compileContext *ctx=(compileContext *)_ctx;
EEL_GROWBUF_RESIZE(&ctx->varNameList,-1);
NSEEL_VM_freeRAM(_ctx);
freeBlocks(&ctx->pblocks);
// these should be 0 normally but just in case
freeBlocks(&ctx->tmpblocks_head); // free blocks
freeBlocks(&ctx->blocks_head); // free blocks
freeBlocks(&ctx->blocks_head_data); // free blocks
#ifndef NSEEL_SUPER_MINIMAL_LEXER
if (ctx->scanner)
{
int nseellex_destroy(void *yyscanner);
nseellex_destroy(ctx->scanner);
}
#endif
ctx->scanner=0;
if (ctx->has_used_global_vars)
{
nseel_globalVarItem *p = NULL;
NSEEL_HOSTSTUB_EnterMutex();
if (--nseel_vms_referencing_globallist_cnt == 0)
{
// clear and free globals
p = nseel_globalreg_list;
nseel_globalreg_list=0;
}
NSEEL_HOSTSTUB_LeaveMutex();
while (p)
{
nseel_globalVarItem *op = p;
p=p->_next;
free(op);
}
}
free(ctx);
}
}
int *NSEEL_code_getstats(NSEEL_CODEHANDLE code)
{
codeHandleType *h = (codeHandleType *)code;
if (h)
{
return h->code_stats;
}
return 0;
}
void NSEEL_VM_SetStringFunc(NSEEL_VMCTX ctx,
EEL_F (*onString)(void *caller_this, struct eelStringSegmentRec *list),
EEL_F (*onNamedString)(void *caller_this, const char *name))
{
if (ctx)
{
compileContext *c=(compileContext*)ctx;
c->onString = onString;
c->onNamedString = onNamedString;
}
}
void NSEEL_VM_SetCustomFuncThis(NSEEL_VMCTX ctx, void *thisptr)
{
if (ctx)
{
compileContext *c=(compileContext*)ctx;
c->caller_this=thisptr;
}
}
void *NSEEL_PProc_RAM(void *data, int data_size, compileContext *ctx)
{
if (data_size>0) data=EEL_GLUE_set_immediate(data, (INT_PTR)ctx->ram_state.blocks);
return data;
}
void *NSEEL_PProc_THIS(void *data, int data_size, compileContext *ctx)
{
if (data_size>0) data=EEL_GLUE_set_immediate(data, (INT_PTR)ctx->caller_this);
return data;
}
static int vartable_lowerbound(compileContext *ctx, const char *name, int *ismatch)
{
int a = 0, c = EEL_GROWBUF_GET_SIZE(&ctx->varNameList);
varNameRec **list = EEL_GROWBUF_GET(&ctx->varNameList);
while (a != c)
{
const int b = (a+c)/2;
const int cmp = strnicmp(name,list[b]->str,NSEEL_MAX_VARIABLE_NAMELEN);
if (cmp > 0) a = b+1;
else if (cmp < 0) c = b;
else
{
*ismatch = 1;
return b;
}
}
*ismatch = 0;
return a;
}
static void vartable_cull_list(compileContext *ctx, int refcnt_chk)
{
const int ni = EEL_GROWBUF_GET_SIZE(&ctx->varNameList);
int i = ni, ndel = 0;
varNameRec **rd = EEL_GROWBUF_GET(&ctx->varNameList), **wr=rd;
while (i--)
{
varNameRec *v = rd[0];
if ((!refcnt_chk || !v->refcnt) && !v->isreg)
{
ndel++;
}
else
{
if (wr != rd) *wr = *rd;
wr++;
}
rd++;
}
if (ndel) EEL_GROWBUF_RESIZE(&ctx->varNameList,ni - ndel);
}
void NSEEL_VM_remove_unused_vars(NSEEL_VMCTX _ctx)
{
compileContext *ctx = (compileContext *)_ctx;
if (ctx) vartable_cull_list(ctx,1);
}
void NSEEL_VM_remove_all_nonreg_vars(NSEEL_VMCTX _ctx)
{
compileContext *ctx = (compileContext *)_ctx;
if (ctx) vartable_cull_list(ctx,0);
}
void NSEEL_VM_clear_var_refcnts(NSEEL_VMCTX _ctx)
{
compileContext *ctx = (compileContext *)_ctx;
if (ctx)
{
int i = EEL_GROWBUF_GET_SIZE(&ctx->varNameList);
varNameRec **rd = EEL_GROWBUF_GET(&ctx->varNameList);
while (i--)
{
rd[0]->refcnt=0;
rd++;
}
}
}
#ifdef NSEEL_EEL1_COMPAT_MODE
static EEL_F __nseel_global_regs[100];
double *NSEEL_getglobalregs() { return __nseel_global_regs; }
#endif
EEL_F *get_global_var(compileContext *ctx, const char *gv, int addIfNotPresent)
{
nseel_globalVarItem *p;
#ifdef NSEEL_EEL1_COMPAT_MODE
if (!strnicmp(gv,"reg",3) && gv[3]>='0' && gv[3] <= '9' && gv[4] >= '0' && gv[4] <= '9' && !gv[5])
{
return __nseel_global_regs + atoi(gv+3);
}
#endif
NSEEL_HOSTSTUB_EnterMutex();
if (!ctx->has_used_global_vars)
{
ctx->has_used_global_vars++;
nseel_vms_referencing_globallist_cnt++;
}
p = nseel_globalreg_list;
while (p)
{
if (!stricmp(p->name,gv)) break;
p=p->_next;
}
if (!p && addIfNotPresent)
{
size_t gvl = strlen(gv);
p = (nseel_globalVarItem*)malloc(sizeof(nseel_globalVarItem) + gvl);
if (p)
{
p->data=0.0;
strcpy(p->name,gv);
p->_next = nseel_globalreg_list;
nseel_globalreg_list=p;
}
}
NSEEL_HOSTSTUB_LeaveMutex();
return p ? &p->data : NULL;
}
EEL_F *nseel_int_register_var(compileContext *ctx, const char *name, int isReg, const char **namePtrOut)
{
int slot, match;
if (isReg == 0 && ctx->getVariable)
{
EEL_F *ret = ctx->getVariable(ctx->getVariable_userctx, name);
if (ret) return ret;
}
if (!strnicmp(name,"_global.",8) && name[8])
{
EEL_F *a=get_global_var(ctx,name+8,isReg >= 0);
if (a) return a;
}
slot = vartable_lowerbound(ctx,name, &match);
if (match)
{
varNameRec *v = EEL_GROWBUF_GET(&ctx->varNameList)[slot];
if (isReg >= 0)
{
v->refcnt++;
if (isReg) v->isreg=isReg;
if (namePtrOut) *namePtrOut = v->str;
}
return v->value;
}
if (isReg < 0) return NULL;
if (ctx->varValueStore_left<1)
{
const int sz=500;
ctx->varValueStore_left = sz;
ctx->varValueStore = (EEL_F *)newCtxDataBlock((int)sizeof(EEL_F)*sz,8);
}
if (ctx->varValueStore)
{
int listsz = EEL_GROWBUF_GET_SIZE(&ctx->varNameList);
size_t l = strlen(name);
varNameRec *vh;
if (l > NSEEL_MAX_VARIABLE_NAMELEN) l = NSEEL_MAX_VARIABLE_NAMELEN;
vh = (varNameRec*) newCtxDataBlock( (int) (sizeof(varNameRec) + l),8);
if (!vh || EEL_GROWBUF_RESIZE(&ctx->varNameList, (listsz+1))) return NULL; // alloc fail
(vh->value = ctx->varValueStore++)[0]=0.0;
ctx->varValueStore_left--;
vh->refcnt=1;
vh->isreg=isReg;
memcpy(vh->str,name,l);
vh->str[l] = 0;
if (namePtrOut) *namePtrOut = vh->str;
if (slot < listsz)
{
memmove(EEL_GROWBUF_GET(&ctx->varNameList) + slot+1,
EEL_GROWBUF_GET(&ctx->varNameList) + slot, (listsz - slot) * sizeof(EEL_GROWBUF_GET(&ctx->varNameList)[0]));
}
EEL_GROWBUF_GET(&ctx->varNameList)[slot] = vh;
return vh->value;
}
return NULL;
}
//------------------------------------------------------------------------------
void NSEEL_VM_enumallvars(NSEEL_VMCTX ctx, int (*func)(const char *name, EEL_F *val, void *ctx), void *userctx)
{
compileContext *tctx = (compileContext *) ctx;
int ni;
varNameRec **rd;
if (!tctx) return;
ni = EEL_GROWBUF_GET_SIZE(&tctx->varNameList);
rd = EEL_GROWBUF_GET(&tctx->varNameList);
while (ni--)
{
if (!func(rd[0]->str,rd[0]->value,userctx)) break;
rd++;
}
}
//------------------------------------------------------------------------------
EEL_F *NSEEL_VM_regvar(NSEEL_VMCTX _ctx, const char *var)
{
compileContext *ctx = (compileContext *)_ctx;
if (!ctx) return 0;
if (!strnicmp(var,"reg",3) && strlen(var) == 5 && isdigit(var[3]) && isdigit(var[4]))
{
EEL_F *a=get_global_var(ctx,var,1);
if (a) return a;
}
return nseel_int_register_var(ctx,var,1,NULL);
}
EEL_F *NSEEL_VM_getvar(NSEEL_VMCTX _ctx, const char *var)
{
compileContext *ctx = (compileContext *)_ctx;
if (!ctx) return 0;
if (!strnicmp(var,"reg",3) && strlen(var) == 5 && isdigit(var[3]) && isdigit(var[4]))
{
EEL_F *a=get_global_var(ctx,var,0);
if (a) return a;
}
return nseel_int_register_var(ctx,var,-1,NULL);
}
int NSEEL_VM_get_var_refcnt(NSEEL_VMCTX _ctx, const char *name)
{
compileContext *ctx = (compileContext *)_ctx;
int slot,match;
if (!ctx) return -1;
slot = vartable_lowerbound(ctx,name, &match);
return match ? EEL_GROWBUF_GET(&ctx->varNameList)[slot]->refcnt : -1;
}
opcodeRec *nseel_createFunctionByName(compileContext *ctx, const char *name, int np, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3)
{
int chkamt=0;
functionType *f=nseel_getFunctionByName(ctx,name,&chkamt);
if (f) while (chkamt-->=0)
{
if ((f->nParams&FUNCTIONTYPE_PARAMETERCOUNTMASK) == np)
{
opcodeRec *o=newOpCode(ctx,NULL, np==3?OPCODETYPE_FUNC3:np==2?OPCODETYPE_FUNC2:OPCODETYPE_FUNC1);
if (o)
{
o->fntype = FUNCTYPE_FUNCTIONTYPEREC;
o->fn = f;
o->parms.parms[0]=code1;
o->parms.parms[1]=code2;
o->parms.parms[2]=code3;
}
return o;
}
f++;
if (stricmp(f->name,name)) break;
}
return NULL;
}
//------------------------------------------------------------------------------
opcodeRec *nseel_translate(compileContext *ctx, const char *tmp, size_t tmplen) // tmplen 0 = null term
{
// this depends on the string being nul terminated eventually, tmplen is used more as a hint than anything else
if ((tmp[0] == '0' || tmp[0] == '$') && toupper(tmp[1])=='X')
{
char *p;
return nseel_createCompiledValue(ctx,(EEL_F)strtoul(tmp+2,&p,16));
}
else if (tmp[0] == '$')
{
if (tmp[1] == '~')
{
char *p=(char*)tmp+2;
unsigned int v=strtoul(tmp+2,&p,10);
if (v>53) v=53;
return nseel_createCompiledValue(ctx,(EEL_F)((((WDL_INT64)1) << v) - 1));
}
else if (!tmplen ? !stricmp(tmp,"$E") : (tmplen == 2 && !strnicmp(tmp,"$E",2)))
return nseel_createCompiledValue(ctx,(EEL_F)2.71828183);
else if (!tmplen ? !stricmp(tmp, "$PI") : (tmplen == 3 && !strnicmp(tmp, "$PI", 3)))
return nseel_createCompiledValue(ctx,(EEL_F)3.141592653589793);
else if (!tmplen ? !stricmp(tmp, "$PHI") : (tmplen == 4 && !strnicmp(tmp, "$PHI", 4)))
return nseel_createCompiledValue(ctx,(EEL_F)1.61803399);
else if ((!tmplen || tmplen == 4) && tmp[1] == '\'' && tmp[2] && tmp[3] == '\'')
return nseel_createCompiledValue(ctx,(EEL_F)tmp[2]);
else return NULL;
}
else if (tmp[0] == '\'')
{
char b[64];
int x,sz;
unsigned int rv=0;
if (!tmplen) // nul terminated tmplen, calculate a workable length
{
// faster than strlen(tmp) if tmp is large, we'll never need more than ~18 chars anyway
while (tmplen < 32 && tmp[tmplen]) tmplen++;
}
sz = tmplen > 0 ? nseel_filter_escaped_string(b,sizeof(b),tmp+1, tmplen - 1, '\'') : 0;
if (sz > 4)
{
if (ctx->last_error_string[0]) lstrcatn(ctx->last_error_string, ", ", sizeof(ctx->last_error_string));
snprintf_append(ctx->last_error_string,sizeof(ctx->last_error_string),"multi-byte character '%.5s...' too long",b);
return NULL; // do not allow 'xyzxy', limit to 4 bytes
}
for (x=0;x<sz;x++) rv = (rv<<8) + ((unsigned char*)b)[x];
return nseel_createCompiledValue(ctx,(EEL_F)rv);
}
else if (tmp[0] == '#')
{
char buf[2048];
if (!tmplen) while (tmplen < sizeof(buf)-1 && tmp[tmplen]) tmplen++;
else if (tmplen > sizeof(buf)-1) tmplen = sizeof(buf)-1;
memcpy(buf,tmp,tmplen);
buf[tmplen]=0;
if (ctx->onNamedString)
{
if (tmplen>0 && buf[1]&&ctx->function_curName)
{
int err=0;
opcodeRec *r = nseel_resolve_named_symbol(ctx,nseel_createCompiledValuePtr(ctx,NULL,buf),-1, &err);
if (r)
{
if (r->opcodeType!=OPCODETYPE_VALUE_FROM_NAMESPACENAME)
{
r->opcodeType = OPCODETYPE_DIRECTVALUE;
r->parms.dv.directValue = ctx->onNamedString(ctx->caller_this,buf+1);
r->parms.dv.valuePtr=NULL;
}
return r;
}
if (err) return NULL;
}
// if not namespaced symbol, return directly
if (!buf[1])
{
opcodeRec *r=newOpCode(ctx,NULL,OPCODETYPE_DIRECTVALUE_TEMPSTRING);
if (r) r->parms.dv.directValue = -10000.0;
return r;
}
return nseel_createCompiledValue(ctx,ctx->onNamedString(ctx->caller_this,buf+1));
}
}
return nseel_createCompiledValue(ctx,(EEL_F)atof(tmp));
}
void NSEEL_VM_set_var_resolver(NSEEL_VMCTX _ctx, EEL_F *(*res)(void *userctx, const char *name), void *userctx)
{
compileContext *ctx = (compileContext *)_ctx;
if (ctx)
{
ctx->getVariable = res;
ctx->getVariable_userctx = userctx;
}
}
#if defined(__ppc__) || defined(EEL_TARGET_PORTABLE)
// blank stubs
void eel_setfp_round() { }
void eel_setfp_trunc() { }
void eel_enterfp(int s[2]) {}
void eel_leavefp(int s[2]) {}
#endif