| 1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
 | require "compilers"
class CxxStdlib
  include CompilerConstants
  class CompatibilityError < StandardError
    def initialize(formula, dep, stdlib)
      super <<-EOS.undent
        #{formula.full_name} dependency #{dep.name} was built with a different C++ standard
        library (#{stdlib.type_string} from #{stdlib.compiler}). This may cause problems at runtime.
        EOS
    end
  end
  def self.create(type, compiler)
    if type && ![:libstdcxx, :libcxx].include?(type)
      raise ArgumentError, "Invalid C++ stdlib type: #{type}"
    end
    klass = (compiler.to_s =~ GNU_GCC_REGEXP) ? GnuStdlib : AppleStdlib
    klass.new(type, compiler)
  end
  def self.check_compatibility(formula, deps, keg, compiler)
    return if formula.skip_cxxstdlib_check?
    stdlib = create(keg.detect_cxx_stdlibs.first, compiler)
    begin
      stdlib.check_dependencies(formula, deps)
    rescue CompatibilityError => e
      opoo e.message
    end
  end
  attr_reader :type, :compiler
  def initialize(type, compiler)
    @type = type
    @compiler = compiler.to_sym
  end
  # If either package doesn't use C++, all is well
  # libstdc++ and libc++ aren't ever intercompatible
  # libstdc++ is compatible across Apple compilers, but
  # not between Apple and GNU compilers, or between GNU compiler versions
  def compatible_with?(other)
    return true if type.nil? || other.type.nil?
    return false unless type == other.type
    apple_compiler? && other.apple_compiler? ||
      !other.apple_compiler? && compiler.to_s[4..6] == other.compiler.to_s[4..6]
  end
  def check_dependencies(formula, deps)
    deps.each do |dep|
      # Software is unlikely to link against libraries from build-time deps, so
      # it doesn't matter if they link against different C++ stdlibs.
      next if dep.build?
      dep_stdlib = Tab.for_formula(dep.to_formula).cxxstdlib
      unless compatible_with? dep_stdlib
        raise CompatibilityError.new(formula, dep, dep_stdlib)
      end
    end
  end
  def type_string
    type.to_s.gsub(/cxx$/, "c++")
  end
  def inspect
    "#<#{self.class.name}: #{compiler} #{type}>"
  end
  class AppleStdlib < CxxStdlib
    def apple_compiler?
      true
    end
  end
  class GnuStdlib < CxxStdlib
    def apple_compiler?
      false
    end
  end
end
 |