Files
epics-base/src/libCom/macLib/macCore.c
2001-07-26 07:59:07 +00:00

793 lines
21 KiB
C

/* $Id$
*
* Implementation of core macro substitution library (macLib)
*
* Just about all functionality proposed in tech-talk on 19-Feb-96 has
* been implemented with only minor departures. Missing features are:
*
* 1. All the attribute-setting routines (for setting and getting
* special characters etc)
*
* 2. Error message verbosity control (although a debug level is
* supported)
*
* 3. The macro substitution tool (although the supplied macTest
* program is in fact a simple version of this)
*
* The implementation is fairly unsophisticated and linked lists are
* used to store macro values. Typically there will will be only a
* small number of macros and performance won't be a problem. Special
* measures are taken to avoid unnecessary expansion of macros whose
* definitions reference other macros. Whenever a macro is created,
* modified or deleted, a "dirty" flag is set; this causes a full
* expansion of all macros the next time a macro value is read
*
* See the files README (original specification from tech-talk) and
* NOTES (valid input file for the macTest program) for further
* information
*
* Testing has been primarily under SunOS, but the code has been built
* and macTest <NOTES run under vxWorks
*
* William Lupton, W. M. Keck Observatory
*/
#include <ctype.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define epicsExportSharedSymbols
#include "dbmf.h"
#include "macLib.h"
/*
* Static function prototypes (these static functions provide an low-level
* library operating on macro entries)
*/
static MAC_ENTRY *first ( MAC_HANDLE *handle );
static MAC_ENTRY *last ( MAC_HANDLE *handle );
static MAC_ENTRY *next ( MAC_ENTRY *entry );
static MAC_ENTRY *previous( MAC_ENTRY *entry );
static MAC_ENTRY *create( MAC_HANDLE *handle, char *name, long special );
static MAC_ENTRY *lookup( MAC_HANDLE *handle, char *name, long special );
static char *rawval( MAC_HANDLE *handle, MAC_ENTRY *entry, char *value );
static void delete( MAC_HANDLE *handle, MAC_ENTRY *entry );
static long expand( MAC_HANDLE *handle );
static void trans ( MAC_HANDLE *handle, MAC_ENTRY *entry, long level,
char *term, char **rawval, char **value, char *valend );
static char *Strdup( char *string );
/*
* Static variables
*/
static MAC_CHARS macDefaultChars = {
MAC_STARTCHARS,
MAC_LEFTCHARS,
MAC_RIGHTCHARS,
MAC_SEPARCHARS,
MAC_ASSIGNCHARS,
MAC_ESCAPECHARS
};
/*
* Create a new macro substitution context and return an opaque handle
* associated with the new context. Also optionally load an initial set
* of macro definitions
*/
long /* 0 = OK; <0 = ERROR */
epicsShareAPI macCreateHandle(
MAC_HANDLE **pHandle, /* address of variable to receive pointer */
/* to new macro substitution context */
char *pairs[] ) /* pointer to NULL-terminated array of */
/* {name,value} pair strings; a NULL */
/* value implies undefined; a NULL */
/* argument implies no macros */
{
MAC_HANDLE *handle; /* pointer to macro substitution context */
/* ensure NULL handle pointer is returned on error */
*pHandle = NULL;
/* allocate macro substitution context */
handle = ( MAC_HANDLE * ) dbmfMalloc( sizeof( MAC_HANDLE ) );
if ( handle == NULL ) {
macErrMessage0( -1, "macCreateHandle: failed to allocate context" );
return -1;
}
/* initialize context */
handle->magic = MAC_MAGIC;
handle->chars = macDefaultChars;
handle->dirty = FALSE;
handle->level = 0;
handle->debug = 0;
handle->suppressWarning = 0;
ellInit( &handle->list );
/* if supplied, load macro definitions */
for ( ; pairs && pairs[0]; pairs += 2 ) {
if ( macPutValue( handle, pairs[0], pairs[1] ) < 0 ) {
dbmfFree( handle );
return -1;
}
}
/* set returned handle pointer */
*pHandle = handle;
return 0;
}
void epicsShareAPI macSuppressWarning(
MAC_HANDLE *handle, /* opaque handle */
int falseTrue /*0 means issue, 1 means suppress*/
)
{
handle->suppressWarning = falseTrue;
}
/*
* Expand a string that may contain macro references and return the
* expanded string
*
* This is a very basic and powerful routine. It's basically a wrapper
* around the the translation "engine" trans()
*/
long /* #chars copied, <0 if any macros are */
/* undefined */
epicsShareAPI macExpandString(
MAC_HANDLE *handle, /* opaque handle */
char *src, /* source string */
char *dest, /* destination string */
long maxlen ) /* maximum number of characters to copy */
/* to destination string */
{
MAC_ENTRY entry;
char *s, *d;
long length;
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macExpandString: NULL or invalid handle\n" );
return -1;
}
/* debug output */
if ( handle->debug & 1 )
printf( "macExpandString( %s, maxlen = %ld )\n", src, maxlen );
/* expand raw values if necessary */
if ( expand( handle ) < 0 )
macErrMessage0( -1, "macExpandString: failed to expand raw values" );
/* fill in necessary fields in fake macro entry struncture */
entry.name = "<string>";
entry.error = FALSE;
/* expand the string */
s = src;
d = dest;
*d = '\0';
trans( handle, &entry, 0, "", &s, &d, d + maxlen );
/* return +/- #chars copied depending on successful expansion */
length = d - dest;
length = ( entry.error ) ? -length : length;
/* debug output */
if ( handle->debug & 1 )
printf( "macExpandString() -> %ld\n", length );
return length;
}
/*
* Define the value of a macro. A NULL value deletes the macro if it
* already existed
*/
long /* length of value */
epicsShareAPI macPutValue(
MAC_HANDLE *handle, /* opaque handle */
char *name, /* macro name */
char *value ) /* macro value */
{
MAC_ENTRY *entry; /* pointer to this macro's entry structure */
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macPutValue: NULL or invalid handle\n" );
return -1;
}
if ( handle->debug & 1 )
epicsPrintf( "macPutValue( %s, %s )\n", name, value ? value : "NULL" );
/* handle NULL value case: if name was found, delete entry (may be
several entries at different scoping levels) */
if ( value == NULL ) {
while ( ( entry = lookup( handle, name, FALSE ) ) != NULL )
delete( handle, entry );
return 0;
}
/* look up macro name */
entry = lookup( handle, name, FALSE );
/* new entry must be created if macro doesn't exist or if it only
exists at a lower scoping level */
if ( entry == NULL || entry->level < handle->level ) {
entry = create( handle, name, FALSE );
if ( entry == NULL ) {
macErrMessage2( -1, "macPutValue: failed to create macro %s = %s",
name, value );
return -1;
}
}
/* copy raw value */
if ( rawval( handle, entry, value ) == NULL ) {
macErrMessage2( -1, "macPutValue: failed to copy macro %s = %s",
name, value ) ;
return -1;
}
/* return length of value */
return strlen( value );
}
/*
* Return the value of a macro
*/
long /* #chars copied (<0 if undefined) */
epicsShareAPI macGetValue(
MAC_HANDLE *handle, /* opaque handle */
char *name, /* macro name or reference */
char *value, /* string to receive macro value or name */
/* argument if macro is undefined */
long maxlen ) /* maximum number of characters to copy */
/* to value */
{
MAC_ENTRY *entry; /* pointer to this macro's entry structure */
long length; /* number of characters returned */
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macGetValue: NULL or invalid handle\n" );
return -1;
}
/* debug output */
if ( handle->debug & 1 )
epicsPrintf( "macGetValue( %s )\n", name );
/* look up macro name */
entry = lookup( handle, name, FALSE );
/* if maxlen <= 0 or VALUE == NULL just return -1 / 0 for undefined /
defined macro */
if ( maxlen <= 0 || value == NULL ) {
return ( entry == NULL ) ? -1 : 0;
}
/* if not found, copy name to value and return minus #chars copied */
if ( entry == NULL ) {
strncpy( value, name, maxlen );
return ( value[maxlen-1] == '\0' ) ? - (long) strlen( name ) : -maxlen;
}
/* expand raw values if necessary; if fail (can only fail because of
memory allocation failure), return same as if not found */
if ( expand( handle ) < 0 ) {
macErrMessage0( -1, "macGetValue: failed to expand raw values" );
strncpy( value, name, maxlen );
return ( value[maxlen-1] == '\0' ) ? - (long) strlen( name ) : -maxlen;
}
/* copy value and return +/- #chars copied depending on successful
expansion */
strncpy( value, entry->value, maxlen );
length = ( value[maxlen-1] == '\0' ) ? entry->length : maxlen;
return ( entry->error ) ? -length : length;
}
/*
* Free up all storage associated with and delete a macro substitution
* context
*/
long /* 0 = OK; <0 = ERROR */
epicsShareAPI macDeleteHandle(
MAC_HANDLE *handle ) /* opaque handle */
{
MAC_ENTRY *entry, *nextEntry;
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macDeleteHandle: NULL or invalid handle\n" );
return -1;
}
/* debug output */
if ( handle->debug & 1 )
printf( "macDeleteHandle()\n" );
/* delete all entries */
for ( entry = first( handle ); entry != NULL; entry = nextEntry ) {
nextEntry = next( entry );
delete( handle, entry );
}
/* clear magic field and free context structure */
handle->magic = 0;
dbmfFree( handle );
return 0;
}
/*
* Mark the start of a new scoping level
*/
long /* 0 = OK; <0 = ERROR */
epicsShareAPI macPushScope(
MAC_HANDLE *handle ) /* opaque handle */
{
MAC_ENTRY *entry;
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macPushScope: NULL or invalid handle\n" );
return -1;
}
/* debug output */
if ( handle->debug & 1 )
printf( "macPushScope()\n" );
/* increment scoping level */
handle->level++;
/* create new "special" entry of name "<scope>" */
entry = create( handle, "<scope>", TRUE );
if ( entry == NULL ) {
handle->level--;
macErrMessage0( -1, "macPushScope: failed to push scope" );
return -1;
}
return 0;
}
/*
* Pop all macros defined since the last call to macPushScope()
*/
long /* 0 = OK; <0 = ERROR */
epicsShareAPI macPopScope(
MAC_HANDLE *handle ) /* opaque handle */
{
MAC_ENTRY *entry, *nextEntry;
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macPopScope: NULL or invalid handle\n" );
return -1;
}
/* debug output */
if ( handle->debug & 1 )
printf( "macPopScope()\n" );
/* check scoping level isn't already zero */
if ( handle->level == 0 ) {
macErrMessage0( -1, "macPopScope: no scope to pop" );
return -1;
}
/* look up most recent scope entry */
entry = lookup( handle, "<scope>", TRUE );
if ( entry == NULL ) {
macErrMessage0( -1, "macPopScope: no scope to pop" );
return -1;
}
/* delete scope entry and all macros defined since it */
for ( ; entry != NULL; entry = nextEntry ) {
nextEntry = next( entry );
delete( handle, entry );
}
/* decrement scoping level */
handle->level--;
return 0;
}
/*
* Report macro details to standard output
*/
long /* 0 = OK; <0 = ERROR */
epicsShareAPI macReportMacros(
MAC_HANDLE *handle ) /* opaque handle */
{
char *format = "%-1s %-16s %-16s %s\n";
MAC_ENTRY *entry;
/* check handle */
if ( handle == NULL || handle->magic != MAC_MAGIC ) {
macErrMessage0( -1, "macReportMacros: NULL or invalid handle\n" );
return -1;
}
/* expand raw values if necessary; report but ignore failure */
if ( expand( handle ) < 0 )
macErrMessage0( -1, "macGetValue: failed to expand raw values" );
/* loop through macros, reporting names and values */
epicsPrintf( format, "e", "name", "rawval", "value" );
epicsPrintf( format, "-", "----", "------", "-----" );
for ( entry = first( handle ); entry != NULL; entry = next( entry ) ) {
/* differentiate between "special" (scope marker) and ordinary
entries */
if ( entry->special )
epicsPrintf( format, "s", "----", "------", "-----" );
else
epicsPrintf( format, entry->error ? "*" : " ", entry->name,
entry->rawval ? entry->rawval : "",
entry->value ? entry->value : "");
}
return 0;
}
/******************** beginning of static functions ********************/
/*
* Return pointer to first macro entry (could be preprocessor macro)
*/
static MAC_ENTRY *first( MAC_HANDLE *handle )
{
return ( MAC_ENTRY * ) ellFirst( &handle->list );
}
/*
* Return pointer to last macro entry (could be preprocessor macro)
*/
static MAC_ENTRY *last( MAC_HANDLE *handle )
{
return ( MAC_ENTRY * ) ellLast( &handle->list );
}
/*
* Return pointer to next macro entry (could be preprocessor macro)
*/
static MAC_ENTRY *next( MAC_ENTRY *entry )
{
return ( MAC_ENTRY * ) ellNext( ( ELLNODE * ) entry );
}
/*
* Return pointer to previous macro entry (could be preprocessor macro)
*/
static MAC_ENTRY *previous( MAC_ENTRY *entry )
{
return ( MAC_ENTRY * ) ellPrevious( ( ELLNODE * ) entry );
}
/*
* Create new macro entry (can assume it doesn't exist)
*/
static MAC_ENTRY *create( MAC_HANDLE *handle, char *name, long special )
{
ELLLIST *list = &handle->list;
MAC_ENTRY *entry = ( MAC_ENTRY * ) dbmfMalloc( sizeof( MAC_ENTRY ) );
if ( entry != NULL ) {
entry->name = Strdup( name );
if ( entry->name == NULL ) {
dbmfFree( entry );
entry = NULL;
}
else {
entry->rawval = NULL;
entry->value = NULL;
entry->length = 0;
entry->error = FALSE;
entry->visited = FALSE;
entry->special = special;
entry->level = handle->level;
ellAdd( list, ( ELLNODE * ) entry );
}
}
return entry;
}
/*
* Look up macro entry with matching "special" attribute by name
*/
static MAC_ENTRY *lookup( MAC_HANDLE *handle, char *name, long special )
{
MAC_ENTRY *entry;
/* search backwards so scoping works */
for ( entry = last( handle ); entry != NULL; entry = previous( entry ) ) {
if ( entry->special != special )
continue;
if ( strcmp( name, entry->name ) == 0 )
break;
}
return entry;
}
/*
* Copy raw value to macro entry
*/
static char *rawval( MAC_HANDLE *handle, MAC_ENTRY *entry, char *value )
{
if ( entry->rawval != NULL )
dbmfFree( entry->rawval );
entry->rawval = Strdup( value );
handle->dirty = TRUE;
return entry->rawval;
}
/*
* Delete a macro entry; requires re-expansion of macro values since this
* macro may be referenced by another one
*/
static void delete( MAC_HANDLE *handle, MAC_ENTRY *entry )
{
ELLLIST *list = &handle->list;
ellDelete( list, ( ELLNODE * ) entry );
dbmfFree( entry->name );
if ( entry->rawval != NULL )
dbmfFree( entry->rawval );
if ( entry->value != NULL )
free( entry->value );
dbmfFree( entry );
handle->dirty = TRUE;
}
/*
* Expand macro definitions (expensive but done very infrequently)
*/
static long expand( MAC_HANDLE *handle )
{
MAC_ENTRY *entry;
char *rawval;
char *value;
if ( !handle->dirty )
return 0;
for ( entry = first( handle ); entry != NULL; entry = next( entry ) ) {
if ( handle->debug & 2 )
epicsPrintf( "\nexpand %s = %s\n", entry->name,
entry->rawval ? entry->rawval : "" );
if ( entry->value == NULL ) {
if ( ( entry->value = malloc( MAC_SIZE + 1 ) ) == NULL ) {
return -1;
}
}
/* start at level 1 so quotes and escapes will be removed from
expanded value */
rawval = entry->rawval;
value = entry->value;
*value = '\0';
entry->error = FALSE;
trans( handle, entry, 1, "", &rawval, &value, entry->value + MAC_SIZE );
entry->length = value - entry->value;
entry->value[MAC_SIZE] = '\0';
}
handle->dirty = FALSE;
return 0;
}
/*
* Translate raw macro value (recursive). This is by far the most complicated
* of the macro routines and calls itself recursively both to translate any
* macros referenced in the name and to translate the resulting name
*
* For now, use default special characters
*/
static void trans( MAC_HANDLE *handle, MAC_ENTRY *entry, long level,
char *term, char **rawval, char **value, char *valend )
{
char quote;
char *r, *v, *n2, *r2;
char name2[MAC_SIZE + 1];
long discard;
long macRef;
char *macEnd;
MAC_ENTRY *entry2;
/* return immediately if raw value is NULL */
if ( *rawval == NULL ) return;
/* discard quotes and escapes if level is > 0 (i.e. if these aren't
the user's quotes and escapes) */
discard = ( level > 0 );
/* debug output */
if ( handle->debug & 2 )
epicsPrintf( "trans-> level = %ld, maxlen = %4d, discard = %s, "
"rawval = %s\n", level, valend - *value, discard ? "T" : "F", *rawval );
/* initially not in quotes */
quote = 0;
/* scan characters until hit terminator or end of string */
for ( r = *rawval, v = *value; strchr( term, *r ) == NULL; r++ ) {
/* handle quoted characters (quotes are discarded if in name) */
if ( quote ) {
if ( *r == quote ) {
quote = 0;
if ( discard ) continue;
}
}
else if ( *r == '"' || *r == '\'' ) {
quote = *r;
if ( discard ) continue;
}
/* macro reference if '$' followed by '(' or '{' */
macRef = ( *r == '$' && *( r + 1 ) != '\0' &&
strchr( "({", *( r + 1 ) ) != NULL );
/* if not macro reference (macros are not expanded in single quotes) */
if ( quote == '\'' || !macRef ) {
/* handle escaped characters (escape is discarded if in name) */
if ( *r == '\\' && *( r + 1 ) != '\0' ) {
if ( v < valend && !discard ) *v++ = '\\';
if ( v < valend ) *v++ = *++r;
}
/* copy character to output */
else {
if ( v < valend ) *v++ = *r;
}
/* ensure string remains properly terminated */
if ( v < valend ) *v = '\0';
}
/* macro reference */
else {
/* step over '$(' or '${' */
r++;
macEnd = ( *r == '(' ) ? ")" : "}";
r++;
/* translate name (may contain macro references); truncated
quietly if too long but always guaranteed zero-terminated */
n2 = name2;
*n2 = '\0';
trans( handle, entry, level + 1, macEnd, &r, &n2, n2 + MAC_SIZE );
name2[MAC_SIZE] = '\0';
/* look up resultant name; if not there, set error flag and
copy reference */
if ( ( entry2 = lookup( handle, name2, FALSE ) ) == NULL ) {
if(!handle->suppressWarning) {
entry->error = TRUE;
macErrMessage2( -1, "%s: %s referenced but undefined",
entry->name, name2 );
}
sprintf( v, "$(%s)", name2 ); v += strlen( v );
}
/* check for recursive reference; if so, set error flag and
copy reference */
else if ( entry2->visited ) {
entry->error = TRUE;
macErrMessage2( -1, "%s: %s referenced recursively",
entry->name, entry2->name );
sprintf( v, "$(%s)", name2 ); v += strlen( v );
}
/* if all expanded values are valid (not "dirty") copy the
value directly */
else if ( !handle->dirty ) {
strcpy( v, entry2->value ); v += strlen( v );
}
/* otherwise, translate raw value */
else {
r2 = entry2->rawval;
entry2->visited = TRUE;
trans( handle, entry, level + 1, "", &r2, &v, valend );
entry2->visited = FALSE;
}
}
}
/* debug output */
if ( handle->debug & 2 )
epicsPrintf( "<-trans level = %ld, length = %4d, value = %s\n",
level, v - *value, *value );
/* update pointers to next characters to scan in raw value and to fill
in in output value (if at end of input, step back so terminator is
still there to be seen) */
*rawval = ( *r == '\0' ) ? r - 1 : r;
*value = v;
return;
}
/*
* strdup() implementation (because it's not always available)
*/
static char *Strdup( char *string )
{
char *copy = dbmfMalloc( strlen( string ) + 1 );
if ( copy != NULL )
strcpy( copy, string );
return copy;
}
/*************************************************************************
* $Log$
* Revision 1.9 1999/05/13 16:06:21 anj
* Cleaned up warnings - the RCS Log: keyword was on the comment line
* that starts with a '/' which RCS replicates for these notes...
*
* Revision 1.8 1998/02/27 01:34:13 jhill
* cleaned up the DLL symbol export
*
* Revision 1.7 1998/02/20 21:45:16 evans
* Made a large number of changes to epicsShareThings in libCom routines
* to get imports and exports straight on WIN32. Not everything is fixed
* at this time.
*
* Revision 1.6 1997/06/05 18:28:48 mrk
* Allow message about referenced but undefined to be suppressed
*
* Revision 1.5 1997/05/01 19:57:32 jhill
* updated dll keywords
*
* Revision 1.4 1997/04/30 19:10:04 mrk
* Fix long memory free list problem
*
* Revision 1.3 1996/11/02 02:05:24 jhill
* %d => %ld
*
* Revision 1.2 1996/09/16 21:07:10 jhill
* fixed warnings
*
* Revision 1.1 1996/07/10 14:49:48 mrk
* added macLib
*
* Revision 1.9 1996/06/26 09:43:14 wlupton
* first released version
*
*/