Skip to content

Commit 78bd857

Browse files
authored
Specify interface for LibGit2 abstract types (#36452)
This improves inference results for a number LibGit2 operations, and reduces invalidations from methods (notably, `peel`) that operate on the values extracted from these fields.
1 parent 1eb8bb5 commit 78bd857

File tree

1 file changed

+36
-0
lines changed

1 file changed

+36
-0
lines changed

stdlib/LibGit2/src/types.jl

+36
Original file line numberDiff line numberDiff line change
@@ -966,11 +966,47 @@ function split_cfg_entry(ce::ConfigEntry)
966966
end
967967

968968
# Abstract object types
969+
970+
"""
971+
AbstractGitObject
972+
973+
`AbstractGitObject`s must obey the following interface:
974+
- `obj.owner`, if present, must be a `Union{Nothing,GitRepo,GitTree}`
975+
- `obj.ptr`, if present, must be a `Union{Ptr{Cvoid},Ptr{SignatureStruct}}`
976+
"""
969977
abstract type AbstractGitObject end
978+
979+
function Base.getproperty(obj::AbstractGitObject, name::Symbol)
980+
# These type-assertions enforce the interface requirements above.
981+
# They assist type-inference in cases where the compiler only knows that it
982+
# has an `AbstractGitObject` without being certain about the concrete type.
983+
# See detailed explanation in https://github.com/JuliaLang/julia/pull/36452.
984+
if name === :owner
985+
return getfield(obj, :owner)::Union{Nothing,GitRepo,GitTree}
986+
elseif name === :ptr
987+
return getfield(obj, :ptr)::Union{Ptr{Cvoid},Ptr{SignatureStruct}}
988+
else
989+
return getfield(obj, name)
990+
end
991+
end
992+
970993
Base.isempty(obj::AbstractGitObject) = (obj.ptr == C_NULL)
971994

995+
# `GitObject`s must obey the following interface:
996+
# - `obj.owner` must be a `GitRepo`
997+
# - `obj.ptr` must be a Ptr{Cvoid}
972998
abstract type GitObject <: AbstractGitObject end
973999

1000+
function Base.getproperty(obj::GitObject, name::Symbol)
1001+
if name === :owner
1002+
return getfield(obj, :owner)::GitRepo
1003+
elseif name === :ptr
1004+
return getfield(obj, :ptr)::Ptr{Cvoid}
1005+
else
1006+
return getfield(obj, name)
1007+
end
1008+
end
1009+
9741010
for (typ, owntyp, sup, cname) in [
9751011
(:GitRepo, nothing, :AbstractGitObject, :git_repository),
9761012
(:GitConfig, :(Union{GitRepo, Nothing}), :AbstractGitObject, :git_config),

0 commit comments

Comments
 (0)