Skip to content

Commit f01f186

Browse files
committed
[Products] Don't dlopen libraries built for an incompatible ISA
1 parent 55c795d commit f01f186

File tree

3 files changed

+47
-2
lines changed

3 files changed

+47
-2
lines changed

Project.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
name = "BinaryBuilderBase"
22
uuid = "7f725544-6523-48cd-82d1-3fa08ff4056e"
33
authors = ["Elliot Saba <[email protected]>"]
4-
version = "1.8.0"
4+
version = "1.8.1"
55

66
[deps]
77
CodecZlib = "944b1d66-785c-5afd-91f1-9de20f533193"

src/Products.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -222,7 +222,7 @@ function locate(lp::LibraryProduct, prefix::Prefix; platform::AbstractPlatform =
222222
end
223223

224224
# If it does, try to `dlopen()` it if the current platform is good
225-
if (!lp.dont_dlopen && !skip_dlopen) && platforms_match(platform, HostPlatform())
225+
if (!lp.dont_dlopen && !skip_dlopen) && platforms_match(platform, augment_microarchitecture!(HostPlatform()))
226226
if isolate
227227
# Isolated dlopen is a lot slower, but safer
228228
if success(`$(Base.julia_cmd()) --startup-file=no -e "import Libdl; Libdl.dlopen(\"$dl_path\")"`)

src/utils.jl

+45
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
# The functions in this file may not be used anywhere in this package but may
22
# needed by different modules of BinaryBuilder.jl
33

4+
using Base.BinaryPlatforms: arch_march_isa_mapping, set_compare_strategy!
5+
using Base.BinaryPlatforms.CPUID
6+
47
with_logfile(f::Function, prefix::Prefix, name::String; subdir="") = with_logfile(f, joinpath(logdir(prefix; subdir), name))
58
function with_logfile(f::Function, logfile::String)
69
mkpath(dirname(logfile))
@@ -60,3 +63,45 @@ end
6063
# We want the AnyPlatform to look like `default_host_platform`,
6164
get_concrete_platform(::AnyPlatform, shards::Vector{CompilerShard}) =
6265
get_concrete_platform(default_host_platform, shards)
66+
67+
function march_comparison_strategy(a::String, b::String, a_requested::Bool, b_requested::Bool)
68+
# If both b and a requested, then we fall back to equality:
69+
if a_requested && b_requested
70+
return a == b
71+
end
72+
73+
function get_arch_isa(isa_name::String)
74+
for (arch, isas) in arch_march_isa_mapping
75+
for (name, isa) in isas
76+
name == isa_name && return arch, isa
77+
end
78+
end
79+
return nothing, nothing
80+
end
81+
82+
a_arch, a_isa = get_arch_isa(a)
83+
b_arch, b_isa = get_arch_isa(b)
84+
if any(isnothing, (a_arch, b_arch)) || a_arch != b_arch
85+
# Architectures are definitely not compatible, exit early
86+
return false
87+
end
88+
89+
if a_requested
90+
# ISA `b` is compatible with ISA `a` only if it's a subset of `a`
91+
return b_isa a_isa
92+
else
93+
# ISA `a` is compatible with ISA `b` only if it's a subset of `b`
94+
return a_isa b_isa
95+
end
96+
end
97+
98+
function augment_microarchitecture!(platform::Platform)
99+
haskey(platform, "march") && return platform
100+
101+
host_arch = arch(HostPlatform())
102+
host_isas = arch_march_isa_mapping[host_arch]
103+
idx = findlast(((name, isa),) -> isa <= CPUID.cpu_isa(), host_isas)
104+
platform["march"] = first(host_isas[idx])
105+
set_compare_strategy!(platform, "march", march_comparison_strategy)
106+
return platform
107+
end

0 commit comments

Comments
 (0)