D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
usr
/
share
/
passenger
/
phusion_passenger
/
platform_info
/
Filename :
operating_system.rb
back
Copy
# 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. require 'rbconfig' require 'rubygems' PhusionPassenger.require_passenger_lib 'platform_info' module PhusionPassenger module PlatformInfo class VersionComparer < String def <=>(other) Gem::Version.new(self) <=> Gem::Version.new(other.to_s) end end # Returns the operating system's name in as simple a form as possible. For example, # Linux is always identified as "linux". OS X is always identified as "macosx" (despite # the actual os name being something like "darwin"). This is useful as a stable indicator # of the os without having to worry about version numbers, etc. # N.B. unrecognized os names will just be returned as-is. def self.os_name_simple if rb_config['target_os'] =~ /darwin/ && (sw_vers = find_command('sw_vers')) 'macosx' elsif rb_config['target_os'] =~ /^linux-/ 'linux' elsif rb_config['target_os'] =~ /solaris/ 'solaris' elsif rb_config['target_os'] =~ /freebsd/ 'freebsd' elsif rb_config['target_os'] =~ /aix/ 'aix' else rb_config['target_os'] end end memoize :os_name_simple # Returns the operating system's name exactly as advertised by the system. While it is # in lowercase and contains no spaces, it can contain things like version number or # may be less intuitive (e.g. "darwin" for OS X). def self.os_name_full rb_config['target_os'] end memoize :os_name_full # Returns the operating system's version number, or nil if unknown. # This includes the patch version, so for example on macOS Sierra # it could return "10.12.5". # # On Debian/Ubuntu, this returns the version number (e.g. "16.04") # as opposed to the codename ("Trusty"). def self.os_version case os_name_simple when 'macosx' VersionComparer.new(`/usr/bin/sw_vers -productVersion`.strip.split.last) when 'linux' # Parse LSB (applicable to e.g. Ubuntu) if read_file('/etc/lsb-release') =~ /DISTRIB_RELEASE=(.+)/ version = $1.delete(%q['"]) return VersionComparer.new(version) if !version.empty? end # Parse CentOS/RedHat data = read_file('/etc/centos-release') data = read_file('/etc/redhat-release') if data.empty? if !data.empty? data =~ /^(.+?) (Linux |Stream )?(release |version )?(.+?)( |$)/i return VersionComparer.new($4) if $4 end # Parse Debian if File.exist?('/etc/debian_version') return VersionComparer.new(read_file('/etc/debian_version').strip) end nil else nil end end # The current platform's shared library extension ('so' on most Unices). def self.library_extension if os_name_simple == "macosx" return "bundle" else return "so" end end # Returns the `uname` command, or nil if `uname` cannot be found. # In addition to looking for `uname` in `PATH`, this method also looks # for `uname` in /bin and /usr/bin, just in case the user didn't # configure its PATH properly. def self.uname_command if result = find_command("uname") result elsif File.exist?("/bin/uname") return "/bin/uname" elsif File.exist?("/usr/bin/uname") return "/usr/bin/uname" else return nil end end # Returns a list of all CPU architecture names that the current machine CPU # supports. If there are multiple such architectures then the first item in # the result denotes that OS runtime's main/preferred architecture. # # This function normalizes some names. For example x86 is always reported # as "x86" regardless of whether the OS reports it as "i386" or "i686". # x86_64 is always reported as "x86_64" even if the OS reports it as "amd64". # # Please note that even if the CPU supports multiple architectures, the # operating system might not. For example most x86 CPUs nowadays also # support x86_64, but x86_64 Linux systems require various x86 compatibility # libraries to be installed before x86 executables can be run. This function # does not detect whether these compatibility libraries are installed. # The only guarantee that you have is that the OS can run executables in # the architecture denoted by the first item in the result. # # For example, on x86_64 Linux this function can return ["x86_64", "x86"]. # This indicates that the CPU supports both of these architectures, and that # the OS's main/preferred architecture is x86_64. Most executables on the # system are thus be x86_64. It is guaranteed that the OS can run x86_64 # executables, but not x86 executables per se. # # Another example: on MacOS X this function will return some subset of # "x86_64", "x86", and "arm". def self.cpu_architectures if os_name_simple == "macosx" # Macs were intel from around 2006 until 2022. # All but the first 3 Intel Macs' CPUs supported both x86_64 and x86, # and every OS X version from 10.4 to 10.14 had both x86 and x86_64 runtimes installed, # from 10.15 on x86 was dropped, and from (11/10.16) on arm (aarch64) was added. major, minor, *rest = os_version.split(".").map(&:to_i) if major >= 11 || (major == 10 && minor >= 16) # Since Big Sur aarch64 is supported, and default on m1 macs. if `#{uname_command} -m` =~ /arm64/ ["arm64", "x86_64"] elsif `sysctl -in sysctl.proc_translated` == "1" ["arm64", "x86_64"] else ["x86_64", "arm64"] end elsif minor == 15 # Since Catalina x86 is gone. ["x86_64"] elsif minor >= 6 # Since Snow Leopard x86_64 is the default on intel macs. ["x86_64", "x86"] else # Before Snow Leopard x86 was the default. ["x86", "x86_64"] end else uname = uname_command raise "The 'uname' command cannot be found" if !uname arch = `#{uname} -p`.strip # On some systems 'uname -p' returns something like # 'Intel(R) Pentium(R) M processor 1400MHz' or # 'Intel(R)_Xeon(R)_CPU___________X7460__@_2.66GHz'. if arch == "unknown" || arch =~ / / || arch =~ /Hz$/ arch = `#{uname} -m`.strip end if arch =~ /^i.86$/ arch = "x86" elsif arch == "amd64" arch = "x86_64" end if arch == "x86" # Most x86 operating systems nowadays are probably running on # a CPU that supports both x86 and x86_64, but we're not gonna # go through the trouble of checking that. The main architecture # is what we usually care about. ["x86"] elsif arch == "x86_64" # I don't think there's a single x86_64 CPU out there # that doesn't support x86 as well. ["x86_64", "x86"] else [arch] end end end memoize :cpu_architectures, true # Returns whether the flock() function is supported on this OS. def self.supports_flock? defined?(File::LOCK_EX) && os_name_simple != 'solaris' end # Returns whether the OS's main CPU architecture supports the # x86/x86_64 sfence instruction. def self.supports_sfence_instruction? arch = cpu_architectures[0] return arch == "x86_64" || (arch == "x86" && try_compile_and_run("Checking for sfence instruction support", :c, %Q{ int main() { __asm__ __volatile__ ("sfence" ::: "memory"); return 0; } })) end memoize :supports_sfence_instruction?, true # Returns whether the OS's main CPU architecture supports the # x86/x86_64 lfence instruction. def self.supports_lfence_instruction? arch = cpu_architectures[0] return arch == "x86_64" || (arch == "x86" && try_compile_and_run("Checking for lfence instruction support", :c, %Q{ int main() { __asm__ __volatile__ ("lfence" ::: "memory"); return 0; } })) end memoize :supports_lfence_instruction?, true def self.requires_no_tls_direct_seg_refs? return File.exist?("/proc/xen/capabilities") && cpu_architectures[0] == "x86" end memoize :requires_no_tls_direct_seg_refs?, true # Returns true if the current os is running in Windows Subsystem for Linux def self.windows_subsystem? uname = uname_command raise "The 'uname' command cannot be found" if !uname `#{uname} -r`.include?("Microsoft") end memoize :windows_subsystem?, true end end # module PhusionPassenger