1 /* -*- linux-c -*- --------------------------------------------------------- *
3 * linux/fs/autofs/waitq.c
5 * Copyright 1997-1998 Transmeta Corporation -- All Rights Reserved
7 * This file is part of the Linux kernel and is made available under
8 * the terms of the GNU General Public License, version 2, or at your
9 * option, any later version, incorporated herein by reference.
11 * ------------------------------------------------------------------------- */
13 #include <linux/slab.h>
14 #include <linux/time.h>
15 #include <linux/signal.h>
16 #include <linux/file.h>
19 /* We make this a static variable rather than a part of the superblock; it
20 is better if we don't reassign numbers easily even across filesystems */
21 static autofs_wqt_t autofs_next_wait_queue
= 1;
23 /* These are the signals we allow interrupting a pending mount */
24 #define SHUTDOWN_SIGS (sigmask(SIGKILL) | sigmask(SIGINT) | sigmask(SIGQUIT))
26 void autofs_catatonic_mode(struct autofs_sb_info
*sbi
)
28 struct autofs_wait_queue
*wq
, *nwq
;
30 DPRINTK(("autofs: entering catatonic mode\n"));
34 sbi
->queues
= NULL
; /* Erase all wait queues */
37 wq
->status
= -ENOENT
; /* Magic is gone - report failure */
43 fput(sbi
->pipe
); /* Close the pipe */
45 autofs_hash_dputall(&sbi
->dirhash
); /* Remove all dentry pointers */
48 static int autofs_write(struct file
*file
, const void *addr
, int bytes
)
50 unsigned long sigpipe
, flags
;
52 const char *data
= (const char *)addr
;
55 /** WARNING: this is not safe for writing more than PIPE_BUF bytes! **/
57 sigpipe
= sigismember(¤t
->pending
.signal
, SIGPIPE
);
59 /* Save pointer to user space and point back to kernel space */
64 (wr
= file
->f_op
->write(file
,data
,bytes
,&file
->f_pos
)) > 0) {
71 /* Keep the currently executing process from receiving a
72 SIGPIPE unless it was already supposed to get one */
73 if (wr
== -EPIPE
&& !sigpipe
) {
74 spin_lock_irqsave(¤t
->sighand
->siglock
, flags
);
75 sigdelset(¤t
->pending
.signal
, SIGPIPE
);
77 spin_unlock_irqrestore(¤t
->sighand
->siglock
, flags
);
83 static void autofs_notify_daemon(struct autofs_sb_info
*sbi
, struct autofs_wait_queue
*wq
)
85 struct autofs_packet_missing pkt
;
87 DPRINTK(("autofs_wait: wait id = 0x%08lx, name = ", wq
->wait_queue_token
));
88 autofs_say(wq
->name
,wq
->len
);
90 memset(&pkt
,0,sizeof pkt
); /* For security reasons */
92 pkt
.hdr
.proto_version
= AUTOFS_PROTO_VERSION
;
93 pkt
.hdr
.type
= autofs_ptype_missing
;
94 pkt
.wait_queue_token
= wq
->wait_queue_token
;
96 memcpy(pkt
.name
, wq
->name
, pkt
.len
);
97 pkt
.name
[pkt
.len
] = '\0';
99 if ( autofs_write(sbi
->pipe
,&pkt
,sizeof(struct autofs_packet_missing
)) )
100 autofs_catatonic_mode(sbi
);
103 int autofs_wait(struct autofs_sb_info
*sbi
, struct qstr
*name
)
105 struct autofs_wait_queue
*wq
;
108 /* In catatonic mode, we don't wait for nobody */
109 if ( sbi
->catatonic
)
112 /* We shouldn't be able to get here, but just in case */
113 if ( name
->len
> NAME_MAX
)
116 for ( wq
= sbi
->queues
; wq
; wq
= wq
->next
) {
117 if ( wq
->hash
== name
->hash
&&
118 wq
->len
== name
->len
&&
119 wq
->name
&& !memcmp(wq
->name
,name
->name
,name
->len
) )
124 /* Create a new wait queue */
125 wq
= kmalloc(sizeof(struct autofs_wait_queue
),GFP_KERNEL
);
129 wq
->name
= kmalloc(name
->len
,GFP_KERNEL
);
134 wq
->wait_queue_token
= autofs_next_wait_queue
++;
135 init_waitqueue_head(&wq
->queue
);
136 wq
->hash
= name
->hash
;
138 wq
->status
= -EINTR
; /* Status return if interrupted */
139 memcpy(wq
->name
, name
->name
, name
->len
);
140 wq
->next
= sbi
->queues
;
143 /* autofs_notify_daemon() may block */
145 autofs_notify_daemon(sbi
,wq
);
149 /* wq->name is NULL if and only if the lock is already released */
151 if ( sbi
->catatonic
) {
152 /* We might have slept, so check again for catatonic mode */
153 wq
->status
= -ENOENT
;
159 /* Block all but "shutdown" signals while waiting */
162 siginitsetinv(&sigmask
, SHUTDOWN_SIGS
);
163 sigprocmask(SIG_BLOCK
, &sigmask
, &sigmask
);
165 interruptible_sleep_on(&wq
->queue
);
167 sigprocmask(SIG_SETMASK
, &sigmask
, NULL
);
169 DPRINTK(("autofs_wait: skipped sleeping\n"));
174 if ( ! --wq
->wait_ctr
) /* Are we the last process to need status? */
181 int autofs_wait_release(struct autofs_sb_info
*sbi
, autofs_wqt_t wait_queue_token
, int status
)
183 struct autofs_wait_queue
*wq
, **wql
;
185 for (wql
= &sbi
->queues
; (wq
= *wql
) != NULL
; wql
= &wq
->next
) {
186 if ( wq
->wait_queue_token
== wait_queue_token
)
192 *wql
= wq
->next
; /* Unlink from chain */
194 wq
->name
= NULL
; /* Do not wait on this queue */
198 if ( ! --wq
->wait_ctr
) /* Is anyone still waiting for this guy? */