Ludovic Courtès <l...@gnu.org> writes:

>> 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*.

>
>> 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.

Fair enough.

>> 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:

--8<---------------cut here---------------start------------->8---
wrap preserves pointer identity, for two pointer objects p1 and p2 that
are equal?, (eq? (wrap p1) (wrap p2)) ⇒ #t.
--8<---------------cut here---------------end--------------->8---

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.

--8<---------------cut here---------------start------------->8---
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
--8<---------------cut here---------------end--------------->8---

>
>> 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.



Reply via email to