Suppose we have the following structs matching some structs in a C header:
struct Foo
a::UInt32
b::Cstring
end
struct Bar
a::UInt32
b::Ptr{Foo}
end
And we have a function that takes a ptr to Bar. ccall will handle one conversion (from Bar to a Bar pointer) for us, and writing a function that takes a Bar and calls ccall will ensure a julia ref exists for the scope of the C function call.
However that’s not true for Bar’s field b. What’s the idiomatic way to create a pointer to a Foo, since pointer_from_objref should not be called on immutables?
As a follow on, what’s the “julian” way to redefine these types so that they use julia strings and refs and conversion “just works”?
I’m confused. Does ccall really do that conversion if you doesn’t overload Base.cconvert? What does your ccall signature look like? To pass Bar to a C function that accepts Ptr{Bar}, the common way is to wrap Bar into a Ref (Ref{Bar}) and then pass it to ccall and the conversion from Ref{Bar} to Ptr{Bar} just works. As for Foo, it’s similar.
fooref = Ref(foo_object)
GC.@preserve fooref begin
ptr = Base.unsafe_convert(Ptr{Foo}, fooref)
....some code using ptr....
end
Here, I used GC.@preserve to make sure that the fooref object lives for as long as I need ptr. Other strategies can also be employed (e.g. stashing fooref in some persistent data structure) if you need ptr to be valid for a longer time.
You are correct, this was a total misconception on my behalf here.
Right, so first we need to create a julia reference, then use unsafe_convert for the pointer conversion, and make sure that reference sticks around so the GC doesn’t clean up the memory while we’re using it.
For rare applications that need to store “raw” pointers to the underlying native Julia object. For example, it is used in PyCall to stuff a reference to a native Julia object into a Python “wrapper” object that can be used to refer to the Julia object from Python.
Would an example use of this be a C api that supports callbacks with user data (for example using a glfwGetWindowUserPointer in a window resize callback)? i.e. cases where we want a callback to be run on a native julia object?
(with the understanding that we’d probably have fewer headaches if we just captured that reference within the callback itself for most use cases)