Merge branch 'futexes-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip
* 'futexes-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip: futex: Fix kernel-doc notation & typos futex: Add lock context annotations futex: Mark restart_block.futex.uaddr[2] __user futex: Change 3rd arg of fetch_robust_entry() to unsigned int*
This commit is contained in:
commit
b61f6a57f1
|
@ -23,12 +23,12 @@ struct restart_block {
|
||||||
};
|
};
|
||||||
/* For futex_wait and futex_wait_requeue_pi */
|
/* For futex_wait and futex_wait_requeue_pi */
|
||||||
struct {
|
struct {
|
||||||
u32 *uaddr;
|
u32 __user *uaddr;
|
||||||
u32 val;
|
u32 val;
|
||||||
u32 flags;
|
u32 flags;
|
||||||
u32 bitset;
|
u32 bitset;
|
||||||
u64 time;
|
u64 time;
|
||||||
u32 *uaddr2;
|
u32 __user *uaddr2;
|
||||||
} futex;
|
} futex;
|
||||||
/* For nanosleep */
|
/* For nanosleep */
|
||||||
struct {
|
struct {
|
||||||
|
|
|
@ -91,6 +91,7 @@ struct futex_pi_state {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* struct futex_q - The hashed futex queue entry, one per waiting task
|
* struct futex_q - The hashed futex queue entry, one per waiting task
|
||||||
|
* @list: priority-sorted list of tasks waiting on this futex
|
||||||
* @task: the task waiting on the futex
|
* @task: the task waiting on the futex
|
||||||
* @lock_ptr: the hash bucket lock
|
* @lock_ptr: the hash bucket lock
|
||||||
* @key: the key the futex is hashed on
|
* @key: the key the futex is hashed on
|
||||||
|
@ -104,7 +105,7 @@ struct futex_pi_state {
|
||||||
*
|
*
|
||||||
* A futex_q has a woken state, just like tasks have TASK_RUNNING.
|
* A futex_q has a woken state, just like tasks have TASK_RUNNING.
|
||||||
* It is considered woken when plist_node_empty(&q->list) || q->lock_ptr == 0.
|
* It is considered woken when plist_node_empty(&q->list) || q->lock_ptr == 0.
|
||||||
* The order of wakup is always to make the first condition true, then
|
* The order of wakeup is always to make the first condition true, then
|
||||||
* the second.
|
* the second.
|
||||||
*
|
*
|
||||||
* PI futexes are typically woken before they are removed from the hash list via
|
* PI futexes are typically woken before they are removed from the hash list via
|
||||||
|
@ -295,7 +296,7 @@ void put_futex_key(int fshared, union futex_key *key)
|
||||||
* Slow path to fixup the fault we just took in the atomic write
|
* Slow path to fixup the fault we just took in the atomic write
|
||||||
* access to @uaddr.
|
* access to @uaddr.
|
||||||
*
|
*
|
||||||
* We have no generic implementation of a non destructive write to the
|
* We have no generic implementation of a non-destructive write to the
|
||||||
* user address. We know that we faulted in the atomic pagefault
|
* user address. We know that we faulted in the atomic pagefault
|
||||||
* disabled section so we can as well avoid the #PF overhead by
|
* disabled section so we can as well avoid the #PF overhead by
|
||||||
* calling get_user_pages() right away.
|
* calling get_user_pages() right away.
|
||||||
|
@ -515,7 +516,7 @@ lookup_pi_state(u32 uval, struct futex_hash_bucket *hb,
|
||||||
*/
|
*/
|
||||||
pi_state = this->pi_state;
|
pi_state = this->pi_state;
|
||||||
/*
|
/*
|
||||||
* Userspace might have messed up non PI and PI futexes
|
* Userspace might have messed up non-PI and PI futexes
|
||||||
*/
|
*/
|
||||||
if (unlikely(!pi_state))
|
if (unlikely(!pi_state))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
@ -736,8 +737,8 @@ static void wake_futex(struct futex_q *q)
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We set q->lock_ptr = NULL _before_ we wake up the task. If
|
* We set q->lock_ptr = NULL _before_ we wake up the task. If
|
||||||
* a non futex wake up happens on another CPU then the task
|
* a non-futex wake up happens on another CPU then the task
|
||||||
* might exit and p would dereference a non existing task
|
* might exit and p would dereference a non-existing task
|
||||||
* struct. Prevent this by holding a reference on p across the
|
* struct. Prevent this by holding a reference on p across the
|
||||||
* wake up.
|
* wake up.
|
||||||
*/
|
*/
|
||||||
|
@ -1131,11 +1132,13 @@ static int futex_proxy_trylock_atomic(u32 __user *pifutex,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* futex_requeue() - Requeue waiters from uaddr1 to uaddr2
|
* futex_requeue() - Requeue waiters from uaddr1 to uaddr2
|
||||||
* uaddr1: source futex user address
|
* @uaddr1: source futex user address
|
||||||
* uaddr2: target futex user address
|
* @fshared: 0 for a PROCESS_PRIVATE futex, 1 for PROCESS_SHARED
|
||||||
* nr_wake: number of waiters to wake (must be 1 for requeue_pi)
|
* @uaddr2: target futex user address
|
||||||
* nr_requeue: number of waiters to requeue (0-INT_MAX)
|
* @nr_wake: number of waiters to wake (must be 1 for requeue_pi)
|
||||||
* requeue_pi: if we are attempting to requeue from a non-pi futex to a
|
* @nr_requeue: number of waiters to requeue (0-INT_MAX)
|
||||||
|
* @cmpval: @uaddr1 expected value (or %NULL)
|
||||||
|
* @requeue_pi: if we are attempting to requeue from a non-pi futex to a
|
||||||
* pi futex (pi to pi requeue is not supported)
|
* pi futex (pi to pi requeue is not supported)
|
||||||
*
|
*
|
||||||
* Requeue waiters on uaddr1 to uaddr2. In the requeue_pi case, try to acquire
|
* Requeue waiters on uaddr1 to uaddr2. In the requeue_pi case, try to acquire
|
||||||
|
@ -1360,6 +1363,7 @@ out:
|
||||||
|
|
||||||
/* The key must be already stored in q->key. */
|
/* The key must be already stored in q->key. */
|
||||||
static inline struct futex_hash_bucket *queue_lock(struct futex_q *q)
|
static inline struct futex_hash_bucket *queue_lock(struct futex_q *q)
|
||||||
|
__acquires(&hb->lock)
|
||||||
{
|
{
|
||||||
struct futex_hash_bucket *hb;
|
struct futex_hash_bucket *hb;
|
||||||
|
|
||||||
|
@ -1372,6 +1376,7 @@ static inline struct futex_hash_bucket *queue_lock(struct futex_q *q)
|
||||||
|
|
||||||
static inline void
|
static inline void
|
||||||
queue_unlock(struct futex_q *q, struct futex_hash_bucket *hb)
|
queue_unlock(struct futex_q *q, struct futex_hash_bucket *hb)
|
||||||
|
__releases(&hb->lock)
|
||||||
{
|
{
|
||||||
spin_unlock(&hb->lock);
|
spin_unlock(&hb->lock);
|
||||||
}
|
}
|
||||||
|
@ -1389,6 +1394,7 @@ queue_unlock(struct futex_q *q, struct futex_hash_bucket *hb)
|
||||||
* an example).
|
* an example).
|
||||||
*/
|
*/
|
||||||
static inline void queue_me(struct futex_q *q, struct futex_hash_bucket *hb)
|
static inline void queue_me(struct futex_q *q, struct futex_hash_bucket *hb)
|
||||||
|
__releases(&hb->lock)
|
||||||
{
|
{
|
||||||
int prio;
|
int prio;
|
||||||
|
|
||||||
|
@ -1469,6 +1475,7 @@ retry:
|
||||||
* and dropped here.
|
* and dropped here.
|
||||||
*/
|
*/
|
||||||
static void unqueue_me_pi(struct futex_q *q)
|
static void unqueue_me_pi(struct futex_q *q)
|
||||||
|
__releases(q->lock_ptr)
|
||||||
{
|
{
|
||||||
WARN_ON(plist_node_empty(&q->list));
|
WARN_ON(plist_node_empty(&q->list));
|
||||||
plist_del(&q->list, &q->list.plist);
|
plist_del(&q->list, &q->list.plist);
|
||||||
|
@ -1841,7 +1848,7 @@ retry:
|
||||||
|
|
||||||
restart = ¤t_thread_info()->restart_block;
|
restart = ¤t_thread_info()->restart_block;
|
||||||
restart->fn = futex_wait_restart;
|
restart->fn = futex_wait_restart;
|
||||||
restart->futex.uaddr = (u32 *)uaddr;
|
restart->futex.uaddr = uaddr;
|
||||||
restart->futex.val = val;
|
restart->futex.val = val;
|
||||||
restart->futex.time = abs_time->tv64;
|
restart->futex.time = abs_time->tv64;
|
||||||
restart->futex.bitset = bitset;
|
restart->futex.bitset = bitset;
|
||||||
|
@ -1865,7 +1872,7 @@ out:
|
||||||
|
|
||||||
static long futex_wait_restart(struct restart_block *restart)
|
static long futex_wait_restart(struct restart_block *restart)
|
||||||
{
|
{
|
||||||
u32 __user *uaddr = (u32 __user *)restart->futex.uaddr;
|
u32 __user *uaddr = restart->futex.uaddr;
|
||||||
int fshared = 0;
|
int fshared = 0;
|
||||||
ktime_t t, *tp = NULL;
|
ktime_t t, *tp = NULL;
|
||||||
|
|
||||||
|
@ -2459,7 +2466,7 @@ retry:
|
||||||
*/
|
*/
|
||||||
static inline int fetch_robust_entry(struct robust_list __user **entry,
|
static inline int fetch_robust_entry(struct robust_list __user **entry,
|
||||||
struct robust_list __user * __user *head,
|
struct robust_list __user * __user *head,
|
||||||
int *pi)
|
unsigned int *pi)
|
||||||
{
|
{
|
||||||
unsigned long uentry;
|
unsigned long uentry;
|
||||||
|
|
||||||
|
@ -2648,7 +2655,7 @@ static int __init futex_init(void)
|
||||||
* of the complex code paths. Also we want to prevent
|
* of the complex code paths. Also we want to prevent
|
||||||
* registration of robust lists in that case. NULL is
|
* registration of robust lists in that case. NULL is
|
||||||
* guaranteed to fault and we get -EFAULT on functional
|
* guaranteed to fault and we get -EFAULT on functional
|
||||||
* implementation, the non functional ones will return
|
* implementation, the non-functional ones will return
|
||||||
* -ENOSYS.
|
* -ENOSYS.
|
||||||
*/
|
*/
|
||||||
curval = cmpxchg_futex_value_locked(NULL, 0, 0);
|
curval = cmpxchg_futex_value_locked(NULL, 0, 0);
|
||||||
|
|
|
@ -19,7 +19,7 @@
|
||||||
*/
|
*/
|
||||||
static inline int
|
static inline int
|
||||||
fetch_robust_entry(compat_uptr_t *uentry, struct robust_list __user **entry,
|
fetch_robust_entry(compat_uptr_t *uentry, struct robust_list __user **entry,
|
||||||
compat_uptr_t __user *head, int *pi)
|
compat_uptr_t __user *head, unsigned int *pi)
|
||||||
{
|
{
|
||||||
if (get_user(*uentry, head))
|
if (get_user(*uentry, head))
|
||||||
return -EFAULT;
|
return -EFAULT;
|
||||||
|
|
Loading…
Reference in New Issue