Skip to content

Commit 91d7e18

Browse files
authored
Ensure hash(::Platform) is stable (#37734)
This allows us to key dictionaries with `Platform` objects more reliably
1 parent 4165fdd commit 91d7e18

File tree

2 files changed

+28
-8
lines changed

2 files changed

+28
-8
lines changed

base/binaryplatforms.jl

+14-3
Original file line numberDiff line numberDiff line change
@@ -142,6 +142,20 @@ function Base.setindex!(p::AbstractPlatform, v::String, k::String)
142142
return p
143143
end
144144

145+
# Hash definitino to ensure that it's stable
146+
function Base.hash(p::Platform, h::UInt)
147+
h += 0x506c6174666f726d % UInt
148+
h = hash(p.tags, h)
149+
h = hash(p.compare_strategies, h)
150+
return h
151+
end
152+
153+
# Simple equality definition; for compatibility testing, use `platforms_match()`
154+
function Base.:(==)(a::Platform, b::Platform)
155+
return a.tags == b.tags && a.compare_strategies == b.compare_strategies
156+
end
157+
158+
145159
# Allow us to easily serialize Platform objects
146160
function Base.repr(p::Platform; context=nothing)
147161
str = string(
@@ -166,9 +180,6 @@ function Base.show(io::IO, p::Platform)
166180
print(io, str)
167181
end
168182

169-
# Simple equality definition; for compatibility testing, use `platforms_match()`
170-
Base.:(==)(a::AbstractPlatform, b::AbstractPlatform) = tags(a) == tags(b)
171-
172183
function validate_tags(tags::Dict)
173184
throw_invalid_key(k) = throw(ArgumentError("Key \"$(k)\" cannot have value \"$(tags[k])\""))
174185
# Validate `arch`

test/binaryplatforms.jl

+14-5
Original file line numberDiff line numberDiff line change
@@ -136,6 +136,16 @@ end
136136

137137
# Test that trying to set illegal tags fails
138138
@test_throws ArgumentError p["os"] = "a+b"
139+
140+
# Test that our `hash()` is stable
141+
@test hash(HostPlatform()) == hash(HostPlatform())
142+
143+
# Test that round-tripping through `triplet` for a does not
144+
# maintain equality, as we end up losing the `compare_strategies`:
145+
p = Platform("x86_64", "linux"; cuda = v"11")
146+
Base.BinaryPlatforms.set_compare_strategy!(p, "cuda", Base.BinaryPlatforms.compare_version_cap)
147+
q = parse(Platform, triplet(p))
148+
@test q != p
139149
end
140150

141151
@testset "Triplet parsing" begin
@@ -186,13 +196,12 @@ end
186196
# Round-trip our little homie through `triplet()`, with some bending
187197
# of the rules for MacOS and FreeBSD, who have incomplete `os_version`
188198
# numbers embedded within their triplets.
189-
p = HostPlatform()
190-
if !Sys.isbsd(p)
191-
@test parse(Platform, triplet(p)) == p
192-
end
199+
p = Platform("x86_64", "linux")
200+
@test parse(Platform, triplet(p)) == p
193201

194202
# Also test round-tripping through `repr()`:
195-
@test eval(Meta.parse(repr(HostPlatform()))) == HostPlatform()
203+
p = Platform("aarch64", "macos"; os_version=v"20", march="armv8_4_crypto_sve")
204+
@test eval(Meta.parse(repr(p))) == p
196205
end
197206

198207
@testset "platforms_match()" begin

0 commit comments

Comments
 (0)