diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 8b92125..e97574f 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -10,8 +10,8 @@ jobs: strategy: matrix: - otp: ['24', '25', '26', '27'] - rebar: ['3.23'] + otp: ['25', '26', '27'] + rebar: ['3.24'] steps: - uses: actions/checkout@v3 diff --git a/README.md b/README.md index 78eb29c..9f01d79 100644 --- a/README.md +++ b/README.md @@ -1,7 +1,5 @@ # katana-code -![build](https://github.com/inaka/katana-code/workflows/build/badge.svg) - Katana Code is an Erlang library application containinig modules useful for processing Erlang code. # Contact Us @@ -12,4 +10,3 @@ If you find any **bugs** or have a **problem** while using this library, please And you can check all of our open-source projects at [inaka.github.io](http://inaka.github.io) - diff --git a/rebar.config b/rebar.config index 867fe26..3fda2b5 100644 --- a/rebar.config +++ b/rebar.config @@ -3,7 +3,7 @@ {erl_opts, [warn_unused_import, warn_export_vars, warnings_as_errors, verbose, report, debug_info]}. -{minimum_otp_vsn, "24"}. +{minimum_otp_vsn, "25"}. {profiles, [{test, [{cover_enabled, true}, {cover_opts, [verbose]}, {ct_opts, [{verbose, true}]}]}]}. diff --git a/src/ktn_dodger.erl b/src/ktn_dodger.erl index f4e69e2..c0cdadb 100644 --- a/src/ktn_dodger.erl +++ b/src/ktn_dodger.erl @@ -21,6 +21,7 @@ %% %% @copyright 2001-2006 Richard Carlsson %% @author Richard Carlsson +%% @end %% ===================================================================== %% @doc `epp_dodger' - bypasses the Erlang preprocessor. @@ -29,9 +30,9 @@ %% expanding preprocessor directives and macro applications, as long as %% these are syntactically "well-behaved". Because the normal parse %% trees of the `erl_parse' module cannot represent these things -%% (normally, they are expanded by the Erlang preprocessor (`epp') -%% before the parser sees them), an extended syntax tree -%% is created, using the `erl_syntax' module.

