1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     Contrib. demo 
   4 // Author:      Aleksandras Gluchovas 
   8 // Copyright:   (c) Aleskandars Gluchovas 
   9 // Licence:       wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13 #  pragma implementation "acell.h" 
  16 // For compilers that support precompilation, includes "wx/wx.h". 
  17 #include "wx/wxprec.h" 
  29 #if defined( wxUSE_TEMPLATE_STL ) 
  40 /***** Implementation for class SJParser *****/ 
  42 // statics used by inline'ed C helper-functions 
  43 static char* _gSrcStart 
= 0; 
  44 static char* _gSrcEnd   
= 0; 
  45 static char* _gLastSuppresedComment 
= 0; 
  46 static int   _gLineNo      
= 0; 
  48 // FOR NOW:: comments queue is static 
  49 #define MAX_CQ_ENTRIES 128 
  50 static char* _gCommentsQueue
[MAX_CQ_ENTRIES
]; 
  51 static int    _gCQSize 
= 0; 
  53 /***** keyword map related structures *****/ 
  57     inline bool operator()( char* x
, char* y
) const 
  58     {     return ( strcmp( x
,y 
) < 0 ); 
  62 //WXSTL_MAP(CharPtrT,CharPtrT, LESS_THEN_FUNCTOR(CharPtrT)); 
  64 #if defined( wxUSE_TEMPLATE_STL ) 
  66     typedef map
< char*, char*, less_c_str 
> KeywordMapT
; 
  70     typedef char* CharPtrT
; 
  71     typedef WXSTL_MAP( CharPtrT
, CharPtrT 
,less_c_str
) KeywordMapT
; 
  75 static KeywordMapT __gMultiLangMap
; 
  76 static int         __gMapReady 
= 0; 
  78 static char* __gKeyWords
[] = 
 114 static void check_keyword_map() 
 120         // "make sure" the address of the first member of non-polimorphic class 
 121         // coinsides with the address of the instance 
 123         char** keyword 
= __gKeyWords
; 
 125         while ( (*keyword
) != 0 ) 
 127             __gMultiLangMap
.insert( 
 128                 KeywordMapT::value_type( *keyword
, *keyword 
) 
 136 /***** helper functions *****/ 
 138 static inline void skip_to_eol( char*& cur 
) 
 140     while( *(cur
) != 10 && *cur 
!= 13 && cur 
< _gSrcEnd
) ++cur
; 
 143 static inline void skip_eol( char*& cur 
) 
 154 static inline bool skip_to_next_comment_in_the_line( char*& cur 
) 
 158         while( cur 
< _gSrcEnd  
&& 
 164         if ( cur 
== _gSrcEnd 
) return FALSE
; 
 168             if ( (*(cur
+1) == '*') || 
 169                  (*(cur
+1) == '/') ) return TRUE
; 
 182 inline static void store_line_no( int& toVar 
) 
 187 inline static void restore_line_no( int storedLineNo 
) 
 189     _gLineNo 
= storedLineNo
; 
 192 inline static int get_line_no() 
 197 static void skip_to_prev_line( char*& cur 
) 
 199     while( cur 
>= _gSrcStart  
&& 
 204     // NOTE:: '\n' is 13,10 for DOS 
 205     //        '\n' is 10 for UNIX 
 207     // NOTE1: '\n' symbol is not used here, 
 208     //        to provide possibility of loading 
 218     if ( *cur 
== 13 ) --cur
; 
 220     while( cur 
>= _gSrcStart  
&& 
 225     ++cur
; // move to the first character in the line 
 228 static inline void skip_comments( char*& cur 
) 
 230     ++cur
; // skip '/' token 
 232     if ( *cur 
!= '/' && *cur 
!= '*' ) return; 
 234     // first, store position of the comment into the queue 
 235     // (which further will be attached to the next context 
 238     if ( cur
-1 != _gLastSuppresedComment 
) 
 240         if ( _gCQSize 
== MAX_CQ_ENTRIES 
) 
 242             size_t i 
= MAX_CQ_ENTRIES
-1; 
 246                 _gCommentsQueue
[i
-1] = _gCommentsQueue
[i
]; 
 253         _gCommentsQueue
[_gCQSize
++] = cur
-1; 
 256     // if signle-line comment, skip it now 
 266     // check for multiline comment (handle nested multiline comments!) 
 274         // TBD:: check eof cond. 
 276         // detect and remove vertical columns of '*''s 
 278         while ( *cur 
!= '/' && cur 
< _gSrcEnd 
) 
 284                         if ( *(cur
+1) != '/' ) 
 294                 case 13 : line_len 
= 0; break; 
 295                 case 10 : { line_len 
= 0; ++_gLineNo
; } break; 
 297                 default : ++line_len
; 
 303         if ( cur 
>= _gSrcEnd  
) return; 
 307         if ( *(cur
-2) == '*' ) 
 325 static inline void clear_commets_queue() 
 330 static inline void skip_quoted_string( char*& cur 
) 
 332     ++cur
; // skip first quote '"' 
 334     // check if quote wasn't prefixed 
 335     if ( *(cur
-2) == '\\' ) 
 340         while ( *cur 
!= '"' && cur 
< _gSrcEnd 
) 
 342             if ( *cur 
== 10 ) ++_gLineNo
; 
 346         if ( cur 
>= _gSrcEnd 
) return; 
 348         ++cur
; // skip the last quote 
 350         // check if it wasn't prefixed 
 352         if ( *(cur
-2) != '\\' ) 
 358 // skips subsequent white space and comments 
 359 // (return false if the end of source code reached) 
 361 static inline bool get_next_token( char*& cur 
) 
 363     for( ; cur 
< _gSrcEnd
; ++cur 
) 
 371             case 10  : { ++_gLineNo
;continue; } 
 373             case '/' : skip_comments( cur 
); 
 383     if ( cur 
>= _gSrcEnd 
) 
 390 static inline void skip_preprocessor_dir( char*& cur 
) 
 396         if ( *(cur
-1) != '\\' ) 
 399         if ( cur 
< _gSrcEnd 
) 
 407 static void skip_token( char*& cur 
) 
 411         skip_quoted_string( cur 
); 
 425     // special case of "!=", "<=", ... 2 character composite tokens 
 439     ++cur
; // leading character is always skipped 
 441     for( ; cur 
< _gSrcEnd 
; ++cur 
) 
 454             // FIXME:: QUICK-HACK:: to treat scope resolution 
 455             //         tokens are a part of the string - e.g. SomeSpace::SubName would 
 458             case ':' : if ( *(cur
+1) == ':' ) 
 477 static inline size_t get_token_len( char* tok 
) 
 483     return size_t( tok 
- start 
); 
 486 // returns true, if given tokens are equel 
 488 static inline bool cmp_tokens( char* tok1
, char* tok2 
) 
 490     // NOTE:: the case one token includes 
 491     //        other in it's entirely is not handled 
 493     size_t len 
= get_token_len( tok1 
); 
 495     // assuming that tokens are non-zero length 
 499         if ( *(tok1
++) != *(tok2
++) ) 
 509 static inline bool cmp_tokens_fast( char* tok1
, char* tok2
, size_t len 
) 
 513         if ( *(tok1
++) != *(tok2
++) ) 
 521 static inline void skip_tempalate_statement( char*& cur 
) 
 525     // go one level deeper 
 526     while( *cur 
!= '<' && cur 
< _gSrcEnd 
) 
 528         if (*cur 
== 10 ) ++_gLineNo
; 
 532     // FIXME:: template should be checked statement for 
 533     //         comments inside of it 
 542         ++cur
; // skip '<' or '>' token 
 547         while( *cur 
!= '<' && *cur 
!= '>' && cur 
< _gSrcEnd 
) 
 549             if (*cur 
== 10 ) ++_gLineNo
; 
 556 static inline void skip_statement( char*& cur 
) 
 558     for( ; cur 
< _gSrcEnd
; ++cur 
) 
 562             case  ';' : ++cur
; // skip statement-terminator token 
 565             case  '"' : skip_quoted_string(cur
); 
 569             case  10  : ++_gLineNo
; 
 572             case  '/' : skip_comments( cur 
); 
 579 // "reversed" versions of skip_token() and get_next_token() 
 581 static inline void skip_token_back( char*& cur 
) 
 583     // FIXME:: now, when moving backwards, neither strings nor 
 584     //         comment blocks are checked 
 586     --cur
; // skip to the trailing character 
 595     for( ; cur 
< _gSrcEnd 
; --cur 
) 
 612     ++cur
; // get to the leading character of the token 
 615 static inline void skip_next_token_back( char*& cur 
) 
 617     --cur
; // skip leading character of the current token 
 628     for( ; cur 
< _gSrcEnd
; --cur 
) 
 645     ++cur
; // position after the trailing charcter of the prev token 
 648 static string 
get_token_str( char* cur 
) 
 650     return string( cur
, get_token_len( cur 
) ); 
 653 // skips token or whole expression which may have 
 654 // nested  expressions between '(' ')' brackets. 
 656 // Upon return, the cursor points to the terminating bracket ')', 
 658 // Return value is the size of the block 
 660 static size_t skip_block( char*& cur 
) 
 662     size_t level 
= 0; // nesting level 
 666     // NOTE:: assumed that block not necessarely starts 
 667     //        with bracket rightaway 
 678         char* savedPos 
= cur
; 
 680         store_line_no( tmpLnNo 
); 
 682         get_next_token( cur 
); 
 684         if ( cur 
>= _gSrcEnd 
) return 0; 
 696                 restore_line_no( tmpLnNo 
); 
 698                 return size_t(cur
-start
); 
 707                 // QUICK-HACK::to easily handle function prototypes , 
 708                 // it works, besause theoretically there should 
 709                 // be no cast-expressions in non-implementation 
 710                 // scope (e.g. "time( (long*)(ptr+1) )" should not 
 711                 // appear in the declarations, thus it is most likelly 
 712                 // for the ")(" fragment to be within a function 
 713                 // prototype in the declarations scope 
 721                 else return size_t(cur
-start
); 
 729                 restore_line_no( tmpLnNo 
); 
 731                 return size_t(cur
-start
); 
 738 // returns 0, if end of source reached 
 739 static inline bool skip_imp_block( char*& cur 
) 
 741     while( *cur 
!= '{' && cur 
< _gSrcEnd 
) 
 744         if ( !get_next_token( cur 
) ) return FALSE
; 
 747     while( *cur 
!= '}' && cur 
< _gSrcEnd 
) 
 750         if ( !get_next_token( cur 
) ) return FALSE
; 
 758 static bool is_class_token( char*& cur 
) 
 760     // FIXME:: the below mess should be cleaned in it's entirely 
 763         if ( *(cur
+1) == 'n' ) 
 765             return cmp_tokens_fast( cur
, "interface", 9 ); 
 768         if ( *(cur
+1) == 'l' ) 
 770             return cmp_tokens_fast( cur
, "class", 5 ); 
 773         if ( *(cur
+1) == 't' ) 
 775             return cmp_tokens_fast( cur
, "struct", 6 ); 
 778         if ( *(cur
+1) == 'n' ) 
 780             return cmp_tokens_fast( cur
, "union", 5 ); 
 785 inline static bool is_forward_decl( char* cur 
) 
 791             case ':' : return FALSE
; 
 792             case '{' : return FALSE
; 
 793             case '(' : return FALSE
; 
 795             case ';' : return TRUE
; 
 802     } while (cur 
< _gSrcEnd
); // prevent running out of bounds 
 807 inline static bool is_function( char* cur
, bool& isAMacro 
) 
 812     store_line_no( tmpLnNo 
); 
 814     // NOTE:: comments and quoted strings are not checked here 
 816     // first,check for "single-line hanginging macros" like: 
 824     get_next_token( cur 
); 
 829         restore_line_no( tmpLnNo 
); 
 834     // it's not a macro, go to the begining of arg. list 
 838         // if bracket found, it's a function or a begining 
 842             restore_line_no( tmpLnNo 
); 
 846         // end of statement found without any brackets in it 
 847         // - it cannot be a function 
 851             restore_line_no( tmpLnNo 
); 
 857     } while( cur 
< _gSrcEnd
); 
 860     restore_line_no( tmpLnNo 
); 
 865 // upon return the cursor is positioned after the 
 866 // terminating curly brace 
 868 static inline void skip_scope_block( char*& cur 
) 
 872     for( ; cur 
< _gSrcEnd 
; ++cur 
) 
 876             case '/' : skip_comments( cur 
); 
 879             case '"' : skip_quoted_string( cur 
); 
 889                            ++cur
; // skip final closing curly brace 
 893             case 10 : ++_gLineNo
; continue; 
 899 // moves tokens like '*' '**', '***', '&' from the name 
 902 static void arrange_indirection_tokens_between( string
& type
, 
 905     // TBD:: FIXME:: return value of operators ! 
 907     while ( identifier
[0u] == '*' || 
 908             identifier
[0u] == '&' 
 911         type 
+= identifier
[0u]; 
 912         identifier
.erase(0,1); 
 914         if ( !identifier
.length() ) return; 
 919 // the only function where multi-lang keyword map is accessed 
 921 static bool is_keyword( char* cur 
) 
 923     size_t len 
= get_token_len( cur 
); 
 925     // put a terminating zero after the given token 
 926     char tmp 
= *(cur 
+ len
); 
 929     KeywordMapT::iterator i
; 
 931     i 
= __gMultiLangMap
.find( cur 
); 
 933     // restore original character suppresed by terminating zero 
 936     return i 
== __gMultiLangMap
.end() ? false : true; 
 939 static inline void get_string_between( char* start
, char* end
, 
 949 static char* set_comment_text( string
& text
, char* start 
) 
 953     // to avoid poluting the queue with this comment 
 954     _gLastSuppresedComment 
= start
; 
 956     skip_comments( end 
); 
 958     if ( *(end
-1) == '/' ) 
 963     // skip multiple leading '/''s or '*''s 
 964     while( *start 
== '/' && start 
< end 
) ++start
; 
 965     while( *start 
== '*' && start 
< end 
) ++start
; 
 967     get_string_between( start
, end
, &text 
); 
 972 /***** Implementation for class CJSourceParser *****/ 
 974 CJSourceParser::CJSourceParser( bool collectCommnets
, bool collectMacros 
) 
 978       mCommentsOn( collectCommnets 
), 
 979       mMacrosOn  ( collectMacros 
) 
 984 spFile
* CJSourceParser::Parse( char* start
, char* end 
) 
 986     // set up state variables 
 987     mCurVis       
= SP_VIS_PRIVATE
; 
 989     spFile
* pTopCtx 
= new spFile(); 
1001     _gSrcEnd   
= mpEnd
; // let all the C-functions "smell" the end of file 
1006     clear_commets_queue(); 
1008     // main parsing loop 
1012         if ( !get_next_token( cur 
) ) 
1013             // end of source reached 
1016         if ( memcmp( cur
, "ScriptSection( const string&", 
1017                      strlen( "ScriptSection( const string&" ) 
1029                     AddMacroNode( cur 
); 
1060         // 'const' is a part of the return type, not a keyword here 
1061         if ( strncmp(cur
, "const", 5) != 0 && is_keyword( cur 
) ) 
1063             // parses, token, if token identifies 
1064             // the container context (e.g. class/namespace) 
1065             // the corresponding context object is created 
1066             // and set as current context 
1068             ParseKeyword( cur 
); 
1072         if ( *cur 
>= '0' && *cur 
<= '9' ) 
1080             if ( mCurCtxType 
!= SP_CTX_CLASS 
) 
1082                 // FOR NOW:: disable the below assertion 
1084                 // DBG:: unexpected closing-bracket found 
1087                 skip_token( cur 
); // just skip it 
1091             if ( mpCurCtx
->GetType() == SP_CTX_CLASS 
) 
1093                 int curOfs 
= ( (cur
+1) - _gSrcStart 
); 
1095                 mpCurCtx
->mContextLength 
= ( curOfs 
- mpCurCtx
->mSrcOffset 
); 
1100             // terminate operation/class/namespace context 
1101             // TBD:: check if it's really this type of context 
1103             wxASSERT( mpCurCtx 
); 
1104             mpCurCtx 
= mpCurCtx
->GetOutterContext(); 
1105             wxASSERT( mpCurCtx 
); 
1107             if ( mNestingLevel 
== 0 ) 
1110                 mCurCtxType 
= SP_CTX_FILE
; 
1112                 // not-nested class delclaration finished, 
1113                 // rest template flag in any case 
1123         if ( is_function( cur
, isAMacro 
) ) 
1131             char* savedPos 
= cur
; 
1134             store_line_no( tmpLnNo 
); 
1138             if ( !ParseNameAndRetVal( cur
, isAMacro 
) ) 
1143                     SkipFunction( cur 
); 
1148             if ( !ParseArguments( cur 
) ) 
1150                 // failure while parsing arguments, 
1151                 // remove enclosing operation context 
1153                 spContext
* pFailed 
= mpCurCtx
; 
1154                 mpCurCtx 
= mpCurCtx
->GetOutterContext(); 
1155                 mpCurCtx
->RemoveChild( pFailed 
); 
1162                 // otherwise, successfully close operation context: 
1164                 clear_commets_queue(); 
1166                 SkipFunctionBody( cur 
); 
1168                 mpCurCtx 
= mpCurCtx
->GetOutterContext(); 
1171                 wxASSERT( mpCurCtx 
); 
1175         else // otherwise it's declaration of a variable; 
1177             // now, the cursor point to the end of statement (';' token) 
1179             if ( mCurCtxType 
!= SP_CTX_CLASS 
) 
1181                 // non-class members are ignored 
1183                 skip_token( cur 
); // skip the end of statement 
1187             ParseMemberVar( cur 
); 
1193 void CJSourceParser::AttachComments( spContext
& ctx
, char* cur 
) 
1195     if ( !mCommentsOn 
) return; 
1197     MCommentListT
& lst 
= ctx
.GetCommentList(); 
1199     char* prevComEnd 
= 0; 
1202     store_line_no( tmpLnNo 
); 
1204     // attach comments which were found before the given context 
1206     for( int i 
= 0; i 
!= _gCQSize
; ++i 
) 
1208         spComment
* pComment 
= new spComment(); 
1209         lst
.push_back( pComment 
); 
1211         // find the end of comment 
1212         char* start 
= _gCommentsQueue
[i
]; 
1214         pComment
->mIsMultiline 
= ( *(start
+1) == '*' ); 
1216         // first comment in the queue and multiline 
1217         // comments are always treated as a begining 
1218         // of the new paragraph in the comment text 
1222             pComment
->mStartsPar 
= TRUE
; 
1224         if ( pComment
->mIsMultiline 
) 
1226             pComment
->mStartsPar 
= TRUE
; 
1229             // find out wheather there is a new-line 
1230             // between to adjecent comments 
1233             char* prevLine 
= start
; 
1234             skip_to_prev_line(prevLine
); 
1236             if ( prevLine 
>= prevComEnd 
) 
1238                 pComment
->mStartsPar 
= TRUE
; 
1240                 pComment
->mStartsPar 
= FALSE
; 
1243         prevComEnd 
= set_comment_text( pComment
->mText
, start 
); 
1247     // attach comments which are at the end of the line 
1248     // of the given context (if any) 
1250     if ( skip_to_next_comment_in_the_line( cur 
) ) 
1252         spComment
* pComment 
= new spComment(); 
1253         lst
.push_back( pComment 
); 
1255         set_comment_text( pComment
->mText
, cur 
); 
1257         pComment
->mStartsPar 
= 1; 
1258         pComment
->mIsMultiline 
= ( *(cur
+1) == '*' ); 
1260         // mark this comment, so that it would not 
1261         // get in the comments list of the next context 
1262         _gLastSuppresedComment 
= cur
; 
1265     restore_line_no( tmpLnNo 
); 
1267     clear_commets_queue(); 
1270 void CJSourceParser::AddMacroNode( char*& cur 
) 
1274     int lineNo 
= get_line_no(); 
1276     skip_preprocessor_dir( cur 
); 
1279     store_line_no( tmpLnNo 
); 
1281     if ( !mMacrosOn 
) return; 
1283     spPreprocessorLine
* pPL 
= new spPreprocessorLine(); 
1284     pPL
->mSrcLineNo 
= lineNo
; 
1286     AttachComments( *pPL
, cur 
); 
1288     get_string_between( start
, cur
, &pPL
->mLine 
); 
1290     ++start
; // skip '#' 
1291     get_next_token( start 
); 
1293     pPL
->mDefType 
= SP_PREP_DEF_OTHER
; 
1295     // if we found a definition or redefinition, 
1296     // determine the type exactly and assign 
1297     // a name to the context 
1299     if ( *start 
== 'd' ) 
1301         if ( cmp_tokens_fast( start
, "define", 6 ) ) 
1303             char* tok 
= start
+6; 
1305             get_next_token( tok 
); 
1307             pPL
->mName 
= get_token_str( tok 
); 
1310             get_next_token( tok
); 
1314                 pPL
->mDefType 
= SP_PREP_DEF_DEFINE_SYMBOL
; 
1316                 pPL
->mDefType 
= SP_PREP_DEF_REDEFINE_SYMBOL
; 
1319     else if ( *start 
== 'i' ) 
1321         if ( cmp_tokens_fast( start
, "include", 7 ) ) 
1323             pPL
->mDefType 
= SP_PREP_DEF_INCLUDE_FILE
; 
1325         else if ( *++start 
== 'f' ) 
1327             // either "#if" or "#ifdef" 
1330             get_next_token( cur 
); 
1332             string condition 
= get_token_str( cur 
); 
1334             // currently, everything except '0' is true 
1335             if ( condition 
== "0" ) { 
1336                 // skip until the following else or enif 
1337                 while ( cur 
< _gSrcEnd 
) { 
1341                     get_next_token( cur 
); 
1342                     if ( *cur
++ == '#' && *cur 
== 'e' ) 
1347             // TODO parse the condition... 
1350     else if ( cmp_tokens_fast( start
, "else", 4 ) ) 
1352         // skip until "#endif" 
1353         while ( cur 
< _gSrcEnd 
) { 
1357             get_next_token( cur 
); 
1358             if ( *cur
++ == '#' && cmp_tokens_fast( cur
, "endif", 5 ) ) 
1363     mpCurCtx
->AddMember( pPL 
); 
1368     restore_line_no( tmpLnNo 
); 
1370     clear_commets_queue(); 
1373 void CJSourceParser::ParseKeyword( char*& cur 
) 
1375     // analyze token, which identifies the begining of a new context 
1377     if ( CheckVisibilty( cur 
) ) 
1383     if ( is_class_token( cur 
) ) 
1385         if ( is_forward_decl( cur 
) ) 
1387             // forward declarations are ignored; 
1392         if ( mNestingLevel 
== 0 ) 
1394             // change context form global class context 
1395             mCurCtxType 
= SP_CTX_CLASS
; 
1400         // add information about new class (name, inheritance, etc) 
1401         AddClassNode( cur 
); 
1403         // the default visiblity for class members is 'private' 
1404         mCurVis 
= SP_VIS_PRIVATE
; 
1409     size_t len 
= get_token_len( cur 
); 
1411     if ( cmp_tokens_fast( cur
, "typedef", len  
) ) 
1414         get_next_token(cur
); 
1416         if ( cmp_tokens_fast( cur
, "struct", len 
) || 
1417              cmp_tokens_fast( cur
, "union",  len 
) || 
1418              cmp_tokens_fast( cur
, "class",  len 
) 
1421             if ( mNestingLevel 
== 0 ) 
1423                 // change context form global class context 
1424                 mCurCtxType 
= SP_CTX_CLASS
; 
1429             // add information about new class (name, inheritance, etc) 
1430             AddClassNode( cur 
); 
1432             // the default visiblity for class members is 'private' 
1433             mCurVis 
= SP_VIS_PRIVATE
; 
1437             // FOR NOW:: typedef struct, etc are also ignored 
1438             //skip_scope_block( cur ); 
1441         if ( cmp_tokens_fast( cur
, "enum", len  
) ) 
1447         AddTypeDefNode( cur 
); 
1452     if ( cmp_tokens_fast( cur
, "enum", len 
) ) 
1458     if ( cmp_tokens_fast( cur
, "extern", len 
) ) 
1460         // extern's are ignored (both extern "C" and extern vars) 
1461         while ( *cur 
!= '{' && 
1465             get_next_token( cur 
); 
1470     if ( cmp_tokens_fast( cur
, "enum", len 
) ) 
1472         // enumeration blocks are ignored 
1474         skip_scope_block( cur 
); 
1476         get_next_token( cur 
); 
1477         skip_token( cur 
); // skip ';' token; 
1481     if ( cmp_tokens_fast( cur
, "package", len  
) ) 
1483         // packages are ignored 
1484         skip_statement( cur 
); 
1488     if ( cmp_tokens_fast( cur
, "import", len  
) ) 
1490         // import statements are ignored 
1491         skip_statement( cur 
); 
1495     if ( cmp_tokens_fast( cur
, "virtual", len  
) ) 
1497         // probably the virtual method is in front of us; 
1503     if ( cmp_tokens_fast( cur
, "template", len  
) ) 
1506         skip_tempalate_statement( cur 
); 
1510     if ( cmp_tokens_fast( cur
, "friend", len  
) ) 
1512         skip_statement( cur 
); 
1516     // ingnore "unsigificant" tokens (i.e. which do not 
1517     // affect the current parsing context) 
1522 bool CJSourceParser::ParseNameAndRetVal( char*& cur
, bool& isAMacro 
) 
1526     // FOR NOW:: all functions in the global 
1527     //           scope are ignored 
1529     int lineNo 
= get_line_no(); 
1533     bool isVirtual 
= false; 
1534     while( *cur 
!= '(' ) 
1536         if ( get_token_str( cur 
) == "virtual" ) 
1540         if ( !get_next_token( cur 
) ) return FALSE
; 
1543     char* bracketPos 
= cur
; 
1544     char* savedPos   
= cur 
+ 1; 
1547     store_line_no( tmpLnNo 
); 
1549     // skip gap between function name and start of paramters list 
1550     while ( *(cur
-1) == ' ' ) 
1553     // check if it's not a macro, and let plugin handle it, if so 
1557         skip_token_back( cur 
); 
1561         if ( mpPlugin
->CanUnderstandContext( tmp
, _gSrcEnd
, mpCurCtx 
) ) 
1565             mpPlugin
->ParseContext( _gSrcStart
, cur
, _gSrcEnd
, mpCurCtx 
); 
1573     spOperation
* pOp 
= new spOperation(); 
1575     pOp
->mSrcLineNo    
= lineNo
; 
1576     pOp
->mSrcOffset    
= int( start 
- _gSrcStart 
); 
1577     pOp
->mHeaderLength 
= int( bracketPos 
- start 
); 
1578     if ( mpCurCtx
->GetContextType() == SP_CTX_CLASS 
) 
1579         pOp
->mScope 
= mpCurCtx
->mName
; 
1581     mpCurCtx
->AddMember( pOp 
); 
1582     pOp
->mVisibility 
= mCurVis
; 
1583     pOp
->mIsVirtual 
= isVirtual
; 
1585     // add comments about operation 
1586     AttachComments( *pOp
, cur 
); 
1588     // go backwards to method name 
1589     skip_token_back( cur 
); 
1591     pOp
->mName 
= get_token_str( cur 
); 
1593     // checker whether it's not an operator 
1594     char chFirst 
= *pOp
->mName
.c_str(); 
1595     if ( !isalpha(chFirst
) && chFirst 
!= '_' && chFirst 
!= '~' ) { 
1597         skip_next_token_back( cur 
); 
1598         skip_token_back( cur 
); 
1600         string lastToken 
= get_token_str( cur 
); 
1601         if ( lastToken 
== "operator" ) { 
1602             lastToken 
+= pOp
->mName
; 
1603             pOp
->mName 
= lastToken
; 
1606             // ok, it wasn't an operator after all 
1610     else if ( pOp
->mName 
== "operator" ) { 
1612         get_next_token( cur 
); 
1613         string oper 
= get_token_str( cur 
); 
1618     // go backwards to method return type 
1619     skip_next_token_back( cur 
); 
1623         string rettype 
= string( start
, size_t( cur
-start 
) ); 
1624         rettype
.Replace("WXDLLEXPORT ", ""); // FIXME just for now... 
1625         pOp
->mRetType 
= rettype
; 
1628     arrange_indirection_tokens_between( pOp
->mRetType
, pOp
->mName 
); 
1631     restore_line_no( tmpLnNo 
); 
1633     // now, enter operation context 
1639 bool CJSourceParser::ParseArguments( char*& cur 
) 
1643     // now cursor position is right after the first opening bracket 
1644     // of the function declaration 
1646     char* blocks    
[16]; // used exclusivelly for iterative "lean out" 
1647                           // of macros and misc. not-obviouse grammar 
1648                           // (dirty,, but we cannot do it very nice, 
1649                           //  we're not preprocessor-free C/C++ code) 
1654         size_t blocksSkipped 
= 0; 
1656         get_next_token( cur 
); 
1660         while( *cur 
!= ')' && *cur 
!= ',' ) 
1662             blocks
[blocksSkipped
] = cur
; 
1669                 blockSizes
[blocksSkipped
] = size_t(cur
-prev
); 
1674                 blockSizes
[blocksSkipped
] = skip_block( cur 
); 
1676             get_next_token( cur 
); 
1681         if ( blocksSkipped 
== 1 ) 
1683             // check if the empty arg. list stressed with "void" inside 
1684             if ( cmp_tokens_fast( blocks
[0] , "void", 4 ) ) 
1691             // FIXME:: TBD:: K&R-style function declarations! 
1693             // if only one block enclosed, than it's probably 
1694             // some macro, there should be at least two blocks, 
1695             // one for argument type and another for it's identifier 
1699         if ( blocksSkipped 
== 0 ) 
1701             if ( *cur 
== 10 ) ++_gLineNo
; 
1704             break; // function without paramters 
1707         // we should be in the operation context now 
1708         spOperation
* pOp 
= (spOperation
*)mpCurCtx
; 
1710         spParameter
* pPar 
= new spParameter(); 
1712         pOp
->AddMember( pPar 
); 
1713         // FOR NOW:: line number is not exact if argument list is mutiline 
1714         pPar
->mSrcLineNo 
= get_line_no(); 
1716         size_t nameBlock 
= blocksSkipped 
- 1; 
1717         size_t typeBlock 
= nameBlock 
- 1; 
1719         // check if default values present 
1720         if ( *blocks
[typeBlock
] == '=' ) 
1722             // expressions like "int = 5" are ignored, 
1723             // since name for paramters is required 
1724             if ( blocksSkipped 
== 3 ) 
1735             pPar
->mInitVal 
= string( blocks
[nameBlock
], blockSizes
[nameBlock
] ); 
1737             nameBlock 
= nameBlock 
- 2; // skip '=' token and default value block 
1738             typeBlock 
= nameBlock 
- 1; 
1741         // attach comments about the parameter 
1742         AttachComments( *pPar
, blocks
[nameBlock
] ); 
1744         // retrieve argument name 
1745         pPar
->mName 
= string( blocks
[nameBlock
], blockSizes
[nameBlock
] ); 
1747         // retreive argument type 
1749         size_t len 
= blockSizes
[ typeBlock 
]; 
1750         len 
= size_t ( (blocks
[ typeBlock 
] + len
) - blocks
[ 0 ] ); 
1752         pPar
->mType 
= string( blocks
[0], len 
); 
1754         arrange_indirection_tokens_between( pPar
->mType
, pPar
->mName 
); 
1762         ++cur
; // skip comma 
1763         get_next_token(cur
); 
1767     // skip possible whitespace between ')' and following "const" 
1768     while ( isspace(*cur
) ) 
1771     // check if it was really a function not a macro, 
1772     // if so, than it should be terminated with semicolon ';' 
1773     // or opening implemenetaton bracket '{' 
1778     store_line_no( tmpLnNo 
); 
1782         if ( *tok 
== '{' || *tok 
== ';' ) 
1784             restore_line_no(tmpLnNo
); 
1788         // check for unexpected tokens 
1789         if ( *tok 
== '=' || *tok 
== '0' ) 
1792             if ( !get_next_token(tok
) ) return FALSE
; 
1796         if ( *tok 
== '}' ) return FALSE
; 
1798         // if initialization list found 
1801             restore_line_no(tmpLnNo
); 
1805         if ( cmp_tokens_fast( tok
, "const", 5 ) ) 
1807             ((spOperation
*)mpCurCtx
)->mIsConstant 
= true; 
1810             if ( !get_next_token(tok
) ) return FALSE
; 
1814         if ( CheckVisibilty( tok 
) ) return FALSE
; 
1816         // if next context found 
1817         if ( is_keyword( tok 
) ) return FALSE
; 
1820         if ( !get_next_token(tok
) ) return FALSE
; 
1827 void CJSourceParser::ParseMemberVar( char*& cur 
) 
1829     MMemberListT
& members 
= mpCurCtx
->GetMembers(); 
1831     bool firstMember 
= 1; 
1837     // jump to the end of statement 
1838     // and start collecting same-type varibles 
1839     // back-to-front towards the type identifier 
1841     skip_statement( cur 
); 
1842     char* savedPos 
= cur
; 
1845     store_line_no( tmpLnNo 
); 
1847     --cur
; // rewind back to ';' 
1851         spAttribute
* pAttr 
= new spAttribute(); 
1852         // FOR NOW:: line not is not exact, if member declaration is multiline 
1853         pAttr
->mSrcLineNo 
= get_line_no(); 
1855         mpCurCtx
->AddMember( pAttr 
); 
1856         pAttr
->mVisibility 
= mCurVis
; 
1858         pAttr
->mIsConstant 
= 0; 
1863             first 
= members
.size() - 1;; 
1866         skip_token_back( cur 
); 
1868         // attach comments about the attribute 
1869         AttachComments( *pAttr
, cur 
); 
1871         pAttr
->mName 
= get_token_str( cur 
); 
1873         // guessing that this going to be variable type 
1874         skip_next_token_back( cur 
); 
1875         skip_token_back( cur 
); 
1877         pAttr
->mType 
= get_token_str( cur 
); 
1879         // if comma, than variable list continues 
1880         // otherwise the variable type reached - stop 
1884             // yes, we've mistaken, it was not a identifier, 
1885             // but it's default value 
1889             // skip default value and '=' symbol 
1890             skip_next_token_back( cur 
); 
1891             skip_token_back( cur 
); 
1893             pAttr
->mName 
= get_token_str( cur 
); 
1895             skip_next_token_back( cur 
); 
1896             skip_token_back( cur 
); 
1901             type 
= get_token_str( cur 
); 
1909     // set up types for all collected (same-type) attributes; 
1910     while ( first 
!= members
.size() - 1 ) 
1912         spAttribute
* pAttr 
= members
[first
++]->CastToAttribute(); 
1916         if ( !pAttr
->mType 
) 
1917             pAttr
->mType 
= type
; 
1918         pAttr
->mVisibility 
= mCurVis
; 
1920         if ( !!pAttr
->mName 
) 
1921             arrange_indirection_tokens_between( pAttr
->mType
, pAttr
->mName 
); 
1925     restore_line_no( tmpLnNo 
); 
1927     clear_commets_queue(); 
1932 void CJSourceParser::SkipFunction( char*& cur 
) 
1934     while ( *cur 
!= '(' && cur 
< _gSrcEnd 
) 
1936         if (*cur 
== 10 ) ++_gLineNo
; 
1940     skip_next_token_back( cur 
); // go back and skip function identifier 
1941     skip_token_back( cur 
);      // go back and skip return type 
1943     skip_block( cur 
);           // now, go ahead and skip whole declaration 
1945     SkipFunctionBody( cur 
); 
1949 void CJSourceParser::SkipFunctionBody( char*& cur 
) 
1951     // FIXME:: check for comments and quoted stirngs here 
1953     bool hasDefinition 
= FALSE
; 
1955     while( *cur 
!= '{' && *cur 
!= ';' ) 
1957         if (*cur 
== 10 ) ++_gLineNo
; 
1967         hasDefinition 
= TRUE
; 
1969         skip_scope_block( cur 
); // skip the whole imp. 
1972     if ( mpCurCtx
->GetType() == SP_CTX_OPERATION 
) 
1974         spOperation
& op 
= *((spOperation
*)mpCurCtx
); 
1976         int curOfs 
= int ( cur 
- _gSrcStart 
); 
1978         op
.mContextLength 
= curOfs 
- mpCurCtx
->mSrcOffset
; 
1980         op
.mHasDefinition 
= hasDefinition
; 
1982         // separate scope resolution token from the name of operation 
1984         for( size_t i 
= 0; i 
!= op
.mName
.length(); ++i 
) 
1986             if ( op
.mName
[i
] == ':' && op
.mName
[i
+1] == ':' ) 
1988                 string 
unscoped( op
.mName
, i
+2, op
.mName
.length() - ( i 
+ 2 ) ); 
1990                 op
.mScope 
= string( op
.mName
, 0, i 
); 
1992                 op
.mName 
= unscoped
; 
2000 bool CJSourceParser::CheckVisibilty( char*& cur 
) 
2002     size_t len 
= get_token_len( cur 
); 
2004     if ( cmp_tokens_fast( cur
, "public:", len 
) ) 
2006         mCurVis 
= SP_VIS_PUBLIC
; 
2010     if ( cmp_tokens_fast( cur
, "protected:", len 
) ) 
2012         mCurVis 
= SP_VIS_PROTECTED
; 
2016     if ( cmp_tokens_fast( cur
, "private:", len 
) ) 
2018         mCurVis 
= SP_VIS_PRIVATE
; 
2025 void CJSourceParser::AddClassNode( char*& cur 
) 
2027     char* ctxStart 
= cur
; 
2029     string classkeyword 
= get_token_str( cur 
); 
2031     skip_token( cur 
); // skip 'class' keyword 
2032     if ( !get_next_token( cur 
) ) return; 
2038         get_next_token( cur 
); 
2041     // by default all class members are private 
2042     mCurVis 
= SP_VIS_PRIVATE
; 
2044     spClass
* pClass 
= new spClass(); 
2045     if ( classkeyword 
== "class" ) 
2046         pClass
->mClassSubType 
= SP_CLTYPE_CLASS
; 
2047     else if ( classkeyword 
== "struct" ) { 
2048         pClass
->mClassSubType 
= SP_CLTYPE_STRUCTURE
; 
2050         mCurVis 
= SP_VIS_PUBLIC
; 
2052     else if ( classkeyword 
== "union" ) { 
2053         pClass
->mClassSubType 
= SP_CLTYPE_UNION
; 
2055         mCurVis 
= SP_VIS_PUBLIC
; 
2057     else if ( classkeyword 
== "interface" ) 
2058         pClass
->mClassSubType 
= SP_CLTYPE_INTERFACE
; 
2060         pClass
->mClassSubType 
= SP_CLTYPE_INVALID
; 
2062         wxFAIL_MSG("unknown class keyword"); 
2065     mpCurCtx
->AddMember( pClass 
); 
2067     // attach comments about the class 
2068     AttachComments( *pClass
, cur 
); 
2070     pClass
->mSrcLineNo 
= get_line_no(); 
2072     pClass
->mSrcOffset 
= int( ctxStart 
- _gSrcStart 
); 
2074     char* nameTok 
= cur
; 
2075     pClass
->mName 
= get_token_str( cur 
); 
2084         if ( !get_next_token( cur 
) ) return; 
2093             store_line_no( tmpLn 
); 
2095             skip_next_token_back( tok 
); 
2096             skip_token_back( tok 
); 
2098             restore_line_no( tmpLn 
); 
2100             // class name should precend ':' colon, thus 
2101             // the one which was captured before was 
2102             // proablty something else (like __dllexport MyClass : ... ) 
2104             if ( nameTok 
!= tok 
) 
2106                 pClass
->mName 
= get_token_str( tok 
); 
2117         size_t len 
= get_token_len( cur 
); 
2119         // skip neglectable C++ modifieres 
2120         if ( cmp_tokens_fast( cur
, "public", len 
) ) 
2123         if ( cmp_tokens_fast( cur
, "protected", len 
) ) 
2126         if ( cmp_tokens_fast( cur
, "private", len 
) ) 
2129         if ( cmp_tokens_fast( cur
, "virtual", len 
) ) 
2132         // skip neglectable JAVA modifieres 
2134         if ( cmp_tokens_fast( cur
, "extends", len 
) ) 
2140         if ( cmp_tokens_fast( cur
, "implements", len 
) ) 
2146         // all we need to know is superclass or interface 
2150         store_line_no( tmpLn 
); 
2153         get_next_token(tok
); 
2155         restore_line_no( tmpLn 
); 
2157         if ( *tok 
!= ':' && *cur 
!= ':' ) 
2159             pClass
->mSuperClassNames
.push_back( string( cur
, len 
) ); 
2166         store_line_no( tmpLn 
); 
2168         while ( pClass
->mSuperClassNames
.size() ) 
2170             pClass
->mSuperClassNames
.erase( &pClass
->mSuperClassNames
[0] ); 
2174         // some non-obviouse token was following "class" keyword - 
2175         // we've confused it with class name - thus now we're reverting this mistake 
2177         skip_next_token_back( tok 
); 
2178         skip_token_back( tok 
); 
2180         pClass
->mName 
= get_token_str( tok 
); 
2182         restore_line_no( tmpLn 
); 
2186     ++cur
; // skip opening curly brace 
2188     pClass
->mHeaderLength 
= ( cur 
- ctxStart 
); 
2190     // now, enter the class context 
2193     clear_commets_queue(); 
2196 void CJSourceParser::AddEnumNode( char*& cur 
) 
2198     // now the cursor is at "enum" keyword 
2201     spEnumeration
* pEnum 
= new spEnumeration(); 
2202     mpCurCtx
->AddMember( pEnum 
); 
2204     pEnum
->mSrcLineNo 
= get_line_no(); 
2207     AttachComments( *pEnum
, cur 
); 
2210     if ( !get_next_token( cur 
) ) return; 
2212     // check if enumeration has got it's identifier 
2215         pEnum
->mName 
= get_token_str( cur 
); 
2218     if ( !skip_imp_block( cur 
) ) return; 
2220     get_string_between( start
, cur
, &pEnum
->mEnumContent 
); 
2222     if ( get_next_token(cur
) ) 
2224         // check if the identifier if after the {...} block 
2227             pEnum
->mName 
= get_token_str( cur 
); 
2230     clear_commets_queue(); 
2233 void CJSourceParser::AddTypeDefNode( char*& cur 
) 
2235     // now the cursor at the token next to "typedef" keyword 
2237     if ( !get_next_token(cur
) ) return; 
2241     spTypeDef
* pTDef 
= new spTypeDef(); 
2242     mpCurCtx
->AddMember( pTDef 
); 
2244     pTDef
->mSrcLineNo 
= get_line_no(); 
2246     AttachComments( *pTDef
, cur 
); 
2248     skip_statement( cur 
); 
2251     store_line_no( tmpLnNo 
); 
2254     skip_next_token_back( tok 
); 
2256     char* nameEnd 
= tok
; 
2258     skip_token_back( tok 
); 
2260     char* nameStart 
= tok
; 
2262     skip_next_token_back( tok 
); 
2264     char* typeEnd 
= tok
; 
2266     // check if it's function prototype 
2267     if ( *nameStart 
== ')' ) 
2269         typeEnd 
= nameStart
+1; 
2271         // skip argument list 
2272         while ( *nameStart 
!= '(' ) --nameStart
; 
2274         // skip to function type definition 
2275         while ( *nameStart 
!= ')' ) --nameStart
; 
2277         skip_next_token_back( nameStart 
); 
2279         nameEnd 
= nameStart
; 
2281         skip_token_back( nameStart 
); 
2283         if ( *nameStart 
== '*' ) ++nameStart
; 
2286     get_string_between( start
, typeEnd
, &pTDef
->mOriginalType 
); 
2288     get_string_between( nameStart
, nameEnd
, &pTDef
->mName 
); 
2290     clear_commets_queue(); 
2292     restore_line_no( tmpLnNo 
);