Whamcloud - gitweb
d9921afdaf3e13cc6b0c74fdd173373a0772503e
[fs/lustre-release.git] / lustre / mdd / mdd_lock.c
1 /*
2  * GPL HEADER START
3  *
4  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License version 2 only,
8  * as published by the Free Software Foundation.
9  *
10  * This program is distributed in the hope that it will be useful, but
11  * WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * General Public License version 2 for more details (a copy is included
14  * in the LICENSE file that accompanied this code).
15  *
16  * You should have received a copy of the GNU General Public License
17  * version 2 along with this program; If not, see
18  * http://www.sun.com/software/products/lustre/docs/GPLv2.pdf
19  *
20  * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
21  * CA 95054 USA or visit www.sun.com if you need additional information or
22  * have any questions.
23  *
24  * GPL HEADER END
25  */
26 /*
27  * Copyright (c) 2007, 2010, Oracle and/or its affiliates. All rights reserved.
28  * Use is subject to license terms.
29  *
30  * Copyright (c) 2012, Intel Corporation.
31  */
32 /*
33  * This file is part of Lustre, http://www.lustre.org/
34  * Lustre is a trademark of Sun Microsystems, Inc.
35  *
36  * lustre/mdd/mdd_lock.c
37  *
38  * Lustre Metadata Server (mdd) routines
39  *
40  * Author: Mike Pershin <tappro@clusterfs.com>
41  */
42
43 #define DEBUG_SUBSYSTEM S_MDS
44
45 #include <linux/module.h>
46 #include <lustre_ver.h>
47 #include "mdd_internal.h"
48
49 void mdd_write_lock(const struct lu_env *env, struct mdd_object *obj,
50                     enum mdd_object_role role)
51 {
52         struct dt_object  *next = mdd_object_child(obj);
53
54         next->do_ops->do_write_lock(env, next, role);
55 }
56
57 void mdd_read_lock(const struct lu_env *env, struct mdd_object *obj,
58                    enum mdd_object_role role)
59 {
60         struct dt_object  *next = mdd_object_child(obj);
61
62         next->do_ops->do_read_lock(env, next, role);
63 }
64
65 void mdd_write_unlock(const struct lu_env *env, struct mdd_object *obj)
66 {
67         struct dt_object  *next = mdd_object_child(obj);
68
69         next->do_ops->do_write_unlock(env, next);
70 }
71
72 void mdd_read_unlock(const struct lu_env *env, struct mdd_object *obj)
73 {
74         struct dt_object  *next = mdd_object_child(obj);
75
76         next->do_ops->do_read_unlock(env, next);
77 }
78
79 int mdd_write_locked(const struct lu_env *env, struct mdd_object *obj)
80 {
81         struct dt_object  *next = mdd_object_child(obj);
82
83         return next->do_ops->do_write_locked(env, next);
84 }
85
86 unsigned long mdd_name2hash(const char *name)
87 {
88         return full_name_hash((unsigned char*)name, strlen(name));
89 }
90
91 /* Methods for parallel directory locking */
92 #if MDD_DISABLE_PDO_LOCK
93
94 static void *pdo_handle = (void *)0xbabecafe;
95
96 void mdd_pdlock_init(struct mdd_object *obj)
97 {
98 }
99
100 void *mdd_pdo_write_lock(const struct lu_env *env, struct mdd_object *obj,
101                          const char *name, enum mdd_object_role role)
102 {
103         return pdo_handle;
104 }
105
106 void *mdd_pdo_read_lock(const struct lu_env *env, struct mdd_object *obj,
107                         const char *name, enum mdd_object_role role)
108 {
109         return pdo_handle;
110 }
111
112 void mdd_pdo_write_unlock(const struct lu_env *env, struct mdd_object *obj,
113                           void *dlh)
114 {
115         LASSERT(dlh == pdo_handle);
116 }
117
118 void mdd_pdo_read_unlock(const struct lu_env *env, struct mdd_object *obj,
119                          void *dlh)
120 {
121         LASSERT(dlh == pdo_handle);
122 }
123
124 #else /* !MDD_DISABLE_PDO_LOCK */
125
126 #ifdef CONFIG_LOCKDEP
127 static struct lock_class_key mdd_pdirop_key;
128
129 #define RETIP ((unsigned long)__builtin_return_address(0))
130
131 static void mdd_lockdep_init(struct mdd_object *obj)
132 {
133         lockdep_set_class_and_name(obj, &mdd_pdirop_key, "pdir");
134 }
135
136 static void mdd_lockdep_pd_acquire(struct mdd_object *obj,
137                                    enum mdd_object_role role)
138 {
139         lock_map_acquire(&obj->dep_map);
140 }
141
142 static void mdd_lockdep_pd_release(struct mdd_object *obj)
143 {
144         lock_map_release(&obj->dep_map);
145 }
146
147 #else /* !CONFIG_LOCKDEP */
148
149 static void mdd_lockdep_init(struct mdd_object *obj)
150 {}
151 static void mdd_lockdep_pd_acquire(struct mdd_object *obj,
152                                    enum mdd_object_role role)
153 {}
154 static void mdd_lockdep_pd_release(struct mdd_object *obj)
155 {}
156
157 #endif /* !CONFIG_LOCKDEP */
158
159 void mdd_pdlock_init(struct mdd_object *obj)
160 {
161         dynlock_init(&obj->mod_pdlock);
162         mdd_lockdep_init(obj);
163 }
164
165 void *mdd_pdo_write_lock(const struct lu_env *env, struct mdd_object *obj,
166                          const char *name, enum mdd_object_role role)
167 {
168         struct dynlock_handle *handle;
169         unsigned long value = mdd_name2hash(name);
170
171         handle = dynlock_lock(&obj->mod_pdlock, value, DLT_WRITE, GFP_NOFS);
172         if (handle != NULL)
173                 mdd_lockdep_pd_acquire(obj, role);
174         return handle;
175 }
176
177 void *mdd_pdo_read_lock(const struct lu_env *env, struct mdd_object *obj,
178                         const char *name, enum mdd_object_role role)
179 {
180         struct dynlock_handle *handle;
181         unsigned long value = mdd_name2hash(name);
182         handle = dynlock_lock(&obj->mod_pdlock, value, DLT_READ, GFP_NOFS);
183         if (handle != NULL)
184                 mdd_lockdep_pd_acquire(obj, role);
185         return handle;
186 }
187
188 void mdd_pdo_write_unlock(const struct lu_env *env, struct mdd_object *obj,
189                           void *dlh)
190 {
191         mdd_lockdep_pd_release(obj);
192         return dynlock_unlock(&obj->mod_pdlock, dlh);
193 }
194
195 void mdd_pdo_read_unlock(const struct lu_env *env, struct mdd_object *obj,
196                          void *dlh)
197 {
198         mdd_lockdep_pd_release(obj);
199         return dynlock_unlock(&obj->mod_pdlock, dlh);
200 }
201
202 #endif /* MDD_DISABLE_PDO_LOCK */