2011-01-31 16:43:31 +00:00
|
|
|
%% -*- erlang-indent-level: 4;indent-tabs-mode: nil -*-
|
2009-12-31 18:42:53 +00:00
|
|
|
%% ex: ts=4 sw=4 et
|
2009-11-30 23:03:45 +00:00
|
|
|
%% -------------------------------------------------------------------
|
|
|
|
%%
|
|
|
|
%% rebar: Erlang Build Tools
|
|
|
|
%%
|
|
|
|
%% Copyright (c) 2009 Dave Smith (dizzyd@dizzyd.com)
|
|
|
|
%%
|
|
|
|
%% Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
%% of this software and associated documentation files (the "Software"), to deal
|
|
|
|
%% in the Software without restriction, including without limitation the rights
|
|
|
|
%% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
%% copies of the Software, and to permit persons to whom the Software is
|
|
|
|
%% furnished to do so, subject to the following conditions:
|
|
|
|
%%
|
|
|
|
%% The above copyright notice and this permission notice shall be included in
|
|
|
|
%% all copies or substantial portions of the Software.
|
|
|
|
%%
|
|
|
|
%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
%% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
%% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
%% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
%% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
%% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
%% THE SOFTWARE.
|
|
|
|
%% -------------------------------------------------------------------
|
|
|
|
-module(rebar_port_compiler).
|
|
|
|
|
|
|
|
-export([compile/2,
|
2011-04-05 14:04:03 +00:00
|
|
|
clean/2,
|
|
|
|
setup_env/1]).
|
2009-11-30 23:03:45 +00:00
|
|
|
|
|
|
|
-include("rebar.hrl").
|
|
|
|
|
|
|
|
%% ===================================================================
|
|
|
|
%% Public API
|
|
|
|
%% ===================================================================
|
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
%% Supported configuration variables:
|
|
|
|
%%
|
2012-01-03 17:59:36 +00:00
|
|
|
%% * port_specs - Erlang list of tuples of the forms
|
2012-02-07 19:15:58 +00:00
|
|
|
%% {ArchRegex, TargetFile, Sources, Options}
|
|
|
|
%% {ArchRegex, TargetFile, Sources}
|
|
|
|
%% {TargetFile, Sources}
|
2010-04-28 23:27:54 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% * port_env - Erlang list of key/value pairs which will control
|
|
|
|
%% the environment when running the compiler and linker.
|
2011-01-28 15:08:27 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% By default, the following variables are defined:
|
|
|
|
%% CC - C compiler
|
|
|
|
%% CXX - C++ compiler
|
|
|
|
%% CFLAGS - C compiler
|
|
|
|
%% CXXFLAGS - C++ compiler
|
|
|
|
%% 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
|
|
|
|
%% 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
|
|
|
|
%% 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
|
|
|
|
%% command template)
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% Note that if you wish to extend (vs. replace) these variables,
|
|
|
|
%% you MUST include a shell-style reference in your definition.
|
|
|
|
%% e.g. to extend CFLAGS, do something like:
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% {port_env, [{"CFLAGS", "$CFLAGS -MyOtherOptions"}]}
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% It is also possible to specify platform specific options
|
|
|
|
%% 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:
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
%% {port_env, [{"x86_64.*-linux", "CFLAGS",
|
|
|
|
%% "$CFLAGS -X86Options"}]}
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
2009-11-30 23:03:45 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
%% TODO: reconsider keeping both sources and objects once
|
|
|
|
%% support for deprecated options has been remove.
|
|
|
|
%% remove [] as valid value for sources, objects, and opts
|
|
|
|
%% when removing deprecated options.
|
|
|
|
-record(spec, {type::'drv' | 'exe',
|
|
|
|
target::file:filename(),
|
|
|
|
sources = [] :: [file:filename(), ...] | [],
|
|
|
|
objects = [] :: [file:filename(), ...] | [],
|
|
|
|
opts = [] ::list() | []}).
|
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
compile(Config, AppFile) ->
|
2012-01-03 17:59:36 +00:00
|
|
|
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"),
|
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
%% TODO: remove SpecType and OldSources make get_specs/2
|
|
|
|
%% return list(#spec{}) when removing deprecated options
|
|
|
|
{SpecType, {OldSources, Specs}} = get_specs(Config, AppFile),
|
|
|
|
|
|
|
|
case {SpecType, OldSources, Specs} of
|
|
|
|
{old, [], _} ->
|
|
|
|
ok; % old specs empty
|
|
|
|
{new, [], []} ->
|
|
|
|
ok; % port_specs empty
|
2012-01-03 17:59:36 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
_ -> % have old/new specs
|
|
|
|
|
|
|
|
SharedEnv = rebar_config:get_env(Config, ?MODULE),
|
2009-12-30 12:13:39 +00:00
|
|
|
|
2009-11-30 23:03:45 +00:00
|
|
|
%% Compile each of the sources
|
2012-02-07 19:15:58 +00:00
|
|
|
NewBins = compile_sources(OldSources, Specs, SharedEnv),
|
2009-12-01 17:38:30 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
%% Make sure that the target directories exist
|
2012-01-03 17:59:36 +00:00
|
|
|
?INFO("Using specs ~p\n", [Specs]),
|
2012-02-07 19:15:58 +00:00
|
|
|
lists:foreach(fun(#spec{target=Target}) ->
|
2012-01-03 17:59:36 +00:00
|
|
|
ok = filelib:ensure_dir(Target)
|
|
|
|
end, Specs),
|
|
|
|
|
|
|
|
%% Only relink if necessary, given the Target
|
2011-01-28 15:08:27 +00:00
|
|
|
%% and list of new binaries
|
|
|
|
lists:foreach(
|
2012-02-07 19:15:58 +00:00
|
|
|
fun(#spec{target=Target, objects=Bins, opts=Opts}) ->
|
2011-05-23 14:26:24 +00:00
|
|
|
AllBins = [sets:from_list(Bins),
|
|
|
|
sets:from_list(NewBins)],
|
2011-05-23 10:24:55 +00:00
|
|
|
Intersection = sets:intersection(AllBins),
|
2012-01-03 17:59:36 +00:00
|
|
|
case needs_link(Target, sets:to_list(Intersection)) of
|
2011-05-23 10:24:55 +00:00
|
|
|
true ->
|
2012-01-03 17:59:36 +00:00
|
|
|
LinkTemplate = select_link_template(Target),
|
2012-02-07 19:15:58 +00:00
|
|
|
Env = proplists:get_value(env, Opts, SharedEnv),
|
2012-01-03 17:59:36 +00:00
|
|
|
Cmd = expand_command(LinkTemplate, Env,
|
2011-04-30 20:47:15 +00:00
|
|
|
string:join(Bins, " "),
|
2012-01-03 17:59:36 +00:00
|
|
|
Target),
|
2011-04-30 20:47:15 +00:00
|
|
|
rebar_utils:sh(Cmd, [{env, Env}]);
|
2011-05-23 10:24:55 +00:00
|
|
|
false ->
|
2012-01-03 17:59:36 +00:00
|
|
|
?INFO("Skipping relink of ~s\n", [Target]),
|
2011-05-23 10:24:55 +00:00
|
|
|
ok
|
|
|
|
end
|
2012-01-03 17:59:36 +00:00
|
|
|
end, Specs)
|
2009-11-30 23:03:45 +00:00
|
|
|
end.
|
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
clean(Config, AppFile) ->
|
2012-02-07 19:15:58 +00:00
|
|
|
%% TODO: remove SpecType and OldSources make get_specs/2
|
|
|
|
%% return list(#spec{}) when removing deprecated options
|
|
|
|
{SpecType, {OldSources, Specs}} = get_specs(Config, AppFile),
|
|
|
|
|
|
|
|
case {SpecType, OldSources, Specs} of
|
|
|
|
{old, [], _} ->
|
|
|
|
ok; % old specs empty
|
|
|
|
{new, [], []} ->
|
|
|
|
ok; % port_specs empty
|
|
|
|
|
|
|
|
_ -> % have old/new specs
|
|
|
|
|
|
|
|
lists:foreach(fun(#spec{target=Target, objects=Objects}) ->
|
|
|
|
rebar_file_utils:delete_each([Target]),
|
|
|
|
rebar_file_utils:delete_each(Objects)
|
|
|
|
end, Specs)
|
|
|
|
end.
|
2009-12-01 17:38:30 +00:00
|
|
|
|
2011-04-05 14:04:03 +00:00
|
|
|
setup_env(Config) ->
|
2012-02-07 19:15:58 +00:00
|
|
|
setup_env(Config, []).
|
|
|
|
|
|
|
|
%% ===================================================================
|
|
|
|
%% Internal functions
|
|
|
|
%% ===================================================================
|
|
|
|
|
|
|
|
setup_env(Config, ExtraEnv) ->
|
2011-04-05 14:04:03 +00:00
|
|
|
%% Extract environment values from the config (if specified) and
|
|
|
|
%% merge with the default for this operating system. This enables
|
|
|
|
%% max flexibility for users.
|
2012-03-09 19:00:15 +00:00
|
|
|
DefaultEnv = filter_env(default_env(), []),
|
2012-02-07 19:15:58 +00:00
|
|
|
PortEnv = filter_env(port_env(Config), []),
|
|
|
|
OverrideEnv = global_defines() ++ PortEnv ++ filter_env(ExtraEnv, []),
|
2012-03-09 19:00:15 +00:00
|
|
|
RawEnv = apply_defaults(os_env(), DefaultEnv) ++ OverrideEnv,
|
2011-04-05 14:04:03 +00:00
|
|
|
expand_vars_loop(merge_each_var(RawEnv, [])).
|
2009-11-30 23:03:45 +00:00
|
|
|
|
2011-10-22 20:25:40 +00:00
|
|
|
global_defines() ->
|
2012-01-24 20:05:35 +00:00
|
|
|
Defines = rebar_config:get_global(defines, []),
|
|
|
|
Flags = string:join(["-D" ++ D || D <- Defines], " "),
|
|
|
|
[{"ERL_CFLAGS", "$ERL_CFLAGS " ++ Flags}].
|
2011-10-22 20:25:40 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
replace_extension(File, NewExt) ->
|
|
|
|
OldExt = filename:extension(File),
|
|
|
|
replace_extension(File, OldExt, NewExt).
|
2012-01-03 17:59:36 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
replace_extension(File, OldExt, NewExt) ->
|
|
|
|
filename:rootname(File, OldExt) ++ NewExt.
|
2012-01-03 17:59:36 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
%%
|
|
|
|
%% == compile and link ==
|
|
|
|
%%
|
2012-01-03 17:59:36 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
compile_sources([], Specs, SharedEnv) -> % port_spec
|
|
|
|
lists:foldl(
|
|
|
|
fun(#spec{sources=Sources, type=Type, opts=Opts}, NewBins) ->
|
|
|
|
Env = proplists:get_value(env, Opts, SharedEnv),
|
|
|
|
compile_each(Sources, Type, Env, NewBins)
|
|
|
|
end, [], Specs);
|
|
|
|
compile_sources(OldSources, _Specs, SharedEnv) -> % deprecated
|
|
|
|
compile_each(OldSources, drv, SharedEnv, []).
|
|
|
|
|
|
|
|
compile_each([], _Type, _Env, NewBins) ->
|
|
|
|
lists:reverse(NewBins);
|
|
|
|
compile_each([Source | Rest], Type, Env, NewBins) ->
|
2009-11-30 23:03:45 +00:00
|
|
|
Ext = filename:extension(Source),
|
2012-02-07 19:15:58 +00:00
|
|
|
Bin = replace_extension(Source, Ext, ".o"),
|
2009-12-01 17:38:30 +00:00
|
|
|
case needs_compile(Source, Bin) of
|
|
|
|
true ->
|
|
|
|
?CONSOLE("Compiling ~s\n", [Source]),
|
2012-01-03 17:59:36 +00:00
|
|
|
Template = select_compile_template(Type, compiler(Ext)),
|
|
|
|
rebar_utils:sh(expand_command(Template, Env, Source, Bin),
|
|
|
|
[{env, Env}]),
|
2012-02-07 19:15:58 +00:00
|
|
|
compile_each(Rest, Type, Env, [Bin | NewBins]);
|
2009-12-01 17:38:30 +00:00
|
|
|
false ->
|
|
|
|
?INFO("Skipping ~s\n", [Source]),
|
2012-02-07 19:15:58 +00:00
|
|
|
compile_each(Rest, Type, Env, NewBins)
|
2009-12-01 17:38:30 +00:00
|
|
|
end.
|
2009-11-30 23:03:45 +00:00
|
|
|
|
|
|
|
needs_compile(Source, Bin) ->
|
2011-01-28 15:08:27 +00:00
|
|
|
%% TODO: Generate depends using gcc -MM so we can also
|
|
|
|
%% check for include changes
|
2009-11-30 23:03:45 +00:00
|
|
|
filelib:last_modified(Bin) < filelib:last_modified(Source).
|
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
needs_link(SoName, []) ->
|
|
|
|
filelib:last_modified(SoName) == 0;
|
|
|
|
needs_link(SoName, NewBins) ->
|
|
|
|
MaxLastMod = lists:max([filelib:last_modified(B) || B <- NewBins]),
|
|
|
|
case filelib:last_modified(SoName) of
|
|
|
|
0 ->
|
2009-12-16 15:27:14 +00:00
|
|
|
?DEBUG("Last mod is 0 on ~s\n", [SoName]),
|
2009-12-01 17:38:30 +00:00
|
|
|
true;
|
|
|
|
Other ->
|
2011-04-28 16:39:46 +00:00
|
|
|
?DEBUG("Checking ~p >= ~p\n", [MaxLastMod, Other]),
|
2009-12-16 15:27:14 +00:00
|
|
|
MaxLastMod >= Other
|
2009-12-01 17:38:30 +00:00
|
|
|
end.
|
2009-12-30 12:13:39 +00:00
|
|
|
|
2012-02-07 19:15:58 +00:00
|
|
|
%%
|
|
|
|
%% == port_specs ==
|
|
|
|
%%
|
|
|
|
|
|
|
|
get_specs(Config, AppFile) ->
|
2012-04-17 14:46:59 +00:00
|
|
|
case rebar_config:get_local(Config, port_specs, undefined) of
|
2012-02-07 19:15:58 +00:00
|
|
|
undefined ->
|
|
|
|
%% TODO: DEPRECATED: remove support for non-port_specs syntax
|
|
|
|
{old, old_get_specs(Config, AppFile)};
|
|
|
|
PortSpecs ->
|
|
|
|
{new, get_port_specs(Config, PortSpecs)}
|
|
|
|
end.
|
|
|
|
|
|
|
|
get_port_specs(Config, PortSpecs) ->
|
|
|
|
Filtered = filter_port_specs(PortSpecs),
|
|
|
|
OsType = os:type(),
|
|
|
|
{[], [get_port_spec(Config, OsType, Spec) || Spec <- Filtered]}.
|
|
|
|
|
|
|
|
filter_port_specs(Specs) ->
|
|
|
|
[S || S <- Specs, filter_port_spec(S)].
|
|
|
|
|
|
|
|
filter_port_spec({ArchRegex, _, _, _}) ->
|
|
|
|
rebar_utils:is_arch(ArchRegex);
|
|
|
|
filter_port_spec({ArchRegex, _, _}) ->
|
|
|
|
rebar_utils:is_arch(ArchRegex);
|
|
|
|
filter_port_spec({_, _}) ->
|
|
|
|
true.
|
|
|
|
|
|
|
|
get_port_spec(Config, OsType, {Target, Sources}) ->
|
|
|
|
get_port_spec(Config, OsType, {undefined, Target, Sources, []});
|
|
|
|
get_port_spec(Config, OsType, {Arch, Target, Sources}) ->
|
|
|
|
get_port_spec(Config, OsType, {Arch, Target, Sources, []});
|
|
|
|
get_port_spec(Config, OsType, {_Arch, Target, Sources, Opts}) ->
|
|
|
|
SourceFiles = port_sources(Sources),
|
|
|
|
ObjectFiles = port_objects(SourceFiles),
|
|
|
|
#spec{type=target_type(Target),
|
|
|
|
target=maybe_switch_extension(OsType, Target),
|
|
|
|
sources=SourceFiles,
|
|
|
|
objects=ObjectFiles,
|
|
|
|
opts=port_opts(Config, Opts)}.
|
|
|
|
|
|
|
|
port_sources(Sources) ->
|
|
|
|
lists:flatmap(fun filelib:wildcard/1, Sources).
|
|
|
|
|
|
|
|
port_objects(SourceFiles) ->
|
|
|
|
[replace_extension(O, ".o") || O <- SourceFiles].
|
|
|
|
|
|
|
|
port_opts(Config, Opts) ->
|
|
|
|
[port_opt(Config, O) || O <- Opts].
|
|
|
|
|
|
|
|
port_opt(Config, {env, Env}) ->
|
|
|
|
{env, setup_env(Config, Env)};
|
|
|
|
port_opt(_Config, Opt) ->
|
|
|
|
Opt.
|
|
|
|
|
|
|
|
maybe_switch_extension({win32, nt}, Target) ->
|
|
|
|
switch_to_dll_or_exe(Target);
|
|
|
|
maybe_switch_extension(_OsType, Target) ->
|
|
|
|
Target.
|
|
|
|
|
|
|
|
switch_to_dll_or_exe(Target) ->
|
|
|
|
case filename:extension(Target) of
|
|
|
|
".so" -> filename:rootname(Target, ".so") ++ ".dll";
|
|
|
|
[] -> Target ++ ".exe";
|
|
|
|
Other -> Other
|
|
|
|
end.
|
|
|
|
|
|
|
|
%% TODO: DEPRECATED: remove support for non-port_specs syntax [old_*()]
|
|
|
|
old_get_specs(Config, AppFile) ->
|
|
|
|
OsType = os:type(),
|
|
|
|
SourceFiles = old_get_sources(Config),
|
|
|
|
Specs =
|
2012-04-17 14:46:59 +00:00
|
|
|
case rebar_config:get_local(Config, so_specs, undefined) of
|
2012-02-07 19:15:58 +00:00
|
|
|
undefined ->
|
|
|
|
Objects = port_objects(SourceFiles),
|
|
|
|
%% New form of so_specs is not provided. See if the old form
|
|
|
|
%% of {so_name} is available instead
|
|
|
|
Dir = "priv",
|
2012-04-17 14:46:59 +00:00
|
|
|
SoName =
|
|
|
|
case rebar_config:get_local(Config, so_name, undefined) of
|
|
|
|
undefined ->
|
|
|
|
%% Ok, neither old nor new form is
|
|
|
|
%% available. Use the app name and
|
|
|
|
%% generate a sensible default.
|
|
|
|
AppName = rebar_app_utils:app_name(AppFile),
|
|
|
|
DrvName = ?FMT("~s_drv.so", [AppName]),
|
|
|
|
filename:join([Dir, DrvName]);
|
|
|
|
AName ->
|
|
|
|
%% Old form is available -- use it
|
|
|
|
filename:join(Dir, AName)
|
|
|
|
end,
|
2012-02-07 19:15:58 +00:00
|
|
|
[old_get_so_spec({SoName, Objects}, OsType)];
|
|
|
|
SoSpecs ->
|
|
|
|
[old_get_so_spec(S, OsType) || S <- SoSpecs]
|
|
|
|
end,
|
|
|
|
{SourceFiles, Specs}.
|
|
|
|
|
|
|
|
old_get_sources(Config) ->
|
2012-04-17 14:46:59 +00:00
|
|
|
RawSources = rebar_config:get_local(Config, port_sources,
|
|
|
|
["c_src/*.c"]),
|
2012-02-07 19:15:58 +00:00
|
|
|
FilteredSources = old_filter_port_sources(RawSources),
|
|
|
|
old_expand_sources(FilteredSources).
|
|
|
|
|
|
|
|
old_filter_port_sources(PortSources) ->
|
|
|
|
[S || S <- PortSources, old_is_arch_port_sources(S)].
|
|
|
|
|
|
|
|
old_is_arch_port_sources({Arch, _Sources}) -> rebar_utils:is_arch(Arch);
|
|
|
|
old_is_arch_port_sources(_Sources) -> true.
|
|
|
|
|
|
|
|
old_expand_sources(Sources) ->
|
|
|
|
lists:flatmap(fun filelib:wildcard/1, Sources).
|
|
|
|
|
|
|
|
old_get_so_spec({Target, Objects}, OsType) ->
|
|
|
|
#spec{type=drv,
|
|
|
|
target=maybe_switch_extension(OsType, Target),
|
|
|
|
sources=[],
|
|
|
|
objects=Objects,
|
|
|
|
opts=[]}.
|
|
|
|
|
|
|
|
%%
|
|
|
|
%% == port_env ==
|
|
|
|
%%
|
2009-11-30 23:03:45 +00:00
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
|
|
|
%% Choose a compiler variable, based on a provided extension
|
|
|
|
%%
|
2009-11-30 23:03:45 +00:00
|
|
|
compiler(".cc") -> "$CXX";
|
|
|
|
compiler(".cp") -> "$CXX";
|
|
|
|
compiler(".cxx") -> "$CXX";
|
|
|
|
compiler(".cpp") -> "$CXX";
|
|
|
|
compiler(".CPP") -> "$CXX";
|
|
|
|
compiler(".c++") -> "$CXX";
|
|
|
|
compiler(".C") -> "$CXX";
|
|
|
|
compiler(_) -> "$CC".
|
2009-12-01 17:38:30 +00:00
|
|
|
|
2011-05-12 21:14:24 +00:00
|
|
|
%%
|
|
|
|
%% Given a list of {Key, Value} variables, and another list of default
|
|
|
|
%% {Key, Value} variables, return a merged list where the rule is if the
|
|
|
|
%% default is expandable expand it with the value of the variable list,
|
|
|
|
%% otherwise just return the value of the variable.
|
|
|
|
%%
|
|
|
|
apply_defaults(Vars, Defaults) ->
|
|
|
|
dict:to_list(
|
2011-05-23 14:26:24 +00:00
|
|
|
dict:merge(fun(Key, VarValue, DefaultValue) ->
|
|
|
|
case is_expandable(DefaultValue) of
|
2011-05-23 10:24:55 +00:00
|
|
|
true ->
|
2011-08-08 18:12:04 +00:00
|
|
|
rebar_utils:expand_env_variable(DefaultValue,
|
2011-08-20 16:20:07 +00:00
|
|
|
Key,
|
|
|
|
VarValue);
|
2011-05-23 10:24:55 +00:00
|
|
|
false -> VarValue
|
2011-05-23 14:26:24 +00:00
|
|
|
end
|
|
|
|
end,
|
|
|
|
dict:from_list(Vars),
|
|
|
|
dict:from_list(Defaults))).
|
|
|
|
|
2010-02-16 14:05:29 +00:00
|
|
|
%%
|
|
|
|
%% Given a list of {Key, Value} environment variables, where Key may be defined
|
|
|
|
%% multiple times, walk the list and expand each self-reference so that we
|
|
|
|
%% end with a list of each variable singly-defined.
|
|
|
|
%%
|
|
|
|
merge_each_var([], Vars) ->
|
|
|
|
Vars;
|
|
|
|
merge_each_var([{Key, Value} | Rest], Vars) ->
|
2011-05-11 22:14:25 +00:00
|
|
|
Evalue = case orddict:find(Key, Vars) of
|
|
|
|
error ->
|
|
|
|
%% Nothing yet defined for this key/value.
|
|
|
|
%% Expand any self-references as blank.
|
2011-08-08 18:12:04 +00:00
|
|
|
rebar_utils:expand_env_variable(Value, Key, "");
|
2011-05-11 22:14:25 +00:00
|
|
|
{ok, Value0} ->
|
|
|
|
%% Use previous definition in expansion
|
2011-08-08 18:12:04 +00:00
|
|
|
rebar_utils:expand_env_variable(Value, Key, Value0)
|
2011-05-11 22:14:25 +00:00
|
|
|
end,
|
2010-02-16 14:05:29 +00:00
|
|
|
merge_each_var(Rest, orddict:store(Key, Evalue, Vars)).
|
|
|
|
|
|
|
|
%%
|
|
|
|
%% Give a unique list of {Key, Value} environment variables, expand each one
|
|
|
|
%% for every other key until no further expansions are possible.
|
|
|
|
%%
|
|
|
|
expand_vars_loop(Vars) ->
|
2011-10-26 05:59:50 +00:00
|
|
|
expand_vars_loop(Vars, [], dict:from_list(Vars), 10).
|
2010-02-16 14:05:29 +00:00
|
|
|
|
2011-10-26 05:59:50 +00:00
|
|
|
expand_vars_loop(_Pending, _Recurse, _Vars, 0) ->
|
2010-02-16 14:05:29 +00:00
|
|
|
?ABORT("Max. expansion reached for ENV vars!\n", []);
|
2011-10-26 05:59:50 +00:00
|
|
|
expand_vars_loop([], [], Vars, _Count) ->
|
|
|
|
lists:keysort(1, dict:to_list(Vars));
|
|
|
|
expand_vars_loop([], Recurse, Vars, Count) ->
|
|
|
|
expand_vars_loop(Recurse, [], Vars, Count-1);
|
|
|
|
expand_vars_loop([{K, V} | Rest], Recurse, Vars, Count) ->
|
|
|
|
%% Identify the variables that need expansion in this value
|
2011-10-26 18:08:35 +00:00
|
|
|
ReOpts = [global, {capture, all_but_first, list}],
|
|
|
|
case re:run(V, "\\\${?(\\w+)}?", ReOpts) of
|
2011-10-26 05:59:50 +00:00
|
|
|
{match, Matches} ->
|
|
|
|
%% Identify the unique variables that need to be expanded
|
|
|
|
UniqueMatches = lists:usort([M || [M] <- Matches]),
|
|
|
|
|
2011-10-26 18:08:35 +00:00
|
|
|
%% For each variable, expand it and return the final
|
|
|
|
%% value. Note that if we have a bunch of unresolvable
|
|
|
|
%% variables, nothing happens and we don't bother
|
|
|
|
%% attempting further expansion
|
2011-10-26 05:59:50 +00:00
|
|
|
case expand_keys_in_value(UniqueMatches, V, Vars) of
|
|
|
|
V ->
|
|
|
|
%% No change after expansion; move along
|
|
|
|
expand_vars_loop(Rest, Recurse, Vars, Count);
|
|
|
|
Expanded ->
|
2011-10-26 18:08:35 +00:00
|
|
|
%% Some expansion occurred; move to next k/v but
|
|
|
|
%% revisit this value in the next loop to check
|
|
|
|
%% for further expansion
|
2011-10-26 05:59:50 +00:00
|
|
|
NewVars = dict:store(K, Expanded, Vars),
|
2011-10-26 18:08:35 +00:00
|
|
|
expand_vars_loop(Rest, [{K, Expanded} | Recurse],
|
|
|
|
NewVars, Count)
|
2011-10-26 05:59:50 +00:00
|
|
|
end;
|
|
|
|
|
|
|
|
nomatch ->
|
|
|
|
%% No values in this variable need expansion; move along
|
|
|
|
expand_vars_loop(Rest, Recurse, Vars, Count)
|
2010-02-16 14:05:29 +00:00
|
|
|
end.
|
|
|
|
|
2011-10-26 05:59:50 +00:00
|
|
|
expand_keys_in_value([], Value, _Vars) ->
|
|
|
|
Value;
|
|
|
|
expand_keys_in_value([Key | Rest], Value, Vars) ->
|
|
|
|
NewValue = case dict:find(Key, Vars) of
|
|
|
|
{ok, KValue} ->
|
|
|
|
rebar_utils:expand_env_variable(Value, Key, KValue);
|
|
|
|
error ->
|
|
|
|
Value
|
|
|
|
end,
|
|
|
|
expand_keys_in_value(Rest, NewValue, Vars).
|
|
|
|
|
2011-04-30 20:47:15 +00:00
|
|
|
expand_command(TmplName, Env, InFiles, OutFile) ->
|
|
|
|
Cmd0 = proplists:get_value(TmplName, Env),
|
2011-08-08 18:12:04 +00:00
|
|
|
Cmd1 = rebar_utils:expand_env_variable(Cmd0, "PORT_IN_FILES", InFiles),
|
|
|
|
Cmd2 = rebar_utils:expand_env_variable(Cmd1, "PORT_OUT_FILE", OutFile),
|
2011-04-30 20:47:15 +00:00
|
|
|
re:replace(Cmd2, "\\\$\\w+|\\\${\\w+}", "", [global, {return, list}]).
|
|
|
|
|
2011-05-12 21:14:24 +00:00
|
|
|
%%
|
|
|
|
%% Given a string, determine if it is expandable
|
|
|
|
%%
|
|
|
|
is_expandable(InStr) ->
|
|
|
|
case re:run(InStr,"\\\$",[{capture,none}]) of
|
|
|
|
match -> true;
|
|
|
|
nomatch -> false
|
|
|
|
end.
|
|
|
|
|
2012-03-09 19:00:15 +00:00
|
|
|
port_env(Config) ->
|
|
|
|
%% TODO: remove support for deprecated port_envs option
|
|
|
|
PortEnv = rebar_utils:get_deprecated_list(Config, port_envs, port_env,
|
|
|
|
[], "soon"),
|
|
|
|
%% TODO: remove migration of deprecated port_env DRV_-/EXE_-less vars
|
2012-01-03 17:59:36 +00:00
|
|
|
%% 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,
|
2012-03-09 19:00:15 +00:00
|
|
|
[Convert(EnvVar) || EnvVar <- PortEnv].
|
2012-01-03 17:59:36 +00:00
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
%%
|
|
|
|
%% Filter a list of env vars such that only those which match the provided
|
|
|
|
%% architecture regex (or do not have a regex) are returned.
|
|
|
|
%%
|
2012-03-09 19:00:15 +00:00
|
|
|
filter_env([], Acc) ->
|
2009-12-01 17:38:30 +00:00
|
|
|
lists:reverse(Acc);
|
2012-03-09 19:00:15 +00:00
|
|
|
filter_env([{ArchRegex, Key, Value} | Rest], Acc) ->
|
2009-12-01 17:38:30 +00:00
|
|
|
case rebar_utils:is_arch(ArchRegex) of
|
|
|
|
true ->
|
2012-03-09 19:00:15 +00:00
|
|
|
filter_env(Rest, [{Key, Value} | Acc]);
|
2009-12-01 17:38:30 +00:00
|
|
|
false ->
|
2012-03-09 19:00:15 +00:00
|
|
|
filter_env(Rest, Acc)
|
2009-12-01 17:38:30 +00:00
|
|
|
end;
|
2012-03-09 19:00:15 +00:00
|
|
|
filter_env([{Key, Value} | Rest], Acc) ->
|
|
|
|
filter_env(Rest, [{Key, Value} | Acc]).
|
2009-11-30 23:03:45 +00:00
|
|
|
|
|
|
|
erts_dir() ->
|
|
|
|
lists:concat([code:root_dir(), "/erts-", erlang:system_info(version)]).
|
|
|
|
|
2010-02-16 14:05:29 +00:00
|
|
|
os_env() ->
|
2011-01-28 15:08:27 +00:00
|
|
|
Os = [list_to_tuple(re:split(S, "=", [{return, list}, {parts, 2}])) ||
|
|
|
|
S <- os:getenv()],
|
2011-04-30 20:47:15 +00:00
|
|
|
%% Drop variables without a name (win32)
|
2011-06-02 19:03:09 +00:00
|
|
|
[T1 || {K, _V} = T1 <- Os, K =/= []].
|
2010-02-16 14:05:29 +00:00
|
|
|
|
2012-01-03 17:59:36 +00:00
|
|
|
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.
|
|
|
|
|
2011-12-08 19:03:18 +00:00
|
|
|
erl_interface_dir(Subdir) ->
|
|
|
|
case code:lib_dir(erl_interface, Subdir) of
|
|
|
|
{error, bad_name} ->
|
|
|
|
throw({error, {erl_interface,Subdir,"code:lib_dir(erl_interface)"
|
|
|
|
"is unable to find the erl_interface library."}});
|
|
|
|
Dir -> Dir
|
|
|
|
end.
|
|
|
|
|
2009-12-01 17:38:30 +00:00
|
|
|
default_env() ->
|
2010-04-28 15:31:38 +00:00
|
|
|
[
|
2012-01-03 17:59:36 +00:00
|
|
|
{"CC" , "cc"},
|
|
|
|
{"CXX", "c++"},
|
|
|
|
{"DRV_CXX_TEMPLATE",
|
2011-04-30 20:47:15 +00:00
|
|
|
"$CXX -c $CXXFLAGS $DRV_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"},
|
2012-01-03 17:59:36 +00:00
|
|
|
{"DRV_CC_TEMPLATE",
|
2011-04-30 20:47:15 +00:00
|
|
|
"$CC -c $CFLAGS $DRV_CFLAGS $PORT_IN_FILES -o $PORT_OUT_FILE"},
|
2012-01-03 17:59:36 +00:00
|
|
|
{"DRV_LINK_TEMPLATE",
|
2011-04-30 20:47:15 +00:00
|
|
|
"$CC $PORT_IN_FILES $LDFLAGS $DRV_LDFLAGS -o $PORT_OUT_FILE"},
|
2012-01-03 17:59:36 +00:00
|
|
|
{"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"},
|
|
|
|
|
2011-12-08 19:03:18 +00:00
|
|
|
{"ERL_CFLAGS", lists:concat([" -I", erl_interface_dir(include),
|
2010-09-29 19:28:20 +00:00
|
|
|
" -I", filename:join(erts_dir(), "include"),
|
2010-02-16 14:05:29 +00:00
|
|
|
" "])},
|
2011-12-08 19:03:18 +00:00
|
|
|
{"ERL_EI_LIBDIR", erl_interface_dir(lib)},
|
2012-01-03 17:59:36 +00:00
|
|
|
{"ERL_LDFLAGS" , " -L$ERL_EI_LIBDIR -lerl_interface -lei"},
|
|
|
|
{"ERLANG_ARCH" , rebar_utils:wordsize()},
|
|
|
|
{"ERLANG_TARGET", rebar_utils:get_arch()},
|
|
|
|
|
2011-01-28 15:08:27 +00:00
|
|
|
{"darwin", "DRV_LDFLAGS",
|
|
|
|
"-bundle -flat_namespace -undefined suppress $ERL_LDFLAGS"},
|
2010-04-28 15:31:38 +00:00
|
|
|
|
2011-05-12 21:14:24 +00:00
|
|
|
%% Solaris specific flags
|
|
|
|
{"solaris.*-64$", "CFLAGS", "-D_REENTRANT -m64 $CFLAGS"},
|
|
|
|
{"solaris.*-64$", "CXXFLAGS", "-D_REENTRANT -m64 $CXXFLAGS"},
|
|
|
|
{"solaris.*-64$", "LDFLAGS", "-m64 $LDFLAGS"},
|
2010-04-28 15:31:38 +00:00
|
|
|
|
2011-05-12 21:14:24 +00:00
|
|
|
%% OS X Leopard flags for 64-bit
|
|
|
|
{"darwin9.*-64$", "CFLAGS", "-m64 $CFLAGS"},
|
|
|
|
{"darwin9.*-64$", "CXXFLAGS", "-m64 $CXXFLAGS"},
|
|
|
|
{"darwin9.*-64$", "LDFLAGS", "-arch x86_64 $LDFLAGS"},
|
2010-04-28 15:31:38 +00:00
|
|
|
|
2011-05-12 21:14:24 +00:00
|
|
|
%% OS X Snow Leopard flags for 32-bit
|
|
|
|
{"darwin10.*-32", "CFLAGS", "-m32 $CFLAGS"},
|
|
|
|
{"darwin10.*-32", "CXXFLAGS", "-m32 $CXXFLAGS"},
|
2011-08-01 14:36:35 +00:00
|
|
|
{"darwin10.*-32", "LDFLAGS", "-arch i386 $LDFLAGS"},
|
|
|
|
|
|
|
|
%% OS X Lion flags for 32-bit
|
|
|
|
{"darwin11.*-32", "CFLAGS", "-m32 $CFLAGS"},
|
|
|
|
{"darwin11.*-32", "CXXFLAGS", "-m32 $CXXFLAGS"},
|
|
|
|
{"darwin11.*-32", "LDFLAGS", "-arch i386 $LDFLAGS"}
|
2010-04-28 15:31:38 +00:00
|
|
|
].
|