diff options
| author | Tuncer Ayaz <tuncer.ayaz@gmail.com> | 2012-01-03 18:59:36 +0100 | 
|---|---|---|
| committer | Tuncer Ayaz <tuncer.ayaz@gmail.com> | 2012-02-03 16:26:14 +0100 | 
| commit | 6898eff2c7ef8e91f7a0d0a033c365de1f97cecf (patch) | |
| tree | 7fa0072c337da57cdea54000499f19b7cde4e9a9 /src | |
| parent | ba538094bafe568031c505ffe88d683945cdecd2 (diff) | |
Rework port compiler support
 * consolidate options
 * add support for building executables
Diffstat (limited to 'src')
| -rw-r--r-- | src/rebar_port_compiler.erl | 275 | 
1 files changed, 200 insertions, 75 deletions
| diff --git a/src/rebar_port_compiler.erl b/src/rebar_port_compiler.erl index bee9644..ef62c4c 100644 --- a/src/rebar_port_compiler.erl +++ b/src/rebar_port_compiler.erl @@ -38,21 +38,14 @@  %% Supported configuration variables:  %% -%% * port_sources - Erlang list of filenames or wildcards to be compiled. May -%%                  also contain a tuple consisting of a regular expression to -%%                  be applied against the system architecture and a list of -%%                  filenames or wildcards to include should the expression -%%                  pass. -%% -%% * so_specs  - Erlang list of tuples of the form -%%               {"priv/so_name.so", ["c_src/object_file_name.o"]} -%%               useful for building multiple *.so files. +%% * port_specs - Erlang list of tuples of the forms +%%                {arch_regex(), "priv/foo.so", ["c_src/foo.c"]} +%%                {"priv/foo", ["c_src/foo.c"]}  %%  %% * port_envs - Erlang list of key/value pairs which will control  %%               the environment when running the compiler and linker.  %% -%%               By default, the following variables -%%               are defined: +%%               By default, the following variables are defined:  %%               CC       - C compiler  %%               CXX      - C++ compiler  %%               CFLAGS   - C compiler @@ -60,12 +53,17 @@  %%               LDFLAGS  - Link flags  %%               ERL_CFLAGS  - default -I paths for erts and ei  %%               ERL_LDFLAGS - default -L and -lerl_interface -lei -%%               DRV_CFLAGS  - flags that will be used for compiling the driver -%%               DRV_LDFLAGS - flags that will be used for linking the driver +%%               DRV_CFLAGS  - flags that will be used for compiling +%%               DRV_LDFLAGS - flags that will be used for linking +%%               EXE_CFLAGS  - flags that will be used for compiling +%%               EXE_LDFLAGS - flags that will be used for linking  %%               ERL_EI_LIBDIR - ei library directory -%%               CXX_TEMPLATE  - C++ command template -%%               CC_TEMPLATE   - C command template -%%               LINK_TEMPLATE - Linker command template +%%               DRV_CXX_TEMPLATE  - C++ command template +%%               DRV_CC_TEMPLATE   - C command template +%%               DRV_LINK_TEMPLATE - Linker command template +%%               EXE_CXX_TEMPLATE  - C++ command template +%%               EXE_CC_TEMPLATE   - C command template +%%               EXE_LINK_TEMPLATE - Linker command template  %%               PORT_IN_FILES - contains a space separated list of input  %%                    file(s), (used in command template)  %%               PORT_OUT_FILE - contains the output filename (used in @@ -78,8 +76,8 @@  %%               {port_envs, [{"CFLAGS", "$CFLAGS -MyOtherOptions"}]}  %%  %%               It is also possible to specify platform specific options -%%               by specifying a tripletwhere the first string is a regex -%%               that is checked against erlang's system architecture string. +%%               by specifying a triplet where the first string is a regex +%%               that is checked against Erlang's system architecture string.  %%               e.g. to specify a CFLAG that only applies to x86_64 on linux  %%               do:  %% @@ -88,64 +86,75 @@  %%  compile(Config, AppFile) -> -    %% Compose list of sources from config file -- defaults to c_src/*.c -    Sources = expand_sources(rebar_config:get_list(Config, port_sources, -                                                   ["c_src/*.c"]), []), -    case Sources of +    rebar_utils:deprecated(port_sources, port_specs, Config, "soon"), +    rebar_utils:deprecated(so_name, port_specs, Config, "soon"), +    rebar_utils:deprecated(so_specs, port_specs, Config, "soon"), + +    SourceFiles = get_sources(Config), + +    case SourceFiles of          [] ->              ok;          _ ->              Env = setup_env(Config),              %% Compile each of the sources -            {NewBins, ExistingBins} = compile_each(Sources, Config, Env, +            {NewBins, ExistingBins} = compile_each(SourceFiles, Config, Env,                                                     [], []), -            %% Construct the driver name and make sure priv/ exists -            SoSpecs = so_specs(Config, AppFile, NewBins ++ ExistingBins), -            ?INFO("Using specs ~p\n", [SoSpecs]), -            lists:foreach(fun({SoName,_}) -> -                                  ok = filelib:ensure_dir(SoName) -                          end, SoSpecs), - -            %% Only relink if necessary, given the SoName +            %% Construct the target filename and make sure that the +            %% target directory exists +            Specs = port_specs(Config, AppFile, NewBins ++ ExistingBins), +            ?INFO("Using specs ~p\n", [Specs]), +            lists:foreach(fun({_, Target,_}) -> +                                  ok = filelib:ensure_dir(Target); +                             ({Target, _}) -> +                                  ok = filelib:ensure_dir(Target) +                          end, Specs), + +            %% Only relink if necessary, given the Target              %% and list of new binaries              lists:foreach( -              fun({SoName,Bins}) -> +              fun({Target, Sources}) -> +                      Bins = lists:map(fun source_to_bin/1, Sources),                        AllBins = [sets:from_list(Bins),                                   sets:from_list(NewBins)],                        Intersection = sets:intersection(AllBins), -                      case needs_link(SoName, sets:to_list(Intersection)) of +                      case needs_link(Target, sets:to_list(Intersection)) of                            true -> -                              Cmd = expand_command("LINK_TEMPLATE", Env, +                              LinkTemplate = select_link_template(Target), +                              Cmd = expand_command(LinkTemplate, Env,                                                     string:join(Bins, " "), -                                                   SoName), +                                                   Target),                                rebar_utils:sh(Cmd, [{env, Env}]);                            false -> -                              ?INFO("Skipping relink of ~s\n", [SoName]), +                              ?INFO("Skipping relink of ~s\n", [Target]),                                ok                        end -              end, SoSpecs) +              end, Specs)      end.  clean(Config, AppFile) ->      %% Build a list of sources so as to derive all the bins we generated -    Sources = expand_sources(rebar_config:get_list(Config, port_sources, -                                                   ["c_src/*.c"]), []), +    Sources = get_sources(Config),      rebar_file_utils:delete_each([source_to_bin(S) || S <- Sources]), -    %% Delete the .so file -    ExtractSoName = fun({SoName, _}) -> SoName end, -    rebar_file_utils:delete_each([ExtractSoName(S) -                                  || S <- so_specs(Config, AppFile, -                                                   expand_objects(Sources))]). +    %% Delete the target file +    ExtractTarget = fun({_, Target, _}) -> +                            Target; +                       ({Target, _}) -> +                            Target +                    end, +    rebar_file_utils:delete_each([ExtractTarget(S) +                                  || S <- port_specs(Config, AppFile, +                                                     expand_objects(Sources))]).  setup_env(Config) ->      %% Extract environment values from the config (if specified) and      %% merge with the default for this operating system. This enables      %% max flexibility for users.      DefaultEnvs  = filter_envs(default_env(), []), -    PortEnvs = rebar_config:get_list(Config, port_envs, []), +    PortEnvs = port_envs(Config),      OverrideEnvs = global_defines() ++ filter_envs(PortEnvs, []),      RawEnv = apply_defaults(os_env(), DefaultEnvs) ++ OverrideEnvs,      expand_vars_loop(merge_each_var(RawEnv, [])). @@ -159,6 +168,36 @@ global_defines() ->      Flags = string:join(["-D" ++ D || D <- Defines], " "),      [{"ERL_CFLAGS", "$ERL_CFLAGS " ++ Flags}]. +get_sources(Config) -> +    case rebar_config:get_list(Config, port_specs, []) of +        [] -> +            %% TODO: DEPRECATED: remove +            expand_sources(rebar_config:get_list(Config, port_sources, +                                                 ["c_src/*.c"]), []); +        PortSpecs -> +            expand_port_specs(PortSpecs) +    end. + +expand_port_specs(Specs) -> +    lists:append(lists:map(fun({_, Target, FileSpecs}) -> +                                   expand_file_specs(Target, FileSpecs); +                              ({Target, FileSpecs}) -> +                                   expand_file_specs(Target, FileSpecs) +                           end, filter_port_specs(Specs))). + +expand_file_specs(Target, FileSpecs) -> +    Sources = lists:append([filelib:wildcard(FS) || FS <- FileSpecs]), +    [{Target, Src} || Src <- Sources]. + +filter_port_specs(Specs) -> +    lists:filter(fun({ArchRegex, _, _}) -> +                         rebar_utils:is_arch(ArchRegex); +                    ({_, _}) -> +                         true +                 end, Specs). + + +%% TODO: DEPRECATED: remove  expand_sources([], Acc) ->      Acc;  expand_sources([{ArchRegex, Spec} | Rest], Acc) -> @@ -174,34 +213,35 @@ expand_sources([Spec | Rest], Acc) ->      expand_sources(Rest, Acc2).  expand_objects(Sources) -> -    [filename:join([filename:dirname(F), filename:basename(F) ++ ".o"]) -     || F <- Sources]. +    [expand_object(".o", Src) || Src <- Sources]. + +expand_object(Ext, {_Target, Source}) -> +    expand_object(Ext, Source); +expand_object(Ext, Source) -> +    filename:join(filename:dirname(Source), filename:basename(Source) ++ Ext).  compile_each([], _Config, _Env, NewBins, ExistingBins) ->      {lists:reverse(NewBins), lists:reverse(ExistingBins)}; -compile_each([Source | Rest], Config, Env, NewBins, ExistingBins) -> +compile_each([RawSource | Rest], Config, Env, NewBins, ExistingBins) -> +    %% TODO: DEPRECATED: remove +    {Type, Source} = source_type(RawSource),      Ext = filename:extension(Source),      Bin = filename:rootname(Source, Ext) ++ ".o",      case needs_compile(Source, Bin) of          true ->              ?CONSOLE("Compiling ~s\n", [Source]), -            case compiler(Ext) of -                "$CC" -> -                    rebar_utils:sh(expand_command("CC_TEMPLATE", Env, -                                                  Source, Bin), -                                   [{env, Env}]); -                "$CXX" -> -                    rebar_utils:sh(expand_command("CXX_TEMPLATE", Env, -                                                  Source, Bin), -                                   [{env, Env}]) -            end, +            Template = select_compile_template(Type, compiler(Ext)), +            rebar_utils:sh(expand_command(Template, Env, Source, Bin), +                           [{env, Env}]),              compile_each(Rest, Config, Env, [Bin | NewBins], ExistingBins); -          false ->              ?INFO("Skipping ~s\n", [Source]),              compile_each(Rest, Config, Env, NewBins, [Bin | ExistingBins])      end. +source_type({Target, Source}) -> {target_type(Target), Source}; +source_type(Source)           -> {drv, Source}. +  needs_compile(Source, Bin) ->      %% TODO: Generate depends using gcc -MM so we can also      %% check for include changes @@ -342,6 +382,33 @@ is_expandable(InStr) ->          nomatch -> false      end. +port_envs(Config) -> +    PortEnvs = rebar_config:get_list(Config, port_envs, []), +    %% TODO: remove migration of deprecated port_envs (DRV_-/EXE_-less vars) +    %%       when the deprecation grace period ends +    WarnAndConvertVar = fun(Var) -> +                                New = "DRV_" ++ Var, +                                rebar_utils:deprecated(Var, New, "soon"), +                                New +                        end, +    ConvertVar = fun(Var="CXX_TEMPLATE") -> WarnAndConvertVar(Var); +                    (Var="CC_TEMPLATE")  -> WarnAndConvertVar(Var); +                    (Var="LINK_TEMPLATE") -> WarnAndConvertVar(Var); +                    (Var) -> Var +                 end, +    %% Also warn about references to deprecated vars? omitted for +    %% performance reasons. +    ReplaceVars = fun(Val) -> +                          re:replace(Val, "\\$(CXX|CC|LINK)(_TEMPLATE)", +                                     "DRV_\\1\\2", [{return,list}, global]) +                  end, +    Convert = fun({ArchRegex, Var, Val}) -> +                      {ArchRegex, ConvertVar(Var), ReplaceVars(Val)}; +                 ({Var, Val}) -> +                      {ConvertVar(Var), ReplaceVars(Val)} +              end, +    [Convert(Env) || Env <- PortEnvs]. +  %%  %% Filter a list of env vars such that only those which match the provided  %% architecture regex (or do not have a regex) are returned. @@ -368,6 +435,30 @@ os_env() ->      %% Drop variables without a name (win32)      [T1 || {K, _V} = T1 <- Os, K =/= []]. +select_compile_template(drv, Compiler) -> +    select_compile_drv_template(Compiler); +select_compile_template(exe, Compiler) -> +    select_compile_exe_template(Compiler). + +select_compile_drv_template("$CC")  -> "DRV_CC_TEMPLATE"; +select_compile_drv_template("$CXX") -> "DRV_CXX_TEMPLATE". + +select_compile_exe_template("$CC")  -> "EXE_CC_TEMPLATE"; +select_compile_exe_template("$CXX") -> "EXE_CXX_TEMPLATE". + +select_link_template(Target) -> +    case target_type(Target) of +        drv -> "DRV_LINK_TEMPLATE"; +        exe -> "EXE_LINK_TEMPLATE" +    end. + +target_type(Target) -> target_type1(filename:extension(Target)). + +target_type1(".so")  -> drv; +target_type1(".dll") -> drv; +target_type1("")     -> exe; +target_type1(".exe") -> exe. +  erl_interface_dir(Subdir) ->      case code:lib_dir(erl_interface, Subdir) of          {error, bad_name} -> @@ -378,25 +469,35 @@ erl_interface_dir(Subdir) ->  default_env() ->      [ -     {"CXX_TEMPLATE", +     {"CC" , "cc"}, +     {"CXX", "c++"}, +     {"DRV_CXX_TEMPLATE",        "$CXX -c $CXXFLAGS $DRV_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"}, -     {"CC_TEMPLATE", +     {"DRV_CC_TEMPLATE",        "$CC -c $CFLAGS $DRV_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"}, -     {"LINK_TEMPLATE", +     {"DRV_LINK_TEMPLATE",        "$CC $PORT_IN_FILES $LDFLAGS $DRV_LDFLAGS -o $PORT_OUT_FILE"}, -     {"CC", "cc"}, -     {"CXX", "c++"}, +     {"EXE_CXX_TEMPLATE", +      "$CXX -c $CXXFLAGS $EXE_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"}, +     {"EXE_CC_TEMPLATE", +      "$CC -c $CFLAGS $EXE_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"}, +     {"EXE_LINK_TEMPLATE", +      "$CC $PORT_IN_FILES $LDFLAGS $EXE_LDFLAGS -o $PORT_OUT_FILE"}, +     {"DRV_CFLAGS" , "-g -Wall -fPIC $ERL_CFLAGS"}, +     {"DRV_LDFLAGS", "-shared $ERL_LDFLAGS"}, +     {"EXE_CFLAGS" , "-g -Wall -fPIC $ERL_CFLAGS"}, +     {"EXE_LDFLAGS", "$ERL_LDFLAGS"}, +       {"ERL_CFLAGS", lists:concat([" -I", erl_interface_dir(include),                                    " -I", filename:join(erts_dir(), "include"),                                    " "])}, -     {"ERL_LDFLAGS", " -L$ERL_EI_LIBDIR -lerl_interface -lei"}, -     {"DRV_CFLAGS", "-g -Wall -fPIC $ERL_CFLAGS"}, -     {"DRV_LDFLAGS", "-shared $ERL_LDFLAGS"},       {"ERL_EI_LIBDIR", erl_interface_dir(lib)}, +     {"ERL_LDFLAGS"  , " -L$ERL_EI_LIBDIR -lerl_interface -lei"}, +     {"ERLANG_ARCH"  , rebar_utils:wordsize()}, +     {"ERLANG_TARGET", rebar_utils:get_arch()}, +       {"darwin", "DRV_LDFLAGS",        "-bundle -flat_namespace -undefined suppress $ERL_LDFLAGS"}, -     {"ERLANG_ARCH", rebar_utils:wordsize()}, -     {"ERLANG_TARGET", rebar_utils:get_arch()},       %% Solaris specific flags       {"solaris.*-64$", "CFLAGS", "-D_REENTRANT -m64 $CFLAGS"}, @@ -419,30 +520,54 @@ default_env() ->       {"darwin11.*-32", "LDFLAGS", "-arch i386 $LDFLAGS"}      ]. - - +source_to_bin({_Target, Source}) -> +    source_to_bin(Source);  source_to_bin(Source) ->      Ext = filename:extension(Source),      filename:rootname(Source, Ext) ++ ".o". -so_specs(Config, AppFile, Bins) -> -    Specs = make_so_specs(Config, AppFile, Bins), +port_specs(Config, AppFile, Bins) -> +    Specs = make_port_specs(Config, AppFile, Bins),      case os:type() of          {win32, nt} -> -            [switch_so_to_dll(SoSpec) || SoSpec <- Specs]; +            [switch_to_dll_or_exe(Spec) || Spec <- Specs];          _ ->              Specs      end. -switch_so_to_dll(Orig = {Name, Spec}) -> +switch_to_dll_or_exe(Orig = {Name, Spec}) ->      case filename:extension(Name) of          ".so" ->              {filename:rootname(Name, ".so") ++ ".dll", Spec}; +        [] -> +            {Name ++ ".exe", Spec};          _ ->              %% Not a .so; leave it              Orig      end. +make_port_specs(Config, AppFile, Bins) -> +    case rebar_config:get(Config, port_specs, undefined) of +        undefined -> +            %% TODO: DEPRECATED: remove +            make_so_specs(Config, AppFile, Bins); +        PortSpecs -> +            %% filter based on ArchRegex +            Specs0 = lists:filter(fun({ArchRegex, _Target, _Sources}) -> +                                          rebar_utils:is_arch(ArchRegex); +                                     (_) -> +                                          true +                                  end, PortSpecs), +            %% TODO: DEPRECATED: remove support for non-port_specs syntax +            %% drop ArchRegex from specs +            lists:map(fun({_, Target, Sources}) -> +                              {Target, Sources}; +                         (Spec) -> +                              Spec +                      end, Specs0) +    end. + +%% DEPRECATED  make_so_specs(Config, AppFile, Bins) ->      case rebar_config:get(Config, so_specs, undefined) of          undefined -> | 
