ruby.rl's string and regex literals shouldn't call @code right after '%'.
[ohcount] / ext / ohcount_native / ragel_parsers / sql.rl
1 // sql.rl written by Mitchell Foral. mitchell<att>caladbolg<dott>net.
2
3 /************************* Required for every parser *************************/
4 #ifndef RAGEL_SQL_PARSER
5 #define RAGEL_SQL_PARSER
6
7 #include "ragel_parser_macros.h"
8
9 // the name of the language
10 const char *SQL_LANG = "sql";
11
12 // the languages entities
13 const char *sql_entities[] = {
14   "space", "comment", "string", "any",
15 };
16
17 // constants associated with the entities
18 enum {
19   SQL_SPACE = 0, SQL_COMMENT, SQL_STRING, SQL_ANY
20 };
21
22 /*****************************************************************************/
23
24 %%{
25   machine sql;
26   write data;
27   include common "common.rl";
28
29   # Line counting machine
30
31   action sql_ccallback {
32     switch(entity) {
33     case SQL_SPACE:
34       ls
35       break;
36     case SQL_ANY:
37       code
38       break;
39     case INTERNAL_NL:
40       std_internal_newline(SQL_LANG)
41       break;
42     case NEWLINE:
43       std_newline(SQL_LANG)
44     }
45   }
46
47   sql_line_comment = ('--' | '#' | '//') @comment nonnewline*;
48   sql_c_block_comment =
49     '/*' @comment (
50       newline %{ entity = INTERNAL_NL; } %sql_ccallback
51       |
52       ws
53       |
54       (nonnewline - ws) @comment
55     )* :>> '*/';
56   sql_block_comment =
57     '{' @comment (
58       newline %{ entity = INTERNAL_NL; } %sql_ccallback
59       |
60       ws
61       |
62       (nonnewline - ws) @comment
63     )* :>> '}';
64   sql_comment = sql_line_comment | sql_c_block_comment | sql_block_comment;
65
66   sql_sq_str = '\'' @code ([^\r\n\f'\\] | '\\' nonnewline)* '\'';
67   sql_dq_str = '"' @code ([^\r\n\f"\\] | '\\' nonnewline)* '"';
68   sql_string = sql_sq_str | sql_dq_str;
69
70   sql_line := |*
71     spaces       ${ entity = SQL_SPACE; } => sql_ccallback;
72     sql_comment;
73     sql_string;
74     newline      ${ entity = NEWLINE;   } => sql_ccallback;
75     ^space       ${ entity = SQL_ANY;   } => sql_ccallback;
76   *|;
77
78   # Entity machine
79
80   action sql_ecallback {
81     callback(SQL_LANG, sql_entities[entity], cint(ts), cint(te));
82   }
83
84   sql_entity := 'TODO:';
85 }%%
86
87 /************************* Required for every parser *************************/
88
89 /* Parses a string buffer with SQL code.
90  *
91  * @param *buffer The string to parse.
92  * @param length The length of the string to parse.
93  * @param count Integer flag specifying whether or not to count lines. If yes,
94  *   uses the Ragel machine optimized for counting. Otherwise uses the Ragel
95  *   machine optimized for returning entity positions.
96  * @param *callback Callback function. If count is set, callback is called for
97  *   every line of code, comment, or blank with 'lcode', 'lcomment', and
98  *   'lblank' respectively. Otherwise callback is called for each entity found.
99  */
100 void parse_sql(char *buffer, int length, int count,
101   void (*callback) (const char *lang, const char *entity, int start, int end)
102   ) {
103   init
104
105   %% write init;
106   cs = (count) ? sql_en_sql_line : sql_en_sql_entity;
107   %% write exec;
108
109   // if no newline at EOF; callback contents of last line
110   if (count) { process_last_line(SQL_LANG) }
111 }
112
113 #endif
114
115 /*****************************************************************************/