ipc/sem: Fix semctl(..., GETPID, ...) between pid namespaces
Today the last process to update a semaphore is remembered and
reported in the pid namespace of that process. If there are processes
in any other pid namespace querying that process id with GETPID the
result will be unusable nonsense as it does not make any
sense in your own pid namespace.
Due to ipc_update_pid I don't think you will be able to get System V
ipc semaphores into a troublesome cache line ping-pong. Using struct
pids from separate process are not a problem because they do not share
a cache line. Using struct pid from different threads of the same
process are unlikely to be a problem as the reference count update
can be avoided.
Further linux futexes are a much better tool for the job of mutual
exclusion between processes than System V semaphores. So I expect
programs that are performance limited by their interprocess mutual
exclusion primitive will be using futexes.
So while it is possible that enhancing the storage of the last
rocess of a System V semaphore from an integer to a struct pid
will cause a performance regression because of the effect
of frequently updating the pid reference count. I don't expect
that to happen in practice.
This change updates semctl(..., GETPID, ...) to return the
process id of the last process to update a semphore inthe
pid namespace of the calling process.
Fixes: b488893a39
("pid namespaces: changes to show virtual ids to user")
Signed-off-by: "Eric W. Biederman" <ebiederm@xmission.com>
This commit is contained in:
parent
39a4940eaa
commit
51d6f2635b
22
ipc/sem.c
22
ipc/sem.c
@ -98,7 +98,7 @@ struct sem {
|
|||||||
* - semctl, via SETVAL and SETALL.
|
* - semctl, via SETVAL and SETALL.
|
||||||
* - at task exit when performing undo adjustments (see exit_sem).
|
* - at task exit when performing undo adjustments (see exit_sem).
|
||||||
*/
|
*/
|
||||||
int sempid;
|
struct pid *sempid;
|
||||||
spinlock_t lock; /* spinlock for fine-grained semtimedop */
|
spinlock_t lock; /* spinlock for fine-grained semtimedop */
|
||||||
struct list_head pending_alter; /* pending single-sop operations */
|
struct list_head pending_alter; /* pending single-sop operations */
|
||||||
/* that alter the semaphore */
|
/* that alter the semaphore */
|
||||||
@ -128,7 +128,7 @@ struct sem_queue {
|
|||||||
struct list_head list; /* queue of pending operations */
|
struct list_head list; /* queue of pending operations */
|
||||||
struct task_struct *sleeper; /* this process */
|
struct task_struct *sleeper; /* this process */
|
||||||
struct sem_undo *undo; /* undo structure */
|
struct sem_undo *undo; /* undo structure */
|
||||||
int pid; /* process id of requesting process */
|
struct pid *pid; /* process id of requesting process */
|
||||||
int status; /* completion status of operation */
|
int status; /* completion status of operation */
|
||||||
struct sembuf *sops; /* array of pending operations */
|
struct sembuf *sops; /* array of pending operations */
|
||||||
struct sembuf *blocking; /* the operation that blocked */
|
struct sembuf *blocking; /* the operation that blocked */
|
||||||
@ -628,7 +628,8 @@ SYSCALL_DEFINE3(semget, key_t, key, int, nsems, int, semflg)
|
|||||||
*/
|
*/
|
||||||
static int perform_atomic_semop_slow(struct sem_array *sma, struct sem_queue *q)
|
static int perform_atomic_semop_slow(struct sem_array *sma, struct sem_queue *q)
|
||||||
{
|
{
|
||||||
int result, sem_op, nsops, pid;
|
int result, sem_op, nsops;
|
||||||
|
struct pid *pid;
|
||||||
struct sembuf *sop;
|
struct sembuf *sop;
|
||||||
struct sem *curr;
|
struct sem *curr;
|
||||||
struct sembuf *sops;
|
struct sembuf *sops;
|
||||||
@ -666,7 +667,7 @@ static int perform_atomic_semop_slow(struct sem_array *sma, struct sem_queue *q)
|
|||||||
sop--;
|
sop--;
|
||||||
pid = q->pid;
|
pid = q->pid;
|
||||||
while (sop >= sops) {
|
while (sop >= sops) {
|
||||||
sma->sems[sop->sem_num].sempid = pid;
|
ipc_update_pid(&sma->sems[sop->sem_num].sempid, pid);
|
||||||
sop--;
|
sop--;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -753,7 +754,7 @@ static int perform_atomic_semop(struct sem_array *sma, struct sem_queue *q)
|
|||||||
un->semadj[sop->sem_num] = undo;
|
un->semadj[sop->sem_num] = undo;
|
||||||
}
|
}
|
||||||
curr->semval += sem_op;
|
curr->semval += sem_op;
|
||||||
curr->sempid = q->pid;
|
ipc_update_pid(&curr->sempid, q->pid);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -1160,6 +1161,7 @@ static void freeary(struct ipc_namespace *ns, struct kern_ipc_perm *ipcp)
|
|||||||
unlink_queue(sma, q);
|
unlink_queue(sma, q);
|
||||||
wake_up_sem_queue_prepare(q, -EIDRM, &wake_q);
|
wake_up_sem_queue_prepare(q, -EIDRM, &wake_q);
|
||||||
}
|
}
|
||||||
|
ipc_update_pid(&sem->sempid, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Remove the semaphore set from the IDR */
|
/* Remove the semaphore set from the IDR */
|
||||||
@ -1352,7 +1354,7 @@ static int semctl_setval(struct ipc_namespace *ns, int semid, int semnum,
|
|||||||
un->semadj[semnum] = 0;
|
un->semadj[semnum] = 0;
|
||||||
|
|
||||||
curr->semval = val;
|
curr->semval = val;
|
||||||
curr->sempid = task_tgid_vnr(current);
|
ipc_update_pid(&curr->sempid, task_tgid(current));
|
||||||
sma->sem_ctime = ktime_get_real_seconds();
|
sma->sem_ctime = ktime_get_real_seconds();
|
||||||
/* maybe some queued-up processes were waiting for this */
|
/* maybe some queued-up processes were waiting for this */
|
||||||
do_smart_update(sma, NULL, 0, 0, &wake_q);
|
do_smart_update(sma, NULL, 0, 0, &wake_q);
|
||||||
@ -1473,7 +1475,7 @@ static int semctl_main(struct ipc_namespace *ns, int semid, int semnum,
|
|||||||
|
|
||||||
for (i = 0; i < nsems; i++) {
|
for (i = 0; i < nsems; i++) {
|
||||||
sma->sems[i].semval = sem_io[i];
|
sma->sems[i].semval = sem_io[i];
|
||||||
sma->sems[i].sempid = task_tgid_vnr(current);
|
ipc_update_pid(&sma->sems[i].sempid, task_tgid(current));
|
||||||
}
|
}
|
||||||
|
|
||||||
ipc_assert_locked_object(&sma->sem_perm);
|
ipc_assert_locked_object(&sma->sem_perm);
|
||||||
@ -1505,7 +1507,7 @@ static int semctl_main(struct ipc_namespace *ns, int semid, int semnum,
|
|||||||
err = curr->semval;
|
err = curr->semval;
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
case GETPID:
|
case GETPID:
|
||||||
err = curr->sempid;
|
err = pid_vnr(curr->sempid);
|
||||||
goto out_unlock;
|
goto out_unlock;
|
||||||
case GETNCNT:
|
case GETNCNT:
|
||||||
err = count_semcnt(sma, semnum, 0);
|
err = count_semcnt(sma, semnum, 0);
|
||||||
@ -2024,7 +2026,7 @@ static long do_semtimedop(int semid, struct sembuf __user *tsops,
|
|||||||
queue.sops = sops;
|
queue.sops = sops;
|
||||||
queue.nsops = nsops;
|
queue.nsops = nsops;
|
||||||
queue.undo = un;
|
queue.undo = un;
|
||||||
queue.pid = task_tgid_vnr(current);
|
queue.pid = task_tgid(current);
|
||||||
queue.alter = alter;
|
queue.alter = alter;
|
||||||
queue.dupsop = dupsop;
|
queue.dupsop = dupsop;
|
||||||
|
|
||||||
@ -2318,7 +2320,7 @@ void exit_sem(struct task_struct *tsk)
|
|||||||
semaphore->semval = 0;
|
semaphore->semval = 0;
|
||||||
if (semaphore->semval > SEMVMX)
|
if (semaphore->semval > SEMVMX)
|
||||||
semaphore->semval = SEMVMX;
|
semaphore->semval = SEMVMX;
|
||||||
semaphore->sempid = task_tgid_vnr(current);
|
ipc_update_pid(&semaphore->sempid, task_tgid(current));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* maybe some queued-up processes were waiting for this */
|
/* maybe some queued-up processes were waiting for this */
|
||||||
|
Loading…
Reference in New Issue
Block a user