1 #include <linux/ceph/ceph_debug.h>
3 #include <linux/types.h>
4 #include <linux/slab.h>
6 #include <linux/ceph/cls_lock_client.h>
7 #include <linux/ceph/decode.h>
10 * ceph_cls_lock - grab rados lock for object
11 * @oid, @oloc: object to lock
12 * @lock_name: the name of the lock
13 * @type: lock type (CEPH_CLS_LOCK_EXCLUSIVE or CEPH_CLS_LOCK_SHARED)
14 * @cookie: user-defined identifier for this instance of the lock
15 * @tag: user-defined tag
16 * @desc: user-defined lock description
19 * All operations on the same lock should use the same tag.
21 int ceph_cls_lock(struct ceph_osd_client
*osdc
,
22 struct ceph_object_id
*oid
,
23 struct ceph_object_locator
*oloc
,
24 char *lock_name
, u8 type
, char *cookie
,
25 char *tag
, char *desc
, u8 flags
)
28 int name_len
= strlen(lock_name
);
29 int cookie_len
= strlen(cookie
);
30 int tag_len
= strlen(tag
);
31 int desc_len
= strlen(desc
);
33 struct page
*lock_op_page
;
34 struct timespec mtime
;
37 lock_op_buf_size
= name_len
+ sizeof(__le32
) +
38 cookie_len
+ sizeof(__le32
) +
39 tag_len
+ sizeof(__le32
) +
40 desc_len
+ sizeof(__le32
) +
41 sizeof(struct ceph_timespec
) +
43 sizeof(u8
) + sizeof(u8
) +
44 CEPH_ENCODING_START_BLK_LEN
;
45 if (lock_op_buf_size
> PAGE_SIZE
)
48 lock_op_page
= alloc_page(GFP_NOIO
);
52 p
= page_address(lock_op_page
);
53 end
= p
+ lock_op_buf_size
;
55 /* encode cls_lock_lock_op struct */
56 ceph_start_encoding(&p
, 1, 1,
57 lock_op_buf_size
- CEPH_ENCODING_START_BLK_LEN
);
58 ceph_encode_string(&p
, end
, lock_name
, name_len
);
59 ceph_encode_8(&p
, type
);
60 ceph_encode_string(&p
, end
, cookie
, cookie_len
);
61 ceph_encode_string(&p
, end
, tag
, tag_len
);
62 ceph_encode_string(&p
, end
, desc
, desc_len
);
63 /* only support infinite duration */
64 memset(&mtime
, 0, sizeof(mtime
));
65 ceph_encode_timespec(p
, &mtime
);
66 p
+= sizeof(struct ceph_timespec
);
67 ceph_encode_8(&p
, flags
);
69 dout("%s lock_name %s type %d cookie %s tag %s desc %s flags 0x%x\n",
70 __func__
, lock_name
, type
, cookie
, tag
, desc
, flags
);
71 ret
= ceph_osdc_call(osdc
, oid
, oloc
, "lock", "lock",
72 CEPH_OSD_FLAG_WRITE
, lock_op_page
,
73 lock_op_buf_size
, NULL
, NULL
);
75 dout("%s: status %d\n", __func__
, ret
);
76 __free_page(lock_op_page
);
79 EXPORT_SYMBOL(ceph_cls_lock
);
82 * ceph_cls_unlock - release rados lock for object
83 * @oid, @oloc: object to lock
84 * @lock_name: the name of the lock
85 * @cookie: user-defined identifier for this instance of the lock
87 int ceph_cls_unlock(struct ceph_osd_client
*osdc
,
88 struct ceph_object_id
*oid
,
89 struct ceph_object_locator
*oloc
,
90 char *lock_name
, char *cookie
)
92 int unlock_op_buf_size
;
93 int name_len
= strlen(lock_name
);
94 int cookie_len
= strlen(cookie
);
96 struct page
*unlock_op_page
;
99 unlock_op_buf_size
= name_len
+ sizeof(__le32
) +
100 cookie_len
+ sizeof(__le32
) +
101 CEPH_ENCODING_START_BLK_LEN
;
102 if (unlock_op_buf_size
> PAGE_SIZE
)
105 unlock_op_page
= alloc_page(GFP_NOIO
);
109 p
= page_address(unlock_op_page
);
110 end
= p
+ unlock_op_buf_size
;
112 /* encode cls_lock_unlock_op struct */
113 ceph_start_encoding(&p
, 1, 1,
114 unlock_op_buf_size
- CEPH_ENCODING_START_BLK_LEN
);
115 ceph_encode_string(&p
, end
, lock_name
, name_len
);
116 ceph_encode_string(&p
, end
, cookie
, cookie_len
);
118 dout("%s lock_name %s cookie %s\n", __func__
, lock_name
, cookie
);
119 ret
= ceph_osdc_call(osdc
, oid
, oloc
, "lock", "unlock",
120 CEPH_OSD_FLAG_WRITE
, unlock_op_page
,
121 unlock_op_buf_size
, NULL
, NULL
);
123 dout("%s: status %d\n", __func__
, ret
);
124 __free_page(unlock_op_page
);
127 EXPORT_SYMBOL(ceph_cls_unlock
);
130 * ceph_cls_break_lock - release rados lock for object for specified client
131 * @oid, @oloc: object to lock
132 * @lock_name: the name of the lock
133 * @cookie: user-defined identifier for this instance of the lock
134 * @locker: current lock owner
136 int ceph_cls_break_lock(struct ceph_osd_client
*osdc
,
137 struct ceph_object_id
*oid
,
138 struct ceph_object_locator
*oloc
,
139 char *lock_name
, char *cookie
,
140 struct ceph_entity_name
*locker
)
142 int break_op_buf_size
;
143 int name_len
= strlen(lock_name
);
144 int cookie_len
= strlen(cookie
);
145 struct page
*break_op_page
;
149 break_op_buf_size
= name_len
+ sizeof(__le32
) +
150 cookie_len
+ sizeof(__le32
) +
151 sizeof(u8
) + sizeof(__le64
) +
152 CEPH_ENCODING_START_BLK_LEN
;
153 if (break_op_buf_size
> PAGE_SIZE
)
156 break_op_page
= alloc_page(GFP_NOIO
);
160 p
= page_address(break_op_page
);
161 end
= p
+ break_op_buf_size
;
163 /* encode cls_lock_break_op struct */
164 ceph_start_encoding(&p
, 1, 1,
165 break_op_buf_size
- CEPH_ENCODING_START_BLK_LEN
);
166 ceph_encode_string(&p
, end
, lock_name
, name_len
);
167 ceph_encode_copy(&p
, locker
, sizeof(*locker
));
168 ceph_encode_string(&p
, end
, cookie
, cookie_len
);
170 dout("%s lock_name %s cookie %s locker %s%llu\n", __func__
, lock_name
,
171 cookie
, ENTITY_NAME(*locker
));
172 ret
= ceph_osdc_call(osdc
, oid
, oloc
, "lock", "break_lock",
173 CEPH_OSD_FLAG_WRITE
, break_op_page
,
174 break_op_buf_size
, NULL
, NULL
);
176 dout("%s: status %d\n", __func__
, ret
);
177 __free_page(break_op_page
);
180 EXPORT_SYMBOL(ceph_cls_break_lock
);
182 int ceph_cls_set_cookie(struct ceph_osd_client
*osdc
,
183 struct ceph_object_id
*oid
,
184 struct ceph_object_locator
*oloc
,
185 char *lock_name
, u8 type
, char *old_cookie
,
186 char *tag
, char *new_cookie
)
188 int cookie_op_buf_size
;
189 int name_len
= strlen(lock_name
);
190 int old_cookie_len
= strlen(old_cookie
);
191 int tag_len
= strlen(tag
);
192 int new_cookie_len
= strlen(new_cookie
);
194 struct page
*cookie_op_page
;
197 cookie_op_buf_size
= name_len
+ sizeof(__le32
) +
198 old_cookie_len
+ sizeof(__le32
) +
199 tag_len
+ sizeof(__le32
) +
200 new_cookie_len
+ sizeof(__le32
) +
201 sizeof(u8
) + CEPH_ENCODING_START_BLK_LEN
;
202 if (cookie_op_buf_size
> PAGE_SIZE
)
205 cookie_op_page
= alloc_page(GFP_NOIO
);
209 p
= page_address(cookie_op_page
);
210 end
= p
+ cookie_op_buf_size
;
212 /* encode cls_lock_set_cookie_op struct */
213 ceph_start_encoding(&p
, 1, 1,
214 cookie_op_buf_size
- CEPH_ENCODING_START_BLK_LEN
);
215 ceph_encode_string(&p
, end
, lock_name
, name_len
);
216 ceph_encode_8(&p
, type
);
217 ceph_encode_string(&p
, end
, old_cookie
, old_cookie_len
);
218 ceph_encode_string(&p
, end
, tag
, tag_len
);
219 ceph_encode_string(&p
, end
, new_cookie
, new_cookie_len
);
221 dout("%s lock_name %s type %d old_cookie %s tag %s new_cookie %s\n",
222 __func__
, lock_name
, type
, old_cookie
, tag
, new_cookie
);
223 ret
= ceph_osdc_call(osdc
, oid
, oloc
, "lock", "set_cookie",
224 CEPH_OSD_FLAG_WRITE
, cookie_op_page
,
225 cookie_op_buf_size
, NULL
, NULL
);
227 dout("%s: status %d\n", __func__
, ret
);
228 __free_page(cookie_op_page
);
231 EXPORT_SYMBOL(ceph_cls_set_cookie
);
233 void ceph_free_lockers(struct ceph_locker
*lockers
, u32 num_lockers
)
237 for (i
= 0; i
< num_lockers
; i
++)
238 kfree(lockers
[i
].id
.cookie
);
241 EXPORT_SYMBOL(ceph_free_lockers
);
243 static int decode_locker(void **p
, void *end
, struct ceph_locker
*locker
)
250 ret
= ceph_start_decoding(p
, end
, 1, "locker_id_t", &struct_v
, &len
);
254 ceph_decode_copy(p
, &locker
->id
.name
, sizeof(locker
->id
.name
));
255 s
= ceph_extract_encoded_string(p
, end
, NULL
, GFP_NOIO
);
259 locker
->id
.cookie
= s
;
261 ret
= ceph_start_decoding(p
, end
, 1, "locker_info_t", &struct_v
, &len
);
265 *p
+= sizeof(struct ceph_timespec
); /* skip expiration */
266 ceph_decode_copy(p
, &locker
->info
.addr
, sizeof(locker
->info
.addr
));
267 ceph_decode_addr(&locker
->info
.addr
);
268 len
= ceph_decode_32(p
);
269 *p
+= len
; /* skip description */
271 dout("%s %s%llu cookie %s addr %s\n", __func__
,
272 ENTITY_NAME(locker
->id
.name
), locker
->id
.cookie
,
273 ceph_pr_addr(&locker
->info
.addr
.in_addr
));
277 static int decode_lockers(void **p
, void *end
, u8
*type
, char **tag
,
278 struct ceph_locker
**lockers
, u32
*num_lockers
)
286 ret
= ceph_start_decoding(p
, end
, 1, "cls_lock_get_info_reply",
287 &struct_v
, &struct_len
);
291 *num_lockers
= ceph_decode_32(p
);
292 *lockers
= kcalloc(*num_lockers
, sizeof(**lockers
), GFP_NOIO
);
296 for (i
= 0; i
< *num_lockers
; i
++) {
297 ret
= decode_locker(p
, end
, *lockers
+ i
);
299 goto err_free_lockers
;
302 *type
= ceph_decode_8(p
);
303 s
= ceph_extract_encoded_string(p
, end
, NULL
, GFP_NOIO
);
306 goto err_free_lockers
;
313 ceph_free_lockers(*lockers
, *num_lockers
);
318 * On success, the caller is responsible for:
321 * ceph_free_lockers(lockers, num_lockers);
323 int ceph_cls_lock_info(struct ceph_osd_client
*osdc
,
324 struct ceph_object_id
*oid
,
325 struct ceph_object_locator
*oloc
,
326 char *lock_name
, u8
*type
, char **tag
,
327 struct ceph_locker
**lockers
, u32
*num_lockers
)
329 int get_info_op_buf_size
;
330 int name_len
= strlen(lock_name
);
331 struct page
*get_info_op_page
, *reply_page
;
332 size_t reply_len
= PAGE_SIZE
;
336 get_info_op_buf_size
= name_len
+ sizeof(__le32
) +
337 CEPH_ENCODING_START_BLK_LEN
;
338 if (get_info_op_buf_size
> PAGE_SIZE
)
341 get_info_op_page
= alloc_page(GFP_NOIO
);
342 if (!get_info_op_page
)
345 reply_page
= alloc_page(GFP_NOIO
);
347 __free_page(get_info_op_page
);
351 p
= page_address(get_info_op_page
);
352 end
= p
+ get_info_op_buf_size
;
354 /* encode cls_lock_get_info_op struct */
355 ceph_start_encoding(&p
, 1, 1,
356 get_info_op_buf_size
- CEPH_ENCODING_START_BLK_LEN
);
357 ceph_encode_string(&p
, end
, lock_name
, name_len
);
359 dout("%s lock_name %s\n", __func__
, lock_name
);
360 ret
= ceph_osdc_call(osdc
, oid
, oloc
, "lock", "get_info",
361 CEPH_OSD_FLAG_READ
, get_info_op_page
,
362 get_info_op_buf_size
, reply_page
, &reply_len
);
364 dout("%s: status %d\n", __func__
, ret
);
366 p
= page_address(reply_page
);
369 ret
= decode_lockers(&p
, end
, type
, tag
, lockers
, num_lockers
);
372 __free_page(get_info_op_page
);
373 __free_page(reply_page
);
376 EXPORT_SYMBOL(ceph_cls_lock_info
);