mirror of
https://github.com/zhigang1992/RubyMotion.git
synced 2026-04-05 17:40:51 +08:00
671 lines
23 KiB
Ruby
671 lines
23 KiB
Ruby
# Copyright (c) 2012, HipByte SPRL and contributors
|
|
# All rights reserved.
|
|
#
|
|
# Redistribution and use in source and binary forms, with or without
|
|
# modification, are permitted provided that the following conditions are met:
|
|
#
|
|
# 1. Redistributions of source code must retain the above copyright notice, this
|
|
# list of conditions and the following disclaimer.
|
|
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
# this list of conditions and the following disclaimer in the documentation
|
|
# and/or other materials provided with the distribution.
|
|
#
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
# ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
require 'thread'
|
|
|
|
module Motion; module Project;
|
|
class Builder
|
|
include Rake::DSL if Object.const_defined?(:Rake) && Rake.const_defined?(:DSL)
|
|
|
|
def build(config, platform, opts)
|
|
datadir = config.datadir
|
|
archs = config.archs[platform]
|
|
|
|
static_library = opts.delete(:static)
|
|
|
|
ruby = File.join(config.bindir, 'ruby')
|
|
llc = File.join(config.bindir, 'llc')
|
|
|
|
if config.spec_mode and (config.spec_files - config.spec_core_files).empty?
|
|
App.fail "No spec files in `#{config.specs_dir}'"
|
|
end
|
|
|
|
config.resources_dirs.flatten!
|
|
|
|
# Locate SDK and compilers.
|
|
sdk = config.sdk(platform)
|
|
cc = config.locate_compiler(platform, 'llvm-gcc')
|
|
cxx = config.locate_compiler(platform, 'clang++')
|
|
|
|
build_dir = File.join(config.versionized_build_dir(platform))
|
|
App.info 'Build', build_dir
|
|
|
|
# Prepare the list of BridgeSupport files needed.
|
|
bs_files = config.bridgesupport_files
|
|
|
|
# Build vendor libraries.
|
|
vendor_libs = []
|
|
config.vendor_projects.each do |vendor_project|
|
|
vendor_project.build(platform)
|
|
vendor_libs.concat(vendor_project.libs)
|
|
bs_files.concat(vendor_project.bs_files)
|
|
end
|
|
|
|
# Validate common build directory.
|
|
if !File.directory?(Builder.common_build_dir) or !File.writable?(Builder.common_build_dir)
|
|
$stderr.puts "Cannot write into the `#{Builder.common_build_dir}' directory, please remove or check permissions and try again."
|
|
exit 1
|
|
end
|
|
|
|
# Prepare embedded frameworks BridgeSupport files (OSX-only).
|
|
embedded_frameworks = (config.respond_to?(:embedded_frameworks) ? config.embedded_frameworks.map { |x| File.expand_path(x) } : [])
|
|
unless embedded_frameworks.empty?
|
|
embedded_frameworks.each do |path|
|
|
headers = Dir.glob(File.join(path, 'Headers/**/*.h'))
|
|
bs_file = File.join(Builder.common_build_dir, File.expand_path(path) + '.bridgesupport')
|
|
if !File.exist?(bs_file) or File.mtime(path) > File.mtime(bs_file)
|
|
FileUtils.mkdir_p(File.dirname(bs_file))
|
|
config.gen_bridge_metadata(platform, headers, bs_file, '', [])
|
|
end
|
|
bs_files << bs_file
|
|
end
|
|
end
|
|
|
|
# Build object files.
|
|
objs_build_dir = File.join(build_dir, 'objs')
|
|
FileUtils.mkdir_p(objs_build_dir)
|
|
any_obj_file_built = false
|
|
project_files = Dir.glob("**/*.rb").map{ |x| File.expand_path(x) }
|
|
is_default_archs = (archs == config.default_archs[platform])
|
|
rubyc_bs_flags = bs_files.map { |x| "--uses-bs \"" + x + "\" " }.join(' ')
|
|
|
|
build_file = Proc.new do |files_build_dir, path|
|
|
rpath = path
|
|
path = File.expand_path(path)
|
|
if is_default_archs && !project_files.include?(path)
|
|
files_build_dir = File.expand_path(File.join(Builder.common_build_dir, files_build_dir))
|
|
end
|
|
obj = File.join(files_build_dir, "#{path}.o")
|
|
should_rebuild = (!File.exist?(obj) \
|
|
or File.mtime(path) > File.mtime(obj) \
|
|
or File.mtime(ruby) > File.mtime(obj))
|
|
|
|
# Generate or retrieve init function.
|
|
init_func = should_rebuild ? "MREP_#{`/usr/bin/uuidgen`.strip.gsub('-', '')}" : `#{config.locate_binary('nm')} \"#{obj}\"`.scan(/T\s+_(MREP_.*)/)[0][0]
|
|
|
|
if should_rebuild
|
|
App.info 'Compile', rpath
|
|
FileUtils.mkdir_p(File.dirname(obj))
|
|
arch_objs = []
|
|
archs.each do |arch|
|
|
# Locate arch kernel.
|
|
kernel = File.join(datadir, platform, "kernel-#{arch}.bc")
|
|
raise "Can't locate kernel file" unless File.exist?(kernel)
|
|
|
|
# LLVM bitcode.
|
|
bc = File.join(files_build_dir, "#{path}.#{arch}.bc")
|
|
arch_cmd = (arch =~ /^arm/) ? "/usr/bin/arch -arch i386" : "/usr/bin/arch -arch #{arch}"
|
|
sh "/usr/bin/env VM_KERNEL_PATH=\"#{kernel}\" VM_OPT_LEVEL=\"#{config.opt_level}\" #{arch_cmd} #{ruby} #{rubyc_bs_flags} --emit-llvm \"#{bc}\" #{init_func} \"#{path}\""
|
|
|
|
# Assembly.
|
|
asm = File.join(files_build_dir, "#{path}.#{arch}.s")
|
|
llc_arch = case arch
|
|
when 'i386'; 'x86'
|
|
when 'x86_64'; 'x86-64'
|
|
when /^arm/; 'arm'
|
|
else; arch
|
|
end
|
|
sh "#{llc} \"#{bc}\" -o=\"#{asm}\" -march=#{llc_arch} -relocation-model=pic -disable-fp-elim -jit-enable-eh -disable-cfi"
|
|
|
|
# Object.
|
|
arch_obj = File.join(files_build_dir, "#{path}.#{arch}.o")
|
|
sh "#{cc} -fexceptions -c -arch #{arch} \"#{asm}\" -o \"#{arch_obj}\""
|
|
|
|
[bc, asm].each { |x| File.unlink(x) } unless ENV['keep_temps']
|
|
arch_objs << arch_obj
|
|
end
|
|
|
|
# Assemble fat binary.
|
|
arch_objs_list = arch_objs.map { |x| "\"#{x}\"" }.join(' ')
|
|
sh "/usr/bin/lipo -create #{arch_objs_list} -output \"#{obj}\""
|
|
end
|
|
|
|
any_obj_file_built = true
|
|
[obj, init_func]
|
|
end
|
|
|
|
# Resolve file dependencies.
|
|
if config.detect_dependencies == true
|
|
config.dependencies = Dependency.new(config.files - config.exclude_from_detect_dependencies, config.dependencies).run
|
|
end
|
|
|
|
parallel = ParallelBuilder.new(objs_build_dir, build_file)
|
|
parallel.files = config.ordered_build_files
|
|
parallel.run
|
|
objs = parallel.objects
|
|
|
|
FileUtils.touch(objs_build_dir) if any_obj_file_built
|
|
|
|
app_objs = objs
|
|
spec_objs = []
|
|
if config.spec_mode
|
|
# Build spec files too, but sequentially.
|
|
parallel = ParallelBuilder.new(objs_build_dir, build_file)
|
|
parallel.files = config.spec_files
|
|
parallel.run
|
|
spec_objs = parallel.objects
|
|
objs += spec_objs
|
|
end
|
|
|
|
# Generate init file.
|
|
init_txt = <<EOS
|
|
extern "C" {
|
|
void ruby_sysinit(int *, char ***);
|
|
void ruby_init(void);
|
|
void ruby_init_loadpath(void);
|
|
void ruby_script(const char *);
|
|
void ruby_set_argv(int, char **);
|
|
void rb_vm_init_compiler(void);
|
|
void rb_vm_init_jit(void);
|
|
void rb_vm_aot_feature_provide(const char *, void *);
|
|
void *rb_vm_top_self(void);
|
|
void rb_define_global_const(const char *, void *);
|
|
void rb_rb2oc_exc_handler(void);
|
|
void rb_exit(int);
|
|
EOS
|
|
app_objs.each do |_, init_func|
|
|
init_txt << "void #{init_func}(void *, void *);\n"
|
|
end
|
|
init_txt << <<EOS
|
|
}
|
|
|
|
extern "C"
|
|
void
|
|
RubyMotionInit(int argc, char **argv)
|
|
{
|
|
static bool initialized = false;
|
|
if (!initialized) {
|
|
ruby_init();
|
|
ruby_init_loadpath();
|
|
if (argc > 0) {
|
|
const char *progname = argv[0];
|
|
ruby_script(progname);
|
|
}
|
|
#if !__LP64__
|
|
try {
|
|
#endif
|
|
void *self = rb_vm_top_self();
|
|
EOS
|
|
init_txt << config.define_global_env_txt
|
|
app_objs.each do |_, init_func|
|
|
init_txt << "#{init_func}(self, 0);\n"
|
|
end
|
|
init_txt << <<EOS
|
|
#if !__LP64__
|
|
}
|
|
catch (...) {
|
|
rb_rb2oc_exc_handler();
|
|
}
|
|
#endif
|
|
initialized = true;
|
|
}
|
|
}
|
|
EOS
|
|
|
|
# Compile init file.
|
|
init = File.join(objs_build_dir, 'init.mm')
|
|
init_o = File.join(objs_build_dir, 'init.o')
|
|
if !(File.exist?(init) and File.exist?(init_o) and File.read(init) == init_txt)
|
|
File.open(init, 'w') { |io| io.write(init_txt) }
|
|
sh "#{cxx} \"#{init}\" #{config.cflags(platform, true)} -c -o \"#{init_o}\""
|
|
end
|
|
|
|
if static_library
|
|
# Create a static archive with all object files + the runtime.
|
|
lib = File.join(config.versionized_build_dir(platform), config.name + '.a')
|
|
App.info 'Create', lib
|
|
libmacruby = File.join(datadir, platform, 'libmacruby-static.a')
|
|
objs_list = objs.map { |path, _| path }.unshift(init_o, *config.frameworks_stubs_objects(platform)).map { |x| "\"#{x}\"" }.join(' ')
|
|
sh "/usr/bin/libtool -static \"#{libmacruby}\" #{objs_list} -o \"#{lib}\""
|
|
return lib
|
|
end
|
|
|
|
# Generate main file.
|
|
main_txt = config.main_cpp_file_txt(spec_objs)
|
|
|
|
# Compile main file.
|
|
main = File.join(objs_build_dir, 'main.mm')
|
|
main_o = File.join(objs_build_dir, 'main.o')
|
|
if !(File.exist?(main) and File.exist?(main_o) and File.read(main) == main_txt)
|
|
File.open(main, 'w') { |io| io.write(main_txt) }
|
|
sh "#{cxx} \"#{main}\" #{config.cflags(platform, true)} -c -o \"#{main_o}\""
|
|
end
|
|
|
|
# Prepare bundle.
|
|
bundle_path = config.app_bundle(platform)
|
|
unless File.exist?(bundle_path)
|
|
App.info 'Create', bundle_path
|
|
FileUtils.mkdir_p(bundle_path)
|
|
end
|
|
|
|
# Link executable.
|
|
main_exec = config.app_bundle_executable(platform)
|
|
unless File.exist?(File.dirname(main_exec))
|
|
App.info 'Create', File.dirname(main_exec)
|
|
FileUtils.mkdir_p(File.dirname(main_exec))
|
|
end
|
|
main_exec_created = false
|
|
if !File.exist?(main_exec) \
|
|
or File.mtime(config.project_file) > File.mtime(main_exec) \
|
|
or objs.any? { |path, _| File.mtime(path) > File.mtime(main_exec) } \
|
|
or File.mtime(main_o) > File.mtime(main_exec) \
|
|
or vendor_libs.any? { |lib| File.mtime(lib) > File.mtime(main_exec) } \
|
|
or File.mtime(File.join(datadir, platform, 'libmacruby-static.a')) > File.mtime(main_exec)
|
|
App.info 'Link', main_exec
|
|
objs_list = objs.map { |path, _| path }.unshift(init_o, main_o, *config.frameworks_stubs_objects(platform)).map { |x| "\"#{x}\"" }.join(' ')
|
|
framework_search_paths = (config.framework_search_paths + embedded_frameworks.map { |x| File.dirname(x) }).uniq.map { |x| "-F#{File.expand_path(x)}" }.join(' ')
|
|
frameworks = (config.frameworks_dependencies + embedded_frameworks.map { |x| File.basename(x, '.framework') }).map { |x| "-framework #{x}" }.join(' ')
|
|
weak_frameworks = config.weak_frameworks.map { |x| "-weak_framework #{x}" }.join(' ')
|
|
vendor_libs = config.vendor_projects.inject([]) do |libs, vendor_project|
|
|
libs << vendor_project.libs.map { |x|
|
|
(vendor_project.opts[:force_load] ? '-force_load ' : '-ObjC ') + "\"#{x}\""
|
|
}
|
|
end.join(' ')
|
|
linker_option = begin
|
|
m = config.deployment_target.match(/(\d+)/)
|
|
if m[0].to_i < 7
|
|
"-stdlib=libstdc++"
|
|
end
|
|
end || ""
|
|
sh "#{cxx} -o \"#{main_exec}\" #{objs_list} #{config.ldflags(platform)} -L#{File.join(datadir, platform)} -lmacruby-static -lobjc -licucore #{linker_option} #{framework_search_paths} #{frameworks} #{weak_frameworks} #{config.libs.join(' ')} #{vendor_libs}"
|
|
main_exec_created = true
|
|
|
|
# Change the install name of embedded frameworks.
|
|
embedded_frameworks.each do |path|
|
|
res = `/usr/bin/otool -L \"#{main_exec}\"`.scan(/(.*#{File.basename(path)}.*)\s\(/)
|
|
if res and res[0] and res[0][0]
|
|
old_path = res[0][0].strip
|
|
new_path = "@executable_path/../Frameworks/" + old_path.scan(/#{File.basename(path)}.*/)[0]
|
|
sh "/usr/bin/install_name_tool -change \"#{old_path}\" \"#{new_path}\" \"#{main_exec}\""
|
|
else
|
|
App.warn "Cannot locate and fix install name path of embedded framework `#{path}' in executable `#{main_exec}', application might not start"
|
|
end
|
|
end
|
|
end
|
|
|
|
# Create bundle/Info.plist.
|
|
bundle_info_plist = File.join(bundle_path, 'Info.plist')
|
|
if !File.exist?(bundle_info_plist) or File.mtime(config.project_file) > File.mtime(bundle_info_plist)
|
|
App.info 'Create', bundle_info_plist
|
|
File.open(bundle_info_plist, 'w') { |io| io.write(config.info_plist_data) }
|
|
sh "/usr/bin/plutil -convert binary1 \"#{bundle_info_plist}\""
|
|
end
|
|
|
|
# Create bundle/PkgInfo.
|
|
bundle_pkginfo = File.join(bundle_path, 'PkgInfo')
|
|
if !File.exist?(bundle_pkginfo) or File.mtime(config.project_file) > File.mtime(bundle_pkginfo)
|
|
App.info 'Create', bundle_pkginfo
|
|
File.open(bundle_pkginfo, 'w') { |io| io.write(config.pkginfo_data) }
|
|
end
|
|
|
|
# Compile IB resources.
|
|
config.resources_dirs.each do |dir|
|
|
if File.exist?(dir)
|
|
ib_resources = []
|
|
ib_resources.concat((Dir.glob(File.join(dir, '**', '*.xib')) + Dir.glob(File.join(dir, '*.lproj', '*.xib'))).map { |xib| [xib, xib.sub(/\.xib$/, '.nib')] })
|
|
ib_resources.concat(Dir.glob(File.join(dir, '**', '*.storyboard')).map { |storyboard| [storyboard, storyboard.sub(/\.storyboard$/, '.storyboardc')] })
|
|
ib_resources.each do |src, dest|
|
|
if !File.exist?(dest) or File.mtime(src) > File.mtime(dest)
|
|
App.info 'Compile', src
|
|
sh "/usr/bin/ibtool --compile \"#{dest}\" \"#{src}\""
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
# Compile CoreData Model resources and SpriteKit atlas files.
|
|
config.resources_dirs.each do |dir|
|
|
if File.exist?(dir)
|
|
Dir.glob(File.join(dir, '*.xcdatamodeld')).each do |model|
|
|
momd = model.sub(/\.xcdatamodeld$/, '.momd')
|
|
if !File.exist?(momd) or File.mtime(model) > File.mtime(momd)
|
|
App.info 'Compile', model
|
|
model = File.expand_path(model) # momc wants absolute paths.
|
|
momd = File.expand_path(momd)
|
|
sh "\"#{App.config.xcode_dir}/usr/bin/momc\" \"#{model}\" \"#{momd}\""
|
|
end
|
|
end
|
|
if cmd = config.spritekit_texture_atlas_compiler
|
|
Dir.glob(File.join(dir, '*.atlas')).each do |atlas|
|
|
if File.directory?(atlas)
|
|
App.info 'Compile', atlas
|
|
sh "\"#{cmd}\" \"#{atlas}\" \"#{bundle_path}\""
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
# Copy embedded frameworks.
|
|
unless embedded_frameworks.empty?
|
|
app_frameworks = File.join(config.app_bundle(platform), 'Frameworks')
|
|
FileUtils.mkdir_p(app_frameworks)
|
|
embedded_frameworks.each do |src_path|
|
|
dest_path = File.join(app_frameworks, File.basename(src_path))
|
|
if !File.exist?(dest_path) or File.mtime(src_path) > File.mtime(dest_path)
|
|
App.info 'Copy', src_path
|
|
FileUtils.cp_r(src_path, dest_path)
|
|
end
|
|
end
|
|
end
|
|
|
|
# Copy resources, handle subdirectories.
|
|
app_resources_dir = config.app_resources_dir(platform)
|
|
FileUtils.mkdir_p(app_resources_dir)
|
|
reserved_app_bundle_files = [
|
|
'_CodeSignature/CodeResources', 'CodeResources', 'embedded.mobileprovision',
|
|
'Info.plist', 'PkgInfo', 'ResourceRules.plist',
|
|
config.name
|
|
]
|
|
resources_paths = []
|
|
config.resources_dirs.each do |dir|
|
|
if File.exist?(dir)
|
|
resources_paths << Dir.chdir(dir) do
|
|
Dir.glob('**{,/*/**}/*').reject { |x| ['.xib', '.storyboard', '.xcdatamodeld', '.lproj', '.atlas'].include?(File.extname(x)) }.map { |file| File.join(dir, file) }
|
|
end
|
|
end
|
|
end
|
|
resources_paths.flatten!
|
|
resources_paths.each do |res_path|
|
|
res = path_on_resources_dirs(config.resources_dirs, res_path)
|
|
if reserved_app_bundle_files.include?(res)
|
|
App.fail "Cannot use `#{res_path}' as a resource file because it's a reserved application bundle file"
|
|
end
|
|
dest_path = File.join(app_resources_dir, res)
|
|
if !File.exist?(dest_path) or File.mtime(res_path) > File.mtime(dest_path)
|
|
FileUtils.mkdir_p(File.dirname(dest_path))
|
|
App.info 'Copy', res_path
|
|
FileUtils.cp_r(res_path, dest_path)
|
|
end
|
|
end
|
|
|
|
# Delete old resource files.
|
|
resources_files = resources_paths.map { |x| path_on_resources_dirs(config.resources_dirs, x) }
|
|
Dir.chdir(app_resources_dir) do
|
|
Dir.glob('*').each do |bundle_res|
|
|
bundle_res = convert_filesystem_encoding(bundle_res)
|
|
next if File.directory?(bundle_res)
|
|
next if reserved_app_bundle_files.include?(bundle_res)
|
|
next if resources_files.include?(bundle_res)
|
|
App.warn "File `#{bundle_res}' found in app bundle but not in resource directories, removing"
|
|
FileUtils.rm_rf(bundle_res)
|
|
end
|
|
end
|
|
|
|
# Generate dSYM.
|
|
dsym_path = config.app_bundle_dsym(platform)
|
|
if !File.exist?(dsym_path) or File.mtime(main_exec) > File.mtime(dsym_path)
|
|
App.info "Create", dsym_path
|
|
sh "/usr/bin/dsymutil \"#{main_exec}\" -o \"#{dsym_path}\""
|
|
end
|
|
|
|
# Strip all symbols. Only in distribution mode.
|
|
if main_exec_created and config.distribution_mode
|
|
App.info "Strip", main_exec
|
|
sh "#{config.locate_binary('strip')} #{config.strip_args} \"#{main_exec}\""
|
|
end
|
|
end
|
|
|
|
def path_on_resources_dirs(dirs, path)
|
|
dir = dirs.each do |dir|
|
|
break dir if path =~ /^#{dir}/
|
|
end
|
|
path = path.sub(/^#{dir}\/*/, '') if dir
|
|
path
|
|
end
|
|
|
|
def convert_filesystem_encoding(string)
|
|
begin
|
|
string.encode("UTF-8", "UTF8-MAC")
|
|
rescue
|
|
# for Ruby 1.8
|
|
require 'iconv'
|
|
Iconv.conv("UTF-8", "UTF8-MAC", string)
|
|
end
|
|
end
|
|
|
|
class << self
|
|
def common_build_dir
|
|
dir = File.expand_path("~/Library/RubyMotion/build")
|
|
unless File.exist?(dir)
|
|
begin
|
|
FileUtils.mkdir_p dir
|
|
rescue
|
|
end
|
|
end
|
|
dir
|
|
end
|
|
end
|
|
end
|
|
|
|
class ParallelBuilder
|
|
attr_accessor :files
|
|
|
|
def initialize(objs_build_dir, builder)
|
|
@builders_count = begin
|
|
if jobs = ENV['jobs']
|
|
jobs.to_i
|
|
else
|
|
`/usr/sbin/sysctl -n machdep.cpu.thread_count`.strip.to_i
|
|
end
|
|
end
|
|
@builders_count = 1 if @builders_count < 1
|
|
|
|
@builders = []
|
|
@builders_count.times do
|
|
queue = []
|
|
th = Thread.new do
|
|
sleep
|
|
objs = []
|
|
while path = queue.shift
|
|
objs << builder.call(objs_build_dir, path)
|
|
end
|
|
queue.concat(objs)
|
|
end
|
|
@builders << [queue, th]
|
|
end
|
|
end
|
|
|
|
def run
|
|
builder_i = 0
|
|
@files.each do |path|
|
|
@builders[builder_i][0] << path
|
|
builder_i += 1
|
|
builder_i = 0 if builder_i == @builders_count
|
|
end
|
|
|
|
# Start build.
|
|
@builders.each do |queue, th|
|
|
sleep 0.01 while th.status != 'sleep'
|
|
th.wakeup
|
|
end
|
|
@builders.each { |queue, th| th.join }
|
|
@builders
|
|
end
|
|
|
|
def objects
|
|
objs = []
|
|
builder_i = 0
|
|
@files.each do |path|
|
|
objs << @builders[builder_i][0].shift
|
|
builder_i += 1
|
|
builder_i = 0 if builder_i == @builders_count
|
|
end
|
|
objs
|
|
end
|
|
end
|
|
|
|
class Dependency
|
|
begin
|
|
require 'ripper'
|
|
rescue LoadError
|
|
$:.unshift(File.expand_path(File.join(File.dirname(__FILE__), '../../ripper18')))
|
|
require 'ripper'
|
|
end
|
|
|
|
@file_paths = []
|
|
|
|
def initialize(paths, dependencies)
|
|
@file_paths = paths.flatten.sort
|
|
@dependencies = dependencies
|
|
end
|
|
|
|
def cyclic?(dependencies, def_path, ref_path)
|
|
deps = dependencies[def_path]
|
|
if deps
|
|
if deps.include?(ref_path)
|
|
return true
|
|
end
|
|
deps.each do |file|
|
|
return true if cyclic?(dependencies, file, ref_path)
|
|
end
|
|
end
|
|
|
|
return false
|
|
end
|
|
|
|
def run
|
|
consts_defined = {}
|
|
consts_referred = {}
|
|
@file_paths.each do |path|
|
|
parser = Constant.new(File.read(path))
|
|
parser.parse
|
|
parser.defined.each do |const|
|
|
consts_defined[const] = path
|
|
end
|
|
parser.referred.each do |const|
|
|
consts_referred[const] ||= []
|
|
consts_referred[const] << path
|
|
end
|
|
end
|
|
|
|
dependency = @dependencies.dup
|
|
consts_defined.each do |const, def_path|
|
|
if consts_referred[const]
|
|
consts_referred[const].each do |ref_path|
|
|
if def_path != ref_path
|
|
if cyclic?(dependency, def_path, ref_path)
|
|
# remove cyclic dependencies
|
|
next
|
|
end
|
|
|
|
dependency[ref_path] ||= []
|
|
dependency[ref_path] << def_path
|
|
dependency[ref_path].uniq!
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
return dependency
|
|
end
|
|
|
|
class Constant < Ripper::SexpBuilder
|
|
attr_accessor :defined
|
|
attr_accessor :referred
|
|
|
|
def initialize(source)
|
|
@defined = []
|
|
@referred = []
|
|
super
|
|
end
|
|
|
|
def on_const_ref(args)
|
|
args
|
|
end
|
|
|
|
def on_var_field(args)
|
|
args
|
|
end
|
|
|
|
def on_var_ref(args)
|
|
type, name, position = args
|
|
if type == :@const
|
|
@referred << name
|
|
return [:referred, name]
|
|
end
|
|
end
|
|
|
|
def on_const_path_ref(parent, args)
|
|
type, name, position = args
|
|
if type == :@const
|
|
@referred << name
|
|
if parent && parent[0] == :referred
|
|
register_referred_constants(parent[1], name)
|
|
end
|
|
end
|
|
args
|
|
end
|
|
|
|
def on_assign(const, *args)
|
|
type, name, position = const
|
|
if type == :@const
|
|
@defined << name
|
|
return [:defined, name]
|
|
end
|
|
end
|
|
|
|
def on_module(const, *args)
|
|
handle_module_class_event(const, args)
|
|
end
|
|
|
|
def on_class(const, *args)
|
|
handle_module_class_event(const, args)
|
|
end
|
|
|
|
def handle_module_class_event(const, *args)
|
|
type, name, position = const
|
|
if type == :@const
|
|
@defined << name
|
|
@referred.delete(name)
|
|
children = args.flatten
|
|
children.each_with_index do |key, i|
|
|
if key == :defined
|
|
register_defined_constants(name, children[i+1])
|
|
end
|
|
end
|
|
return [:defined, name]
|
|
end
|
|
end
|
|
|
|
def register_defined_constants(parent, child)
|
|
construct_nest_constants!(@defined, parent, child)
|
|
end
|
|
|
|
def register_referred_constants(parent, child)
|
|
construct_nest_constants!(@referred, parent, child)
|
|
end
|
|
|
|
def construct_nest_constants!(consts, parent, child)
|
|
nested = []
|
|
consts.each do |const|
|
|
if md = const.match(/^([^:]+)/)
|
|
nested << "#{parent}::#{const}" if md[0] == child
|
|
end
|
|
end
|
|
consts.concat(nested)
|
|
end
|
|
end
|
|
end
|
|
end; end
|