1254 lines
23 KiB
C++
1254 lines
23 KiB
C++
/*
|
|
* This module contains the definitions for the inline functions used by the
|
|
* name undecorator. It is intended that this file should be included
|
|
* somewhere in the source file for the undecorator to maximise the chance
|
|
* that they will be truly inlined.
|
|
*/
|
|
|
|
// The following class is a special node class, used in the implementation
|
|
// of the internal chaining mechanism of the 'DName's
|
|
|
|
class charNode;
|
|
class pcharNode;
|
|
class pDNameNode;
|
|
class DNameStatusNode;
|
|
|
|
|
|
#if ( NO_VIRTUAL )
|
|
enum NodeType
|
|
{
|
|
charNode_t,
|
|
pcharNode_t,
|
|
pDNameNode_t,
|
|
DNameStatusNode_t
|
|
|
|
};
|
|
#endif // NO_VIRTUAL
|
|
|
|
|
|
class DNameNode
|
|
{
|
|
private:
|
|
|
|
#if NO_VIRTUAL
|
|
NodeType typeIndex;
|
|
#endif // NO_VIRTUAL
|
|
|
|
DNameNode * next;
|
|
|
|
protected:
|
|
|
|
#if ( !NO_VIRTUAL )
|
|
DNameNode ();
|
|
#else // } elif NO_VIRTUAL {
|
|
DNameNode ( NodeType );
|
|
#endif // NO_VIRTUAL
|
|
|
|
DNameNode ( const DNameNode & );
|
|
|
|
public:
|
|
|
|
virtual int length () const PURE;
|
|
virtual char getLastChar () const PURE;
|
|
virtual pchar_t getString ( pchar_t, int ) const PURE;
|
|
DNameNode * clone ();
|
|
DNameNode * nextNode () const;
|
|
|
|
DNameNode & operator += ( DNameNode * );
|
|
|
|
};
|
|
|
|
|
|
class charNode : public DNameNode
|
|
{
|
|
private:
|
|
char me;
|
|
|
|
public:
|
|
charNode ( char );
|
|
|
|
virtual int length () const;
|
|
virtual char getLastChar () const;
|
|
virtual pchar_t getString ( pchar_t, int ) const;
|
|
|
|
};
|
|
|
|
|
|
class pcharNode : public DNameNode
|
|
{
|
|
private:
|
|
pchar_t me;
|
|
int myLen;
|
|
|
|
public:
|
|
pcharNode ( pcchar_t, int = 0 );
|
|
|
|
virtual int length () const;
|
|
virtual char getLastChar () const;
|
|
virtual pchar_t getString ( pchar_t, int ) const;
|
|
|
|
};
|
|
|
|
|
|
class pDNameNode : public DNameNode
|
|
{
|
|
private:
|
|
DName * me;
|
|
|
|
public:
|
|
pDNameNode ( DName * );
|
|
|
|
virtual int length () const;
|
|
virtual char getLastChar () const;
|
|
virtual pchar_t getString ( pchar_t, int ) const;
|
|
|
|
};
|
|
|
|
|
|
class DNameStatusNode : public DNameNode
|
|
{
|
|
private:
|
|
#define TruncationMessage (" ?? ")
|
|
#define TruncationMessageLength (4)
|
|
|
|
DNameStatus me;
|
|
int myLen;
|
|
|
|
public:
|
|
DNameStatusNode ( DNameStatus );
|
|
|
|
virtual int length () const;
|
|
virtual char getLastChar () const;
|
|
virtual pchar_t getString ( pchar_t, int ) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Memory allocation functions
|
|
|
|
inline void * operator new ( size_t sz, HeapManager &, int noBuffer )
|
|
{ return heap.getMemory ( sz, noBuffer ); }
|
|
|
|
void * HeapManager::getMemory ( size_t sz, int noBuffer )
|
|
{
|
|
// Align the allocation on an appropriate boundary
|
|
|
|
sz = (( sz + PACK_SIZE-1 ) & ~(PACK_SIZE-1) );
|
|
|
|
if ( noBuffer )
|
|
return ( *pOpNew )( sz );
|
|
else
|
|
{
|
|
// Handle a potential request for no space
|
|
|
|
if ( !sz )
|
|
sz = PACK_SIZE;
|
|
|
|
if ( blockLeft < sz )
|
|
{
|
|
// Is the request greater than the largest buffer size ?
|
|
|
|
if ( sz > memBlockSize )
|
|
return 0; // If it is, there is nothing we can do
|
|
|
|
|
|
// Allocate a new block
|
|
|
|
Block * pNewBlock = rnew Block;
|
|
|
|
|
|
// Did the allocation succeed ? If so connect it up
|
|
|
|
if ( pNewBlock )
|
|
{
|
|
// Handle the initial state
|
|
|
|
if ( tail )
|
|
tail = tail->next = pNewBlock;
|
|
else
|
|
head = tail = pNewBlock;
|
|
|
|
// Compute the remaining space
|
|
|
|
blockLeft = memBlockSize - sz;
|
|
|
|
} // End of IF then
|
|
else
|
|
return 0; // Oh-oh! Memory allocation failure
|
|
|
|
} // End of IF then
|
|
else
|
|
blockLeft -= sz; // Deduct the allocated amount
|
|
|
|
// And return the buffer address
|
|
|
|
return &( tail->memBlock[ blockLeft ]);
|
|
|
|
} // End of IF else
|
|
} // End of "HeapManager" FUNCTION "getMemory(size_t,int)"
|
|
|
|
|
|
|
|
|
|
// Friend functions of 'DName'
|
|
|
|
inline DName operator + ( char c, const DName & rd )
|
|
{ return DName ( c ) + rd; }
|
|
|
|
inline DName operator + ( DNameStatus st, const DName & rd )
|
|
{ return DName ( st ) + rd; }
|
|
|
|
inline DName operator + ( pcchar_t s, const DName & rd )
|
|
{ return DName ( s ) + rd; }
|
|
|
|
|
|
// The 'DName' constructors
|
|
|
|
inline DName::DName () { node = 0; stat = DN_valid; isIndir = 0; isAUDC = 0; isAUDTThunk = 0; isArrayType = 0; NoTE = 0; }
|
|
inline DName::DName ( DNameNode * pd ) { node = pd; stat = DN_valid; isIndir = 0; isAUDC = 0; isAUDTThunk = 0; isArrayType = 0; NoTE = 0; }
|
|
|
|
DName::DName ( char c )
|
|
{
|
|
stat = DN_valid;
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
node = 0;
|
|
NoTE = 0;
|
|
|
|
// The NULL character is boring, do not copy
|
|
|
|
if ( c )
|
|
doPchar ( &c, 1 );
|
|
|
|
} // End of "DName" CONSTRUCTOR '(char)'
|
|
|
|
|
|
inline DName::DName ( const DName & rd )
|
|
{
|
|
stat = rd.stat;
|
|
isIndir = rd.isIndir;
|
|
isAUDC = rd.isAUDC;
|
|
isAUDTThunk = rd.isAUDTThunk;
|
|
isArrayType = rd.isArrayType;
|
|
node = rd.node;
|
|
NoTE = rd.NoTE;
|
|
} // End of "DName" CONSTRUCTOR '(const DName&)'
|
|
|
|
|
|
DName::DName ( DName * pd )
|
|
{
|
|
if ( pd )
|
|
{
|
|
node = gnew pDNameNode ( pd );
|
|
stat = ( node ? DN_valid : DN_error );
|
|
|
|
} // End of IF else
|
|
else
|
|
{
|
|
stat = DN_valid;
|
|
node = 0;
|
|
|
|
} // End of IF else
|
|
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
NoTE = 0;
|
|
|
|
} // End of "DName" CONSTRUCTOR '( DName* )'
|
|
|
|
|
|
DName::DName ( pcchar_t s )
|
|
{
|
|
stat = DN_valid;
|
|
node = 0;
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
NoTE = 0;
|
|
|
|
if ( s )
|
|
doPchar ( s, und_strlen ( s ));
|
|
|
|
} // End of "DName" CONSTRUCTOR '(pcchar_t)'
|
|
|
|
|
|
DName::DName ( pcchar_t & name, char terminator )
|
|
{
|
|
stat = DN_valid;
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
node = 0;
|
|
NoTE = 0;
|
|
|
|
// Is there a string ?
|
|
|
|
if ( name )
|
|
if ( *name )
|
|
{
|
|
int len = 0;
|
|
|
|
|
|
// How long is the string ?
|
|
|
|
pcchar_t s;
|
|
for ( s = name; *name && ( *name != terminator ); name++ )
|
|
if ( isValidIdentChar ( *name ) || UnDecorator::doNoIdentCharCheck () )
|
|
len++;
|
|
else
|
|
{
|
|
stat = DN_invalid;
|
|
|
|
return;
|
|
|
|
} // End of IF else
|
|
|
|
// Copy the name string fragment
|
|
|
|
doPchar ( s, len );
|
|
|
|
// Now gobble the terminator if present, handle error conditions
|
|
|
|
if ( *name )
|
|
{
|
|
if ( *name++ != terminator )
|
|
{
|
|
stat = DN_error;
|
|
node = 0;
|
|
|
|
} // End of IF then
|
|
else
|
|
stat = DN_valid;
|
|
|
|
} // End of IF then
|
|
elif ( status () == DN_valid )
|
|
stat = DN_truncated;
|
|
|
|
} // End of IF then
|
|
else
|
|
stat = DN_truncated;
|
|
else
|
|
stat = DN_invalid;
|
|
|
|
} // End of "DName" CONSTRUCTOR '(pcchar_t&,char)'
|
|
|
|
|
|
DName::DName ( unsigned __int64 num )
|
|
{
|
|
char buf[ 21 ];
|
|
char * pBuf = buf + 20;
|
|
|
|
|
|
stat = DN_valid;
|
|
node = 0;
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
NoTE = 0;
|
|
|
|
// Essentially, 'ultoa ( num, buf, 10 )' :-
|
|
|
|
*pBuf = 0;
|
|
|
|
do
|
|
{
|
|
*( --pBuf ) = (char)(( num % 10ui64 ) + '0' );
|
|
num /= 10ui64;
|
|
|
|
} while ( num );
|
|
|
|
doPchar ( pBuf, ( 20 - (int) ( pBuf - buf )));
|
|
|
|
} // End of "DName" CONSTRUCTOR '(unsigned long)'
|
|
|
|
DName::DName ( __int64 num )
|
|
{
|
|
char buf[ 22 ];
|
|
char * pBuf = buf + 21;
|
|
|
|
stat = DN_valid;
|
|
node = 0;
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
NoTE = 0;
|
|
|
|
// Essentially, 'ltoa ( num, buf, 10 )' :-
|
|
|
|
*pBuf = 0;
|
|
|
|
bool fSigned = false;
|
|
if (num < 0) {
|
|
fSigned = true;
|
|
num = -num;
|
|
}
|
|
|
|
do
|
|
{
|
|
*( --pBuf ) = (char)(( num % 10ui64 ) + '0' );
|
|
num /= 10ui64;
|
|
|
|
} while ( num );
|
|
|
|
if (fSigned) {
|
|
*(--pBuf) = '-';
|
|
}
|
|
|
|
doPchar ( pBuf, ( 21 - (int) ( pBuf - buf )));
|
|
}
|
|
|
|
|
|
DName::DName ( DNameStatus st )
|
|
{
|
|
stat = ((( st == DN_invalid ) || ( st == DN_error )) ? st : DN_valid );
|
|
node = gnew DNameStatusNode ( st );
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
NoTE = 0;
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
|
|
} // End of "DName" CONSTRUCTOR '(DNameStatus)'
|
|
|
|
|
|
|
|
// Now the member functions for 'DName'
|
|
|
|
int DName::isValid () const { return (( status () == DN_valid ) || ( status () == DN_truncated )); }
|
|
int DName::isEmpty () const { return (( node == 0 ) || !isValid ()); }
|
|
|
|
inline DNameStatus DName::status () const { return (DNameStatus)stat; } // The cast is to keep Glockenspiel quiet
|
|
inline void DName::clearStatus () { stat = DN_valid; }
|
|
|
|
inline DName & DName::setPtrRef () { isIndir = 1; return *this; }
|
|
inline int DName::isPtrRef () const { return isIndir; }
|
|
inline int DName::isUDC () const { return ( !isEmpty () && isAUDC ); }
|
|
inline void DName::setIsUDC () { if ( !isEmpty ()) isAUDC = TRUE; }
|
|
inline int DName::isUDTThunk () const { return ( !isEmpty () && isAUDTThunk ); }
|
|
inline void DName::setIsUDTThunk () { if ( !isEmpty ()) isAUDTThunk = TRUE; }
|
|
inline void DName::setIsArray () { isArrayType = 1; }
|
|
inline int DName::isArray () const { return isArrayType; }
|
|
inline int DName::isNoTE () const { return NoTE; }
|
|
inline void DName::setIsNoTE () { NoTE = TRUE; }
|
|
|
|
int DName::length () const
|
|
{
|
|
int len = 0;
|
|
|
|
|
|
if ( !isEmpty ())
|
|
for ( DNameNode * pNode = node; pNode; pNode = pNode->nextNode ())
|
|
len += pNode->length ();
|
|
|
|
return len;
|
|
|
|
} // End of "DName" FUNCTION "length"
|
|
|
|
|
|
char DName::getLastChar () const
|
|
{
|
|
DNameNode * pLast = 0;
|
|
|
|
if ( !isEmpty ())
|
|
for ( DNameNode * pNode = node; pNode; pNode = pNode->nextNode ())
|
|
if ( pNode->length () != 0 )
|
|
pLast = pNode;
|
|
|
|
return pLast != 0 ? pLast->getLastChar () : '\0';
|
|
|
|
} // End of "DName" FUNCTION "getLastChar"
|
|
|
|
|
|
pchar_t DName::getString ( pchar_t buf, int max ) const
|
|
{
|
|
if ( !isEmpty ())
|
|
{
|
|
// Does the caller want a buffer allocated ?
|
|
|
|
if ( !buf )
|
|
{
|
|
max = length () + 1;
|
|
buf = gnew char[ max ]; // Get a buffer big enough
|
|
|
|
} // End of IF then
|
|
|
|
// If memory allocation failure, then return no buffer
|
|
|
|
if ( buf )
|
|
{
|
|
// Now, go through the process of filling the buffer (until max is reached)
|
|
|
|
int curLen = max;
|
|
DNameNode * curNode = node;
|
|
pchar_t curBuf = buf;
|
|
|
|
|
|
while ( curNode && ( curLen > 0 ))
|
|
{
|
|
int fragLen = curNode->length ();
|
|
pchar_t fragBuf = 0;
|
|
|
|
|
|
// Skip empty nodes
|
|
|
|
if ( fragLen )
|
|
{
|
|
// Handle buffer overflow
|
|
|
|
if (( curLen - fragLen ) < 0 )
|
|
fragLen = curLen;
|
|
|
|
// Now copy 'len' number of bytes of the piece to the buffer
|
|
|
|
fragBuf = curNode->getString ( curBuf, fragLen );
|
|
|
|
// Should never happen, but handle it anyway
|
|
|
|
if ( fragBuf )
|
|
{
|
|
// Update string position
|
|
|
|
curLen -= fragLen;
|
|
curBuf += fragLen;
|
|
|
|
} // End of IF
|
|
} // End of IF
|
|
|
|
// Move on to the next name fragment
|
|
|
|
curNode = curNode->nextNode ();
|
|
|
|
} // End of WHILE
|
|
|
|
*curBuf = 0; // Always NULL terminate the resulting string
|
|
|
|
} // End of IF
|
|
} // End of IF then
|
|
elif ( buf )
|
|
*buf = 0;
|
|
|
|
// Return the buffer
|
|
|
|
return buf;
|
|
|
|
} // End of "DName" FUNCTION "getString(pchar_t,int)"
|
|
|
|
|
|
DName DName::operator + ( char ch ) const
|
|
{
|
|
DName local ( *this );
|
|
|
|
|
|
if ( local.isEmpty ())
|
|
local = ch;
|
|
else
|
|
local += ch;
|
|
|
|
// And return the newly formed 'DName'
|
|
|
|
return local;
|
|
|
|
} // End of "DName" OPERATOR "+(char)"
|
|
|
|
|
|
DName DName::operator + ( pcchar_t str ) const
|
|
{
|
|
DName local ( *this );
|
|
|
|
|
|
if ( local.isEmpty ())
|
|
local = str;
|
|
else
|
|
local += str;
|
|
|
|
// And return the newly formed 'DName'
|
|
|
|
return local;
|
|
|
|
} // End of "DName" OPERATOR "+(pcchar_t)"
|
|
|
|
|
|
DName DName::operator + ( const DName & rd ) const
|
|
{
|
|
DName local ( *this );
|
|
|
|
|
|
if ( local.isEmpty ())
|
|
local = rd;
|
|
elif ( rd.isEmpty ())
|
|
local += rd.status ();
|
|
else
|
|
local += rd;
|
|
|
|
// And return the newly formed 'DName'
|
|
|
|
return local;
|
|
|
|
} // End of "DName" OPERATOR "+(const DName&)"
|
|
|
|
|
|
DName DName::operator + ( DName * pd ) const
|
|
{
|
|
DName local ( *this );
|
|
|
|
|
|
if ( local.isEmpty ())
|
|
local = pd;
|
|
else
|
|
local += pd;
|
|
|
|
// And return the newly formed 'DName'
|
|
|
|
return local;
|
|
|
|
} // End of "DName" OPERATOR "+(DName*)"
|
|
|
|
|
|
DName DName::operator + ( DNameStatus st ) const
|
|
{
|
|
DName local ( *this );
|
|
|
|
|
|
if ( local.isEmpty ())
|
|
local = st;
|
|
else
|
|
local += st;
|
|
|
|
// And return the newly formed 'DName'
|
|
|
|
return local;
|
|
|
|
} // End of "DName" OPERATOR "+(DNameStatus)"
|
|
|
|
|
|
|
|
DName & DName::operator += ( char ch )
|
|
{
|
|
if ( ch )
|
|
if ( isEmpty ())
|
|
*this = ch;
|
|
else
|
|
{
|
|
node = node->clone ();
|
|
|
|
if ( node )
|
|
*node += gnew charNode ( ch );
|
|
else
|
|
stat = DN_error;
|
|
|
|
} // End of IF
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR "+=(char)"
|
|
|
|
|
|
DName & DName::operator += ( pcchar_t str )
|
|
{
|
|
if ( str && *str )
|
|
if ( isEmpty ())
|
|
*this = str;
|
|
else
|
|
{
|
|
node = node->clone ();
|
|
|
|
if ( node )
|
|
*node += gnew pcharNode ( str );
|
|
else
|
|
stat = DN_error;
|
|
|
|
} // End of IF
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR "+=(pcchar_t)"
|
|
|
|
|
|
DName & DName::operator += ( const DName & rd )
|
|
{
|
|
if ( rd.isEmpty ())
|
|
*this += rd.status ();
|
|
else
|
|
if ( isEmpty ())
|
|
*this = rd;
|
|
else
|
|
{
|
|
node = node->clone ();
|
|
|
|
if ( node )
|
|
*node += rd.node;
|
|
else
|
|
stat = DN_error;
|
|
|
|
} // End of IF
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR "+=(const DName&)"
|
|
|
|
|
|
DName & DName::operator += ( DName * pd )
|
|
{
|
|
if ( pd )
|
|
if ( isEmpty ())
|
|
*this = pd;
|
|
elif (( pd->status () == DN_valid ) || ( pd->status () == DN_truncated ))
|
|
{
|
|
DNameNode * pNew = gnew pDNameNode ( pd );
|
|
|
|
|
|
if ( pNew )
|
|
{
|
|
node = node->clone ();
|
|
|
|
if ( node )
|
|
*node += pNew;
|
|
|
|
} // End of IF then
|
|
else
|
|
node = 0;
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
|
|
} // End of IF then
|
|
else
|
|
*this += pd->status ();
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR "+=(DName*)"
|
|
|
|
|
|
DName & DName::operator += ( DNameStatus st )
|
|
{
|
|
if ( isEmpty () || (( st == DN_invalid ) || ( st == DN_error )))
|
|
*this = st;
|
|
else
|
|
{
|
|
DNameNode * pNew = gnew DNameStatusNode ( st );
|
|
|
|
|
|
if ( pNew )
|
|
{
|
|
node = node->clone ();
|
|
|
|
if ( node )
|
|
*node += pNew;
|
|
|
|
} // End of IF then
|
|
else
|
|
node = 0;
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
|
|
} // End of IF else
|
|
|
|
// Return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR "+=(DNameStatus)"
|
|
|
|
|
|
|
|
DName & DName::operator |= ( const DName & rd )
|
|
{
|
|
// Attenuate the error status. Always becomes worse. Don't propogate truncation
|
|
|
|
if (( status () != DN_error ) && !rd.isValid ())
|
|
stat = rd.status ();
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '|=(const DName&)'
|
|
|
|
|
|
|
|
DName & DName::operator = ( char ch )
|
|
{
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
|
|
doPchar ( &ch, 1 );
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '=(char)'
|
|
|
|
|
|
DName & DName::operator = ( pcchar_t str )
|
|
{
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
|
|
doPchar ( str, und_strlen ( str ));
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '=(pcchar_t)'
|
|
|
|
|
|
DName & DName::operator = ( const DName & rd )
|
|
{
|
|
if (( status () == DN_valid ) || ( status () == DN_truncated ))
|
|
{
|
|
stat = rd.stat;
|
|
isIndir = rd.isIndir;
|
|
isAUDC = rd.isAUDC;
|
|
isAUDTThunk = rd.isAUDTThunk;
|
|
isArrayType = rd.isArrayType;
|
|
node = rd.node;
|
|
|
|
} // End of IF
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '=(const DName&)'
|
|
|
|
|
|
DName & DName::operator = ( DName * pd )
|
|
{
|
|
if (( status () == DN_valid ) || ( status () == DN_truncated ))
|
|
if ( pd )
|
|
{
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
node = gnew pDNameNode ( pd );
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
|
|
} // End of IF then
|
|
else
|
|
*this = DN_error;
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '=(DName*)'
|
|
|
|
|
|
DName & DName::operator = ( DNameStatus st )
|
|
{
|
|
if (( st == DN_invalid ) || ( st == DN_error ))
|
|
{
|
|
node = 0;
|
|
|
|
if ( status () != DN_error )
|
|
stat = st;
|
|
|
|
} // End of IF then
|
|
elif (( status () == DN_valid ) || ( status () == DN_truncated ))
|
|
{
|
|
isIndir = 0;
|
|
isAUDC = 0;
|
|
isAUDTThunk = 0;
|
|
isArrayType = 0;
|
|
node = gnew DNameStatusNode ( st );
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
|
|
} // End of ELIF then
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DName" OPERATOR '=(DNameStatus)'
|
|
|
|
|
|
// Private implementation functions for 'DName'
|
|
|
|
void DName::doPchar ( pcchar_t str, int len )
|
|
{
|
|
if ( !(( status () == DN_invalid ) || ( status () == DN_error )))
|
|
if ( node )
|
|
*this = DN_error;
|
|
elif ( str && len )
|
|
{
|
|
// Allocate as economically as possible
|
|
|
|
switch ( len )
|
|
{
|
|
case 0:
|
|
stat = DN_error;
|
|
break;
|
|
|
|
case 1:
|
|
node = gnew charNode ( *str );
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
break;
|
|
|
|
default:
|
|
node = gnew pcharNode ( str, len );
|
|
|
|
if ( !node )
|
|
stat = DN_error;
|
|
break;
|
|
|
|
} // End of SWITCH
|
|
} // End of ELIF
|
|
else
|
|
stat = DN_invalid;
|
|
|
|
} // End of "DName" FUNCTION "doPchar(pcchar_t,int)"
|
|
|
|
|
|
|
|
// The member functions for the 'Replicator'
|
|
|
|
inline int Replicator::isFull () const { return ( index == 9 ); }
|
|
inline Replicator::Replicator ()
|
|
: ErrorDName ( DN_error ), InvalidDName ( DN_invalid )
|
|
{ index = -1; }
|
|
|
|
|
|
|
|
Replicator & Replicator::operator += ( const DName & rd )
|
|
{
|
|
if ( !isFull () && !rd.isEmpty ())
|
|
{
|
|
DName * pNew = gnew DName ( rd );
|
|
|
|
|
|
// Don't update if failed
|
|
|
|
if ( pNew )
|
|
dNameBuffer[ ++index ] = pNew;
|
|
|
|
} // End of IF
|
|
|
|
return *this;
|
|
|
|
} // End of "Replicator" OPERATOR '+=(const DName&)'
|
|
|
|
|
|
const DName & Replicator::operator [] ( int x ) const
|
|
{
|
|
if (( x < 0 ) || ( x > 9 ))
|
|
return ErrorDName;
|
|
elif (( index == -1 ) || ( x > index ))
|
|
return InvalidDName;
|
|
else
|
|
return *dNameBuffer[ x ];
|
|
|
|
} // End of "Replicator" OPERATOR '[](int)'
|
|
|
|
|
|
|
|
// The member functions for the 'DNameNode' classes
|
|
|
|
#if ( !NO_VIRTUAL )
|
|
DNameNode::DNameNode ()
|
|
#else // } elif NO_VIRTUAL {
|
|
DNameNode::DNameNode ( NodeType ndTy )
|
|
: typeIndex ( ndTy )
|
|
#endif // NO_VIRTUAL
|
|
{ next = 0; }
|
|
|
|
inline DNameNode::DNameNode ( const DNameNode & rd ) { next = (( rd.next ) ? rd.next->clone () : 0 ); }
|
|
|
|
inline DNameNode * DNameNode::nextNode () const { return next; }
|
|
|
|
DNameNode * DNameNode::clone ()
|
|
{
|
|
return gnew pDNameNode ( gnew DName ( this ));
|
|
}
|
|
|
|
#if ( NO_VIRTUAL )
|
|
int DNameNode::length () const
|
|
{ // Pure function, should not be called
|
|
|
|
switch ( typeIndex )
|
|
{
|
|
case charNode_t:
|
|
return ((charNode*)this )->length ();
|
|
|
|
case pcharNode_t:
|
|
return ((pcharNode*)this )->length ();
|
|
|
|
case pDNameNode_t:
|
|
return ((pDNameNode*)this )->length ();
|
|
|
|
case DNameStatusNode_t:
|
|
return ((DNameStatusNode*)this )->length ();
|
|
|
|
} // End of SWITCH
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
char DNameNode::getLastChar () const
|
|
{ // Pure function, should not be called
|
|
|
|
switch ( typeIndex )
|
|
{
|
|
case charNode_t:
|
|
return ((charNode*)this )->getLastChar ();
|
|
|
|
case pcharNode_t:
|
|
return ((pcharNode*)this )->getLastChar ();
|
|
|
|
case pDNameNode_t:
|
|
return ((pDNameNode*)this )->getLastChar ();
|
|
|
|
case DNameStatusNode_t:
|
|
return ((DNameStatusNode*)this )->getLastChar ();
|
|
|
|
} // End of SWITCH
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
pchar_t DNameNode::getString ( pchar_t s, int l ) const
|
|
{ // Pure function, should not be called
|
|
|
|
switch ( typeIndex )
|
|
{
|
|
case charNode_t:
|
|
return ((charNode*)this )->getString ( s, l );
|
|
|
|
case pcharNode_t:
|
|
return ((pcharNode*)this )->getString ( s, l );
|
|
|
|
case pDNameNode_t:
|
|
return ((pDNameNode*)this )->getString ( s, l );
|
|
|
|
case DNameStatusNode_t:
|
|
return ((DNameStatusNode*)this )->getString ( s, l );
|
|
|
|
} // End of SWITCH
|
|
|
|
return 0;
|
|
}
|
|
#endif // NO_VIRTUAL
|
|
|
|
|
|
DNameNode & DNameNode::operator += ( DNameNode * pNode )
|
|
{
|
|
if ( pNode )
|
|
{
|
|
if ( next )
|
|
{
|
|
// Skip to the end of the chain
|
|
|
|
DNameNode* pScan;
|
|
for ( pScan = next; pScan->next; pScan = pScan->next )
|
|
;
|
|
|
|
// And append the new node
|
|
|
|
pScan->next = pNode;
|
|
|
|
} // End of IF then
|
|
else
|
|
next = pNode;
|
|
|
|
} // End of IF
|
|
|
|
// And return self
|
|
|
|
return *this;
|
|
|
|
} // End of "DNameNode" OPERATOR '+=(DNameNode*)'
|
|
|
|
|
|
|
|
// The 'charNode' virtual functions
|
|
|
|
charNode::charNode ( char ch )
|
|
#if ( NO_VIRTUAL )
|
|
: DNameNode ( charNode_t )
|
|
#endif // NO_VIRTUAL
|
|
{ me = ch; }
|
|
|
|
inline int charNode::length () const { return 1; }
|
|
|
|
inline char charNode::getLastChar () const { return me; }
|
|
|
|
pchar_t charNode::getString ( pchar_t buf, int len ) const
|
|
{
|
|
if ( buf && len )
|
|
*buf = me;
|
|
else
|
|
buf = 0;
|
|
|
|
// Now return the character
|
|
|
|
return buf;
|
|
|
|
} // End of "charNode" FUNCTION "getString(pchar_t,int)"
|
|
|
|
|
|
|
|
// The 'pcharNode' virtual functions
|
|
|
|
inline int pcharNode::length () const { return myLen; }
|
|
|
|
inline char pcharNode::getLastChar () const { return ( myLen ? me[ myLen - 1 ] : '\0' ); }
|
|
|
|
pcharNode::pcharNode ( pcchar_t str, int len )
|
|
#if ( NO_VIRTUAL )
|
|
: DNameNode ( pcharNode_t )
|
|
#endif // NO_VIRTUAL
|
|
{
|
|
// Get length if not supplied
|
|
|
|
if ( !len && str )
|
|
len = und_strlen ( str );
|
|
|
|
// Allocate a new string buffer if valid state
|
|
|
|
if ( len && str )
|
|
{
|
|
me = gnew char[ len ];
|
|
myLen = len;
|
|
|
|
if ( me )
|
|
und_strncpy ( me, str, len );
|
|
|
|
} // End of IF then
|
|
else
|
|
{
|
|
me = 0;
|
|
myLen = 0;
|
|
|
|
} // End of IF else
|
|
} // End of "pcharNode" CONSTRUCTOR '(pcchar_t,int)'
|
|
|
|
|
|
pchar_t pcharNode::getString ( pchar_t buf, int len ) const
|
|
{
|
|
// Use the shorter of the two lengths (may not be NULL terminated)
|
|
|
|
if ( len > pcharNode::length ())
|
|
len = pcharNode::length ();
|
|
|
|
// Do the copy as appropriate
|
|
|
|
return (( me && buf && len ) ? und_strncpy ( buf, me, len ) : 0 );
|
|
|
|
} // End of "pcharNode" FUNCTION "getString(pchar_t,int)"
|
|
|
|
|
|
|
|
// The 'pDNameNode' virtual functions
|
|
|
|
pDNameNode::pDNameNode ( DName * pName )
|
|
#if ( NO_VIRTUAL )
|
|
: DNameNode ( pDNameNode_t )
|
|
#endif // NO_VIRTUAL
|
|
{ me = (( pName && (( pName->status () == DN_invalid ) || ( pName->status () == DN_error ))) ? 0 : pName ); }
|
|
|
|
inline int pDNameNode::length () const { return ( me ? me->length () : 0 ); }
|
|
|
|
inline char pDNameNode::getLastChar () const { return ( me ? me->getLastChar () : '\0' ); }
|
|
|
|
pchar_t pDNameNode::getString ( pchar_t buf, int len ) const
|
|
{ return (( me && buf && len ) ? me->getString ( buf, len ) : 0 ); }
|
|
|
|
|
|
|
|
// The 'DNameStatusNode' virtual functions
|
|
|
|
DNameStatusNode::DNameStatusNode ( DNameStatus stat )
|
|
#if ( NO_VIRTUAL )
|
|
: DNameNode ( DNameStatusNode_t )
|
|
#endif // NO_VIRTUAL
|
|
{ me = stat; myLen = (( me == DN_truncated ) ? TruncationMessageLength : 0 ); }
|
|
|
|
inline int DNameStatusNode::length () const { return myLen; }
|
|
|
|
inline char DNameStatusNode::getLastChar () const
|
|
{ return (( me == DN_truncated ) ? TruncationMessage[ TruncationMessageLength - 1 ] : '\0' ); }
|
|
|
|
pchar_t DNameStatusNode::getString ( pchar_t buf, int len ) const
|
|
{
|
|
// Use the shorter of the two lengths (may not be NULL terminated)
|
|
|
|
if ( len > DNameStatusNode::length ())
|
|
len = DNameStatusNode::length ();
|
|
|
|
// Do the copy as appropriate
|
|
|
|
return ((( me == DN_truncated ) && buf && len ) ? und_strncpy ( buf, TruncationMessage, len ) : 0 );
|
|
|
|
} // End of "DNameStatusNode" FUNCTION "getString(pchar_t,int)"
|
|
|
|
|
|
static unsigned int und_strlen ( pcchar_t str )
|
|
{
|
|
unsigned int len;
|
|
|
|
for ( len = 0; *str; str++ )
|
|
len++;
|
|
|
|
return len;
|
|
|
|
} // End of FUNCTION "und_strlen"
|
|
|
|
|
|
static pchar_t und_strncpy ( pchar_t dst, pcchar_t src, unsigned int len )
|
|
{
|
|
for ( char * d = dst; ( len && ( *d = *src )); d++, src++, len-- )
|
|
;
|
|
|
|
return dst;
|
|
|
|
} // End of FUNCTION "und_strncpy"
|
|
|
|
static unsigned int und_strncmp ( pcchar_t first, pcchar_t last, unsigned int count)
|
|
{
|
|
if (!count)
|
|
return(0);
|
|
|
|
while (--count && *first && *first == *last)
|
|
{
|
|
first++;
|
|
last++;
|
|
}
|
|
|
|
return( *(unsigned char *)first - *(unsigned char *)last );
|
|
}
|
|
|