feat(lexer): match regular words
This commit is contained in:
parent
dcc52e2850
commit
ec076a56a5
6 changed files with 82 additions and 5 deletions
|
|
@ -55,6 +55,11 @@ void mrk_lexer_advance_eq(mrk_lexer *lexer, char c);
|
|||
*/
|
||||
char mrk_lexer_advance_n(mrk_lexer *lexer, size_t n);
|
||||
|
||||
/**
|
||||
* Advance the lexer as long as the next character is part of a regular word.
|
||||
*/
|
||||
void mrk_lexer_advance_word(mrk_lexer *lexer);
|
||||
|
||||
/**
|
||||
* Reset the lexer's current token context.
|
||||
*/
|
||||
|
|
@ -66,4 +71,9 @@ void mrk_lexer_reset(mrk_lexer *lexer);
|
|||
*/
|
||||
void mrk_lexer_emit(mrk_token *out, mrk_lexer *lexer, mrk_token_type type);
|
||||
|
||||
/**
|
||||
* Returns whether the given character can be part of a word, and is thus not a
|
||||
* special character.
|
||||
*/
|
||||
bool mrk_is_special_char(char c);
|
||||
#endif
|
||||
|
|
|
|||
|
|
@ -78,6 +78,12 @@ char mrk_lexer_peek(mrk_lexer *lexer) {
|
|||
return lexer->buf.s[lexer->pos.buf_index];
|
||||
}
|
||||
|
||||
void mrk_lexer_advance_word(mrk_lexer *lexer) {
|
||||
while (!mrk_is_special_char(mrk_lexer_peek(lexer))) {
|
||||
mrk_lexer_advance(lexer);
|
||||
}
|
||||
}
|
||||
|
||||
char mrk_lexer_peek_n(mrk_lexer *lexer, size_t n) {
|
||||
// Check whether the lexer would be done in n steps
|
||||
bool done_in_n = false;
|
||||
|
|
@ -201,9 +207,34 @@ mrk_lexer_err mrk_lexer_next(mrk_token *out, mrk_lexer *lexer) {
|
|||
mrk_lexer_emit(out, lexer, mrk_token_type_spaces);
|
||||
}
|
||||
} break;
|
||||
default: {
|
||||
// Match ordered list headers
|
||||
if (isdigit(c)) {
|
||||
mrk_lexer_advance(lexer);
|
||||
|
||||
while (isdigit(mrk_lexer_peek(lexer))) {
|
||||
mrk_lexer_advance(lexer);
|
||||
}
|
||||
|
||||
if (mrk_lexer_peek(lexer) == '.') {
|
||||
mrk_lexer_advance(lexer);
|
||||
mrk_lexer_emit(out, lexer, mrk_token_type_dotted_number);
|
||||
}
|
||||
// Doesn't end with a dot, so it's just a word that happens to start
|
||||
// with a number
|
||||
else {
|
||||
mrk_lexer_advance_word(lexer);
|
||||
mrk_lexer_emit(out, lexer, mrk_token_type_word);
|
||||
}
|
||||
}
|
||||
// Any other special scenarios we simply parse as a word
|
||||
else {
|
||||
mrk_lexer_advance_word(lexer);
|
||||
mrk_lexer_emit(out, lexer, mrk_token_type_word);
|
||||
}
|
||||
} break;
|
||||
}
|
||||
}
|
||||
|
||||
return lexer->token.emitted ? mrk_lexer_err_ok : mrk_lexer_err_done;
|
||||
;
|
||||
}
|
||||
|
|
|
|||
15
src/lexer/util.c
Normal file
15
src/lexer/util.c
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
#include "mrk/lexer_internal.h"
|
||||
|
||||
const char special_chars[] = {'#', '`', '-', '_', '*', '=', '\t', '>',
|
||||
'!', '[', ']', '(', ')', '\\', '\n', ' '};
|
||||
const size_t special_chars_len = sizeof(special_chars);
|
||||
|
||||
bool mrk_is_special_char(char c) {
|
||||
bool is_special = false;
|
||||
|
||||
for (size_t i = 0; i < special_chars_len && !is_special; i++) {
|
||||
is_special = c == special_chars[i];
|
||||
}
|
||||
|
||||
return is_special;
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue