]> bbs.cooldavid.org Git - net-next-2.6.git/blobdiff - fs/fscache/main.c
fscache: drop references to slow-work
[net-next-2.6.git] / fs / fscache / main.c
index add6bdb53f04400779ca899370d07bd2480e108a..500936d9fff21f222069f05ccbbca4136618e49f 100644 (file)
@@ -15,6 +15,7 @@
 #include <linux/sched.h>
 #include <linux/completion.h>
 #include <linux/slab.h>
+#include <linux/seq_file.h>
 #include "internal.h"
 
 MODULE_DESCRIPTION("FS Cache Manager");
@@ -40,22 +41,105 @@ MODULE_PARM_DESC(fscache_debug,
                 "FS-Cache debugging mask");
 
 struct kobject *fscache_root;
+struct workqueue_struct *fscache_object_wq;
+struct workqueue_struct *fscache_op_wq;
+
+DEFINE_PER_CPU(wait_queue_head_t, fscache_object_cong_wait);
+
+/* these values serve as lower bounds, will be adjusted in fscache_init() */
+static unsigned fscache_object_max_active = 4;
+static unsigned fscache_op_max_active = 2;
+
+#ifdef CONFIG_SYSCTL
+static struct ctl_table_header *fscache_sysctl_header;
+
+static int fscache_max_active_sysctl(struct ctl_table *table, int write,
+                                    void __user *buffer,
+                                    size_t *lenp, loff_t *ppos)
+{
+       struct workqueue_struct **wqp = table->extra1;
+       unsigned int *datap = table->data;
+       int ret;
+
+       ret = proc_dointvec(table, write, buffer, lenp, ppos);
+       if (ret == 0)
+               workqueue_set_max_active(*wqp, *datap);
+       return ret;
+}
+
+ctl_table fscache_sysctls[] = {
+       {
+               .procname       = "object_max_active",
+               .data           = &fscache_object_max_active,
+               .maxlen         = sizeof(unsigned),
+               .mode           = 0644,
+               .proc_handler   = fscache_max_active_sysctl,
+               .extra1         = &fscache_object_wq,
+       },
+       {
+               .procname       = "operation_max_active",
+               .data           = &fscache_op_max_active,
+               .maxlen         = sizeof(unsigned),
+               .mode           = 0644,
+               .proc_handler   = fscache_max_active_sysctl,
+               .extra1         = &fscache_op_wq,
+       },
+       {}
+};
+
+ctl_table fscache_sysctls_root[] = {
+       {
+               .procname       = "fscache",
+               .mode           = 0555,
+               .child          = fscache_sysctls,
+       },
+       {}
+};
+#endif
 
 /*
  * initialise the fs caching module
  */
 static int __init fscache_init(void)
 {
+       unsigned int nr_cpus = num_possible_cpus();
+       unsigned int cpu;
        int ret;
 
-       ret = slow_work_register_user(THIS_MODULE);
-       if (ret < 0)
-               goto error_slow_work;
+       fscache_object_max_active =
+               clamp_val(nr_cpus,
+                         fscache_object_max_active, WQ_UNBOUND_MAX_ACTIVE);
+
+       ret = -ENOMEM;
+       fscache_object_wq = alloc_workqueue("fscache_object", WQ_UNBOUND,
+                                           fscache_object_max_active);
+       if (!fscache_object_wq)
+               goto error_object_wq;
+
+       fscache_op_max_active =
+               clamp_val(fscache_object_max_active / 2,
+                         fscache_op_max_active, WQ_UNBOUND_MAX_ACTIVE);
+
+       ret = -ENOMEM;
+       fscache_op_wq = alloc_workqueue("fscache_operation", WQ_UNBOUND,
+                                       fscache_op_max_active);
+       if (!fscache_op_wq)
+               goto error_op_wq;
+
+       for_each_possible_cpu(cpu)
+               init_waitqueue_head(&per_cpu(fscache_object_cong_wait, cpu));
 
        ret = fscache_proc_init();
        if (ret < 0)
                goto error_proc;
 
+#ifdef CONFIG_SYSCTL
+       ret = -ENOMEM;
+       fscache_sysctl_header = register_sysctl_table(fscache_sysctls_root);
+       if (!fscache_sysctl_header)
+               goto error_sysctl;
+#endif
+
        fscache_cookie_jar = kmem_cache_create("fscache_cookie_jar",
                                               sizeof(struct fscache_cookie),
                                               0,
@@ -78,10 +162,16 @@ static int __init fscache_init(void)
 error_kobj:
        kmem_cache_destroy(fscache_cookie_jar);
 error_cookie_jar:
+#ifdef CONFIG_SYSCTL
+       unregister_sysctl_table(fscache_sysctl_header);
+error_sysctl:
+#endif
        fscache_proc_cleanup();
 error_proc:
-       slow_work_unregister_user(THIS_MODULE);
-error_slow_work:
+       destroy_workqueue(fscache_op_wq);
+error_op_wq:
+       destroy_workqueue(fscache_object_wq);
+error_object_wq:
        return ret;
 }
 
@@ -96,8 +186,10 @@ static void __exit fscache_exit(void)
 
        kobject_put(fscache_root);
        kmem_cache_destroy(fscache_cookie_jar);
+       unregister_sysctl_table(fscache_sysctl_header);
        fscache_proc_cleanup();
-       slow_work_unregister_user(THIS_MODULE);
+       destroy_workqueue(fscache_op_wq);
+       destroy_workqueue(fscache_object_wq);
        printk(KERN_NOTICE "FS-Cache: Unloaded\n");
 }