fully parallelize the Preprocess builder

This commit is contained in:
Josh Holtrop 2017-05-25 15:54:33 -04:00
parent 178940cd5d
commit 80a80a7cb0
3 changed files with 31 additions and 70 deletions

View File

@ -79,8 +79,9 @@ module Rscons
# #
# @param options [Hash] Builder run options. # @param options [Hash] Builder run options.
# #
# @return [ThreadedCommand] # @return [String, ThreadedCommand]
# Threaded command to execute. # Target file name if target is up to date or a {ThreadedCommand}
# to execute to build the target.
def run(options) def run(options)
target, sources, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars) target, sources, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
vars = vars.merge({ vars = vars.merge({
@ -88,14 +89,14 @@ module Rscons
'_SOURCES' => sources, '_SOURCES' => sources,
'_DEPFILE' => Rscons.set_suffix(target, env.expand_varref("${DEPFILESUFFIX}", vars)), '_DEPFILE' => Rscons.set_suffix(target, env.expand_varref("${DEPFILESUFFIX}", vars)),
}) })
# Store vars back into options so new keys are accessible in #finalize.
options[:vars] = vars
com_prefix = KNOWN_SUFFIXES.find do |compiler, suffix_var| com_prefix = KNOWN_SUFFIXES.find do |compiler, suffix_var|
sources.first.end_with?(*env.expand_varref("${#{suffix_var}}")) sources.first.end_with?(*env.expand_varref("${#{suffix_var}}"))
end.tap do |v| end.tap do |v|
v.nil? and raise "Error: unknown input file type: #{sources.first.inspect}" v.nil? and raise "Error: unknown input file type: #{sources.first.inspect}"
end.first end.first
command = env.build_command("${#{com_prefix}CMD}", vars) command = env.build_command("${#{com_prefix}CMD}", vars)
# Store vars back into options so new keys are accessible in #finalize.
options[:vars] = vars
standard_threaded_build("#{com_prefix} #{target}", target, command, sources, env, cache) standard_threaded_build("#{com_prefix} #{target}", target, command, sources, env, cache)
end end
@ -103,7 +104,7 @@ module Rscons
# #
# @param options [Hash] Builder finalize options. # @param options [Hash] Builder finalize options.
# #
# @return [String,nil] # @return [String, nil]
# Name of the target file on success or nil on failure. # Name of the target file on success or nil on failure.
def finalize(options) def finalize(options)
if options[:command_status] if options[:command_status]

View File

@ -20,15 +20,13 @@ module Rscons
# Run the builder to produce a build target. # Run the builder to produce a build target.
# #
# @param target [String] Target file name. # @param options [Hash] Builder run options.
# @param sources [Array<String>] Source file name(s).
# @param cache [Cache] The Cache object.
# @param env [Environment] The Environment executing the builder.
# @param vars [Hash,VarSet] Extra construction variables.
# #
# @return [String,false] # @return [String, ThreadedCommand]
# Name of the target file on success or false on failure. # Target file name if target is up to date or a {ThreadedCommand}
def run(target, sources, cache, env, vars) # to execute to build the target.
def run(options)
target, sources, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
if sources.find {|s| s.end_with?(*env.expand_varref("${CXXSUFFIX}", vars))} if sources.find {|s| s.end_with?(*env.expand_varref("${CXXSUFFIX}", vars))}
pp_cc = "${CXX}" pp_cc = "${CXX}"
depgen = "${CXXDEPGEN}" depgen = "${CXXDEPGEN}"
@ -42,17 +40,27 @@ module Rscons
"_SOURCES" => sources, "_SOURCES" => sources,
"_DEPFILE" => Rscons.set_suffix(target, env.expand_varref("${DEPFILESUFFIX}", vars))) "_DEPFILE" => Rscons.set_suffix(target, env.expand_varref("${DEPFILESUFFIX}", vars)))
command = env.build_command("${CPP_CMD}", vars) command = env.build_command("${CPP_CMD}", vars)
unless cache.up_to_date?(target, command, sources, env) # Store vars back into options so new keys are accessible in #finalize.
cache.mkdir_p(File.dirname(target)) options[:vars] = vars
return false unless env.execute("Preprocess #{target}", command) standard_threaded_build("#{name} #{target}", target, command, sources, env, cache)
deps = sources end
if File.exists?(vars["_DEPFILE"])
deps += Environment.parse_makefile_deps(vars["_DEPFILE"], nil) # Finalize the build operation.
FileUtils.rm_f(vars["_DEPFILE"]) #
# @param options [Hash] Builder finalize options.
#
# @return [String, nil]
# Name of the target file on success or nil on failure.
def finalize(options)
if options[:command_status]
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
if File.exists?(vars['_DEPFILE'])
deps += Environment.parse_makefile_deps(vars['_DEPFILE'], nil)
FileUtils.rm_f(vars['_DEPFILE'])
end end
cache.register_build(target, command, deps.uniq, env) cache.register_build(target, options[:tc].command, deps.uniq, env)
target
end end
target
end end
end end

View File

@ -1,48 +0,0 @@
module Rscons
module Builders
describe Preprocess do
let(:env) {Environment.new}
subject {Preprocess.new}
it "supports overriding CC construction variable" do
cache = double(Cache)
command = %w[my_cpp -E -MMD -MF module.mf -o module.pp module.c]
expect(cache).to receive(:up_to_date?).with("module.pp", command, %w[module.c], env).and_return(false)
expect(cache).to receive(:mkdir_p).with(".")
expect(env).to receive(:execute).with("Preprocess module.pp", command).and_return(true)
expect(File).to receive(:exists?).with("module.mf").and_return(true)
expect(Environment).to receive(:parse_makefile_deps).with("module.mf", nil).and_return(%w[module.c one.h two.h])
expect(FileUtils).to receive(:rm_f).with("module.mf")
expect(cache).to receive(:register_build).with("module.pp", command, %w[module.c one.h two.h], env)
expect(subject.run("module.pp", ["module.c"], cache, env, "CC" => "my_cpp")).to eq("module.pp")
end
it "supports overriding CPP_CMD construction variable" do
cache = double(Cache)
command = %w[my_cpp module.c]
expect(cache).to receive(:up_to_date?).with("module.pp", command, %w[module.c], env).and_return(false)
expect(cache).to receive(:mkdir_p).with(".")
expect(env).to receive(:execute).with("Preprocess module.pp", command).and_return(true)
expect(File).to receive(:exists?).with("module.mf").and_return(true)
expect(Environment).to receive(:parse_makefile_deps).with("module.mf", nil).and_return(%w[module.c one.h two.h])
expect(FileUtils).to receive(:rm_f).with("module.mf")
expect(cache).to receive(:register_build).with("module.pp", command, %w[module.c one.h two.h], env)
expect(subject.run("module.pp", ["module.c"], cache, env, "CPP_CMD" => ["my_cpp", "${_SOURCES}"])).to eq("module.pp")
end
it "returns false if executing the preprocessor fails" do
cache = double(Cache)
command = %w[gcc -E -MMD -MF module.mf -o module.pp module.c]
expect(cache).to receive(:up_to_date?).with("module.pp", command, %w[module.c], env).and_return(false)
expect(cache).to receive(:mkdir_p).with(".")
expect(env).to receive(:execute).with("Preprocess module.pp", command).and_return(false)
expect(subject.run("module.pp", ["module.c"], cache, env, {})).to eq(false)
end
end
end
end