CMDscan.l 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625
  1. %{
  2. // flex --nounput -o CMDscan.cpp -P CMD CMDscan.l
  3. #define YYLMAX 4096
  4. #define YY_NO_UNISTD_H
  5. #include <stdio.h>
  6. #include "platform/platform.h"
  7. #include "core/stringTable.h"
  8. #include "console/console.h"
  9. #include "console/compiler.h"
  10. #include "console/dynamicTypes.h"
  11. #include "core/strings/stringFunctions.h"
  12. template< typename T >
  13. struct Token
  14. {
  15. T value;
  16. S32 lineNumber;
  17. };
  18. // Can't have ctors in structs used in unions, so we have this.
  19. template< typename T >
  20. inline Token< T > MakeToken( T value, U32 lineNumber )
  21. {
  22. Token< T > result;
  23. result.value = value;
  24. result.lineNumber = lineNumber;
  25. return result;
  26. }
  27. #include "console/cmdgram.h"
  28. using namespace Compiler;
  29. #define YY_NEVER_INTERACTIVE 1
  30. // Some basic parsing primitives...
  31. static int Sc_ScanDocBlock();
  32. static int Sc_ScanString(int ret);
  33. static int Sc_ScanNum();
  34. static int Sc_ScanVar();
  35. static int Sc_ScanHex();
  36. static int Sc_ScanIdent();
  37. // Deal with debuggability of FLEX.
  38. #ifdef TORQUE_DEBUG
  39. #define FLEX_DEBUG 1
  40. #else
  41. #define FLEX_DEBUG 0
  42. #endif
  43. // Install our own input code...
  44. #undef CMDgetc
  45. int CMDgetc();
  46. // Hack to make windows lex happy.
  47. #ifndef isatty
  48. inline int isatty(int) { return 0; }
  49. #endif
  50. // Wrap our getc, so that lex doesn't try to do its own buffering/file IO.
  51. #define YY_INPUT(buf,result,max_size) \
  52. { \
  53. int c = '*', n; \
  54. for ( n = 0; n < max_size && \
  55. (c = CMDgetc()) != EOF && c != '\n'; ++n ) \
  56. buf[n] = (char) c; \
  57. if ( c == '\n' ) \
  58. buf[n++] = (char) c; \
  59. result = n; \
  60. }
  61. // General helper stuff.
  62. static int lineIndex;
  63. // File state
  64. void CMDSetScanBuffer(const char *sb, const char *fn);
  65. const char * CMDgetFileLine(int &lineNumber);
  66. // Error reporting
  67. void CMDerror(char * s, ...);
  68. // Reset the parser.
  69. void CMDrestart(FILE *in);
  70. %}
  71. DIGIT [0-9]
  72. INTEGER {DIGIT}+
  73. FLOAT ({INTEGER}?\.{INTEGER})|({INTEGER}(\.{INTEGER})?[eE][+-]?{INTEGER})
  74. LETTER [A-Za-z_]
  75. FILECHAR [A-Za-z_\.]
  76. VARMID [:A-Za-z0-9_]
  77. IDTAIL [A-Za-z0-9_]
  78. VARTAIL {VARMID}*{IDTAIL}
  79. VAR [$%]{LETTER}{VARTAIL}*
  80. ID {LETTER}{IDTAIL}*
  81. ILID [$%]{DIGIT}+{LETTER}{VARTAIL}*
  82. FILENAME {FILECHAR}+
  83. SPACE [ \t\v\f]
  84. HEXDIGIT [a-fA-F0-9]
  85. %%
  86. ;
  87. {SPACE}+ { }
  88. ("///"([^/\n\r][^\n\r]*)?[\n\r]+)+ { return(Sc_ScanDocBlock()); }
  89. "//"[^\n\r]* ;
  90. [\r] ;
  91. [\n] {lineIndex++;}
  92. \"(\\.|[^\\"\n\r])*\" { return(Sc_ScanString(STRATOM)); }
  93. \'(\\.|[^\\'\n\r])*\' { return(Sc_ScanString(TAGATOM)); }
  94. "==" { CMDlval.i = MakeToken< int >( opEQ, lineIndex ); return opEQ; }
  95. "!=" { CMDlval.i = MakeToken< int >( opNE, lineIndex ); return opNE; }
  96. ">=" { CMDlval.i = MakeToken< int >( opGE, lineIndex ); return opGE; }
  97. "<=" { CMDlval.i = MakeToken< int >( opLE, lineIndex ); return opLE; }
  98. "&&" { CMDlval.i = MakeToken< int >( opAND, lineIndex ); return opAND; }
  99. "||" { CMDlval.i = MakeToken< int >( opOR, lineIndex ); return opOR; }
  100. "::" { CMDlval.i = MakeToken< int >( opCOLONCOLON, lineIndex ); return opCOLONCOLON; }
  101. "--" { CMDlval.i = MakeToken< int >( opMINUSMINUS, lineIndex ); return opMINUSMINUS; }
  102. "++" { CMDlval.i = MakeToken< int >( opPLUSPLUS, lineIndex ); return opPLUSPLUS; }
  103. "$=" { CMDlval.i = MakeToken< int >( opSTREQ, lineIndex ); return opSTREQ; }
  104. "!$=" { CMDlval.i = MakeToken< int >( opSTRNE, lineIndex ); return opSTRNE; }
  105. "<<" { CMDlval.i = MakeToken< int >( opSHL, lineIndex ); return opSHL; }
  106. ">>" { CMDlval.i = MakeToken< int >( opSHR, lineIndex ); return opSHR; }
  107. "+=" { CMDlval.i = MakeToken< int >( opPLASN, lineIndex ); return opPLASN; }
  108. "-=" { CMDlval.i = MakeToken< int >( opMIASN, lineIndex ); return opMIASN; }
  109. "*=" { CMDlval.i = MakeToken< int >( opMLASN, lineIndex ); return opMLASN; }
  110. "/=" { CMDlval.i = MakeToken< int >( opDVASN, lineIndex ); return opDVASN; }
  111. "%=" { CMDlval.i = MakeToken< int >( opMODASN, lineIndex ); return opMODASN; }
  112. "&=" { CMDlval.i = MakeToken< int >( opANDASN, lineIndex ); return opANDASN; }
  113. "^=" { CMDlval.i = MakeToken< int >( opXORASN, lineIndex ); return opXORASN; }
  114. "|=" { CMDlval.i = MakeToken< int >( opORASN, lineIndex ); return opORASN; }
  115. "<<=" { CMDlval.i = MakeToken< int >( opSLASN, lineIndex ); return opSLASN; }
  116. ">>=" { CMDlval.i = MakeToken< int >( opSRASN, lineIndex ); return opSRASN; }
  117. "->" { CMDlval.i = MakeToken< int >( opINTNAME, lineIndex ); return opINTNAME; }
  118. "-->" { CMDlval.i = MakeToken< int >( opINTNAMER, lineIndex ); return opINTNAMER; }
  119. "NL" { CMDlval.i = MakeToken< int >( '\n', lineIndex ); return '@'; }
  120. "TAB" { CMDlval.i = MakeToken< int >( '\t', lineIndex ); return '@'; }
  121. "SPC" { CMDlval.i = MakeToken< int >( ' ', lineIndex ); return '@'; }
  122. "@" { CMDlval.i = MakeToken< int >( 0, lineIndex ); return '@'; }
  123. "/*" { /* this comment stops syntax highlighting from getting messed up when editing the lexer in TextPad */
  124. register int c = 0, l;
  125. for ( ; ; )
  126. {
  127. l = c;
  128. c = yyinput();
  129. // Is this an open comment?
  130. if ( c == EOF )
  131. {
  132. CMDerror( "unexpected end of file found in comment" );
  133. break;
  134. }
  135. // Increment line numbers.
  136. else if ( c == '\n' )
  137. lineIndex++;
  138. // Did we find the end of the comment?
  139. else if ( l == '*' && c == '/' )
  140. break;
  141. }
  142. }
  143. "?" |
  144. "[" |
  145. "]" |
  146. "(" |
  147. ")" |
  148. "+" |
  149. "-" |
  150. "*" |
  151. "/" |
  152. "<" |
  153. ">" |
  154. "|" |
  155. "." |
  156. "!" |
  157. ":" |
  158. ";" |
  159. "{" |
  160. "}" |
  161. "," |
  162. "&" |
  163. "%" |
  164. "^" |
  165. "~" |
  166. "=" { CMDlval.i = MakeToken< int >( CMDtext[ 0 ], lineIndex ); return CMDtext[ 0 ]; }
  167. "in" { CMDlval.i = MakeToken< int >( rwIN, lineIndex ); return(rwIN); }
  168. "or" { CMDlval.i = MakeToken< int >( rwCASEOR, lineIndex ); return(rwCASEOR); }
  169. "break" { CMDlval.i = MakeToken< int >( rwBREAK, lineIndex ); return(rwBREAK); }
  170. "return" { CMDlval.i = MakeToken< int >( rwRETURN, lineIndex ); return(rwRETURN); }
  171. "else" { CMDlval.i = MakeToken< int >( rwELSE, lineIndex ); return(rwELSE); }
  172. "assert" { CMDlval.i = MakeToken< int >( rwASSERT, lineIndex ); return(rwASSERT); }
  173. "while" { CMDlval.i = MakeToken< int >( rwWHILE, lineIndex ); return(rwWHILE); }
  174. "do" { CMDlval.i = MakeToken< int >( rwDO, lineIndex ); return(rwDO); }
  175. "if" { CMDlval.i = MakeToken< int >( rwIF, lineIndex ); return(rwIF); }
  176. "foreach$" { CMDlval.i = MakeToken< int >( rwFOREACHSTR, lineIndex ); return(rwFOREACHSTR); }
  177. "foreach" { CMDlval.i = MakeToken< int >( rwFOREACH, lineIndex ); return(rwFOREACH); }
  178. "for" { CMDlval.i = MakeToken< int >( rwFOR, lineIndex ); return(rwFOR); }
  179. "continue" { CMDlval.i = MakeToken< int >( rwCONTINUE, lineIndex ); return(rwCONTINUE); }
  180. "function" { CMDlval.i = MakeToken< int >( rwDEFINE, lineIndex ); return(rwDEFINE); }
  181. "new" { CMDlval.i = MakeToken< int >( rwDECLARE, lineIndex ); return(rwDECLARE); }
  182. "singleton" { CMDlval.i = MakeToken< int >( rwDECLARESINGLETON, lineIndex ); return(rwDECLARESINGLETON); }
  183. "datablock" { CMDlval.i = MakeToken< int >( rwDATABLOCK, lineIndex ); return(rwDATABLOCK); }
  184. "case" { CMDlval.i = MakeToken< int >( rwCASE, lineIndex ); return(rwCASE); }
  185. "switch$" { CMDlval.i = MakeToken< int >( rwSWITCHSTR, lineIndex ); return(rwSWITCHSTR); }
  186. "switch" { CMDlval.i = MakeToken< int >( rwSWITCH, lineIndex ); return(rwSWITCH); }
  187. "default" { CMDlval.i = MakeToken< int >( rwDEFAULT, lineIndex ); return(rwDEFAULT); }
  188. "package" { CMDlval.i = MakeToken< int >( rwPACKAGE, lineIndex ); return(rwPACKAGE); }
  189. "namespace" { CMDlval.i = MakeToken< int >( rwNAMESPACE, lineIndex ); return(rwNAMESPACE); }
  190. "true" { CMDlval.i = MakeToken< int >( 1, lineIndex ); return INTCONST; }
  191. "false" { CMDlval.i = MakeToken< int >( 0, lineIndex ); return INTCONST; }
  192. {VAR} { return(Sc_ScanVar()); }
  193. {ID} { return Sc_ScanIdent(); }
  194. 0[xX]{HEXDIGIT}+ return(Sc_ScanHex());
  195. {INTEGER} { CMDtext[CMDleng] = 0; CMDlval.i = MakeToken< int >( dAtoi(CMDtext), lineIndex ); return INTCONST; }
  196. {FLOAT} return Sc_ScanNum();
  197. {ILID} return(ILLEGAL_TOKEN);
  198. . return(ILLEGAL_TOKEN);
  199. %%
  200. static const char *scanBuffer;
  201. static const char *fileName;
  202. static int scanIndex;
  203. const char * CMDGetCurrentFile()
  204. {
  205. return fileName;
  206. }
  207. int CMDGetCurrentLine()
  208. {
  209. return lineIndex;
  210. }
  211. extern bool gConsoleSyntaxError;
  212. void CMDerror(char *format, ...)
  213. {
  214. Compiler::gSyntaxError = true;
  215. const int BUFMAX = 1024;
  216. char tempBuf[BUFMAX];
  217. va_list args;
  218. va_start( args, format );
  219. #ifdef TORQUE_OS_WIN
  220. _vsnprintf( tempBuf, BUFMAX, format, args );
  221. #else
  222. vsnprintf( tempBuf, BUFMAX, format, args );
  223. #endif
  224. va_end(args);
  225. if(fileName)
  226. {
  227. Con::errorf(ConsoleLogEntry::Script, "%s Line: %d - %s", fileName, lineIndex, tempBuf);
  228. #ifndef NO_ADVANCED_ERROR_REPORT
  229. // dhc - lineIndex is bogus. let's try to add some sanity back in.
  230. int i,j,n;
  231. char c;
  232. int linediv = 1;
  233. // first, walk the buffer, trying to detect line ending type.
  234. // this is imperfect, if inconsistant line endings exist...
  235. for (i=0; i<scanIndex; i++)
  236. {
  237. c = scanBuffer[i];
  238. if (c=='\r' && scanBuffer[i+1]=='\n') linediv = 2; // crlf detected
  239. if (c=='\r' || c=='\n' || c==0) break; // enough for us to stop.
  240. }
  241. // grab some of the chars starting at the error location - lineending.
  242. i = 1; j = 0; n = 1;
  243. // find prev lineending
  244. while (n<BUFMAX-8 && i<scanIndex) // cap at file start
  245. {
  246. c = scanBuffer[scanIndex-i];
  247. if ((c=='\r' || c=='\n') && i>BUFMAX>>2) break; // at least get a little data
  248. n++; i++;
  249. }
  250. // find next lineending
  251. while (n<BUFMAX-8 && j<BUFMAX>>1) // cap at half-buf-size forward
  252. {
  253. c = scanBuffer[scanIndex+j];
  254. if (c==0) break;
  255. if ((c=='\r' || c=='\n') && j>BUFMAX>>2) break; // at least get a little data
  256. n++; j++;
  257. }
  258. if (i) i--; // chop off extra linefeed.
  259. if (j) j--; // chop off extra linefeed.
  260. // build our little text block
  261. if (i) dStrncpy(tempBuf,scanBuffer+scanIndex-i,i);
  262. dStrncpy(tempBuf+i,"##", 2); // bracketing.
  263. tempBuf[i+2] = scanBuffer[scanIndex]; // copy the halt character.
  264. dStrncpy(tempBuf+i+3,"##", 2); // bracketing.
  265. if (j) dStrncpy(tempBuf+i+5,scanBuffer+scanIndex+1,j); // +1 to go past current char.
  266. tempBuf[i+j+5] = 0; // null terminate
  267. for(n=0; n<i+j+5; n++) // convert CR to LF if alone...
  268. if (tempBuf[n]=='\r' && tempBuf[n+1]!='\n') tempBuf[n] = '\n';
  269. // write out to console the advanced error report
  270. Con::warnf(ConsoleLogEntry::Script, ">>> Advanced script error report. Line %d.", lineIndex);
  271. Con::warnf(ConsoleLogEntry::Script, ">>> Some error context, with ## on sides of error halt:");
  272. Con::errorf(ConsoleLogEntry::Script, "%s", tempBuf);
  273. Con::warnf(ConsoleLogEntry::Script, ">>> Error report complete.\n");
  274. #endif
  275. // Update the script-visible error buffer.
  276. const char *prevStr = Con::getVariable("$ScriptError");
  277. if (prevStr[0])
  278. dSprintf(tempBuf, sizeof(tempBuf), "%s\n%s Line: %d - Syntax error.", prevStr, fileName, lineIndex);
  279. else
  280. dSprintf(tempBuf, sizeof(tempBuf), "%s Line: %d - Syntax error.", fileName, lineIndex);
  281. Con::setVariable("$ScriptError", tempBuf);
  282. // We also need to mark that we came up with a new error.
  283. static S32 sScriptErrorHash=1000;
  284. Con::setIntVariable("$ScriptErrorHash", sScriptErrorHash++);
  285. }
  286. else
  287. Con::errorf(ConsoleLogEntry::Script, tempBuf);
  288. }
  289. void CMDSetScanBuffer(const char *sb, const char *fn)
  290. {
  291. scanBuffer = sb;
  292. fileName = fn;
  293. scanIndex = 0;
  294. lineIndex = 1;
  295. }
  296. int CMDgetc()
  297. {
  298. int ret = scanBuffer[scanIndex];
  299. if(ret)
  300. scanIndex++;
  301. else
  302. ret = -1;
  303. return ret;
  304. }
  305. int CMDwrap()
  306. {
  307. return 1;
  308. }
  309. static int Sc_ScanVar()
  310. {
  311. // Truncate the temp buffer...
  312. CMDtext[CMDleng] = 0;
  313. // Make it a stringtable string!
  314. CMDlval.s = MakeToken< StringTableEntry >( StringTable->insert(CMDtext), lineIndex );
  315. return(VAR);
  316. }
  317. static int charConv(int in)
  318. {
  319. switch(in)
  320. {
  321. case 'r':
  322. return '\r';
  323. case 'n':
  324. return '\n';
  325. case 't':
  326. return '\t';
  327. default:
  328. return in;
  329. }
  330. }
  331. static int getHexDigit(char c)
  332. {
  333. if(c >= '0' && c <= '9')
  334. return c - '0';
  335. if(c >= 'A' && c <= 'F')
  336. return c - 'A' + 10;
  337. if(c >= 'a' && c <= 'f')
  338. return c - 'a' + 10;
  339. return -1;
  340. }
  341. static int Sc_ScanDocBlock()
  342. {
  343. S32 len = dStrlen(CMDtext);
  344. char* text = (char *) consoleAlloc(len + 1);
  345. S32 line = lineIndex;
  346. for( S32 i = 0, j = 0; j <= len; j++ )
  347. {
  348. if( ( j <= (len - 2) ) && ( CMDtext[j] == '/' ) && ( CMDtext[j + 1] == '/' ) && ( CMDtext[j + 2] == '/' ) )
  349. {
  350. j += 2;
  351. continue;
  352. }
  353. if( CMDtext[j] == '\r' )
  354. continue;
  355. if( CMDtext[j] == '\n' )
  356. lineIndex++;
  357. text[i++] = CMDtext[j];
  358. }
  359. CMDlval.str = MakeToken< char* >( text, line );
  360. return(DOCBLOCK);
  361. }
  362. static int Sc_ScanString(int ret)
  363. {
  364. CMDtext[CMDleng - 1] = 0;
  365. if(!collapseEscape(CMDtext+1))
  366. return -1;
  367. char* buffer = ( char* ) consoleAlloc( dStrlen( CMDtext ) );
  368. dStrcpy( buffer, CMDtext + 1 );
  369. CMDlval.str = MakeToken< char* >( buffer, lineIndex );
  370. return ret;
  371. }
  372. static int Sc_ScanIdent()
  373. {
  374. ConsoleBaseType *type;
  375. CMDtext[CMDleng] = 0;
  376. if((type = ConsoleBaseType::getTypeByName(CMDtext)) != NULL)
  377. {
  378. /* It's a type */
  379. CMDlval.i = MakeToken< int >( type->getTypeID(), lineIndex );
  380. return TYPEIDENT;
  381. }
  382. /* It's an identifier */
  383. CMDlval.s = MakeToken< StringTableEntry >( StringTable->insert(CMDtext), lineIndex );
  384. return IDENT;
  385. }
  386. void expandEscape(char *dest, const char *src)
  387. {
  388. U8 c;
  389. while((c = (U8) *src++) != 0)
  390. {
  391. if(c == '\"')
  392. {
  393. *dest++ = '\\';
  394. *dest++ = '\"';
  395. }
  396. else if(c == '\\')
  397. {
  398. *dest++ = '\\';
  399. *dest++ = '\\';
  400. }
  401. else if(c == '\r')
  402. {
  403. *dest++ = '\\';
  404. *dest++ = 'r';
  405. }
  406. else if(c == '\n')
  407. {
  408. *dest++ = '\\';
  409. *dest++ = 'n';
  410. }
  411. else if(c == '\t')
  412. {
  413. *dest++ = '\\';
  414. *dest++ = 't';
  415. }
  416. else if(c == '\'')
  417. {
  418. *dest++ = '\\';
  419. *dest++ = '\'';
  420. }
  421. else if((c >= 1 && c <= 7) ||
  422. (c >= 11 && c <= 12) ||
  423. (c >= 14 && c <= 15))
  424. {
  425. /* Remap around: \b = 0x8, \t = 0x9, \n = 0xa, \r = 0xd */
  426. static U8 expandRemap[15] = { 0x0,
  427. 0x0,
  428. 0x1,
  429. 0x2,
  430. 0x3,
  431. 0x4,
  432. 0x5,
  433. 0x6,
  434. 0x0,
  435. 0x0,
  436. 0x0,
  437. 0x7,
  438. 0x8,
  439. 0x0,
  440. 0x9 };
  441. *dest++ = '\\';
  442. *dest++ = 'c';
  443. if(c == 15)
  444. *dest++ = 'r';
  445. else if(c == 16)
  446. *dest++ = 'p';
  447. else if(c == 17)
  448. *dest++ = 'o';
  449. else
  450. *dest++ = expandRemap[c] + '0';
  451. }
  452. else if(c < 32)
  453. {
  454. *dest++ = '\\';
  455. *dest++ = 'x';
  456. S32 dig1 = c >> 4;
  457. S32 dig2 = c & 0xf;
  458. if(dig1 < 10)
  459. dig1 += '0';
  460. else
  461. dig1 += 'A' - 10;
  462. if(dig2 < 10)
  463. dig2 += '0';
  464. else
  465. dig2 += 'A' - 10;
  466. *dest++ = dig1;
  467. *dest++ = dig2;
  468. }
  469. else
  470. *dest++ = c;
  471. }
  472. *dest = '\0';
  473. }
  474. bool collapseEscape(char *buf)
  475. {
  476. S32 len = dStrlen(buf) + 1;
  477. for(S32 i = 0; i < len;)
  478. {
  479. if(buf[i] == '\\')
  480. {
  481. if(buf[i+1] == 'x')
  482. {
  483. S32 dig1 = getHexDigit(buf[i+2]);
  484. if(dig1 == -1)
  485. return false;
  486. S32 dig2 = getHexDigit(buf[i+3]);
  487. if(dig2 == -1)
  488. return false;
  489. buf[i] = dig1 * 16 + dig2;
  490. dMemmove(buf + i + 1, buf + i + 4, len - i - 3);
  491. len -= 3;
  492. i++;
  493. }
  494. else if(buf[i+1] == 'c')
  495. {
  496. /* Remap around: \b = 0x8, \t = 0x9, \n = 0xa, \r = 0xd */
  497. static U8 collapseRemap[10] = { 0x1,
  498. 0x2,
  499. 0x3,
  500. 0x4,
  501. 0x5,
  502. 0x6,
  503. 0x7,
  504. 0xb,
  505. 0xc,
  506. 0xe };
  507. if(buf[i+2] == 'r')
  508. buf[i] = 15;
  509. else if(buf[i+2] == 'p')
  510. buf[i] = 16;
  511. else if(buf[i+2] == 'o')
  512. buf[i] = 17;
  513. else
  514. {
  515. int dig1 = buf[i+2] - '0';
  516. if(dig1 < 0 || dig1 > 9)
  517. return false;
  518. buf[i] = collapseRemap[dig1];
  519. }
  520. // Make sure we don't put 0x1 at the beginning of the string.
  521. if ((buf[i] == 0x1) && (i == 0))
  522. {
  523. buf[i] = 0x2;
  524. buf[i+1] = 0x1;
  525. dMemmove(buf + i + 2, buf + i + 3, len - i - 1);
  526. len -= 1;
  527. }
  528. else
  529. {
  530. dMemmove(buf + i + 1, buf + i + 3, len - i - 2);
  531. len -= 2;
  532. }
  533. i++;
  534. }
  535. else
  536. {
  537. buf[i] = charConv(buf[i+1]);
  538. dMemmove(buf + i + 1, buf + i + 2, len - i - 1);
  539. len--;
  540. i++;
  541. }
  542. }
  543. else
  544. i++;
  545. }
  546. return true;
  547. }
  548. static int Sc_ScanNum()
  549. {
  550. CMDtext[CMDleng] = 0;
  551. CMDlval.f = MakeToken< double >( dAtof(CMDtext), lineIndex );
  552. return(FLTCONST);
  553. }
  554. static int Sc_ScanHex()
  555. {
  556. S32 val = 0;
  557. dSscanf(CMDtext, "%x", &val);
  558. CMDlval.i = MakeToken< int >( val, lineIndex );
  559. return INTCONST;
  560. }
  561. void CMD_reset()
  562. {
  563. CMDrestart(NULL);
  564. }