Ludovic Courtès <ludo@gnu.org> writes:
Toggle quote (7 lines)
>> I did not know guile-git was supposed to guarantee it. I assumed it is
>> just a binding to libgit2, so I have expected it to have the same
>> semantics.
>
> Well, ‘eq?’ only makes sense for the Scheme objects that wrap the
> underlying C objects.
In strict sense of the word, sure, but even in C being able to compare
pointers is useful. My understanding of eq? for objects wrapping C data
structures basically is "pointer equality", so in that sense I have
expected eq? on <commit> to be the equivalent of == on git_commit*.
Toggle quote (8 lines)
>
>> I think I would just fix this in Guix, and drop the promise on
>> guile-git's side. I am obviously unsure whether people rely on it or
>> not, but given it does not work I am tempted to guess.
>
> (guix git) relies on it and I probably wrote pieces of code with that
> assumption in mind.
Toggle quote (9 lines)
>> However I do see the appeal in being able to use eq?. The correct
>> action probably depends on in what direction you want to take guile-git.
>> Should it stay just a bindings wrapper, or should it provide extra
>> features and guarantees?
>
> It’s not really an extra feature, it’s really a core part of defining
> bindings in my view; ‘define-wrapped-pointer-type’ in Guile implements
> those semantics.
I have read through the documentation for define-wrapped-pointer-type,
and I must say I do not see this part in the documentation. Only part
talking about eq? is this:
Toggle snippet (4 lines)
wrap preserves pointer identity, for two pointer objects p1 and p2 that
are equal?, (eq? (wrap p1) (wrap p2)) ⇒ #t.
But the two commit pointers in this case are not equal?, so why should
they eq??
However on my first (and second) reading of the documentation I did not
get it, and expected eq? in the below snippet to return #f (since the
pointer being wrapped does equal? to itself). So maybe I am missing
something else as well.
Toggle snippet (15 lines)
scheme@(guile-user)> ,use (system foreign)
scheme@(guile-user)> ,use (system foreign-library)
scheme@(guile-user)> (define-wrapped-pointer-type foo foo? wrap-foo unwrap-foo #t)
scheme@(guile-user)> (define-wrapped-pointer-type bar bar? wrap-bar unwrap-bar #t)
scheme@(guile-user)> (define open* (foreign-library-function #f "open"))
scheme@(guile-user)> (wrap-foo open*)
$8 = #<foo 7fec7aac1010>
scheme@(guile-user)> (wrap-bar open*)
$9 = #<bar 7fec7d3dd510>
scheme@(guile-user)> (eq? $8 $9)
$10 = #f
scheme@(guile-user)> (equal? open* open*)
$12 = #t
Toggle quote (8 lines)
>
>> But then again, I do not have much experience with weak tables, and
>> guile-git internals, so maybe I overestimate the complexity. I would be
>> scared I miss some paths that return commits though.
>
> I’m happy to provide guidance if you want to give it a try, lemme
> know!
One aspect I was thinking about is whether this makes sense to do in
libgit2 directly and contribute to the upstream. This ("stable" mapping
of hash -> pointer) does seem useful, and there already is
git_commit_free, so maybe it could just reference count? In that way
all user, not just Guile, would benefit.
Tomas
--
There are only two hard things in Computer Science:
cache invalidation, naming things and off-by-one errors.