# Phusion Passenger - https://www.phusionpassenger.com/ # Copyright (c) 2010-2017 Phusion Holding B.V. # # "Passenger", "Phusion Passenger" and "Union Station" are registered # trademarks of Phusion Holding B.V. # # 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. # Rake functions for compiling/linking C++ stuff. def run_compiler(*command) PhusionPassenger.require_passenger_lib 'utils/ansi_colors' if !defined?(PhusionPassenger::Utils::AnsiColors) show_command = command.join(' ') puts show_command if !system(*command) colors = PhusionPassenger::Utils::AnsiColors.new if $? && $?.exitstatus == 4 # This probably means the compiler ran out of memory. msg = "" + "-----------------------------------------------\n" + "Your compiler failed with the exit status 4. This " + "probably means that it ran out of memory. To solve " + "this problem, try increasing your swap space: " + "https://www.digitalocean.com/community/articles/how-to-add-swap-on-ubuntu-12-04" + "" fail(colors.ansi_colorize(msg)) elsif $? && $?.termsig == 9 msg = "" + "-----------------------------------------------\n" + "Your compiler was killed by the operating system. This " + "probably means that it ran out of memory. To solve " + "this problem, try increasing your swap space: " + "https://www.digitalocean.com/community/articles/how-to-add-swap-on-ubuntu-12-04" + "" fail(colors.ansi_colorize(msg)) else fail "Command failed with status (#{$? ? $?.exitstatus : 1}): [#{show_command}]" end end end def build_compiler_flags_from_options_or_flags(options_or_flags) if options_or_flags.is_a?(Hash) result = [] options = options_or_flags (options[:include_paths] || []).each do |path| result << "-I#{path}" end if flags = options[:flags] result.concat([flags].flatten) end result.flatten.reject{ |x| x.nil? || x.empty? }.join(" ") elsif options_or_flags.is_a?(String) options_or_flags elsif options_or_flags.respond_to?(:call) build_compiler_flags_from_options_or_flags(options_or_flags.call) elsif options_or_flags.nil? "" else raise ArgumentError, "Invalid argument type: #{options_or_flags.inspect}" end end def generate_compilation_task_dependencies(source, options = nil) result = [source] if dependencies = CXX_DEPENDENCY_MAP[source] result.concat(dependencies) end options = maybe_eval_lambda(options) if options && options[:deps] result.concat([options[:deps]].flatten.compact) end result end def compile_c(object, source, options_or_flags = nil) flags = build_compiler_flags_from_options_or_flags(options_or_flags) ensure_target_directory_exists(object) run_compiler("#{cc} -o #{object} #{EXTRA_PRE_CFLAGS} #{flags} #{extra_cflags} -c #{source}") end def compile_cxx(object, source, options_or_flags = nil) flags = build_compiler_flags_from_options_or_flags(options_or_flags) ensure_target_directory_exists(object) run_compiler("#{cxx} -o #{object} #{EXTRA_PRE_CXXFLAGS} #{flags} #{extra_cxxflags} -c #{source}") end def create_c_executable(target, objects, options_or_flags = nil) objects = [objects].flatten.join(" ") flags = build_compiler_flags_from_options_or_flags(options_or_flags) ensure_target_directory_exists(target) run_compiler("#{cc} -o #{target} #{objects} #{EXTRA_PRE_C_LDFLAGS} #{flags} #{extra_c_ldflags}") end def create_cxx_executable(target, objects, options_or_flags = nil) objects = [objects].flatten.join(" ") flags = build_compiler_flags_from_options_or_flags(options_or_flags) ensure_target_directory_exists(target) run_compiler("#{cxx} -o #{target} #{objects} #{EXTRA_PRE_CXX_LDFLAGS} #{flags} #{extra_cxx_ldflags}") end def create_static_library(target, objects) # On OS X, 'ar cru' will sometimes fail with an obscure error: # # ar: foo.a is a fat file (use libtool(1) or lipo(1) and ar(1) on it) # ar: foo.a: Inappropriate file type or format # # So here we delete the ar file before creating it, which bypasses this problem. objects = [objects].flatten.join(" ") ensure_target_directory_exists(target) sh "rm -rf #{target}" sh "ar cru #{target} #{objects}" sh "ranlib #{target}" end def create_shared_library(target, objects, options_or_flags = nil) if PlatformInfo.os_name_simple == "macosx" shlib_flag = "-flat_namespace -bundle -undefined dynamic_lookup" else shlib_flag = "-shared" end if PhusionPassenger::PlatformInfo.cxx_is_sun_studio? fPIC = "-KPIC" else fPIC = "-fPIC" end objects = [objects].flatten.join(" ") flags = build_compiler_flags_from_options_or_flags(options_or_flags) ensure_target_directory_exists(target) run_compiler("#{cxx} #{shlib_flag} #{objects} #{fPIC} -o #{target} #{flags}") end def define_c_object_compilation_task(object, source, options_or_flags = nil) options = options_or_flags if options_or_flags.is_a?(Hash) || options_or_flags.respond_to?(:call) file(object => generate_compilation_task_dependencies(source, options)) do compile_c(object, source, options_or_flags) end end def define_cxx_object_compilation_task(object, source, options_or_flags = nil) options = options_or_flags if options_or_flags.is_a?(Hash) || options_or_flags.respond_to?(:call) file(object => generate_compilation_task_dependencies(source, options)) do compile_cxx(object, source, options_or_flags) end end def define_c_or_cxx_object_compilation_task(object, source, options_or_flags = nil) if source =~ /\.c$/ define_c_object_compilation_task(object, source, options_or_flags) else define_cxx_object_compilation_task(object, source, options_or_flags) end end