|
|
/* * 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 ); }
|