Skip site navigation (1)Skip section navigation (2)
Date:      Thu, 07 Feb 2002 17:46:18 -0800
From:      Terry Lambert <tlambert2@mindspring.com>
To:        Julian Elischer <julian@elischer.org>
Cc:        current@freebsd.org
Subject:   Re: ucred for threads
Message-ID:  <3C632DEA.F522339D@mindspring.com>
References:  <Pine.BSF.4.21.0202071633430.91961-100000@InterJet.elischer.org>

next in thread | previous in thread | raw e-mail | index | archive | help
Julian Elischer wrote:
> In the KSE code I have:
>  in trap(), and syscall()
> 
>         if (td->td_ucred != p->p_ucred) {
>                 PROC_LOCK(p);
>                 if (td->td_ucred) {
>                         crfree(td->td_ucred);
>                         td->td_ucred = NULL;
>                 }
>                 if (p->p_ucred != NULL) {
>                         td->td_ucred = crhold(p->p_ucred);
>                 }
>                 PROC_UNLOCK(p);
>         }
> 
> THis is actually not dependent on KSE (though I originally needed it
> for KSE I have since decided that it would be a good idea anyhow).
> 
> Do those who deal in ucreds (probably jhb and Robert W)
> agree or disagree.. (if so I'll happily commit it as it shrinks the KSE
> patches a bit more :-)

You can't have a process with a thread without a ucred, right?

If so, then the idea is sound, but the code is broken.

In the case that they are not equal, you free it, null
the pointer, and don't allocate a replacement.

What is the default state of td->td_ucred?

In theory, it should be initialized on creation, and then
left initialized (you are effectively lazy binding the cred
references).

Since the reference count is positive in both cases, an
unlocked pointer comparison is fine.  In the case where
there is a race on a credential change in the unlocked
value, that race exists in the calling code, anyway.  The
failure case in a change for->to instead of to->from is
also safe, since the pointers will be inequal during the
update, the lock will be held during the update, so the
subsequent release and regrab with the increment never
dropping below 1.

If that's the case, then the code should be:

        if (td->td_ucred != p->p_ucred) {
                PROC_LOCK(p);
                if (td->td_ucred) {
                        crfree(td->td_ucred);
                }
                td->td_ucred = crhold(p->p_ucred);
                PROC_UNLOCK(p);
        }

Is a thread lock required?  I think it's guaranteed to
be non-reentrant, so the answer should be no...

-- Terry

To Unsubscribe: send mail to majordomo@FreeBSD.org
with "unsubscribe freebsd-current" in the body of the message




Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?3C632DEA.F522339D>