diff options
| author | Hans Bolinder <[email protected]> | 2016-02-17 13:01:46 +0100 | 
|---|---|---|
| committer | Hans Bolinder <[email protected]> | 2016-02-17 13:01:46 +0100 | 
| commit | f524d3baff6f63846efa571851cc3cc4a55916bd (patch) | |
| tree | fe8a5d13ab678425e40a60117519c6a641a07dfc /lib/stdlib/src | |
| parent | 05a24fed7539f861c7d725829acb196a3991923c (diff) | |
| parent | 6285f2874977e033d02b3c0b77da78e35e771961 (diff) | |
| download | otp-f524d3baff6f63846efa571851cc3cc4a55916bd.tar.gz otp-f524d3baff6f63846efa571851cc3cc4a55916bd.tar.bz2 otp-f524d3baff6f63846efa571851cc3cc4a55916bd.zip | |
Merge branch 'hb/stdlib/epp_typed_record_fields/OTP-13148'
* hb/stdlib/epp_typed_record_fields/OTP-13148:
  stdlib: Update qlc_pt to handle typed record fields
  stdlib: Let the linter detect old typed records
  stdlib: Update the linter to handle typed record fields
  stdlib: Update erl_expand_records to handle typed record fields
  compiler: Update the compiler to handle typed record fields
  stdlib: Update erl_id_trans to handle types and specs
  stdlib: Add a Cover test with typed record field
  stdlib: Update module shell to handle typed record fields
  stdlib: Update ms_transform to handle typed record fields
  stdlib: Modify the preprocessor as to expose typed record fields
