Windows2000/private/windbg64/langapi/undname/undname.cxx

3355 lines
94 KiB
C++
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
// Make sure all dependent defines exist and have a valid value
#ifndef NO_COMPILER_NAMES
#define NO_COMPILER_NAMES 0
#endif
#ifndef VERS_32BIT
#define VERS_32BIT 1
#endif
#ifndef PACK_SIZE
#if !VERS_32BIT
#define PACK_SIZE 2
#elif defined(_X86_)
#define PACK_SIZE 4
#else
#define PACK_SIZE 8
#endif
#endif
// Check for version inconsistancies, and setup version flags
#ifdef VERS_BSC
#undef NO_COMPILER_NAMES
#define NO_COMPILER_NAMES 1
#pragma inline_depth ( 3 )
#pragma check_stack ( off )
#else
#pragma inline_depth ( 3 )
#pragma check_stack ( off )
#endif
#define PURE =0
#include <stddef.h>
#include "undname.hxx"
static unsigned int __near __pascal strlen ( pcchar_t );
static pchar_t __near __pascal strncpy ( pchar_t, pcchar_t, unsigned int );
class DName;
class DNameNode;
class Replicator;
class HeapManager;
class UnDecorator;
const unsigned int memBlockSize = 508; // A '512' byte block including the header
class HeapManager
{
private:
Alloc_t pOpNew;
Free_t pOpDelete;
struct Block
{
Block * next;
char memBlock[ memBlockSize ];
__near Block () { next = 0; }
};
Block * head;
Block * tail;
unsigned int blockLeft;
public:
void __near Constructor ( Alloc_t pAlloc, Free_t pFree )
{ pOpNew = pAlloc;
pOpDelete = pFree;
blockLeft = 0;
head = 0;
tail = 0;
}
void __far * __near getMemory ( unsigned int, int );
void __near Destructor ( void )
{ if ( pOpDelete != 0 )
while ( tail = head )
{
head = tail->next;
( *pOpDelete )( tail );
}
}
#define gnew new(heap,0)
#define rnew new(heap,1)
};
void * __near __pascal operator new ( unsigned int, HeapManager &, int = 0 );
static HeapManager heap;
// The MS Token table
enum Tokens
{
#if !VERS_32BIT
TOK_near,
TOK_nearSp,
TOK_nearP,
TOK_far,
TOK_farSp,
TOK_farP,
TOK_huge,
TOK_hugeSp,
TOK_hugeP,
#endif
TOK_basedLp,
TOK_cdecl,
TOK_pascal,
TOK_stdcall,
TOK_thiscall,
TOK_fastcall,
TOK_ptr64,
TOK_restrict,
#if !VERS_32BIT
TOK_interrupt,
TOK_saveregs,
TOK_self,
TOK_segment,
TOK_segnameLpQ,
#endif
TOK__last
};
static const pcchar_t __near tokenTable[] =
{
#if !VERS_32BIT
"__near", // TOK_near
"__near ", // TOK_nearSp
"__near*", // TOK_nearP
"__far", // TOK_far
"__far ", // TOK_farSp
"__far*", // TOK_farP
"__huge", // TOK_huge
"__huge ", // TOK_hugeSp
"__huge*", // TOK_hugeP
#endif
"__based(", // TOK_basedLp
"__cdecl", // TOK_cdecl
"__pascal", // TOK_pascal
"__stdcall", // TOK_stdcall
"__thiscall", // TOK_thiscall
"__fastcall", // TOK_fastcall
"__ptr64", // TOK_ptr64
"__restrict", // TOK_restrict
#if !VERS_32BIT
"__interrupt", // TOK_interrupt
"__saveregs", // TOK_saveregs
"__self", // TOK_self
"__segment", // TOK_segment
"__segname(\"", // TOK_segnameLpQ
#endif
""
};
// The operator mapping table
static const pcchar_t __near nameTable[] =
{
" new",
" delete",
"=",
">>",
"<<",
"!",
"==",
"!=",
"[]",
"operator",
"->",
"*",
"++",
"--",
"-",
"+",
"&",
"->*",
"/",
"%",
"<",
"<=",
">",
">=",
",",
"()",
"~",
"^",
"|",
"&&",
"||",
"*=",
"+=",
"-=",
"/=",
"%=",
">>=",
"<<=",
"&=",
"|=",
"^=",
#if ( !NO_COMPILER_NAMES )
"`vftable'",
"`vbtable'",
"`vcall'",
"`typeof'",
"`local static guard'",
"`string'",
"`vbase destructor'",
"`vector deleting destructor'",
"`default constructor closure'",
"`scalar deleting destructor'",
"`vector constructor iterator'",
"`vector destructor iterator'",
"`vector vbase constructor iterator'",
"`virtual displacement map",
"`eh vector constructor iterator'",
"`eh vector destructor iterator'",
"`eh vector vbase constructor iterator'",
"`copy constructor closure'",
"`udt returning'",
"`EH", //eh initialized struct
"`RTTI", //rtti initialized struct
"`local vftable'",
"`local vftable constructor closure'",
#endif // !NO_COMPILER_NAMES
" new[]",
" delete[]",
#if ( !NO_COMPILER_NAMES )
"`omni callsig'",
"`placement delete closure'",
"`placement delete[] closure'",
#endif
""
};
static const pcchar_t __near ehTable[] =
{
" Ptr to Member Data'",
" Catchable Type'",
" Catchable Type Array'",
" ThrowInfo'",
};
static const pcchar_t __near rttiTable[] =
{
" Type Descriptor'",
" Base Class Descriptor at (",
" Base Class Array'",
" Class Hierarchy Descriptor'",
" Complete Object Locator'",
};
// The following 'enum' should really be nested inside 'class DName', but to
// make the code compile better with Glockenspiel, I have extracted it
enum DNameStatus
{
DN_valid,
DN_invalid,
DN_truncated,
DN_error
};
class DName
{
public:
__near DName ();
__near DName ( char );
#if 1
__near DName ( const DName & ); // Shallow copy
#endif
__near DName ( DNameNode * );
__near DName ( pcchar_t );
__near DName ( pcchar_t&, char );
__near DName ( DNameStatus );
__near DName ( DName * );
__near DName ( unsigned long );
int __near isValid () const;
int __near isEmpty () const;
DNameStatus __near status () const;
DName & __near setPtrRef ();
int __near isPtrRef () const;
int __near isUDC () const;
void __near setIsUDC ();
int __near isUDTThunk () const;
void __near setIsUDTThunk ();
int __near length () const;
pchar_t __near getString ( pchar_t, int ) const;
DName __near operator + ( pcchar_t ) const;
DName __near operator + ( const DName & ) const;
DName __near operator + ( char ) const;
DName __near operator + ( DName * ) const;
DName __near operator + ( DNameStatus ) const;
DName & __near operator += ( char );
DName & __near operator += ( pcchar_t );
DName & __near operator += ( DName * );
DName & __near operator += ( DNameStatus );
DName & __near operator += ( const DName & );
DName & __near operator |= ( const DName & );
DName & __near operator = ( pcchar_t );
DName & __near operator = ( const DName & );
DName & __near operator = ( char );
DName & __near operator = ( DName * );
DName & __near operator = ( DNameStatus );
// Friends :
friend DName __near __pascal operator + ( char, const DName & );
friend DName __near __pascal operator + ( pcchar_t, const DName & );
friend DName __near __pascal operator + ( DNameStatus, const DName & );
private:
DNameNode * node;
DNameStatus stat : 4;
unsigned int isIndir : 1;
unsigned int isAUDC : 1;
unsigned int isAUDTThunk : 1;
void __near doPchar ( pcchar_t, int );
};
class Replicator
{
private:
// Declare, in order to suppress automatic generation
void operator = ( const Replicator& );
int index;
DName * dNameBuffer[ 10 ];
const DName ErrorDName;
const DName InvalidDName;
public:
__near Replicator ();
int __near isFull () const;
Replicator & __near operator += ( const DName & );
const DName & __near operator [] ( int ) const;
};
class UnDecorator
{
private:
// Declare, in order to suppress automatic generation
void operator = ( const UnDecorator& );
Replicator ArgList;
static Replicator * pArgList;
Replicator ZNameList;
static Replicator * pZNameList;
static Replicator * pTemplateArgList;
static pcchar_t gName;
static pcchar_t name;
static pchar_t outputString;
static int maxStringLength;
static unsigned short disableFlags;
static DName __near getDecoratedName ( void );
static DName __near getSymbolName ( void );
static DName __near getZName ( void );
static DName __near getOperatorName ( void );
static DName __near getScope ( void );
static DName getScopedName ( void );
static DName __near getSignedDimension ( void );
static DName __near getDimension ( void );
static int __near getNumberOfDimensions ( void );
static DName __near getTemplateName ( void );
static DName __near getTemplateArgumentList( void );
static DName __near getTemplateConstant( void );
static DName __near composeDeclaration ( const DName & );
static int __near getTypeEncoding ( void );
static DName __near getBasedType ( void );
static DName __near getECSUName ( void );
static DName __near getEnumName ( void );
static DName __near getCallingConvention ( void );
static DName __near getReturnType ( DName * = 0 );
static DName __near getDataType ( DName * );
static DName __near getPrimaryDataType ( const DName & );
static DName __near getDataIndirectType ( const DName &, char, const DName &, int = FALSE );
static DName __near getDataIndirectType ();
static DName __near getBasicDataType ( const DName & );
static DName __near getECSUDataType ( int = 0 );
static int __near getECSUDataIndirectType ();
static DName __near getPtrRefType ( const DName &, const DName &, int );
static DName __near getPtrRefDataType ( const DName &, int );
static DName __near getArrayType ( const DName& );
static DName getFunctionIndirectType( const DName & superType );
static DName __near getArgumentTypes ( void );
static DName __near getArgumentList ( void );
static DName __near getThrowTypes ( void );
static DName __near getLexicalFrame ( void );
static DName __near getStorageConvention ( void );
static DName __near getThisType ( void );
static DName __near getPointerType ( const DName &, const DName & );
static DName __near getReferenceType ( const DName &, const DName & );
static DName __near getExternalDataType ( const DName & );
static DName __near getSegmentName ( void );
#if ( !NO_COMPILER_NAMES )
static DName __near getDisplacement ( void );
static DName __near getCallIndex ( void );
static DName __near getGuardNumber ( void );
static DName __near getVfTableType ( const DName & );
static DName __near getVbTableType ( const DName & );
static DName __near getVCallThunkType ( void );
#endif // !NO_COMPILER_NAMES
public:
__near UnDecorator ( pchar_t, pcchar_t, int, unsigned short );
static int __near doUnderScore ();
static int __near doMSKeywords ();
static int __near doFunctionReturns ();
static int __near doAllocationModel ();
static int __near doAllocationLanguage ();
#if 0
static int __near doMSThisType ();
static int __near doCVThisType ();
#endif
static int __near doThisTypes ();
static int __near doAccessSpecifiers ();
static int __near doThrowTypes ();
static int __near doMemberTypes ();
static int __near doReturnUDTModel ();
static int __near do32BitNear ();
static int __near doNameOnly ();
static int __near doTypeOnly ();
static pcchar_t __near UScore ( Tokens );
__near operator pchar_t ();
};
Replicator * UnDecorator::pArgList;
Replicator * UnDecorator::pZNameList = 0;
Replicator * UnDecorator::pTemplateArgList = 0;
pcchar_t UnDecorator::gName = 0;
pcchar_t UnDecorator::name = 0;
pchar_t UnDecorator::outputString = 0;
int UnDecorator::maxStringLength = 0;
unsigned short UnDecorator::disableFlags = 0;
#ifdef _CRTBLD
pchar_t __far _CRTIMP __loadds __unDName ( pchar_t outputString,
#else
pchar_t __far __cdecl __loadds unDName ( pchar_t outputString,
#endif
pcchar_t name,
int maxStringLength, // Note, COMMA is leading following optional arguments
Alloc_t pAlloc,
Free_t pFree,
unsigned short disableFlags
)
/*
* This function will undecorate a name, returning the string corresponding to
* the C++ declaration needed to produce the name. Its has a similar interface
* to 'strncpy'.
*
* If the target string 'outputString' is specified to be NULL, a string of
* suitable length will be allocated and its address returned. If the returned
* string is allocated by 'unDName', then it is the programmers responsibility
* to deallocate it. It will have been allocated on the far heap.
*
* If the target string is not NULL, then the parameter 'maxStringLength' will
* specify the maximum number of characters which may be placed in the string.
* In this case, the returned value is the same as 'outputString'.
*
* Both the input parameter 'name' and the returned string are NULL terminated
* strings of characters.
*
* If the returned value is NULL, it indicates that the undecorator ran out of
* memory, or an internal error occurred, and was unable to complete its task.
*/
{
// Must have an allocator and a deallocator (and we MUST trust them)
if ( !( pAlloc ))
return 0;
else
heap.Constructor ( pAlloc, pFree );
// Create the undecorator object, and get the result
UnDecorator unDecorate ( outputString,
name,
maxStringLength,
disableFlags
);
pchar_t unDecoratedName = unDecorate;
// Destruct the heap (would use a destructor, but that causes DLL problems)
heap.Destructor ();
// And return the composed name
return unDecoratedName;
} // End of FUNCTION "unDName"
// The 'UnDecorator' member functions
inline __near UnDecorator::UnDecorator ( pchar_t output,
pcchar_t dName,
int maxLen,
unsigned short disable
)
{
name = dName;
gName = name;
maxStringLength = maxLen;
outputString = output;
pZNameList = &ZNameList;
pArgList = &ArgList;
disableFlags = disable;
} // End of "UnDecorator" CONSTRUCTOR '()'
inline __near UnDecorator::operator pchar_t ()
{
DName result;
DName unDName;
// Find out if the name is a decorated name or not. Could be a reserved
// CodeView variant of a decorated name
if ( name )
{
if (( *name == '?' ) && ( name[ 1 ] == '@' ))
{
#if ( !NO_COMPILER_NAMES )
gName += 2;
result = "CV: " + getDecoratedName ();
#else // } elif NO_COMPILER_NAMES
result = DN_invalid;
#endif // NO_COMPILER_NAMES
} // End of IF then
elif (( *name == '?' ) && ( name[1] == '$' ))
result = getTemplateName ();
else
result = getDecoratedName ();
} // End of IF then
// If the name was not a valid name, then make the name the same as the original
// It is also invalid if there are any remaining characters in the name (except when
// we're giving the name only)
if ( result.status () == DN_error )
return 0;
elif ( (*gName && !doNameOnly ()) || ( result.status () == DN_invalid ))
unDName = name; // Return the original name
else
unDName = result;
// Construct the return string
if ( !outputString )
{
maxStringLength = unDName.length () + 1;
outputString = rnew char[ maxStringLength ];
} // End of IF
if ( outputString )
unDName.getString ( outputString, maxStringLength );
// Return the result
return outputString;
} // End of "UnDecorator" OPERATOR 'pchar_t'
DName __near UnDecorator::getDecoratedName ( void )
{
// Ensure that it is intended to be a decorated name
if ( doTypeOnly() )
{
// Disable the type-only flag, so that if we get here recursively, eg.
// in a template tag, we do full name undecoration.
disableFlags &= ~UNDNAME_TYPE_ONLY;
// If we're decoding just a type, process it as the type for an abstract
// declarator, by giving an empty symbol name.
DName result = getDataType ( NULL );
disableFlags |= UNDNAME_TYPE_ONLY;
return result;
}
elif ( *gName == '?' )
{
// Extract the basic symbol name
gName++; // Advance the original name pointer
DName symbolName = getSymbolName ();
int udcSeen = symbolName.isUDC ();
// Abort if the symbol name is invalid
if ( !symbolName.isValid ())
return symbolName;
// Extract, and prefix the scope qualifiers
if ( *gName && ( *gName != '@' )) {
DName scope = getScope ();
if ( !scope.isEmpty() )
symbolName = scope + "::" + symbolName;
}
if ( udcSeen )
symbolName.setIsUDC ();
// Now compose declaration
if ( symbolName.isEmpty () || (doNameOnly () && !udcSeen))
{
return symbolName;
}
elif ( !*gName || ( *gName == '@' ) )
{
if ( *gName )
gName++;
return composeDeclaration ( symbolName );
} // End of ELIF then
else
return DN_invalid;
} // End of IF then
elif ( *gName )
return DN_invalid;
else
return DN_truncated;
} // End of "UnDecorator" FUNCTION "getDecoratedName"
inline DName __near UnDecorator::getSymbolName ( void )
{
if ( *gName == '?' )
{
gName++;
return getOperatorName ();
} // End of IF then
else
return getZName ();
} // End of "UnDecorator" FUNCTION "getSymbolName"
DName __near UnDecorator::getZName ( void )
{
int zNameIndex = *gName - '0';
// Handle 'zname-replicators', otherwise an actual name
if (( zNameIndex >= 0 ) && ( zNameIndex <= 9 ))
{
gName++; // Skip past the replicator
// And return the indexed name
return ( *pZNameList )[ zNameIndex ];
} // End of IF then
else
{
DName zName;
if ( *gName == '?' )
{
zName = getTemplateName ();
if ( *gName++ != '@' )
zName = *--gName ? DN_invalid : DN_truncated;
}
else
// Extract the 'zname' to the terminator
zName = DName( gName, '@' ); // This constructor updates 'name'
// Add it to the current list of 'zname's
if ( !pZNameList->isFull ())
*pZNameList += zName;
// And return the symbol name
return zName;
} // End of IF else
} // End of "UnDecorator" FUNCTION "getZName"
inline DName __near UnDecorator::getOperatorName ( void )
{
DName operatorName;
DName tmpName;
int udcSeen = FALSE;
// So what type of operator is it ?
switch ( *gName++ )
{
case 0:
gName--; // End of string, better back-track
return DN_truncated;
case OC_ctor:
case OC_dtor:
//
// The constructor and destructor are special:
// Their operator name is the name of their first enclosing scope, which
// will always be a tag, which may be a template specialization!
//
{
// Use a temporary. Don't want to advance the name pointer
pcchar_t pName = gName;
operatorName = getZName ();
gName = pName; // Undo our lookahead
if ( !operatorName.isEmpty () && ( gName[ -1 ] == OC_dtor ))
operatorName = '~' + operatorName;
return operatorName;
} // End of CASE 'OC_ctor,OC_dtor'
break;
case OC_new:
case OC_delete:
case OC_assign:
case OC_rshift:
case OC_lshift:
case OC_not:
case OC_equal:
case OC_unequal:
operatorName = nameTable[ gName[ -1 ] - OC_new ];
break;
case OC_udc:
udcSeen = TRUE;
// No break
case OC_index:
case OC_pointer:
case OC_star:
case OC_incr:
case OC_decr:
case OC_minus:
case OC_plus:
case OC_amper:
case OC_ptrmem:
case OC_divide:
case OC_modulo:
case OC_less:
case OC_leq:
case OC_greater:
case OC_geq:
case OC_comma:
case OC_call:
case OC_compl:
case OC_xor:
case OC_or:
case OC_land:
case OC_lor:
case OC_asmul:
case OC_asadd:
case OC_assub: // Regular operators from the first group
operatorName = nameTable[ gName[ -1 ] - OC_index + ( OC_unequal - OC_new + 1 )];
break;
case '_':
switch ( *gName++ )
{
case 0:
gName--; // End of string, better back-track
return DN_truncated;
case OC_asdiv:
case OC_asmod:
case OC_asrshift:
case OC_aslshift:
case OC_asand:
case OC_asor:
case OC_asxor: // Regular operators from the extended group
operatorName = nameTable[ gName[ -1 ] - OC_asdiv + ( OC_assub - OC_index + 1 ) + ( OC_unequal - OC_new + 1 )];
break;
#if ( !NO_COMPILER_NAMES )
case OC_vftable:
case OC_vbtable:
case OC_vcall:
return nameTable[ gName[ -1 ] - OC_asdiv + ( OC_assub - OC_index + 1 ) + ( OC_unequal - OC_new + 1 )];
case OC_metatype:
case OC_guard:
case OC_uctor:
case OC_udtor:
case OC_vdeldtor:
case OC_defctor:
case OC_sdeldtor:
case OC_vctor:
case OC_vdtor:
case OC_vallctor:
case OC_ehvctor:
case OC_ehvdtor:
case OC_ehvctorvb:
case OC_copyctorclosure:
case OC_locvfctorclosure:
case OC_locvftable: // Special purpose names
case OC_placementDeleteClosure:
case OC_placementArrayDeleteClosure:
return nameTable[ gName[ -1 ] - OC_metatype + ( OC_vcall - OC_asdiv + 1 ) + ( OC_assub - OC_index + 1 ) + ( OC_unequal - OC_new + 1 )];
case OC_udtthunk:
operatorName = nameTable[ gName[ -1 ] - OC_metatype + ( OC_vcall - OC_asdiv + 1 ) + ( OC_assub - OC_index + 1 ) + ( OC_unequal - OC_new + 1 )];
tmpName = getOperatorName();
if ( !tmpName.isEmpty() && tmpName.isUDTThunk() )
return DN_invalid;
return operatorName + tmpName;
break;
case OC_eh_init:
break;
case OC_rtti_init:
operatorName = nameTable[ gName[ -1 ] - OC_metatype + ( OC_vcall - OC_asdiv + 1 ) + ( OC_assub - OC_index + 1 ) + ( OC_unequal - OC_new + 1 )];
tmpName = rttiTable[ gName[0] - OC_rtti_TD ];
switch ( *gName++ )
{
case OC_rtti_TD:
{
DName result = getDataType ( NULL );
return result + ' ' + operatorName + tmpName;
}
break;
case OC_rtti_BCD:
{
DName result = operatorName + tmpName;
result += getSignedDimension() + ',';
result += getSignedDimension() + ',';
result += getSignedDimension() + ',';
result += getDimension() + ')';
return result + '\'';
}
break;
case OC_rtti_BCA:
case OC_rtti_CHD:
case OC_rtti_COL:
return operatorName + tmpName;
break;
default:
gName--;
return DN_truncated;
break;
}
break;
#endif // !NO_COMPILER_NAMES
case OC_arrayNew:
case OC_arrayDelete:
operatorName = DName(nameTable[ gName[ -1 ] - OC_new ]) + "[]";
break;
default:
return DN_invalid;
} // End of SWITCH
break;
default:
return DN_invalid;
} // End of SWITCH
// This really is an operator name, so prefix it with 'operator'
if ( udcSeen )
operatorName.setIsUDC ();
elif ( !operatorName.isEmpty ())
operatorName = "operator" + operatorName;
return operatorName;
} // End of "UnDecorator" FUNCTION "getOperatorName"
DName __near UnDecorator::getScope ( void )
{
DName scope;
// Get the list of scopes
while (( scope.status () == DN_valid ) && *gName && ( *gName != '@' ))
{ // Insert the scope operator if not the first scope
if ( !scope.isEmpty() )
scope = "::" + scope;
// Determine what kind of scope it is
if ( *gName == '?' )
switch ( *++gName )
{
case '?':
if ( !doNameOnly() )
scope = '`' + getDecoratedName () + '\'' + scope;
else
getDecoratedName(); // Skip lexical scope info
break;
case '$':
// It's a template name, which is a kind of zname; back up
// and handle like a zname.
gName--;
scope = getZName () + scope;
break;
case '%':
//
// It an anonymous namespace, skip the (unreadable) name and instead insert
// an appropriate string
//
while ( *gName != '@' ) {
gName++;
}
gName++;
scope = "`anonymous namespace'" + scope;
break;
default:
if ( !doNameOnly() )
scope = getLexicalFrame () + scope;
else
getLexicalFrame(); // Skip lexical scope info
break;
} // End of SWITCH
else
scope = getZName () + scope;
} // End of WHILE
// Catch error conditions
switch ( *gName )
{
case 0:
if ( scope.isEmpty() )
scope = DN_truncated;
else
scope = DName ( DN_truncated ) + "::" + scope;
break;
case '@': // '@' expected to end the scope list
break;
default:
scope = DN_invalid;
break;
} // End of SWITCH
// Return the composed scope
return scope;
} // End of "UnDecorator" FUNCTION "getScope"
DName __near UnDecorator::getSignedDimension ( void )
{
if ( !*gName )
return DN_truncated;
elif ( *gName == '?' ) {
gName++; // skip the '?'
return '-' + getDimension();
}
else
return getDimension();
} // End of "Undecorator" FUNCTION "getSignedDimension"
DName __near UnDecorator::getDimension ( void )
{
if ( !*gName )
return DN_truncated;
elif (( *gName >= '0' ) && ( *gName <= '9' ))
return DName ((unsigned long)( *gName++ - '0' + 1 ));
else
{
unsigned long dim = 0L;
// Don't bother detecting overflow, it's not worth it
while ( *gName != '@' )
{
if ( !*gName )
return DN_truncated;
elif (( *gName >= 'A' ) && ( *gName <= 'P' ))
dim = ( dim << 4 ) + ( *gName - 'A' );
else
return DN_invalid;
gName++;
} // End of WHILE
// Ensure integrity, and return
if ( *gName++ != '@' )
return DN_invalid; // Should never get here
return dim;
} // End of ELIF else
} // End of "UnDecorator" FUNCTION "getDimension"
int __near UnDecorator::getNumberOfDimensions ( void )
{
if ( !*gName )
return 0;
elif (( *gName >= '0' ) && ( *gName <= '9' ))
return (( *gName++ - '0' ) + 1 );
else
{
int dim = 0;
// Don't bother detecting overflow, it's not worth it
while ( *gName != '@' )
{
if ( !*gName )
return 0;
elif (( *gName >= 'A' ) && ( *gName <= 'P' ))
dim = ( dim << 4 ) + ( *gName - 'A' );
else
return -1;
gName++;
} // End of WHILE
// Ensure integrity, and return
if ( *gName++ != '@' )
return -1; // Should never get here
return dim;
} // End of ELIF else
} // End of "UnDecorator" FUNCTION "getNumberOfDimensions"
DName __near UnDecorator::getTemplateName ( void )
{
//
// First make sure we're really looking at a template name
//
if ( gName[0] != '?' || gName[1] != '$' )
return DN_invalid;
gName += 2; // Skip the marker characters
//
// Stack the replicators, since template names are their own replicator scope:
//
Replicator * pSaveArgList = pArgList;
Replicator * pSaveZNameList = pZNameList;
Replicator * pSaveTemplateArgList = pTemplateArgList;
Replicator localArgList, localZNameList, localTemplateArgList;
pArgList = &localArgList;
pZNameList = &localZNameList;
pTemplateArgList = &localTemplateArgList;
//
// Crack the template name:
//
DName templateName = getZName ();
if ( !templateName.isEmpty ())
templateName += '<' + getTemplateArgumentList () + '>';
//
// Restore the previous replicators:
//
pArgList = pSaveArgList;
pZNameList = pSaveZNameList;
pTemplateArgList = pSaveTemplateArgList;
// Return the completed 'template-name'
return templateName;
} // End of "UnDecorator" FUNCTION "getTemplateName"
DName __near UnDecorator::getTemplateArgumentList ( void )
{
int first = TRUE;
DName aList;
while (( aList.status () == DN_valid ) && *gName && ( *gName != AT_endoflist ))
{
// Insert the argument list separator if not the first argument
if ( first )
first = FALSE;
else
aList += ',';
// Get the individual argument type
int argIndex = *gName - '0';
// Handle 'template-argument-replicators', otherwise a new argument type
if (( argIndex >= 0 ) && ( argIndex <= 9 ))
{
gName++; // Skip past the replicator
// Append to the argument list
aList += ( *pTemplateArgList )[ argIndex ];
} // End of IF then
else
{
pcchar_t oldGName = gName;
DName arg;
//
// Extract the 'argument' type
//
if ( *gName == DT_void ) {
gName++;
arg = "void";
}
elif ( (*gName == '$') && (gName[1] != '$')) {
gName++;
arg = getTemplateConstant();
}
elif ( *gName == '?' ) {
//
// This is a template-parameter, i.e. we have a "specialization" of
// X<T>. so get the template-parameter-index and use a "generic" name
// for this parameter
//
arg = "`template-parameter" + getSignedDimension() +"'";
}
else {
arg = getPrimaryDataType ( DName() );
}
// Add it to the current list of 'template-argument's, if it is bigger than a one byte encoding
if ((( gName - oldGName ) > 1 ) && !pTemplateArgList->isFull ())
*pTemplateArgList += arg;
// Append to the argument list
aList += arg;
} // End of IF else
} // End of WHILE
// Return the completed template argument list
return aList;
} // End of "UnDecorator" FUNCTION "getTemplateArgumentList"
DName __near UnDecorator::getTemplateConstant(void)
{
//
// template-constant ::=
// '0' <template-integral-constant>
// '1' <template-address-constant>
// '2' <template-floating-point-constant>
//
switch ( *gName++ )
{
//
// template-integral-constant ::=
// <signed-dimension>
//
case TC_integral:
return getSignedDimension ();
//
// template-address-constant ::=
// '@' // Null pointer
// <decorated-name>
//
case TC_address:
if ( *gName == TC_nullptr )
return "NULL";
else
return getDecoratedName ();
//
// template-floating-point-constant ::=
// <normalized-mantissa><exponent>
//
case TC_fp:
{
DName mantissa ( getSignedDimension () );
DName exponent ( getSignedDimension () );
if ( mantissa.isValid() && exponent.isValid() )
{
//
// Get string representation of mantissa
//
char buf[100]; // Way overkill for a compiler generated fp constant
if ( !mantissa.getString( &(buf[1]), 100 ) )
return DN_invalid;
//
// Insert decimal point
//
buf[0] = buf[1];
if ( buf[0] == '-' )
{
buf[1] = buf[2];
buf[2] = '.';
}
else
buf[1] = '.';
//
// String it all together
//
return DName( buf ) + 'e' + exponent;
} // End of IF then
else
return DN_truncated;
} // End of BLOCK case TC_fp
case '\0':
--gName;
return DN_truncated;
default:
return DN_invalid;
} // End of SWITCH
} // End of "UnDecorator" FUNCTION "getTemplateConstant"
inline DName __near UnDecorator::composeDeclaration ( const DName & symbol )
{
DName declaration;
unsigned int typeCode = getTypeEncoding ();
int symIsUDC = symbol.isUDC ();
// Handle bad typeCode's, or truncation
if ( TE_isbadtype ( typeCode ))
return DN_invalid;
elif ( TE_istruncated ( typeCode ))
return ( DN_truncated + symbol );
// This is a very complex part. The type of the declaration must be
// determined, and the exact composition must be dictated by this type.
// Is it any type of a function ?
// However, for ease of decoding, treat the 'localdtor' thunk as data, since
// its decoration is a function of the variable to which it belongs and not
// a usual function type of decoration.
#if ( NO_COMPILER_NAMES )
if ( TE_isthunk ( typeCode ))
return DN_invalid;
if ( TE_isfunction ( typeCode ))
#else // } elif !NO_COMPILER_NAMES {
if ( TE_isfunction ( typeCode ) && !(( TE_isthunk ( typeCode ) && TE_islocaldtor ( typeCode )) ||
( TE_isthunk ( typeCode ) && ( TE_istemplatector ( typeCode ) || TE_istemplatedtor ( typeCode )))))
#endif // !NO_COMPILER_NAMES
{
// If it is based, then compose the 'based' prefix for the name
if ( TE_isbased ( typeCode ))
if ( doMSKeywords () && doAllocationModel ())
declaration = ' ' + getBasedType ();
else
declaration |= getBasedType (); // Just lose the 'based-type'
#if ( !NO_COMPILER_NAMES )
// Check for some of the specially composed 'thunk's
if ( TE_isthunk ( typeCode ) && TE_isvcall ( typeCode ))
{
declaration += symbol + '{' + getCallIndex () + ',';
declaration += getVCallThunkType () + "}' ";
if ( doMSKeywords () && doAllocationLanguage ())
declaration = ' ' + getCallingConvention () + ' ' + declaration; // What calling convention ?
else
declaration |= getCallingConvention (); // Just lose the 'calling-convention'
} // End of IF then
else
#endif // !NO_COMPILER_NAMES
{
DName vtorDisp;
DName adjustment;
DName thisType;
#if ( !NO_COMPILER_NAMES )
if ( TE_isthunk ( typeCode ))
{
if ( TE_isvtoradj ( typeCode ))
vtorDisp = getDisplacement ();
adjustment = getDisplacement ();
} // End of IF else
#endif // !NO_COMPILER_NAMES
// Get the 'this-type' for non-static function members
if ( TE_ismember ( typeCode ) && !TE_isstatic ( typeCode ))
if ( doThisTypes ())
thisType = getThisType ();
else
thisType |= getThisType ();
if ( doMSKeywords ())
{
// Attach the calling convention
if ( doAllocationLanguage ())
declaration = getCallingConvention () + declaration; // What calling convention ?
else
declaration |= getCallingConvention (); // Just lose the 'calling-convention'
// Any model specifiers ?
#if !VERS_32BIT
if ( doAllocationModel ())
if ( TE_isnear ( typeCode ))
declaration = UScore ( TOK_nearSp ) + declaration;
elif ( TE_isfar ( typeCode ))
declaration = UScore ( TOK_farSp ) + declaration;
#endif
} // End of IF
else
declaration |= getCallingConvention (); // Just lose the 'calling-convention'
// Now put them all together
if ( !symbol.isEmpty ())
if ( !declaration.isEmpty () && !doNameOnly() ) // And the symbol name
declaration += ' ' + symbol;
else
declaration = symbol;
// Compose the return type, catching the UDC case
DName * pDeclarator = 0;
DName returnType;
if ( symIsUDC ) // Is the symbol a UDC operator ?
{
declaration += " " + getReturnType ();
if ( doNameOnly() )
return declaration;
}
else
{
pDeclarator = gnew DName;
returnType = getReturnType ( pDeclarator );
} // End of IF else
#if ( !NO_COMPILER_NAMES )
// Add the displacements for virtual function thunks
if ( TE_isthunk ( typeCode ))
{
if ( TE_isvtoradj ( typeCode ))
declaration += "`vtordisp{" + vtorDisp + ',';
else
declaration += "`adjustor{";
declaration += adjustment + "}' ";
} // End of IF
#endif // !NO_COMPILER_NAMES
// Add the function argument prototype
declaration += '(' + getArgumentTypes () + ')';
// If this is a non-static member function, append the 'this' modifiers
if ( TE_ismember ( typeCode ) && !TE_isstatic ( typeCode ))
declaration += thisType;
// Add the 'throw' signature
if ( doThrowTypes ())
declaration += getThrowTypes ();
else
declaration |= getThrowTypes (); // Just lose the 'throw-types'
// If it has a declarator, then insert it into the declaration,
// sensitive to the return type composition
if ( doFunctionReturns () && pDeclarator )
{
*pDeclarator = declaration;
declaration = returnType;
} // End of IF
} // End of IF else
} // End of IF then
else
{
declaration += symbol;
// Catch the special handling cases
#if ( !NO_COMPILER_NAMES )
if ( TE_isvftable ( typeCode ))
return getVfTableType ( declaration );
elif ( TE_isvbtable ( typeCode ))
return getVbTableType ( declaration );
elif ( TE_isguard ( typeCode ))
return ( declaration + '{' + getGuardNumber () + "}'" );
elif ( TE_isthunk ( typeCode ) && TE_islocaldtor ( typeCode ))
declaration += "`local static destructor helper'";
elif ( TE_isthunk ( typeCode ) && TE_istemplatector ( typeCode ))
declaration += "`template static data member constructor helper'";
elif ( TE_isthunk ( typeCode ) && TE_istemplatedtor ( typeCode ))
declaration += "`template static data member destructor helper'";
elif ( TE_ismetaclass ( typeCode ))
//
// Meta-class information has its information in its operator id
//
return declaration;
#else // } elif NO_COMPILER_NAMES {
if ( TE_isvftable ( typeCode )
|| TE_isvbtable ( typeCode )
|| TE_isguard ( typeCode )
|| TE_ismetaclass ( typeCode ))
return DN_invalid;
#endif // NO_COMPILER_NAMES
if ( TE_isthunk( typeCode ) && ( TE_istemplatector( typeCode ) || TE_istemplatedtor( typeCode ))) {
//
// Insert a space before the declaration
//
declaration = " " + declaration;
}
else {
// All others are decorated as data symbols
declaration = getExternalDataType ( declaration );
}
} // End of IF else
// Prepend the 'virtual' and 'static' attributes for members
if ( TE_ismember ( typeCode ))
{
if ( doMemberTypes ())
{
if ( TE_isstatic ( typeCode ))
declaration = "static " + declaration;
if ( TE_isvirtual ( typeCode ) || ( TE_isthunk ( typeCode ) && ( TE_isvtoradj ( typeCode ) || TE_isadjustor ( typeCode ))))
declaration = "virtual " + declaration;
} // End of IF
// Prepend the access specifiers
if ( doAccessSpecifiers ())
if ( TE_isprivate ( typeCode ))
declaration = "private: " + declaration;
elif ( TE_isprotected ( typeCode ))
declaration = "protected: " + declaration;
elif ( TE_ispublic ( typeCode ))
declaration = "public: " + declaration;
} // End of IF
#if ( !NO_COMPILER_NAMES )
// If it is a thunk, mark it appropriately
if ( TE_isthunk ( typeCode ))
declaration = "[thunk]:" + declaration;
#endif // !NO_COMPILER_NAMES
// Return the composed declaration
return declaration;
} // End of "UnDecorator" FUNCTION "composeDeclaration"
inline int __near UnDecorator::getTypeEncoding ( void )
{
unsigned int typeCode = 0u;
// Strip any leading '_' which indicates that it is based
if ( *gName == '_' )
{
TE_setisbased ( typeCode );
gName++;
} // End of IF
// Now handle the code proper :-
if (( *gName >= 'A' ) && ( *gName <= 'Z' )) // Is it some sort of function ?
{
int code = *gName++ - 'A';
// Now determine the function type
TE_setisfunction ( typeCode ); // All of them are functions ?
// Determine the calling model
if ( code & TE_far )
TE_setisfar ( typeCode );
else
TE_setisnear ( typeCode );
// Is it a member function or not ?
if ( code < TE_external )
{
// Record the fact that it is a member
TE_setismember ( typeCode );
// What access permissions does it have
switch ( code & TE_access )
{
case TE_private:
TE_setisprivate ( typeCode );
break;
case TE_protect:
TE_setisprotected ( typeCode );
break;
case TE_public:
TE_setispublic ( typeCode );
break;
default:
TE_setisbadtype ( typeCode );
return typeCode;
} // End of SWITCH
// What type of a member function is it ?
switch ( code & TE_adjustor )
{
case TE_adjustor:
TE_setisadjustor ( typeCode );
break;
case TE_virtual:
TE_setisvirtual ( typeCode );
break;
case TE_static:
TE_setisstatic ( typeCode );
break;
case TE_member:
break;
default:
TE_setisbadtype ( typeCode );
return typeCode;
} // End of SWITCH
} // End of IF
} // End of IF then
elif ( *gName == '$' ) // Extended set ? Special handling
{
// What type of symbol is it ?
switch ( *( ++gName ))
{
case SHF_localdtor: // A destructor helper for a local static ?
TE_setislocaldtor ( typeCode );
break;
case SHF_vcall: // A VCall-thunk ?
TE_setisvcall ( typeCode );
break;
case SHF_templateStaticDataMemberCtor: // A constructor helper for template static data members
TE_setistemplatector ( typeCode );
break;
case SHF_templateStaticDataMemberDtor: // A destructor helper for template static data members
TE_setistemplatedtor ( typeCode );
break;
case 0:
TE_setistruncated ( typeCode );
break;
case '0':
case '1':
case '2':
case '3':
case '4':
case '5': // Construction displacement adjustor thunks
{
int code = *gName - '0';
// Set up the principal type information
TE_setisfunction ( typeCode );
TE_setismember ( typeCode );
TE_setisvtoradj ( typeCode );
// Is it 'near' or 'far' ?
if ( code & TE_far )
TE_setisfar ( typeCode );
else
TE_setisnear ( typeCode );
// What type of access protection ?
switch ( code & TE_access_vadj )
{
case TE_private_vadj:
TE_setisprivate ( typeCode );
break;
case TE_protect_vadj:
TE_setisprotected ( typeCode );
break;
case TE_public_vadj:
TE_setispublic ( typeCode );
break;
default:
TE_setisbadtype ( typeCode );
return typeCode;
} // End of SWITCH
} // End of CASE '0,1,2,3,4,5'
break;
default:
TE_setisbadtype ( typeCode );
return typeCode;
} // End of SWITCH
// Advance past the code character
gName++;
} // End of ELIF then
elif (( *gName >= TE_static_d ) && ( *gName <= TE_metatype )) // Non function decorations ?
{
int code = *gName++;
TE_setisdata ( typeCode );
// What type of symbol is it ?
switch ( code )
{
case ( TE_static_d | TE_private_d ):
TE_setisstatic ( typeCode );
TE_setisprivate ( typeCode );
break;
case ( TE_static_d | TE_protect_d ):
TE_setisstatic ( typeCode );
TE_setisprotected ( typeCode );
break;
case ( TE_static_d | TE_public_d ):
TE_setisstatic ( typeCode );
TE_setispublic ( typeCode );
break;
case TE_global:
TE_setisglobal ( typeCode );
break;
case TE_guard:
TE_setisguard ( typeCode );
break;
case TE_local:
TE_setislocal ( typeCode );
break;
case TE_vftable:
TE_setisvftable ( typeCode );
break;
case TE_vbtable:
TE_setisvbtable ( typeCode );
break;
case TE_metatype:
TE_setismetaclass ( typeCode );
break;
default:
TE_setisbadtype ( typeCode );
return typeCode;
} // End of SWITCH
} // End of ELIF then
elif ( *gName )
TE_setisbadtype ( typeCode );
else
TE_setistruncated ( typeCode );
// Return the composed type code
return typeCode;
} // End of "UnDecorator" FUNCTION "getTypeEncoding"
DName __near UnDecorator::getBasedType ( void )
{
DName basedDecl ( UScore ( TOK_basedLp ));
// What type of 'based' is it ?
if ( *gName )
{
switch ( *gName++ )
{
#if !VERS_32BIT
case BT_segname:
basedDecl += UScore ( TOK_segnameLpQ ) + getSegmentName () + "\")";
break;
case BT_segment:
basedDecl += DName ( "NYI:" ) + UScore ( TOK_segment );
break;
#endif
case BT_void:
basedDecl += "void";
break;
#if !VERS_32BIT
case BT_self:
basedDecl += UScore ( TOK_self );
break;
case BT_nearptr:
basedDecl += DName ( "NYI:" ) + UScore ( TOK_nearP );
break;
case BT_farptr:
basedDecl += DName ( "NYI:" ) + UScore ( TOK_farP );
break;
case BT_hugeptr:
basedDecl += DName ( "NYI:" ) + UScore ( TOK_hugeP );
break;
case BT_segaddr:
basedDecl += "NYI:<segment-address-of-variable>";
break;
#else
case BT_nearptr:
basedDecl += getScopedName();
break;
#endif
case BT_basedptr:
//
// Note: based pointer on based pointer is reserved
//
return DN_invalid;
} // End of SWITCH
} // End of IF else
else
basedDecl += DN_truncated;
// Close the based syntax
basedDecl += ") ";
// Return completed based declaration
return basedDecl;
} // End of "UnDecorator" FUNCTION "getBasedType"
DName __near UnDecorator::getScopedName ( void )
{
DName name;
// Get the beginning of the name
name = getZName ();
// Now the scope (if any)
if (( name.status () == DN_valid ) && *gName && ( *gName != '@' ))
name = getScope () + "::" + name;
// Skip the trailing '@'
if ( *gName == '@' )
gName++;
elif ( *gName )
name = DN_invalid;
elif ( name.isEmpty ())
name = DN_truncated;
else
name = DName ( DN_truncated ) + "::" + name;
// And return the complete name
return name;
} // End of "UnDecorator" FUNCTION "getECSUName"
inline DName UnDecorator::getECSUName ( void ) { return getScopedName(); }
inline DName __near UnDecorator::getEnumName ( void )
{
DName ecsuName;
if ( *gName )
{
// What type of an 'enum' is it ?
switch ( *gName )
{
case ET_schar:
case ET_uchar:
ecsuName = "char ";
break;
case ET_sshort:
case ET_ushort:
ecsuName = "short ";
break;
case ET_sint:
break;
case ET_uint:
ecsuName = "int ";
break;
case ET_slong:
case ET_ulong:
ecsuName = "long ";
break;
default:
return DN_invalid;
} // End of SWITCH
// Add the 'unsigned'ness if appropriate
switch ( *gName++ )
{
case ET_uchar:
case ET_ushort:
case ET_uint:
case ET_ulong:
ecsuName = "unsigned " + ecsuName;
break;
} // End of SWITCH
// Now return the composed name
return ecsuName + getECSUName ();
} // End of IF then
else
return DN_truncated;
} // End of "UnDecorator" FUNCTION "getEnumName"
DName __near UnDecorator::getCallingConvention ( void )
{
if ( *gName )
{
unsigned int callCode = ((unsigned int)*gName++ ) - 'A';
// What is the primary calling convention
if (( callCode >= CC_cdecl ) && ( callCode <= CC_interrupt ))
{
DName callType;
// Now, what type of 'calling-convention' is it, 'interrupt' is special ?
if ( doMSKeywords ())
#if !VERS_32BIT
if ( callCode == CC_interrupt )
callType = UScore ( TOK_interrupt );
else
#endif
{
switch ( callCode & ~CC_saveregs )
{
case CC_cdecl:
callType = UScore ( TOK_cdecl );
break;
case CC_pascal:
callType = UScore ( TOK_pascal );
break;
case CC_thiscall:
callType = UScore ( TOK_thiscall );
break;
case CC_stdcall:
callType = UScore ( TOK_stdcall );
break;
case CC_fastcall:
callType = UScore ( TOK_fastcall );
break;
} // End of SWITCH
// Has it also got 'saveregs' marked ?
#if !VERS_32BIT
if ( callCode & CC_saveregs )
callType += ' ' + UScore ( TOK_saveregs );
#endif
} // End of IF else
// And return
return callType;
} // End of IF then
else
return DN_invalid;
} // End of IF then
else
return DN_truncated;
} // End of "UnDecorator" FUNCTION "getCallingConvention"
DName __near UnDecorator::getReturnType ( DName * pDeclarator )
{
if ( *gName == '@' ) // Return type for constructors and destructors ?
{
gName++;
return DName ( pDeclarator );
} // End of IF then
else
return getDataType ( pDeclarator );
} // End of "UnDecorator" FUNCTION "getReturnType"
DName __near UnDecorator::getDataType ( DName * pDeclarator )
{
DName superType ( pDeclarator );
// What type is it ?
switch ( *gName )
{
case 0:
return ( DN_truncated + superType );
case DT_void:
gName++;
if ( superType.isEmpty ())
return "void";
else
return "void " + superType;
case '?':
{
gName++; // Skip the '?'
superType = getDataIndirectType ( superType, 0, DName (), 0);
return getPrimaryDataType ( superType );
return superType;
} // End of CASE '?'
default:
return getPrimaryDataType ( superType );
} // End of SWITCH
} // End of "UnDecorator" FUNCTION "getDataType"
DName __near UnDecorator::getPrimaryDataType ( const DName & superType )
{
DName cvType;
switch ( *gName )
{
case 0:
return ( DN_truncated + superType );
case PDT_volatileReference:
cvType = "volatile";
if ( !superType.isEmpty ())
cvType += ' ';
// No break
case PDT_reference:
{
DName super ( superType );
gName++;
return getReferenceType ( cvType, super.setPtrRef ());
} // End of CASE 'PDT_reference'
case PDT_extend:
{
//
// Extended Primary Data Type (items overlooked in original design):
// prefixed by '$$'.
//
if ( gName[1] != PDT_extend )
if ( gName[1] == '\0' )
return DN_truncated + superType;
else
return DN_invalid;
gName += 2;
switch ( *gName )
{
case PDT_ex_function:
gName++;
return getFunctionIndirectType( superType );
case PDT_ex_other:
gName++;
return getPtrRefDataType( superType, /* isPtr = */ TRUE );
case PDT_ex_qualified:
gName++;
return(getBasicDataType(getDataIndirectType ( superType, 0, DName (), 0)));
case 0:
return ( DN_truncated + superType );
default:
return DN_invalid;
}
}
default:
return getBasicDataType ( superType );
} // End of SWITCH
} // End of "UnDecorator" FUNCTION "getPrimaryDataType"
DName __near UnDecorator::getArgumentTypes ( void )
{
switch ( *gName )
{
case AT_ellipsis:
return ( gName++, "..." );
case AT_void:
return ( gName++, "void" );
default:
{
DName arguments ( getArgumentList ());
// Now, is it a varargs function or not ?
if ( arguments.status () == DN_valid )
switch ( *gName )
{
case 0:
return arguments;
case AT_ellipsis:
return ( gName++, arguments + ",..." );
case AT_endoflist:
return ( gName++, arguments );
default:
return DN_invalid;
} // End of SWITCH
else
return arguments;
} // End of DEFAULT
} // End of SWITCH
} // End of "UnDecorator" FUNCTION "getArgumentTypes"
DName __near UnDecorator::getArgumentList ( void )
{
int first = TRUE;
DName aList;
while (( aList.status () == DN_valid ) && ( *gName != AT_endoflist ) && ( *gName != AT_ellipsis ))
{
// Insert the argument list separator if not the first argument
if ( first )
first = FALSE;
else
aList += ',';
// Get the individual argument type
if ( *gName )
{
int argIndex = *gName - '0';
// Handle 'argument-replicators', otherwise a new argument type
if (( argIndex >= 0 ) && ( argIndex <= 9 ))
{
gName++; // Skip past the replicator
// Append to the argument list
aList += ( *pArgList )[ argIndex ];
} // End of IF then
else
{
pcchar_t oldGName = gName;
// Extract the 'argument' type
DName arg ( getPrimaryDataType ( DName ()));
// Add it to the current list of 'argument's, if it is bigger than a one byte encoding
if ((( gName - oldGName ) > 1 ) && !pArgList->isFull ())
*pArgList += arg;
// Append to the argument list
aList += arg;
} // End of IF else
} // End of IF then
else
{
aList += DN_truncated;
break;
} // End of IF else
} // End of WHILE
// Return the completed argument list
return aList;
} // End of "UnDecorator" FUNCTION "getArgumentList"
DName __near UnDecorator::getThrowTypes ( void )
{
if ( *gName )
if ( *gName == AT_ellipsis ) // Handle ellipsis here to suppress the 'throw' signature
return ( gName++, DName ());
else
return ( " throw(" + getArgumentTypes () + ')' );
else
return ( DName ( " throw(" ) + DN_truncated + ')' );
} // End of "UnDecorator" FUNCTION "getThrowTypes"
DName __near UnDecorator::getBasicDataType ( const DName & superType )
{
if ( *gName )
{
unsigned char bdtCode = *gName++;
unsigned char extended_bdtCode;
int pCvCode = -1;
DName basicDataType;
// Extract the principal type information itself, and validate the codes
switch ( bdtCode )
{
case BDT_schar:
case BDT_char:
case ( BDT_char | BDT_unsigned ):
basicDataType = "char";
break;
case BDT_short:
case ( BDT_short | BDT_unsigned ):
basicDataType = "short";
break;
case BDT_int:
case ( BDT_int | BDT_unsigned ):
basicDataType = "int";
break;
case BDT_long:
case ( BDT_long | BDT_unsigned ):
basicDataType = "long";
break;
#if !VERS_32BIT
case BDT_segment:
basicDataType = UScore ( TOK_segment );
break;
#endif
case BDT_float:
basicDataType = "float";
break;
case BDT_longdouble:
basicDataType = "long ";
// No break
case BDT_double:
basicDataType += "double";
break;
case BDT_pointer:
case ( BDT_pointer | BDT_const ):
case ( BDT_pointer | BDT_volatile ):
case ( BDT_pointer | BDT_const | BDT_volatile ):
pCvCode = ( bdtCode & ( BDT_const | BDT_volatile ));
break;
case BDT_extend:
switch(extended_bdtCode = *gName++) {
case BDT_bool:
basicDataType = "bool";
break;
case BDT_int8:
case ( BDT_int8 | BDT_unsigned ):
basicDataType = "__int8";
break;
case BDT_int16:
case ( BDT_int16 | BDT_unsigned ):
basicDataType = "__int16";
break;
case BDT_int32:
case ( BDT_int32 | BDT_unsigned ):
basicDataType = "__int32";
break;
case BDT_int64:
case ( BDT_int64 | BDT_unsigned ):
basicDataType = "__int64";
break;
case BDT_int128:
case ( BDT_int128 | BDT_unsigned ):
basicDataType = "__int128";
break;
case BDT_wchar_t:
basicDataType = "wchar_t";
break;
default:
basicDataType = "UNKNOWN";
break;
}
break;
default:
gName--; // Backup, since 'ecsu-data-type' does it's own decoding
basicDataType = getECSUDataType ();
if ( basicDataType.isEmpty ())
return basicDataType;
break;
} // End of SWITCH
// What type of basic data type composition is involved ?
if ( pCvCode == -1 ) // Simple ?
{
// Determine the 'signed/unsigned'ness
switch ( bdtCode )
{
case ( BDT_char | BDT_unsigned ):
case ( BDT_short | BDT_unsigned ):
case ( BDT_int | BDT_unsigned ):
case ( BDT_long | BDT_unsigned ):
basicDataType = "unsigned " + basicDataType;
break;
case BDT_schar:
basicDataType = "signed " + basicDataType;
break;
case BDT_extend:
switch ( extended_bdtCode )
{
case ( BDT_int8 | BDT_unsigned ):
case ( BDT_int16 | BDT_unsigned ):
case ( BDT_int32 | BDT_unsigned ):
case ( BDT_int64 | BDT_unsigned ):
case ( BDT_int128 | BDT_unsigned ):
basicDataType = "unsigned " + basicDataType;
break;
} // End of SWITCH
break;
} // End of SWITCH
// Add the indirection type to the type
if ( !superType.isEmpty ())
basicDataType += ' ' + superType;
// And return the completed type
return basicDataType;
} // End of IF then
else
{
DName cvType;
DName super ( superType );
// Is it 'const/volatile' qualified ?
if ( superType . isEmpty() )
{
//
// const/volatile are redundantly encoded, except at the start
// of a "type only" context. In such a context, the super-type
// is empty.
//
if ( pCvCode & BDT_const )
{
cvType = "const";
if ( pCvCode & BDT_volatile )
cvType += " volatile";
} // End of IF then
elif ( pCvCode & BDT_volatile )
cvType = "volatile";
} // End of IF then
// Construct the appropriate pointer type declaration
return getPointerType ( cvType, super );
} // End of IF else
} // End of IF then
else
return ( DN_truncated + superType );
} // End of "UnDecorator" FUNCTION "getBasicDataType"
DName __near UnDecorator::getECSUDataType ( int ecsuMods )
{
DName ecsuDataType;
// Get the 'model' modifiers if applicable
if ( ecsuMods )
if ( ecsuMods == ECSU_invalid )
return DN_invalid;
elif ( ecsuMods == ECSU_truncated )
ecsuDataType = DN_truncated;
else
switch ( ecsuMods & ECSU_modelmask )
{
#if !VERS_32BIT
case ECSU_near:
if ( doMSKeywords () && doReturnUDTModel ())
ecsuDataType = UScore ( TOK_nearSp );
break;
case ECSU_far:
if ( doMSKeywords () && doReturnUDTModel ())
ecsuDataType = UScore ( TOK_farSp );
break;
case ECSU_huge:
if ( doMSKeywords () && doReturnUDTModel ())
ecsuDataType = UScore ( TOK_hugeSp );
break;
#endif
case ECSU_based:
if ( doMSKeywords () && doReturnUDTModel ())
ecsuDataType = getBasedType ();
else
ecsuDataType |= getBasedType (); // Just lose the 'based-type'
break;
} // End of SWITCH
// Extract the principal type information itself, and validate the codes
switch ( *gName++ )
{
case 0:
gName--; // Backup to permit later error recovery to work safely
return "`unknown ecsu'" + ecsuDataType + DN_truncated;
case BDT_union:
if ( !doNameOnly() )
ecsuDataType = "union " + ecsuDataType;
break;
case BDT_struct:
if ( !doNameOnly() )
ecsuDataType = "struct " + ecsuDataType;
break;
case BDT_class:
if ( !doNameOnly() )
ecsuDataType = "class " + ecsuDataType;
break;
case BDT_enum:
return "enum " + ecsuDataType + getEnumName ();
// default:
// return DN_invalid;
} // End of SWITCH
// Get the UDT 'const/volatile' modifiers if applicable
if (ecsuMods & ECSU_volatile)
ecsuDataType = "volatile " + ecsuDataType;
if (ecsuMods & ECSU_const)
ecsuDataType = "const " + ecsuDataType;
// Get the 'class/struct/union' name
ecsuDataType += getECSUName ();
// And return the formed 'ecsu-data-type'
return ecsuDataType;
} // End of "UnDecorator" FUNCTION "getECSUDataType"
// Undecorator::getFunctionIndirectType
// Note: this function gets both the function-indirect-type and the function-type.
DName UnDecorator::getFunctionIndirectType( const DName & superType )
{
if ( ! *gName )
return DN_truncated + superType;
if ( ! IT_isfunction( *gName ))
return DN_invalid;
int fitCode = *gName++ - '6';
if ( fitCode == ( '_' - '6' ))
{
if ( *gName )
{
fitCode = *gName++ - 'A' + FIT_based;
if (( fitCode < FIT_based ) || ( fitCode > ( FIT_based | FIT_far | FIT_member )))
fitCode = -1;
} // End of IF then
else
return ( DN_truncated + superType );
} // End of IF then
elif (( fitCode < FIT_near ) || ( fitCode > ( FIT_far | FIT_member )))
fitCode = -1;
// Return if invalid name
if ( fitCode == -1 )
return DN_invalid;
// Otherwise, what are the function indirect attributes
DName thisType;
DName fitType = superType;
// Is it a pointer to member function ?
if ( fitCode & FIT_member )
{
fitType = "::" + fitType;
if ( *gName )
fitType = ' ' + getScope () + fitType;
else
fitType = DN_truncated + fitType;
if ( *gName )
if ( *gName == '@' )
gName++;
else
return DN_invalid;
else
return ( DN_truncated + fitType );
if ( doThisTypes ())
thisType = getThisType ();
else
thisType |= getThisType ();
} // End of IF
// Is it a based allocated function ?
if ( fitCode & FIT_based )
if ( doMSKeywords ())
fitType = ' ' + getBasedType () + fitType;
else
fitType |= getBasedType (); // Just lose the 'based-type'
// Get the 'calling-convention'
if ( doMSKeywords ())
{
fitType = getCallingConvention () + fitType;
// Is it a near or far function pointer
#if !VERS_32BIT
fitType = UScore ((( fitCode & FIT_far ) ? TOK_farSp : TOK_nearSp )) + fitType;
#endif
} // End of IF then
else
fitType |= getCallingConvention (); // Just lose the 'calling-convention'
// Parenthesise the indirection component, and work on the rest
if ( ! superType . isEmpty() ) {
fitType = '(' + fitType + ')';
}
// Get the rest of the 'function-type' pieces
DName * pDeclarator = gnew DName;
DName returnType ( getReturnType ( pDeclarator ));
fitType += '(' + getArgumentTypes () + ')';
if ( doThisTypes () && ( fitCode & FIT_member ))
fitType += thisType;
if ( doThrowTypes ())
fitType += getThrowTypes ();
else
fitType |= getThrowTypes (); // Just lose the 'throw-types'
// Now insert the indirected declarator, catch the allocation failure here
if ( pDeclarator )
*pDeclarator = fitType;
else
return DN_error;
// And return the composed function type (now in 'returnType' )
return returnType;
}
DName __near UnDecorator::getPtrRefType ( const DName & cvType, const DName & superType, int isPtr )
{
// Doubles up as 'pointer-type' and 'reference-type'
if ( *gName )
if ( IT_isfunction ( *gName )) // Is it a function or data indirection ?
{
DName fitType = ( isPtr ? '*' : '&' );
if ( !cvType.isEmpty () && ( superType.isEmpty () || !superType.isPtrRef ()))
fitType += cvType;
if ( !superType.isEmpty ())
fitType += superType;
return getFunctionIndirectType( fitType );
} // End of IF then
else
{
// Otherwise, it is either a pointer or a reference to some data type
DName innerType ( getDataIndirectType ( superType, ( isPtr ? '*' : '&' ), cvType ));
return getPtrRefDataType ( innerType, isPtr );
} // End of IF else
else
{
DName trunk ( DN_truncated );
trunk += ( isPtr ? '*' : '&' );
if ( !cvType.isEmpty ())
trunk += cvType;
if ( !superType.isEmpty ())
{
if ( !cvType.isEmpty ())
trunk += ' ';
trunk += superType;
} // End of IF
return trunk;
} // End of IF else
} // End of "UnDecorator" FUNCTION "getPtrRefType"
DName __near UnDecorator::getDataIndirectType ( const DName & superType, char prType, const DName & cvType, int thisFlag )
{
if ( *gName )
{
unsigned int ditCode = ( *gName - (( *gName >= 'A' ) ? (unsigned int)'A': (unsigned int)( '0' - 26 )));
DName msExtension;
if ( doMSKeywords ())
{
int fContinue = TRUE;
do
{
switch ( ditCode & DIT_modelmask )
{
case DIT_ptr64:
if ( !msExtension.isEmpty())
msExtension = msExtension + ' ' + UScore( TOK_ptr64 );
else
msExtension = UScore( TOK_ptr64 );
gName++;
ditCode = ( *gName - (( *gName >= 'A' ) ? (unsigned int)'A': (unsigned int)( '0' - 26 )));
break;
case DIT_restrict:
if ( !msExtension.isEmpty())
msExtension = msExtension + ' ' + UScore( TOK_restrict );
else
msExtension = UScore( TOK_restrict );
gName++;
ditCode = ( *gName - (( *gName >= 'A' ) ? (unsigned int)'A': (unsigned int)( '0' - 26 )));
break;
default:
fContinue = FALSE;
break;
}
} while (fContinue);
}
gName++; // Skip to next character in name
// Is it a valid 'data-indirection-type' ?
if (( ditCode >= DIT_near ) && ( ditCode <= ( DIT_const | DIT_volatile | DIT_modelmask | DIT_member )))
{
DName ditType ( prType );
if ( !msExtension.isEmpty())
ditType = ditType + ' ' + msExtension;
// If it is a member, then these attributes immediately precede the indirection token
if ( ditCode & DIT_member )
{
// If it is really 'this-type', then it cannot be any form of pointer to member
if ( thisFlag )
return DN_invalid;
// Otherwise, extract the scope for the PM
if ( prType != '\0' )
{
ditType = "::" + ditType;
if ( *gName )
ditType = getScope () + ditType;
else
ditType = DN_truncated + ditType;
}
elif ( *gName )
{
//
// The scope is ignored for special uses of data-indirect-type, such
// as storage-convention. I think it's a bug that we ever mark things
// with Member storage convention, as that is already covered in the
// scope of the name. However, we don't want to change the dname scheme,
// so we're stuck with it.
//
ditType |= getScope ();
}
// Now skip the scope terminator
if ( !*gName )
ditType += DN_truncated;
elif ( *gName++ != '@' )
return DN_invalid;
} // End of IF
// Add the 'model' attributes (prefixed) as appropriate
if ( doMSKeywords ()) {
switch ( ditCode & DIT_modelmask )
{
#if !VERS_32BIT
case DIT_near:
if ( do32BitNear ())
ditType = UScore ( TOK_near ) + ditType;
break;
case DIT_far:
ditType = UScore ( TOK_far ) + ditType;
break;
case DIT_huge:
ditType = UScore ( TOK_huge ) + ditType;
break;
#endif
case DIT_based:
// The 'this-type' can never be 'based'
if ( thisFlag )
return DN_invalid;
ditType = getBasedType () + ditType;
break;
} // End of SWITCH
} // End of IF
elif (( ditCode & DIT_modelmask ) == DIT_based )
ditType |= getBasedType (); // Just lose the 'based-type'
// Handle the 'const' and 'volatile' attributes
if ( ditCode & DIT_volatile )
ditType = "volatile " + ditType;
if ( ditCode & DIT_const )
ditType = "const " + ditType;
// Append the supertype, if not 'this-type'
if ( !thisFlag )
if ( !superType.isEmpty ())
{
// Is the super context included 'cv' information, ensure that it is added appropriately
if ( superType.isPtrRef () || cvType.isEmpty ())
ditType += ' ' + superType;
else
ditType += ' ' + cvType + ' ' + superType;
} // End of IF then
elif ( !cvType.isEmpty ())
ditType += ' ' + cvType;
// Make sure qualifiers aren't re-applied
ditType.setPtrRef ();
// Finally, return the composed 'data-indirection-type' (with embedded sub-type)
return ditType;
} // End of IF then
else
return DN_invalid;
} // End of IF then
elif ( !thisFlag && !superType.isEmpty ())
{
// Is the super context included 'cv' information, ensure that it is added appropriately
if ( superType.isPtrRef () || cvType.isEmpty ())
return ( DN_truncated + superType );
else
return ( DN_truncated + cvType + ' ' + superType );
} // End of ELIF then
elif ( !thisFlag && !cvType.isEmpty ())
return ( DN_truncated + cvType );
else
return DN_truncated;
} // End of "UnDecorator" FUNCTION "getDataIndirectType"
inline int __near UnDecorator::getECSUDataIndirectType ()
{
if ( *gName )
{
unsigned int ecsuCode = *gName++ - 'A';
// Is it a valid 'ecsu-data-indirection-type' ?
if (( ecsuCode >= ECSU_near ) && ( ecsuCode <= ( ECSU_const | ECSU_volatile | ECSU_modelmask )))
return ( ecsuCode | ECSU_valid );
else
return ECSU_invalid;
} // End of IF then
else
return ECSU_truncated;
} // End of "UnDecorator" FUNCTION "getECSUDataIndirectType"
inline DName __near UnDecorator::getPtrRefDataType ( const DName & superType, int isPtr )
{
// Doubles up as 'pointer-data-type' and 'reference-data-type'
if ( *gName )
{
// Is this a 'pointer-data-type' ?
if ( isPtr && ( *gName == PoDT_void ))
{
gName++; // Skip this character
if ( superType.isEmpty ())
return "void";
else
return "void " + superType;
} // End of IF
// Otherwise it may be a 'reference-data-type'
if ( *gName == RDT_array ) // An array ?
{
gName++;
return getArrayType( superType );
} // End of IF
// Otherwise, it is a 'basic-data-type'
return getBasicDataType ( superType );
} // End of IF then
else
return ( DN_truncated + superType );
} // End of "UnDecorator" FUNCTION "getPtrRefDataType"
inline DName __near UnDecorator::getArrayType ( const DName & superType )
{
if ( *gName )
{
int noDimensions = getNumberOfDimensions ();
if ( !noDimensions )
return getBasicDataType ( DName ( '[' ) + DN_truncated + ']' );
else
{
DName arrayType;
while ( noDimensions-- )
arrayType += '[' + getDimension () + ']';
// If it is indirect, then parenthesise the 'super-type'
if ( !superType.isEmpty ())
arrayType = '(' + superType + ')' + arrayType;
// Return the finished array dimension information
return getPrimaryDataType ( arrayType );
} // End of IF else
} // End of IF
elif ( !superType.isEmpty ())
return getBasicDataType ( '(' + superType + ")[" + DN_truncated + ']' );
else
return getBasicDataType ( DName ( '[' ) + DN_truncated + ']' );
} // End of "UnDecorator" FUNCTION "getArrayType"
inline DName __near UnDecorator::getLexicalFrame ( void ) { return '`' + getDimension () + '\''; }
inline DName __near UnDecorator::getStorageConvention ( void ) { return getDataIndirectType (); }
inline DName __near UnDecorator::getDataIndirectType () { return getDataIndirectType ( DName (), 0, DName ()); }
inline DName __near UnDecorator::getThisType ( void ) { return getDataIndirectType ( DName (), 0, DName (), TRUE ); }
inline DName __near UnDecorator::getPointerType ( const DName & cv, const DName & name )
{ return getPtrRefType ( cv, name, TRUE ); }
inline DName __near UnDecorator::getReferenceType ( const DName & cv, const DName & name )
{ return getPtrRefType ( cv, name, FALSE ); }
inline DName __near UnDecorator::getSegmentName ( void ) { return getZName (); }
#if ( !NO_COMPILER_NAMES )
inline DName __near UnDecorator::getDisplacement ( void ) { return getDimension (); }
inline DName __near UnDecorator::getCallIndex ( void ) { return getDimension (); }
inline DName __near UnDecorator::getGuardNumber ( void ) { return getDimension (); }
inline DName __near UnDecorator::getVbTableType ( const DName & superType )
{ return getVfTableType ( superType ); }
inline DName __near UnDecorator::getVCallThunkType ( void )
{
#if VERS_32BIT
switch (*gName) {
case VMT_nTnCnV:
++gName;
return DName("{flat}");
case 0:
return DN_truncated;
default:
return DN_invalid;
}
#else
DName vcallType = '{';
// Get the 'this' model, and validate all values
switch ( *gName )
{
case VMT_nTnCnV:
case VMT_nTfCnV:
case VMT_nTnCfV:
case VMT_nTfCfV:
case VMT_nTnCbV:
case VMT_nTfCbV:
vcallType += UScore ( TOK_nearSp );
break;
case VMT_fTnCnV:
case VMT_fTfCnV:
case VMT_fTnCfV:
case VMT_fTfCfV:
case VMT_fTnCbV:
case VMT_fTfCbV:
vcallType += UScore ( TOK_farSp );
break;
case 0:
return DN_truncated;
default:
return DN_invalid;
} // End of SWITCH
// Always append 'this'
vcallType += "this, ";
// Get the 'call' model
switch ( *gName )
{
case VMT_nTnCnV:
case VMT_fTnCnV:
case VMT_nTnCfV:
case VMT_fTnCfV:
case VMT_nTnCbV:
case VMT_fTnCbV:
vcallType += UScore ( TOK_nearSp );
break;
case VMT_nTfCnV:
case VMT_fTfCnV:
case VMT_nTfCfV:
case VMT_fTfCfV:
case VMT_nTfCbV:
case VMT_fTfCbV:
vcallType += UScore ( TOK_farSp );
break;
} // End of SWITCH
// Always append 'call'
vcallType += "call, ";
// Get the 'vfptr' model
switch ( *gName++ ) // Last time, so advance the pointer
{
case VMT_nTnCnV:
case VMT_nTfCnV:
case VMT_fTnCnV:
case VMT_fTfCnV:
vcallType += UScore ( TOK_nearSp );
break;
case VMT_nTnCfV:
case VMT_nTfCfV:
case VMT_fTnCfV:
case VMT_fTfCfV:
vcallType += UScore ( TOK_farSp );
break;
case VMT_nTfCbV:
case VMT_fTnCbV:
case VMT_fTfCbV:
case VMT_nTnCbV:
vcallType += getBasedType ();
break;
} // End of SWITCH
// Always append 'vfptr'
vcallType += "vfptr}";
// And return the resultant 'vcall-model-type'
return vcallType;
#endif
} // End of "UnDecorator" FUNCTION "getVCallThunk"
inline DName __near UnDecorator::getVfTableType ( const DName & superType )
{
DName vxTableName = superType;
if ( vxTableName.isValid () && *gName )
{
vxTableName = getStorageConvention () + ' ' + vxTableName;
if ( vxTableName.isValid ())
{
if ( *gName != '@' )
{
vxTableName += "{for ";
while ( vxTableName.isValid () && *gName && ( *gName != '@' ))
{
vxTableName += '`' + getScope () + '\'';
// Skip the scope delimiter
if ( *gName == '@' )
gName++;
// Close the current scope, and add a conjunction for the next (if any)
if ( vxTableName.isValid () && ( *gName != '@' ))
vxTableName += "s ";
} // End of WHILE
if ( vxTableName.isValid ())
{
if ( !*gName )
vxTableName += DN_truncated;
vxTableName += '}';
} // End of IF
} // End of IF
// Skip the 'vpath-name' terminator
if ( *gName == '@' )
gName++;
} // End of IF
} // End of IF then
elif ( vxTableName.isValid ())
vxTableName = DN_truncated + vxTableName;
return vxTableName;
} // End of "UnDecorator" FUNCTION "getVfTableType"
#endif // !NO_COMPILER_NAMES
inline DName __near UnDecorator::getExternalDataType ( const DName & superType )
{
// Create an indirect declarator for the the rest
DName * pDeclarator = gnew DName ();
DName declaration = getDataType ( pDeclarator );
// Now insert the declarator into the declaration along with its 'storage-convention'
*pDeclarator = getStorageConvention () + ' ' + superType;
return declaration;
} // End of "UnDecorator" FUNCTION "getExternalDataType"
inline int __near UnDecorator::doUnderScore () { return !( disableFlags & UNDNAME_NO_LEADING_UNDERSCORES ); }
inline int __near UnDecorator::doMSKeywords () { return !( disableFlags & UNDNAME_NO_MS_KEYWORDS ); }
inline int __near UnDecorator::doFunctionReturns () { return !( disableFlags & UNDNAME_NO_FUNCTION_RETURNS ); }
inline int __near UnDecorator::doAllocationModel () { return !( disableFlags & UNDNAME_NO_ALLOCATION_MODEL ); }
inline int __near UnDecorator::doAllocationLanguage () { return !( disableFlags & UNDNAME_NO_ALLOCATION_LANGUAGE ); }
#if 0
inline int __near UnDecorator::doMSThisType () { return !( disableFlags & UNDNAME_NO_MS_THISTYPE ); }
inline int __near UnDecorator::doCVThisType () { return !( disableFlags & UNDNAME_NO_CV_THISTYPE ); }
#endif
inline int __near UnDecorator::doThisTypes () { return (( disableFlags & UNDNAME_NO_THISTYPE ) != UNDNAME_NO_THISTYPE ); }
inline int __near UnDecorator::doAccessSpecifiers () { return !( disableFlags & UNDNAME_NO_ACCESS_SPECIFIERS ); }
inline int __near UnDecorator::doThrowTypes () { return !( disableFlags & UNDNAME_NO_THROW_SIGNATURES ); }
inline int __near UnDecorator::doMemberTypes () { return !( disableFlags & UNDNAME_NO_MEMBER_TYPE ); }
inline int __near UnDecorator::doReturnUDTModel () { return !( disableFlags & UNDNAME_NO_RETURN_UDT_MODEL ); }
inline int __near UnDecorator::do32BitNear () { return !( disableFlags & UNDNAME_32_BIT_DECODE ); }
inline int __near UnDecorator::doNameOnly () { return ( disableFlags & UNDNAME_NAME_ONLY ); }
inline int __near UnDecorator::doTypeOnly () { return ( disableFlags & UNDNAME_TYPE_ONLY ); }
pcchar_t __near UnDecorator::UScore ( Tokens tok )
{
#if !VERS_32BIT
if ((( tok == TOK_nearSp ) || ( tok == TOK_nearP )) && !do32BitNear ())
return tokenTable[ tok ] + 6; // Skip '__near'
#endif
if ( doUnderScore ())
return tokenTable[ tok ];
else
return tokenTable[ tok ] + 2 ;
} // End of "UnDecorator" FUNCTION "UScore"
// Include the string composition support classes. Mostly inline stuff, and
// not important to the algorithm.
#include "undname.inl"