mirror of
https://github.com/rsyslog/rsyslog.git
synced 2025-12-21 02:00:42 +01:00
233 lines
5.7 KiB
C
233 lines
5.7 KiB
C
/* vmstk.c - the arithmetic stack of a virtual machine.
|
|
*
|
|
* Module begun 2008-02-21 by Rainer Gerhards
|
|
*
|
|
* Copyright 2008-2012 Rainer Gerhards and Adiscon GmbH.
|
|
*
|
|
* This file is part of the rsyslog runtime library.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
* -or-
|
|
* see COPYING.ASL20 in the source distribution
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
|
|
#include "rsyslog.h"
|
|
#include "obj.h"
|
|
#include "vmstk.h"
|
|
|
|
/* static data */
|
|
DEFobjStaticHelpers
|
|
DEFobjCurrIf(var)
|
|
|
|
|
|
/* Standard-Constructor
|
|
*/
|
|
BEGINobjConstruct(vmstk) /* be sure to specify the object type also in END macro! */
|
|
ENDobjConstruct(vmstk)
|
|
|
|
|
|
/* ConstructionFinalizer
|
|
* rgerhards, 2008-01-09
|
|
*/
|
|
static rsRetVal
|
|
vmstkConstructFinalize(vmstk_t __attribute__((unused)) *pThis)
|
|
{
|
|
DEFiRet;
|
|
ISOBJ_TYPE_assert(pThis, vmstk);
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* destructor for the vmstk object */
|
|
BEGINobjDestruct(vmstk) /* be sure to specify the object type also in END and CODESTART macros! */
|
|
CODESTARTobjDestruct(vmstk)
|
|
ENDobjDestruct(vmstk)
|
|
|
|
|
|
/* debugprint for the vmstk object */
|
|
BEGINobjDebugPrint(vmstk) /* be sure to specify the object type also in END and CODESTART macros! */
|
|
CODESTARTobjDebugPrint(vmstk)
|
|
dbgoprint((obj_t*) pThis, "stack contents:\n");
|
|
ENDobjDebugPrint(vmstk)
|
|
|
|
|
|
/* push a value on the stack. The provided pVar is now owned
|
|
* by the stack. If the user intends to continue use it, it
|
|
* must be duplicated.
|
|
*/
|
|
static rsRetVal
|
|
push(vmstk_t *pThis, var_t *pVar)
|
|
{
|
|
DEFiRet;
|
|
|
|
ISOBJ_TYPE_assert(pThis, vmstk);
|
|
ISOBJ_TYPE_assert(pVar, var);
|
|
|
|
if(pThis->iStkPtr >= VMSTK_SIZE)
|
|
ABORT_FINALIZE(RS_RET_OUT_OF_STACKSPACE);
|
|
|
|
pThis->vStk[pThis->iStkPtr++] = pVar;
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* pop a value from the stack
|
|
* IMPORTANT: the stack pointer always points to the NEXT FREE entry. So in
|
|
* order to pop, we must access the element one below the stack pointer.
|
|
* The user is responsible for destructing the ppVar returned.
|
|
*/
|
|
static rsRetVal
|
|
pop(vmstk_t *pThis, var_t **ppVar)
|
|
{
|
|
DEFiRet;
|
|
|
|
ISOBJ_TYPE_assert(pThis, vmstk);
|
|
ASSERT(ppVar != NULL);
|
|
|
|
if(pThis->iStkPtr == 0)
|
|
ABORT_FINALIZE(RS_RET_STACK_EMPTY);
|
|
|
|
*ppVar = pThis->vStk[--pThis->iStkPtr];
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* pop a boolean value from the stack
|
|
* The user is responsible for destructing the ppVar returned.
|
|
*/
|
|
static rsRetVal
|
|
popBool(vmstk_t *pThis, var_t **ppVar)
|
|
{
|
|
DEFiRet;
|
|
|
|
/* assertions are done in pop(), we do not duplicate here */
|
|
CHKiRet(pop(pThis, ppVar));
|
|
CHKiRet(var.ConvToBool(*ppVar));
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* pop a number value from the stack
|
|
* The user is responsible for destructing the ppVar returned.
|
|
*/
|
|
static rsRetVal
|
|
popNumber(vmstk_t *pThis, var_t **ppVar)
|
|
{
|
|
DEFiRet;
|
|
|
|
/* assertions are done in pop(), we do not duplicate here */
|
|
CHKiRet(pop(pThis, ppVar));
|
|
CHKiRet(var.ConvToNumber(*ppVar));
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* pop a number value from the stack
|
|
* The user is responsible for destructing the ppVar returned.
|
|
*/
|
|
static rsRetVal
|
|
popString(vmstk_t *pThis, var_t **ppVar)
|
|
{
|
|
DEFiRet;
|
|
|
|
/* assertions are done in pop(), we do not duplicate here */
|
|
CHKiRet(pop(pThis, ppVar));
|
|
CHKiRet(var.ConvToString(*ppVar));
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* pop two variables for a common operation, e.g. a compare. When this
|
|
* functions returns, both variables have the same type, but the type
|
|
* is not set to anything specific.
|
|
* The user is responsible for destructing the ppVar's returned.
|
|
* A quick note on the name: it means pop 2 variable for a common
|
|
* opertion - just in case you wonder (I don't really like the name,
|
|
* but I didn't come up with a better one...).
|
|
* rgerhards, 2008-02-25
|
|
*/
|
|
static rsRetVal
|
|
pop2CommOp(vmstk_t *pThis, var_t **ppVar1, var_t **ppVar2)
|
|
{
|
|
DEFiRet;
|
|
|
|
/* assertions are done in pop(), we do not duplicate here */
|
|
/* operand two must be popped first, because it is at the top of stack */
|
|
CHKiRet(pop(pThis, ppVar2));
|
|
CHKiRet(pop(pThis, ppVar1));
|
|
CHKiRet(var.ConvForOperation(*ppVar1, *ppVar2));
|
|
|
|
finalize_it:
|
|
RETiRet;
|
|
}
|
|
|
|
|
|
/* queryInterface function
|
|
* rgerhards, 2008-02-21
|
|
*/
|
|
BEGINobjQueryInterface(vmstk)
|
|
CODESTARTobjQueryInterface(vmstk)
|
|
if(pIf->ifVersion != vmstkCURR_IF_VERSION) { /* check for current version, increment on each change */
|
|
ABORT_FINALIZE(RS_RET_INTERFACE_NOT_SUPPORTED);
|
|
}
|
|
|
|
/* ok, we have the right interface, so let's fill it
|
|
* Please note that we may also do some backwards-compatibility
|
|
* work here (if we can support an older interface version - that,
|
|
* of course, also affects the "if" above).
|
|
*/
|
|
pIf->Construct = vmstkConstruct;
|
|
pIf->ConstructFinalize = vmstkConstructFinalize;
|
|
pIf->Destruct = vmstkDestruct;
|
|
pIf->DebugPrint = vmstkDebugPrint;
|
|
pIf->Push = push;
|
|
pIf->Pop = pop;
|
|
pIf->PopBool = popBool;
|
|
pIf->PopNumber = popNumber;
|
|
pIf->PopString = popString;
|
|
pIf->Pop2CommOp = pop2CommOp;
|
|
|
|
finalize_it:
|
|
ENDobjQueryInterface(vmstk)
|
|
|
|
|
|
/* Initialize the vmstk class. Must be called as the very first method
|
|
* before anything else is called inside this class.
|
|
* rgerhards, 2008-02-19
|
|
*/
|
|
BEGINObjClassInit(vmstk, 1, OBJ_IS_CORE_MODULE) /* class, version */
|
|
/* request objects we use */
|
|
CHKiRet(objUse(var, CORE_COMPONENT));
|
|
|
|
/* set our own handlers */
|
|
OBJSetMethodHandler(objMethod_DEBUGPRINT, vmstkDebugPrint);
|
|
OBJSetMethodHandler(objMethod_CONSTRUCTION_FINALIZER, vmstkConstructFinalize);
|
|
ENDObjClassInit(vmstk)
|
|
|
|
/* vi:set ai:
|
|
*/
|