|
@@ -36,8 +36,10 @@ void autofs4_catatonic_mode(struct autofs_sb_info *sbi)
|
|
while (wq) {
|
|
while (wq) {
|
|
nwq = wq->next;
|
|
nwq = wq->next;
|
|
wq->status = -ENOENT; /* Magic is gone - report failure */
|
|
wq->status = -ENOENT; /* Magic is gone - report failure */
|
|
- kfree(wq->name);
|
|
|
|
- wq->name = NULL;
|
|
|
|
|
|
+ if (wq->name.name) {
|
|
|
|
+ kfree(wq->name.name);
|
|
|
|
+ wq->name.name = NULL;
|
|
|
|
+ }
|
|
wake_up_interruptible(&wq->queue);
|
|
wake_up_interruptible(&wq->queue);
|
|
wq = nwq;
|
|
wq = nwq;
|
|
}
|
|
}
|
|
@@ -92,7 +94,7 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi,
|
|
size_t pktsz;
|
|
size_t pktsz;
|
|
|
|
|
|
DPRINTK("wait id = 0x%08lx, name = %.*s, type=%d",
|
|
DPRINTK("wait id = 0x%08lx, name = %.*s, type=%d",
|
|
- wq->wait_queue_token, wq->len, wq->name, type);
|
|
|
|
|
|
+ wq->wait_queue_token, wq->name.len, wq->name.name, type);
|
|
|
|
|
|
memset(&pkt,0,sizeof pkt); /* For security reasons */
|
|
memset(&pkt,0,sizeof pkt); /* For security reasons */
|
|
|
|
|
|
@@ -107,9 +109,9 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi,
|
|
pktsz = sizeof(*mp);
|
|
pktsz = sizeof(*mp);
|
|
|
|
|
|
mp->wait_queue_token = wq->wait_queue_token;
|
|
mp->wait_queue_token = wq->wait_queue_token;
|
|
- mp->len = wq->len;
|
|
|
|
- memcpy(mp->name, wq->name, wq->len);
|
|
|
|
- mp->name[wq->len] = '\0';
|
|
|
|
|
|
+ mp->len = wq->name.len;
|
|
|
|
+ memcpy(mp->name, wq->name.name, wq->name.len);
|
|
|
|
+ mp->name[wq->name.len] = '\0';
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
case autofs_ptype_expire_multi:
|
|
case autofs_ptype_expire_multi:
|
|
@@ -119,9 +121,9 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi,
|
|
pktsz = sizeof(*ep);
|
|
pktsz = sizeof(*ep);
|
|
|
|
|
|
ep->wait_queue_token = wq->wait_queue_token;
|
|
ep->wait_queue_token = wq->wait_queue_token;
|
|
- ep->len = wq->len;
|
|
|
|
- memcpy(ep->name, wq->name, wq->len);
|
|
|
|
- ep->name[wq->len] = '\0';
|
|
|
|
|
|
+ ep->len = wq->name.len;
|
|
|
|
+ memcpy(ep->name, wq->name.name, wq->name.len);
|
|
|
|
+ ep->name[wq->name.len] = '\0';
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
/*
|
|
/*
|
|
@@ -138,9 +140,9 @@ static void autofs4_notify_daemon(struct autofs_sb_info *sbi,
|
|
pktsz = sizeof(*packet);
|
|
pktsz = sizeof(*packet);
|
|
|
|
|
|
packet->wait_queue_token = wq->wait_queue_token;
|
|
packet->wait_queue_token = wq->wait_queue_token;
|
|
- packet->len = wq->len;
|
|
|
|
- memcpy(packet->name, wq->name, wq->len);
|
|
|
|
- packet->name[wq->len] = '\0';
|
|
|
|
|
|
+ packet->len = wq->name.len;
|
|
|
|
+ memcpy(packet->name, wq->name.name, wq->name.len);
|
|
|
|
+ packet->name[wq->name.len] = '\0';
|
|
packet->dev = wq->dev;
|
|
packet->dev = wq->dev;
|
|
packet->ino = wq->ino;
|
|
packet->ino = wq->ino;
|
|
packet->uid = wq->uid;
|
|
packet->uid = wq->uid;
|
|
@@ -191,15 +193,15 @@ static int autofs4_getpath(struct autofs_sb_info *sbi,
|
|
}
|
|
}
|
|
|
|
|
|
static struct autofs_wait_queue *
|
|
static struct autofs_wait_queue *
|
|
-autofs4_find_wait(struct autofs_sb_info *sbi,
|
|
|
|
- char *name, unsigned int hash, unsigned int len)
|
|
|
|
|
|
+autofs4_find_wait(struct autofs_sb_info *sbi, struct qstr *qstr)
|
|
{
|
|
{
|
|
struct autofs_wait_queue *wq;
|
|
struct autofs_wait_queue *wq;
|
|
|
|
|
|
for (wq = sbi->queues; wq; wq = wq->next) {
|
|
for (wq = sbi->queues; wq; wq = wq->next) {
|
|
- if (wq->hash == hash &&
|
|
|
|
- wq->len == len &&
|
|
|
|
- wq->name && !memcmp(wq->name, name, len))
|
|
|
|
|
|
+ if (wq->name.hash == qstr->hash &&
|
|
|
|
+ wq->name.len == qstr->len &&
|
|
|
|
+ wq->name.name &&
|
|
|
|
+ !memcmp(wq->name.name, qstr->name, qstr->len))
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
return wq;
|
|
return wq;
|
|
@@ -210,9 +212,8 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
{
|
|
{
|
|
struct autofs_info *ino;
|
|
struct autofs_info *ino;
|
|
struct autofs_wait_queue *wq;
|
|
struct autofs_wait_queue *wq;
|
|
|
|
+ struct qstr qstr;
|
|
char *name;
|
|
char *name;
|
|
- unsigned int len = 0;
|
|
|
|
- unsigned int hash = 0;
|
|
|
|
int status, type;
|
|
int status, type;
|
|
|
|
|
|
/* In catatonic mode, we don't wait for nobody */
|
|
/* In catatonic mode, we don't wait for nobody */
|
|
@@ -225,22 +226,23 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
|
|
|
|
/* If this is a direct mount request create a dummy name */
|
|
/* If this is a direct mount request create a dummy name */
|
|
if (IS_ROOT(dentry) && (sbi->type & AUTOFS_TYPE_DIRECT))
|
|
if (IS_ROOT(dentry) && (sbi->type & AUTOFS_TYPE_DIRECT))
|
|
- len = sprintf(name, "%p", dentry);
|
|
|
|
|
|
+ qstr.len = sprintf(name, "%p", dentry);
|
|
else {
|
|
else {
|
|
- len = autofs4_getpath(sbi, dentry, &name);
|
|
|
|
- if (!len) {
|
|
|
|
|
|
+ qstr.len = autofs4_getpath(sbi, dentry, &name);
|
|
|
|
+ if (!qstr.len) {
|
|
kfree(name);
|
|
kfree(name);
|
|
return -ENOENT;
|
|
return -ENOENT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- hash = full_name_hash(name, len);
|
|
|
|
|
|
+ qstr.name = name;
|
|
|
|
+ qstr.hash = full_name_hash(name, qstr.len);
|
|
|
|
|
|
if (mutex_lock_interruptible(&sbi->wq_mutex)) {
|
|
if (mutex_lock_interruptible(&sbi->wq_mutex)) {
|
|
- kfree(name);
|
|
|
|
|
|
+ kfree(qstr.name);
|
|
return -EINTR;
|
|
return -EINTR;
|
|
}
|
|
}
|
|
|
|
|
|
- wq = autofs4_find_wait(sbi, name, hash, len);
|
|
|
|
|
|
+ wq = autofs4_find_wait(sbi, &qstr);
|
|
ino = autofs4_dentry_ino(dentry);
|
|
ino = autofs4_dentry_ino(dentry);
|
|
if (!wq && ino && notify == NFY_NONE) {
|
|
if (!wq && ino && notify == NFY_NONE) {
|
|
/*
|
|
/*
|
|
@@ -254,10 +256,10 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
schedule_timeout_interruptible(HZ/10);
|
|
schedule_timeout_interruptible(HZ/10);
|
|
if (mutex_lock_interruptible(&sbi->wq_mutex)) {
|
|
if (mutex_lock_interruptible(&sbi->wq_mutex)) {
|
|
- kfree(name);
|
|
|
|
|
|
+ kfree(qstr.name);
|
|
return -EINTR;
|
|
return -EINTR;
|
|
}
|
|
}
|
|
- wq = autofs4_find_wait(sbi, name, hash, len);
|
|
|
|
|
|
+ wq = autofs4_find_wait(sbi, &qstr);
|
|
if (wq)
|
|
if (wq)
|
|
break;
|
|
break;
|
|
}
|
|
}
|
|
@@ -268,7 +270,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
* return status of the wait.
|
|
* return status of the wait.
|
|
*/
|
|
*/
|
|
if (!wq) {
|
|
if (!wq) {
|
|
- kfree(name);
|
|
|
|
|
|
+ kfree(qstr.name);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
@@ -278,7 +280,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
/* Create a new wait queue */
|
|
/* Create a new wait queue */
|
|
wq = kmalloc(sizeof(struct autofs_wait_queue),GFP_KERNEL);
|
|
wq = kmalloc(sizeof(struct autofs_wait_queue),GFP_KERNEL);
|
|
if (!wq) {
|
|
if (!wq) {
|
|
- kfree(name);
|
|
|
|
|
|
+ kfree(qstr.name);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
return -ENOMEM;
|
|
return -ENOMEM;
|
|
}
|
|
}
|
|
@@ -289,9 +291,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
wq->next = sbi->queues;
|
|
wq->next = sbi->queues;
|
|
sbi->queues = wq;
|
|
sbi->queues = wq;
|
|
init_waitqueue_head(&wq->queue);
|
|
init_waitqueue_head(&wq->queue);
|
|
- wq->hash = hash;
|
|
|
|
- wq->name = name;
|
|
|
|
- wq->len = len;
|
|
|
|
|
|
+ memcpy(&wq->name, &qstr, sizeof(struct qstr));
|
|
wq->dev = autofs4_get_dev(sbi);
|
|
wq->dev = autofs4_get_dev(sbi);
|
|
wq->ino = autofs4_get_ino(sbi);
|
|
wq->ino = autofs4_get_ino(sbi);
|
|
wq->uid = current->uid;
|
|
wq->uid = current->uid;
|
|
@@ -319,16 +319,18 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
}
|
|
}
|
|
|
|
|
|
DPRINTK("new wait id = 0x%08lx, name = %.*s, nfy=%d\n",
|
|
DPRINTK("new wait id = 0x%08lx, name = %.*s, nfy=%d\n",
|
|
- (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify);
|
|
|
|
|
|
+ (unsigned long) wq->wait_queue_token, wq->name.len,
|
|
|
|
+ wq->name.name, notify);
|
|
|
|
|
|
/* autofs4_notify_daemon() may block */
|
|
/* autofs4_notify_daemon() may block */
|
|
autofs4_notify_daemon(sbi, wq, type);
|
|
autofs4_notify_daemon(sbi, wq, type);
|
|
} else {
|
|
} else {
|
|
atomic_inc(&wq->wait_ctr);
|
|
atomic_inc(&wq->wait_ctr);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
- kfree(name);
|
|
|
|
|
|
+ kfree(qstr.name);
|
|
DPRINTK("existing wait id = 0x%08lx, name = %.*s, nfy=%d",
|
|
DPRINTK("existing wait id = 0x%08lx, name = %.*s, nfy=%d",
|
|
- (unsigned long) wq->wait_queue_token, wq->len, wq->name, notify);
|
|
|
|
|
|
+ (unsigned long) wq->wait_queue_token, wq->name.len,
|
|
|
|
+ wq->name.name, notify);
|
|
}
|
|
}
|
|
|
|
|
|
/* wq->name is NULL if and only if the lock is already released */
|
|
/* wq->name is NULL if and only if the lock is already released */
|
|
@@ -336,11 +338,13 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
if (sbi->catatonic) {
|
|
if (sbi->catatonic) {
|
|
/* We might have slept, so check again for catatonic mode */
|
|
/* We might have slept, so check again for catatonic mode */
|
|
wq->status = -ENOENT;
|
|
wq->status = -ENOENT;
|
|
- kfree(wq->name);
|
|
|
|
- wq->name = NULL;
|
|
|
|
|
|
+ if (wq->name.name) {
|
|
|
|
+ kfree(wq->name.name);
|
|
|
|
+ wq->name.name = NULL;
|
|
|
|
+ }
|
|
}
|
|
}
|
|
|
|
|
|
- if (wq->name) {
|
|
|
|
|
|
+ if (wq->name.name) {
|
|
/* Block all but "shutdown" signals while waiting */
|
|
/* Block all but "shutdown" signals while waiting */
|
|
sigset_t oldset;
|
|
sigset_t oldset;
|
|
unsigned long irqflags;
|
|
unsigned long irqflags;
|
|
@@ -351,7 +355,7 @@ int autofs4_wait(struct autofs_sb_info *sbi, struct dentry *dentry,
|
|
recalc_sigpending();
|
|
recalc_sigpending();
|
|
spin_unlock_irqrestore(¤t->sighand->siglock, irqflags);
|
|
spin_unlock_irqrestore(¤t->sighand->siglock, irqflags);
|
|
|
|
|
|
- wait_event_interruptible(wq->queue, wq->name == NULL);
|
|
|
|
|
|
+ wait_event_interruptible(wq->queue, wq->name.name == NULL);
|
|
|
|
|
|
spin_lock_irqsave(¤t->sighand->siglock, irqflags);
|
|
spin_lock_irqsave(¤t->sighand->siglock, irqflags);
|
|
current->blocked = oldset;
|
|
current->blocked = oldset;
|
|
@@ -388,8 +392,8 @@ int autofs4_wait_release(struct autofs_sb_info *sbi, autofs_wqt_t wait_queue_tok
|
|
|
|
|
|
*wql = wq->next; /* Unlink from chain */
|
|
*wql = wq->next; /* Unlink from chain */
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
mutex_unlock(&sbi->wq_mutex);
|
|
- kfree(wq->name);
|
|
|
|
- wq->name = NULL; /* Do not wait on this queue */
|
|
|
|
|
|
+ kfree(wq->name.name);
|
|
|
|
+ wq->name.name = NULL; /* Do not wait on this queue */
|
|
|
|
|
|
wq->status = status;
|
|
wq->status = status;
|
|
|
|
|