%%---------------------------------------------------------------------- %% %% %CopyrightBegin% %% %% Copyright Ericsson AB 1999-2016. All Rights Reserved. %% %% Licensed under the Apache License, Version 2.0 (the "License"); %% you may not use this file except in compliance with the License. %% You may obtain a copy of the License at %% %% http://www.apache.org/licenses/LICENSE-2.0 %% %% Unless required by applicable law or agreed to in writing, software %% distributed under the License is distributed on an "AS IS" BASIS, %% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. %% See the License for the specific language governing permissions and %% limitations under the License. %% %% %CopyrightEnd% %% %% %%---------------------------------------------------------------------- %% File : cosNotification_Scanner.erl %% Purpose : Scan and pre-process a grammar. %%---------------------------------------------------------------------- -module('cosNotification_Scanner'). -export([scan/1]). scan(Str) -> RSL = scan(Str, 1, [], any), {ok, lists:reverse(RSL)}. %% Guard macros used at top scan functions only -define(is_number(X), X >= $0, X =< $9). -define(is_upper(X), X >= $A, X =< $Z). -define(is_lower(X), X >= $a, X =< $z). %%---------------------------------------------------------------------- %% scan %% %% A-Z scan([X|Str], Line, Out, Type) when ?is_upper(X) -> scan_name(Str, [X], Line, Out, Type); %% a-z scan([X|Str], Line, Out, Type) when ?is_lower(X) -> scan_name(Str, [X], Line, Out, Type); %% 0-9 scan([X|Str], Line, Out, Type) when ?is_number(X) -> scan_number(Str, [X], Line, Out, Type); %% RELOP:s == != <= >= > < scan([$=,$= | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '=='} | Out], any); scan([$!,$= | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '!='} | Out], any); scan([$<,$= | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '<='} | Out], any); scan([$>,$= | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '>='} | Out], any); scan([$> | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '>'} | Out], any); scan([$< | Str], Line, Out, _Type) -> scan(Str, Line, [{'RELOP', '<'} | Out], any); %% ADDOP:s + - scan([$+ | Str], Line, Out, Type) -> scan(Str, Line, [{'ADDOP', '+'} | Out], Type); scan([$- | Str], Line, Out, Type) -> scan(Str, Line, [{'ADDOP', '-'} | Out], Type); %% MULOP:s * / scan([$* | Str], Line, Out, _Type) -> scan(Str, Line, [{'MULOP', '*'} | Out], any); scan([$/ | Str], Line, Out, _Type) -> scan(Str, Line, [{'MULOP', '/'} | Out], any); %% TAB scan([9| T], Line, Out, Type) -> scan(T, Line, Out, Type); %% SP scan([32| T], Line, Out, Type) -> scan(T, Line, Out, Type); %% CR scan([$\r|Str], Line, Out, Type) -> scan(Str, Line, Out, Type); %% LF scan([$\n|Str], Line, Out, Type) -> scan(Str, Line+1, Out, Type); %% \\ scan([92, 92 | Str], Line, Out, Type) -> scan(Str, Line, [{'dbslsh', Line} | Out], Type); %% \' scan([92, 39 | Str], Line, Out, Type) -> scan(Str, Line, [{'bslshd', Line} | Out], Type); %% '\' scan([92 | Str], Line, Out, Type) -> scan(Str, Line, [{'bslsh', Line} | Out], Type); %% '_' scan([$_ | Str], Line, Out, dollar) -> scan_name(Str, [$_], Line, Out, dollar); %% '$' scan([$$, 92 | Str], Line, Out, _Type) -> scan(Str, Line, [{'bslsh', Line}, {'dollar', Line} | Out], dollar); scan([$$ | Str], Line, Out, _Type) -> scan(Str, Line, [{'dollar', Line} | Out], dollar); scan([$"|Str], Line, Out, Type) -> scan_const(char, Str, [], Line, Out, Type); scan([$'|Str], Line, Out, Type) -> scan_const(string, Str, [], Line, Out, Type); %% Writing '+.' is not allowed ('+' or '-' are only allowed %% as unary for (within a component) which must be en integer). scan([$. | Str], Line, [{'ADDOP', Op}|Out], _) -> scan_frac(Str, [$.], Line, [{'ADDOP', Op}|Out], any); %% Must be a scan([$. | Str], Line, Out, dollar) -> scan(Str, Line, [{'.',Line} | Out], dollar); %% Number scan([$. | Str], Line, Out, Type) -> scan_frac(Str, [$.], Line, Out, Type); scan([C|Str], Line, Out, Type) -> scan(Str, Line, [{list_to_atom([C]), Line} | Out], Type); scan([], _Line, Out, _Type) -> Out. %%---------------------------------------------------------------------- %% scan_name %% scan_number([X|Str], Accum, Line, Out, Type) when ?is_number(X) -> scan_number(Str, [X|Accum], Line, Out, Type); scan_number([X|Str], Accum, Line, Out, dollar) when X==$. -> scan(Str, Line, [{'.', Line}, {'int', list_to_integer(lists:reverse(Accum))} | Out], dollar); scan_number([X|Str], Accum, Line, Out, Type) when X==$. -> scan_frac(Str, [X|Accum], Line, Out, Type); scan_number([X|Str], Accum, Line, Out, Type) when X==$e -> scan_exp(Str, [X|Accum], Line, Out, Type); scan_number([X|Str], Accum, Line, Out, Type) when X==$E -> scan_exp(Str, [X|Accum], Line, Out, Type); scan_number(Str, Accum, Line, Out, Type) -> scan(Str, Line, [{'int', list_to_integer(lists:reverse(Accum))} | Out], Type). %% Floating point number scan. %% %% Non trivial scan. A float consists of an integral part, a %% decimal point, a fraction part, an e or E and a signed integer %% exponent. Either the integer part or the fraction part but not %% both may be missing, and either the decimal point or the %% exponent part but not both may be missing. The exponent part %% must consist of an e or E and a possibly signed exponent. %% %% Analysis shows that "1." ".7" "1e2" ".5e-3" "1.7e2" "1.7e-2" %% is allowed and "1" ".e9" is not. The sign is only allowed just %% after an e or E. The scanner reads a number as an integer %% until it encounters a "." so the integer part only error case %% will not be caught in the scanner (but rather in expression %% evaluation) scan_frac([$e | _Str], [$.], _Line, _Out, _Type) -> {error, "illegal_float"}; scan_frac([$E | _Str], [$.], _Line, _Out, _Type) -> {error, "illegal_float"}; scan_frac(Str, Accum, Line, Out, Type) -> scan_frac2(Str, Accum, Line, Out, Type). scan_frac2([X|Str], Accum, Line, Out, Type) when ?is_number(X) -> scan_frac2(Str, [X|Accum], Line, Out, Type); scan_frac2([X|Str], Accum, Line, Out, Type) when X==$e -> scan_exp(Str, [X|Accum], Line, Out, Type); scan_frac2([X|Str], Accum, Line, Out, Type) when X==$E -> scan_exp(Str, [X|Accum], Line, Out, Type); %% Since '.2' is allowed, we add '0' in front to be sure (erlang do not allow %% list_to_float(".2") and list_to_float("0.2") eq. list_to_float("00.2")). scan_frac2(Str, Accum, Line, Out, Type) -> scan(Str, Line, [{'num', list_to_float([$0|lists:reverse(Accum)])} | Out], Type). scan_exp([X|Str], Accum, Line, Out, Type) when X==$- -> scan_exp2(Str, [X|Accum], Line, Out, Type); scan_exp(Str, Accum, Line, Out, Type) -> scan_exp2(Str, Accum, Line, Out, Type). scan_exp2([X|Str], Accum, Line, Out, Type) when ?is_number(X) -> scan_exp2(Str, [X|Accum], Line, Out, Type); %% Since '.2' is allowed, we add '0' in front to be sure (erlang do not allow %% list_to_float(".2")). scan_exp2(Str, Accum, Line, Out, Type) -> scan(Str, Line, [{'num', list_to_float([$0|lists:reverse(Accum)])} | Out], Type). scan_name([X|Str], Accum, Line, Out, Type) when ?is_upper(X) -> scan_name(Str, [X|Accum], Line, Out, Type); scan_name([X|Str], Accum, Line, Out, Type) when ?is_lower(X) -> scan_name(Str, [X|Accum], Line, Out, Type); scan_name([X|Str], Accum, Line, Out, Type) when ?is_number(X) -> scan_name(Str, [X|Accum], Line, Out, Type); scan_name([$_|Str], Accum, Line, Out, dollar) -> scan_name(Str, [$_|Accum], Line, Out, dollar); scan_name(S, Accum, Line, [{bslsh,LL} | Out], Type) -> %% An escaped identifier. L = lists:reverse(Accum), scan(S, Line, [{'ident', L}, {bslsh,LL} | Out], Type); scan_name(S, Accum, Line, Out, Type) -> L = lists:reverse(Accum), {X, NewType} = case check_name(L) of false -> {{'ident', L}, Type}; _ -> {{list_to_atom(L), Line}, any} end, scan(S, Line, [X | Out], NewType). %% Shall scan a constant scan_const(char, [$" | Rest], Accum, Line, Out, Type) -> scan(Rest, Line, [{'ident', list_to_atom(lists:reverse(Accum))} | Out], Type); scan_const(char, [], _Accum, _Line, Out, _Type) -> %% Bad string % {error, "bad_string"}; Out; scan_const(string, [$' | Rest], Accum, Line, Out, Type) -> scan(Rest, Line, [{'string', lists:reverse(Accum)} | Out], Type); scan_const(Mode, [$\\, C | Rest], Accum, Line, Out, Type) -> case escaped_char(C) of error -> %% Bad escape character %% {error, "bad_escape_character"}; scan_const(Mode, Rest, [C | Accum], Line, Out, Type); EC -> scan_const(Mode, Rest, [EC | Accum], Line, Out, Type) end; scan_const(Mode, [C | Rest], Accum, Line, Out, Type) -> scan_const(Mode, Rest, [C | Accum], Line, Out, Type). %% Escaped character. Escaped chars are repr as two characters in the %% input list of letters and this is translated into one char. escaped_char($n) -> $\n; escaped_char($t) -> $\t; escaped_char($v) -> $\v; escaped_char($b) -> $\b; escaped_char($r) -> $ ; escaped_char($f) -> $\f; escaped_char($a) -> $\a; escaped_char($\\) -> $\\; escaped_char($?) -> $?; escaped_char($') -> $'; escaped_char($") -> $"; %% Error escaped_char(_Other) -> error. check_name("exist") -> true; check_name("default") -> true; check_name("_length") -> true; check_name("_d") -> true; check_name("_type_id") -> true; check_name("_repos_id") -> true; check_name("not") -> true; check_name("or") -> true; check_name("and") -> true; check_name("FALSE") -> true; check_name("TRUE") -> true; check_name("in") -> true; check_name(_) -> false.