]>
git.saurik.com Git - wxWidgets.git/blob - src/stc/scintilla/src/LexVHDL.cxx
1 // Scintilla source code edit control
4 ** Written by Phil Reid,
6 ** - The Verilog Lexer by Avi Yegudin
7 ** - The Fortran Lexer by Chuan-jian Shen
8 ** - The C++ lexer by Neil Hodgson
10 // Copyright 1998-2002 by Neil Hodgson <neilh@scintilla.org>
11 // The License.txt file describes the conditions under which this software may be distributed.
23 #include "StyleContext.h"
25 #include "Scintilla.h"
29 using namespace Scintilla
;
32 static void ColouriseVHDLDoc(
33 unsigned int startPos
,
36 WordList
*keywordlists
[],
40 /***************************************/
41 static inline bool IsAWordChar(const int ch
) {
42 return (ch
< 0x80) && (isalnum(ch
) || ch
== '.' || ch
== '_' );
45 /***************************************/
46 static inline bool IsAWordStart(const int ch
) {
47 return (ch
< 0x80) && (isalnum(ch
) || ch
== '_');
50 /***************************************/
51 inline bool IsABlank(unsigned int ch
) {
52 return (ch
== ' ') || (ch
== 0x09) || (ch
== 0x0b) ;
55 /***************************************/
56 static void ColouriseVHDLDoc(
57 unsigned int startPos
,
60 WordList
*keywordlists
[],
63 WordList
&Keywords
= *keywordlists
[0];
64 WordList
&Operators
= *keywordlists
[1];
65 WordList
&Attributes
= *keywordlists
[2];
66 WordList
&Functions
= *keywordlists
[3];
67 WordList
&Packages
= *keywordlists
[4];
68 WordList
&Types
= *keywordlists
[5];
69 WordList
&User
= *keywordlists
[6];
71 StyleContext
sc(startPos
, length
, initStyle
, styler
);
73 for (; sc
.More(); sc
.Forward())
76 // Determine if the current state should terminate.
77 if (sc
.state
== SCE_VHDL_OPERATOR
) {
78 sc
.SetState(SCE_VHDL_DEFAULT
);
79 } else if (sc
.state
== SCE_VHDL_NUMBER
) {
80 if (!IsAWordChar(sc
.ch
) && (sc
.ch
!= '#')) {
81 sc
.SetState(SCE_VHDL_DEFAULT
);
83 } else if (sc
.state
== SCE_VHDL_IDENTIFIER
) {
84 if (!IsAWordChar(sc
.ch
) || (sc
.ch
== '.')) {
86 sc
.GetCurrentLowered(s
, sizeof(s
));
87 if (Keywords
.InList(s
)) {
88 sc
.ChangeState(SCE_VHDL_KEYWORD
);
89 } else if (Operators
.InList(s
)) {
90 sc
.ChangeState(SCE_VHDL_STDOPERATOR
);
91 } else if (Attributes
.InList(s
)) {
92 sc
.ChangeState(SCE_VHDL_ATTRIBUTE
);
93 } else if (Functions
.InList(s
)) {
94 sc
.ChangeState(SCE_VHDL_STDFUNCTION
);
95 } else if (Packages
.InList(s
)) {
96 sc
.ChangeState(SCE_VHDL_STDPACKAGE
);
97 } else if (Types
.InList(s
)) {
98 sc
.ChangeState(SCE_VHDL_STDTYPE
);
99 } else if (User
.InList(s
)) {
100 sc
.ChangeState(SCE_VHDL_USERWORD
);
102 sc
.SetState(SCE_VHDL_DEFAULT
);
104 } else if (sc
.state
== SCE_VHDL_COMMENT
|| sc
.state
== SCE_V_COMMENTLINEBANG
) {
106 sc
.SetState(SCE_VHDL_DEFAULT
);
108 } else if (sc
.state
== SCE_VHDL_STRING
) {
110 if (sc
.chNext
== '\"' || sc
.chNext
== '\'' || sc
.chNext
== '\\') {
113 } else if (sc
.ch
== '\"') {
114 sc
.ForwardSetState(SCE_VHDL_DEFAULT
);
115 } else if (sc
.atLineEnd
) {
116 sc
.ChangeState(SCE_V_STRINGEOL
);
117 sc
.ForwardSetState(SCE_VHDL_DEFAULT
);
121 // Determine if a new state should be entered.
122 if (sc
.state
== SCE_VHDL_DEFAULT
) {
123 if (IsADigit(sc
.ch
) || (sc
.ch
== '.' && IsADigit(sc
.chNext
))) {
124 sc
.SetState(SCE_VHDL_NUMBER
);
125 } else if (IsAWordStart(sc
.ch
)) {
126 sc
.SetState(SCE_VHDL_IDENTIFIER
);
127 } else if (sc
.Match('-', '-')) {
128 sc
.SetState(SCE_VHDL_COMMENT
);
130 } else if (sc
.Match('-', '-')) {
131 if (sc
.Match("--!")) // Nice to have a different comment style
132 sc
.SetState(SCE_VHDL_COMMENTLINEBANG
);
134 sc
.SetState(SCE_VHDL_COMMENT
);
135 } else if (sc
.ch
== '\"') {
136 sc
.SetState(SCE_VHDL_STRING
);
137 } else if (isoperator(static_cast<char>(sc
.ch
))) {
138 sc
.SetState(SCE_VHDL_OPERATOR
);
144 //=============================================================================
145 static bool IsCommentLine(int line
, Accessor
&styler
) {
146 int pos
= styler
.LineStart(line
);
147 int eol_pos
= styler
.LineStart(line
+ 1) - 1;
148 for (int i
= pos
; i
< eol_pos
; i
++) {
150 char chNext
= styler
[i
+1];
151 if ((ch
== '-') && (chNext
== '-'))
153 else if (ch
!= ' ' && ch
!= '\t')
159 //=============================================================================
161 static void FoldNoBoxVHDLDoc(
162 unsigned int startPos
,
167 // Decided it would be smarter to have the lexer have all keywords included. Therefore I
168 // don't check if the style for the keywords that I use to adjust the levels.
170 "architecture begin case component else elsif end entity generate loop package process record then "
171 "procedure function when";
175 bool foldComment
= styler
.GetPropertyInt("fold.comment", 1) != 0;
176 bool foldCompact
= styler
.GetPropertyInt("fold.compact", 1) != 0;
177 bool foldAtElse
= styler
.GetPropertyInt("fold.at.else", 1) != 0;
178 bool foldAtBegin
= styler
.GetPropertyInt("fold.at.Begin", 1) != 0;
179 bool foldAtParenthese
= styler
.GetPropertyInt("fold.at.Parenthese", 1) != 0;
180 //bool foldAtWhen = styler.GetPropertyInt("fold.at.When", 1) != 0; //< fold at when in case statements
182 int visibleChars
= 0;
183 unsigned int endPos
= startPos
+ length
;
185 int lineCurrent
= styler
.GetLine(startPos
);
186 int levelCurrent
= SC_FOLDLEVELBASE
;
188 levelCurrent
= styler
.LevelAt(lineCurrent
-1) >> 16;
189 //int levelMinCurrent = levelCurrent;
190 int levelMinCurrentElse
= levelCurrent
; //< Used for folding at 'else'
191 int levelMinCurrentBegin
= levelCurrent
; //< Used for folding at 'begin'
192 int levelNext
= levelCurrent
;
194 /***************************************/
196 char prevWord
[32] = "";
198 /***************************************/
200 // The logic for going up or down a level depends on a the previous keyword
201 // This code could be cleaned up.
204 for(j
= startPos
; j
>0; j
--)
206 char ch
= styler
.SafeGetCharAt(j
);
207 char chPrev
= styler
.SafeGetCharAt(j
-1);
208 int style
= styler
.StyleAt(j
);
209 int stylePrev
= styler
.StyleAt(j
-1);
210 if ((stylePrev
!= SCE_VHDL_COMMENT
) && (stylePrev
!= SCE_VHDL_STRING
))
212 if(IsAWordChar(chPrev
) && !IsAWordChar(ch
))
217 if ((style
!= SCE_VHDL_COMMENT
) && (style
!= SCE_VHDL_STRING
))
219 if(!IsAWordChar(chPrev
) && IsAWordStart(ch
) && (end
!= 0))
223 for(k
=0; (k
<31 ) && (k
<end
-j
+1 ); k
++) {
224 s
[k
] = static_cast<char>(tolower(styler
[j
+k
]));
228 if(keywords
.InList(s
)) {
235 for(j
=j
+strlen(prevWord
); j
<endPos
; j
++)
237 char ch
= styler
.SafeGetCharAt(j
);
238 int style
= styler
.StyleAt(j
);
239 if ((style
!= SCE_VHDL_COMMENT
) && (style
!= SCE_VHDL_STRING
))
241 if((ch
== ';') && (strcmp(prevWord
, "end") == 0))
243 strcpy(prevWord
, ";");
248 char chNext
= styler
[startPos
];
251 int styleNext
= styler
.StyleAt(startPos
);
252 int style
= initStyle
;
253 //Platform::DebugPrintf("Line[%04d] Prev[%20s] ************************* Level[%x]\n", lineCurrent+1, prevWord, levelCurrent);
255 /***************************************/
256 for (unsigned int i
= startPos
; i
< endPos
; i
++)
259 chNext
= styler
.SafeGetCharAt(i
+ 1);
260 chPrev
= styler
.SafeGetCharAt(i
- 1);
261 chNextNonBlank
= chNext
;
262 unsigned int j
= i
+1;
263 while(IsABlank(chNextNonBlank
) && j
<endPos
)
266 chNextNonBlank
= styler
.SafeGetCharAt(j
);
269 styleNext
= styler
.StyleAt(i
+ 1);
270 bool atEOL
= (ch
== '\r' && chNext
!= '\n') || (ch
== '\n');
272 if (foldComment
&& atEOL
&& IsCommentLine(lineCurrent
, styler
))
274 if(!IsCommentLine(lineCurrent
-1, styler
) && IsCommentLine(lineCurrent
+1, styler
))
278 else if(IsCommentLine(lineCurrent
-1, styler
) && !IsCommentLine(lineCurrent
+1, styler
))
284 if ((style
== SCE_VHDL_OPERATOR
) && foldAtParenthese
)
288 } else if (ch
== ')') {
293 if ((style
!= SCE_VHDL_COMMENT
) && (style
!= SCE_VHDL_STRING
))
295 if((ch
== ';') && (strcmp(prevWord
, "end") == 0))
297 strcpy(prevWord
, ";");
300 if(!IsAWordChar(chPrev
) && IsAWordStart(ch
))
305 if(iswordchar(ch
) && !iswordchar(chNext
)) {
308 for(k
=0; (k
<31 ) && (k
<i
-lastStart
+1 ); k
++) {
309 s
[k
] = static_cast<char>(tolower(styler
[lastStart
+k
]));
313 if(keywords
.InList(s
))
316 strcmp(s
, "architecture") == 0 ||
317 strcmp(s
, "case") == 0 ||
318 strcmp(s
, "component") == 0 ||
319 strcmp(s
, "entity") == 0 ||
320 strcmp(s
, "generate") == 0 ||
321 strcmp(s
, "loop") == 0 ||
322 strcmp(s
, "package") ==0 ||
323 strcmp(s
, "process") == 0 ||
324 strcmp(s
, "record") == 0 ||
325 strcmp(s
, "then") == 0)
327 if (strcmp(prevWord
, "end") != 0)
329 if (levelMinCurrentElse
> levelNext
) {
330 levelMinCurrentElse
= levelNext
;
335 strcmp(s
, "procedure") == 0 ||
336 strcmp(s
, "function") == 0)
338 if (strcmp(prevWord
, "end") != 0) // check for "end procedure" etc.
339 { // This code checks to see if the procedure / function is a definition within a "package"
340 // rather than the actual code in the body.
341 int BracketLevel
= 0;
342 for(int j
=i
+1; j
<styler
.Length(); j
++)
344 int LocalStyle
= styler
.StyleAt(j
);
345 char LocalCh
= styler
.SafeGetCharAt(j
);
346 if(LocalCh
== '(') BracketLevel
++;
347 if(LocalCh
== ')') BracketLevel
--;
349 (BracketLevel
== 0) &&
350 (LocalStyle
!= SCE_VHDL_COMMENT
) &&
351 (LocalStyle
!= SCE_VHDL_STRING
) &&
352 !iswordchar(styler
.SafeGetCharAt(j
-1)) &&
353 styler
.Match(j
, "is") &&
354 !iswordchar(styler
.SafeGetCharAt(j
+2)))
356 if (levelMinCurrentElse
> levelNext
) {
357 levelMinCurrentElse
= levelNext
;
362 if((BracketLevel
== 0) && (LocalCh
== ';'))
369 } else if (strcmp(s
, "end") == 0) {
371 } else if(strcmp(s
, "elsif") == 0) { // elsif is followed by then so folding occurs correctly
373 } else if (strcmp(s
, "else") == 0) {
374 if(strcmp(prevWord
, "when") != 0) // ignore a <= x when y else z;
376 levelMinCurrentElse
= levelNext
- 1; // VHDL else is all on its own so just dec. the min level
379 ((strcmp(s
, "begin") == 0) && (strcmp(prevWord
, "architecture") == 0)) ||
380 ((strcmp(s
, "begin") == 0) && (strcmp(prevWord
, "function") == 0)) ||
381 ((strcmp(s
, "begin") == 0) && (strcmp(prevWord
, "procedure") == 0)))
383 levelMinCurrentBegin
= levelNext
- 1;
385 //Platform::DebugPrintf("Line[%04d] Prev[%20s] Cur[%20s] Level[%x]\n", lineCurrent+1, prevWord, s, levelCurrent);
391 int levelUse
= levelCurrent
;
393 if (foldAtElse
&& (levelMinCurrentElse
< levelUse
)) {
394 levelUse
= levelMinCurrentElse
;
396 if (foldAtBegin
&& (levelMinCurrentBegin
< levelUse
)) {
397 levelUse
= levelMinCurrentBegin
;
399 int lev
= levelUse
| levelNext
<< 16;
400 if (visibleChars
== 0 && foldCompact
)
401 lev
|= SC_FOLDLEVELWHITEFLAG
;
403 if (levelUse
< levelNext
)
404 lev
|= SC_FOLDLEVELHEADERFLAG
;
405 if (lev
!= styler
.LevelAt(lineCurrent
)) {
406 styler
.SetLevel(lineCurrent
, lev
);
408 //Platform::DebugPrintf("Line[%04d] ---------------------------------------------------- Level[%x]\n", lineCurrent+1, levelCurrent);
410 levelCurrent
= levelNext
;
411 //levelMinCurrent = levelCurrent;
412 levelMinCurrentElse
= levelCurrent
;
413 levelMinCurrentBegin
= levelCurrent
;
416 /***************************************/
417 if (!isspacechar(ch
)) visibleChars
++;
420 /***************************************/
421 // Platform::DebugPrintf("Line[%04d] ---------------------------------------------------- Level[%x]\n", lineCurrent+1, levelCurrent);
424 //=============================================================================
425 static void FoldVHDLDoc(unsigned int startPos
, int length
, int initStyle
, WordList
*[],
427 FoldNoBoxVHDLDoc(startPos
, length
, initStyle
, styler
);
430 //=============================================================================
431 static const char * const VHDLWordLists
[] = {
435 "Standard Functions",
443 LexerModule
lmVHDL(SCLEX_VHDL
, ColouriseVHDLDoc
, "vhdl", FoldVHDLDoc
, VHDLWordLists
);
447 // access after alias all architecture array assert attribute begin block body buffer bus case component
448 // configuration constant disconnect downto else elsif end entity exit file for function generate generic
449 // group guarded if impure in inertial inout is label library linkage literal loop map new next null of
450 // on open others out package port postponed procedure process pure range record register reject report
451 // return select severity shared signal subtype then to transport type unaffected units until use variable
452 // wait when while with
455 // abs and mod nand nor not or rem rol ror sla sll sra srl xnor xor
458 // left right low high ascending image value pos val succ pred leftof rightof base range reverse_range
459 // length delayed stable quiet transaction event active last_event last_active last_value driving
460 // driving_value simple_name path_name instance_name
463 // now readline read writeline write endfile resolved to_bit to_bitvector to_stdulogic to_stdlogicvector
464 // to_stdulogicvector to_x01 to_x01z to_UX01 rising_edge falling_edge is_x shift_left shift_right rotate_left
465 // rotate_right resize to_integer to_unsigned to_signed std_match to_01
468 // std ieee work standard textio std_logic_1164 std_logic_arith std_logic_misc std_logic_signed
469 // std_logic_textio std_logic_unsigned numeric_bit numeric_std math_complex math_real vital_primitives
473 // boolean bit character severity_level integer real time delay_length natural positive string bit_vector
474 // file_open_kind file_open_status line text side width std_ulogic std_ulogic_vector std_logic
475 // std_logic_vector X01 X01Z UX01 UX01Z unsigned signed