332 lines
11 KiB
C
332 lines
11 KiB
C
/*
|
|
Nullsoft Expression Evaluator Library (NS-EEL)
|
|
Copyright (C) 1999-2003 Nullsoft, Inc.
|
|
|
|
ns-eel-int.h: internal code definition header.
|
|
|
|
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.
|
|
*/
|
|
|
|
#ifndef __NS_EELINT_H__
|
|
#define __NS_EELINT_H__
|
|
|
|
#ifdef _WIN32
|
|
#include <arch.h>
|
|
#else
|
|
#include "../wdltypes.h"
|
|
#endif
|
|
|
|
#include "ns-eel.h"
|
|
#include "ns-eel-addfuncs.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
|
|
enum {
|
|
|
|
// these ignore fn in opcodes, just use fntype to determine function
|
|
FN_MULTIPLY=0,
|
|
FN_DIVIDE,
|
|
FN_JOIN_STATEMENTS,
|
|
FN_DENORMAL_LIKELY,
|
|
FN_DENORMAL_UNLIKELY,
|
|
FN_ADD,
|
|
FN_SUB,
|
|
FN_AND,
|
|
FN_OR,
|
|
FN_UMINUS,
|
|
FN_NOT,
|
|
FN_NOTNOT,
|
|
FN_XOR,
|
|
FN_SHL,
|
|
FN_SHR,
|
|
FN_MOD,
|
|
FN_POW,
|
|
FN_LT,
|
|
FN_GT,
|
|
FN_LTE,
|
|
FN_GTE,
|
|
FN_EQ,
|
|
FN_EQ_EXACT,
|
|
FN_NE,
|
|
FN_NE_EXACT,
|
|
FN_LOGICAL_AND,
|
|
FN_LOGICAL_OR,
|
|
FN_IF_ELSE,
|
|
FN_MEMORY,
|
|
FN_GMEMORY,
|
|
FN_NONCONST_BEGIN,
|
|
FN_ASSIGN=FN_NONCONST_BEGIN,
|
|
|
|
FN_ADD_OP,
|
|
FN_SUB_OP,
|
|
FN_MOD_OP,
|
|
FN_OR_OP,
|
|
FN_AND_OP,
|
|
FN_XOR_OP,
|
|
FN_DIV_OP,
|
|
FN_MUL_OP,
|
|
FN_POW_OP,
|
|
|
|
FN_WHILE,
|
|
FN_LOOP,
|
|
|
|
FUNCTYPE_SIMPLEMAX,
|
|
|
|
|
|
FUNCTYPE_FUNCTIONTYPEREC=1000, // fn is a functionType *
|
|
FUNCTYPE_EELFUNC, // fn is a _codeHandleFunctionRec *
|
|
};
|
|
|
|
|
|
|
|
#define YYSTYPE opcodeRec *
|
|
|
|
#define NSEEL_CLOSEFACTOR 0.00001
|
|
|
|
|
|
typedef struct opcodeRec opcodeRec;
|
|
|
|
typedef struct _codeHandleFunctionRec
|
|
{
|
|
struct _codeHandleFunctionRec *next; // main linked list (only used for high level functions)
|
|
struct _codeHandleFunctionRec *derivedCopies; // separate linked list, _head being the main function, other copies being derived versions
|
|
|
|
void *startptr; // compiled code (may be cleared + recompiled when shared)
|
|
opcodeRec *opcodes;
|
|
|
|
int startptr_size; // 0=no code. -1 = needs calculation. >0 = size.
|
|
int tmpspace_req;
|
|
|
|
int num_params;
|
|
|
|
int rvMode; // RETURNVALUE_*
|
|
int fpStackUsage; // 0-8, usually
|
|
int canHaveDenormalOutput;
|
|
|
|
// local storage's first items are the parameters, then locals. Note that the opcodes will reference localstorage[] via VARPTRPTR, but
|
|
// the values localstorage[x] points are reallocated from context-to-context, if it is a common function.
|
|
|
|
// separately allocated list of pointers, the contents of the list should be zeroed on context changes if a common function
|
|
// note that when making variations on a function (context), it is shared, but since it is zeroed on context changes, it is context-local
|
|
int localstorage_size;
|
|
EEL_F **localstorage;
|
|
|
|
int isCommonFunction;
|
|
int usesNamespaces;
|
|
unsigned int parameterAsNamespaceMask;
|
|
|
|
char fname[NSEEL_MAX_FUNCSIG_NAME+1];
|
|
} _codeHandleFunctionRec;
|
|
|
|
#define LLB_DSIZE (65536-64)
|
|
typedef struct _llBlock {
|
|
struct _llBlock *next;
|
|
int sizeused;
|
|
char block[LLB_DSIZE];
|
|
} llBlock;
|
|
|
|
typedef struct {
|
|
llBlock *blocks,
|
|
*blocks_data;
|
|
void *workTable; // references a chunk in blocks_data
|
|
|
|
void *code;
|
|
int code_size; // in case the caller wants to write it out
|
|
int code_stats[4];
|
|
|
|
int want_stack;
|
|
void *stack; // references a chunk in blocks_data, somewhere within the complete NSEEL_STACK_SIZE aligned at NSEEL_STACK_SIZE
|
|
|
|
void *ramPtr;
|
|
|
|
int workTable_size; // size (minus padding/extra space) of workTable -- only used if EEL_VALIDATE_WORKTABLE_USE set, but might be handy to have around too
|
|
int compile_flags;
|
|
} codeHandleType;
|
|
|
|
typedef struct
|
|
{
|
|
EEL_F *value;
|
|
int refcnt;
|
|
char isreg;
|
|
char str[1];
|
|
} varNameRec;
|
|
|
|
typedef struct
|
|
{
|
|
void *ptr;
|
|
int size, alloc;
|
|
} eel_growbuf;
|
|
#define EEL_GROWBUF(type) union { eel_growbuf _growbuf; type *_tval; }
|
|
#define EEL_GROWBUF_RESIZE(gb, newsz) __growbuf_resize(&(gb)->_growbuf, (newsz)*(int)sizeof((gb)->_tval[0])) // <0 to free, does not realloc down otherwise
|
|
#define EEL_GROWBUF_GET(gb) ((gb)->_tval)
|
|
#define EEL_GROWBUF_GET_SIZE(gb) ((gb)->_growbuf.size/(int)sizeof((gb)->_tval[0]))
|
|
|
|
typedef struct _compileContext
|
|
{
|
|
eel_function_table *registered_func_tab;
|
|
const char *(*func_check)(const char *fn_name, void *user); // return error message if not permitted
|
|
void *func_check_user;
|
|
|
|
EEL_GROWBUF(varNameRec *) varNameList;
|
|
EEL_F *varValueStore;
|
|
int varValueStore_left;
|
|
|
|
int errVar,gotEndOfInput;
|
|
opcodeRec *result;
|
|
char last_error_string[256];
|
|
|
|
void *scanner;
|
|
const char *rdbuf_start, *rdbuf, *rdbuf_end;
|
|
|
|
llBlock *tmpblocks_head, // used while compiling, and freed after compiling
|
|
|
|
*blocks_head, // used while compiling, transferred to code context (these are pages marked as executable)
|
|
*blocks_head_data, // used while compiling, transferred to code context
|
|
|
|
*pblocks; // persistent blocks, stores data used by varTable_Names, varTable_Values, etc.
|
|
|
|
int l_stats[4]; // source bytes, static code bytes, call code bytes, data bytes
|
|
int has_used_global_vars;
|
|
|
|
_codeHandleFunctionRec *functions_local, *functions_common;
|
|
|
|
// state used while generating functions
|
|
int optimizeDisableFlags;
|
|
int current_compile_flags;
|
|
struct opcodeRec *directValueCache; // linked list using fn as next
|
|
|
|
int isSharedFunctions;
|
|
int isGeneratingCommonFunction;
|
|
int function_usesNamespaces;
|
|
int function_globalFlag; // set if restrict globals to function_localTable_Names[2]
|
|
// [0] is parameter+local symbols (combined space)
|
|
// [1] is symbols which get implied "this." if used
|
|
// [2] is globals permitted
|
|
int function_localTable_Size[3]; // for parameters only
|
|
char **function_localTable_Names[3]; // lists of pointers
|
|
EEL_F **function_localTable_ValuePtrs;
|
|
const char *function_curName; // name of current function
|
|
|
|
EEL_F (*onString)(void *caller_this, struct eelStringSegmentRec *list);
|
|
EEL_F (*onNamedString)(void *caller_this, const char *name);
|
|
|
|
EEL_F *(*getVariable)(void *userctx, const char *name);
|
|
void *getVariable_userctx;
|
|
|
|
codeHandleType *tmpCodeHandle;
|
|
|
|
struct
|
|
{
|
|
int needfree;
|
|
int maxblocks;
|
|
double closefact;
|
|
EEL_F *blocks[NSEEL_RAM_BLOCKS];
|
|
} ram_state
|
|
#ifdef __GNUC__
|
|
__attribute__ ((aligned (8)))
|
|
#endif
|
|
;
|
|
|
|
void *gram_blocks;
|
|
|
|
void *caller_this;
|
|
}
|
|
compileContext;
|
|
|
|
#define NSEEL_NPARAMS_FLAG_CONST 0x80000
|
|
typedef struct functionType {
|
|
const char *name;
|
|
void *afunc;
|
|
void *func_e;
|
|
int nParams;
|
|
void *replptrs[4];
|
|
NSEEL_PPPROC pProc;
|
|
} functionType;
|
|
|
|
functionType *nseel_getFunctionByName(compileContext *ctx, const char *name, int *mchk); // sets mchk (if non-NULL) to how far allowed to scan forward for duplicate names
|
|
|
|
opcodeRec *nseel_createCompiledValue(compileContext *ctx, EEL_F value);
|
|
opcodeRec *nseel_createCompiledValuePtr(compileContext *ctx, EEL_F *addrValue, const char *namestr);
|
|
|
|
opcodeRec *nseel_createMoreParametersOpcode(compileContext *ctx, opcodeRec *code1, opcodeRec *code2);
|
|
opcodeRec *nseel_createSimpleCompiledFunction(compileContext *ctx, int fn, int np, opcodeRec *code1, opcodeRec *code2);
|
|
opcodeRec *nseel_createMemoryAccess(compileContext *ctx, opcodeRec *code1, opcodeRec *code2);
|
|
opcodeRec *nseel_createIfElse(compileContext *ctx, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3);
|
|
opcodeRec *nseel_createFunctionByName(compileContext *ctx, const char *name, int np, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3);
|
|
|
|
// converts a generic identifier (VARPTR) opcode into either an actual variable reference (parmcnt = -1),
|
|
// or if parmcnt >= 0, to a function call (see nseel_setCompiledFunctionCallParameters())
|
|
opcodeRec *nseel_resolve_named_symbol(compileContext *ctx, opcodeRec *rec, int parmcnt, int *errOut);
|
|
|
|
// sets parameters and calculates parameter count for opcode, and calls nseel_resolve_named_symbol() with the right
|
|
// parameter count
|
|
opcodeRec *nseel_setCompiledFunctionCallParameters(compileContext *ctx, opcodeRec *fn, opcodeRec *code1, opcodeRec *code2, opcodeRec *code3, opcodeRec *postCode, int *errOut);
|
|
// errOut will be set if return NULL:
|
|
// -1 if postCode set when not wanted (i.e. not while())
|
|
// 0 if func not found,
|
|
// 1 if function requires 2+ parameters but was given more
|
|
// 2 if function needs more parameters
|
|
// 4 if function requires 1 parameter but was given more
|
|
|
|
|
|
|
|
struct eelStringSegmentRec *nseel_createStringSegmentRec(compileContext *ctx, const char *str, int len);
|
|
opcodeRec *nseel_eelMakeOpcodeFromStringSegments(compileContext *ctx, struct eelStringSegmentRec *rec);
|
|
|
|
EEL_F *nseel_int_register_var(compileContext *ctx, const char *name, int isReg, const char **namePtrOut);
|
|
_codeHandleFunctionRec *eel_createFunctionNamespacedInstance(compileContext *ctx, _codeHandleFunctionRec *fr, const char *nameptr);
|
|
|
|
typedef struct nseel_globalVarItem
|
|
{
|
|
EEL_F data;
|
|
struct nseel_globalVarItem *_next;
|
|
char name[1]; // varlen, does not include _global. prefix
|
|
} nseel_globalVarItem;
|
|
|
|
extern nseel_globalVarItem *nseel_globalreg_list; // if NSEEL_EEL1_COMPAT_MODE, must use NSEEL_getglobalregs() for regxx values
|
|
|
|
#include "y.tab.h"
|
|
|
|
// nseel_simple_tokenizer will return comments as tokens if state is non-NULL
|
|
const char *nseel_simple_tokenizer(const char **ptr, const char *endptr, int *lenOut, int *state);
|
|
int nseel_filter_escaped_string(char *outbuf, int outbuf_sz, const char *rdptr, size_t rdptr_size, char delim_char); // returns length used, minus NUL char
|
|
|
|
opcodeRec *nseel_translate(compileContext *ctx, const char *tmp, size_t tmplen); // tmplen=0 for nul-term
|
|
int nseel_lookup(compileContext *ctx, opcodeRec **opOut, const char *sname);
|
|
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAMAlloc(EEL_F **blocks, unsigned int w);
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAMAllocGMEM(EEL_F ***blocks, unsigned int w);
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAM_MemSet(EEL_F **blocks,EEL_F *dest, EEL_F *v, EEL_F *lenptr);
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAM_MemFree(void *blocks, EEL_F *which);
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAM_MemTop(void *blocks, EEL_F *which);
|
|
EEL_F * NSEEL_CGEN_CALL __NSEEL_RAM_MemCpy(EEL_F **blocks,EEL_F *dest, EEL_F *src, EEL_F *lenptr);
|
|
EEL_F NSEEL_CGEN_CALL __NSEEL_RAM_Mem_SetValues(EEL_F **blocks, INT_PTR np, EEL_F **parms);
|
|
EEL_F NSEEL_CGEN_CALL __NSEEL_RAM_Mem_GetValues(EEL_F **blocks, INT_PTR np, EEL_F **parms);
|
|
|
|
extern EEL_F nseel_ramalloc_onfail; // address returned by __NSEEL_RAMAlloc et al on failure
|
|
extern EEL_F * volatile nseel_gmembuf_default; // can free/zero this on DLL unload if needed
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
|
|
#endif//__NS_EELINT_H__
|