]> git.saurik.com Git - wxWidgets.git/blob - src/stc/scintilla/lexers/LexCPP.cxx
wxRTC text box layout fixes
[wxWidgets.git] / src / stc / scintilla / lexers / LexCPP.cxx
1 // Scintilla source code edit control
2 /** @file LexCPP.cxx
3 ** Lexer for C++, C, Java, and JavaScript.
4 ** Further folding features and configuration properties added by "Udo Lechner" <dlchnr(at)gmx(dot)net>
5 **/
6 // Copyright 1998-2005 by Neil Hodgson <neilh@scintilla.org>
7 // The License.txt file describes the conditions under which this software may be distributed.
8
9 #include <stdlib.h>
10 #include <string.h>
11 #include <ctype.h>
12 #include <stdio.h>
13 #include <stdarg.h>
14 #include <assert.h>
15
16 #include <string>
17 #include <vector>
18 #include <map>
19 #include <algorithm>
20
21 #include "ILexer.h"
22 #include "Scintilla.h"
23 #include "SciLexer.h"
24
25 #include "WordList.h"
26 #include "LexAccessor.h"
27 #include "Accessor.h"
28 #include "StyleContext.h"
29 #include "CharacterSet.h"
30 #include "LexerModule.h"
31 #include "OptionSet.h"
32 #include "SparseState.h"
33
34 #ifdef SCI_NAMESPACE
35 using namespace Scintilla;
36 #endif
37
38 static bool IsSpaceEquiv(int state) {
39 return (state <= SCE_C_COMMENTDOC) ||
40 // including SCE_C_DEFAULT, SCE_C_COMMENT, SCE_C_COMMENTLINE
41 (state == SCE_C_COMMENTLINEDOC) || (state == SCE_C_COMMENTDOCKEYWORD) ||
42 (state == SCE_C_COMMENTDOCKEYWORDERROR);
43 }
44
45 // Preconditions: sc.currentPos points to a character after '+' or '-'.
46 // The test for pos reaching 0 should be redundant,
47 // and is in only for safety measures.
48 // Limitation: this code will give the incorrect answer for code like
49 // a = b+++/ptn/...
50 // Putting a space between the '++' post-inc operator and the '+' binary op
51 // fixes this, and is highly recommended for readability anyway.
52 static bool FollowsPostfixOperator(StyleContext &sc, LexAccessor &styler) {
53 int pos = (int) sc.currentPos;
54 while (--pos > 0) {
55 char ch = styler[pos];
56 if (ch == '+' || ch == '-') {
57 return styler[pos - 1] == ch;
58 }
59 }
60 return false;
61 }
62
63 static bool followsReturnKeyword(StyleContext &sc, LexAccessor &styler) {
64 // Don't look at styles, so no need to flush.
65 int pos = (int) sc.currentPos;
66 int currentLine = styler.GetLine(pos);
67 int lineStartPos = styler.LineStart(currentLine);
68 char ch;
69 while (--pos > lineStartPos) {
70 ch = styler.SafeGetCharAt(pos);
71 if (ch != ' ' && ch != '\t') {
72 break;
73 }
74 }
75 const char *retBack = "nruter";
76 const char *s = retBack;
77 while (*s
78 && pos >= lineStartPos
79 && styler.SafeGetCharAt(pos) == *s) {
80 s++;
81 pos--;
82 }
83 return !*s;
84 }
85
86 static std::string GetRestOfLine(LexAccessor &styler, int start, bool allowSpace) {
87 std::string restOfLine;
88 int i =0;
89 char ch = styler.SafeGetCharAt(start, '\n');
90 while ((ch != '\r') && (ch != '\n')) {
91 if (allowSpace || (ch != ' '))
92 restOfLine += ch;
93 i++;
94 ch = styler.SafeGetCharAt(start + i, '\n');
95 }
96 return restOfLine;
97 }
98
99 static bool IsStreamCommentStyle(int style) {
100 return style == SCE_C_COMMENT ||
101 style == SCE_C_COMMENTDOC ||
102 style == SCE_C_COMMENTDOCKEYWORD ||
103 style == SCE_C_COMMENTDOCKEYWORDERROR;
104 }
105
106 static std::vector<std::string> Tokenize(const std::string &s) {
107 // Break into space separated tokens
108 std::string word;
109 std::vector<std::string> tokens;
110 for (const char *cp = s.c_str(); *cp; cp++) {
111 if ((*cp == ' ') || (*cp == '\t')) {
112 if (!word.empty()) {
113 tokens.push_back(word);
114 word = "";
115 }
116 } else {
117 word += *cp;
118 }
119 }
120 if (!word.empty()) {
121 tokens.push_back(word);
122 }
123 return tokens;
124 }
125
126 struct PPDefinition {
127 int line;
128 std::string key;
129 std::string value;
130 PPDefinition(int line_, const std::string &key_, const std::string &value_) :
131 line(line_), key(key_), value(value_) {
132 }
133 };
134
135 class LinePPState {
136 int state;
137 int ifTaken;
138 int level;
139 bool ValidLevel() const {
140 return level >= 0 && level < 32;
141 }
142 int maskLevel() const {
143 return 1 << level;
144 }
145 public:
146 LinePPState() : state(0), ifTaken(0), level(-1) {
147 }
148 bool IsInactive() const {
149 return state != 0;
150 }
151 bool CurrentIfTaken() {
152 return (ifTaken & maskLevel()) != 0;
153 }
154 void StartSection(bool on) {
155 level++;
156 if (ValidLevel()) {
157 if (on) {
158 state &= ~maskLevel();
159 ifTaken |= maskLevel();
160 } else {
161 state |= maskLevel();
162 ifTaken &= ~maskLevel();
163 }
164 }
165 }
166 void EndSection() {
167 if (ValidLevel()) {
168 state &= ~maskLevel();
169 ifTaken &= ~maskLevel();
170 }
171 level--;
172 }
173 void InvertCurrentLevel() {
174 if (ValidLevel()) {
175 state ^= maskLevel();
176 ifTaken |= maskLevel();
177 }
178 }
179 };
180
181 // Hold the preprocessor state for each line seen.
182 // Currently one entry per line but could become sparse with just one entry per preprocessor line.
183 class PPStates {
184 std::vector<LinePPState> vlls;
185 public:
186 LinePPState ForLine(int line) {
187 if ((line > 0) && (vlls.size() > static_cast<size_t>(line))) {
188 return vlls[line];
189 } else {
190 return LinePPState();
191 }
192 }
193 void Add(int line, LinePPState lls) {
194 vlls.resize(line+1);
195 vlls[line] = lls;
196 }
197 };
198
199 // An individual named option for use in an OptionSet
200
201 // Options used for LexerCPP
202 struct OptionsCPP {
203 bool stylingWithinPreprocessor;
204 bool identifiersAllowDollars;
205 bool trackPreprocessor;
206 bool updatePreprocessor;
207 bool triplequotedStrings;
208 bool hashquotedStrings;
209 bool fold;
210 bool foldSyntaxBased;
211 bool foldComment;
212 bool foldCommentMultiline;
213 bool foldCommentExplicit;
214 std::string foldExplicitStart;
215 std::string foldExplicitEnd;
216 bool foldExplicitAnywhere;
217 bool foldPreprocessor;
218 bool foldCompact;
219 bool foldAtElse;
220 OptionsCPP() {
221 stylingWithinPreprocessor = false;
222 identifiersAllowDollars = true;
223 trackPreprocessor = true;
224 updatePreprocessor = true;
225 triplequotedStrings = false;
226 hashquotedStrings = false;
227 fold = false;
228 foldSyntaxBased = true;
229 foldComment = false;
230 foldCommentMultiline = true;
231 foldCommentExplicit = true;
232 foldExplicitStart = "";
233 foldExplicitEnd = "";
234 foldExplicitAnywhere = false;
235 foldPreprocessor = false;
236 foldCompact = false;
237 foldAtElse = false;
238 }
239 };
240
241 static const char *const cppWordLists[] = {
242 "Primary keywords and identifiers",
243 "Secondary keywords and identifiers",
244 "Documentation comment keywords",
245 "Global classes and typedefs",
246 "Preprocessor definitions",
247 0,
248 };
249
250 struct OptionSetCPP : public OptionSet<OptionsCPP> {
251 OptionSetCPP() {
252 DefineProperty("styling.within.preprocessor", &OptionsCPP::stylingWithinPreprocessor,
253 "For C++ code, determines whether all preprocessor code is styled in the "
254 "preprocessor style (0, the default) or only from the initial # to the end "
255 "of the command word(1).");
256
257 DefineProperty("lexer.cpp.allow.dollars", &OptionsCPP::identifiersAllowDollars,
258 "Set to 0 to disallow the '$' character in identifiers with the cpp lexer.");
259
260 DefineProperty("lexer.cpp.track.preprocessor", &OptionsCPP::trackPreprocessor,
261 "Set to 1 to interpret #if/#else/#endif to grey out code that is not active.");
262
263 DefineProperty("lexer.cpp.update.preprocessor", &OptionsCPP::updatePreprocessor,
264 "Set to 1 to update preprocessor definitions when #define found.");
265
266 DefineProperty("lexer.cpp.triplequoted.strings", &OptionsCPP::triplequotedStrings,
267 "Set to 1 to enable highlighting of triple-quoted strings.");
268
269 DefineProperty("lexer.cpp.hashquoted.strings", &OptionsCPP::hashquotedStrings,
270 "Set to 1 to enable highlighting of hash-quoted strings.");
271
272 DefineProperty("fold", &OptionsCPP::fold);
273
274 DefineProperty("fold.cpp.syntax.based", &OptionsCPP::foldSyntaxBased,
275 "Set this property to 0 to disable syntax based folding.");
276
277 DefineProperty("fold.comment", &OptionsCPP::foldComment,
278 "This option enables folding multi-line comments and explicit fold points when using the C++ lexer. "
279 "Explicit fold points allows adding extra folding by placing a //{ comment at the start and a //} "
280 "at the end of a section that should fold.");
281
282 DefineProperty("fold.cpp.comment.multiline", &OptionsCPP::foldCommentMultiline,
283 "Set this property to 0 to disable folding multi-line comments when fold.comment=1.");
284
285 DefineProperty("fold.cpp.comment.explicit", &OptionsCPP::foldCommentExplicit,
286 "Set this property to 0 to disable folding explicit fold points when fold.comment=1.");
287
288 DefineProperty("fold.cpp.explicit.start", &OptionsCPP::foldExplicitStart,
289 "The string to use for explicit fold start points, replacing the standard //{.");
290
291 DefineProperty("fold.cpp.explicit.end", &OptionsCPP::foldExplicitEnd,
292 "The string to use for explicit fold end points, replacing the standard //}.");
293
294 DefineProperty("fold.cpp.explicit.anywhere", &OptionsCPP::foldExplicitAnywhere,
295 "Set this property to 1 to enable explicit fold points anywhere, not just in line comments.");
296
297 DefineProperty("fold.preprocessor", &OptionsCPP::foldPreprocessor,
298 "This option enables folding preprocessor directives when using the C++ lexer. "
299 "Includes C#'s explicit #region and #endregion folding directives.");
300
301 DefineProperty("fold.compact", &OptionsCPP::foldCompact);
302
303 DefineProperty("fold.at.else", &OptionsCPP::foldAtElse,
304 "This option enables C++ folding on a \"} else {\" line of an if statement.");
305
306 DefineWordListSets(cppWordLists);
307 }
308 };
309
310 class LexerCPP : public ILexer {
311 bool caseSensitive;
312 CharacterSet setWord;
313 CharacterSet setNegationOp;
314 CharacterSet setArithmethicOp;
315 CharacterSet setRelOp;
316 CharacterSet setLogicalOp;
317 PPStates vlls;
318 std::vector<PPDefinition> ppDefineHistory;
319 WordList keywords;
320 WordList keywords2;
321 WordList keywords3;
322 WordList keywords4;
323 WordList ppDefinitions;
324 std::map<std::string, std::string> preprocessorDefinitionsStart;
325 OptionsCPP options;
326 OptionSetCPP osCPP;
327 SparseState<std::string> rawStringTerminators;
328 enum { activeFlag = 0x40 };
329 public:
330 LexerCPP(bool caseSensitive_) :
331 caseSensitive(caseSensitive_),
332 setWord(CharacterSet::setAlphaNum, "._", 0x80, true),
333 setNegationOp(CharacterSet::setNone, "!"),
334 setArithmethicOp(CharacterSet::setNone, "+-/*%"),
335 setRelOp(CharacterSet::setNone, "=!<>"),
336 setLogicalOp(CharacterSet::setNone, "|&") {
337 }
338 virtual ~LexerCPP() {
339 }
340 void SCI_METHOD Release() {
341 delete this;
342 }
343 int SCI_METHOD Version() const {
344 return lvOriginal;
345 }
346 const char * SCI_METHOD PropertyNames() {
347 return osCPP.PropertyNames();
348 }
349 int SCI_METHOD PropertyType(const char *name) {
350 return osCPP.PropertyType(name);
351 }
352 const char * SCI_METHOD DescribeProperty(const char *name) {
353 return osCPP.DescribeProperty(name);
354 }
355 int SCI_METHOD PropertySet(const char *key, const char *val);
356 const char * SCI_METHOD DescribeWordListSets() {
357 return osCPP.DescribeWordListSets();
358 }
359 int SCI_METHOD WordListSet(int n, const char *wl);
360 void SCI_METHOD Lex(unsigned int startPos, int length, int initStyle, IDocument *pAccess);
361 void SCI_METHOD Fold(unsigned int startPos, int length, int initStyle, IDocument *pAccess);
362
363 void * SCI_METHOD PrivateCall(int, void *) {
364 return 0;
365 }
366
367 static ILexer *LexerFactoryCPP() {
368 return new LexerCPP(true);
369 }
370 static ILexer *LexerFactoryCPPInsensitive() {
371 return new LexerCPP(false);
372 }
373 static int MaskActive(int style) {
374 return style & ~activeFlag;
375 }
376 void EvaluateTokens(std::vector<std::string> &tokens);
377 bool EvaluateExpression(const std::string &expr, const std::map<std::string, std::string> &preprocessorDefinitions);
378 };
379
380 int SCI_METHOD LexerCPP::PropertySet(const char *key, const char *val) {
381 if (osCPP.PropertySet(&options, key, val)) {
382 if (strcmp(key, "lexer.cpp.allow.dollars") == 0) {
383 setWord = CharacterSet(CharacterSet::setAlphaNum, "._", 0x80, true);
384 if (options.identifiersAllowDollars) {
385 setWord.Add('$');
386 }
387 }
388 return 0;
389 }
390 return -1;
391 }
392
393 int SCI_METHOD LexerCPP::WordListSet(int n, const char *wl) {
394 WordList *wordListN = 0;
395 switch (n) {
396 case 0:
397 wordListN = &keywords;
398 break;
399 case 1:
400 wordListN = &keywords2;
401 break;
402 case 2:
403 wordListN = &keywords3;
404 break;
405 case 3:
406 wordListN = &keywords4;
407 break;
408 case 4:
409 wordListN = &ppDefinitions;
410 break;
411 }
412 int firstModification = -1;
413 if (wordListN) {
414 WordList wlNew;
415 wlNew.Set(wl);
416 if (*wordListN != wlNew) {
417 wordListN->Set(wl);
418 firstModification = 0;
419 if (n == 4) {
420 // Rebuild preprocessorDefinitions
421 preprocessorDefinitionsStart.clear();
422 for (int nDefinition = 0; nDefinition < ppDefinitions.len; nDefinition++) {
423 char *cpDefinition = ppDefinitions.words[nDefinition];
424 char *cpEquals = strchr(cpDefinition, '=');
425 if (cpEquals) {
426 std::string name(cpDefinition, cpEquals - cpDefinition);
427 std::string val(cpEquals+1);
428 preprocessorDefinitionsStart[name] = val;
429 } else {
430 std::string name(cpDefinition);
431 std::string val("1");
432 preprocessorDefinitionsStart[name] = val;
433 }
434 }
435 }
436 }
437 }
438 return firstModification;
439 }
440
441 // Functor used to truncate history
442 struct After {
443 int line;
444 After(int line_) : line(line_) {}
445 bool operator()(PPDefinition &p) const {
446 return p.line > line;
447 }
448 };
449
450 void SCI_METHOD LexerCPP::Lex(unsigned int startPos, int length, int initStyle, IDocument *pAccess) {
451 LexAccessor styler(pAccess);
452
453 CharacterSet setOKBeforeRE(CharacterSet::setNone, "([{=,:;!%^&*|?~+-");
454 CharacterSet setCouldBePostOp(CharacterSet::setNone, "+-");
455
456 CharacterSet setDoxygen(CharacterSet::setAlpha, "$@\\&<>#{}[]");
457
458 CharacterSet setWordStart(CharacterSet::setAlpha, "_", 0x80, true);
459
460 if (options.identifiersAllowDollars) {
461 setWordStart.Add('$');
462 }
463
464 int chPrevNonWhite = ' ';
465 int visibleChars = 0;
466 bool lastWordWasUUID = false;
467 int styleBeforeDCKeyword = SCE_C_DEFAULT;
468 bool continuationLine = false;
469 bool isIncludePreprocessor = false;
470
471 int lineCurrent = styler.GetLine(startPos);
472 if ((MaskActive(initStyle) == SCE_C_PREPROCESSOR) ||
473 (MaskActive(initStyle) == SCE_C_COMMENTLINE) ||
474 (MaskActive(initStyle) == SCE_C_COMMENTLINEDOC)) {
475 // Set continuationLine if last character of previous line is '\'
476 if (lineCurrent > 0) {
477 int chBack = styler.SafeGetCharAt(startPos-1, 0);
478 int chBack2 = styler.SafeGetCharAt(startPos-2, 0);
479 int lineEndChar = '!';
480 if (chBack2 == '\r' && chBack == '\n') {
481 lineEndChar = styler.SafeGetCharAt(startPos-3, 0);
482 } else if (chBack == '\n' || chBack == '\r') {
483 lineEndChar = chBack2;
484 }
485 continuationLine = lineEndChar == '\\';
486 }
487 }
488
489 // look back to set chPrevNonWhite properly for better regex colouring
490 if (startPos > 0) {
491 int back = startPos;
492 while (--back && IsSpaceEquiv(MaskActive(styler.StyleAt(back))))
493 ;
494 if (MaskActive(styler.StyleAt(back)) == SCE_C_OPERATOR) {
495 chPrevNonWhite = styler.SafeGetCharAt(back);
496 }
497 }
498
499 StyleContext sc(startPos, length, initStyle, styler, 0x7f);
500 LinePPState preproc = vlls.ForLine(lineCurrent);
501
502 bool definitionsChanged = false;
503
504 // Truncate ppDefineHistory before current line
505
506 if (!options.updatePreprocessor)
507 ppDefineHistory.clear();
508
509 std::vector<PPDefinition>::iterator itInvalid = std::find_if(ppDefineHistory.begin(), ppDefineHistory.end(), After(lineCurrent-1));
510 if (itInvalid != ppDefineHistory.end()) {
511 ppDefineHistory.erase(itInvalid, ppDefineHistory.end());
512 definitionsChanged = true;
513 }
514
515 std::map<std::string, std::string> preprocessorDefinitions = preprocessorDefinitionsStart;
516 for (std::vector<PPDefinition>::iterator itDef = ppDefineHistory.begin(); itDef != ppDefineHistory.end(); ++itDef) {
517 preprocessorDefinitions[itDef->key] = itDef->value;
518 }
519
520 std::string rawStringTerminator = rawStringTerminators.ValueAt(lineCurrent-1);
521 SparseState<std::string> rawSTNew(lineCurrent);
522
523 int activitySet = preproc.IsInactive() ? activeFlag : 0;
524
525 for (; sc.More();) {
526
527 if (sc.atLineStart) {
528 // Using MaskActive() is not needed in the following statement.
529 // Inside inactive preprocessor declaration, state will be reset anyway at the end of this block.
530 if ((sc.state == SCE_C_STRING) || (sc.state == SCE_C_CHARACTER)) {
531 // Prevent SCE_C_STRINGEOL from leaking back to previous line which
532 // ends with a line continuation by locking in the state upto this position.
533 sc.SetState(sc.state);
534 }
535 if ((MaskActive(sc.state) == SCE_C_PREPROCESSOR) && (!continuationLine)) {
536 sc.SetState(SCE_C_DEFAULT|activitySet);
537 }
538 // Reset states to begining of colourise so no surprises
539 // if different sets of lines lexed.
540 visibleChars = 0;
541 lastWordWasUUID = false;
542 isIncludePreprocessor = false;
543 if (preproc.IsInactive()) {
544 activitySet = activeFlag;
545 sc.SetState(sc.state | activitySet);
546 }
547 }
548
549 if (sc.atLineEnd) {
550 lineCurrent++;
551 vlls.Add(lineCurrent, preproc);
552 if (rawStringTerminator != "") {
553 rawSTNew.Set(lineCurrent-1, rawStringTerminator);
554 }
555 }
556
557 // Handle line continuation generically.
558 if (sc.ch == '\\') {
559 if (sc.chNext == '\n' || sc.chNext == '\r') {
560 lineCurrent++;
561 vlls.Add(lineCurrent, preproc);
562 sc.Forward();
563 if (sc.ch == '\r' && sc.chNext == '\n') {
564 sc.Forward();
565 }
566 continuationLine = true;
567 sc.Forward();
568 continue;
569 }
570 }
571
572 const bool atLineEndBeforeSwitch = sc.atLineEnd;
573
574 // Determine if the current state should terminate.
575 switch (MaskActive(sc.state)) {
576 case SCE_C_OPERATOR:
577 sc.SetState(SCE_C_DEFAULT|activitySet);
578 break;
579 case SCE_C_NUMBER:
580 // We accept almost anything because of hex. and number suffixes
581 if (!(setWord.Contains(sc.ch) || ((sc.ch == '+' || sc.ch == '-') && (sc.chPrev == 'e' || sc.chPrev == 'E')))) {
582 sc.SetState(SCE_C_DEFAULT|activitySet);
583 }
584 break;
585 case SCE_C_IDENTIFIER:
586 if (!setWord.Contains(sc.ch) || (sc.ch == '.')) {
587 char s[1000];
588 if (caseSensitive) {
589 sc.GetCurrent(s, sizeof(s));
590 } else {
591 sc.GetCurrentLowered(s, sizeof(s));
592 }
593 if (keywords.InList(s)) {
594 lastWordWasUUID = strcmp(s, "uuid") == 0;
595 sc.ChangeState(SCE_C_WORD|activitySet);
596 } else if (keywords2.InList(s)) {
597 sc.ChangeState(SCE_C_WORD2|activitySet);
598 } else if (keywords4.InList(s)) {
599 sc.ChangeState(SCE_C_GLOBALCLASS|activitySet);
600 }
601 const bool literalString = sc.ch == '\"';
602 if (literalString || sc.ch == '\'') {
603 size_t lenS = strlen(s);
604 const bool raw = literalString && sc.chPrev == 'R';
605 if (raw)
606 s[lenS--] = '\0';
607 bool valid =
608 (lenS == 0) ||
609 ((lenS == 1) && ((s[0] == 'L') || (s[0] == 'u') || (s[0] == 'U'))) ||
610 ((lenS == 2) && literalString && (s[0] == 'u') && (s[1] == '8'));
611 if (valid) {
612 if (literalString)
613 sc.ChangeState((raw ? SCE_C_STRINGRAW : SCE_C_STRING)|activitySet);
614 else
615 sc.ChangeState(SCE_C_CHARACTER|activitySet);
616 }
617 }
618 sc.SetState(SCE_C_DEFAULT|activitySet);
619 }
620 break;
621 case SCE_C_PREPROCESSOR:
622 if (options.stylingWithinPreprocessor) {
623 if (IsASpace(sc.ch)) {
624 sc.SetState(SCE_C_DEFAULT|activitySet);
625 }
626 } else {
627 if (sc.Match('/', '*')) {
628 sc.SetState(SCE_C_PREPROCESSORCOMMENT|activitySet);
629 sc.Forward(); // Eat the *
630 } else if (sc.Match('/', '/')) {
631 sc.SetState(SCE_C_DEFAULT|activitySet);
632 }
633 }
634 break;
635 case SCE_C_PREPROCESSORCOMMENT:
636 if (sc.Match('*', '/')) {
637 sc.Forward();
638 sc.ForwardSetState(SCE_C_PREPROCESSOR|activitySet);
639 continue; // Without advancing in case of '\'.
640 }
641 break;
642 case SCE_C_COMMENT:
643 if (sc.Match('*', '/')) {
644 sc.Forward();
645 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
646 }
647 break;
648 case SCE_C_COMMENTDOC:
649 if (sc.Match('*', '/')) {
650 sc.Forward();
651 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
652 } else if (sc.ch == '@' || sc.ch == '\\') { // JavaDoc and Doxygen support
653 // Verify that we have the conditions to mark a comment-doc-keyword
654 if ((IsASpace(sc.chPrev) || sc.chPrev == '*') && (!IsASpace(sc.chNext))) {
655 styleBeforeDCKeyword = SCE_C_COMMENTDOC;
656 sc.SetState(SCE_C_COMMENTDOCKEYWORD|activitySet);
657 }
658 }
659 break;
660 case SCE_C_COMMENTLINE:
661 if (sc.atLineStart && !continuationLine) {
662 sc.SetState(SCE_C_DEFAULT|activitySet);
663 }
664 break;
665 case SCE_C_COMMENTLINEDOC:
666 if (sc.atLineStart && !continuationLine) {
667 sc.SetState(SCE_C_DEFAULT|activitySet);
668 } else if (sc.ch == '@' || sc.ch == '\\') { // JavaDoc and Doxygen support
669 // Verify that we have the conditions to mark a comment-doc-keyword
670 if ((IsASpace(sc.chPrev) || sc.chPrev == '/' || sc.chPrev == '!') && (!IsASpace(sc.chNext))) {
671 styleBeforeDCKeyword = SCE_C_COMMENTLINEDOC;
672 sc.SetState(SCE_C_COMMENTDOCKEYWORD|activitySet);
673 }
674 }
675 break;
676 case SCE_C_COMMENTDOCKEYWORD:
677 if ((styleBeforeDCKeyword == SCE_C_COMMENTDOC) && sc.Match('*', '/')) {
678 sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR);
679 sc.Forward();
680 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
681 } else if (!setDoxygen.Contains(sc.ch)) {
682 char s[100];
683 if (caseSensitive) {
684 sc.GetCurrent(s, sizeof(s));
685 } else {
686 sc.GetCurrentLowered(s, sizeof(s));
687 }
688 if (!IsASpace(sc.ch) || !keywords3.InList(s + 1)) {
689 sc.ChangeState(SCE_C_COMMENTDOCKEYWORDERROR|activitySet);
690 }
691 sc.SetState(styleBeforeDCKeyword|activitySet);
692 }
693 break;
694 case SCE_C_STRING:
695 if (sc.atLineEnd) {
696 sc.ChangeState(SCE_C_STRINGEOL|activitySet);
697 } else if (isIncludePreprocessor) {
698 if (sc.ch == '>') {
699 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
700 isIncludePreprocessor = false;
701 }
702 } else if (sc.ch == '\\') {
703 if (sc.chNext == '\"' || sc.chNext == '\'' || sc.chNext == '\\') {
704 sc.Forward();
705 }
706 } else if (sc.ch == '\"') {
707 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
708 }
709 break;
710 case SCE_C_HASHQUOTEDSTRING:
711 if (sc.ch == '\\') {
712 if (sc.chNext == '\"' || sc.chNext == '\'' || sc.chNext == '\\') {
713 sc.Forward();
714 }
715 } else if (sc.ch == '\"') {
716 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
717 }
718 break;
719 case SCE_C_STRINGRAW:
720 if (sc.Match(rawStringTerminator.c_str())) {
721 for (size_t termPos=rawStringTerminator.size(); termPos; termPos--)
722 sc.Forward();
723 sc.SetState(SCE_C_DEFAULT|activitySet);
724 rawStringTerminator = "";
725 }
726 break;
727 case SCE_C_CHARACTER:
728 if (sc.atLineEnd) {
729 sc.ChangeState(SCE_C_STRINGEOL|activitySet);
730 } else if (sc.ch == '\\') {
731 if (sc.chNext == '\"' || sc.chNext == '\'' || sc.chNext == '\\') {
732 sc.Forward();
733 }
734 } else if (sc.ch == '\'') {
735 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
736 }
737 break;
738 case SCE_C_REGEX:
739 if (sc.atLineStart) {
740 sc.SetState(SCE_C_DEFAULT|activitySet);
741 } else if (sc.ch == '/') {
742 sc.Forward();
743 while ((sc.ch < 0x80) && islower(sc.ch))
744 sc.Forward(); // gobble regex flags
745 sc.SetState(SCE_C_DEFAULT|activitySet);
746 } else if (sc.ch == '\\') {
747 // Gobble up the quoted character
748 if (sc.chNext == '\\' || sc.chNext == '/') {
749 sc.Forward();
750 }
751 }
752 break;
753 case SCE_C_STRINGEOL:
754 if (sc.atLineStart) {
755 sc.SetState(SCE_C_DEFAULT|activitySet);
756 }
757 break;
758 case SCE_C_VERBATIM:
759 if (sc.ch == '\"') {
760 if (sc.chNext == '\"') {
761 sc.Forward();
762 } else {
763 sc.ForwardSetState(SCE_C_DEFAULT|activitySet);
764 }
765 }
766 break;
767 case SCE_C_TRIPLEVERBATIM:
768 if (sc.Match("\"\"\"")) {
769 while (sc.Match('"')) {
770 sc.Forward();
771 }
772 sc.SetState(SCE_C_DEFAULT|activitySet);
773 }
774 break;
775 case SCE_C_UUID:
776 if (sc.ch == '\r' || sc.ch == '\n' || sc.ch == ')') {
777 sc.SetState(SCE_C_DEFAULT|activitySet);
778 }
779 }
780
781 if (sc.atLineEnd && !atLineEndBeforeSwitch) {
782 // State exit processing consumed characters up to end of line.
783 lineCurrent++;
784 vlls.Add(lineCurrent, preproc);
785 }
786
787 // Determine if a new state should be entered.
788 if (MaskActive(sc.state) == SCE_C_DEFAULT) {
789 if (sc.Match('@', '\"')) {
790 sc.SetState(SCE_C_VERBATIM|activitySet);
791 sc.Forward();
792 } else if (options.triplequotedStrings && sc.Match("\"\"\"")) {
793 sc.SetState(SCE_C_TRIPLEVERBATIM|activitySet);
794 sc.Forward(2);
795 } else if (options.hashquotedStrings && sc.Match('#', '\"')) {
796 sc.SetState(SCE_C_HASHQUOTEDSTRING|activitySet);
797 sc.Forward();
798 } else if (IsADigit(sc.ch) || (sc.ch == '.' && IsADigit(sc.chNext))) {
799 if (lastWordWasUUID) {
800 sc.SetState(SCE_C_UUID|activitySet);
801 lastWordWasUUID = false;
802 } else {
803 sc.SetState(SCE_C_NUMBER|activitySet);
804 }
805 } else if (setWordStart.Contains(sc.ch) || (sc.ch == '@')) {
806 if (lastWordWasUUID) {
807 sc.SetState(SCE_C_UUID|activitySet);
808 lastWordWasUUID = false;
809 } else {
810 sc.SetState(SCE_C_IDENTIFIER|activitySet);
811 }
812 } else if (sc.Match('/', '*')) {
813 if (sc.Match("/**") || sc.Match("/*!")) { // Support of Qt/Doxygen doc. style
814 sc.SetState(SCE_C_COMMENTDOC|activitySet);
815 } else {
816 sc.SetState(SCE_C_COMMENT|activitySet);
817 }
818 sc.Forward(); // Eat the * so it isn't used for the end of the comment
819 } else if (sc.Match('/', '/')) {
820 if ((sc.Match("///") && !sc.Match("////")) || sc.Match("//!"))
821 // Support of Qt/Doxygen doc. style
822 sc.SetState(SCE_C_COMMENTLINEDOC|activitySet);
823 else
824 sc.SetState(SCE_C_COMMENTLINE|activitySet);
825 } else if (sc.ch == '/'
826 && (setOKBeforeRE.Contains(chPrevNonWhite)
827 || followsReturnKeyword(sc, styler))
828 && (!setCouldBePostOp.Contains(chPrevNonWhite)
829 || !FollowsPostfixOperator(sc, styler))) {
830 sc.SetState(SCE_C_REGEX|activitySet); // JavaScript's RegEx
831 } else if (sc.ch == '\"') {
832 if (sc.chPrev == 'R') {
833 styler.Flush();
834 if (MaskActive(styler.StyleAt(sc.currentPos - 1)) == SCE_C_STRINGRAW) {
835 sc.SetState(SCE_C_STRINGRAW|activitySet);
836 rawStringTerminator = ")";
837 for (int termPos = sc.currentPos + 1;; termPos++) {
838 char chTerminator = styler.SafeGetCharAt(termPos, '(');
839 if (chTerminator == '(')
840 break;
841 rawStringTerminator += chTerminator;
842 }
843 rawStringTerminator += '\"';
844 } else {
845 sc.SetState(SCE_C_STRING|activitySet);
846 }
847 } else {
848 sc.SetState(SCE_C_STRING|activitySet);
849 }
850 isIncludePreprocessor = false; // ensure that '>' won't end the string
851 } else if (isIncludePreprocessor && sc.ch == '<') {
852 sc.SetState(SCE_C_STRING|activitySet);
853 } else if (sc.ch == '\'') {
854 sc.SetState(SCE_C_CHARACTER|activitySet);
855 } else if (sc.ch == '#' && visibleChars == 0) {
856 // Preprocessor commands are alone on their line
857 sc.SetState(SCE_C_PREPROCESSOR|activitySet);
858 // Skip whitespace between # and preprocessor word
859 do {
860 sc.Forward();
861 } while ((sc.ch == ' ' || sc.ch == '\t') && sc.More());
862 if (sc.atLineEnd) {
863 sc.SetState(SCE_C_DEFAULT|activitySet);
864 } else if (sc.Match("include")) {
865 isIncludePreprocessor = true;
866 } else {
867 if (options.trackPreprocessor) {
868 if (sc.Match("ifdef") || sc.Match("ifndef")) {
869 bool isIfDef = sc.Match("ifdef");
870 int i = isIfDef ? 5 : 6;
871 std::string restOfLine = GetRestOfLine(styler, sc.currentPos + i + 1, false);
872 bool foundDef = preprocessorDefinitions.find(restOfLine) != preprocessorDefinitions.end();
873 preproc.StartSection(isIfDef == foundDef);
874 } else if (sc.Match("if")) {
875 std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 2, true);
876 bool ifGood = EvaluateExpression(restOfLine, preprocessorDefinitions);
877 preproc.StartSection(ifGood);
878 } else if (sc.Match("else")) {
879 if (!preproc.CurrentIfTaken()) {
880 preproc.InvertCurrentLevel();
881 activitySet = preproc.IsInactive() ? activeFlag : 0;
882 if (!activitySet)
883 sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
884 } else if (!preproc.IsInactive()) {
885 preproc.InvertCurrentLevel();
886 activitySet = preproc.IsInactive() ? activeFlag : 0;
887 if (!activitySet)
888 sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
889 }
890 } else if (sc.Match("elif")) {
891 // Ensure only one chosen out of #if .. #elif .. #elif .. #else .. #endif
892 if (!preproc.CurrentIfTaken()) {
893 // Similar to #if
894 std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 2, true);
895 bool ifGood = EvaluateExpression(restOfLine, preprocessorDefinitions);
896 if (ifGood) {
897 preproc.InvertCurrentLevel();
898 activitySet = preproc.IsInactive() ? activeFlag : 0;
899 if (!activitySet)
900 sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
901 }
902 } else if (!preproc.IsInactive()) {
903 preproc.InvertCurrentLevel();
904 activitySet = preproc.IsInactive() ? activeFlag : 0;
905 if (!activitySet)
906 sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
907 }
908 } else if (sc.Match("endif")) {
909 preproc.EndSection();
910 activitySet = preproc.IsInactive() ? activeFlag : 0;
911 sc.ChangeState(SCE_C_PREPROCESSOR|activitySet);
912 } else if (sc.Match("define")) {
913 if (options.updatePreprocessor && !preproc.IsInactive()) {
914 std::string restOfLine = GetRestOfLine(styler, sc.currentPos + 6, true);
915 if (restOfLine.find(")") == std::string::npos) { // Don't handle macros with arguments
916 std::vector<std::string> tokens = Tokenize(restOfLine);
917 std::string key;
918 std::string value("1");
919 if (tokens.size() >= 1) {
920 key = tokens[0];
921 if (tokens.size() >= 2) {
922 value = tokens[1];
923 }
924 preprocessorDefinitions[key] = value;
925 ppDefineHistory.push_back(PPDefinition(lineCurrent, key, value));
926 definitionsChanged = true;
927 }
928 }
929 }
930 }
931 }
932 }
933 } else if (isoperator(static_cast<char>(sc.ch))) {
934 sc.SetState(SCE_C_OPERATOR|activitySet);
935 }
936 }
937
938 if (!IsASpace(sc.ch) && !IsSpaceEquiv(MaskActive(sc.state))) {
939 chPrevNonWhite = sc.ch;
940 visibleChars++;
941 }
942 continuationLine = false;
943 sc.Forward();
944 }
945 const bool rawStringsChanged = rawStringTerminators.Merge(rawSTNew, lineCurrent);
946 if (definitionsChanged || rawStringsChanged)
947 styler.ChangeLexerState(startPos, startPos + length);
948 sc.Complete();
949 }
950
951 // Store both the current line's fold level and the next lines in the
952 // level store to make it easy to pick up with each increment
953 // and to make it possible to fiddle the current level for "} else {".
954
955 void SCI_METHOD LexerCPP::Fold(unsigned int startPos, int length, int initStyle, IDocument *pAccess) {
956
957 if (!options.fold)
958 return;
959
960 LexAccessor styler(pAccess);
961
962 unsigned int endPos = startPos + length;
963 int visibleChars = 0;
964 int lineCurrent = styler.GetLine(startPos);
965 int levelCurrent = SC_FOLDLEVELBASE;
966 if (lineCurrent > 0)
967 levelCurrent = styler.LevelAt(lineCurrent-1) >> 16;
968 int levelMinCurrent = levelCurrent;
969 int levelNext = levelCurrent;
970 char chNext = styler[startPos];
971 int styleNext = MaskActive(styler.StyleAt(startPos));
972 int style = MaskActive(initStyle);
973 const bool userDefinedFoldMarkers = !options.foldExplicitStart.empty() && !options.foldExplicitEnd.empty();
974 for (unsigned int i = startPos; i < endPos; i++) {
975 char ch = chNext;
976 chNext = styler.SafeGetCharAt(i + 1);
977 int stylePrev = style;
978 style = styleNext;
979 styleNext = MaskActive(styler.StyleAt(i + 1));
980 bool atEOL = (ch == '\r' && chNext != '\n') || (ch == '\n');
981 if (options.foldComment && options.foldCommentMultiline && IsStreamCommentStyle(style)) {
982 if (!IsStreamCommentStyle(stylePrev) && (stylePrev != SCE_C_COMMENTLINEDOC)) {
983 levelNext++;
984 } else if (!IsStreamCommentStyle(styleNext) && (styleNext != SCE_C_COMMENTLINEDOC) && !atEOL) {
985 // Comments don't end at end of line and the next character may be unstyled.
986 levelNext--;
987 }
988 }
989 if (options.foldComment && options.foldCommentExplicit && ((style == SCE_C_COMMENTLINE) || options.foldExplicitAnywhere)) {
990 if (userDefinedFoldMarkers) {
991 if (styler.Match(i, options.foldExplicitStart.c_str())) {
992 levelNext++;
993 } else if (styler.Match(i, options.foldExplicitEnd.c_str())) {
994 levelNext--;
995 }
996 } else {
997 if ((ch == '/') && (chNext == '/')) {
998 char chNext2 = styler.SafeGetCharAt(i + 2);
999 if (chNext2 == '{') {
1000 levelNext++;
1001 } else if (chNext2 == '}') {
1002 levelNext--;
1003 }
1004 }
1005 }
1006 }
1007 if (options.foldPreprocessor && (style == SCE_C_PREPROCESSOR)) {
1008 if (ch == '#') {
1009 unsigned int j = i + 1;
1010 while ((j < endPos) && IsASpaceOrTab(styler.SafeGetCharAt(j))) {
1011 j++;
1012 }
1013 if (styler.Match(j, "region") || styler.Match(j, "if")) {
1014 levelNext++;
1015 } else if (styler.Match(j, "end")) {
1016 levelNext--;
1017 }
1018 }
1019 }
1020 if (options.foldSyntaxBased && (style == SCE_C_OPERATOR)) {
1021 if (ch == '{') {
1022 // Measure the minimum before a '{' to allow
1023 // folding on "} else {"
1024 if (levelMinCurrent > levelNext) {
1025 levelMinCurrent = levelNext;
1026 }
1027 levelNext++;
1028 } else if (ch == '}') {
1029 levelNext--;
1030 }
1031 }
1032 if (!IsASpace(ch))
1033 visibleChars++;
1034 if (atEOL || (i == endPos-1)) {
1035 int levelUse = levelCurrent;
1036 if (options.foldSyntaxBased && options.foldAtElse) {
1037 levelUse = levelMinCurrent;
1038 }
1039 int lev = levelUse | levelNext << 16;
1040 if (visibleChars == 0 && options.foldCompact)
1041 lev |= SC_FOLDLEVELWHITEFLAG;
1042 if (levelUse < levelNext)
1043 lev |= SC_FOLDLEVELHEADERFLAG;
1044 if (lev != styler.LevelAt(lineCurrent)) {
1045 styler.SetLevel(lineCurrent, lev);
1046 }
1047 lineCurrent++;
1048 levelCurrent = levelNext;
1049 levelMinCurrent = levelCurrent;
1050 if (atEOL && (i == static_cast<unsigned int>(styler.Length()-1))) {
1051 // There is an empty line at end of file so give it same level and empty
1052 styler.SetLevel(lineCurrent, (levelCurrent | levelCurrent << 16) | SC_FOLDLEVELWHITEFLAG);
1053 }
1054 visibleChars = 0;
1055 }
1056 }
1057 }
1058
1059 void LexerCPP::EvaluateTokens(std::vector<std::string> &tokens) {
1060
1061 // Evaluate defined() statements to either 0 or 1
1062 for (size_t i=0; (i+2)<tokens.size();) {
1063 if ((tokens[i] == "defined") && (tokens[i+1] == "(")) {
1064 const char *val = "0";
1065 if (tokens[i+2] == ")") {
1066 // defined()
1067 tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 3);
1068 } else if (((i+3)<tokens.size()) && (tokens[i+3] == ")")) {
1069 // defined(<int>)
1070 tokens.erase(tokens.begin() + i + 1, tokens.begin() + i + 4);
1071 val = "1";
1072 }
1073 tokens[i] = val;
1074 } else {
1075 i++;
1076 }
1077 }
1078
1079 // Find bracketed subexpressions and recurse on them
1080 std::vector<std::string>::iterator itBracket = std::find(tokens.begin(), tokens.end(), "(");
1081 std::vector<std::string>::iterator itEndBracket = std::find(tokens.begin(), tokens.end(), ")");
1082 while ((itBracket != tokens.end()) && (itEndBracket != tokens.end()) && (itEndBracket > itBracket)) {
1083 std::vector<std::string> inBracket(itBracket + 1, itEndBracket);
1084 EvaluateTokens(inBracket);
1085
1086 // The insertion is done before the removal because there were failures with the opposite approach
1087 tokens.insert(itBracket, inBracket.begin(), inBracket.end());
1088 itBracket = std::find(tokens.begin(), tokens.end(), "(");
1089 itEndBracket = std::find(tokens.begin(), tokens.end(), ")");
1090 tokens.erase(itBracket, itEndBracket + 1);
1091
1092 itBracket = std::find(tokens.begin(), tokens.end(), "(");
1093 itEndBracket = std::find(tokens.begin(), tokens.end(), ")");
1094 }
1095
1096 // Evaluate logical negations
1097 for (size_t j=0; (j+1)<tokens.size();) {
1098 if (setNegationOp.Contains(tokens[j][0])) {
1099 int isTrue = atoi(tokens[j+1].c_str());
1100 if (tokens[j] == "!")
1101 isTrue = !isTrue;
1102 std::vector<std::string>::iterator itInsert =
1103 tokens.erase(tokens.begin() + j, tokens.begin() + j + 2);
1104 tokens.insert(itInsert, isTrue ? "1" : "0");
1105 } else {
1106 j++;
1107 }
1108 }
1109
1110 // Evaluate expressions in precedence order
1111 enum precedence { precArithmetic, precRelative, precLogical };
1112 for (int prec=precArithmetic; prec <= precLogical; prec++) {
1113 // Looking at 3 tokens at a time so end at 2 before end
1114 for (size_t k=0; (k+2)<tokens.size();) {
1115 char chOp = tokens[k+1][0];
1116 if (
1117 ((prec==precArithmetic) && setArithmethicOp.Contains(chOp)) ||
1118 ((prec==precRelative) && setRelOp.Contains(chOp)) ||
1119 ((prec==precLogical) && setLogicalOp.Contains(chOp))
1120 ) {
1121 int valA = atoi(tokens[k].c_str());
1122 int valB = atoi(tokens[k+2].c_str());
1123 int result = 0;
1124 if (tokens[k+1] == "+")
1125 result = valA + valB;
1126 else if (tokens[k+1] == "-")
1127 result = valA - valB;
1128 else if (tokens[k+1] == "*")
1129 result = valA * valB;
1130 else if (tokens[k+1] == "/")
1131 result = valA / (valB ? valB : 1);
1132 else if (tokens[k+1] == "%")
1133 result = valA % (valB ? valB : 1);
1134 else if (tokens[k+1] == "<")
1135 result = valA < valB;
1136 else if (tokens[k+1] == "<=")
1137 result = valA <= valB;
1138 else if (tokens[k+1] == ">")
1139 result = valA > valB;
1140 else if (tokens[k+1] == ">=")
1141 result = valA >= valB;
1142 else if (tokens[k+1] == "==")
1143 result = valA == valB;
1144 else if (tokens[k+1] == "!=")
1145 result = valA != valB;
1146 else if (tokens[k+1] == "||")
1147 result = valA || valB;
1148 else if (tokens[k+1] == "&&")
1149 result = valA && valB;
1150 char sResult[30];
1151 sprintf(sResult, "%d", result);
1152 std::vector<std::string>::iterator itInsert =
1153 tokens.erase(tokens.begin() + k, tokens.begin() + k + 3);
1154 tokens.insert(itInsert, sResult);
1155 } else {
1156 k++;
1157 }
1158 }
1159 }
1160 }
1161
1162 bool LexerCPP::EvaluateExpression(const std::string &expr, const std::map<std::string, std::string> &preprocessorDefinitions) {
1163 // Break into tokens, replacing with definitions
1164 std::string word;
1165 std::vector<std::string> tokens;
1166 const char *cp = expr.c_str();
1167 for (;;) {
1168 if (setWord.Contains(*cp)) {
1169 word += *cp;
1170 } else {
1171 std::map<std::string, std::string>::const_iterator it = preprocessorDefinitions.find(word);
1172 if (it != preprocessorDefinitions.end()) {
1173 tokens.push_back(it->second);
1174 } else if (!word.empty() && ((word[0] >= '0' && word[0] <= '9') || (word == "defined"))) {
1175 tokens.push_back(word);
1176 }
1177 word = "";
1178 if (!*cp) {
1179 break;
1180 }
1181 if ((*cp != ' ') && (*cp != '\t')) {
1182 std::string op(cp, 1);
1183 if (setRelOp.Contains(*cp)) {
1184 if (setRelOp.Contains(cp[1])) {
1185 op += cp[1];
1186 cp++;
1187 }
1188 } else if (setLogicalOp.Contains(*cp)) {
1189 if (setLogicalOp.Contains(cp[1])) {
1190 op += cp[1];
1191 cp++;
1192 }
1193 }
1194 tokens.push_back(op);
1195 }
1196 }
1197 cp++;
1198 }
1199
1200 EvaluateTokens(tokens);
1201
1202 // "0" or "" -> false else true
1203 bool isFalse = tokens.empty() ||
1204 ((tokens.size() == 1) && ((tokens[0] == "") || tokens[0] == "0"));
1205 return !isFalse;
1206 }
1207
1208 LexerModule lmCPP(SCLEX_CPP, LexerCPP::LexerFactoryCPP, "cpp", cppWordLists);
1209 LexerModule lmCPPNoCase(SCLEX_CPPNOCASE, LexerCPP::LexerFactoryCPPInsensitive, "cppnocase", cppWordLists);