Compare commits
23 Commits
Author | SHA1 | Date | |
---|---|---|---|
b970490de2 | |||
87f1b92e7a | |||
a65247db0c | |||
b6f0e61052 | |||
93c5561190 | |||
355a8383d0 | |||
64a7e56ee1 | |||
|
24ed649982 | ||
eff9a10d0c | |||
6ad3fb0785 | |||
32e8a55b4e | |||
3733cb6114 | |||
5beddd63ac | |||
4967f89a76 | |||
7cc68b2ce1 | |||
be058dd18b | |||
ad703a5c84 | |||
|
fb5d107581 | ||
d5c1a4b41a | |||
3f63bf5d13 | |||
ce47bd3599 | |||
|
68cab5e24d | ||
|
f92dd62ebd |
30
CHANGELOG.md
30
CHANGELOG.md
@ -1,5 +1,35 @@
|
|||||||
## ChangeLog
|
## ChangeLog
|
||||||
|
|
||||||
|
### v1.19.0
|
||||||
|
|
||||||
|
#### New Features
|
||||||
|
|
||||||
|
- #117 - ruby 2.7 compatibility
|
||||||
|
|
||||||
|
#### Fixes
|
||||||
|
|
||||||
|
- #115 - Check that user dependencies exist before running builders.
|
||||||
|
|
||||||
|
### v1.18.0
|
||||||
|
|
||||||
|
#### Performance Improvements
|
||||||
|
|
||||||
|
- Backport 2.x VarSet performance improvements to 1.x branch.
|
||||||
|
- Remove Cache dirty-tracking
|
||||||
|
- Reduce recursion of VarSet#expand_varref by expanding vars earlier.
|
||||||
|
|
||||||
|
#### Fixes
|
||||||
|
|
||||||
|
- #73 - fix inconsistent type handling in VarSet#expand_varref
|
||||||
|
- #75 - avoid mkdir() race conditions
|
||||||
|
|
||||||
|
### v1.17.0
|
||||||
|
|
||||||
|
#### New Features
|
||||||
|
|
||||||
|
- allow construction variable expansion on `true` and `false` values.
|
||||||
|
- remove makefile target name check when parsing dependencies
|
||||||
|
|
||||||
### v1.16.0
|
### v1.16.0
|
||||||
|
|
||||||
#### New Features
|
#### New Features
|
||||||
|
38
Gemfile.lock
38
Gemfile.lock
@ -1,7 +1,7 @@
|
|||||||
PATH
|
PATH
|
||||||
remote: .
|
remote: .
|
||||||
specs:
|
specs:
|
||||||
rscons (1.16.0)
|
rscons (1.19.0)
|
||||||
json (>= 1.8, < 3.0)
|
json (>= 1.8, < 3.0)
|
||||||
|
|
||||||
GEM
|
GEM
|
||||||
@ -9,28 +9,28 @@ GEM
|
|||||||
specs:
|
specs:
|
||||||
diff-lcs (1.3)
|
diff-lcs (1.3)
|
||||||
docile (1.1.5)
|
docile (1.1.5)
|
||||||
json (2.1.0)
|
json (2.3.0)
|
||||||
rake (12.3.1)
|
rake (13.0.1)
|
||||||
rdoc (5.1.0)
|
rdoc (6.2.1)
|
||||||
rspec (3.6.0)
|
rspec (3.9.0)
|
||||||
rspec-core (~> 3.6.0)
|
rspec-core (~> 3.9.0)
|
||||||
rspec-expectations (~> 3.6.0)
|
rspec-expectations (~> 3.9.0)
|
||||||
rspec-mocks (~> 3.6.0)
|
rspec-mocks (~> 3.9.0)
|
||||||
rspec-core (3.6.0)
|
rspec-core (3.9.1)
|
||||||
rspec-support (~> 3.6.0)
|
rspec-support (~> 3.9.1)
|
||||||
rspec-expectations (3.6.0)
|
rspec-expectations (3.9.1)
|
||||||
diff-lcs (>= 1.2.0, < 2.0)
|
diff-lcs (>= 1.2.0, < 2.0)
|
||||||
rspec-support (~> 3.6.0)
|
rspec-support (~> 3.9.0)
|
||||||
rspec-mocks (3.6.0)
|
rspec-mocks (3.9.0)
|
||||||
diff-lcs (>= 1.2.0, < 2.0)
|
diff-lcs (>= 1.2.0, < 2.0)
|
||||||
rspec-support (~> 3.6.0)
|
rspec-support (~> 3.9.0)
|
||||||
rspec-support (3.6.0)
|
rspec-support (3.9.2)
|
||||||
simplecov (0.15.0)
|
simplecov (0.15.1)
|
||||||
docile (~> 1.1.0)
|
docile (~> 1.1.0)
|
||||||
json (>= 1.8, < 3)
|
json (>= 1.8, < 3)
|
||||||
simplecov-html (~> 0.10.0)
|
simplecov-html (~> 0.10.0)
|
||||||
simplecov-html (0.10.2)
|
simplecov-html (0.10.2)
|
||||||
yard (0.9.12)
|
yard (0.9.24)
|
||||||
|
|
||||||
PLATFORMS
|
PLATFORMS
|
||||||
ruby
|
ruby
|
||||||
@ -41,8 +41,8 @@ DEPENDENCIES
|
|||||||
rdoc
|
rdoc
|
||||||
rscons!
|
rscons!
|
||||||
rspec
|
rspec
|
||||||
simplecov
|
simplecov (~> 0.15.0)
|
||||||
yard
|
yard
|
||||||
|
|
||||||
BUNDLED WITH
|
BUNDLED WITH
|
||||||
1.16.2
|
2.1.4
|
||||||
|
1
build_tests/simple/new_dep
Normal file
1
build_tests/simple/new_dep
Normal file
@ -0,0 +1 @@
|
|||||||
|
# This file is referenced by cache_debugging.rb
|
@ -5,6 +5,7 @@ require_relative "rscons/cache"
|
|||||||
require_relative "rscons/environment"
|
require_relative "rscons/environment"
|
||||||
require_relative "rscons/job_set"
|
require_relative "rscons/job_set"
|
||||||
require_relative "rscons/threaded_command"
|
require_relative "rscons/threaded_command"
|
||||||
|
require_relative "rscons/util"
|
||||||
require_relative "rscons/varset"
|
require_relative "rscons/varset"
|
||||||
require_relative "rscons/version"
|
require_relative "rscons/version"
|
||||||
|
|
||||||
|
@ -30,7 +30,7 @@ module Rscons
|
|||||||
'ASSUFFIX' => ['.S'],
|
'ASSUFFIX' => ['.S'],
|
||||||
'ASPPPATH' => '${CPPPATH}',
|
'ASPPPATH' => '${CPPPATH}',
|
||||||
'ASPPFLAGS' => '${CPPFLAGS}',
|
'ASPPFLAGS' => '${CPPFLAGS}',
|
||||||
'ASDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}', '-MT', 'TARGET'],
|
'ASDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}'],
|
||||||
'ASCMD' => ['${AS}', '-c', '-o', '${_TARGET}', '${ASDEPGEN}', '${INCPREFIX}${ASPPPATH}', '${ASPPFLAGS}', '${ASFLAGS}', '${_SOURCES}'],
|
'ASCMD' => ['${AS}', '-c', '-o', '${_TARGET}', '${ASDEPGEN}', '${INCPREFIX}${ASPPPATH}', '${ASPPFLAGS}', '${ASFLAGS}', '${_SOURCES}'],
|
||||||
|
|
||||||
'CPPFLAGS' => ['${CPPDEFPREFIX}${CPPDEFINES}'],
|
'CPPFLAGS' => ['${CPPDEFPREFIX}${CPPDEFINES}'],
|
||||||
@ -42,19 +42,19 @@ module Rscons
|
|||||||
'CC' => 'gcc',
|
'CC' => 'gcc',
|
||||||
'CFLAGS' => [],
|
'CFLAGS' => [],
|
||||||
'CSUFFIX' => ['.c'],
|
'CSUFFIX' => ['.c'],
|
||||||
'CCDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}', '-MT', 'TARGET'],
|
'CCDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}'],
|
||||||
'CCCMD' => ['${CC}', '-c', '-o', '${_TARGET}', '${CCDEPGEN}', '${INCPREFIX}${CPPPATH}', '${CPPFLAGS}', '${CFLAGS}', '${CCFLAGS}', '${_SOURCES}'],
|
'CCCMD' => ['${CC}', '-c', '-o', '${_TARGET}', '${CCDEPGEN}', '${INCPREFIX}${CPPPATH}', '${CPPFLAGS}', '${CFLAGS}', '${CCFLAGS}', '${_SOURCES}'],
|
||||||
|
|
||||||
'CXX' => 'g++',
|
'CXX' => 'g++',
|
||||||
'CXXFLAGS' => [],
|
'CXXFLAGS' => [],
|
||||||
'CXXSUFFIX' => ['.cc', '.cpp', '.cxx', '.C'],
|
'CXXSUFFIX' => ['.cc', '.cpp', '.cxx', '.C'],
|
||||||
'CXXDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}', '-MT', 'TARGET'],
|
'CXXDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}'],
|
||||||
'CXXCMD' =>['${CXX}', '-c', '-o', '${_TARGET}', '${CXXDEPGEN}', '${INCPREFIX}${CPPPATH}', '${CPPFLAGS}', '${CXXFLAGS}', '${CCFLAGS}', '${_SOURCES}'],
|
'CXXCMD' =>['${CXX}', '-c', '-o', '${_TARGET}', '${CXXDEPGEN}', '${INCPREFIX}${CPPPATH}', '${CPPFLAGS}', '${CXXFLAGS}', '${CCFLAGS}', '${_SOURCES}'],
|
||||||
|
|
||||||
'DC' => 'gdc',
|
'DC' => 'gdc',
|
||||||
'DFLAGS' => [],
|
'DFLAGS' => [],
|
||||||
'DSUFFIX' => ['.d'],
|
'DSUFFIX' => ['.d'],
|
||||||
'DDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}', '-MT', 'TARGET'],
|
'DDEPGEN' => ['-MMD', '-MF', '${_DEPFILE}'],
|
||||||
'D_IMPORT_PATH' => [],
|
'D_IMPORT_PATH' => [],
|
||||||
'DCCMD' => ['${DC}', '-c', '-o', '${_TARGET}', '${DDEPGEN}', '${INCPREFIX}${D_IMPORT_PATH}', '${DFLAGS}', '${_SOURCES}'],
|
'DCCMD' => ['${DC}', '-c', '-o', '${_TARGET}', '${DDEPGEN}', '${INCPREFIX}${D_IMPORT_PATH}', '${DFLAGS}', '${_SOURCES}'],
|
||||||
}
|
}
|
||||||
@ -115,7 +115,7 @@ module Rscons
|
|||||||
if options[:command_status]
|
if options[:command_status]
|
||||||
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
||||||
if File.exists?(vars['_DEPFILE'])
|
if File.exists?(vars['_DEPFILE'])
|
||||||
deps += Environment.parse_makefile_deps(vars['_DEPFILE'], 'TARGET')
|
deps += Environment.parse_makefile_deps(vars['_DEPFILE'])
|
||||||
FileUtils.rm_f(vars['_DEPFILE'])
|
FileUtils.rm_f(vars['_DEPFILE'])
|
||||||
end
|
end
|
||||||
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
||||||
|
@ -55,7 +55,7 @@ module Rscons
|
|||||||
if options[:command_status]
|
if options[:command_status]
|
||||||
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
||||||
if File.exists?(vars['_DEPFILE'])
|
if File.exists?(vars['_DEPFILE'])
|
||||||
deps += Environment.parse_makefile_deps(vars['_DEPFILE'], nil)
|
deps += Environment.parse_makefile_deps(vars['_DEPFILE'])
|
||||||
FileUtils.rm_f(vars['_DEPFILE'])
|
FileUtils.rm_f(vars['_DEPFILE'])
|
||||||
end
|
end
|
||||||
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
||||||
|
@ -100,7 +100,7 @@ module Rscons
|
|||||||
if options[:command_status]
|
if options[:command_status]
|
||||||
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
target, deps, cache, env, vars = options.values_at(:target, :sources, :cache, :env, :vars)
|
||||||
if File.exists?(vars['_DEPFILE'])
|
if File.exists?(vars['_DEPFILE'])
|
||||||
deps += Environment.parse_makefile_deps(vars['_DEPFILE'], 'TARGET')
|
deps += Environment.parse_makefile_deps(vars['_DEPFILE'])
|
||||||
FileUtils.rm_f(vars['_DEPFILE'])
|
FileUtils.rm_f(vars['_DEPFILE'])
|
||||||
end
|
end
|
||||||
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
cache.register_build(target, options[:tc].command, deps.uniq, env)
|
||||||
|
@ -85,17 +85,14 @@ module Rscons
|
|||||||
@lookup_checksums = {}
|
@lookup_checksums = {}
|
||||||
end
|
end
|
||||||
|
|
||||||
# Write the cache to disk to be loaded next time.
|
# Write the cache to disk.
|
||||||
#
|
#
|
||||||
# @return [void]
|
# @return [void]
|
||||||
def write
|
def write
|
||||||
if @dirty || (@cache["version"] != VERSION)
|
@cache["version"] = VERSION
|
||||||
@cache["version"] = VERSION
|
File.open(CACHE_FILE, "w") do |fh|
|
||||||
File.open(CACHE_FILE, "w") do |fh|
|
fh.puts(JSON.dump(@cache))
|
||||||
fh.puts(JSON.dump(@cache))
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
@dirty = false
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Check if target(s) are up to date.
|
# Check if target(s) are up to date.
|
||||||
@ -249,7 +246,6 @@ module Rscons
|
|||||||
}
|
}
|
||||||
end,
|
end,
|
||||||
}
|
}
|
||||||
@dirty = true
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -272,9 +268,8 @@ module Rscons
|
|||||||
next if parts[i] == ""
|
next if parts[i] == ""
|
||||||
subpath = File.join(*parts[0, i + 1])
|
subpath = File.join(*parts[0, i + 1])
|
||||||
unless File.exists?(subpath)
|
unless File.exists?(subpath)
|
||||||
FileUtils.mkdir(subpath)
|
FileUtils.mkdir_p(subpath)
|
||||||
@cache["directories"][subpath] = true
|
@cache["directories"][subpath] = true
|
||||||
@dirty = true
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -287,6 +282,16 @@ module Rscons
|
|||||||
@cache["directories"].keys
|
@cache["directories"].keys
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Return a file's checksum, or the previously calculated checksum for
|
||||||
|
# the same file.
|
||||||
|
#
|
||||||
|
# @param file [String] The file name.
|
||||||
|
#
|
||||||
|
# @return [String] The file's checksum.
|
||||||
|
def lookup_checksum(file)
|
||||||
|
@lookup_checksums[file] || calculate_checksum(file)
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
# Return a String key based on the target name to use in the on-disk cache.
|
# Return a String key based on the target name to use in the on-disk cache.
|
||||||
@ -315,17 +320,6 @@ module Rscons
|
|||||||
@cache["targets"] ||= {}
|
@cache["targets"] ||= {}
|
||||||
@cache["directories"] ||= {}
|
@cache["directories"] ||= {}
|
||||||
@lookup_checksums = {}
|
@lookup_checksums = {}
|
||||||
@dirty = false
|
|
||||||
end
|
|
||||||
|
|
||||||
# Return a file's checksum, or the previously calculated checksum for
|
|
||||||
# the same file.
|
|
||||||
#
|
|
||||||
# @param file [String] The file name.
|
|
||||||
#
|
|
||||||
# @return [String] The file's checksum.
|
|
||||||
def lookup_checksum(file)
|
|
||||||
@lookup_checksums[file] || calculate_checksum(file)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
# Calculate and return a file's checksum.
|
# Calculate and return a file's checksum.
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
require "fileutils"
|
require "fileutils"
|
||||||
require "set"
|
require "set"
|
||||||
require "shellwords"
|
require "shellwords"
|
||||||
require "thwait"
|
|
||||||
|
|
||||||
module Rscons
|
module Rscons
|
||||||
# The Environment class is the main programmatic interface to Rscons. It
|
# The Environment class is the main programmatic interface to Rscons. It
|
||||||
@ -280,6 +279,24 @@ module Rscons
|
|||||||
@varset.send(:[], *args)
|
@varset.send(:[], *args)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Access the value of a construction variable.
|
||||||
|
#
|
||||||
|
# @since 1.18.0
|
||||||
|
#
|
||||||
|
# This method is similar to #[] but does not make a copy-on-access copy of
|
||||||
|
# the variable accessed. This means that the returned value is NOT safe to
|
||||||
|
# be modified by the caller. Thus the caller must guarantee that it does
|
||||||
|
# not modify the returned value.
|
||||||
|
#
|
||||||
|
# @param key [String, Symbol]
|
||||||
|
# The construction variable name.
|
||||||
|
#
|
||||||
|
# @return [Object]
|
||||||
|
# The construction variable's value.
|
||||||
|
def get_var(key)
|
||||||
|
@varset.get_var(key)
|
||||||
|
end
|
||||||
|
|
||||||
# Set a construction variable's value.
|
# Set a construction variable's value.
|
||||||
#
|
#
|
||||||
# @see VarSet#[]=
|
# @see VarSet#[]=
|
||||||
@ -323,6 +340,7 @@ module Rscons
|
|||||||
cache.clear_checksum_cache!
|
cache.clear_checksum_cache!
|
||||||
|
|
||||||
if job
|
if job
|
||||||
|
validate_user_deps(job[:target], @user_deps[job[:target]], cache)
|
||||||
result = run_builder(job[:builder],
|
result = run_builder(job[:builder],
|
||||||
job[:target],
|
job[:target],
|
||||||
job[:sources],
|
job[:sources],
|
||||||
@ -385,12 +403,12 @@ module Rscons
|
|||||||
|
|
||||||
# Expand a construction variable reference.
|
# Expand a construction variable reference.
|
||||||
#
|
#
|
||||||
# @param varref [Array, String] Variable reference to expand.
|
# @param varref [nil, String, Array, Proc, Symbol, TrueClass, FalseClass] Variable reference to expand.
|
||||||
# @param extra_vars [Hash, VarSet]
|
# @param extra_vars [Hash, VarSet]
|
||||||
# Extra variables to use in addition to (or replace) the Environment's
|
# Extra variables to use in addition to (or replace) the Environment's
|
||||||
# construction variables when expanding the variable reference.
|
# construction variables when expanding the variable reference.
|
||||||
#
|
#
|
||||||
# @return [Array, String] Expansion of the variable reference.
|
# @return [nil, String, Array, Symbol, TrueClass, FalseClass] Expansion of the variable reference.
|
||||||
def expand_varref(varref, extra_vars = nil)
|
def expand_varref(varref, extra_vars = nil)
|
||||||
vars = if extra_vars.nil?
|
vars = if extra_vars.nil?
|
||||||
@varset
|
@varset
|
||||||
@ -782,7 +800,7 @@ module Rscons
|
|||||||
append["LDFLAGS", ["-arch", val]]
|
append["LDFLAGS", ["-arch", val]]
|
||||||
end
|
end
|
||||||
skip = true
|
skip = true
|
||||||
elsif word =~ /^#{self["CPPDEFPREFIX"]}(.*)$/
|
elsif word =~ /^#{get_var("CPPDEFPREFIX")}(.*)$/
|
||||||
handle["CPPDEFINES", $1]
|
handle["CPPDEFINES", $1]
|
||||||
elsif word == "-include"
|
elsif word == "-include"
|
||||||
if val = words[i + 1]
|
if val = words[i + 1]
|
||||||
@ -795,11 +813,11 @@ module Rscons
|
|||||||
append["LDFLAGS", ["-isysroot", val]]
|
append["LDFLAGS", ["-isysroot", val]]
|
||||||
end
|
end
|
||||||
skip = true
|
skip = true
|
||||||
elsif word =~ /^#{self["INCPREFIX"]}(.*)$/
|
elsif word =~ /^#{get_var("INCPREFIX")}(.*)$/
|
||||||
handle["CPPPATH", $1]
|
handle["CPPPATH", $1]
|
||||||
elsif word =~ /^#{self["LIBLINKPREFIX"]}(.*)$/
|
elsif word =~ /^#{get_var("LIBLINKPREFIX")}(.*)$/
|
||||||
handle["LIBS", $1]
|
handle["LIBS", $1]
|
||||||
elsif word =~ /^#{self["LIBDIRPREFIX"]}(.*)$/
|
elsif word =~ /^#{get_var("LIBDIRPREFIX")}(.*)$/
|
||||||
handle["LIBPATH", $1]
|
handle["LIBPATH", $1]
|
||||||
elsif word == "-mno-cygwin"
|
elsif word == "-mno-cygwin"
|
||||||
append["CCFLAGS", [word]]
|
append["CCFLAGS", [word]]
|
||||||
@ -848,7 +866,7 @@ module Rscons
|
|||||||
# @return [void]
|
# @return [void]
|
||||||
def merge_flags(flags)
|
def merge_flags(flags)
|
||||||
flags.each_pair do |key, val|
|
flags.each_pair do |key, val|
|
||||||
if self[key].is_a?(Array) and val.is_a?(Array)
|
if self.get_var(key).is_a?(Array) and val.is_a?(Array)
|
||||||
self[key] += val
|
self[key] += val
|
||||||
else
|
else
|
||||||
self[key] = val
|
self[key] = val
|
||||||
@ -993,10 +1011,7 @@ module Rscons
|
|||||||
!thread.alive?
|
!thread.alive?
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
if threads.empty?
|
Util.wait_for_thread(*threads)
|
||||||
raise "No threads to wait for"
|
|
||||||
end
|
|
||||||
ThreadsWait.new(*threads).next_wait
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -1065,16 +1080,14 @@ module Rscons
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
# Parse dependencies for a given target from a Makefile.
|
# Parse dependencies from a Makefile.
|
||||||
#
|
#
|
||||||
# This method is used internally by Rscons builders.
|
# This method is used internally by Rscons builders.
|
||||||
#
|
#
|
||||||
# @param mf_fname [String] File name of the Makefile to read.
|
# @param mf_fname [String] File name of the Makefile to read.
|
||||||
# @param target [String, nil]
|
|
||||||
# Name of the target to gather dependencies for, nil for any/all.
|
|
||||||
#
|
#
|
||||||
# @return [Array<String>] Paths of dependency files.
|
# @return [Array<String>] Paths of dependency files.
|
||||||
def self.parse_makefile_deps(mf_fname, target)
|
def self.parse_makefile_deps(mf_fname)
|
||||||
deps = []
|
deps = []
|
||||||
buildup = ''
|
buildup = ''
|
||||||
File.read(mf_fname).each_line do |line|
|
File.read(mf_fname).each_line do |line|
|
||||||
@ -1082,16 +1095,34 @@ module Rscons
|
|||||||
buildup += ' ' + $1
|
buildup += ' ' + $1
|
||||||
else
|
else
|
||||||
buildup += ' ' + line
|
buildup += ' ' + line
|
||||||
if buildup =~ /^(.*): (.*)$/
|
if buildup =~ /^.*: (.*)$/
|
||||||
mf_target, mf_deps = $1.strip, $2
|
mf_deps = $1
|
||||||
if target.nil? or mf_target == target
|
deps += mf_deps.split(' ').map(&:strip)
|
||||||
deps += mf_deps.split(' ').map(&:strip)
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
buildup = ''
|
buildup = ''
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
deps
|
deps
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Ensures that user dependencies exist with valid checksums in the
|
||||||
|
# cache. Raises an exception if any dependency is invalid.
|
||||||
|
#
|
||||||
|
# @param target [String]
|
||||||
|
# Target to be built
|
||||||
|
# @param user_deps [Array<String>, nil]
|
||||||
|
# User dependencies of the target
|
||||||
|
# @param cache [Cache]
|
||||||
|
# Rscons cache instance
|
||||||
|
#
|
||||||
|
# @return [void]
|
||||||
|
def validate_user_deps(target, user_deps, cache)
|
||||||
|
return if user_deps.nil?
|
||||||
|
user_deps.each do |dep|
|
||||||
|
if cache.lookup_checksum(dep) == ""
|
||||||
|
raise "User dependency #{dep} of target #{target} is invalid"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
34
lib/rscons/util.rb
Normal file
34
lib/rscons/util.rb
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
module Rscons
|
||||||
|
# A collection of stand-alone utility methods.
|
||||||
|
module Util
|
||||||
|
class << self
|
||||||
|
|
||||||
|
# Wait for any of a number of threads to complete.
|
||||||
|
#
|
||||||
|
# @param threads [Array<Thread>]
|
||||||
|
# Threads to wait for.
|
||||||
|
#
|
||||||
|
# @return [Thread]
|
||||||
|
# The Thread that completed.
|
||||||
|
def wait_for_thread(*threads)
|
||||||
|
if threads.empty?
|
||||||
|
raise "No threads to wait for"
|
||||||
|
end
|
||||||
|
queue = Queue.new
|
||||||
|
threads.each do |thread|
|
||||||
|
# Create a wait thread for each thread we're waiting for.
|
||||||
|
Thread.new do
|
||||||
|
begin
|
||||||
|
thread.join
|
||||||
|
ensure
|
||||||
|
queue.push(thread)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
# Wait for any thread to complete.
|
||||||
|
queue.pop
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
@ -11,11 +11,13 @@ module Rscons
|
|||||||
append(vars)
|
append(vars)
|
||||||
end
|
end
|
||||||
|
|
||||||
# Access the value of variable.
|
# Access the value of a variable.
|
||||||
#
|
#
|
||||||
# @param key [String, Symbol] The variable name.
|
# @param key [String, Symbol]
|
||||||
|
# The variable name.
|
||||||
#
|
#
|
||||||
# @return [Object] The variable's value.
|
# @return [Object]
|
||||||
|
# The variable's value.
|
||||||
def [](key)
|
def [](key)
|
||||||
if @my_vars.include?(key)
|
if @my_vars.include?(key)
|
||||||
@my_vars[key]
|
@my_vars[key]
|
||||||
@ -30,11 +32,38 @@ module Rscons
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Access the value of a variable.
|
||||||
|
#
|
||||||
|
# This method is similar to #[] but does not make a copy-on-access copy of
|
||||||
|
# the variable accessed. This means that the returned value is NOT safe to
|
||||||
|
# be modified by the caller. Thus the caller must guarantee that it does
|
||||||
|
# not modify the returned value.
|
||||||
|
#
|
||||||
|
# @param key [String, Symbol]
|
||||||
|
# The variable name.
|
||||||
|
#
|
||||||
|
# @return [Object]
|
||||||
|
# The variable's value.
|
||||||
|
def get_var(key)
|
||||||
|
if @my_vars.include?(key)
|
||||||
|
@my_vars[key]
|
||||||
|
else
|
||||||
|
@coa_vars.each do |coa_vars|
|
||||||
|
if coa_vars.include?(key)
|
||||||
|
return coa_vars[key]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
# Assign a value to a variable.
|
# Assign a value to a variable.
|
||||||
#
|
#
|
||||||
# @param key [String, Symbol] The variable name.
|
# @param key [String, Symbol]
|
||||||
|
# The variable name.
|
||||||
#
|
#
|
||||||
# @param val [Object] The value to set.
|
# @param val [Object]
|
||||||
|
# The value to set.
|
||||||
def []=(key, val)
|
def []=(key, val)
|
||||||
@my_vars[key] = val
|
@my_vars[key] = val
|
||||||
end
|
end
|
||||||
@ -86,40 +115,50 @@ module Rscons
|
|||||||
# Replace "$!{var}" variable references in varref with the expanded
|
# Replace "$!{var}" variable references in varref with the expanded
|
||||||
# variables' values, recursively.
|
# variables' values, recursively.
|
||||||
#
|
#
|
||||||
# @param varref [nil, String, Array, Proc]
|
# @param varref [nil, String, Array, Proc, Symbol, TrueClass, FalseClass]
|
||||||
# Value containing references to variables.
|
# Value containing references to variables.
|
||||||
# @param lambda_args [Array]
|
# @param lambda_args [Array]
|
||||||
# Arguments to pass to any lambda variable values to be expanded.
|
# Arguments to pass to any lambda variable values to be expanded.
|
||||||
#
|
#
|
||||||
# @return [nil, String, Array]
|
# @return [nil, String, Array, Symbol, TrueClass, FalseClass]
|
||||||
# Expanded value with "$!{var}" variable references replaced.
|
# Expanded value with "$!{var}" variable references replaced.
|
||||||
def expand_varref(varref, lambda_args)
|
def expand_varref(varref, lambda_args)
|
||||||
if varref.is_a?(String)
|
case varref
|
||||||
|
when String
|
||||||
if varref =~ /^(.*)\$\{([^}]+)\}(.*)$/
|
if varref =~ /^(.*)\$\{([^}]+)\}(.*)$/
|
||||||
prefix, varname, suffix = $1, $2, $3
|
prefix, varname, suffix = $1, $2, $3
|
||||||
varval = expand_varref(self[varname], lambda_args)
|
prefix = expand_varref(prefix, lambda_args) unless prefix.empty?
|
||||||
if varval.is_a?(String) or varval.nil?
|
varval = expand_varref(get_var(varname), lambda_args)
|
||||||
expand_varref("#{prefix}#{varval}#{suffix}", lambda_args)
|
# suffix needs no expansion since the regex matches the last occurence
|
||||||
elsif varval.is_a?(Array)
|
case varval
|
||||||
varval.map {|vv| expand_varref("#{prefix}#{vv}#{suffix}", lambda_args)}.flatten
|
when Array
|
||||||
|
if prefix.is_a?(Array)
|
||||||
|
varval.map {|vv| prefix.map {|p| "#{p}#{vv}#{suffix}"}}.flatten
|
||||||
|
else
|
||||||
|
varval.map {|vv| "#{prefix}#{vv}#{suffix}"}
|
||||||
|
end
|
||||||
|
when String, Symbol, true, false, nil
|
||||||
|
if prefix.is_a?(Array)
|
||||||
|
prefix.map {|p| "#{p}#{varval}#{suffix}"}
|
||||||
|
else
|
||||||
|
"#{prefix}#{varval}#{suffix}"
|
||||||
|
end
|
||||||
else
|
else
|
||||||
raise "I do not know how to expand a variable reference to a #{varval.class.name} (from #{varname.inspect} => #{self[varname].inspect})"
|
raise "Unknown construction variable type: #{varval.class} (from #{varname.inspect} => #{get_var(varname).inspect})"
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
varref
|
varref
|
||||||
end
|
end
|
||||||
elsif varref.is_a?(Array)
|
when Array
|
||||||
varref.map do |ent|
|
varref.map do |ent|
|
||||||
expand_varref(ent, lambda_args)
|
expand_varref(ent, lambda_args)
|
||||||
end.flatten
|
end.flatten
|
||||||
elsif varref.is_a?(Proc)
|
when Symbol, true, false, nil
|
||||||
expand_varref(varref[*lambda_args], lambda_args)
|
|
||||||
elsif varref.nil?
|
|
||||||
nil
|
|
||||||
elsif varref.is_a?(Symbol)
|
|
||||||
varref
|
varref
|
||||||
|
when Proc
|
||||||
|
expand_varref(varref[*lambda_args], lambda_args)
|
||||||
else
|
else
|
||||||
raise "Unknown varref type: #{varref.class} (#{varref.inspect})"
|
raise "Unknown construction variable type: #{varref.class} (#{varref.inspect})"
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -184,16 +223,16 @@ module Rscons
|
|||||||
#
|
#
|
||||||
# @return [Object] Deep copied value.
|
# @return [Object] Deep copied value.
|
||||||
def deep_dup(obj)
|
def deep_dup(obj)
|
||||||
obj_class = obj.class
|
case obj
|
||||||
if obj_class == Hash
|
when String
|
||||||
|
obj.dup
|
||||||
|
when Array
|
||||||
|
obj.map { |v| deep_dup(v) }
|
||||||
|
when Hash
|
||||||
obj.reduce({}) do |result, (k, v)|
|
obj.reduce({}) do |result, (k, v)|
|
||||||
result[k] = deep_dup(v)
|
result[k] = deep_dup(v)
|
||||||
result
|
result
|
||||||
end
|
end
|
||||||
elsif obj_class == Array
|
|
||||||
obj.map { |v| deep_dup(v) }
|
|
||||||
elsif obj_class == String
|
|
||||||
obj.dup
|
|
||||||
else
|
else
|
||||||
obj
|
obj
|
||||||
end
|
end
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
module Rscons
|
module Rscons
|
||||||
# gem version
|
# gem version
|
||||||
VERSION = "1.16.0"
|
VERSION = "1.19.0"
|
||||||
end
|
end
|
||||||
|
@ -22,7 +22,7 @@ Gem::Specification.new do |gem|
|
|||||||
|
|
||||||
gem.add_development_dependency "rspec"
|
gem.add_development_dependency "rspec"
|
||||||
gem.add_development_dependency "rake"
|
gem.add_development_dependency "rake"
|
||||||
gem.add_development_dependency "simplecov"
|
gem.add_development_dependency "simplecov", "~> 0.15.0"
|
||||||
gem.add_development_dependency "yard"
|
gem.add_development_dependency "yard"
|
||||||
gem.add_development_dependency "rdoc"
|
gem.add_development_dependency "rdoc"
|
||||||
end
|
end
|
||||||
|
@ -74,6 +74,8 @@ describe Rscons do
|
|||||||
command_name = "#{command_prefix}#{@statics[:build_test_id]}"
|
command_name = "#{command_prefix}#{@statics[:build_test_id]}"
|
||||||
File.open("_simplecov_setup.rb", "w") do |fh|
|
File.open("_simplecov_setup.rb", "w") do |fh|
|
||||||
fh.puts <<EOF
|
fh.puts <<EOF
|
||||||
|
require "bundler"
|
||||||
|
Bundler.setup
|
||||||
require "simplecov"
|
require "simplecov"
|
||||||
class MyFormatter
|
class MyFormatter
|
||||||
def format(*args)
|
def format(*args)
|
||||||
@ -95,7 +97,7 @@ ENV["TERM"] = nil
|
|||||||
EOF
|
EOF
|
||||||
end
|
end
|
||||||
stdout, stderr, status = nil, nil, nil
|
stdout, stderr, status = nil, nil, nil
|
||||||
Bundler.with_clean_env do
|
Bundler.with_unbundled_env do
|
||||||
stdout, stderr, status = Open3.capture3(*command)
|
stdout, stderr, status = Open3.capture3(*command)
|
||||||
end
|
end
|
||||||
# Remove output lines generated as a result of the test environment
|
# Remove output lines generated as a result of the test environment
|
||||||
@ -126,7 +128,7 @@ EOF
|
|||||||
result = run_test(rsconsfile: "command.rb")
|
result = run_test(rsconsfile: "command.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(lines(result.stdout)).to eq [
|
expect(lines(result.stdout)).to eq [
|
||||||
'gcc -c -o build/simple.o -MMD -MF build/simple.mf -MT TARGET simple.c',
|
'gcc -c -o build/simple.o -MMD -MF build/simple.mf simple.c',
|
||||||
"gcc -o simple.exe build/simple.o",
|
"gcc -o simple.exe build/simple.o",
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
@ -195,7 +197,7 @@ EOF
|
|||||||
result = run_test(rsconsfile: "command.rb")
|
result = run_test(rsconsfile: "command.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(lines(result.stdout)).to eq [
|
expect(lines(result.stdout)).to eq [
|
||||||
'gcc -c -o build/simple.o -MMD -MF build/simple.mf -MT TARGET simple.c',
|
'gcc -c -o build/simple.o -MMD -MF build/simple.mf simple.c',
|
||||||
"gcc -o simple.exe build/simple.o",
|
"gcc -o simple.exe build/simple.o",
|
||||||
]
|
]
|
||||||
result = run_test(rsconsfile: "link_flag_change.rb")
|
result = run_test(rsconsfile: "link_flag_change.rb")
|
||||||
@ -250,8 +252,8 @@ EOF
|
|||||||
result = run_test(rsconsfile: "carat.rb")
|
result = run_test(rsconsfile: "carat.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
%q{gcc -c -o build_root/one.o -MMD -MF build_root/one.mf -MT TARGET -Isrc -Isrc/one -Isrc/two build_root/one.c},
|
%q{gcc -c -o build_root/one.o -MMD -MF build_root/one.mf -Isrc -Isrc/one -Isrc/two build_root/one.c},
|
||||||
%q{gcc -c -o build_root/src/two/two.o -MMD -MF build_root/src/two/two.mf -MT TARGET -Isrc -Isrc/one -Isrc/two src/two/two.c},
|
%q{gcc -c -o build_root/src/two/two.o -MMD -MF build_root/src/two/two.mf -Isrc -Isrc/one -Isrc/two src/two/two.c},
|
||||||
%Q{gcc -o build_dir.exe build_root/src/two/two.o build_root/one.o},
|
%Q{gcc -o build_dir.exe build_root/src/two/two.o build_root/one.o},
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
@ -330,9 +332,9 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
%q{gcc -c -o debug/program.o -MMD -MF debug/program.mf -MT TARGET '-DSTRING="Debug Version"' -O2 src/program.c},
|
%q{gcc -c -o debug/program.o -MMD -MF debug/program.mf '-DSTRING="Debug Version"' -O2 src/program.c},
|
||||||
%Q{gcc -o program-debug.exe debug/program.o},
|
%Q{gcc -o program-debug.exe debug/program.o},
|
||||||
%q{gcc -c -o release/program.o -MMD -MF release/program.mf -MT TARGET '-DSTRING="Release Version"' -O2 src/program.c},
|
%q{gcc -c -o release/program.o -MMD -MF release/program.mf '-DSTRING="Release Version"' -O2 src/program.c},
|
||||||
%Q{gcc -o program-release.exe release/program.o},
|
%Q{gcc -o program-release.exe release/program.o},
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
@ -342,7 +344,7 @@ EOF
|
|||||||
result = run_test(rsconsfile: "clone_all.rb")
|
result = run_test(rsconsfile: "clone_all.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(lines(result.stdout)).to eq [
|
expect(lines(result.stdout)).to eq [
|
||||||
%q{gcc -c -o build/program.o -MMD -MF build/program.mf -MT TARGET -DSTRING="Hello" -O2 src/program.c},
|
%q{gcc -c -o build/program.o -MMD -MF build/program.mf -DSTRING="Hello" -O2 src/program.c},
|
||||||
%q{post build/program.o},
|
%q{post build/program.o},
|
||||||
%Q{gcc -o program.exe build/program.o},
|
%Q{gcc -o program.exe build/program.o},
|
||||||
%q{post program.exe},
|
%q{post program.exe},
|
||||||
@ -365,8 +367,8 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
'gcc -c -o one.o -MMD -MF one.mf -MT TARGET -DONE one.c',
|
'gcc -c -o one.o -MMD -MF one.mf -DONE one.c',
|
||||||
'gcc -c -o build/two.o -MMD -MF build/two.mf -MT TARGET two.c',
|
'gcc -c -o build/two.o -MMD -MF build/two.mf two.c',
|
||||||
"gcc -o two_sources.exe one.o build/two.o",
|
"gcc -o two_sources.exe one.o build/two.o",
|
||||||
]
|
]
|
||||||
expect(File.exists?("two_sources.exe")).to be_truthy
|
expect(File.exists?("two_sources.exe")).to be_truthy
|
||||||
@ -378,10 +380,10 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
'gcc -c -o build/one.o -MMD -MF build/one.mf -MT TARGET -Dmake_lib one.c',
|
'gcc -c -o build/one.o -MMD -MF build/one.mf -Dmake_lib one.c',
|
||||||
'gcc -c -o build/two.o -MMD -MF build/two.mf -MT TARGET -Dmake_lib two.c',
|
'gcc -c -o build/two.o -MMD -MF build/two.mf -Dmake_lib two.c',
|
||||||
'ar rcs lib.a build/one.o build/two.o',
|
'ar rcs lib.a build/one.o build/two.o',
|
||||||
'gcc -c -o build/three.o -MMD -MF build/three.mf -MT TARGET three.c',
|
'gcc -c -o build/three.o -MMD -MF build/three.mf three.c',
|
||||||
"gcc -o library.exe lib.a build/three.o",
|
"gcc -o library.exe lib.a build/three.o",
|
||||||
]
|
]
|
||||||
expect(File.exists?("library.exe")).to be_truthy
|
expect(File.exists?("library.exe")).to be_truthy
|
||||||
@ -393,8 +395,8 @@ EOF
|
|||||||
result = run_test(rsconsfile: "build_hooks.rb")
|
result = run_test(rsconsfile: "build_hooks.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
'gcc -c -o build_one/one.o -MMD -MF build_one/one.mf -MT TARGET -Isrc/one -Isrc/two -O1 src/one/one.c',
|
'gcc -c -o build_one/one.o -MMD -MF build_one/one.mf -Isrc/one -Isrc/two -O1 src/one/one.c',
|
||||||
'gcc -c -o build_two/two.o -MMD -MF build_two/two.mf -MT TARGET -Isrc/one -Isrc/two -O2 src/two/two.c',
|
'gcc -c -o build_two/two.o -MMD -MF build_two/two.mf -Isrc/one -Isrc/two -O2 src/two/two.c',
|
||||||
'gcc -o build_hook.exe build_one/one.o build_two/two.o',
|
'gcc -o build_hook.exe build_one/one.o build_two/two.o',
|
||||||
]
|
]
|
||||||
expect(`./build_hook.exe`).to eq "Hello from two()\n"
|
expect(`./build_hook.exe`).to eq "Hello from two()\n"
|
||||||
@ -415,14 +417,14 @@ EOF
|
|||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(lines(result.stdout)).to eq(["LD simple.exe"])
|
expect(lines(result.stdout)).to eq(["LD simple.exe"])
|
||||||
|
|
||||||
File.unlink("program.ld")
|
|
||||||
result = run_test(rsconsfile: "user_dependencies.rb")
|
|
||||||
expect(result.stderr).to eq ""
|
|
||||||
expect(lines(result.stdout)).to eq ["LD simple.exe"]
|
|
||||||
|
|
||||||
result = run_test(rsconsfile: "user_dependencies.rb")
|
result = run_test(rsconsfile: "user_dependencies.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(result.stdout).to eq ""
|
expect(result.stdout).to eq ""
|
||||||
|
|
||||||
|
File.unlink("program.ld")
|
||||||
|
result = run_test(rsconsfile: "user_dependencies.rb")
|
||||||
|
expect(result.stderr).to match /User dependency program\.ld of target simple\.exe is invalid/
|
||||||
|
expect(result.status).to_not eq 0
|
||||||
end
|
end
|
||||||
|
|
||||||
unless ENV["omit_gdc_tests"]
|
unless ENV["omit_gdc_tests"]
|
||||||
@ -431,8 +433,8 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
slines = lines(result.stdout)
|
slines = lines(result.stdout)
|
||||||
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf -MT TARGET main.d")
|
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf main.d")
|
||||||
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf -MT TARGET mod.d")
|
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf mod.d")
|
||||||
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
||||||
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 42!"
|
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 42!"
|
||||||
end
|
end
|
||||||
@ -442,8 +444,8 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
slines = lines(result.stdout)
|
slines = lines(result.stdout)
|
||||||
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf -MT TARGET main.d")
|
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf main.d")
|
||||||
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf -MT TARGET mod.d")
|
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf mod.d")
|
||||||
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
||||||
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 42!"
|
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 42!"
|
||||||
fcontents = File.read("mod.d", mode: "rb").sub("42", "33")
|
fcontents = File.read("mod.d", mode: "rb").sub("42", "33")
|
||||||
@ -451,8 +453,8 @@ EOF
|
|||||||
result = run_test
|
result = run_test
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
slines = lines(result.stdout)
|
slines = lines(result.stdout)
|
||||||
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf -MT TARGET main.d")
|
expect(slines).to include("gdc -c -o build/main.o -MMD -MF build/main.mf main.d")
|
||||||
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf -MT TARGET mod.d")
|
expect(slines).to include("gdc -c -o build/mod.o -MMD -MF build/mod.mf mod.d")
|
||||||
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
expect(slines.last).to eq("gdc -o hello-d.exe build/main.o build/mod.o")
|
||||||
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 33!"
|
expect(`./hello-d.exe`.rstrip).to eq "Hello from D, value is 33!"
|
||||||
end
|
end
|
||||||
@ -894,8 +896,8 @@ EOF
|
|||||||
result = run_test(rsconsfile: "backward_compatible_build_hooks.rb")
|
result = run_test(rsconsfile: "backward_compatible_build_hooks.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(Set[*lines(result.stdout)]).to eq Set[
|
expect(Set[*lines(result.stdout)]).to eq Set[
|
||||||
'gcc -c -o one.o -MMD -MF one.mf -MT TARGET -Isrc -Isrc/one -Isrc/two -O1 src/two/two.c',
|
'gcc -c -o one.o -MMD -MF one.mf -Isrc -Isrc/one -Isrc/two -O1 src/two/two.c',
|
||||||
'gcc -c -o two.o -MMD -MF two.mf -MT TARGET -Isrc -Isrc/one -Isrc/two -O2 src/two/two.c'
|
'gcc -c -o two.o -MMD -MF two.mf -Isrc -Isrc/one -Isrc/two -O2 src/two/two.c'
|
||||||
]
|
]
|
||||||
expect(File.exists?('one.o')).to be_truthy
|
expect(File.exists?('one.o')).to be_truthy
|
||||||
expect(File.exists?('two.o')).to be_truthy
|
expect(File.exists?('two.o')).to be_truthy
|
||||||
@ -1354,7 +1356,7 @@ EOF
|
|||||||
result = run_test(rsconsfile: "override_depfilesuffix.rb")
|
result = run_test(rsconsfile: "override_depfilesuffix.rb")
|
||||||
expect(result.stderr).to eq ""
|
expect(result.stderr).to eq ""
|
||||||
expect(lines(result.stdout)).to eq [
|
expect(lines(result.stdout)).to eq [
|
||||||
"gcc -c -o simple.o -MMD -MF simple.deppy -MT TARGET simple.c",
|
"gcc -c -o simple.o -MMD -MF simple.deppy simple.c",
|
||||||
]
|
]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -26,13 +26,13 @@ module Rscons
|
|||||||
cache = build_from(_cache)
|
cache = build_from(_cache)
|
||||||
expect(File).to receive(:exists?).with("one").and_return(true)
|
expect(File).to receive(:exists?).with("one").and_return(true)
|
||||||
expect(File).to receive(:exists?).with("one/two").and_return(false)
|
expect(File).to receive(:exists?).with("one/two").and_return(false)
|
||||||
expect(FileUtils).to receive(:mkdir).with("one/two")
|
expect(FileUtils).to receive(:mkdir_p).with("one/two")
|
||||||
expect(File).to receive(:exists?).with("one/two/three").and_return(false)
|
expect(File).to receive(:exists?).with("one/two/three").and_return(false)
|
||||||
expect(FileUtils).to receive(:mkdir).with("one/two/three")
|
expect(FileUtils).to receive(:mkdir_p).with("one/two/three")
|
||||||
expect(File).to receive(:exists?).with("one").and_return(true)
|
expect(File).to receive(:exists?).with("one").and_return(true)
|
||||||
expect(File).to receive(:exists?).with("one/two").and_return(true)
|
expect(File).to receive(:exists?).with("one/two").and_return(true)
|
||||||
expect(File).to receive(:exists?).with("one/two/four").and_return(false)
|
expect(File).to receive(:exists?).with("one/two/four").and_return(false)
|
||||||
expect(FileUtils).to receive(:mkdir).with("one/two/four")
|
expect(FileUtils).to receive(:mkdir_p).with("one/two/four")
|
||||||
cache.mkdir_p("one/two/three")
|
cache.mkdir_p("one/two/three")
|
||||||
cache.mkdir_p("one\\two\\four")
|
cache.mkdir_p("one\\two\\four")
|
||||||
expect(cache.directories).to eq ["one/two", "one/two/three", "one/two/four"]
|
expect(cache.directories).to eq ["one/two", "one/two/three", "one/two/four"]
|
||||||
@ -43,7 +43,7 @@ module Rscons
|
|||||||
cache = build_from(_cache)
|
cache = build_from(_cache)
|
||||||
expect(File).to receive(:exists?).with("/one").and_return(true)
|
expect(File).to receive(:exists?).with("/one").and_return(true)
|
||||||
expect(File).to receive(:exists?).with("/one/two").and_return(false)
|
expect(File).to receive(:exists?).with("/one/two").and_return(false)
|
||||||
expect(FileUtils).to receive(:mkdir).with("/one/two")
|
expect(FileUtils).to receive(:mkdir_p).with("/one/two")
|
||||||
cache.mkdir_p("/one/two")
|
cache.mkdir_p("/one/two")
|
||||||
expect(cache.directories).to eq ["/one/two"]
|
expect(cache.directories).to eq ["/one/two"]
|
||||||
end
|
end
|
||||||
|
@ -184,7 +184,7 @@ module Rscons
|
|||||||
env["foo"] = {}
|
env["foo"] = {}
|
||||||
expect(env.expand_varref(["-p${path}", "${flags}"])).to eq ["-pdir1", "-pdir2", "-x", "-y", "-z"]
|
expect(env.expand_varref(["-p${path}", "${flags}"])).to eq ["-pdir1", "-pdir2", "-x", "-y", "-z"]
|
||||||
expect(env.expand_varref("foo")).to eq "foo"
|
expect(env.expand_varref("foo")).to eq "foo"
|
||||||
expect {env.expand_varref("${foo}")}.to raise_error /Unknown.varref.type/
|
expect {env.expand_varref("${foo}")}.to raise_error /Unknown.construction.variable.type/
|
||||||
expect(env.expand_varref("${specialflag}")).to eq "-z"
|
expect(env.expand_varref("${specialflag}")).to eq "-z"
|
||||||
expect(env.expand_varref("${path}")).to eq ["dir1", "dir2"]
|
expect(env.expand_varref("${path}")).to eq ["dir1", "dir2"]
|
||||||
end
|
end
|
||||||
@ -329,7 +329,7 @@ module Rscons
|
|||||||
expect(File).to receive(:read).with('makefile').and_return(<<EOS)
|
expect(File).to receive(:read).with('makefile').and_return(<<EOS)
|
||||||
module.o: source.cc
|
module.o: source.cc
|
||||||
EOS
|
EOS
|
||||||
expect(Environment.parse_makefile_deps('makefile', 'module.o')).to eq ['source.cc']
|
expect(Environment.parse_makefile_deps('makefile')).to eq ['source.cc']
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'handles dependencies split across many lines' do
|
it 'handles dependencies split across many lines' do
|
||||||
@ -338,7 +338,7 @@ module.o: module.c \\
|
|||||||
module.h \\
|
module.h \\
|
||||||
other.h
|
other.h
|
||||||
EOS
|
EOS
|
||||||
expect(Environment.parse_makefile_deps('makefile', 'module.o')).to eq [
|
expect(Environment.parse_makefile_deps('makefile')).to eq [
|
||||||
'module.c', 'module.h', 'other.h']
|
'module.c', 'module.h', 'other.h']
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -159,14 +159,33 @@ module Rscons
|
|||||||
it "calls a lambda with the given lambda arguments" do
|
it "calls a lambda with the given lambda arguments" do
|
||||||
expect(v.expand_varref("${lambda}", [v: "fez"])).to eq("fez--12")
|
expect(v.expand_varref("${lambda}", [v: "fez"])).to eq("fez--12")
|
||||||
end
|
end
|
||||||
|
it "returns true when passed true" do
|
||||||
|
expect(v.expand_varref(true, :lambda_args)).to eq(true)
|
||||||
|
end
|
||||||
|
it "returns false when passed false" do
|
||||||
|
expect(v.expand_varref(false, :lambda_args)).to eq(false)
|
||||||
|
end
|
||||||
it "raises an error when given an invalid argument" do
|
it "raises an error when given an invalid argument" do
|
||||||
expect { v.expand_varref({a: :b}, :lambda_args) }.to raise_error /Unknown varref type: Hash/
|
expect { v.expand_varref({a: :b}, :lambda_args) }.to raise_error /Unknown construction variable type: Hash/
|
||||||
end
|
end
|
||||||
it "raises an error when an expanded variable is an unexpected type" do
|
it "raises an error when an expanded variable is an unexpected type" do
|
||||||
expect(v).to receive(:[]).at_least(1).times.with("bad").and_return("bad_val")
|
expect(v).to receive(:get_var).at_least(1).times.with("bad").and_return("bad_val")
|
||||||
expect(v).to receive(:expand_varref).with("bad_val", :lambda_args).and_return({a: :b})
|
expect(v).to receive(:expand_varref).with("bad_val", :lambda_args).and_return({a: :b})
|
||||||
expect(v).to receive(:expand_varref).and_call_original
|
expect(v).to receive(:expand_varref).and_call_original
|
||||||
expect { v.expand_varref("${bad}", :lambda_args) }.to raise_error /I do not know how to expand a variable reference to a Hash/
|
expect { v.expand_varref("${bad}", :lambda_args) }.to raise_error /Unknown construction variable type: Hash/
|
||||||
|
end
|
||||||
|
it "expands symbols within a string outside an array" do
|
||||||
|
v['var'] = :a_symbol
|
||||||
|
expect(v.expand_varref("this is ${var}", :lambda_args)).to eq "this is a_symbol"
|
||||||
|
end
|
||||||
|
it "expands booleans within a string outside an array" do
|
||||||
|
v['var'] = false
|
||||||
|
expect(v.expand_varref("this is ${var}", :lambda_args)).to eq "this is false"
|
||||||
|
end
|
||||||
|
it "expands symbols and booleans in an array" do
|
||||||
|
v['var'] = [:a_symbol, false]
|
||||||
|
expanded = v.expand_varref("this is ${var}", :lambda_args)
|
||||||
|
expect(expanded).to eq(["this is a_symbol", "this is false"])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user