1 /* kafsasyncd.c: AFS asynchronous operation daemon
 
   3  * Copyright (C) 2002 Red Hat, Inc. All Rights Reserved.
 
   4  * Written by David Howells (dhowells@redhat.com)
 
   6  * This program is free software; you can redistribute it and/or
 
   7  * modify it under the terms of the GNU General Public License
 
   8  * as published by the Free Software Foundation; either version
 
   9  * 2 of the License, or (at your option) any later version.
 
  12  * The AFS async daemon is used to the following:
 
  13  * - probe "dead" servers to see whether they've come back to life yet.
 
  14  * - probe "live" servers that we haven't talked to for a while to see if they are better
 
  15  *   candidates for serving than what we're currently using
 
  16  * - poll volume location servers to keep up to date volume location lists
 
  19 #include <linux/module.h>
 
  20 #include <linux/init.h>
 
  21 #include <linux/sched.h>
 
  22 #include <linux/completion.h>
 
  26 #include "kafsasyncd.h"
 
  27 #include "kafstimod.h"
 
  28 #include <rxrpc/call.h>
 
  29 #include <asm/errno.h>
 
  32 static DECLARE_COMPLETION(kafsasyncd_alive);
 
  33 static DECLARE_COMPLETION(kafsasyncd_dead);
 
  34 static DECLARE_WAIT_QUEUE_HEAD(kafsasyncd_sleepq);
 
  35 static struct task_struct *kafsasyncd_task;
 
  36 static int kafsasyncd_die;
 
  38 static int kafsasyncd(void *arg);
 
  40 static LIST_HEAD(kafsasyncd_async_attnq);
 
  41 static LIST_HEAD(kafsasyncd_async_busyq);
 
  42 static DEFINE_SPINLOCK(kafsasyncd_async_lock);
 
  44 static void kafsasyncd_null_call_attn_func(struct rxrpc_call *call)
 
  48 static void kafsasyncd_null_call_error_func(struct rxrpc_call *call)
 
  52 /*****************************************************************************/
 
  54  * start the async daemon
 
  56 int afs_kafsasyncd_start(void)
 
  60         ret = kernel_thread(kafsasyncd, NULL, 0);
 
  64         wait_for_completion(&kafsasyncd_alive);
 
  67 } /* end afs_kafsasyncd_start() */
 
  69 /*****************************************************************************/
 
  71  * stop the async daemon
 
  73 void afs_kafsasyncd_stop(void)
 
  75         /* get rid of my daemon */
 
  77         wake_up(&kafsasyncd_sleepq);
 
  78         wait_for_completion(&kafsasyncd_dead);
 
  80 } /* end afs_kafsasyncd_stop() */
 
  82 /*****************************************************************************/
 
  86 static int kafsasyncd(void *arg)
 
  88         struct afs_async_op *op;
 
  91         DECLARE_WAITQUEUE(myself, current);
 
  93         kafsasyncd_task = current;
 
  95         printk("kAFS: Started kafsasyncd %d\n", current->pid);
 
  97         daemonize("kafsasyncd");
 
  99         complete(&kafsasyncd_alive);
 
 101         /* loop around looking for things to attend to */
 
 103                 set_current_state(TASK_INTERRUPTIBLE);
 
 104                 add_wait_queue(&kafsasyncd_sleepq, &myself);
 
 107                         if (!list_empty(&kafsasyncd_async_attnq) ||
 
 108                             signal_pending(current) ||
 
 113                         set_current_state(TASK_INTERRUPTIBLE);
 
 116                 remove_wait_queue(&kafsasyncd_sleepq, &myself);
 
 117                 set_current_state(TASK_RUNNING);
 
 121                 /* discard pending signals */
 
 122                 afs_discard_my_signals();
 
 124                 die = kafsasyncd_die;
 
 126                 /* deal with the next asynchronous operation requiring
 
 128                 if (!list_empty(&kafsasyncd_async_attnq)) {
 
 129                         struct afs_async_op *op;
 
 131                         _debug("@@@ Begin Asynchronous Operation");
 
 134                         spin_lock(&kafsasyncd_async_lock);
 
 136                         if (!list_empty(&kafsasyncd_async_attnq)) {
 
 137                                 op = list_entry(kafsasyncd_async_attnq.next,
 
 138                                                 struct afs_async_op, link);
 
 140                                 list_add_tail(&op->link,
 
 141                                               &kafsasyncd_async_busyq);
 
 144                         spin_unlock(&kafsasyncd_async_lock);
 
 146                         _debug("@@@ Operation %p {%p}\n",
 
 147                                op, op ? op->ops : NULL);
 
 152                         _debug("@@@ End Asynchronous Operation");
 
 157         /* need to kill all outstanding asynchronous operations before
 
 159         kafsasyncd_task = NULL;
 
 160         spin_lock(&kafsasyncd_async_lock);
 
 162         /* fold the busy and attention queues together */
 
 163         list_splice_init(&kafsasyncd_async_busyq,
 
 164                          &kafsasyncd_async_attnq);
 
 166         /* dequeue kafsasyncd from all their wait queues */
 
 167         list_for_each_entry(op, &kafsasyncd_async_attnq, link) {
 
 168                 op->call->app_attn_func = kafsasyncd_null_call_attn_func;
 
 169                 op->call->app_error_func = kafsasyncd_null_call_error_func;
 
 170                 remove_wait_queue(&op->call->waitq, &op->waiter);
 
 173         spin_unlock(&kafsasyncd_async_lock);
 
 175         /* abort all the operations */
 
 176         while (!list_empty(&kafsasyncd_async_attnq)) {
 
 177                 op = list_entry(kafsasyncd_async_attnq.next, struct afs_async_op, link);
 
 178                 list_del_init(&op->link);
 
 180                 rxrpc_call_abort(op->call, -EIO);
 
 181                 rxrpc_put_call(op->call);
 
 184                 op->ops->discard(op);
 
 189         complete_and_exit(&kafsasyncd_dead, 0);
 
 191 } /* end kafsasyncd() */
 
 193 /*****************************************************************************/
 
 196  * - place operation on busy queue
 
 198 void afs_kafsasyncd_begin_op(struct afs_async_op *op)
 
 202         spin_lock(&kafsasyncd_async_lock);
 
 204         init_waitqueue_entry(&op->waiter, kafsasyncd_task);
 
 205         add_wait_queue(&op->call->waitq, &op->waiter);
 
 208         list_add_tail(&op->link, &kafsasyncd_async_busyq);
 
 210         spin_unlock(&kafsasyncd_async_lock);
 
 213 } /* end afs_kafsasyncd_begin_op() */
 
 215 /*****************************************************************************/
 
 217  * request attention for an operation
 
 218  * - move to attention queue
 
 220 void afs_kafsasyncd_attend_op(struct afs_async_op *op)
 
 224         spin_lock(&kafsasyncd_async_lock);
 
 227         list_add_tail(&op->link, &kafsasyncd_async_attnq);
 
 229         spin_unlock(&kafsasyncd_async_lock);
 
 231         wake_up(&kafsasyncd_sleepq);
 
 234 } /* end afs_kafsasyncd_attend_op() */
 
 236 /*****************************************************************************/
 
 238  * terminate an operation
 
 239  * - remove from either queue
 
 241 void afs_kafsasyncd_terminate_op(struct afs_async_op *op)
 
 245         spin_lock(&kafsasyncd_async_lock);
 
 247         if (!list_empty(&op->link)) {
 
 248                 list_del_init(&op->link);
 
 249                 remove_wait_queue(&op->call->waitq, &op->waiter);
 
 252         spin_unlock(&kafsasyncd_async_lock);
 
 254         wake_up(&kafsasyncd_sleepq);
 
 257 } /* end afs_kafsasyncd_terminate_op() */