+%% (normally, they are expanded by the Erlang preprocessor {@link +%% //stdlib/epp} before the parser sees them), an extended syntax tree +%% is created, using the {@link erl_syntax} module.

%% NOTES: @@ -93,7 +94,7 @@ %% This is a so-called Erlang I/O ErrorInfo structure; see the {@link %% //stdlib/io} module for details. --type errorinfo() :: {integer(), atom(), term()}. +-type errorinfo() :: erl_scan:error_info(). -type option() :: atom() | {atom(), term()}. @@ -121,8 +122,8 @@ parse_file(File) -> %%
%%
{@type {no_fail, boolean()@}}
%%
If `true', this makes `epp_dodger' replace any program forms -%% that could not be parsed with nodes of type `text' (see -%% `erl_syntax:text/1'), representing the raw token sequence of the +%% that could not be parsed with nodes of type `text' (see {@link +%% erl_syntax:text/1}), representing the raw token sequence of the %% form, instead of reporting a parse error. The default value is %% `false'.
%%
{@type {clever, boolean()@}}
@@ -157,7 +158,7 @@ quick_parse_file(File) -> %% can usually handle more strange cases than the normal, more exact %% parsing. %% -%% Options: see `parse_file/2'. Note however that for +%% Options: see {@link parse_file/2}. Note however that for %% `quick_parse_file/2', the option `no_fail' is `true' by default. %% %% @see quick_parse/2 @@ -201,7 +202,7 @@ do_parse_file(DefEncoding, File, Parser, Options) -> find_invalid_unicode([H | T]) -> case H of - {error, {_Line, file_io_server, invalid_unicode}} -> + {error, {_Location, file_io_server, invalid_unicode}} -> invalid_unicode; _Other -> find_invalid_unicode(T) @@ -216,17 +217,17 @@ find_invalid_unicode([]) -> parse(Dev) -> parse(Dev, 1). -%% @equiv parse(IODevice, StartLine, []) +%% @equiv parse(IODevice, StartLocation, []) %% @see parse/1 --spec parse(file:io_device(), integer()) -> {ok, erl_syntax:forms()}. +-spec parse(file:io_device(), erl_anno:location()) -> {ok, erl_syntax:forms()}. parse(Dev, L) -> parse(Dev, L, []). %% @doc Reads and parses program text from an I/O stream. Characters are %% read from `IODevice' until end-of-file; apart from this, the -%% behaviour is the same as for `parse_file/2'. `StartLine' is the -%% initial line number, which should be a positive integer. +%% behaviour is the same as for {@link parse_file/2}. `StartLocation' is the +%% initial location. %% %% @see parse/2 %% @see parse_file/2 @@ -244,10 +245,10 @@ parse(Dev, L0, Options) -> quick_parse(Dev) -> quick_parse(Dev, 1). -%% @equiv quick_parse(IODevice, StartLine, []) +%% @equiv quick_parse(IODevice, StartLocation, []) %% @see quick_parse/1 --spec quick_parse(file:io_device(), integer()) -> {ok, erl_syntax:forms()}. +-spec quick_parse(file:io_device(), erl_anno:location()) -> {ok, erl_syntax:forms()}. quick_parse(Dev, L) -> quick_parse(Dev, L, []). @@ -259,7 +260,7 @@ quick_parse(Dev, L) -> %% @see quick_parse_form/2 %% @see parse/3 --spec quick_parse(file:io_device(), integer(), [option()]) -> {ok, erl_syntax:forms()}. +-spec quick_parse(file:io_device(), erl_anno:location(), [option()]) -> {ok, erl_syntax:forms()}. quick_parse(Dev, L0, Options) -> parse(Dev, L0, fun quick_parse_form/3, Options). @@ -280,14 +281,14 @@ parse(Dev, L0, Fs, Parser, Options) -> %% ===================================================================== -%% @equiv parse_form(IODevice, StartLine, []) +%% @equiv parse_form(IODevice, StartLocation, []) %% %% @see quick_parse_form/2 --spec parse_form(file:io_device(), non_neg_integer()) -> - {ok, erl_syntax:forms(), non_neg_integer()} | - {eof, non_neg_integer()} | - {error, errorinfo(), non_neg_integer()}. +-spec parse_form(file:io_device(), erl_anno:location()) -> + {ok, erl_syntax:forms(), erl_anno:location()} | + {eof, erl_anno:location()} | + {error, errorinfo(), erl_anno:location()}. parse_form(Dev, L0) -> parse_form(Dev, L0, []). @@ -296,28 +297,28 @@ parse_form(Dev, L0) -> %% marker is found (a period character followed by whitespace), or until %% end-of-file; apart from this, the behaviour is similar to that of %% `parse/3', except that the return values also contain the -%% final line number given that `StartLine' is the initial -%% line number, and that `{eof, LineNo}' may be returned. +%% final location given that `StartLocation' is the initial +%% location, and that `{eof, Location}' may be returned. %% %% @see parse/3 %% @see parse_form/2 %% @see quick_parse_form/3 --spec parse_form(file:io_device(), integer(), [option()]) -> - {ok, erl_syntax:forms(), integer()} | - {eof, integer()} | - {error, errorinfo(), integer()}. +-spec parse_form(file:io_device(), erl_anno:location(), [option()]) -> + {ok, erl_syntax:forms(), erl_anno:location()} | + {eof, erl_anno:location()} | + {error, errorinfo(), erl_anno:location()}. parse_form(Dev, L0, Options) -> parse_form(Dev, L0, fun normal_parser/2, Options). -%% @equiv quick_parse_form(IODevice, StartLine, []) +%% @equiv quick_parse_form(IODevice, StartLocation, []) %% %% @see parse_form/2 --spec quick_parse_form(file:io_device(), non_neg_integer()) -> - {ok, erl_syntax:forms(), non_neg_integer()} | - {eof, non_neg_integer()} | - {error, errorinfo(), non_neg_integer()}. +-spec quick_parse_form(file:io_device(), erl_anno:location()) -> + {ok, erl_syntax:forms(), erl_anno:location()} | + {eof, erl_anno:location()} | + {error, errorinfo(), erl_anno:location()}. quick_parse_form(Dev, L0) -> quick_parse_form(Dev, L0, []). @@ -328,10 +329,10 @@ quick_parse_form(Dev, L0) -> %% @see quick_parse_form/2 %% @see parse_form/3 --spec quick_parse_form(file:io_device(), integer(), [option()]) -> - {ok, erl_syntax:forms(), integer()} | - {eof, integer()} | - {error, errorinfo(), integer()}. +-spec quick_parse_form(file:io_device(), erl_anno:location(), [option()]) -> + {ok, erl_syntax:forms(), erl_anno:location()} | + {eof, erl_anno:location()} | + {error, errorinfo(), erl_anno:location()}. quick_parse_form(Dev, L0, Options) -> parse_form(Dev, L0, fun quick_parser/2, Options). @@ -353,7 +354,12 @@ parse_form(Dev, L0, Parser, Options) -> compact_strings = proplists:get_bool(compact_strings, Options), pre_fixer = proplists:get_value(pre_fixer, Options, fun no_fix/1), post_fixer = proplists:get_value(post_fixer, Options, fun no_fix/1)}, - ScanOpts = proplists:get_value(scan_opts, Options, []), + + %% This has the *potential* to read options for enabling/disabling + %% (i.e. `{feature, TheFeature, enable}') when parsing the file. + {ok, {_Ftrs, ResWordFun}} = erl_features:keyword_fun(Options, fun reserved_word/1), + + ScanOpts = [{reserved_word_fun, ResWordFun} | proplists:get_value(scan_opts, Options, [])], case io:scan_erl_form(Dev, "", L0, ScanOpts) of {ok, Ts, L1} -> case extract_escript_header(Ts) of @@ -396,7 +402,8 @@ parse_form(Parser, Ts, L1, NoFail, Opt) -> {parse_error, _IoErr} when NoFail -> {ok, erl_syntax:set_pos( - erl_syntax:text(tokens_to_string(Ts)), start_pos(Ts, L1)), + erl_syntax:text(tokens_to_string(Ts)), + erl_anno:new(start_pos(Ts, L1))), L1}; {parse_error, IoErr} -> {error, IoErr, L1}; @@ -408,7 +415,7 @@ io_error(L, Desc) -> {L, ?MODULE, Desc}. start_pos([T | _Ts], _L) -> - erl_anno:line(element(2, T)); + erl_anno:location(element(2, T)); start_pos([], L) -> L. @@ -484,32 +491,36 @@ expression_dot() -> quick_parser(Ts, _Opt) -> filter_form(parse_tokens(quickscan_form(Ts))). -quickscan_form([{'-', _L}, {atom, La, define} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, undef} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, include} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, include_lib} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, ifdef} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, ifndef} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {'if', La} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, elif} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, 'else'} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', _L}, {atom, La, endif} | _Ts]) -> - kill_form(La); -quickscan_form([{'-', L}, {'?', _}, {Type, _, _} = N | [{'(', _} | _] = Ts]) +quickscan_form([{'-', _Anno}, {atom, AnnoA, define} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, undef} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, include} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, include_lib} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, ifdef} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, ifndef} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {'if', AnnoA} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, elif} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, 'else'} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {'else', AnnoA} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, endif} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', _Anno}, {atom, AnnoA, feature} | _Ts]) -> + kill_form(AnnoA); +quickscan_form([{'-', Anno}, {'?', _}, {Type, _, _} = N | [{'(', _} | _] = Ts]) when Type =:= atom; Type =:= var -> %% minus, macro and open parenthesis at start of form - assume that %% the macro takes no arguments; e.g. `-?foo(...).' - do_quickscan_macros(N, Ts, [{'-', L}]); -quickscan_form([{'?', _L}, {Type, _, _} = N | [{'(', _} | _] = Ts]) + do_quickscan_macros(N, Ts, [{'-', Anno}]); +quickscan_form([{'?', _Anno}, {Type, _, _} = N | [{'(', _} | _] = Ts]) when Type =:= atom; Type =:= var -> %% macro and open parenthesis at start of form - assume that the %% macro takes no arguments (see scan_macros for details) @@ -517,18 +528,18 @@ quickscan_form([{'?', _L}, {Type, _, _} = N | [{'(', _} | _] = Ts]) quickscan_form(Ts) -> quickscan_macros(Ts). -kill_form(L) -> - [{atom, L, ?pp_form}, {'(', L}, {')', L}, {'->', L}, {atom, L, kill}, {dot, L}]. +kill_form(A) -> + [{atom, A, ?pp_form}, {'(', A}, {')', A}, {'->', A}, {atom, A, kill}, {dot, A}]. quickscan_macros(Ts) -> quickscan_macros(Ts, []). -quickscan_macros([{'?', _}, {Type, _, A} | Ts], [{string, L, S} | As]) +quickscan_macros([{'?', _}, {Type, _, A} | Ts], [{string, AnnoS, S} | As]) when Type =:= atom; Type =:= var -> %% macro after a string literal: change to a single string {_, Ts1} = skip_macro_args(Ts), S1 = S ++ quick_macro_string(A), - quickscan_macros(Ts1, [{string, L, S1} | As]); + quickscan_macros(Ts1, [{string, AnnoS, S1} | As]); quickscan_macros([{'?', _}, {Type, _, _} = N | [{'(', _} | _] = Ts], [{':', _} | _] = As) when Type =:= atom; Type =:= var -> %% macro and open parenthesis after colon - check the token @@ -553,13 +564,13 @@ quickscan_macros([], As) -> lists:reverse(As). %% (after a macro has been found and the arglist skipped, if any) -do_quickscan_macros({_Type, _, A}, [{string, L, S} | Ts], As) -> +do_quickscan_macros({_Type, _, A}, [{string, AnnoS, S} | Ts], As) -> %% string literal following macro: change to single string S1 = quick_macro_string(A) ++ S, - quickscan_macros(Ts, [{string, L, S1} | As]); -do_quickscan_macros({_Type, L, A}, Ts, As) -> + quickscan_macros(Ts, [{string, AnnoS, S1} | As]); +do_quickscan_macros({_Type, AnnoA, A}, Ts, As) -> %% normal case - just replace the macro with an atom - quickscan_macros(Ts, [{atom, L, quick_macro_atom(A)} | As]). + quickscan_macros(Ts, [{atom, AnnoA, quick_macro_atom(A)} | As]). quick_macro_atom(A) -> list_to_atom("?" ++ atom_to_list(A)). @@ -592,15 +603,9 @@ skip_macro_args([{'receive', _} = T | Ts], Es, As) -> skip_macro_args(Ts, ['end' | Es], [T | As]); skip_macro_args([{'try', _} = T | Ts], Es, As) -> skip_macro_args(Ts, ['end' | Es], [T | As]); -skip_macro_args([{'cond', _} = T | Ts], Es, As) -> - skip_macro_args(Ts, ['end' | Es], [T | As]); -skip_macro_args([{E, _} = T | Ts], - [E], - As) -> %final close +skip_macro_args([{E, _} = T | Ts], [E], As) -> %final close {lists:reverse([T | As]), Ts}; -skip_macro_args([{E, _} = T | Ts], - [E | Es], - As) -> %matching close +skip_macro_args([{E, _} = T | Ts], [E | Es], As) -> %matching close skip_macro_args(Ts, Es, [T | As]); skip_macro_args([T | Ts], Es, As) -> skip_macro_args(Ts, Es, [T | As]); @@ -664,58 +669,67 @@ default_prefix(#opt{parse_macro_definitions = false, compact_strings = true}) -> end end. -scan_form([{'-', _L}, {atom, La, define} | Ts], #opt{parse_macro_definitions = false}) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, define} | Ts]; -scan_form([{'-', _L}, {atom, La, define} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, define} +scan_form([{'-', _Anno}, {atom, AnnoA, define} | Ts], #opt{parse_macro_definitions = false}) -> + [ + {atom, AnnoA, ?pp_form}, + {'(', AnnoA}, + {')', AnnoA}, + {'->', AnnoA}, + {atom, AnnoA, define} + | Ts]; +scan_form([{'-', _Anno}, {atom, AnnoA, define} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, define} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, undef} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, undef} +scan_form([{'-', _Anno}, {atom, AnnoA, undef} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, undef} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, include} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, include} +scan_form([{'-', _Anno}, {atom, AnnoA, include} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, include} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, include_lib} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, include_lib} +scan_form([{'-', _Anno}, {atom, AnnoA, include_lib} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, include_lib} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, ifdef} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, ifdef} +scan_form([{'-', _Anno}, {atom, AnnoA, ifdef} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, ifdef} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, ifndef} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, ifndef} +scan_form([{'-', _Anno}, {atom, AnnoA, ifndef} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, ifndef} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {'if', La} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, 'if'} +scan_form([{'-', _Anno}, {'if', AnnoA} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, 'if'} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, elif} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, elif} +scan_form([{'-', _Anno}, {atom, AnnoA, elif} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, elif} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, 'else'} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, 'else'} +scan_form([{'-', _Anno}, {atom, AnnoA, 'else'} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, 'else'} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, endif} | Ts], Opt) -> - [{atom, La, ?pp_form}, {'(', La}, {')', La}, {'->', La}, {atom, La, endif} +scan_form([{'-', Anno}, {'else', AnnoA} | Ts], Opt) -> + %% See previous clause + scan_form([{'-', Anno}, {atom, AnnoA, 'else'} | Ts], Opt); +scan_form([{'-', _Anno}, {atom, AnnoA, endif} | Ts], Opt) -> + [{atom, AnnoA, ?pp_form}, {'(', AnnoA}, {')', AnnoA}, {'->', AnnoA}, {atom, AnnoA, endif} | scan_macros(Ts, Opt)]; -scan_form([{'-', _L}, {atom, La, error} | Ts], _Opt) -> +scan_form([{'-', _Anno}, {atom, AnnoA, error} | Ts], _Opt) -> Desc = build_info_string("-error", Ts), - ErrorInfo = {La, ?MODULE, {error, Desc}}, + ErrorInfo = {erl_anno:location(AnnoA), ?MODULE, {error, Desc}}, erl_syntax:error_marker(ErrorInfo); -scan_form([{'-', _L}, {atom, La, warning} | Ts], _Opt) -> +scan_form([{'-', _Anno}, {atom, AnnoA, warning} | Ts], _Opt) -> Desc = build_info_string("-warning", Ts), - ErrorInfo = {La, ?MODULE, {warning, Desc}}, + ErrorInfo = {erl_anno:location(AnnoA), ?MODULE, {warning, Desc}}, erl_syntax:error_marker(ErrorInfo); -scan_form([{'-', L}, {'?', L1}, {Type, _, _} = N | [{'(', _} | _] = Ts], Opt) +scan_form([{'-', A}, {'?', A1}, {Type, _, _} = N | [{'(', _} | _] = Ts], Opt) when Type =:= atom; Type =:= var -> %% minus, macro and open parenthesis at start of form - assume that %% the macro takes no arguments; e.g. `-?foo(...).' - macro(L1, N, Ts, [{'-', L}], Opt); -scan_form([{'?', L}, {Type, _, _} = N | [{'(', _} | _] = Ts], Opt) + macro(A1, N, Ts, [{'-', A}], Opt); +scan_form([{'?', A}, {Type, _, _} = N | [{'(', _} | _] = Ts], Opt) when Type =:= atom; Type =:= var -> %% macro and open parenthesis at start of form - assume that the %% macro takes no arguments; probably a function declaration on the %% form `?m(...) -> ...', which will not parse if it is rewritten as %% `(?m(...)) -> ...', so it must be handled as `(?m)(...) -> ...' - macro(L, N, Ts, [], Opt); + macro(A, N, Ts, [], Opt); scan_form(Ts, Opt) -> scan_macros(Ts, Opt). @@ -728,12 +742,12 @@ scan_macros(Ts, Opt) -> scan_macros(Ts, [], Opt). scan_macros([{'?', _} = M, {Type, _, _} = N | Ts], - [{string, L, _} = S | As], + [{string, AnnoS, _} = S | As], #opt{clever = true} = Opt) when Type =:= atom; Type =:= var -> %% macro after a string literal: be clever and insert ++ - scan_macros([M, N | Ts], [{'++', L}, S | As], Opt); -scan_macros([{'?', L}, {Type, _, _} = N | [{'(', _} | _] = Ts], [{':', _} | _] = As, Opt) + scan_macros([M, N | Ts], [{'++', AnnoS}, S | As], Opt); +scan_macros([{'?', Anno}, {Type, _, _} = N | [{'(', _} | _] = Ts], [{':', _} | _] = As, Opt) when Type =:= atom; Type =:= var -> %% macro and open parentheses after colon - probably a call %% `m:?F(...)' so the argument list might belong to the call, not @@ -743,21 +757,21 @@ scan_macros([{'?', L}, {Type, _, _} = N | [{'(', _} | _] = Ts], [{':', _} | _] = {Args, Rest} = skip_macro_args(Ts), case Rest of [{'->', _} | _] -> - macro_call(Args, L, N, Rest, As, Opt); + macro_call(Args, Anno, N, Rest, As, Opt); [{'when', _} | _] -> - macro_call(Args, L, N, Rest, As, Opt); + macro_call(Args, Anno, N, Rest, As, Opt); _ -> - macro(L, N, Ts, As, Opt) + macro(Anno, N, Ts, As, Opt) end; -scan_macros([{'?', L}, {Type, _, _} = N | [{'(', _} | _] = Ts], As, Opt) +scan_macros([{'?', Anno}, {Type, _, _} = N | [{'(', _} | _] = Ts], As, Opt) when Type =:= atom; Type =:= var -> %% macro with arguments {Args, Rest} = skip_macro_args(Ts), - macro_call(Args, L, N, Rest, As, Opt); -scan_macros([{'?', L}, {Type, _, _} = N | Ts], As, Opt) + macro_call(Args, Anno, N, Rest, As, Opt); +scan_macros([{'?', Anno}, {Type, _, _} = N | Ts], As, Opt) when Type =:= atom; Type =:= var -> %% macro without arguments - macro(L, N, Ts, As, Opt); + macro(Anno, N, Ts, As, Opt); scan_macros([T | Ts], As, Opt) -> scan_macros(Ts, [T | As], Opt); scan_macros([], As, _Opt) -> @@ -766,15 +780,18 @@ scan_macros([], As, _Opt) -> %% Rewriting to a call which will be recognized by the post-parse pass %% (we insert parentheses to preserve the precedences when parsing). -macro(L, {Type, _, A}, Rest, As, Opt) -> - do_scan_macros([], Rest, [{atom, L, macro_atom(Type, A)} | As], Opt). +macro(Anno, {Type, _, A}, Rest, As, Opt) -> + do_scan_macros([], Rest, [{atom, Anno, macro_atom(Type, A)} | As], Opt). -macro_call([{'(', _}, {')', _}], L, {_, Ln, _} = N, Rest, As, Opt) -> +macro_call([{'(', _}, {')', _}], Anno, {_, AnnoN, _} = N, Rest, As, Opt) -> {Open, Close} = parentheses(As), do_scan_macros([], Rest, - lists:reverse(Open ++ [{atom, L, ?macro_call}, {'(', L}, N, {')', Ln}] ++ Close, - As), + lists:reverse( + Open ++ + [{atom, Anno, ?macro_call}, {'(', Anno}, N, {')', AnnoN}] ++ + Close, + As), Opt); macro_call([{'(', _} | Args], L, {_, Ln, _} = N, Rest, As, Opt) -> {Open, Close} = parentheses(As), @@ -798,19 +815,19 @@ parentheses(_) -> {[{'(', 0}], [{')', 0}]}. %% (after a macro has been found and the arglist skipped, if any) -do_scan_macros(Args, [{string, L, _} | _] = Rest, As, #opt{clever = true} = Opt) -> +do_scan_macros(Args, [{string, AnnoS, _} | _] = Rest, As, #opt{clever = true} = Opt) -> %% string literal following macro: be clever and insert ++ - scan_macros(Args ++ [{'++', L} | Rest], As, Opt); + scan_macros(Args ++ [{'++', AnnoS} | Rest], As, Opt); do_scan_macros(Args, Rest, As, Opt) -> %% normal case - continue scanning scan_macros(Args ++ Rest, As, Opt). -rewrite_form({function, L, ?pp_form, _, [{clause, _, [], [], [{call, _, A, As}]}]}) -> +rewrite_form({function, Anno, ?pp_form, _, [{clause, _, [], [], [{call, _, A, As}]}]}) -> erl_syntax:set_pos( - erl_syntax:attribute(A, rewrite_list(As)), L); -rewrite_form({function, L, ?pp_form, _, [{clause, _, [], [], [A]}]}) -> + erl_syntax:attribute(A, rewrite_list(As)), Anno); +rewrite_form({function, Anno, ?pp_form, _, [{clause, _, [], [], [A]}]}) -> erl_syntax:set_pos( - erl_syntax:attribute(A), L); + erl_syntax:attribute(A), Anno); rewrite_form(T) -> rewrite(T). @@ -904,11 +921,11 @@ fix_stringyfied_macros([{'?', Pos}, {atom, Pos, MacroName} | Rest], Ts) -> fix_stringyfied_macros([Other | Rest], Ts) -> fix_stringyfied_macros(Rest, [Other | Ts]). -fix_define([{atom, L, ?pp_form}, +fix_define([{atom, Anno, ?pp_form}, {'(', _}, {')', _}, {'->', _}, - {atom, La, define}, + {atom, AnnoA, define}, {'(', _}, N, {',', _} @@ -916,12 +933,12 @@ fix_define([{atom, L, ?pp_form}, [{dot, _}, {')', _} | Ts1] = lists:reverse(Ts), S = tokens_to_string(lists:reverse(Ts1)), A = erl_syntax:set_pos( - erl_syntax:atom(define), La), + erl_syntax:atom(define), AnnoA), Txt = erl_syntax:set_pos( - erl_syntax:text(S), La), + erl_syntax:text(S), AnnoA), {form, erl_syntax:set_pos( - erl_syntax:attribute(A, [N, Txt]), L)}; + erl_syntax:attribute(A, [N, Txt]), Anno)}; fix_define(_Ts) -> no_fix. @@ -1049,3 +1066,10 @@ errormsg(String) -> %% ===================================================================== +%% @doc The dodger currently does not process feature attributes +%% correctly, so temporarily consider the `else' and `maybe' atoms +%% always as keywords +-spec reserved_word(Atom :: atom()) -> boolean(). +reserved_word('else') -> true; +reserved_word('maybe') -> true; +reserved_word(Atom) -> erl_scan:f_reserved_word(Atom).