-
Notifications
You must be signed in to change notification settings - Fork 23
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
Why do we use cloneGeom
for every output?
#196
Comments
Yeah, this is something I encountered as well and partially fixed for points, just like you in the other PR. There are double inits, as getX already is a X. Most constructors do clone if given the chance. Clonegeom is only needed on init with another existing geometry. |
However, we must be careful. Your exteriorring instance must be cloned per https://libgeos.org/doxygen/geos__c_8h.html#afc5d54a2905996833747c2d566da2af3. |
I will make a PR |
I wondered... instead of cloning could we skip the finaliser on polygon rings? The important thing is not to delete them? (I guess you could get a ring then delete its parent polygon and trigger a segfault) |
Are you proposing we leak memory? ;) Everything we create in GEOS needs to be destroyed again. |
We arent creating anything with polygon rings, and theyre attached to the polygons finaliser. Right? Why else should we never delete them? |
Ah, I misunderstood. You meant like a view or similar? The problem is that Julia doesn't know the ring is linked to the Polygon, so it could garbage collect the Polygon, invalidating the ring. We could make structs like prepared geometry, linking back to the Polygon object? Anyway, not sure how much performance you gain by it. |
I think its more than half the cost of loading a polygon. Probably we would need an One day when everything else is done 😅 |
Let's not burden GI with unsafe, but let's see what I can make with a GC.preserve, never used it extensively. |
Well, with the following code (including a third argument to the constructor to run a finalizer or not), the tests run, but I can trigger segfaults. Any ideas? GC.@preserve obj begin
return LinearRing(result, context, false)
end But this segfaults: julia> import LibGEOS as LG
julia> ring = LG.exteriorRing(pLG)^C
julia> pLG = LG.Polygon([
[
[0.0, 5.0], [2.0, 2.0], [5.0, 2.0], [2.0, -2.0], [5.0, -5.0],
[0.0, -2.0], [-5.0, -5.0], [-2.0, -2.0], [-5.0, 2.0], [-2.0, 2.0],
[0.0, 5.0],
],
])
POLYGON ((0 5, 2 2, 5 2, 2 -2, 5 -5, 0 -2, -5 -5, -2 -2, -5 2, -2 2, 0 5))
julia> ring = LG.exteriorRing(pLG)
LINEARRING (0 5, 2 2, 5 2, 2 -2, 5 -5, 0 -2, -5 -5, -2 -2, -5 2, -2 2, 0 5)
julia> pLG = nothing
julia> GC.gc()
julia> ring
# Segfaults |
Thats why we need a method with a promise the contents dont escape the closure! I dont think it can work otherwise |
Or something like this, with AbstractLinearRing replacing LinearRing in most signatures. abstract type AbstractLinearRing <: AbstractGeometry end
struct LinearRingView <: AbstractLinearRing
ptr::GEOSGeom
context::GEOSContext
parent::AbstractGeometry
function LinearRingView(
obj::GEOSGeom,
parent::AbstractGeometry,
context::GEOSContext = get_global_context(),
)
new(obj, context, parent)
end
end
mutable struct LinearRing <: AbstractLinearRing |
Wont it still segfault if you delete the polygon? |
Nope, because the polygon is linked in the parent field. |
We often seem to do this multiple times. Clearly we can remove some of these double clones. But do we need to clone at all? I cant see why...
This one seems like an obvious unnecessary copy:
LibGEOS.jl/src/geos_functions.jl
Line 1416 in feb0055
Because then we copy again here:
LibGEOS.jl/src/geos_types.jl
Line 179 in feb0055
Removing these copies makes everything much faster. This is our GeometryOps.jl
area
method` (with a few branches checked out) on LG polygon. We don't need LibGEOS, its just for testing. But its a good example:JuliaGeo/GeometryOps.jl#32
With
cloneGeom
Removing the first
cloneGeom
:And the first and second
cloneGeom
And finally if I remove the one one the line above when a
LineString
rewrapsanother
LineString
:We shouldn't change this one, but it shouldn't be happening in
getgeom
anyway.The text was updated successfully, but these errors were encountered: