Skip to content

Commit 7badc10

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

File tree

4 files changed

+91
-2
lines changed

4 files changed

+91
-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

+48
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,48 @@ 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+
if haskey(platform, "march")
100+
set_compare_strategy!(platform, "march", march_comparison_strategy)
101+
return platform
102+
end
103+
104+
host_arch = arch(HostPlatform())
105+
host_isas = arch_march_isa_mapping[host_arch]
106+
idx = findlast(((name, isa),) -> isa <= CPUID.cpu_isa(), host_isas)
107+
platform["march"] = first(host_isas[idx])
108+
set_compare_strategy!(platform, "march", march_comparison_strategy)
109+
return platform
110+
end

test/compat.jl

+41
Original file line numberDiff line numberDiff line change
@@ -71,3 +71,44 @@ using BinaryBuilderBase: download_verify, list_tarball_files
7171
end
7272
end
7373
end
74+
75+
using BinaryBuilderBase: march_comparison_strategy
76+
using Base.BinaryPlatforms: Platform, platforms_match, set_compare_strategy!
77+
@testset "Microarchitecture augmentation" begin
78+
linux_x86_64 = Platform("x86_64", "linux")
79+
linux_avx = Platform("x86_64", "linux"; march="avx")
80+
linux_avx2 = Platform("x86_64", "linux"; march="avx2")
81+
linux_avx512 = Platform("x86_64", "linux"; march="avx512")
82+
# Platform with non-existing microarchitecture
83+
linux_bad = Platform("x86_64", "linux"; march="bad")
84+
85+
# Without any custom comparison strategy, the base platform without march matches
86+
# everything, but the others are all incompatible
87+
@test platforms_match(linux_x86_64, linux_avx)
88+
@test platforms_match(linux_x86_64, linux_avx2)
89+
@test platforms_match(linux_x86_64, linux_avx512)
90+
@test platforms_match(linux_x86_64, linux_bad)
91+
@test !platforms_match(linux_avx, linux_avx2)
92+
@test !platforms_match(linux_avx, linux_avx512)
93+
@test !platforms_match(linux_avx, linux_bad)
94+
@test !platforms_match(linux_avx2, linux_bad)
95+
@test !platforms_match(linux_avx2, linux_avx512)
96+
@test !platforms_match(linux_avx512, linux_bad)
97+
98+
# Teach AVX2 platform how to compare the others
99+
set_compare_strategy!(linux_avx2, "march", march_comparison_strategy)
100+
for compatible_p in (linux_x86_64, linux_avx)
101+
@test platforms_match(compatible_p, linux_avx2)
102+
@test platforms_match(linux_avx2, compatible_p)
103+
end
104+
for incompatible_p in (linux_avx512, linux_bad)
105+
@test !platforms_match(incompatible_p, linux_avx2)
106+
@test !platforms_match(linux_avx2, incompatible_p)
107+
end
108+
109+
# Teach also AVX platform how to compare
110+
set_compare_strategy!(linux_avx, "march", march_comparison_strategy)
111+
# Now when we compare AVX and AVX2, they must be equal
112+
@test !platforms_match(linux_avx, linux_avx2)
113+
@test !platforms_match(linux_avx2, linux_avx)
114+
end

0 commit comments

Comments
 (0)