-
Notifications
You must be signed in to change notification settings - Fork 41
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Avoid overspecialization #400
Comments
Since people are describing this as "gratuitous dynamism" and even "bad design", I gave this some more thought. One radical redesign of the type hierarchy would involve making abstract type AbstractValue end
LLVM.name(val::AbstractValue) = API.LLVMGetValueName(val.ref)
struct Value <: AbstractValue
ref::ValueRef
end
abstract type AbstractUser <: AbstractValue end
operands(user::AbstractUser) = API.LLVMGetOperands(user.ref)
struct Instruction <: AbstractUser
ref::ValueRef
end Then, we'd make APIs return This would bring the level of dynamism back to what C++ provides, expecting the user to check and convert to leaf types based on what's expected or what functionality is required. In turn, this should make functions type stable, as there's no unknown conversion from However, I'm not sure this would solve the problem some people are experiencing. AFAIU, the issue isn't necessarily the type instabilities, but the fact that lots of code is being compiled (specialized), which is separate from the type instabilities. And with the proposed solution here, removing our current dynamism to go back to the C++ level, the API functions would still be invoked with lots of leaf types (in fact, it would be worse as it would also concretize the previously abstract types like In addition, this would be a massively breaking change, requiring all LLVM.jl users to update their code to include manual conversions. It seems like the only way to make affected people happy is to strip all dynamism to the level of the C API, such that there are no type instabilities (everything is a A more fitting example may be the functionality we have around julia> ca = ConstantArray([1, 2, 3])
[3 x i64] [i64 1, i64 2, i64 3]
julia> ca[1]
i64 1 If everything were a Bottom line, given the design of LLVM, it seems like the goal of having a user-friendly wrapper is at odds with a wrapper that works without generating any superfluous code or run-time dynamism. I'd love to be proven wrong though, so if anybody has any ideas, feel free to suggest them here or try them out. If not, the best path forwards to me seems to improve the Julia compiler in a way that we can annotate these structures to avoid specialization in (at least) the API wrappers and related conversion methods (see top post). As should be obvious by now, this design was not chosen willy nilly, but was a careful attempt to try and make the use of LLVM's C APIs as user friendly and safe as possible. The overheads associated with it are unfortunate, but very hard to avoid without sacrificing much of the value of LLVM.jl, at which point it's almost like cc @chriselrod |
We currently duplicate the C++ type hierarchy of LLVM objects (e.g. Argument, Instruction, Function, etc), even though the C API only exposes a flattened one (Value). This has several advantages:
unsafe_delete!
,parent
,name
, automatically dispatching to the correct C method based on the type... but also a disadvantage: code gets overspecialized because we have many representations of a Value. For example:
Here you can see that the
add!
method was specialized multiple times, even though the generated code is basically identical: accessing the handle and passing it to C.Is this even a problem?
I'm not sure. Although we overspecialize, we only do so for tiny methods, so I would think the added latency is minimal.
Possible solution 1: hint the compiler
We could add
@nospecialize(x::Value)
annotations to functions that do not need to be specialized. That's a bit cumbersome, and also doesn't suffice. For example:Even though this successfully avoids overspecialization of
add!
, the underlying wrapper function is still specialized:Adding
@nospecialize
over there (which isn't a trivial thing to add to our wrapper generators) pushes the specialization down another layer:... but that one is probably acceptable, and would result in code that can get reused.
Still, lots of changes across the codebase. And FWIW,
Base.Experimental.@max_methods 1 function add! end
doesn't help, neither does adding that to the LLVM.API submodule.Base.@nospecializeinfer
also doesn't help.Possible solution 2: Rework the type hierarchy
We could rework the type hierarchy to be either flat (only Value) or maybe trait based. The former would mean losing a lot of functionality, the latter reimplementing what Julia does for us (e.g. checking trait as function preconditions), so I'm not inclined to implement this.
Furthermore, we currently do actually sometimes use the ability to store extra fields in Value subtypes, e.g., to store lists of roots, or parent objects. Flattening everything would remove that ability.
The text was updated successfully, but these errors were encountered: