Whamcloud - gitweb
LU-2558 test: recovery-small 19a FAIL no eviction
[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 #ifdef HAVE_LOCK_MAP_ACQUIRE
140         lock_map_acquire(&obj->dep_map);
141 #else
142         lock_acquire(&obj->dep_map, role, 0, 1, 2, RETIP);
143 #endif
144 }
145
146 static void mdd_lockdep_pd_release(struct mdd_object *obj)
147 {
148 #ifdef HAVE_LOCK_MAP_ACQUIRE
149         lock_map_release(&obj->dep_map);
150 #else
151         lock_release(&obj->dep_map, 0, RETIP);
152 #endif
153 }
154
155 #else /* !CONFIG_LOCKDEP */
156
157 static void mdd_lockdep_init(struct mdd_object *obj)
158 {}
159 static void mdd_lockdep_pd_acquire(struct mdd_object *obj,
160                                    enum mdd_object_role role)
161 {}
162 static void mdd_lockdep_pd_release(struct mdd_object *obj)
163 {}
164
165 #endif /* !CONFIG_LOCKDEP */
166
167 void mdd_pdlock_init(struct mdd_object *obj)
168 {
169         dynlock_init(&obj->mod_pdlock);
170         mdd_lockdep_init(obj);
171 }
172
173 void *mdd_pdo_write_lock(const struct lu_env *env, struct mdd_object *obj,
174                          const char *name, enum mdd_object_role role)
175 {
176         struct dynlock_handle *handle;
177         unsigned long value = mdd_name2hash(name);
178
179         handle = dynlock_lock(&obj->mod_pdlock, value, DLT_WRITE, GFP_NOFS);
180         if (handle != NULL)
181                 mdd_lockdep_pd_acquire(obj, role);
182         return handle;
183 }
184
185 void *mdd_pdo_read_lock(const struct lu_env *env, struct mdd_object *obj,
186                         const char *name, enum mdd_object_role role)
187 {
188         struct dynlock_handle *handle;
189         unsigned long value = mdd_name2hash(name);
190         handle = dynlock_lock(&obj->mod_pdlock, value, DLT_READ, GFP_NOFS);
191         if (handle != NULL)
192                 mdd_lockdep_pd_acquire(obj, role);
193         return handle;
194 }
195
196 void mdd_pdo_write_unlock(const struct lu_env *env, struct mdd_object *obj,
197                           void *dlh)
198 {
199         mdd_lockdep_pd_release(obj);
200         return dynlock_unlock(&obj->mod_pdlock, dlh);
201 }
202
203 void mdd_pdo_read_unlock(const struct lu_env *env, struct mdd_object *obj,
204                          void *dlh)
205 {
206         mdd_lockdep_pd_release(obj);
207         return dynlock_unlock(&obj->mod_pdlock, dlh);
208 }
209
210 #endif /* MDD_DISABLE_PDO_LOCK */