Diffstat (limited to 'lib/stdlib/src')
| -rw-r--r-- | lib/stdlib/src/epp.erl | 15 | ||||
| -rw-r--r-- | lib/stdlib/src/erl_expand_records.erl | 36 | ||||
| -rw-r--r-- | lib/stdlib/src/erl_lint.erl | 29 | ||||
| -rw-r--r-- | lib/stdlib/src/ms_transform.erl | 13 | ||||
| -rw-r--r-- | lib/stdlib/src/qlc_pt.erl | 6 | ||||
| -rw-r--r-- | lib/stdlib/src/shell.erl | 8 | 
6 files changed, 38 insertions, 69 deletions
| diff --git a/lib/stdlib/src/epp.erl b/lib/stdlib/src/epp.erl index 45f616bb02..be7c2ec346 100644 --- a/lib/stdlib/src/epp.erl +++ b/lib/stdlib/src/epp.erl @@ -258,20 +258,7 @@ parse_file(Ifile, Options) ->  parse_file(Epp) ->      case parse_erl_form(Epp) of  	{ok,Form} -> -	    case Form of -		{attribute,La,record,{Record, Fields}} -> -		    case normalize_typed_record_fields(Fields) of -			{typed, NewFields} -> -			    [{attribute, La, record, {Record, NewFields}}, -			     {attribute, La, type, -			      {{record, Record}, Fields, []}} -			     |parse_file(Epp)]; -			not_typed -> -			    [Form|parse_file(Epp)] -		    end; -		_ -> -		    [Form|parse_file(Epp)] -	    end; +            [Form|parse_file(Epp)];  	{error,E} ->  	    [{error,E}|parse_file(Epp)];  	{eof,Location} -> diff --git a/lib/stdlib/src/erl_expand_records.erl b/lib/stdlib/src/erl_expand_records.erl index bcfeef7321..9c0a7fb7d5 100644 --- a/lib/stdlib/src/erl_expand_records.erl +++ b/lib/stdlib/src/erl_expand_records.erl @@ -33,8 +33,6 @@                   vcount=0,            % Variable counter                   imports=[],          % Imports                   records=dict:new(),  % Record definitions -		 trecords=sets:new(), % Typed records -		 uses_types=false,    % Are there -spec or -type in the module                   strict_ra=[],        % strict record accesses                   checked_ra=[]        % successfully accessed records                  }). @@ -47,45 +45,18 @@  %% erl_lint without errors.  module(Fs0, Opts0) ->      Opts = compiler_options(Fs0) ++ Opts0, -    TRecs = typed_records(Fs0), -    UsesTypes = uses_types(Fs0), -    St0 = #exprec{compile = Opts, trecords = TRecs, uses_types = UsesTypes}, +    St0 = #exprec{compile = Opts},      {Fs,_St} = forms(Fs0, St0),      Fs.  compiler_options(Forms) ->      lists:flatten([C || {attribute,_,compile,C} <- Forms]). -typed_records(Fs) -> -    typed_records(Fs, sets:new()). - -typed_records([{attribute,_L,type,{{record, Name},_Defs,[]}} | Fs], Trecs) -> -    typed_records(Fs, sets:add_element(Name, Trecs)); -typed_records([_|Fs], Trecs) -> -    typed_records(Fs, Trecs); -typed_records([], Trecs) -> -    Trecs. - -uses_types([{attribute,_L,spec,_}|_]) -> true; -uses_types([{attribute,_L,type,_}|_]) -> true; -uses_types([{attribute,_L,opaque,_}|_]) -> true; -uses_types([_|Fs]) -> uses_types(Fs); -uses_types([]) -> false. -     -forms([{attribute,L,record,{Name,Defs}} | Fs], St0) -> +forms([{attribute,_,record,{Name,Defs}}=Attr | Fs], St0) ->      NDefs = normalise_fields(Defs),      St = St0#exprec{records=dict:store(Name, NDefs, St0#exprec.records)},      {Fs1, St1} = forms(Fs, St), -    %% Check if we need to keep the record information for usage in types. -    case St#exprec.uses_types of -	true -> -	    case sets:is_element(Name, St#exprec.trecords) of -		true -> {Fs1, St1}; -		false -> {[{attribute,L,type,{{record,Name},Defs,[]}}|Fs1], St1} -	    end; -	false -> -	    {Fs1, St1} -    end; +    {[Attr | Fs1], St1};  forms([{attribute,L,import,Is} | Fs0], St0) ->      St1 = import(Is, St0),      {Fs,St2} = forms(Fs0, St1), @@ -513,7 +484,6 @@ lc_tq(Line, [F0 | Qs0], St0) ->  lc_tq(_Line, [], St0) ->      {[],St0#exprec{checked_ra = []}}. -  %% normalise_fields([RecDef]) -> [Field].  %%  Normalise the field definitions to always have a default value. If  %%  none has been given then use 'undefined'. diff --git a/lib/stdlib/src/erl_lint.erl b/lib/stdlib/src/erl_lint.erl index 8d2df1cf66..4ca9a609a8 100644 --- a/lib/stdlib/src/erl_lint.erl +++ b/lib/stdlib/src/erl_lint.erl @@ -360,6 +360,9 @@ format_error({redefine_type, {TypeName, Arity}}) ->  		  [TypeName, gen_type_paren(Arity)]);  format_error({type_syntax, Constr}) ->      io_lib:format("bad ~w type", [Constr]); +format_error(old_abstract_code) -> +    io_lib:format("abstract code generated before Erlang/OTP 19.0 and " +                  "having typed record fields cannot be compiled", []);  format_error({redefine_spec, {M, F, A}}) ->      io_lib:format("spec for ~w:~w/~w already defined", [M, F, A]);  format_error({redefine_spec, {F, A}}) -> @@ -1143,7 +1146,7 @@ check_untyped_records(Forms, St0) ->  	    RecNames = dict:fetch_keys(St0#lint.records),  	    %% these are the records with field(s) containing type info  	    TRecNames = [Name || -			    {attribute,_,type,{{record,Name},Fields,_}} <- Forms, +			    {attribute,_,record,{Name,Fields}} <- Forms,  			    lists:all(fun ({typed_record_field,_,_}) -> true;  					  (_) -> false  				      end, Fields)], @@ -1153,7 +1156,8 @@ check_untyped_records(Forms, St0) ->  			      [] -> St; % exclude records with no fields  			      [_|_] -> add_warning(L, {untyped_record, N}, St)  			  end -		  end, St0, RecNames -- TRecNames); +		  end, St0, ordsets:subtract(ordsets:from_list(RecNames), +                                             ordsets:from_list(TRecNames)));  	false ->  	    St0      end. @@ -2443,7 +2447,10 @@ record_def(Line, Name, Fs0, St0) ->          true -> add_error(Line, {redefine_record,Name}, St0);          false ->              {Fs1,St1} = def_fields(normalise_fields(Fs0), Name, St0), -            St1#lint{records=dict:store(Name, {Line,Fs1}, St1#lint.records)} +            St2 = St1#lint{records=dict:store(Name, {Line,Fs1}, +                                              St1#lint.records)}, +            Types = [T || {typed_record_field, _, T} <- Fs0], +            check_type({type, nowarn(), product, Types}, St2)      end.  %% def_fields([RecDef], RecordName, State) -> {[DefField],State}. @@ -2646,11 +2653,6 @@ find_field(_F, []) -> error.  %%    Attr :: 'type' | 'opaque'  %% Checks that a type definition is valid. -type_def(_Attr, _Line, {record, _RecName}, Fields, [], St0) -> -    %% The record field names and such are checked in the record format. -    %% We only need to check the types. -    Types = [T || {typed_record_field, _, T} <- Fields], -    check_type({type, nowarn(), product, Types}, St0);  type_def(Attr, Line, TypeName, ProtoType, Args, St0) ->      TypeDefs = St0#lint.types,      Arity = length(Args), @@ -2813,6 +2815,8 @@ check_type({user_type, L, TypeName, Args}, SeenVars, St) ->      lists:foldl(fun(T, {AccSeenVars, AccSt}) ->  			check_type(T, AccSeenVars, AccSt)  		end, {SeenVars, St1}, Args); +check_type([{typed_record_field,Field,_T}|_], SeenVars, St) -> +    {SeenVars, add_error(element(2, Field), old_abstract_code, St)};  check_type(I, SeenVars, St) ->      case erl_eval:partial_eval(I) of          {integer,_ILn,_Integer} -> {SeenVars, St}; @@ -2994,9 +2998,10 @@ add_missing_spec_warnings(Forms, St0, Type) ->  		[{FA,L} || {function,L,F,A,_} <- Forms,  			   not lists:member(FA = {F,A}, Specs)];  	    exported -> -		Exps = gb_sets:to_list(St0#lint.exports) -- pseudolocals(), +		Exps0 = gb_sets:to_list(St0#lint.exports) -- pseudolocals(), +                Exps = Exps0 -- Specs,  		[{FA,L} || {function,L,F,A,_} <- Forms, -			   member(FA = {F,A}, Exps -- Specs)] +			   member(FA = {F,A}, Exps)]  	end,      foldl(fun ({FA,L}, St) ->  		  add_warning(L, {missing_spec,FA}, St) @@ -3009,7 +3014,9 @@ check_unused_types(Forms, #lint{usage=Usage, types=Ts, exp_types=ExpTs}=St) ->  	    L = gb_sets:to_list(ExpTs) ++ dict:fetch_keys(D),  	    UsedTypes = gb_sets:from_list(L),  	    FoldFun = -		fun(Type, #typeinfo{line = FileLine}, AccSt) -> +                fun({{record, _}=_Type, 0}, _, AccSt) -> +                        AccSt; % Before Erlang/OTP 19.0 +                   (Type, #typeinfo{line = FileLine}, AccSt) ->                          case loc(FileLine, AccSt) of  			    {FirstFile, _} ->  				case gb_sets:is_member(Type, UsedTypes) of diff --git a/lib/stdlib/src/ms_transform.erl b/lib/stdlib/src/ms_transform.erl index b67b6f75d7..24b5fde1db 100644 --- a/lib/stdlib/src/ms_transform.erl +++ b/lib/stdlib/src/ms_transform.erl @@ -307,15 +307,18 @@ cleanup_filename({Old,OldRec,OldWarnings}) ->  add_record_definition({Name,FieldList}) ->      {KeyList,_} = lists:foldl( -		    fun({record_field,_,{atom,Line0,FieldName}},{L,C}) -> -			    {[{FieldName,C,{atom,Line0,undefined}}|L],C+1}; -		       ({record_field,_,{atom,_,FieldName},Def},{L,C}) -> -			    {[{FieldName,C,Def}|L],C+1} -		    end, +                    fun(F, {L,C}) -> {[record_field(F, C)|L],C+1} end,  		    {[],2},  		    FieldList),      put_records([{Name,KeyList}|get_records()]). +record_field({record_field,_,{atom,Line0,FieldName}}, C) -> +    {FieldName,C,{atom,Line0,undefined}}; +record_field({record_field,_,{atom,_,FieldName},Def}, C) -> +    {FieldName,C,Def}; +record_field({typed_record_field,Field,_Type}, C) -> +    record_field(Field, C). +  forms([F0|Fs0]) ->      F1 = form(F0),      Fs1 = forms(Fs0), diff --git a/lib/stdlib/src/qlc_pt.erl b/lib/stdlib/src/qlc_pt.erl index 9f69cd5003..e4b9768b12 100644 --- a/lib/stdlib/src/qlc_pt.erl +++ b/lib/stdlib/src/qlc_pt.erl @@ -1914,9 +1914,9 @@ expand_pattern_records(P, State) ->  expand_expr_records(E, State) ->      RecordDefs = State#state.records,      A = anno1(), -    Forms = RecordDefs ++ [{function,A,foo,0,[{clause,A,[],[],[pe(E)]}]}], -    [{function,_,foo,0,[{clause,_,[],[],[NE]}]}] =  -        erl_expand_records:module(Forms, [no_strict_record_tests]), +    Forms0 = RecordDefs ++ [{function,A,foo,0,[{clause,A,[],[],[pe(E)]}]}], +    Forms = erl_expand_records:module(Forms0, [no_strict_record_tests]), +    {function,_,foo,0,[{clause,_,[],[],[NE]}]} = lists:last(Forms),      NE.  %% Partial evaluation. diff --git a/lib/stdlib/src/shell.erl b/lib/stdlib/src/shell.erl index ce1d9eb0ff..82a3a2be4f 100644 --- a/lib/stdlib/src/shell.erl +++ b/lib/stdlib/src/shell.erl @@ -917,9 +917,9 @@ expand_records(UsedRecords, E0) ->      RecordDefs = [Def || {_Name,Def} <- UsedRecords],      L = erl_anno:new(1),      E = prep_rec(E0), -    Forms = RecordDefs ++ [{function,L,foo,0,[{clause,L,[],[],[E]}]}], -    [{function,L,foo,0,[{clause,L,[],[],[NE]}]}] =  -        erl_expand_records:module(Forms, [strict_record_tests]),  +    Forms0 = RecordDefs ++ [{function,L,foo,0,[{clause,L,[],[],[E]}]}], +    Forms = erl_expand_records:module(Forms0, [strict_record_tests]), +    {function,L,foo,0,[{clause,L,[],[],[NE]}]} = lists:last(Forms),      prep_rec(NE).  prep_rec({value,_CommandN,_V}=Value) -> @@ -1081,6 +1081,8 @@ record_fields([{record_field,_,{atom,_,Field}} | Fs]) ->      [Field | record_fields(Fs)];  record_fields([{record_field,_,{atom,_,Field},_} | Fs]) ->      [Field | record_fields(Fs)]; +record_fields([{typed_record_field,Field,_Type} | Fs]) -> +    record_fields([Field | Fs]);  record_fields([]) ->      []. | 
