blob: c6638c4b062c3b2e0325648b74d8c33d1091f169 [file] [log] [blame]
Philipp Reisnerb411b362009-09-25 16:07:19 -07001/*
2 drbd_receiver.c
3
4 This file is part of DRBD by Philipp Reisner and Lars Ellenberg.
5
6 Copyright (C) 2001-2008, LINBIT Information Technologies GmbH.
7 Copyright (C) 1999-2008, Philipp Reisner <philipp.reisner@linbit.com>.
8 Copyright (C) 2002-2008, Lars Ellenberg <lars.ellenberg@linbit.com>.
9
10 drbd is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 2, or (at your option)
13 any later version.
14
15 drbd is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
19
20 You should have received a copy of the GNU General Public License
21 along with drbd; see the file COPYING. If not, write to
22 the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
23 */
24
25
Philipp Reisnerb411b362009-09-25 16:07:19 -070026#include <linux/module.h>
27
28#include <asm/uaccess.h>
29#include <net/sock.h>
30
Philipp Reisnerb411b362009-09-25 16:07:19 -070031#include <linux/drbd.h>
32#include <linux/fs.h>
33#include <linux/file.h>
34#include <linux/in.h>
35#include <linux/mm.h>
36#include <linux/memcontrol.h>
37#include <linux/mm_inline.h>
38#include <linux/slab.h>
Philipp Reisnerb411b362009-09-25 16:07:19 -070039#include <linux/pkt_sched.h>
40#define __KERNEL_SYSCALLS__
41#include <linux/unistd.h>
42#include <linux/vmalloc.h>
43#include <linux/random.h>
Philipp Reisnerb411b362009-09-25 16:07:19 -070044#include <linux/string.h>
45#include <linux/scatterlist.h>
46#include "drbd_int.h"
Philipp Reisnerb411b362009-09-25 16:07:19 -070047#include "drbd_req.h"
48
49#include "drbd_vli.h"
50
Philipp Reisner77351055b2011-02-07 17:24:26 +010051struct packet_info {
52 enum drbd_packet cmd;
53 int size;
54 int vnr;
55};
56
Philipp Reisnerb411b362009-09-25 16:07:19 -070057enum finish_epoch {
58 FE_STILL_LIVE,
59 FE_DESTROYED,
60 FE_RECYCLED,
61};
62
Philipp Reisnera4fbda82011-03-16 11:13:17 +010063enum mdev_or_conn {
64 MDEV,
65 CONN,
66};
67
Philipp Reisner65d11ed2011-02-07 17:35:59 +010068static int drbd_do_handshake(struct drbd_tconn *tconn);
Philipp Reisner13e60372011-02-08 09:54:40 +010069static int drbd_do_auth(struct drbd_tconn *tconn);
Philipp Reisner360cc742011-02-08 14:29:53 +010070static int drbd_disconnected(int vnr, void *p, void *data);
Philipp Reisnerb411b362009-09-25 16:07:19 -070071
72static enum finish_epoch drbd_may_finish_epoch(struct drbd_conf *, struct drbd_epoch *, enum epoch_event);
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +010073static int e_end_block(struct drbd_work *, int);
Philipp Reisnerb411b362009-09-25 16:07:19 -070074
Philipp Reisnerb411b362009-09-25 16:07:19 -070075
76#define GFP_TRY (__GFP_HIGHMEM | __GFP_NOWARN)
77
Lars Ellenberg45bb9122010-05-14 17:10:48 +020078/*
79 * some helper functions to deal with single linked page lists,
80 * page->private being our "next" pointer.
81 */
82
83/* If at least n pages are linked at head, get n pages off.
84 * Otherwise, don't modify head, and return NULL.
85 * Locking is the responsibility of the caller.
86 */
87static struct page *page_chain_del(struct page **head, int n)
88{
89 struct page *page;
90 struct page *tmp;
91
92 BUG_ON(!n);
93 BUG_ON(!head);
94
95 page = *head;
Philipp Reisner23ce4222010-05-20 13:35:31 +020096
97 if (!page)
98 return NULL;
99
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200100 while (page) {
101 tmp = page_chain_next(page);
102 if (--n == 0)
103 break; /* found sufficient pages */
104 if (tmp == NULL)
105 /* insufficient pages, don't use any of them. */
106 return NULL;
107 page = tmp;
108 }
109
110 /* add end of list marker for the returned list */
111 set_page_private(page, 0);
112 /* actual return value, and adjustment of head */
113 page = *head;
114 *head = tmp;
115 return page;
116}
117
118/* may be used outside of locks to find the tail of a (usually short)
119 * "private" page chain, before adding it back to a global chain head
120 * with page_chain_add() under a spinlock. */
121static struct page *page_chain_tail(struct page *page, int *len)
122{
123 struct page *tmp;
124 int i = 1;
125 while ((tmp = page_chain_next(page)))
126 ++i, page = tmp;
127 if (len)
128 *len = i;
129 return page;
130}
131
132static int page_chain_free(struct page *page)
133{
134 struct page *tmp;
135 int i = 0;
136 page_chain_for_each_safe(page, tmp) {
137 put_page(page);
138 ++i;
139 }
140 return i;
141}
142
143static void page_chain_add(struct page **head,
144 struct page *chain_first, struct page *chain_last)
145{
146#if 1
147 struct page *tmp;
148 tmp = page_chain_tail(chain_first, NULL);
149 BUG_ON(tmp != chain_last);
150#endif
151
152 /* add chain to head */
153 set_page_private(chain_last, (unsigned long)*head);
154 *head = chain_first;
155}
156
157static struct page *drbd_pp_first_pages_or_try_alloc(struct drbd_conf *mdev, int number)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700158{
159 struct page *page = NULL;
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200160 struct page *tmp = NULL;
161 int i = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700162
163 /* Yes, testing drbd_pp_vacant outside the lock is racy.
164 * So what. It saves a spin_lock. */
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200165 if (drbd_pp_vacant >= number) {
Philipp Reisnerb411b362009-09-25 16:07:19 -0700166 spin_lock(&drbd_pp_lock);
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200167 page = page_chain_del(&drbd_pp_pool, number);
168 if (page)
169 drbd_pp_vacant -= number;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700170 spin_unlock(&drbd_pp_lock);
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200171 if (page)
172 return page;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700173 }
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200174
Philipp Reisnerb411b362009-09-25 16:07:19 -0700175 /* GFP_TRY, because we must not cause arbitrary write-out: in a DRBD
176 * "criss-cross" setup, that might cause write-out on some other DRBD,
177 * which in turn might block on the other node at this very place. */
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200178 for (i = 0; i < number; i++) {
179 tmp = alloc_page(GFP_TRY);
180 if (!tmp)
181 break;
182 set_page_private(tmp, (unsigned long)page);
183 page = tmp;
184 }
185
186 if (i == number)
187 return page;
188
189 /* Not enough pages immediately available this time.
190 * No need to jump around here, drbd_pp_alloc will retry this
191 * function "soon". */
192 if (page) {
193 tmp = page_chain_tail(page, NULL);
194 spin_lock(&drbd_pp_lock);
195 page_chain_add(&drbd_pp_pool, page, tmp);
196 drbd_pp_vacant += i;
197 spin_unlock(&drbd_pp_lock);
198 }
199 return NULL;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700200}
201
Philipp Reisnerb411b362009-09-25 16:07:19 -0700202static void reclaim_net_ee(struct drbd_conf *mdev, struct list_head *to_be_freed)
203{
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100204 struct drbd_peer_request *peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700205 struct list_head *le, *tle;
206
207 /* The EEs are always appended to the end of the list. Since
208 they are sent in order over the wire, they have to finish
209 in order. As soon as we see the first not finished we can
210 stop to examine the list... */
211
212 list_for_each_safe(le, tle, &mdev->net_ee) {
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100213 peer_req = list_entry(le, struct drbd_peer_request, w.list);
214 if (drbd_ee_has_active_page(peer_req))
Philipp Reisnerb411b362009-09-25 16:07:19 -0700215 break;
216 list_move(le, to_be_freed);
217 }
218}
219
220static void drbd_kick_lo_and_reclaim_net(struct drbd_conf *mdev)
221{
222 LIST_HEAD(reclaimed);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100223 struct drbd_peer_request *peer_req, *t;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700224
Philipp Reisner87eeee42011-01-19 14:16:30 +0100225 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700226 reclaim_net_ee(mdev, &reclaimed);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100227 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700228
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100229 list_for_each_entry_safe(peer_req, t, &reclaimed, w.list)
230 drbd_free_net_ee(mdev, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700231}
232
233/**
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200234 * drbd_pp_alloc() - Returns @number pages, retries forever (or until signalled)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700235 * @mdev: DRBD device.
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200236 * @number: number of pages requested
237 * @retry: whether to retry, if not enough pages are available right now
Philipp Reisnerb411b362009-09-25 16:07:19 -0700238 *
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200239 * Tries to allocate number pages, first from our own page pool, then from
240 * the kernel, unless this allocation would exceed the max_buffers setting.
241 * Possibly retry until DRBD frees sufficient pages somewhere else.
242 *
243 * Returns a page chain linked via page->private.
Philipp Reisnerb411b362009-09-25 16:07:19 -0700244 */
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200245static struct page *drbd_pp_alloc(struct drbd_conf *mdev, unsigned number, bool retry)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700246{
247 struct page *page = NULL;
248 DEFINE_WAIT(wait);
249
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200250 /* Yes, we may run up to @number over max_buffers. If we
251 * follow it strictly, the admin will get it wrong anyways. */
Philipp Reisner89e58e72011-01-19 13:12:45 +0100252 if (atomic_read(&mdev->pp_in_use) < mdev->tconn->net_conf->max_buffers)
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200253 page = drbd_pp_first_pages_or_try_alloc(mdev, number);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700254
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200255 while (page == NULL) {
Philipp Reisnerb411b362009-09-25 16:07:19 -0700256 prepare_to_wait(&drbd_pp_wait, &wait, TASK_INTERRUPTIBLE);
257
258 drbd_kick_lo_and_reclaim_net(mdev);
259
Philipp Reisner89e58e72011-01-19 13:12:45 +0100260 if (atomic_read(&mdev->pp_in_use) < mdev->tconn->net_conf->max_buffers) {
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200261 page = drbd_pp_first_pages_or_try_alloc(mdev, number);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700262 if (page)
263 break;
264 }
265
266 if (!retry)
267 break;
268
269 if (signal_pending(current)) {
270 dev_warn(DEV, "drbd_pp_alloc interrupted!\n");
271 break;
272 }
273
274 schedule();
275 }
276 finish_wait(&drbd_pp_wait, &wait);
277
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200278 if (page)
279 atomic_add(number, &mdev->pp_in_use);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700280 return page;
281}
282
283/* Must not be used from irq, as that may deadlock: see drbd_pp_alloc.
Philipp Reisner87eeee42011-01-19 14:16:30 +0100284 * Is also used from inside an other spin_lock_irq(&mdev->tconn->req_lock);
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200285 * Either links the page chain back to the global pool,
286 * or returns all pages to the system. */
Lars Ellenberg435f0742010-09-06 12:30:25 +0200287static void drbd_pp_free(struct drbd_conf *mdev, struct page *page, int is_net)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700288{
Lars Ellenberg435f0742010-09-06 12:30:25 +0200289 atomic_t *a = is_net ? &mdev->pp_in_use_by_net : &mdev->pp_in_use;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700290 int i;
Lars Ellenberg435f0742010-09-06 12:30:25 +0200291
Philipp Reisner81a5d602011-02-22 19:53:16 -0500292 if (drbd_pp_vacant > (DRBD_MAX_BIO_SIZE/PAGE_SIZE) * minor_count)
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200293 i = page_chain_free(page);
294 else {
295 struct page *tmp;
296 tmp = page_chain_tail(page, &i);
297 spin_lock(&drbd_pp_lock);
298 page_chain_add(&drbd_pp_pool, page, tmp);
299 drbd_pp_vacant += i;
300 spin_unlock(&drbd_pp_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700301 }
Lars Ellenberg435f0742010-09-06 12:30:25 +0200302 i = atomic_sub_return(i, a);
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200303 if (i < 0)
Lars Ellenberg435f0742010-09-06 12:30:25 +0200304 dev_warn(DEV, "ASSERTION FAILED: %s: %d < 0\n",
305 is_net ? "pp_in_use_by_net" : "pp_in_use", i);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700306 wake_up(&drbd_pp_wait);
307}
308
309/*
310You need to hold the req_lock:
311 _drbd_wait_ee_list_empty()
312
313You must not have the req_lock:
314 drbd_free_ee()
315 drbd_alloc_ee()
316 drbd_init_ee()
317 drbd_release_ee()
318 drbd_ee_fix_bhs()
319 drbd_process_done_ee()
320 drbd_clear_done_ee()
321 drbd_wait_ee_list_empty()
322*/
323
Andreas Gruenbacherf6ffca92011-02-04 15:30:34 +0100324struct drbd_peer_request *
325drbd_alloc_ee(struct drbd_conf *mdev, u64 id, sector_t sector,
326 unsigned int data_size, gfp_t gfp_mask) __must_hold(local)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700327{
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100328 struct drbd_peer_request *peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700329 struct page *page;
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200330 unsigned nr_pages = (data_size + PAGE_SIZE -1) >> PAGE_SHIFT;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700331
Andreas Gruenbacher0cf9d272010-12-07 10:43:29 +0100332 if (drbd_insert_fault(mdev, DRBD_FAULT_AL_EE))
Philipp Reisnerb411b362009-09-25 16:07:19 -0700333 return NULL;
334
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100335 peer_req = mempool_alloc(drbd_ee_mempool, gfp_mask & ~__GFP_HIGHMEM);
336 if (!peer_req) {
Philipp Reisnerb411b362009-09-25 16:07:19 -0700337 if (!(gfp_mask & __GFP_NOWARN))
338 dev_err(DEV, "alloc_ee: Allocation of an EE failed\n");
339 return NULL;
340 }
341
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200342 page = drbd_pp_alloc(mdev, nr_pages, (gfp_mask & __GFP_WAIT));
343 if (!page)
344 goto fail;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700345
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100346 drbd_clear_interval(&peer_req->i);
347 peer_req->i.size = data_size;
348 peer_req->i.sector = sector;
349 peer_req->i.local = false;
350 peer_req->i.waiting = false;
Andreas Gruenbacher53840642011-01-28 10:31:04 +0100351
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100352 peer_req->epoch = NULL;
Philipp Reisnera21e9292011-02-08 15:08:49 +0100353 peer_req->w.mdev = mdev;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100354 peer_req->pages = page;
355 atomic_set(&peer_req->pending_bios, 0);
356 peer_req->flags = 0;
Andreas Gruenbacher9a8e7752011-01-11 14:04:09 +0100357 /*
358 * The block_id is opaque to the receiver. It is not endianness
359 * converted, and sent back to the sender unchanged.
360 */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100361 peer_req->block_id = id;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700362
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100363 return peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700364
Lars Ellenberg45bb9122010-05-14 17:10:48 +0200365 fail:
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100366 mempool_free(peer_req, drbd_ee_mempool);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700367 return NULL;
368}
369
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100370void drbd_free_some_ee(struct drbd_conf *mdev, struct drbd_peer_request *peer_req,
Andreas Gruenbacherf6ffca92011-02-04 15:30:34 +0100371 int is_net)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700372{
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100373 if (peer_req->flags & EE_HAS_DIGEST)
374 kfree(peer_req->digest);
375 drbd_pp_free(mdev, peer_req->pages, is_net);
376 D_ASSERT(atomic_read(&peer_req->pending_bios) == 0);
377 D_ASSERT(drbd_interval_empty(&peer_req->i));
378 mempool_free(peer_req, drbd_ee_mempool);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700379}
380
381int drbd_release_ee(struct drbd_conf *mdev, struct list_head *list)
382{
383 LIST_HEAD(work_list);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100384 struct drbd_peer_request *peer_req, *t;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700385 int count = 0;
Lars Ellenberg435f0742010-09-06 12:30:25 +0200386 int is_net = list == &mdev->net_ee;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700387
Philipp Reisner87eeee42011-01-19 14:16:30 +0100388 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700389 list_splice_init(list, &work_list);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100390 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700391
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100392 list_for_each_entry_safe(peer_req, t, &work_list, w.list) {
393 drbd_free_some_ee(mdev, peer_req, is_net);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700394 count++;
395 }
396 return count;
397}
398
399
Philipp Reisner32862ec2011-02-08 16:41:01 +0100400/* See also comments in _req_mod(,BARRIER_ACKED)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700401 * and receive_Barrier.
402 *
403 * Move entries from net_ee to done_ee, if ready.
404 * Grab done_ee, call all callbacks, free the entries.
405 * The callbacks typically send out ACKs.
406 */
407static int drbd_process_done_ee(struct drbd_conf *mdev)
408{
409 LIST_HEAD(work_list);
410 LIST_HEAD(reclaimed);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100411 struct drbd_peer_request *peer_req, *t;
Andreas Gruenbachere2b30322011-03-16 17:16:12 +0100412 int err = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700413
Philipp Reisner87eeee42011-01-19 14:16:30 +0100414 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700415 reclaim_net_ee(mdev, &reclaimed);
416 list_splice_init(&mdev->done_ee, &work_list);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100417 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700418
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100419 list_for_each_entry_safe(peer_req, t, &reclaimed, w.list)
420 drbd_free_net_ee(mdev, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700421
422 /* possible callbacks here:
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +0100423 * e_end_block, and e_end_resync_block, e_send_discard_write.
Philipp Reisnerb411b362009-09-25 16:07:19 -0700424 * all ignore the last argument.
425 */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100426 list_for_each_entry_safe(peer_req, t, &work_list, w.list) {
Andreas Gruenbachere2b30322011-03-16 17:16:12 +0100427 int err2;
428
Philipp Reisnerb411b362009-09-25 16:07:19 -0700429 /* list_del not necessary, next/prev members not touched */
Andreas Gruenbachere2b30322011-03-16 17:16:12 +0100430 err2 = peer_req->w.cb(&peer_req->w, !!err);
431 if (!err)
432 err = err2;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +0100433 drbd_free_ee(mdev, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700434 }
435 wake_up(&mdev->ee_wait);
436
Andreas Gruenbachere2b30322011-03-16 17:16:12 +0100437 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700438}
439
440void _drbd_wait_ee_list_empty(struct drbd_conf *mdev, struct list_head *head)
441{
442 DEFINE_WAIT(wait);
443
444 /* avoids spin_lock/unlock
445 * and calling prepare_to_wait in the fast path */
446 while (!list_empty(head)) {
447 prepare_to_wait(&mdev->ee_wait, &wait, TASK_UNINTERRUPTIBLE);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100448 spin_unlock_irq(&mdev->tconn->req_lock);
Jens Axboe7eaceac2011-03-10 08:52:07 +0100449 io_schedule();
Philipp Reisnerb411b362009-09-25 16:07:19 -0700450 finish_wait(&mdev->ee_wait, &wait);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100451 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700452 }
453}
454
455void drbd_wait_ee_list_empty(struct drbd_conf *mdev, struct list_head *head)
456{
Philipp Reisner87eeee42011-01-19 14:16:30 +0100457 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700458 _drbd_wait_ee_list_empty(mdev, head);
Philipp Reisner87eeee42011-01-19 14:16:30 +0100459 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700460}
461
462/* see also kernel_accept; which is only present since 2.6.18.
463 * also we want to log which part of it failed, exactly */
Philipp Reisner76536202011-02-07 14:09:54 +0100464static int drbd_accept(const char **what, struct socket *sock, struct socket **newsock)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700465{
466 struct sock *sk = sock->sk;
467 int err = 0;
468
469 *what = "listen";
470 err = sock->ops->listen(sock, 5);
471 if (err < 0)
472 goto out;
473
474 *what = "sock_create_lite";
475 err = sock_create_lite(sk->sk_family, sk->sk_type, sk->sk_protocol,
476 newsock);
477 if (err < 0)
478 goto out;
479
480 *what = "accept";
481 err = sock->ops->accept(sock, *newsock, 0);
482 if (err < 0) {
483 sock_release(*newsock);
484 *newsock = NULL;
485 goto out;
486 }
487 (*newsock)->ops = sock->ops;
488
489out:
490 return err;
491}
492
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100493static int drbd_recv_short(struct socket *sock, void *buf, size_t size, int flags)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700494{
495 mm_segment_t oldfs;
496 struct kvec iov = {
497 .iov_base = buf,
498 .iov_len = size,
499 };
500 struct msghdr msg = {
501 .msg_iovlen = 1,
502 .msg_iov = (struct iovec *)&iov,
503 .msg_flags = (flags ? flags : MSG_WAITALL | MSG_NOSIGNAL)
504 };
505 int rv;
506
507 oldfs = get_fs();
508 set_fs(KERNEL_DS);
509 rv = sock_recvmsg(sock, &msg, size, msg.msg_flags);
510 set_fs(oldfs);
511
512 return rv;
513}
514
Philipp Reisnerde0ff332011-02-07 16:56:20 +0100515static int drbd_recv(struct drbd_tconn *tconn, void *buf, size_t size)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700516{
517 mm_segment_t oldfs;
518 struct kvec iov = {
519 .iov_base = buf,
520 .iov_len = size,
521 };
522 struct msghdr msg = {
523 .msg_iovlen = 1,
524 .msg_iov = (struct iovec *)&iov,
525 .msg_flags = MSG_WAITALL | MSG_NOSIGNAL
526 };
527 int rv;
528
529 oldfs = get_fs();
530 set_fs(KERNEL_DS);
531
532 for (;;) {
Philipp Reisnerde0ff332011-02-07 16:56:20 +0100533 rv = sock_recvmsg(tconn->data.socket, &msg, size, msg.msg_flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700534 if (rv == size)
535 break;
536
537 /* Note:
538 * ECONNRESET other side closed the connection
539 * ERESTARTSYS (on sock) we got a signal
540 */
541
542 if (rv < 0) {
543 if (rv == -ECONNRESET)
Philipp Reisnerde0ff332011-02-07 16:56:20 +0100544 conn_info(tconn, "sock was reset by peer\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700545 else if (rv != -ERESTARTSYS)
Philipp Reisnerde0ff332011-02-07 16:56:20 +0100546 conn_err(tconn, "sock_recvmsg returned %d\n", rv);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700547 break;
548 } else if (rv == 0) {
Philipp Reisnerde0ff332011-02-07 16:56:20 +0100549 conn_info(tconn, "sock was shut down by peer\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700550 break;
551 } else {
552 /* signal came in, or peer/link went down,
553 * after we read a partial message
554 */
555 /* D_ASSERT(signal_pending(current)); */
556 break;
557 }
558 };
559
560 set_fs(oldfs);
561
562 if (rv != size)
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100563 conn_request_state(tconn, NS(conn, C_BROKEN_PIPE), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700564
565 return rv;
566}
567
Andreas Gruenbacherc6967742011-03-17 17:15:20 +0100568static int drbd_recv_all(struct drbd_tconn *tconn, void *buf, size_t size)
569{
570 int err;
571
572 err = drbd_recv(tconn, buf, size);
573 if (err != size) {
574 if (err >= 0)
575 err = -EIO;
576 } else
577 err = 0;
578 return err;
579}
580
Andreas Gruenbachera5c31902011-03-24 03:28:04 +0100581static int drbd_recv_all_warn(struct drbd_tconn *tconn, void *buf, size_t size)
582{
583 int err;
584
585 err = drbd_recv_all(tconn, buf, size);
586 if (err && !signal_pending(current))
587 conn_warn(tconn, "short read (expected size %d)\n", (int)size);
588 return err;
589}
590
Lars Ellenberg5dbf1672010-05-25 16:18:01 +0200591/* quoting tcp(7):
592 * On individual connections, the socket buffer size must be set prior to the
593 * listen(2) or connect(2) calls in order to have it take effect.
594 * This is our wrapper to do so.
595 */
596static void drbd_setbufsize(struct socket *sock, unsigned int snd,
597 unsigned int rcv)
598{
599 /* open coded SO_SNDBUF, SO_RCVBUF */
600 if (snd) {
601 sock->sk->sk_sndbuf = snd;
602 sock->sk->sk_userlocks |= SOCK_SNDBUF_LOCK;
603 }
604 if (rcv) {
605 sock->sk->sk_rcvbuf = rcv;
606 sock->sk->sk_userlocks |= SOCK_RCVBUF_LOCK;
607 }
608}
609
Philipp Reisnereac3e992011-02-07 14:05:07 +0100610static struct socket *drbd_try_connect(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700611{
612 const char *what;
613 struct socket *sock;
614 struct sockaddr_in6 src_in6;
615 int err;
616 int disconnect_on_error = 1;
617
Philipp Reisnereac3e992011-02-07 14:05:07 +0100618 if (!get_net_conf(tconn))
Philipp Reisnerb411b362009-09-25 16:07:19 -0700619 return NULL;
620
621 what = "sock_create_kern";
Philipp Reisnereac3e992011-02-07 14:05:07 +0100622 err = sock_create_kern(((struct sockaddr *)tconn->net_conf->my_addr)->sa_family,
Philipp Reisnerb411b362009-09-25 16:07:19 -0700623 SOCK_STREAM, IPPROTO_TCP, &sock);
624 if (err < 0) {
625 sock = NULL;
626 goto out;
627 }
628
629 sock->sk->sk_rcvtimeo =
Philipp Reisnereac3e992011-02-07 14:05:07 +0100630 sock->sk->sk_sndtimeo = tconn->net_conf->try_connect_int*HZ;
631 drbd_setbufsize(sock, tconn->net_conf->sndbuf_size,
632 tconn->net_conf->rcvbuf_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700633
634 /* explicitly bind to the configured IP as source IP
635 * for the outgoing connections.
636 * This is needed for multihomed hosts and to be
637 * able to use lo: interfaces for drbd.
638 * Make sure to use 0 as port number, so linux selects
639 * a free one dynamically.
640 */
Philipp Reisnereac3e992011-02-07 14:05:07 +0100641 memcpy(&src_in6, tconn->net_conf->my_addr,
642 min_t(int, tconn->net_conf->my_addr_len, sizeof(src_in6)));
643 if (((struct sockaddr *)tconn->net_conf->my_addr)->sa_family == AF_INET6)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700644 src_in6.sin6_port = 0;
645 else
646 ((struct sockaddr_in *)&src_in6)->sin_port = 0; /* AF_INET & AF_SCI */
647
648 what = "bind before connect";
649 err = sock->ops->bind(sock,
650 (struct sockaddr *) &src_in6,
Philipp Reisnereac3e992011-02-07 14:05:07 +0100651 tconn->net_conf->my_addr_len);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700652 if (err < 0)
653 goto out;
654
655 /* connect may fail, peer not yet available.
656 * stay C_WF_CONNECTION, don't go Disconnecting! */
657 disconnect_on_error = 0;
658 what = "connect";
659 err = sock->ops->connect(sock,
Philipp Reisnereac3e992011-02-07 14:05:07 +0100660 (struct sockaddr *)tconn->net_conf->peer_addr,
661 tconn->net_conf->peer_addr_len, 0);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700662
663out:
664 if (err < 0) {
665 if (sock) {
666 sock_release(sock);
667 sock = NULL;
668 }
669 switch (-err) {
670 /* timeout, busy, signal pending */
671 case ETIMEDOUT: case EAGAIN: case EINPROGRESS:
672 case EINTR: case ERESTARTSYS:
673 /* peer not (yet) available, network problem */
674 case ECONNREFUSED: case ENETUNREACH:
675 case EHOSTDOWN: case EHOSTUNREACH:
676 disconnect_on_error = 0;
677 break;
678 default:
Philipp Reisnereac3e992011-02-07 14:05:07 +0100679 conn_err(tconn, "%s failed, err = %d\n", what, err);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700680 }
681 if (disconnect_on_error)
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100682 conn_request_state(tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700683 }
Philipp Reisnereac3e992011-02-07 14:05:07 +0100684 put_net_conf(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700685 return sock;
686}
687
Philipp Reisner76536202011-02-07 14:09:54 +0100688static struct socket *drbd_wait_for_connect(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700689{
690 int timeo, err;
691 struct socket *s_estab = NULL, *s_listen;
692 const char *what;
693
Philipp Reisner76536202011-02-07 14:09:54 +0100694 if (!get_net_conf(tconn))
Philipp Reisnerb411b362009-09-25 16:07:19 -0700695 return NULL;
696
697 what = "sock_create_kern";
Philipp Reisner76536202011-02-07 14:09:54 +0100698 err = sock_create_kern(((struct sockaddr *)tconn->net_conf->my_addr)->sa_family,
Philipp Reisnerb411b362009-09-25 16:07:19 -0700699 SOCK_STREAM, IPPROTO_TCP, &s_listen);
700 if (err) {
701 s_listen = NULL;
702 goto out;
703 }
704
Philipp Reisner76536202011-02-07 14:09:54 +0100705 timeo = tconn->net_conf->try_connect_int * HZ;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700706 timeo += (random32() & 1) ? timeo / 7 : -timeo / 7; /* 28.5% random jitter */
707
708 s_listen->sk->sk_reuse = 1; /* SO_REUSEADDR */
709 s_listen->sk->sk_rcvtimeo = timeo;
710 s_listen->sk->sk_sndtimeo = timeo;
Philipp Reisner76536202011-02-07 14:09:54 +0100711 drbd_setbufsize(s_listen, tconn->net_conf->sndbuf_size,
712 tconn->net_conf->rcvbuf_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700713
714 what = "bind before listen";
715 err = s_listen->ops->bind(s_listen,
Philipp Reisner76536202011-02-07 14:09:54 +0100716 (struct sockaddr *) tconn->net_conf->my_addr,
717 tconn->net_conf->my_addr_len);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700718 if (err < 0)
719 goto out;
720
Philipp Reisner76536202011-02-07 14:09:54 +0100721 err = drbd_accept(&what, s_listen, &s_estab);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700722
723out:
724 if (s_listen)
725 sock_release(s_listen);
726 if (err < 0) {
727 if (err != -EAGAIN && err != -EINTR && err != -ERESTARTSYS) {
Philipp Reisner76536202011-02-07 14:09:54 +0100728 conn_err(tconn, "%s failed, err = %d\n", what, err);
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100729 conn_request_state(tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700730 }
731 }
Philipp Reisner76536202011-02-07 14:09:54 +0100732 put_net_conf(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700733
734 return s_estab;
735}
736
Philipp Reisnerd38e7872011-02-07 15:32:04 +0100737static int drbd_send_fp(struct drbd_tconn *tconn, struct socket *sock, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700738{
Andreas Gruenbacher5a87d922011-03-24 21:17:52 +0100739 struct p_header *h = tconn->data.sbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700740
Andreas Gruenbacherecf23632011-03-15 23:48:25 +0100741 return !_conn_send_cmd(tconn, 0, sock, cmd, h, sizeof(*h), 0);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700742}
743
Philipp Reisnera25b63f2011-02-07 15:43:45 +0100744static enum drbd_packet drbd_recv_fp(struct drbd_tconn *tconn, struct socket *sock)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700745{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +0100746 struct p_header80 *h = tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700747 int rr;
748
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100749 rr = drbd_recv_short(sock, h, sizeof(*h), 0);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700750
Andreas Gruenbacherca9bc122011-01-11 13:47:24 +0100751 if (rr == sizeof(*h) && h->magic == cpu_to_be32(DRBD_MAGIC))
Philipp Reisnerb411b362009-09-25 16:07:19 -0700752 return be16_to_cpu(h->command);
753
754 return 0xffff;
755}
756
757/**
758 * drbd_socket_okay() - Free the socket if its connection is not okay
Philipp Reisnerb411b362009-09-25 16:07:19 -0700759 * @sock: pointer to the pointer to the socket.
760 */
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100761static int drbd_socket_okay(struct socket **sock)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700762{
763 int rr;
764 char tb[4];
765
766 if (!*sock)
Andreas Gruenbacher81e84652010-12-09 15:03:57 +0100767 return false;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700768
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100769 rr = drbd_recv_short(*sock, tb, 4, MSG_DONTWAIT | MSG_PEEK);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700770
771 if (rr > 0 || rr == -EAGAIN) {
Andreas Gruenbacher81e84652010-12-09 15:03:57 +0100772 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700773 } else {
774 sock_release(*sock);
775 *sock = NULL;
Andreas Gruenbacher81e84652010-12-09 15:03:57 +0100776 return false;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700777 }
778}
Philipp Reisner2325eb62011-03-15 16:56:18 +0100779/* Gets called if a connection is established, or if a new minor gets created
780 in a connection */
781int drbd_connected(int vnr, void *p, void *data)
Philipp Reisner907599e2011-02-08 11:25:37 +0100782{
783 struct drbd_conf *mdev = (struct drbd_conf *)p;
Andreas Gruenbacher0829f5e2011-03-24 14:31:22 +0100784 int err;
Philipp Reisner907599e2011-02-08 11:25:37 +0100785
786 atomic_set(&mdev->packet_seq, 0);
787 mdev->peer_seq = 0;
788
Philipp Reisner8410da8f02011-02-11 20:11:10 +0100789 mdev->state_mutex = mdev->tconn->agreed_pro_version < 100 ?
790 &mdev->tconn->cstate_mutex :
791 &mdev->own_state_mutex;
792
Andreas Gruenbacher0829f5e2011-03-24 14:31:22 +0100793 err = drbd_send_sync_param(mdev);
794 if (!err)
795 err = drbd_send_sizes(mdev, 0, 0);
796 if (!err)
797 err = drbd_send_uuids(mdev);
798 if (!err)
799 err = drbd_send_state(mdev);
Philipp Reisner907599e2011-02-08 11:25:37 +0100800 clear_bit(USE_DEGR_WFC_T, &mdev->flags);
801 clear_bit(RESIZE_PENDING, &mdev->flags);
Andreas Gruenbacher0829f5e2011-03-24 14:31:22 +0100802 return err;
Philipp Reisner907599e2011-02-08 11:25:37 +0100803}
804
Philipp Reisnerb411b362009-09-25 16:07:19 -0700805/*
806 * return values:
807 * 1 yes, we have a valid connection
808 * 0 oops, did not work out, please try again
809 * -1 peer talks different language,
810 * no point in trying again, please go standalone.
811 * -2 We do not have a network config...
812 */
Philipp Reisner907599e2011-02-08 11:25:37 +0100813static int drbd_connect(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700814{
815 struct socket *s, *sock, *msock;
816 int try, h, ok;
817
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100818 if (conn_request_state(tconn, NS(conn, C_WF_CONNECTION), CS_VERBOSE) < SS_SUCCESS)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700819 return -2;
820
Philipp Reisner907599e2011-02-08 11:25:37 +0100821 clear_bit(DISCARD_CONCURRENT, &tconn->flags);
Andreas Gruenbacher0916e0e2011-03-21 14:10:15 +0100822
823 /* Assume that the peer only understands protocol 80 until we know better. */
824 tconn->agreed_pro_version = 80;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700825
826 sock = NULL;
827 msock = NULL;
828
829 do {
830 for (try = 0;;) {
831 /* 3 tries, this should take less than a second! */
Philipp Reisner907599e2011-02-08 11:25:37 +0100832 s = drbd_try_connect(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700833 if (s || ++try >= 3)
834 break;
835 /* give the other side time to call bind() & listen() */
Philipp Reisner20ee6392011-01-18 15:28:59 +0100836 schedule_timeout_interruptible(HZ / 10);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700837 }
838
839 if (s) {
840 if (!sock) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100841 drbd_send_fp(tconn, s, P_HAND_SHAKE_S);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700842 sock = s;
843 s = NULL;
844 } else if (!msock) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100845 drbd_send_fp(tconn, s, P_HAND_SHAKE_M);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700846 msock = s;
847 s = NULL;
848 } else {
Philipp Reisner907599e2011-02-08 11:25:37 +0100849 conn_err(tconn, "Logic error in drbd_connect()\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700850 goto out_release_sockets;
851 }
852 }
853
854 if (sock && msock) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100855 schedule_timeout_interruptible(tconn->net_conf->ping_timeo*HZ/10);
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100856 ok = drbd_socket_okay(&sock);
857 ok = drbd_socket_okay(&msock) && ok;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700858 if (ok)
859 break;
860 }
861
862retry:
Philipp Reisner907599e2011-02-08 11:25:37 +0100863 s = drbd_wait_for_connect(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700864 if (s) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100865 try = drbd_recv_fp(tconn, s);
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100866 drbd_socket_okay(&sock);
867 drbd_socket_okay(&msock);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700868 switch (try) {
869 case P_HAND_SHAKE_S:
870 if (sock) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100871 conn_warn(tconn, "initial packet S crossed\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700872 sock_release(sock);
873 }
874 sock = s;
875 break;
876 case P_HAND_SHAKE_M:
877 if (msock) {
Philipp Reisner907599e2011-02-08 11:25:37 +0100878 conn_warn(tconn, "initial packet M crossed\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700879 sock_release(msock);
880 }
881 msock = s;
Philipp Reisner907599e2011-02-08 11:25:37 +0100882 set_bit(DISCARD_CONCURRENT, &tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700883 break;
884 default:
Philipp Reisner907599e2011-02-08 11:25:37 +0100885 conn_warn(tconn, "Error receiving initial packet\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700886 sock_release(s);
887 if (random32() & 1)
888 goto retry;
889 }
890 }
891
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100892 if (tconn->cstate <= C_DISCONNECTING)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700893 goto out_release_sockets;
894 if (signal_pending(current)) {
895 flush_signals(current);
896 smp_rmb();
Philipp Reisner907599e2011-02-08 11:25:37 +0100897 if (get_t_state(&tconn->receiver) == EXITING)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700898 goto out_release_sockets;
899 }
900
901 if (sock && msock) {
Philipp Reisnerdbd9eea2011-02-07 15:34:16 +0100902 ok = drbd_socket_okay(&sock);
903 ok = drbd_socket_okay(&msock) && ok;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700904 if (ok)
905 break;
906 }
907 } while (1);
908
909 msock->sk->sk_reuse = 1; /* SO_REUSEADDR */
910 sock->sk->sk_reuse = 1; /* SO_REUSEADDR */
911
912 sock->sk->sk_allocation = GFP_NOIO;
913 msock->sk->sk_allocation = GFP_NOIO;
914
915 sock->sk->sk_priority = TC_PRIO_INTERACTIVE_BULK;
916 msock->sk->sk_priority = TC_PRIO_INTERACTIVE;
917
Philipp Reisnerb411b362009-09-25 16:07:19 -0700918 /* NOT YET ...
Philipp Reisner907599e2011-02-08 11:25:37 +0100919 * sock->sk->sk_sndtimeo = tconn->net_conf->timeout*HZ/10;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700920 * sock->sk->sk_rcvtimeo = MAX_SCHEDULE_TIMEOUT;
921 * first set it to the P_HAND_SHAKE timeout,
922 * which we set to 4x the configured ping_timeout. */
923 sock->sk->sk_sndtimeo =
Philipp Reisner907599e2011-02-08 11:25:37 +0100924 sock->sk->sk_rcvtimeo = tconn->net_conf->ping_timeo*4*HZ/10;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700925
Philipp Reisner907599e2011-02-08 11:25:37 +0100926 msock->sk->sk_sndtimeo = tconn->net_conf->timeout*HZ/10;
927 msock->sk->sk_rcvtimeo = tconn->net_conf->ping_int*HZ;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700928
929 /* we don't want delays.
Lucas De Marchi25985ed2011-03-30 22:57:33 -0300930 * we use TCP_CORK where appropriate, though */
Philipp Reisnerb411b362009-09-25 16:07:19 -0700931 drbd_tcp_nodelay(sock);
932 drbd_tcp_nodelay(msock);
933
Philipp Reisner907599e2011-02-08 11:25:37 +0100934 tconn->data.socket = sock;
935 tconn->meta.socket = msock;
936 tconn->last_received = jiffies;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700937
Philipp Reisner907599e2011-02-08 11:25:37 +0100938 h = drbd_do_handshake(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700939 if (h <= 0)
940 return h;
941
Philipp Reisner907599e2011-02-08 11:25:37 +0100942 if (tconn->cram_hmac_tfm) {
Philipp Reisnerb411b362009-09-25 16:07:19 -0700943 /* drbd_request_state(mdev, NS(conn, WFAuth)); */
Philipp Reisner907599e2011-02-08 11:25:37 +0100944 switch (drbd_do_auth(tconn)) {
Johannes Thomab10d96c2010-01-07 16:02:50 +0100945 case -1:
Philipp Reisner907599e2011-02-08 11:25:37 +0100946 conn_err(tconn, "Authentication of peer failed\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -0700947 return -1;
Johannes Thomab10d96c2010-01-07 16:02:50 +0100948 case 0:
Philipp Reisner907599e2011-02-08 11:25:37 +0100949 conn_err(tconn, "Authentication of peer failed, trying again.\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +0100950 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700951 }
952 }
953
Philipp Reisnerbbeb6412011-02-10 13:45:46 +0100954 if (conn_request_state(tconn, NS(conn, C_WF_REPORT_PARAMS), CS_VERBOSE) < SS_SUCCESS)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700955 return 0;
956
Philipp Reisner907599e2011-02-08 11:25:37 +0100957 sock->sk->sk_sndtimeo = tconn->net_conf->timeout*HZ/10;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700958 sock->sk->sk_rcvtimeo = MAX_SCHEDULE_TIMEOUT;
959
Philipp Reisner907599e2011-02-08 11:25:37 +0100960 drbd_thread_start(&tconn->asender);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700961
Andreas Gruenbacher387eb302011-03-16 01:05:37 +0100962 if (drbd_send_protocol(tconn) == -EOPNOTSUPP)
Philipp Reisner7e2455c2010-04-22 14:50:23 +0200963 return -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700964
Philipp Reisner907599e2011-02-08 11:25:37 +0100965 return !idr_for_each(&tconn->volumes, drbd_connected, tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -0700966
967out_release_sockets:
968 if (sock)
969 sock_release(sock);
970 if (msock)
971 sock_release(msock);
972 return -1;
973}
974
Andreas Gruenbacher8172f3e2011-03-16 17:22:39 +0100975static int decode_header(struct drbd_tconn *tconn, struct p_header *h, struct packet_info *pi)
Philipp Reisnerb411b362009-09-25 16:07:19 -0700976{
Philipp Reisnerfd340c12011-01-19 16:57:39 +0100977 if (h->h80.magic == cpu_to_be32(DRBD_MAGIC)) {
Philipp Reisner77351055b2011-02-07 17:24:26 +0100978 pi->cmd = be16_to_cpu(h->h80.command);
979 pi->size = be16_to_cpu(h->h80.length);
Philipp Reisnereefc2f72011-02-08 12:55:24 +0100980 pi->vnr = 0;
Andreas Gruenbacherca9bc122011-01-11 13:47:24 +0100981 } else if (h->h95.magic == cpu_to_be16(DRBD_MAGIC_BIG)) {
Philipp Reisner77351055b2011-02-07 17:24:26 +0100982 pi->cmd = be16_to_cpu(h->h95.command);
983 pi->size = be32_to_cpu(h->h95.length) & 0x00ffffff;
984 pi->vnr = 0;
Philipp Reisner02918be2010-08-20 14:35:10 +0200985 } else {
Philipp Reisnerce243852011-02-07 17:27:47 +0100986 conn_err(tconn, "magic?? on data m: 0x%08x c: %d l: %d\n",
Lars Ellenberg004352f2010-10-05 20:13:58 +0200987 be32_to_cpu(h->h80.magic),
988 be16_to_cpu(h->h80.command),
989 be16_to_cpu(h->h80.length));
Andreas Gruenbacher8172f3e2011-03-16 17:22:39 +0100990 return -EINVAL;
Philipp Reisnerb411b362009-09-25 16:07:19 -0700991 }
Andreas Gruenbacher8172f3e2011-03-16 17:22:39 +0100992 return 0;
Philipp Reisner257d0af2011-01-26 12:15:29 +0100993}
994
Philipp Reisner9ba7aa02011-02-07 17:32:41 +0100995static int drbd_recv_header(struct drbd_tconn *tconn, struct packet_info *pi)
Philipp Reisner257d0af2011-01-26 12:15:29 +0100996{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +0100997 struct p_header *h = tconn->data.rbuf;
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +0100998 int err;
Philipp Reisner257d0af2011-01-26 12:15:29 +0100999
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001000 err = drbd_recv_all_warn(tconn, h, sizeof(*h));
1001 if (err)
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01001002 return err;
Philipp Reisner257d0af2011-01-26 12:15:29 +01001003
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01001004 err = decode_header(tconn, h, pi);
Philipp Reisner9ba7aa02011-02-07 17:32:41 +01001005 tconn->last_received = jiffies;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001006
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01001007 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001008}
1009
Philipp Reisner2451fc32010-08-24 13:43:11 +02001010static void drbd_flush(struct drbd_conf *mdev)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001011{
1012 int rv;
1013
1014 if (mdev->write_ordering >= WO_bdev_flush && get_ldev(mdev)) {
Dmitry Monakhovfbd9b092010-04-28 17:55:06 +04001015 rv = blkdev_issue_flush(mdev->ldev->backing_bdev, GFP_KERNEL,
Christoph Hellwigdd3932e2010-09-16 20:51:46 +02001016 NULL);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001017 if (rv) {
1018 dev_err(DEV, "local disk flush failed with status %d\n", rv);
1019 /* would rather check on EOPNOTSUPP, but that is not reliable.
1020 * don't try again for ANY return value != 0
1021 * if (rv == -EOPNOTSUPP) */
1022 drbd_bump_write_ordering(mdev, WO_drain_io);
1023 }
1024 put_ldev(mdev);
1025 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07001026}
1027
1028/**
1029 * drbd_may_finish_epoch() - Applies an epoch_event to the epoch's state, eventually finishes it.
1030 * @mdev: DRBD device.
1031 * @epoch: Epoch object.
1032 * @ev: Epoch event.
1033 */
1034static enum finish_epoch drbd_may_finish_epoch(struct drbd_conf *mdev,
1035 struct drbd_epoch *epoch,
1036 enum epoch_event ev)
1037{
Philipp Reisner2451fc32010-08-24 13:43:11 +02001038 int epoch_size;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001039 struct drbd_epoch *next_epoch;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001040 enum finish_epoch rv = FE_STILL_LIVE;
1041
1042 spin_lock(&mdev->epoch_lock);
1043 do {
1044 next_epoch = NULL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001045
1046 epoch_size = atomic_read(&epoch->epoch_size);
1047
1048 switch (ev & ~EV_CLEANUP) {
1049 case EV_PUT:
1050 atomic_dec(&epoch->active);
1051 break;
1052 case EV_GOT_BARRIER_NR:
1053 set_bit(DE_HAVE_BARRIER_NUMBER, &epoch->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001054 break;
1055 case EV_BECAME_LAST:
1056 /* nothing to do*/
1057 break;
1058 }
1059
Philipp Reisnerb411b362009-09-25 16:07:19 -07001060 if (epoch_size != 0 &&
1061 atomic_read(&epoch->active) == 0 &&
Philipp Reisner2451fc32010-08-24 13:43:11 +02001062 test_bit(DE_HAVE_BARRIER_NUMBER, &epoch->flags)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07001063 if (!(ev & EV_CLEANUP)) {
1064 spin_unlock(&mdev->epoch_lock);
1065 drbd_send_b_ack(mdev, epoch->barrier_nr, epoch_size);
1066 spin_lock(&mdev->epoch_lock);
1067 }
1068 dec_unacked(mdev);
1069
1070 if (mdev->current_epoch != epoch) {
1071 next_epoch = list_entry(epoch->list.next, struct drbd_epoch, list);
1072 list_del(&epoch->list);
1073 ev = EV_BECAME_LAST | (ev & EV_CLEANUP);
1074 mdev->epochs--;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001075 kfree(epoch);
1076
1077 if (rv == FE_STILL_LIVE)
1078 rv = FE_DESTROYED;
1079 } else {
1080 epoch->flags = 0;
1081 atomic_set(&epoch->epoch_size, 0);
Uwe Kleine-König698f9312010-07-02 20:41:51 +02001082 /* atomic_set(&epoch->active, 0); is already zero */
Philipp Reisnerb411b362009-09-25 16:07:19 -07001083 if (rv == FE_STILL_LIVE)
1084 rv = FE_RECYCLED;
Philipp Reisner2451fc32010-08-24 13:43:11 +02001085 wake_up(&mdev->ee_wait);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001086 }
1087 }
1088
1089 if (!next_epoch)
1090 break;
1091
1092 epoch = next_epoch;
1093 } while (1);
1094
1095 spin_unlock(&mdev->epoch_lock);
1096
Philipp Reisnerb411b362009-09-25 16:07:19 -07001097 return rv;
1098}
1099
1100/**
1101 * drbd_bump_write_ordering() - Fall back to an other write ordering method
1102 * @mdev: DRBD device.
1103 * @wo: Write ordering method to try.
1104 */
1105void drbd_bump_write_ordering(struct drbd_conf *mdev, enum write_ordering_e wo) __must_hold(local)
1106{
1107 enum write_ordering_e pwo;
1108 static char *write_ordering_str[] = {
1109 [WO_none] = "none",
1110 [WO_drain_io] = "drain",
1111 [WO_bdev_flush] = "flush",
Philipp Reisnerb411b362009-09-25 16:07:19 -07001112 };
1113
1114 pwo = mdev->write_ordering;
1115 wo = min(pwo, wo);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001116 if (wo == WO_bdev_flush && mdev->ldev->dc.no_disk_flush)
1117 wo = WO_drain_io;
1118 if (wo == WO_drain_io && mdev->ldev->dc.no_disk_drain)
1119 wo = WO_none;
1120 mdev->write_ordering = wo;
Philipp Reisner2451fc32010-08-24 13:43:11 +02001121 if (pwo != mdev->write_ordering || wo == WO_bdev_flush)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001122 dev_info(DEV, "Method to ensure write ordering: %s\n", write_ordering_str[mdev->write_ordering]);
1123}
1124
1125/**
Andreas Gruenbacherfbe29de2011-02-17 16:38:35 +01001126 * drbd_submit_peer_request()
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001127 * @mdev: DRBD device.
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001128 * @peer_req: peer request
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001129 * @rw: flag field, see bio->bi_rw
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01001130 *
1131 * May spread the pages to multiple bios,
1132 * depending on bio_add_page restrictions.
1133 *
1134 * Returns 0 if all bios have been submitted,
1135 * -ENOMEM if we could not allocate enough bios,
1136 * -ENOSPC (any better suggestion?) if we have not been able to bio_add_page a
1137 * single page to an empty bio (which should never happen and likely indicates
1138 * that the lower level IO stack is in some way broken). This has been observed
1139 * on certain Xen deployments.
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001140 */
1141/* TODO allocate from our own bio_set. */
Andreas Gruenbacherfbe29de2011-02-17 16:38:35 +01001142int drbd_submit_peer_request(struct drbd_conf *mdev,
1143 struct drbd_peer_request *peer_req,
1144 const unsigned rw, const int fault_type)
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001145{
1146 struct bio *bios = NULL;
1147 struct bio *bio;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001148 struct page *page = peer_req->pages;
1149 sector_t sector = peer_req->i.sector;
1150 unsigned ds = peer_req->i.size;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001151 unsigned n_bios = 0;
1152 unsigned nr_pages = (ds + PAGE_SIZE -1) >> PAGE_SHIFT;
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01001153 int err = -ENOMEM;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001154
1155 /* In most cases, we will only need one bio. But in case the lower
1156 * level restrictions happen to be different at this offset on this
1157 * side than those of the sending peer, we may need to submit the
Lars Ellenbergda4a75d2011-02-23 17:02:01 +01001158 * request in more than one bio.
1159 *
1160 * Plain bio_alloc is good enough here, this is no DRBD internally
1161 * generated bio, but a bio allocated on behalf of the peer.
1162 */
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001163next_bio:
1164 bio = bio_alloc(GFP_NOIO, nr_pages);
1165 if (!bio) {
1166 dev_err(DEV, "submit_ee: Allocation of a bio failed\n");
1167 goto fail;
1168 }
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001169 /* > peer_req->i.sector, unless this is the first bio */
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001170 bio->bi_sector = sector;
1171 bio->bi_bdev = mdev->ldev->backing_bdev;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001172 bio->bi_rw = rw;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001173 bio->bi_private = peer_req;
Andreas Gruenbacherfcefa622011-02-17 16:46:59 +01001174 bio->bi_end_io = drbd_peer_request_endio;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001175
1176 bio->bi_next = bios;
1177 bios = bio;
1178 ++n_bios;
1179
1180 page_chain_for_each(page) {
1181 unsigned len = min_t(unsigned, ds, PAGE_SIZE);
1182 if (!bio_add_page(bio, page, len, 0)) {
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01001183 /* A single page must always be possible!
1184 * But in case it fails anyways,
1185 * we deal with it, and complain (below). */
1186 if (bio->bi_vcnt == 0) {
1187 dev_err(DEV,
1188 "bio_add_page failed for len=%u, "
1189 "bi_vcnt=0 (bi_sector=%llu)\n",
1190 len, (unsigned long long)bio->bi_sector);
1191 err = -ENOSPC;
1192 goto fail;
1193 }
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001194 goto next_bio;
1195 }
1196 ds -= len;
1197 sector += len >> 9;
1198 --nr_pages;
1199 }
1200 D_ASSERT(page == NULL);
1201 D_ASSERT(ds == 0);
1202
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001203 atomic_set(&peer_req->pending_bios, n_bios);
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001204 do {
1205 bio = bios;
1206 bios = bios->bi_next;
1207 bio->bi_next = NULL;
1208
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001209 drbd_generic_make_request(mdev, fault_type, bio);
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001210 } while (bios);
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001211 return 0;
1212
1213fail:
1214 while (bios) {
1215 bio = bios;
1216 bios = bios->bi_next;
1217 bio_put(bio);
1218 }
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01001219 return err;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001220}
1221
Andreas Gruenbacher53840642011-01-28 10:31:04 +01001222static void drbd_remove_epoch_entry_interval(struct drbd_conf *mdev,
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001223 struct drbd_peer_request *peer_req)
Andreas Gruenbacher53840642011-01-28 10:31:04 +01001224{
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001225 struct drbd_interval *i = &peer_req->i;
Andreas Gruenbacher53840642011-01-28 10:31:04 +01001226
1227 drbd_remove_interval(&mdev->write_requests, i);
1228 drbd_clear_interval(i);
1229
Andreas Gruenbacher6c852be2011-02-04 15:38:52 +01001230 /* Wake up any processes waiting for this peer request to complete. */
Andreas Gruenbacher53840642011-01-28 10:31:04 +01001231 if (i->waiting)
1232 wake_up(&mdev->misc_wait);
1233}
1234
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01001235static int receive_Barrier(struct drbd_conf *mdev, enum drbd_packet cmd,
1236 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001237{
Philipp Reisner2451fc32010-08-24 13:43:11 +02001238 int rv;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01001239 struct p_barrier *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001240 struct drbd_epoch *epoch;
1241
Philipp Reisnerb411b362009-09-25 16:07:19 -07001242 inc_unacked(mdev);
1243
Philipp Reisnerb411b362009-09-25 16:07:19 -07001244 mdev->current_epoch->barrier_nr = p->barrier;
1245 rv = drbd_may_finish_epoch(mdev, mdev->current_epoch, EV_GOT_BARRIER_NR);
1246
1247 /* P_BARRIER_ACK may imply that the corresponding extent is dropped from
1248 * the activity log, which means it would not be resynced in case the
1249 * R_PRIMARY crashes now.
1250 * Therefore we must send the barrier_ack after the barrier request was
1251 * completed. */
1252 switch (mdev->write_ordering) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07001253 case WO_none:
1254 if (rv == FE_RECYCLED)
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001255 return 0;
Philipp Reisner2451fc32010-08-24 13:43:11 +02001256
1257 /* receiver context, in the writeout path of the other node.
1258 * avoid potential distributed deadlock */
1259 epoch = kmalloc(sizeof(struct drbd_epoch), GFP_NOIO);
1260 if (epoch)
1261 break;
1262 else
1263 dev_warn(DEV, "Allocation of an epoch failed, slowing down\n");
1264 /* Fall through */
Philipp Reisnerb411b362009-09-25 16:07:19 -07001265
1266 case WO_bdev_flush:
1267 case WO_drain_io:
Philipp Reisnerb411b362009-09-25 16:07:19 -07001268 drbd_wait_ee_list_empty(mdev, &mdev->active_ee);
Philipp Reisner2451fc32010-08-24 13:43:11 +02001269 drbd_flush(mdev);
1270
1271 if (atomic_read(&mdev->current_epoch->epoch_size)) {
1272 epoch = kmalloc(sizeof(struct drbd_epoch), GFP_NOIO);
1273 if (epoch)
1274 break;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001275 }
1276
Philipp Reisner2451fc32010-08-24 13:43:11 +02001277 epoch = mdev->current_epoch;
1278 wait_event(mdev->ee_wait, atomic_read(&epoch->epoch_size) == 0);
1279
1280 D_ASSERT(atomic_read(&epoch->active) == 0);
1281 D_ASSERT(epoch->flags == 0);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001282
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001283 return 0;
Philipp Reisner2451fc32010-08-24 13:43:11 +02001284 default:
1285 dev_err(DEV, "Strangeness in mdev->write_ordering %d\n", mdev->write_ordering);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001286 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001287 }
1288
1289 epoch->flags = 0;
1290 atomic_set(&epoch->epoch_size, 0);
1291 atomic_set(&epoch->active, 0);
1292
1293 spin_lock(&mdev->epoch_lock);
1294 if (atomic_read(&mdev->current_epoch->epoch_size)) {
1295 list_add(&epoch->list, &mdev->current_epoch->list);
1296 mdev->current_epoch = epoch;
1297 mdev->epochs++;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001298 } else {
1299 /* The current_epoch got recycled while we allocated this one... */
1300 kfree(epoch);
1301 }
1302 spin_unlock(&mdev->epoch_lock);
1303
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001304 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001305}
1306
1307/* used from receive_RSDataReply (recv_resync_read)
1308 * and from receive_Data */
Andreas Gruenbacherf6ffca92011-02-04 15:30:34 +01001309static struct drbd_peer_request *
1310read_in_block(struct drbd_conf *mdev, u64 id, sector_t sector,
1311 int data_size) __must_hold(local)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001312{
Lars Ellenberg66660322010-04-06 12:15:04 +02001313 const sector_t capacity = drbd_get_capacity(mdev->this_bdev);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001314 struct drbd_peer_request *peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001315 struct page *page;
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001316 int dgs, ds, err;
Philipp Reisnera0638452011-01-19 14:31:32 +01001317 void *dig_in = mdev->tconn->int_dig_in;
1318 void *dig_vv = mdev->tconn->int_dig_vv;
Philipp Reisner6b4388a2010-04-26 14:11:45 +02001319 unsigned long *data;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001320
Philipp Reisnera0638452011-01-19 14:31:32 +01001321 dgs = (mdev->tconn->agreed_pro_version >= 87 && mdev->tconn->integrity_r_tfm) ?
1322 crypto_hash_digestsize(mdev->tconn->integrity_r_tfm) : 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001323
1324 if (dgs) {
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001325 err = drbd_recv_all_warn(mdev->tconn, dig_in, dgs);
1326 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001327 return NULL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001328 }
1329
1330 data_size -= dgs;
1331
Andreas Gruenbacher841ce242010-12-15 19:31:20 +01001332 if (!expect(data_size != 0))
1333 return NULL;
1334 if (!expect(IS_ALIGNED(data_size, 512)))
1335 return NULL;
1336 if (!expect(data_size <= DRBD_MAX_BIO_SIZE))
1337 return NULL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001338
Lars Ellenberg66660322010-04-06 12:15:04 +02001339 /* even though we trust out peer,
1340 * we sometimes have to double check. */
1341 if (sector + (data_size>>9) > capacity) {
Lars Ellenbergfdda6542011-01-24 15:11:01 +01001342 dev_err(DEV, "request from peer beyond end of local disk: "
1343 "capacity: %llus < sector: %llus + size: %u\n",
Lars Ellenberg66660322010-04-06 12:15:04 +02001344 (unsigned long long)capacity,
1345 (unsigned long long)sector, data_size);
1346 return NULL;
1347 }
1348
Philipp Reisnerb411b362009-09-25 16:07:19 -07001349 /* GFP_NOIO, because we must not cause arbitrary write-out: in a DRBD
1350 * "criss-cross" setup, that might cause write-out on some other DRBD,
1351 * which in turn might block on the other node at this very place. */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001352 peer_req = drbd_alloc_ee(mdev, id, sector, data_size, GFP_NOIO);
1353 if (!peer_req)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001354 return NULL;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001355
Philipp Reisnerb411b362009-09-25 16:07:19 -07001356 ds = data_size;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001357 page = peer_req->pages;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001358 page_chain_for_each(page) {
1359 unsigned len = min_t(int, ds, PAGE_SIZE);
Philipp Reisner6b4388a2010-04-26 14:11:45 +02001360 data = kmap(page);
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001361 err = drbd_recv_all_warn(mdev->tconn, data, len);
Andreas Gruenbacher0cf9d272010-12-07 10:43:29 +01001362 if (drbd_insert_fault(mdev, DRBD_FAULT_RECEIVE)) {
Philipp Reisner6b4388a2010-04-26 14:11:45 +02001363 dev_err(DEV, "Fault injection: Corrupting data on receive\n");
1364 data[0] = data[0] ^ (unsigned long)-1;
1365 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07001366 kunmap(page);
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001367 if (err) {
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001368 drbd_free_ee(mdev, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001369 return NULL;
1370 }
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001371 ds -= len;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001372 }
1373
1374 if (dgs) {
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001375 drbd_csum_ee(mdev, mdev->tconn->integrity_r_tfm, peer_req, dig_vv);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001376 if (memcmp(dig_in, dig_vv, dgs)) {
Lars Ellenberg470be442010-11-10 10:36:52 +01001377 dev_err(DEV, "Digest integrity check FAILED: %llus +%u\n",
1378 (unsigned long long)sector, data_size);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001379 drbd_free_ee(mdev, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001380 return NULL;
1381 }
1382 }
1383 mdev->recv_cnt += data_size>>9;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001384 return peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001385}
1386
1387/* drbd_drain_block() just takes a data block
1388 * out of the socket input buffer, and discards it.
1389 */
1390static int drbd_drain_block(struct drbd_conf *mdev, int data_size)
1391{
1392 struct page *page;
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001393 int err = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001394 void *data;
1395
Lars Ellenbergc3470cd2010-04-01 16:57:19 +02001396 if (!data_size)
Andreas Gruenbacherfc5be832011-03-16 17:50:50 +01001397 return 0;
Lars Ellenbergc3470cd2010-04-01 16:57:19 +02001398
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001399 page = drbd_pp_alloc(mdev, 1, 1);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001400
1401 data = kmap(page);
1402 while (data_size) {
Andreas Gruenbacherfc5be832011-03-16 17:50:50 +01001403 unsigned int len = min_t(int, data_size, PAGE_SIZE);
1404
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001405 err = drbd_recv_all_warn(mdev->tconn, data, len);
1406 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001407 break;
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001408 data_size -= len;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001409 }
1410 kunmap(page);
Lars Ellenberg435f0742010-09-06 12:30:25 +02001411 drbd_pp_free(mdev, page, 0);
Andreas Gruenbacherfc5be832011-03-16 17:50:50 +01001412 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001413}
1414
1415static int recv_dless_read(struct drbd_conf *mdev, struct drbd_request *req,
1416 sector_t sector, int data_size)
1417{
1418 struct bio_vec *bvec;
1419 struct bio *bio;
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001420 int dgs, err, i, expect;
Philipp Reisnera0638452011-01-19 14:31:32 +01001421 void *dig_in = mdev->tconn->int_dig_in;
1422 void *dig_vv = mdev->tconn->int_dig_vv;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001423
Philipp Reisnera0638452011-01-19 14:31:32 +01001424 dgs = (mdev->tconn->agreed_pro_version >= 87 && mdev->tconn->integrity_r_tfm) ?
1425 crypto_hash_digestsize(mdev->tconn->integrity_r_tfm) : 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001426
1427 if (dgs) {
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001428 err = drbd_recv_all_warn(mdev->tconn, dig_in, dgs);
1429 if (err)
1430 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001431 }
1432
1433 data_size -= dgs;
1434
1435 /* optimistically update recv_cnt. if receiving fails below,
1436 * we disconnect anyways, and counters will be reset. */
1437 mdev->recv_cnt += data_size>>9;
1438
1439 bio = req->master_bio;
1440 D_ASSERT(sector == bio->bi_sector);
1441
1442 bio_for_each_segment(bvec, bio, i) {
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001443 void *mapped = kmap(bvec->bv_page) + bvec->bv_offset;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001444 expect = min_t(int, data_size, bvec->bv_len);
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001445 err = drbd_recv_all_warn(mdev->tconn, mapped, expect);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001446 kunmap(bvec->bv_page);
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01001447 if (err)
1448 return err;
1449 data_size -= expect;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001450 }
1451
1452 if (dgs) {
Philipp Reisnera0638452011-01-19 14:31:32 +01001453 drbd_csum_bio(mdev, mdev->tconn->integrity_r_tfm, bio, dig_vv);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001454 if (memcmp(dig_in, dig_vv, dgs)) {
1455 dev_err(DEV, "Digest integrity check FAILED. Broken NICs?\n");
Andreas Gruenbacher28284ce2011-03-16 17:54:02 +01001456 return -EINVAL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001457 }
1458 }
1459
1460 D_ASSERT(data_size == 0);
Andreas Gruenbacher28284ce2011-03-16 17:54:02 +01001461 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001462}
1463
1464/* e_end_resync_block() is called via
1465 * drbd_process_done_ee() by asender only */
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001466static int e_end_resync_block(struct drbd_work *w, int unused)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001467{
Andreas Gruenbacher8050e6d2011-02-18 16:12:48 +01001468 struct drbd_peer_request *peer_req =
1469 container_of(w, struct drbd_peer_request, w);
Philipp Reisner00d56942011-02-09 18:09:48 +01001470 struct drbd_conf *mdev = w->mdev;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001471 sector_t sector = peer_req->i.sector;
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001472 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001473
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001474 D_ASSERT(drbd_interval_empty(&peer_req->i));
Philipp Reisnerb411b362009-09-25 16:07:19 -07001475
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001476 if (likely((peer_req->flags & EE_WAS_ERROR) == 0)) {
1477 drbd_set_in_sync(mdev, sector, peer_req->i.size);
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001478 err = drbd_send_ack(mdev, P_RS_WRITE_ACK, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001479 } else {
1480 /* Record failure to sync */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001481 drbd_rs_failed_io(mdev, sector, peer_req->i.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001482
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001483 err = drbd_send_ack(mdev, P_NEG_ACK, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001484 }
1485 dec_unacked(mdev);
1486
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001487 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001488}
1489
1490static int recv_resync_read(struct drbd_conf *mdev, sector_t sector, int data_size) __releases(local)
1491{
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001492 struct drbd_peer_request *peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001493
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001494 peer_req = read_in_block(mdev, ID_SYNCER, sector, data_size);
1495 if (!peer_req)
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001496 goto fail;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001497
1498 dec_rs_pending(mdev);
1499
Philipp Reisnerb411b362009-09-25 16:07:19 -07001500 inc_unacked(mdev);
1501 /* corresponding dec_unacked() in e_end_resync_block()
1502 * respective _drbd_clear_done_ee */
1503
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001504 peer_req->w.cb = e_end_resync_block;
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001505
Philipp Reisner87eeee42011-01-19 14:16:30 +01001506 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001507 list_add(&peer_req->w.list, &mdev->sync_ee);
Philipp Reisner87eeee42011-01-19 14:16:30 +01001508 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001509
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02001510 atomic_add(data_size >> 9, &mdev->rs_sect_ev);
Andreas Gruenbacherfbe29de2011-02-17 16:38:35 +01001511 if (drbd_submit_peer_request(mdev, peer_req, WRITE, DRBD_FAULT_RS_WR) == 0)
Andreas Gruenbachere1c1b0f2011-03-16 17:58:27 +01001512 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001513
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01001514 /* don't care for the reason here */
1515 dev_err(DEV, "submit failed, triggering re-connect\n");
Philipp Reisner87eeee42011-01-19 14:16:30 +01001516 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001517 list_del(&peer_req->w.list);
Philipp Reisner87eeee42011-01-19 14:16:30 +01001518 spin_unlock_irq(&mdev->tconn->req_lock);
Lars Ellenberg22cc37a2010-09-14 20:40:41 +02001519
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001520 drbd_free_ee(mdev, peer_req);
Lars Ellenberg45bb9122010-05-14 17:10:48 +02001521fail:
1522 put_ldev(mdev);
Andreas Gruenbachere1c1b0f2011-03-16 17:58:27 +01001523 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001524}
1525
Andreas Gruenbacher668eebc2011-01-20 17:14:26 +01001526static struct drbd_request *
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01001527find_request(struct drbd_conf *mdev, struct rb_root *root, u64 id,
1528 sector_t sector, bool missing_ok, const char *func)
Andreas Gruenbacher668eebc2011-01-20 17:14:26 +01001529{
Andreas Gruenbacher668eebc2011-01-20 17:14:26 +01001530 struct drbd_request *req;
1531
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01001532 /* Request object according to our peer */
1533 req = (struct drbd_request *)(unsigned long)id;
Andreas Gruenbacher5e472262011-01-27 14:42:51 +01001534 if (drbd_contains_interval(root, sector, &req->i) && req->i.local)
Andreas Gruenbacher668eebc2011-01-20 17:14:26 +01001535 return req;
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01001536 if (!missing_ok) {
1537 dev_err(DEV, "%s: failed to find request %lu, sector %llus\n", func,
1538 (unsigned long)id, (unsigned long long)sector);
1539 }
Andreas Gruenbacher668eebc2011-01-20 17:14:26 +01001540 return NULL;
1541}
1542
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01001543static int receive_DataReply(struct drbd_conf *mdev, enum drbd_packet cmd,
1544 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001545{
1546 struct drbd_request *req;
1547 sector_t sector;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001548 int err;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01001549 struct p_data *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001550
1551 sector = be64_to_cpu(p->sector);
1552
Philipp Reisner87eeee42011-01-19 14:16:30 +01001553 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01001554 req = find_request(mdev, &mdev->read_requests, p->block_id, sector, false, __func__);
Philipp Reisner87eeee42011-01-19 14:16:30 +01001555 spin_unlock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01001556 if (unlikely(!req))
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001557 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001558
Bart Van Assche24c48302011-05-21 18:32:29 +02001559 /* hlist_del(&req->collision) is done in _req_may_be_done, to avoid
Philipp Reisnerb411b362009-09-25 16:07:19 -07001560 * special casing it there for the various failure cases.
1561 * still no race with drbd_fail_pending_reads */
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001562 err = recv_dless_read(mdev, req, sector, data_size);
1563 if (!err)
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01001564 req_mod(req, DATA_RECEIVED);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001565 /* else: nothing. handled from drbd_disconnect...
1566 * I don't think we may complete this just yet
1567 * in case we are "on-disconnect: freeze" */
1568
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001569 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001570}
1571
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01001572static int receive_RSDataReply(struct drbd_conf *mdev, enum drbd_packet cmd,
1573 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001574{
1575 sector_t sector;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001576 int err;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01001577 struct p_data *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001578
1579 sector = be64_to_cpu(p->sector);
1580 D_ASSERT(p->block_id == ID_SYNCER);
1581
1582 if (get_ldev(mdev)) {
1583 /* data is submitted to disk within recv_resync_read.
1584 * corresponding put_ldev done below on error,
Andreas Gruenbacherfcefa622011-02-17 16:46:59 +01001585 * or in drbd_peer_request_endio. */
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001586 err = recv_resync_read(mdev, sector, data_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001587 } else {
1588 if (__ratelimit(&drbd_ratelimit_state))
1589 dev_err(DEV, "Can not write resync data to local disk.\n");
1590
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001591 err = drbd_drain_block(mdev, data_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001592
Lars Ellenberg2b2bf212010-10-06 11:46:55 +02001593 drbd_send_ack_dp(mdev, P_NEG_ACK, p, data_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001594 }
1595
Philipp Reisner778f2712010-07-06 11:14:00 +02001596 atomic_add(data_size >> 9, &mdev->rs_sect_in);
1597
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001598 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001599}
1600
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001601static int w_restart_write(struct drbd_work *w, int cancel)
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001602{
1603 struct drbd_request *req = container_of(w, struct drbd_request, w);
1604 struct drbd_conf *mdev = w->mdev;
1605 struct bio *bio;
1606 unsigned long start_time;
1607 unsigned long flags;
1608
1609 spin_lock_irqsave(&mdev->tconn->req_lock, flags);
1610 if (!expect(req->rq_state & RQ_POSTPONED)) {
1611 spin_unlock_irqrestore(&mdev->tconn->req_lock, flags);
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001612 return -EIO;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001613 }
1614 bio = req->master_bio;
1615 start_time = req->start_time;
1616 /* Postponed requests will not have their master_bio completed! */
1617 __req_mod(req, DISCARD_WRITE, NULL);
1618 spin_unlock_irqrestore(&mdev->tconn->req_lock, flags);
1619
1620 while (__drbd_make_request(mdev, bio, start_time))
1621 /* retry */ ;
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001622 return 0;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001623}
1624
1625static void restart_conflicting_writes(struct drbd_conf *mdev,
1626 sector_t sector, int size)
1627{
1628 struct drbd_interval *i;
1629 struct drbd_request *req;
1630
1631 drbd_for_each_overlap(i, &mdev->write_requests, sector, size) {
1632 if (!i->local)
1633 continue;
1634 req = container_of(i, struct drbd_request, i);
1635 if (req->rq_state & RQ_LOCAL_PENDING ||
1636 !(req->rq_state & RQ_POSTPONED))
1637 continue;
1638 if (expect(list_empty(&req->w.list))) {
1639 req->w.mdev = mdev;
1640 req->w.cb = w_restart_write;
1641 drbd_queue_work(&mdev->tconn->data.work, &req->w);
1642 }
1643 }
1644}
1645
Philipp Reisnerb411b362009-09-25 16:07:19 -07001646/* e_end_block() is called via drbd_process_done_ee().
1647 * this means this function only runs in the asender thread
1648 */
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001649static int e_end_block(struct drbd_work *w, int cancel)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001650{
Andreas Gruenbacher8050e6d2011-02-18 16:12:48 +01001651 struct drbd_peer_request *peer_req =
1652 container_of(w, struct drbd_peer_request, w);
Philipp Reisner00d56942011-02-09 18:09:48 +01001653 struct drbd_conf *mdev = w->mdev;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001654 sector_t sector = peer_req->i.sector;
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001655 int err = 0, pcmd;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001656
Philipp Reisner89e58e72011-01-19 13:12:45 +01001657 if (mdev->tconn->net_conf->wire_protocol == DRBD_PROT_C) {
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001658 if (likely((peer_req->flags & EE_WAS_ERROR) == 0)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07001659 pcmd = (mdev->state.conn >= C_SYNC_SOURCE &&
1660 mdev->state.conn <= C_PAUSED_SYNC_T &&
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001661 peer_req->flags & EE_MAY_SET_IN_SYNC) ?
Philipp Reisnerb411b362009-09-25 16:07:19 -07001662 P_RS_WRITE_ACK : P_WRITE_ACK;
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001663 err = drbd_send_ack(mdev, pcmd, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001664 if (pcmd == P_RS_WRITE_ACK)
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001665 drbd_set_in_sync(mdev, sector, peer_req->i.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001666 } else {
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001667 err = drbd_send_ack(mdev, P_NEG_ACK, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001668 /* we expect it to be marked out of sync anyways...
1669 * maybe assert this? */
1670 }
1671 dec_unacked(mdev);
1672 }
1673 /* we delete from the conflict detection hash _after_ we sent out the
1674 * P_WRITE_ACK / P_NEG_ACK, to get the sequence number right. */
Philipp Reisner89e58e72011-01-19 13:12:45 +01001675 if (mdev->tconn->net_conf->two_primaries) {
Philipp Reisner87eeee42011-01-19 14:16:30 +01001676 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001677 D_ASSERT(!drbd_interval_empty(&peer_req->i));
1678 drbd_remove_epoch_entry_interval(mdev, peer_req);
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001679 if (peer_req->flags & EE_RESTART_REQUESTS)
1680 restart_conflicting_writes(mdev, sector, peer_req->i.size);
Philipp Reisner87eeee42011-01-19 14:16:30 +01001681 spin_unlock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherbb3bfe92011-01-21 15:59:23 +01001682 } else
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001683 D_ASSERT(drbd_interval_empty(&peer_req->i));
Philipp Reisnerb411b362009-09-25 16:07:19 -07001684
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001685 drbd_may_finish_epoch(mdev, peer_req->epoch, EV_PUT + (cancel ? EV_CLEANUP : 0));
Philipp Reisnerb411b362009-09-25 16:07:19 -07001686
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001687 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001688}
1689
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001690static int e_send_ack(struct drbd_work *w, enum drbd_packet ack)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001691{
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001692 struct drbd_conf *mdev = w->mdev;
Andreas Gruenbacher8050e6d2011-02-18 16:12:48 +01001693 struct drbd_peer_request *peer_req =
1694 container_of(w, struct drbd_peer_request, w);
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001695 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001696
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001697 err = drbd_send_ack(mdev, ack, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001698 dec_unacked(mdev);
1699
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001700 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001701}
1702
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001703static int e_send_discard_write(struct drbd_work *w, int unused)
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001704{
1705 return e_send_ack(w, P_DISCARD_WRITE);
1706}
1707
Andreas Gruenbacher99920dc2011-03-16 15:31:39 +01001708static int e_send_retry_write(struct drbd_work *w, int unused)
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001709{
1710 struct drbd_tconn *tconn = w->mdev->tconn;
1711
1712 return e_send_ack(w, tconn->agreed_pro_version >= 100 ?
1713 P_RETRY_WRITE : P_DISCARD_WRITE);
1714}
1715
Andreas Gruenbacher3e394da2011-01-26 18:36:55 +01001716static bool seq_greater(u32 a, u32 b)
1717{
1718 /*
1719 * We assume 32-bit wrap-around here.
1720 * For 24-bit wrap-around, we would have to shift:
1721 * a <<= 8; b <<= 8;
1722 */
1723 return (s32)a - (s32)b > 0;
1724}
1725
1726static u32 seq_max(u32 a, u32 b)
1727{
1728 return seq_greater(a, b) ? a : b;
1729}
1730
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001731static bool need_peer_seq(struct drbd_conf *mdev)
1732{
1733 struct drbd_tconn *tconn = mdev->tconn;
1734
1735 /*
1736 * We only need to keep track of the last packet_seq number of our peer
1737 * if we are in dual-primary mode and we have the discard flag set; see
1738 * handle_write_conflicts().
1739 */
1740 return tconn->net_conf->two_primaries &&
1741 test_bit(DISCARD_CONCURRENT, &tconn->flags);
1742}
1743
Andreas Gruenbacher43ae0772011-02-03 18:42:08 +01001744static void update_peer_seq(struct drbd_conf *mdev, unsigned int peer_seq)
Andreas Gruenbacher3e394da2011-01-26 18:36:55 +01001745{
Lars Ellenberg3c13b682011-02-23 16:10:01 +01001746 unsigned int newest_peer_seq;
Andreas Gruenbacher3e394da2011-01-26 18:36:55 +01001747
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001748 if (need_peer_seq(mdev)) {
1749 spin_lock(&mdev->peer_seq_lock);
Lars Ellenberg3c13b682011-02-23 16:10:01 +01001750 newest_peer_seq = seq_max(mdev->peer_seq, peer_seq);
1751 mdev->peer_seq = newest_peer_seq;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001752 spin_unlock(&mdev->peer_seq_lock);
Lars Ellenberg3c13b682011-02-23 16:10:01 +01001753 /* wake up only if we actually changed mdev->peer_seq */
1754 if (peer_seq == newest_peer_seq)
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001755 wake_up(&mdev->seq_wait);
1756 }
Andreas Gruenbacher3e394da2011-01-26 18:36:55 +01001757}
1758
Philipp Reisnerb411b362009-09-25 16:07:19 -07001759/* Called from receive_Data.
1760 * Synchronize packets on sock with packets on msock.
1761 *
1762 * This is here so even when a P_DATA packet traveling via sock overtook an Ack
1763 * packet traveling on msock, they are still processed in the order they have
1764 * been sent.
1765 *
1766 * Note: we don't care for Ack packets overtaking P_DATA packets.
1767 *
1768 * In case packet_seq is larger than mdev->peer_seq number, there are
1769 * outstanding packets on the msock. We wait for them to arrive.
1770 * In case we are the logically next packet, we update mdev->peer_seq
1771 * ourselves. Correctly handles 32bit wrap around.
1772 *
1773 * Assume we have a 10 GBit connection, that is about 1<<30 byte per second,
1774 * about 1<<21 sectors per second. So "worst" case, we have 1<<3 == 8 seconds
1775 * for the 24bit wrap (historical atomic_t guarantee on some archs), and we have
1776 * 1<<9 == 512 seconds aka ages for the 32bit wrap around...
1777 *
1778 * returns 0 if we may process the packet,
1779 * -ERESTARTSYS if we were interrupted (by disconnect signal). */
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001780static int wait_for_and_update_peer_seq(struct drbd_conf *mdev, const u32 peer_seq)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001781{
1782 DEFINE_WAIT(wait);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001783 long timeout;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001784 int ret;
1785
1786 if (!need_peer_seq(mdev))
1787 return 0;
1788
Philipp Reisnerb411b362009-09-25 16:07:19 -07001789 spin_lock(&mdev->peer_seq_lock);
1790 for (;;) {
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001791 if (!seq_greater(peer_seq - 1, mdev->peer_seq)) {
1792 mdev->peer_seq = seq_max(mdev->peer_seq, peer_seq);
1793 ret = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001794 break;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001795 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07001796 if (signal_pending(current)) {
1797 ret = -ERESTARTSYS;
1798 break;
1799 }
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001800 prepare_to_wait(&mdev->seq_wait, &wait, TASK_INTERRUPTIBLE);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001801 spin_unlock(&mdev->peer_seq_lock);
Andreas Gruenbacher71b1c1e2011-03-01 15:40:43 +01001802 timeout = mdev->tconn->net_conf->ping_timeo*HZ/10;
1803 timeout = schedule_timeout(timeout);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001804 spin_lock(&mdev->peer_seq_lock);
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001805 if (!timeout) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07001806 ret = -ETIMEDOUT;
Andreas Gruenbacher71b1c1e2011-03-01 15:40:43 +01001807 dev_err(DEV, "Timed out waiting for missing ack packets; disconnecting\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07001808 break;
1809 }
1810 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07001811 spin_unlock(&mdev->peer_seq_lock);
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001812 finish_wait(&mdev->seq_wait, &wait);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001813 return ret;
1814}
1815
Lars Ellenberg688593c2010-11-17 22:25:03 +01001816/* see also bio_flags_to_wire()
1817 * DRBD_REQ_*, because we need to semantically map the flags to data packet
1818 * flags and back. We may replicate to other kernel versions. */
1819static unsigned long wire_flags_to_bio(struct drbd_conf *mdev, u32 dpf)
Philipp Reisner76d2e7e2010-08-25 11:58:05 +02001820{
Lars Ellenberg688593c2010-11-17 22:25:03 +01001821 return (dpf & DP_RW_SYNC ? REQ_SYNC : 0) |
1822 (dpf & DP_FUA ? REQ_FUA : 0) |
1823 (dpf & DP_FLUSH ? REQ_FLUSH : 0) |
1824 (dpf & DP_DISCARD ? REQ_DISCARD : 0);
Philipp Reisner76d2e7e2010-08-25 11:58:05 +02001825}
1826
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001827static void fail_postponed_requests(struct drbd_conf *mdev, sector_t sector,
1828 unsigned int size)
1829{
1830 struct drbd_interval *i;
1831
1832 repeat:
1833 drbd_for_each_overlap(i, &mdev->write_requests, sector, size) {
1834 struct drbd_request *req;
1835 struct bio_and_error m;
1836
1837 if (!i->local)
1838 continue;
1839 req = container_of(i, struct drbd_request, i);
1840 if (!(req->rq_state & RQ_POSTPONED))
1841 continue;
1842 req->rq_state &= ~RQ_POSTPONED;
1843 __req_mod(req, NEG_ACKED, &m);
1844 spin_unlock_irq(&mdev->tconn->req_lock);
1845 if (m.bio)
1846 complete_master_bio(mdev, &m);
1847 spin_lock_irq(&mdev->tconn->req_lock);
1848 goto repeat;
1849 }
1850}
1851
1852static int handle_write_conflicts(struct drbd_conf *mdev,
1853 struct drbd_peer_request *peer_req)
1854{
1855 struct drbd_tconn *tconn = mdev->tconn;
1856 bool resolve_conflicts = test_bit(DISCARD_CONCURRENT, &tconn->flags);
1857 sector_t sector = peer_req->i.sector;
1858 const unsigned int size = peer_req->i.size;
1859 struct drbd_interval *i;
1860 bool equal;
1861 int err;
1862
1863 /*
1864 * Inserting the peer request into the write_requests tree will prevent
1865 * new conflicting local requests from being added.
1866 */
1867 drbd_insert_interval(&mdev->write_requests, &peer_req->i);
1868
1869 repeat:
1870 drbd_for_each_overlap(i, &mdev->write_requests, sector, size) {
1871 if (i == &peer_req->i)
1872 continue;
1873
1874 if (!i->local) {
1875 /*
1876 * Our peer has sent a conflicting remote request; this
1877 * should not happen in a two-node setup. Wait for the
1878 * earlier peer request to complete.
1879 */
1880 err = drbd_wait_misc(mdev, i);
1881 if (err)
1882 goto out;
1883 goto repeat;
1884 }
1885
1886 equal = i->sector == sector && i->size == size;
1887 if (resolve_conflicts) {
1888 /*
1889 * If the peer request is fully contained within the
1890 * overlapping request, it can be discarded; otherwise,
1891 * it will be retried once all overlapping requests
1892 * have completed.
1893 */
1894 bool discard = i->sector <= sector && i->sector +
1895 (i->size >> 9) >= sector + (size >> 9);
1896
1897 if (!equal)
1898 dev_alert(DEV, "Concurrent writes detected: "
1899 "local=%llus +%u, remote=%llus +%u, "
1900 "assuming %s came first\n",
1901 (unsigned long long)i->sector, i->size,
1902 (unsigned long long)sector, size,
1903 discard ? "local" : "remote");
1904
1905 inc_unacked(mdev);
1906 peer_req->w.cb = discard ? e_send_discard_write :
1907 e_send_retry_write;
1908 list_add_tail(&peer_req->w.list, &mdev->done_ee);
1909 wake_asender(mdev->tconn);
1910
1911 err = -ENOENT;
1912 goto out;
1913 } else {
1914 struct drbd_request *req =
1915 container_of(i, struct drbd_request, i);
1916
1917 if (!equal)
1918 dev_alert(DEV, "Concurrent writes detected: "
1919 "local=%llus +%u, remote=%llus +%u\n",
1920 (unsigned long long)i->sector, i->size,
1921 (unsigned long long)sector, size);
1922
1923 if (req->rq_state & RQ_LOCAL_PENDING ||
1924 !(req->rq_state & RQ_POSTPONED)) {
1925 /*
1926 * Wait for the node with the discard flag to
1927 * decide if this request will be discarded or
1928 * retried. Requests that are discarded will
1929 * disappear from the write_requests tree.
1930 *
1931 * In addition, wait for the conflicting
1932 * request to finish locally before submitting
1933 * the conflicting peer request.
1934 */
1935 err = drbd_wait_misc(mdev, &req->i);
1936 if (err) {
1937 _conn_request_state(mdev->tconn,
1938 NS(conn, C_TIMEOUT),
1939 CS_HARD);
1940 fail_postponed_requests(mdev, sector, size);
1941 goto out;
1942 }
1943 goto repeat;
1944 }
1945 /*
1946 * Remember to restart the conflicting requests after
1947 * the new peer request has completed.
1948 */
1949 peer_req->flags |= EE_RESTART_REQUESTS;
1950 }
1951 }
1952 err = 0;
1953
1954 out:
1955 if (err)
1956 drbd_remove_epoch_entry_interval(mdev, peer_req);
1957 return err;
1958}
1959
Philipp Reisnerb411b362009-09-25 16:07:19 -07001960/* mirrored write */
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01001961static int receive_Data(struct drbd_conf *mdev, enum drbd_packet cmd,
1962 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07001963{
1964 sector_t sector;
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001965 struct drbd_peer_request *peer_req;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01001966 struct p_data *p = mdev->tconn->data.rbuf;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001967 u32 peer_seq = be32_to_cpu(p->seq_num);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001968 int rw = WRITE;
1969 u32 dp_flags;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001970 int err;
1971
Philipp Reisnerb411b362009-09-25 16:07:19 -07001972 if (!get_ldev(mdev)) {
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001973 int err2;
1974
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01001975 err = wait_for_and_update_peer_seq(mdev, peer_seq);
Lars Ellenberg2b2bf212010-10-06 11:46:55 +02001976 drbd_send_ack_dp(mdev, P_NEG_ACK, p, data_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07001977 atomic_inc(&mdev->current_epoch->epoch_size);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001978 err2 = drbd_drain_block(mdev, data_size);
1979 if (!err)
1980 err = err2;
1981 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001982 }
1983
Andreas Gruenbacherfcefa622011-02-17 16:46:59 +01001984 /*
1985 * Corresponding put_ldev done either below (on various errors), or in
1986 * drbd_peer_request_endio, if we successfully submit the data at the
1987 * end of this function.
1988 */
Philipp Reisnerb411b362009-09-25 16:07:19 -07001989
1990 sector = be64_to_cpu(p->sector);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001991 peer_req = read_in_block(mdev, p->block_id, sector, data_size);
1992 if (!peer_req) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07001993 put_ldev(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01001994 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001995 }
1996
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01001997 peer_req->w.cb = e_end_block;
Philipp Reisnerb411b362009-09-25 16:07:19 -07001998
Lars Ellenberg688593c2010-11-17 22:25:03 +01001999 dp_flags = be32_to_cpu(p->dp_flags);
2000 rw |= wire_flags_to_bio(mdev, dp_flags);
2001
2002 if (dp_flags & DP_MAY_SET_IN_SYNC)
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002003 peer_req->flags |= EE_MAY_SET_IN_SYNC;
Lars Ellenberg688593c2010-11-17 22:25:03 +01002004
Philipp Reisnerb411b362009-09-25 16:07:19 -07002005 spin_lock(&mdev->epoch_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002006 peer_req->epoch = mdev->current_epoch;
2007 atomic_inc(&peer_req->epoch->epoch_size);
2008 atomic_inc(&peer_req->epoch->active);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002009 spin_unlock(&mdev->epoch_lock);
2010
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01002011 if (mdev->tconn->net_conf->two_primaries) {
2012 err = wait_for_and_update_peer_seq(mdev, peer_seq);
2013 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002014 goto out_interrupted;
Philipp Reisner87eeee42011-01-19 14:16:30 +01002015 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01002016 err = handle_write_conflicts(mdev, peer_req);
2017 if (err) {
2018 spin_unlock_irq(&mdev->tconn->req_lock);
2019 if (err == -ENOENT) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002020 put_ldev(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002021 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002022 }
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01002023 goto out_interrupted;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002024 }
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01002025 } else
2026 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002027 list_add(&peer_req->w.list, &mdev->active_ee);
Philipp Reisner87eeee42011-01-19 14:16:30 +01002028 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002029
Philipp Reisner89e58e72011-01-19 13:12:45 +01002030 switch (mdev->tconn->net_conf->wire_protocol) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002031 case DRBD_PROT_C:
2032 inc_unacked(mdev);
2033 /* corresponding dec_unacked() in e_end_block()
2034 * respective _drbd_clear_done_ee */
2035 break;
2036 case DRBD_PROT_B:
2037 /* I really don't like it that the receiver thread
2038 * sends on the msock, but anyways */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002039 drbd_send_ack(mdev, P_RECV_ACK, peer_req);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002040 break;
2041 case DRBD_PROT_A:
2042 /* nothing to do */
2043 break;
2044 }
2045
Lars Ellenberg6719fb02010-10-18 23:04:07 +02002046 if (mdev->state.pdsk < D_INCONSISTENT) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002047 /* In case we have the only disk of the cluster, */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002048 drbd_set_out_of_sync(mdev, peer_req->i.sector, peer_req->i.size);
2049 peer_req->flags |= EE_CALL_AL_COMPLETE_IO;
2050 peer_req->flags &= ~EE_MAY_SET_IN_SYNC;
2051 drbd_al_begin_io(mdev, peer_req->i.sector);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002052 }
2053
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002054 err = drbd_submit_peer_request(mdev, peer_req, rw, DRBD_FAULT_DT_WR);
2055 if (!err)
2056 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002057
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01002058 /* don't care for the reason here */
2059 dev_err(DEV, "submit failed, triggering re-connect\n");
Philipp Reisner87eeee42011-01-19 14:16:30 +01002060 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002061 list_del(&peer_req->w.list);
2062 drbd_remove_epoch_entry_interval(mdev, peer_req);
Philipp Reisner87eeee42011-01-19 14:16:30 +01002063 spin_unlock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002064 if (peer_req->flags & EE_CALL_AL_COMPLETE_IO)
2065 drbd_al_complete_io(mdev, peer_req->i.sector);
Lars Ellenberg22cc37a2010-09-14 20:40:41 +02002066
Philipp Reisnerb411b362009-09-25 16:07:19 -07002067out_interrupted:
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002068 drbd_may_finish_epoch(mdev, peer_req->epoch, EV_PUT + EV_CLEANUP);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002069 put_ldev(mdev);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002070 drbd_free_ee(mdev, peer_req);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002071 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002072}
2073
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002074/* We may throttle resync, if the lower device seems to be busy,
2075 * and current sync rate is above c_min_rate.
2076 *
2077 * To decide whether or not the lower device is busy, we use a scheme similar
2078 * to MD RAID is_mddev_idle(): if the partition stats reveal "significant"
2079 * (more than 64 sectors) of activity we cannot account for with our own resync
2080 * activity, it obviously is "busy".
2081 *
2082 * The current sync rate used here uses only the most recent two step marks,
2083 * to have a short time average so we can react faster.
2084 */
Philipp Reisnere3555d82010-11-07 15:56:29 +01002085int drbd_rs_should_slow_down(struct drbd_conf *mdev, sector_t sector)
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002086{
2087 struct gendisk *disk = mdev->ldev->backing_bdev->bd_contains->bd_disk;
2088 unsigned long db, dt, dbdt;
Philipp Reisnere3555d82010-11-07 15:56:29 +01002089 struct lc_element *tmp;
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002090 int curr_events;
2091 int throttle = 0;
2092
2093 /* feature disabled? */
Lars Ellenbergf3990022011-03-23 14:31:09 +01002094 if (mdev->ldev->dc.c_min_rate == 0)
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002095 return 0;
2096
Philipp Reisnere3555d82010-11-07 15:56:29 +01002097 spin_lock_irq(&mdev->al_lock);
2098 tmp = lc_find(mdev->resync, BM_SECT_TO_EXT(sector));
2099 if (tmp) {
2100 struct bm_extent *bm_ext = lc_entry(tmp, struct bm_extent, lce);
2101 if (test_bit(BME_PRIORITY, &bm_ext->flags)) {
2102 spin_unlock_irq(&mdev->al_lock);
2103 return 0;
2104 }
2105 /* Do not slow down if app IO is already waiting for this extent */
2106 }
2107 spin_unlock_irq(&mdev->al_lock);
2108
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002109 curr_events = (int)part_stat_read(&disk->part0, sectors[0]) +
2110 (int)part_stat_read(&disk->part0, sectors[1]) -
2111 atomic_read(&mdev->rs_sect_ev);
Philipp Reisnere3555d82010-11-07 15:56:29 +01002112
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002113 if (!mdev->rs_last_events || curr_events - mdev->rs_last_events > 64) {
2114 unsigned long rs_left;
2115 int i;
2116
2117 mdev->rs_last_events = curr_events;
2118
2119 /* sync speed average over the last 2*DRBD_SYNC_MARK_STEP,
2120 * approx. */
Lars Ellenberg2649f082010-11-05 10:05:47 +01002121 i = (mdev->rs_last_mark + DRBD_SYNC_MARKS-1) % DRBD_SYNC_MARKS;
2122
2123 if (mdev->state.conn == C_VERIFY_S || mdev->state.conn == C_VERIFY_T)
2124 rs_left = mdev->ov_left;
2125 else
2126 rs_left = drbd_bm_total_weight(mdev) - mdev->rs_failed;
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002127
2128 dt = ((long)jiffies - (long)mdev->rs_mark_time[i]) / HZ;
2129 if (!dt)
2130 dt++;
2131 db = mdev->rs_mark_left[i] - rs_left;
2132 dbdt = Bit2KB(db/dt);
2133
Lars Ellenbergf3990022011-03-23 14:31:09 +01002134 if (dbdt > mdev->ldev->dc.c_min_rate)
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002135 throttle = 1;
2136 }
2137 return throttle;
2138}
2139
2140
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01002141static int receive_DataRequest(struct drbd_conf *mdev, enum drbd_packet cmd,
2142 unsigned int digest_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002143{
2144 sector_t sector;
2145 const sector_t capacity = drbd_get_capacity(mdev->this_bdev);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002146 struct drbd_peer_request *peer_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002147 struct digest_info *di = NULL;
Philipp Reisnerb18b37b2010-10-13 15:32:44 +02002148 int size, verb;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002149 unsigned int fault_type;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01002150 struct p_block_req *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002151
2152 sector = be64_to_cpu(p->sector);
2153 size = be32_to_cpu(p->blksize);
2154
Andreas Gruenbacherc670a392011-02-21 12:41:39 +01002155 if (size <= 0 || !IS_ALIGNED(size, 512) || size > DRBD_MAX_BIO_SIZE) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002156 dev_err(DEV, "%s:%d: sector: %llus, size: %u\n", __FILE__, __LINE__,
2157 (unsigned long long)sector, size);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002158 return -EINVAL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002159 }
2160 if (sector + (size>>9) > capacity) {
2161 dev_err(DEV, "%s:%d: sector: %llus, size: %u\n", __FILE__, __LINE__,
2162 (unsigned long long)sector, size);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002163 return -EINVAL;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002164 }
2165
2166 if (!get_ldev_if_state(mdev, D_UP_TO_DATE)) {
Philipp Reisnerb18b37b2010-10-13 15:32:44 +02002167 verb = 1;
2168 switch (cmd) {
2169 case P_DATA_REQUEST:
2170 drbd_send_ack_rp(mdev, P_NEG_DREPLY, p);
2171 break;
2172 case P_RS_DATA_REQUEST:
2173 case P_CSUM_RS_REQUEST:
2174 case P_OV_REQUEST:
2175 drbd_send_ack_rp(mdev, P_NEG_RS_DREPLY , p);
2176 break;
2177 case P_OV_REPLY:
2178 verb = 0;
2179 dec_rs_pending(mdev);
2180 drbd_send_ack_ex(mdev, P_OV_RESULT, sector, size, ID_IN_SYNC);
2181 break;
2182 default:
2183 dev_err(DEV, "unexpected command (%s) in receive_DataRequest\n",
2184 cmdname(cmd));
2185 }
2186 if (verb && __ratelimit(&drbd_ratelimit_state))
Philipp Reisnerb411b362009-09-25 16:07:19 -07002187 dev_err(DEV, "Can not satisfy peer's read request, "
2188 "no local data.\n");
Philipp Reisnerb18b37b2010-10-13 15:32:44 +02002189
Lars Ellenberga821cc42010-09-06 12:31:37 +02002190 /* drain possibly payload */
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002191 return drbd_drain_block(mdev, digest_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002192 }
2193
2194 /* GFP_NOIO, because we must not cause arbitrary write-out: in a DRBD
2195 * "criss-cross" setup, that might cause write-out on some other DRBD,
2196 * which in turn might block on the other node at this very place. */
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002197 peer_req = drbd_alloc_ee(mdev, p->block_id, sector, size, GFP_NOIO);
2198 if (!peer_req) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002199 put_ldev(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002200 return -ENOMEM;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002201 }
2202
Philipp Reisner02918be2010-08-20 14:35:10 +02002203 switch (cmd) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002204 case P_DATA_REQUEST:
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002205 peer_req->w.cb = w_e_end_data_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002206 fault_type = DRBD_FAULT_DT_RD;
Lars Ellenberg80a40e42010-08-11 23:28:00 +02002207 /* application IO, don't drbd_rs_begin_io */
2208 goto submit;
2209
Philipp Reisnerb411b362009-09-25 16:07:19 -07002210 case P_RS_DATA_REQUEST:
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002211 peer_req->w.cb = w_e_end_rsdata_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002212 fault_type = DRBD_FAULT_RS_RD;
Lars Ellenberg5f9915b2010-11-09 14:15:24 +01002213 /* used in the sector offset progress display */
2214 mdev->bm_resync_fo = BM_SECT_TO_BIT(sector);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002215 break;
2216
2217 case P_OV_REPLY:
2218 case P_CSUM_RS_REQUEST:
2219 fault_type = DRBD_FAULT_RS_RD;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002220 di = kmalloc(sizeof(*di) + digest_size, GFP_NOIO);
2221 if (!di)
2222 goto out_free_e;
2223
2224 di->digest_size = digest_size;
2225 di->digest = (((char *)di)+sizeof(struct digest_info));
2226
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002227 peer_req->digest = di;
2228 peer_req->flags |= EE_HAS_DIGEST;
Lars Ellenbergc36c3ce2010-08-11 20:42:55 +02002229
Philipp Reisnerde0ff332011-02-07 16:56:20 +01002230 if (drbd_recv(mdev->tconn, di->digest, digest_size) != digest_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002231 goto out_free_e;
2232
Philipp Reisner02918be2010-08-20 14:35:10 +02002233 if (cmd == P_CSUM_RS_REQUEST) {
Philipp Reisner31890f42011-01-19 14:12:51 +01002234 D_ASSERT(mdev->tconn->agreed_pro_version >= 89);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002235 peer_req->w.cb = w_e_end_csum_rs_req;
Lars Ellenberg5f9915b2010-11-09 14:15:24 +01002236 /* used in the sector offset progress display */
2237 mdev->bm_resync_fo = BM_SECT_TO_BIT(sector);
Philipp Reisner02918be2010-08-20 14:35:10 +02002238 } else if (cmd == P_OV_REPLY) {
Lars Ellenberg2649f082010-11-05 10:05:47 +01002239 /* track progress, we may need to throttle */
2240 atomic_add(size >> 9, &mdev->rs_sect_in);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002241 peer_req->w.cb = w_e_end_ov_reply;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002242 dec_rs_pending(mdev);
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002243 /* drbd_rs_begin_io done when we sent this request,
2244 * but accounting still needs to be done. */
2245 goto submit_for_resync;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002246 }
2247 break;
2248
2249 case P_OV_REQUEST:
Philipp Reisnerb411b362009-09-25 16:07:19 -07002250 if (mdev->ov_start_sector == ~(sector_t)0 &&
Philipp Reisner31890f42011-01-19 14:12:51 +01002251 mdev->tconn->agreed_pro_version >= 90) {
Lars Ellenbergde228bb2010-11-05 09:43:15 +01002252 unsigned long now = jiffies;
2253 int i;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002254 mdev->ov_start_sector = sector;
2255 mdev->ov_position = sector;
Lars Ellenberg30b743a2010-11-05 09:39:06 +01002256 mdev->ov_left = drbd_bm_bits(mdev) - BM_SECT_TO_BIT(sector);
2257 mdev->rs_total = mdev->ov_left;
Lars Ellenbergde228bb2010-11-05 09:43:15 +01002258 for (i = 0; i < DRBD_SYNC_MARKS; i++) {
2259 mdev->rs_mark_left[i] = mdev->ov_left;
2260 mdev->rs_mark_time[i] = now;
2261 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07002262 dev_info(DEV, "Online Verify start sector: %llu\n",
2263 (unsigned long long)sector);
2264 }
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002265 peer_req->w.cb = w_e_end_ov_req;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002266 fault_type = DRBD_FAULT_RS_RD;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002267 break;
2268
Philipp Reisnerb411b362009-09-25 16:07:19 -07002269 default:
2270 dev_err(DEV, "unexpected command (%s) in receive_DataRequest\n",
Philipp Reisner02918be2010-08-20 14:35:10 +02002271 cmdname(cmd));
Philipp Reisnerb411b362009-09-25 16:07:19 -07002272 fault_type = DRBD_FAULT_MAX;
Lars Ellenberg80a40e42010-08-11 23:28:00 +02002273 goto out_free_e;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002274 }
2275
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002276 /* Throttle, drbd_rs_begin_io and submit should become asynchronous
2277 * wrt the receiver, but it is not as straightforward as it may seem.
2278 * Various places in the resync start and stop logic assume resync
2279 * requests are processed in order, requeuing this on the worker thread
2280 * introduces a bunch of new code for synchronization between threads.
2281 *
2282 * Unlimited throttling before drbd_rs_begin_io may stall the resync
2283 * "forever", throttling after drbd_rs_begin_io will lock that extent
2284 * for application writes for the same time. For now, just throttle
2285 * here, where the rest of the code expects the receiver to sleep for
2286 * a while, anyways.
2287 */
Philipp Reisnerb411b362009-09-25 16:07:19 -07002288
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002289 /* Throttle before drbd_rs_begin_io, as that locks out application IO;
2290 * this defers syncer requests for some time, before letting at least
2291 * on request through. The resync controller on the receiving side
2292 * will adapt to the incoming rate accordingly.
2293 *
2294 * We cannot throttle here if remote is Primary/SyncTarget:
2295 * we would also throttle its application reads.
2296 * In that case, throttling is done on the SyncTarget only.
2297 */
Philipp Reisnere3555d82010-11-07 15:56:29 +01002298 if (mdev->state.peer != R_PRIMARY && drbd_rs_should_slow_down(mdev, sector))
2299 schedule_timeout_uninterruptible(HZ/10);
2300 if (drbd_rs_begin_io(mdev, sector))
Lars Ellenberg80a40e42010-08-11 23:28:00 +02002301 goto out_free_e;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002302
Lars Ellenberg0f0601f2010-08-11 23:40:24 +02002303submit_for_resync:
2304 atomic_add(size >> 9, &mdev->rs_sect_ev);
2305
Lars Ellenberg80a40e42010-08-11 23:28:00 +02002306submit:
Philipp Reisnerb411b362009-09-25 16:07:19 -07002307 inc_unacked(mdev);
Philipp Reisner87eeee42011-01-19 14:16:30 +01002308 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002309 list_add_tail(&peer_req->w.list, &mdev->read_ee);
Philipp Reisner87eeee42011-01-19 14:16:30 +01002310 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002311
Andreas Gruenbacherfbe29de2011-02-17 16:38:35 +01002312 if (drbd_submit_peer_request(mdev, peer_req, READ, fault_type) == 0)
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002313 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002314
Lars Ellenberg10f6d9922011-01-24 14:47:09 +01002315 /* don't care for the reason here */
2316 dev_err(DEV, "submit failed, triggering re-connect\n");
Philipp Reisner87eeee42011-01-19 14:16:30 +01002317 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002318 list_del(&peer_req->w.list);
Philipp Reisner87eeee42011-01-19 14:16:30 +01002319 spin_unlock_irq(&mdev->tconn->req_lock);
Lars Ellenberg22cc37a2010-09-14 20:40:41 +02002320 /* no drbd_rs_complete_io(), we are dropping the connection anyways */
2321
Philipp Reisnerb411b362009-09-25 16:07:19 -07002322out_free_e:
Philipp Reisnerb411b362009-09-25 16:07:19 -07002323 put_ldev(mdev);
Andreas Gruenbacherdb830c42011-02-04 15:57:48 +01002324 drbd_free_ee(mdev, peer_req);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002325 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002326}
2327
2328static int drbd_asb_recover_0p(struct drbd_conf *mdev) __must_hold(local)
2329{
2330 int self, peer, rv = -100;
2331 unsigned long ch_self, ch_peer;
2332
2333 self = mdev->ldev->md.uuid[UI_BITMAP] & 1;
2334 peer = mdev->p_uuid[UI_BITMAP] & 1;
2335
2336 ch_peer = mdev->p_uuid[UI_SIZE];
2337 ch_self = mdev->comm_bm_set;
2338
Philipp Reisner89e58e72011-01-19 13:12:45 +01002339 switch (mdev->tconn->net_conf->after_sb_0p) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002340 case ASB_CONSENSUS:
2341 case ASB_DISCARD_SECONDARY:
2342 case ASB_CALL_HELPER:
2343 dev_err(DEV, "Configuration error.\n");
2344 break;
2345 case ASB_DISCONNECT:
2346 break;
2347 case ASB_DISCARD_YOUNGER_PRI:
2348 if (self == 0 && peer == 1) {
2349 rv = -1;
2350 break;
2351 }
2352 if (self == 1 && peer == 0) {
2353 rv = 1;
2354 break;
2355 }
2356 /* Else fall through to one of the other strategies... */
2357 case ASB_DISCARD_OLDER_PRI:
2358 if (self == 0 && peer == 1) {
2359 rv = 1;
2360 break;
2361 }
2362 if (self == 1 && peer == 0) {
2363 rv = -1;
2364 break;
2365 }
2366 /* Else fall through to one of the other strategies... */
Lars Ellenbergad19bf62009-10-14 09:36:49 +02002367 dev_warn(DEV, "Discard younger/older primary did not find a decision\n"
Philipp Reisnerb411b362009-09-25 16:07:19 -07002368 "Using discard-least-changes instead\n");
2369 case ASB_DISCARD_ZERO_CHG:
2370 if (ch_peer == 0 && ch_self == 0) {
Philipp Reisner25703f82011-02-07 14:35:25 +01002371 rv = test_bit(DISCARD_CONCURRENT, &mdev->tconn->flags)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002372 ? -1 : 1;
2373 break;
2374 } else {
2375 if (ch_peer == 0) { rv = 1; break; }
2376 if (ch_self == 0) { rv = -1; break; }
2377 }
Philipp Reisner89e58e72011-01-19 13:12:45 +01002378 if (mdev->tconn->net_conf->after_sb_0p == ASB_DISCARD_ZERO_CHG)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002379 break;
2380 case ASB_DISCARD_LEAST_CHG:
2381 if (ch_self < ch_peer)
2382 rv = -1;
2383 else if (ch_self > ch_peer)
2384 rv = 1;
2385 else /* ( ch_self == ch_peer ) */
2386 /* Well, then use something else. */
Philipp Reisner25703f82011-02-07 14:35:25 +01002387 rv = test_bit(DISCARD_CONCURRENT, &mdev->tconn->flags)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002388 ? -1 : 1;
2389 break;
2390 case ASB_DISCARD_LOCAL:
2391 rv = -1;
2392 break;
2393 case ASB_DISCARD_REMOTE:
2394 rv = 1;
2395 }
2396
2397 return rv;
2398}
2399
2400static int drbd_asb_recover_1p(struct drbd_conf *mdev) __must_hold(local)
2401{
Andreas Gruenbacher6184ea22010-12-09 14:23:27 +01002402 int hg, rv = -100;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002403
Philipp Reisner89e58e72011-01-19 13:12:45 +01002404 switch (mdev->tconn->net_conf->after_sb_1p) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002405 case ASB_DISCARD_YOUNGER_PRI:
2406 case ASB_DISCARD_OLDER_PRI:
2407 case ASB_DISCARD_LEAST_CHG:
2408 case ASB_DISCARD_LOCAL:
2409 case ASB_DISCARD_REMOTE:
2410 dev_err(DEV, "Configuration error.\n");
2411 break;
2412 case ASB_DISCONNECT:
2413 break;
2414 case ASB_CONSENSUS:
2415 hg = drbd_asb_recover_0p(mdev);
2416 if (hg == -1 && mdev->state.role == R_SECONDARY)
2417 rv = hg;
2418 if (hg == 1 && mdev->state.role == R_PRIMARY)
2419 rv = hg;
2420 break;
2421 case ASB_VIOLENTLY:
2422 rv = drbd_asb_recover_0p(mdev);
2423 break;
2424 case ASB_DISCARD_SECONDARY:
2425 return mdev->state.role == R_PRIMARY ? 1 : -1;
2426 case ASB_CALL_HELPER:
2427 hg = drbd_asb_recover_0p(mdev);
2428 if (hg == -1 && mdev->state.role == R_PRIMARY) {
Andreas Gruenbacherbb437942010-12-09 14:02:35 +01002429 enum drbd_state_rv rv2;
2430
2431 drbd_set_role(mdev, R_SECONDARY, 0);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002432 /* drbd_change_state() does not sleep while in SS_IN_TRANSIENT_STATE,
2433 * we might be here in C_WF_REPORT_PARAMS which is transient.
2434 * we do not need to wait for the after state change work either. */
Andreas Gruenbacherbb437942010-12-09 14:02:35 +01002435 rv2 = drbd_change_state(mdev, CS_VERBOSE, NS(role, R_SECONDARY));
2436 if (rv2 != SS_SUCCESS) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002437 drbd_khelper(mdev, "pri-lost-after-sb");
2438 } else {
2439 dev_warn(DEV, "Successfully gave up primary role.\n");
2440 rv = hg;
2441 }
2442 } else
2443 rv = hg;
2444 }
2445
2446 return rv;
2447}
2448
2449static int drbd_asb_recover_2p(struct drbd_conf *mdev) __must_hold(local)
2450{
Andreas Gruenbacher6184ea22010-12-09 14:23:27 +01002451 int hg, rv = -100;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002452
Philipp Reisner89e58e72011-01-19 13:12:45 +01002453 switch (mdev->tconn->net_conf->after_sb_2p) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002454 case ASB_DISCARD_YOUNGER_PRI:
2455 case ASB_DISCARD_OLDER_PRI:
2456 case ASB_DISCARD_LEAST_CHG:
2457 case ASB_DISCARD_LOCAL:
2458 case ASB_DISCARD_REMOTE:
2459 case ASB_CONSENSUS:
2460 case ASB_DISCARD_SECONDARY:
2461 dev_err(DEV, "Configuration error.\n");
2462 break;
2463 case ASB_VIOLENTLY:
2464 rv = drbd_asb_recover_0p(mdev);
2465 break;
2466 case ASB_DISCONNECT:
2467 break;
2468 case ASB_CALL_HELPER:
2469 hg = drbd_asb_recover_0p(mdev);
2470 if (hg == -1) {
Andreas Gruenbacherbb437942010-12-09 14:02:35 +01002471 enum drbd_state_rv rv2;
2472
Philipp Reisnerb411b362009-09-25 16:07:19 -07002473 /* drbd_change_state() does not sleep while in SS_IN_TRANSIENT_STATE,
2474 * we might be here in C_WF_REPORT_PARAMS which is transient.
2475 * we do not need to wait for the after state change work either. */
Andreas Gruenbacherbb437942010-12-09 14:02:35 +01002476 rv2 = drbd_change_state(mdev, CS_VERBOSE, NS(role, R_SECONDARY));
2477 if (rv2 != SS_SUCCESS) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002478 drbd_khelper(mdev, "pri-lost-after-sb");
2479 } else {
2480 dev_warn(DEV, "Successfully gave up primary role.\n");
2481 rv = hg;
2482 }
2483 } else
2484 rv = hg;
2485 }
2486
2487 return rv;
2488}
2489
2490static void drbd_uuid_dump(struct drbd_conf *mdev, char *text, u64 *uuid,
2491 u64 bits, u64 flags)
2492{
2493 if (!uuid) {
2494 dev_info(DEV, "%s uuid info vanished while I was looking!\n", text);
2495 return;
2496 }
2497 dev_info(DEV, "%s %016llX:%016llX:%016llX:%016llX bits:%llu flags:%llX\n",
2498 text,
2499 (unsigned long long)uuid[UI_CURRENT],
2500 (unsigned long long)uuid[UI_BITMAP],
2501 (unsigned long long)uuid[UI_HISTORY_START],
2502 (unsigned long long)uuid[UI_HISTORY_END],
2503 (unsigned long long)bits,
2504 (unsigned long long)flags);
2505}
2506
2507/*
2508 100 after split brain try auto recover
2509 2 C_SYNC_SOURCE set BitMap
2510 1 C_SYNC_SOURCE use BitMap
2511 0 no Sync
2512 -1 C_SYNC_TARGET use BitMap
2513 -2 C_SYNC_TARGET set BitMap
2514 -100 after split brain, disconnect
2515-1000 unrelated data
Philipp Reisner4a23f262011-01-11 17:42:17 +01002516-1091 requires proto 91
2517-1096 requires proto 96
Philipp Reisnerb411b362009-09-25 16:07:19 -07002518 */
2519static int drbd_uuid_compare(struct drbd_conf *mdev, int *rule_nr) __must_hold(local)
2520{
2521 u64 self, peer;
2522 int i, j;
2523
2524 self = mdev->ldev->md.uuid[UI_CURRENT] & ~((u64)1);
2525 peer = mdev->p_uuid[UI_CURRENT] & ~((u64)1);
2526
2527 *rule_nr = 10;
2528 if (self == UUID_JUST_CREATED && peer == UUID_JUST_CREATED)
2529 return 0;
2530
2531 *rule_nr = 20;
2532 if ((self == UUID_JUST_CREATED || self == (u64)0) &&
2533 peer != UUID_JUST_CREATED)
2534 return -2;
2535
2536 *rule_nr = 30;
2537 if (self != UUID_JUST_CREATED &&
2538 (peer == UUID_JUST_CREATED || peer == (u64)0))
2539 return 2;
2540
2541 if (self == peer) {
2542 int rct, dc; /* roles at crash time */
2543
2544 if (mdev->p_uuid[UI_BITMAP] == (u64)0 && mdev->ldev->md.uuid[UI_BITMAP] != (u64)0) {
2545
Philipp Reisner31890f42011-01-19 14:12:51 +01002546 if (mdev->tconn->agreed_pro_version < 91)
Philipp Reisner4a23f262011-01-11 17:42:17 +01002547 return -1091;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002548
2549 if ((mdev->ldev->md.uuid[UI_BITMAP] & ~((u64)1)) == (mdev->p_uuid[UI_HISTORY_START] & ~((u64)1)) &&
2550 (mdev->ldev->md.uuid[UI_HISTORY_START] & ~((u64)1)) == (mdev->p_uuid[UI_HISTORY_START + 1] & ~((u64)1))) {
2551 dev_info(DEV, "was SyncSource, missed the resync finished event, corrected myself:\n");
2552 drbd_uuid_set_bm(mdev, 0UL);
2553
2554 drbd_uuid_dump(mdev, "self", mdev->ldev->md.uuid,
2555 mdev->state.disk >= D_NEGOTIATING ? drbd_bm_total_weight(mdev) : 0, 0);
2556 *rule_nr = 34;
2557 } else {
2558 dev_info(DEV, "was SyncSource (peer failed to write sync_uuid)\n");
2559 *rule_nr = 36;
2560 }
2561
2562 return 1;
2563 }
2564
2565 if (mdev->ldev->md.uuid[UI_BITMAP] == (u64)0 && mdev->p_uuid[UI_BITMAP] != (u64)0) {
2566
Philipp Reisner31890f42011-01-19 14:12:51 +01002567 if (mdev->tconn->agreed_pro_version < 91)
Philipp Reisner4a23f262011-01-11 17:42:17 +01002568 return -1091;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002569
2570 if ((mdev->ldev->md.uuid[UI_HISTORY_START] & ~((u64)1)) == (mdev->p_uuid[UI_BITMAP] & ~((u64)1)) &&
2571 (mdev->ldev->md.uuid[UI_HISTORY_START + 1] & ~((u64)1)) == (mdev->p_uuid[UI_HISTORY_START] & ~((u64)1))) {
2572 dev_info(DEV, "was SyncTarget, peer missed the resync finished event, corrected peer:\n");
2573
2574 mdev->p_uuid[UI_HISTORY_START + 1] = mdev->p_uuid[UI_HISTORY_START];
2575 mdev->p_uuid[UI_HISTORY_START] = mdev->p_uuid[UI_BITMAP];
2576 mdev->p_uuid[UI_BITMAP] = 0UL;
2577
2578 drbd_uuid_dump(mdev, "peer", mdev->p_uuid, mdev->p_uuid[UI_SIZE], mdev->p_uuid[UI_FLAGS]);
2579 *rule_nr = 35;
2580 } else {
2581 dev_info(DEV, "was SyncTarget (failed to write sync_uuid)\n");
2582 *rule_nr = 37;
2583 }
2584
2585 return -1;
2586 }
2587
2588 /* Common power [off|failure] */
2589 rct = (test_bit(CRASHED_PRIMARY, &mdev->flags) ? 1 : 0) +
2590 (mdev->p_uuid[UI_FLAGS] & 2);
2591 /* lowest bit is set when we were primary,
2592 * next bit (weight 2) is set when peer was primary */
2593 *rule_nr = 40;
2594
2595 switch (rct) {
2596 case 0: /* !self_pri && !peer_pri */ return 0;
2597 case 1: /* self_pri && !peer_pri */ return 1;
2598 case 2: /* !self_pri && peer_pri */ return -1;
2599 case 3: /* self_pri && peer_pri */
Philipp Reisner25703f82011-02-07 14:35:25 +01002600 dc = test_bit(DISCARD_CONCURRENT, &mdev->tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002601 return dc ? -1 : 1;
2602 }
2603 }
2604
2605 *rule_nr = 50;
2606 peer = mdev->p_uuid[UI_BITMAP] & ~((u64)1);
2607 if (self == peer)
2608 return -1;
2609
2610 *rule_nr = 51;
2611 peer = mdev->p_uuid[UI_HISTORY_START] & ~((u64)1);
2612 if (self == peer) {
Philipp Reisner31890f42011-01-19 14:12:51 +01002613 if (mdev->tconn->agreed_pro_version < 96 ?
Philipp Reisner4a23f262011-01-11 17:42:17 +01002614 (mdev->ldev->md.uuid[UI_HISTORY_START] & ~((u64)1)) ==
2615 (mdev->p_uuid[UI_HISTORY_START + 1] & ~((u64)1)) :
2616 peer + UUID_NEW_BM_OFFSET == (mdev->p_uuid[UI_BITMAP] & ~((u64)1))) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002617 /* The last P_SYNC_UUID did not get though. Undo the last start of
2618 resync as sync source modifications of the peer's UUIDs. */
2619
Philipp Reisner31890f42011-01-19 14:12:51 +01002620 if (mdev->tconn->agreed_pro_version < 91)
Philipp Reisner4a23f262011-01-11 17:42:17 +01002621 return -1091;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002622
2623 mdev->p_uuid[UI_BITMAP] = mdev->p_uuid[UI_HISTORY_START];
2624 mdev->p_uuid[UI_HISTORY_START] = mdev->p_uuid[UI_HISTORY_START + 1];
Philipp Reisner4a23f262011-01-11 17:42:17 +01002625
2626 dev_info(DEV, "Did not got last syncUUID packet, corrected:\n");
2627 drbd_uuid_dump(mdev, "peer", mdev->p_uuid, mdev->p_uuid[UI_SIZE], mdev->p_uuid[UI_FLAGS]);
2628
Philipp Reisnerb411b362009-09-25 16:07:19 -07002629 return -1;
2630 }
2631 }
2632
2633 *rule_nr = 60;
2634 self = mdev->ldev->md.uuid[UI_CURRENT] & ~((u64)1);
2635 for (i = UI_HISTORY_START; i <= UI_HISTORY_END; i++) {
2636 peer = mdev->p_uuid[i] & ~((u64)1);
2637 if (self == peer)
2638 return -2;
2639 }
2640
2641 *rule_nr = 70;
2642 self = mdev->ldev->md.uuid[UI_BITMAP] & ~((u64)1);
2643 peer = mdev->p_uuid[UI_CURRENT] & ~((u64)1);
2644 if (self == peer)
2645 return 1;
2646
2647 *rule_nr = 71;
2648 self = mdev->ldev->md.uuid[UI_HISTORY_START] & ~((u64)1);
2649 if (self == peer) {
Philipp Reisner31890f42011-01-19 14:12:51 +01002650 if (mdev->tconn->agreed_pro_version < 96 ?
Philipp Reisner4a23f262011-01-11 17:42:17 +01002651 (mdev->ldev->md.uuid[UI_HISTORY_START + 1] & ~((u64)1)) ==
2652 (mdev->p_uuid[UI_HISTORY_START] & ~((u64)1)) :
2653 self + UUID_NEW_BM_OFFSET == (mdev->ldev->md.uuid[UI_BITMAP] & ~((u64)1))) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002654 /* The last P_SYNC_UUID did not get though. Undo the last start of
2655 resync as sync source modifications of our UUIDs. */
2656
Philipp Reisner31890f42011-01-19 14:12:51 +01002657 if (mdev->tconn->agreed_pro_version < 91)
Philipp Reisner4a23f262011-01-11 17:42:17 +01002658 return -1091;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002659
2660 _drbd_uuid_set(mdev, UI_BITMAP, mdev->ldev->md.uuid[UI_HISTORY_START]);
2661 _drbd_uuid_set(mdev, UI_HISTORY_START, mdev->ldev->md.uuid[UI_HISTORY_START + 1]);
2662
Philipp Reisner4a23f262011-01-11 17:42:17 +01002663 dev_info(DEV, "Last syncUUID did not get through, corrected:\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002664 drbd_uuid_dump(mdev, "self", mdev->ldev->md.uuid,
2665 mdev->state.disk >= D_NEGOTIATING ? drbd_bm_total_weight(mdev) : 0, 0);
2666
2667 return 1;
2668 }
2669 }
2670
2671
2672 *rule_nr = 80;
Philipp Reisnerd8c2a362009-11-18 15:52:51 +01002673 peer = mdev->p_uuid[UI_CURRENT] & ~((u64)1);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002674 for (i = UI_HISTORY_START; i <= UI_HISTORY_END; i++) {
2675 self = mdev->ldev->md.uuid[i] & ~((u64)1);
2676 if (self == peer)
2677 return 2;
2678 }
2679
2680 *rule_nr = 90;
2681 self = mdev->ldev->md.uuid[UI_BITMAP] & ~((u64)1);
2682 peer = mdev->p_uuid[UI_BITMAP] & ~((u64)1);
2683 if (self == peer && self != ((u64)0))
2684 return 100;
2685
2686 *rule_nr = 100;
2687 for (i = UI_HISTORY_START; i <= UI_HISTORY_END; i++) {
2688 self = mdev->ldev->md.uuid[i] & ~((u64)1);
2689 for (j = UI_HISTORY_START; j <= UI_HISTORY_END; j++) {
2690 peer = mdev->p_uuid[j] & ~((u64)1);
2691 if (self == peer)
2692 return -100;
2693 }
2694 }
2695
2696 return -1000;
2697}
2698
2699/* drbd_sync_handshake() returns the new conn state on success, or
2700 CONN_MASK (-1) on failure.
2701 */
2702static enum drbd_conns drbd_sync_handshake(struct drbd_conf *mdev, enum drbd_role peer_role,
2703 enum drbd_disk_state peer_disk) __must_hold(local)
2704{
2705 int hg, rule_nr;
2706 enum drbd_conns rv = C_MASK;
2707 enum drbd_disk_state mydisk;
2708
2709 mydisk = mdev->state.disk;
2710 if (mydisk == D_NEGOTIATING)
2711 mydisk = mdev->new_state_tmp.disk;
2712
2713 dev_info(DEV, "drbd_sync_handshake:\n");
2714 drbd_uuid_dump(mdev, "self", mdev->ldev->md.uuid, mdev->comm_bm_set, 0);
2715 drbd_uuid_dump(mdev, "peer", mdev->p_uuid,
2716 mdev->p_uuid[UI_SIZE], mdev->p_uuid[UI_FLAGS]);
2717
2718 hg = drbd_uuid_compare(mdev, &rule_nr);
2719
2720 dev_info(DEV, "uuid_compare()=%d by rule %d\n", hg, rule_nr);
2721
2722 if (hg == -1000) {
2723 dev_alert(DEV, "Unrelated data, aborting!\n");
2724 return C_MASK;
2725 }
Philipp Reisner4a23f262011-01-11 17:42:17 +01002726 if (hg < -1000) {
2727 dev_alert(DEV, "To resolve this both sides have to support at least protocol %d\n", -hg - 1000);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002728 return C_MASK;
2729 }
2730
2731 if ((mydisk == D_INCONSISTENT && peer_disk > D_INCONSISTENT) ||
2732 (peer_disk == D_INCONSISTENT && mydisk > D_INCONSISTENT)) {
2733 int f = (hg == -100) || abs(hg) == 2;
2734 hg = mydisk > D_INCONSISTENT ? 1 : -1;
2735 if (f)
2736 hg = hg*2;
2737 dev_info(DEV, "Becoming sync %s due to disk states.\n",
2738 hg > 0 ? "source" : "target");
2739 }
2740
Adam Gandelman3a11a482010-04-08 16:48:23 -07002741 if (abs(hg) == 100)
2742 drbd_khelper(mdev, "initial-split-brain");
2743
Philipp Reisner89e58e72011-01-19 13:12:45 +01002744 if (hg == 100 || (hg == -100 && mdev->tconn->net_conf->always_asbp)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002745 int pcount = (mdev->state.role == R_PRIMARY)
2746 + (peer_role == R_PRIMARY);
2747 int forced = (hg == -100);
2748
2749 switch (pcount) {
2750 case 0:
2751 hg = drbd_asb_recover_0p(mdev);
2752 break;
2753 case 1:
2754 hg = drbd_asb_recover_1p(mdev);
2755 break;
2756 case 2:
2757 hg = drbd_asb_recover_2p(mdev);
2758 break;
2759 }
2760 if (abs(hg) < 100) {
2761 dev_warn(DEV, "Split-Brain detected, %d primaries, "
2762 "automatically solved. Sync from %s node\n",
2763 pcount, (hg < 0) ? "peer" : "this");
2764 if (forced) {
2765 dev_warn(DEV, "Doing a full sync, since"
2766 " UUIDs where ambiguous.\n");
2767 hg = hg*2;
2768 }
2769 }
2770 }
2771
2772 if (hg == -100) {
Philipp Reisner89e58e72011-01-19 13:12:45 +01002773 if (mdev->tconn->net_conf->want_lose && !(mdev->p_uuid[UI_FLAGS]&1))
Philipp Reisnerb411b362009-09-25 16:07:19 -07002774 hg = -1;
Philipp Reisner89e58e72011-01-19 13:12:45 +01002775 if (!mdev->tconn->net_conf->want_lose && (mdev->p_uuid[UI_FLAGS]&1))
Philipp Reisnerb411b362009-09-25 16:07:19 -07002776 hg = 1;
2777
2778 if (abs(hg) < 100)
2779 dev_warn(DEV, "Split-Brain detected, manually solved. "
2780 "Sync from %s node\n",
2781 (hg < 0) ? "peer" : "this");
2782 }
2783
2784 if (hg == -100) {
Lars Ellenberg580b9762010-02-26 23:15:23 +01002785 /* FIXME this log message is not correct if we end up here
2786 * after an attempted attach on a diskless node.
2787 * We just refuse to attach -- well, we drop the "connection"
2788 * to that disk, in a way... */
Adam Gandelman3a11a482010-04-08 16:48:23 -07002789 dev_alert(DEV, "Split-Brain detected but unresolved, dropping connection!\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002790 drbd_khelper(mdev, "split-brain");
2791 return C_MASK;
2792 }
2793
2794 if (hg > 0 && mydisk <= D_INCONSISTENT) {
2795 dev_err(DEV, "I shall become SyncSource, but I am inconsistent!\n");
2796 return C_MASK;
2797 }
2798
2799 if (hg < 0 && /* by intention we do not use mydisk here. */
2800 mdev->state.role == R_PRIMARY && mdev->state.disk >= D_CONSISTENT) {
Philipp Reisner89e58e72011-01-19 13:12:45 +01002801 switch (mdev->tconn->net_conf->rr_conflict) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002802 case ASB_CALL_HELPER:
2803 drbd_khelper(mdev, "pri-lost");
2804 /* fall through */
2805 case ASB_DISCONNECT:
2806 dev_err(DEV, "I shall become SyncTarget, but I am primary!\n");
2807 return C_MASK;
2808 case ASB_VIOLENTLY:
2809 dev_warn(DEV, "Becoming SyncTarget, violating the stable-data"
2810 "assumption\n");
2811 }
2812 }
2813
Philipp Reisner8169e412011-03-15 18:40:27 +01002814 if (mdev->tconn->net_conf->dry_run || test_bit(CONN_DRY_RUN, &mdev->tconn->flags)) {
Philipp Reisnercf14c2e2010-02-02 21:03:50 +01002815 if (hg == 0)
2816 dev_info(DEV, "dry-run connect: No resync, would become Connected immediately.\n");
2817 else
2818 dev_info(DEV, "dry-run connect: Would become %s, doing a %s resync.",
2819 drbd_conn_str(hg > 0 ? C_SYNC_SOURCE : C_SYNC_TARGET),
2820 abs(hg) >= 2 ? "full" : "bit-map based");
2821 return C_MASK;
2822 }
2823
Philipp Reisnerb411b362009-09-25 16:07:19 -07002824 if (abs(hg) >= 2) {
2825 dev_info(DEV, "Writing the whole bitmap, full sync required after drbd_sync_handshake.\n");
Lars Ellenberg20ceb2b2011-01-21 10:56:44 +01002826 if (drbd_bitmap_io(mdev, &drbd_bmio_set_n_write, "set_n_write from sync_handshake",
2827 BM_LOCKED_SET_ALLOWED))
Philipp Reisnerb411b362009-09-25 16:07:19 -07002828 return C_MASK;
2829 }
2830
2831 if (hg > 0) { /* become sync source. */
2832 rv = C_WF_BITMAP_S;
2833 } else if (hg < 0) { /* become sync target */
2834 rv = C_WF_BITMAP_T;
2835 } else {
2836 rv = C_CONNECTED;
2837 if (drbd_bm_total_weight(mdev)) {
2838 dev_info(DEV, "No resync, but %lu bits in bitmap!\n",
2839 drbd_bm_total_weight(mdev));
2840 }
2841 }
2842
2843 return rv;
2844}
2845
2846/* returns 1 if invalid */
2847static int cmp_after_sb(enum drbd_after_sb_p peer, enum drbd_after_sb_p self)
2848{
2849 /* ASB_DISCARD_REMOTE - ASB_DISCARD_LOCAL is valid */
2850 if ((peer == ASB_DISCARD_REMOTE && self == ASB_DISCARD_LOCAL) ||
2851 (self == ASB_DISCARD_REMOTE && peer == ASB_DISCARD_LOCAL))
2852 return 0;
2853
2854 /* any other things with ASB_DISCARD_REMOTE or ASB_DISCARD_LOCAL are invalid */
2855 if (peer == ASB_DISCARD_REMOTE || peer == ASB_DISCARD_LOCAL ||
2856 self == ASB_DISCARD_REMOTE || self == ASB_DISCARD_LOCAL)
2857 return 1;
2858
2859 /* everything else is valid if they are equal on both sides. */
2860 if (peer == self)
2861 return 0;
2862
2863 /* everything es is invalid. */
2864 return 1;
2865}
2866
Philipp Reisner7204624c2011-03-15 18:51:47 +01002867static int receive_protocol(struct drbd_tconn *tconn, enum drbd_packet cmd,
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01002868 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002869{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01002870 struct p_protocol *p = tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002871 int p_proto, p_after_sb_0p, p_after_sb_1p, p_after_sb_2p;
Philipp Reisnercf14c2e2010-02-02 21:03:50 +01002872 int p_want_lose, p_two_primaries, cf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002873 char p_integrity_alg[SHARED_SECRET_MAX] = "";
2874
Philipp Reisnerb411b362009-09-25 16:07:19 -07002875 p_proto = be32_to_cpu(p->protocol);
2876 p_after_sb_0p = be32_to_cpu(p->after_sb_0p);
2877 p_after_sb_1p = be32_to_cpu(p->after_sb_1p);
2878 p_after_sb_2p = be32_to_cpu(p->after_sb_2p);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002879 p_two_primaries = be32_to_cpu(p->two_primaries);
Philipp Reisnercf14c2e2010-02-02 21:03:50 +01002880 cf = be32_to_cpu(p->conn_flags);
2881 p_want_lose = cf & CF_WANT_LOSE;
2882
Philipp Reisner7204624c2011-03-15 18:51:47 +01002883 clear_bit(CONN_DRY_RUN, &tconn->flags);
Philipp Reisnercf14c2e2010-02-02 21:03:50 +01002884
2885 if (cf & CF_DRY_RUN)
Philipp Reisner7204624c2011-03-15 18:51:47 +01002886 set_bit(CONN_DRY_RUN, &tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002887
Philipp Reisner7204624c2011-03-15 18:51:47 +01002888 if (p_proto != tconn->net_conf->wire_protocol) {
2889 conn_err(tconn, "incompatible communication protocols\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002890 goto disconnect;
2891 }
2892
Philipp Reisner7204624c2011-03-15 18:51:47 +01002893 if (cmp_after_sb(p_after_sb_0p, tconn->net_conf->after_sb_0p)) {
2894 conn_err(tconn, "incompatible after-sb-0pri settings\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002895 goto disconnect;
2896 }
2897
Philipp Reisner7204624c2011-03-15 18:51:47 +01002898 if (cmp_after_sb(p_after_sb_1p, tconn->net_conf->after_sb_1p)) {
2899 conn_err(tconn, "incompatible after-sb-1pri settings\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002900 goto disconnect;
2901 }
2902
Philipp Reisner7204624c2011-03-15 18:51:47 +01002903 if (cmp_after_sb(p_after_sb_2p, tconn->net_conf->after_sb_2p)) {
2904 conn_err(tconn, "incompatible after-sb-2pri settings\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002905 goto disconnect;
2906 }
2907
Philipp Reisner7204624c2011-03-15 18:51:47 +01002908 if (p_want_lose && tconn->net_conf->want_lose) {
2909 conn_err(tconn, "both sides have the 'want_lose' flag set\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002910 goto disconnect;
2911 }
2912
Philipp Reisner7204624c2011-03-15 18:51:47 +01002913 if (p_two_primaries != tconn->net_conf->two_primaries) {
2914 conn_err(tconn, "incompatible setting of the two-primaries options\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002915 goto disconnect;
2916 }
2917
Philipp Reisner7204624c2011-03-15 18:51:47 +01002918 if (tconn->agreed_pro_version >= 87) {
2919 unsigned char *my_alg = tconn->net_conf->integrity_alg;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002920 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002921
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002922 err = drbd_recv_all(tconn, p_integrity_alg, data_size);
2923 if (err)
2924 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002925
2926 p_integrity_alg[SHARED_SECRET_MAX-1] = 0;
2927 if (strcmp(p_integrity_alg, my_alg)) {
Philipp Reisner7204624c2011-03-15 18:51:47 +01002928 conn_err(tconn, "incompatible setting of the data-integrity-alg\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07002929 goto disconnect;
2930 }
Philipp Reisner7204624c2011-03-15 18:51:47 +01002931 conn_info(tconn, "data-integrity-alg: %s\n",
Philipp Reisnerb411b362009-09-25 16:07:19 -07002932 my_alg[0] ? my_alg : (unsigned char *)"<not-used>");
2933 }
2934
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002935 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002936
2937disconnect:
Philipp Reisner7204624c2011-03-15 18:51:47 +01002938 conn_request_state(tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002939 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002940}
2941
2942/* helper function
2943 * input: alg name, feature name
2944 * return: NULL (alg name was "")
2945 * ERR_PTR(error) if something goes wrong
2946 * or the crypto hash ptr, if it worked out ok. */
2947struct crypto_hash *drbd_crypto_alloc_digest_safe(const struct drbd_conf *mdev,
2948 const char *alg, const char *name)
2949{
2950 struct crypto_hash *tfm;
2951
2952 if (!alg[0])
2953 return NULL;
2954
2955 tfm = crypto_alloc_hash(alg, 0, CRYPTO_ALG_ASYNC);
2956 if (IS_ERR(tfm)) {
2957 dev_err(DEV, "Can not allocate \"%s\" as %s (reason: %ld)\n",
2958 alg, name, PTR_ERR(tfm));
2959 return tfm;
2960 }
2961 if (!drbd_crypto_is_hash(crypto_hash_tfm(tfm))) {
2962 crypto_free_hash(tfm);
2963 dev_err(DEV, "\"%s\" is not a digest (%s)\n", alg, name);
2964 return ERR_PTR(-EINVAL);
2965 }
2966 return tfm;
2967}
2968
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01002969static int receive_SyncParam(struct drbd_conf *mdev, enum drbd_packet cmd,
2970 unsigned int packet_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07002971{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01002972 struct p_rs_param_95 *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002973 unsigned int header_size, data_size, exp_max_sz;
2974 struct crypto_hash *verify_tfm = NULL;
2975 struct crypto_hash *csums_tfm = NULL;
Philipp Reisner31890f42011-01-19 14:12:51 +01002976 const int apv = mdev->tconn->agreed_pro_version;
Philipp Reisner778f2712010-07-06 11:14:00 +02002977 int *rs_plan_s = NULL;
2978 int fifo_size = 0;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002979 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002980
2981 exp_max_sz = apv <= 87 ? sizeof(struct p_rs_param)
2982 : apv == 88 ? sizeof(struct p_rs_param)
2983 + SHARED_SECRET_MAX
Philipp Reisner8e26f9c2010-07-06 17:25:54 +02002984 : apv <= 94 ? sizeof(struct p_rs_param_89)
2985 : /* apv >= 95 */ sizeof(struct p_rs_param_95);
Philipp Reisnerb411b362009-09-25 16:07:19 -07002986
Philipp Reisner02918be2010-08-20 14:35:10 +02002987 if (packet_size > exp_max_sz) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07002988 dev_err(DEV, "SyncParam packet too long: received %u, expected <= %u bytes\n",
Philipp Reisner02918be2010-08-20 14:35:10 +02002989 packet_size, exp_max_sz);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01002990 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002991 }
2992
2993 if (apv <= 88) {
Philipp Reisner257d0af2011-01-26 12:15:29 +01002994 header_size = sizeof(struct p_rs_param) - sizeof(struct p_header);
Philipp Reisner02918be2010-08-20 14:35:10 +02002995 data_size = packet_size - header_size;
Philipp Reisner8e26f9c2010-07-06 17:25:54 +02002996 } else if (apv <= 94) {
Philipp Reisner257d0af2011-01-26 12:15:29 +01002997 header_size = sizeof(struct p_rs_param_89) - sizeof(struct p_header);
Philipp Reisner02918be2010-08-20 14:35:10 +02002998 data_size = packet_size - header_size;
Philipp Reisnerb411b362009-09-25 16:07:19 -07002999 D_ASSERT(data_size == 0);
Philipp Reisner8e26f9c2010-07-06 17:25:54 +02003000 } else {
Philipp Reisner257d0af2011-01-26 12:15:29 +01003001 header_size = sizeof(struct p_rs_param_95) - sizeof(struct p_header);
Philipp Reisner02918be2010-08-20 14:35:10 +02003002 data_size = packet_size - header_size;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003003 D_ASSERT(data_size == 0);
3004 }
3005
3006 /* initialize verify_alg and csums_alg */
3007 memset(p->verify_alg, 0, 2 * SHARED_SECRET_MAX);
3008
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003009 err = drbd_recv_all(mdev->tconn, &p->head.payload, header_size);
3010 if (err)
3011 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003012
Lars Ellenbergf3990022011-03-23 14:31:09 +01003013 if (get_ldev(mdev)) {
3014 mdev->ldev->dc.resync_rate = be32_to_cpu(p->rate);
3015 put_ldev(mdev);
3016 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003017
3018 if (apv >= 88) {
3019 if (apv == 88) {
3020 if (data_size > SHARED_SECRET_MAX) {
3021 dev_err(DEV, "verify-alg too long, "
3022 "peer wants %u, accepting only %u byte\n",
3023 data_size, SHARED_SECRET_MAX);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003024 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003025 }
3026
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003027 err = drbd_recv_all(mdev->tconn, p->verify_alg, data_size);
3028 if (err)
3029 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003030
3031 /* we expect NUL terminated string */
3032 /* but just in case someone tries to be evil */
3033 D_ASSERT(p->verify_alg[data_size-1] == 0);
3034 p->verify_alg[data_size-1] = 0;
3035
3036 } else /* apv >= 89 */ {
3037 /* we still expect NUL terminated strings */
3038 /* but just in case someone tries to be evil */
3039 D_ASSERT(p->verify_alg[SHARED_SECRET_MAX-1] == 0);
3040 D_ASSERT(p->csums_alg[SHARED_SECRET_MAX-1] == 0);
3041 p->verify_alg[SHARED_SECRET_MAX-1] = 0;
3042 p->csums_alg[SHARED_SECRET_MAX-1] = 0;
3043 }
3044
Lars Ellenbergf3990022011-03-23 14:31:09 +01003045 if (strcmp(mdev->tconn->net_conf->verify_alg, p->verify_alg)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07003046 if (mdev->state.conn == C_WF_REPORT_PARAMS) {
3047 dev_err(DEV, "Different verify-alg settings. me=\"%s\" peer=\"%s\"\n",
Lars Ellenbergf3990022011-03-23 14:31:09 +01003048 mdev->tconn->net_conf->verify_alg, p->verify_alg);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003049 goto disconnect;
3050 }
3051 verify_tfm = drbd_crypto_alloc_digest_safe(mdev,
3052 p->verify_alg, "verify-alg");
3053 if (IS_ERR(verify_tfm)) {
3054 verify_tfm = NULL;
3055 goto disconnect;
3056 }
3057 }
3058
Lars Ellenbergf3990022011-03-23 14:31:09 +01003059 if (apv >= 89 && strcmp(mdev->tconn->net_conf->csums_alg, p->csums_alg)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07003060 if (mdev->state.conn == C_WF_REPORT_PARAMS) {
3061 dev_err(DEV, "Different csums-alg settings. me=\"%s\" peer=\"%s\"\n",
Lars Ellenbergf3990022011-03-23 14:31:09 +01003062 mdev->tconn->net_conf->csums_alg, p->csums_alg);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003063 goto disconnect;
3064 }
3065 csums_tfm = drbd_crypto_alloc_digest_safe(mdev,
3066 p->csums_alg, "csums-alg");
3067 if (IS_ERR(csums_tfm)) {
3068 csums_tfm = NULL;
3069 goto disconnect;
3070 }
3071 }
3072
Lars Ellenbergf3990022011-03-23 14:31:09 +01003073 if (apv > 94 && get_ldev(mdev)) {
3074 mdev->ldev->dc.resync_rate = be32_to_cpu(p->rate);
3075 mdev->ldev->dc.c_plan_ahead = be32_to_cpu(p->c_plan_ahead);
3076 mdev->ldev->dc.c_delay_target = be32_to_cpu(p->c_delay_target);
3077 mdev->ldev->dc.c_fill_target = be32_to_cpu(p->c_fill_target);
3078 mdev->ldev->dc.c_max_rate = be32_to_cpu(p->c_max_rate);
Philipp Reisner778f2712010-07-06 11:14:00 +02003079
Lars Ellenbergf3990022011-03-23 14:31:09 +01003080 fifo_size = (mdev->ldev->dc.c_plan_ahead * 10 * SLEEP_TIME) / HZ;
Philipp Reisner778f2712010-07-06 11:14:00 +02003081 if (fifo_size != mdev->rs_plan_s.size && fifo_size > 0) {
3082 rs_plan_s = kzalloc(sizeof(int) * fifo_size, GFP_KERNEL);
3083 if (!rs_plan_s) {
3084 dev_err(DEV, "kmalloc of fifo_buffer failed");
Lars Ellenbergf3990022011-03-23 14:31:09 +01003085 put_ldev(mdev);
Philipp Reisner778f2712010-07-06 11:14:00 +02003086 goto disconnect;
3087 }
3088 }
Lars Ellenbergf3990022011-03-23 14:31:09 +01003089 put_ldev(mdev);
Philipp Reisner8e26f9c2010-07-06 17:25:54 +02003090 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003091
3092 spin_lock(&mdev->peer_seq_lock);
3093 /* lock against drbd_nl_syncer_conf() */
3094 if (verify_tfm) {
Lars Ellenbergf3990022011-03-23 14:31:09 +01003095 strcpy(mdev->tconn->net_conf->verify_alg, p->verify_alg);
3096 mdev->tconn->net_conf->verify_alg_len = strlen(p->verify_alg) + 1;
3097 crypto_free_hash(mdev->tconn->verify_tfm);
3098 mdev->tconn->verify_tfm = verify_tfm;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003099 dev_info(DEV, "using verify-alg: \"%s\"\n", p->verify_alg);
3100 }
3101 if (csums_tfm) {
Lars Ellenbergf3990022011-03-23 14:31:09 +01003102 strcpy(mdev->tconn->net_conf->csums_alg, p->csums_alg);
3103 mdev->tconn->net_conf->csums_alg_len = strlen(p->csums_alg) + 1;
3104 crypto_free_hash(mdev->tconn->csums_tfm);
3105 mdev->tconn->csums_tfm = csums_tfm;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003106 dev_info(DEV, "using csums-alg: \"%s\"\n", p->csums_alg);
3107 }
Philipp Reisner778f2712010-07-06 11:14:00 +02003108 if (fifo_size != mdev->rs_plan_s.size) {
3109 kfree(mdev->rs_plan_s.values);
3110 mdev->rs_plan_s.values = rs_plan_s;
3111 mdev->rs_plan_s.size = fifo_size;
3112 mdev->rs_planed = 0;
3113 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003114 spin_unlock(&mdev->peer_seq_lock);
3115 }
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003116 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003117
Philipp Reisnerb411b362009-09-25 16:07:19 -07003118disconnect:
3119 /* just for completeness: actually not needed,
3120 * as this is not reached if csums_tfm was ok. */
3121 crypto_free_hash(csums_tfm);
3122 /* but free the verify_tfm again, if csums_tfm did not work out */
3123 crypto_free_hash(verify_tfm);
Philipp Reisner38fa9982011-03-15 18:24:49 +01003124 conn_request_state(mdev->tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003125 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003126}
3127
Philipp Reisnerb411b362009-09-25 16:07:19 -07003128/* warn if the arguments differ by more than 12.5% */
3129static void warn_if_differ_considerably(struct drbd_conf *mdev,
3130 const char *s, sector_t a, sector_t b)
3131{
3132 sector_t d;
3133 if (a == 0 || b == 0)
3134 return;
3135 d = (a > b) ? (a - b) : (b - a);
3136 if (d > (a>>3) || d > (b>>3))
3137 dev_warn(DEV, "Considerable difference in %s: %llus vs. %llus\n", s,
3138 (unsigned long long)a, (unsigned long long)b);
3139}
3140
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003141static int receive_sizes(struct drbd_conf *mdev, enum drbd_packet cmd,
3142 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003143{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003144 struct p_sizes *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003145 enum determine_dev_size dd = unchanged;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003146 sector_t p_size, p_usize, my_usize;
3147 int ldsc = 0; /* local disk size changed */
Philipp Reisnere89b5912010-03-24 17:11:33 +01003148 enum dds_flags ddsf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003149
Philipp Reisnerb411b362009-09-25 16:07:19 -07003150 p_size = be64_to_cpu(p->d_size);
3151 p_usize = be64_to_cpu(p->u_size);
3152
Philipp Reisnerb411b362009-09-25 16:07:19 -07003153 /* just store the peer's disk size for now.
3154 * we still need to figure out whether we accept that. */
3155 mdev->p_size = p_size;
3156
Philipp Reisnerb411b362009-09-25 16:07:19 -07003157 if (get_ldev(mdev)) {
3158 warn_if_differ_considerably(mdev, "lower level device sizes",
3159 p_size, drbd_get_max_capacity(mdev->ldev));
3160 warn_if_differ_considerably(mdev, "user requested size",
3161 p_usize, mdev->ldev->dc.disk_size);
3162
3163 /* if this is the first connect, or an otherwise expected
3164 * param exchange, choose the minimum */
3165 if (mdev->state.conn == C_WF_REPORT_PARAMS)
3166 p_usize = min_not_zero((sector_t)mdev->ldev->dc.disk_size,
3167 p_usize);
3168
3169 my_usize = mdev->ldev->dc.disk_size;
3170
3171 if (mdev->ldev->dc.disk_size != p_usize) {
3172 mdev->ldev->dc.disk_size = p_usize;
3173 dev_info(DEV, "Peer sets u_size to %lu sectors\n",
3174 (unsigned long)mdev->ldev->dc.disk_size);
3175 }
3176
3177 /* Never shrink a device with usable data during connect.
3178 But allow online shrinking if we are connected. */
Philipp Reisnera393db62009-12-22 13:35:52 +01003179 if (drbd_new_dev_size(mdev, mdev->ldev, 0) <
Philipp Reisnerb411b362009-09-25 16:07:19 -07003180 drbd_get_capacity(mdev->this_bdev) &&
3181 mdev->state.disk >= D_OUTDATED &&
3182 mdev->state.conn < C_CONNECTED) {
3183 dev_err(DEV, "The peer's disk size is too small!\n");
Philipp Reisner38fa9982011-03-15 18:24:49 +01003184 conn_request_state(mdev->tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003185 mdev->ldev->dc.disk_size = my_usize;
3186 put_ldev(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003187 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003188 }
3189 put_ldev(mdev);
3190 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003191
Philipp Reisnere89b5912010-03-24 17:11:33 +01003192 ddsf = be16_to_cpu(p->dds_flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003193 if (get_ldev(mdev)) {
Bart Van Assche24c48302011-05-21 18:32:29 +02003194 dd = drbd_determine_dev_size(mdev, ddsf);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003195 put_ldev(mdev);
3196 if (dd == dev_size_error)
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003197 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003198 drbd_md_sync(mdev);
3199 } else {
3200 /* I am diskless, need to accept the peer's size. */
3201 drbd_set_my_capacity(mdev, p_size);
3202 }
3203
Philipp Reisner99432fc2011-05-20 16:39:13 +02003204 mdev->peer_max_bio_size = be32_to_cpu(p->max_bio_size);
3205 drbd_reconsider_max_bio_size(mdev);
3206
Philipp Reisnerb411b362009-09-25 16:07:19 -07003207 if (get_ldev(mdev)) {
3208 if (mdev->ldev->known_size != drbd_get_capacity(mdev->ldev->backing_bdev)) {
3209 mdev->ldev->known_size = drbd_get_capacity(mdev->ldev->backing_bdev);
3210 ldsc = 1;
3211 }
3212
Philipp Reisnerb411b362009-09-25 16:07:19 -07003213 put_ldev(mdev);
3214 }
3215
3216 if (mdev->state.conn > C_WF_REPORT_PARAMS) {
3217 if (be64_to_cpu(p->c_size) !=
3218 drbd_get_capacity(mdev->this_bdev) || ldsc) {
3219 /* we have different sizes, probably peer
3220 * needs to know my new size... */
Philipp Reisnere89b5912010-03-24 17:11:33 +01003221 drbd_send_sizes(mdev, 0, ddsf);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003222 }
3223 if (test_and_clear_bit(RESIZE_PENDING, &mdev->flags) ||
3224 (dd == grew && mdev->state.conn == C_CONNECTED)) {
3225 if (mdev->state.pdsk >= D_INCONSISTENT &&
Philipp Reisnere89b5912010-03-24 17:11:33 +01003226 mdev->state.disk >= D_INCONSISTENT) {
3227 if (ddsf & DDSF_NO_RESYNC)
3228 dev_info(DEV, "Resync of new storage suppressed with --assume-clean\n");
3229 else
3230 resync_after_online_grow(mdev);
3231 } else
Philipp Reisnerb411b362009-09-25 16:07:19 -07003232 set_bit(RESYNC_AFTER_NEG, &mdev->flags);
3233 }
3234 }
3235
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003236 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003237}
3238
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003239static int receive_uuids(struct drbd_conf *mdev, enum drbd_packet cmd,
3240 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003241{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003242 struct p_uuids *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003243 u64 *p_uuid;
Lars Ellenberg62b0da32011-01-20 13:25:21 +01003244 int i, updated_uuids = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003245
Philipp Reisnerb411b362009-09-25 16:07:19 -07003246 p_uuid = kmalloc(sizeof(u64)*UI_EXTENDED_SIZE, GFP_NOIO);
3247
3248 for (i = UI_CURRENT; i < UI_EXTENDED_SIZE; i++)
3249 p_uuid[i] = be64_to_cpu(p->uuid[i]);
3250
3251 kfree(mdev->p_uuid);
3252 mdev->p_uuid = p_uuid;
3253
3254 if (mdev->state.conn < C_CONNECTED &&
3255 mdev->state.disk < D_INCONSISTENT &&
3256 mdev->state.role == R_PRIMARY &&
3257 (mdev->ed_uuid & ~((u64)1)) != (p_uuid[UI_CURRENT] & ~((u64)1))) {
3258 dev_err(DEV, "Can only connect to data with current UUID=%016llX\n",
3259 (unsigned long long)mdev->ed_uuid);
Philipp Reisner38fa9982011-03-15 18:24:49 +01003260 conn_request_state(mdev->tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003261 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003262 }
3263
3264 if (get_ldev(mdev)) {
3265 int skip_initial_sync =
3266 mdev->state.conn == C_CONNECTED &&
Philipp Reisner31890f42011-01-19 14:12:51 +01003267 mdev->tconn->agreed_pro_version >= 90 &&
Philipp Reisnerb411b362009-09-25 16:07:19 -07003268 mdev->ldev->md.uuid[UI_CURRENT] == UUID_JUST_CREATED &&
3269 (p_uuid[UI_FLAGS] & 8);
3270 if (skip_initial_sync) {
3271 dev_info(DEV, "Accepted new current UUID, preparing to skip initial sync\n");
3272 drbd_bitmap_io(mdev, &drbd_bmio_clear_n_write,
Lars Ellenberg20ceb2b2011-01-21 10:56:44 +01003273 "clear_n_write from receive_uuids",
3274 BM_LOCKED_TEST_ALLOWED);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003275 _drbd_uuid_set(mdev, UI_CURRENT, p_uuid[UI_CURRENT]);
3276 _drbd_uuid_set(mdev, UI_BITMAP, 0);
3277 _drbd_set_state(_NS2(mdev, disk, D_UP_TO_DATE, pdsk, D_UP_TO_DATE),
3278 CS_VERBOSE, NULL);
3279 drbd_md_sync(mdev);
Lars Ellenberg62b0da32011-01-20 13:25:21 +01003280 updated_uuids = 1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003281 }
3282 put_ldev(mdev);
Philipp Reisner18a50fa2010-06-21 14:14:15 +02003283 } else if (mdev->state.disk < D_INCONSISTENT &&
3284 mdev->state.role == R_PRIMARY) {
3285 /* I am a diskless primary, the peer just created a new current UUID
3286 for me. */
Lars Ellenberg62b0da32011-01-20 13:25:21 +01003287 updated_uuids = drbd_set_ed_uuid(mdev, p_uuid[UI_CURRENT]);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003288 }
3289
3290 /* Before we test for the disk state, we should wait until an eventually
3291 ongoing cluster wide state change is finished. That is important if
3292 we are primary and are detaching from our disk. We need to see the
3293 new disk state... */
Philipp Reisner8410da8f02011-02-11 20:11:10 +01003294 mutex_lock(mdev->state_mutex);
3295 mutex_unlock(mdev->state_mutex);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003296 if (mdev->state.conn >= C_CONNECTED && mdev->state.disk < D_INCONSISTENT)
Lars Ellenberg62b0da32011-01-20 13:25:21 +01003297 updated_uuids |= drbd_set_ed_uuid(mdev, p_uuid[UI_CURRENT]);
3298
3299 if (updated_uuids)
3300 drbd_print_uuids(mdev, "receiver updated UUIDs to");
Philipp Reisnerb411b362009-09-25 16:07:19 -07003301
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003302 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003303}
3304
3305/**
3306 * convert_state() - Converts the peer's view of the cluster state to our point of view
3307 * @ps: The state as seen by the peer.
3308 */
3309static union drbd_state convert_state(union drbd_state ps)
3310{
3311 union drbd_state ms;
3312
3313 static enum drbd_conns c_tab[] = {
3314 [C_CONNECTED] = C_CONNECTED,
3315
3316 [C_STARTING_SYNC_S] = C_STARTING_SYNC_T,
3317 [C_STARTING_SYNC_T] = C_STARTING_SYNC_S,
3318 [C_DISCONNECTING] = C_TEAR_DOWN, /* C_NETWORK_FAILURE, */
3319 [C_VERIFY_S] = C_VERIFY_T,
3320 [C_MASK] = C_MASK,
3321 };
3322
3323 ms.i = ps.i;
3324
3325 ms.conn = c_tab[ps.conn];
3326 ms.peer = ps.role;
3327 ms.role = ps.peer;
3328 ms.pdsk = ps.disk;
3329 ms.disk = ps.pdsk;
3330 ms.peer_isp = (ps.aftr_isp | ps.user_isp);
3331
3332 return ms;
3333}
3334
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003335static int receive_req_state(struct drbd_conf *mdev, enum drbd_packet cmd,
3336 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003337{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003338 struct p_req_state *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003339 union drbd_state mask, val;
Andreas Gruenbacherbf885f82010-12-08 00:39:32 +01003340 enum drbd_state_rv rv;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003341
Philipp Reisnerb411b362009-09-25 16:07:19 -07003342 mask.i = be32_to_cpu(p->mask);
3343 val.i = be32_to_cpu(p->val);
3344
Philipp Reisner25703f82011-02-07 14:35:25 +01003345 if (test_bit(DISCARD_CONCURRENT, &mdev->tconn->flags) &&
Philipp Reisner8410da8f02011-02-11 20:11:10 +01003346 mutex_is_locked(mdev->state_mutex)) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07003347 drbd_send_sr_reply(mdev, SS_CONCURRENT_ST_CHG);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003348 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003349 }
3350
3351 mask = convert_state(mask);
3352 val = convert_state(val);
3353
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003354 rv = drbd_change_state(mdev, CS_VERBOSE, mask, val);
3355 drbd_send_sr_reply(mdev, rv);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003356
Philipp Reisnerb411b362009-09-25 16:07:19 -07003357 drbd_md_sync(mdev);
3358
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003359 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003360}
3361
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003362static int receive_req_conn_state(struct drbd_tconn *tconn, enum drbd_packet cmd,
3363 unsigned int data_size)
3364{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003365 struct p_req_state *p = tconn->data.rbuf;
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003366 union drbd_state mask, val;
3367 enum drbd_state_rv rv;
3368
3369 mask.i = be32_to_cpu(p->mask);
3370 val.i = be32_to_cpu(p->val);
3371
3372 if (test_bit(DISCARD_CONCURRENT, &tconn->flags) &&
3373 mutex_is_locked(&tconn->cstate_mutex)) {
3374 conn_send_sr_reply(tconn, SS_CONCURRENT_ST_CHG);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003375 return 0;
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003376 }
3377
3378 mask = convert_state(mask);
3379 val = convert_state(val);
3380
3381 rv = conn_request_state(tconn, mask, val, CS_VERBOSE | CS_LOCAL_ONLY);
3382 conn_send_sr_reply(tconn, rv);
3383
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003384 return 0;
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003385}
3386
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003387static int receive_state(struct drbd_conf *mdev, enum drbd_packet cmd,
3388 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003389{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003390 struct p_state *p = mdev->tconn->data.rbuf;
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003391 union drbd_state os, ns, peer_state;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003392 enum drbd_disk_state real_peer_disk;
Philipp Reisner65d922c2010-06-16 16:18:09 +02003393 enum chg_state_flags cs_flags;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003394 int rv;
3395
Philipp Reisnerb411b362009-09-25 16:07:19 -07003396 peer_state.i = be32_to_cpu(p->state);
3397
3398 real_peer_disk = peer_state.disk;
3399 if (peer_state.disk == D_NEGOTIATING) {
3400 real_peer_disk = mdev->p_uuid[UI_FLAGS] & 4 ? D_INCONSISTENT : D_CONSISTENT;
3401 dev_info(DEV, "real peer disk state = %s\n", drbd_disk_str(real_peer_disk));
3402 }
3403
Philipp Reisner87eeee42011-01-19 14:16:30 +01003404 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003405 retry:
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003406 os = ns = mdev->state;
Philipp Reisner87eeee42011-01-19 14:16:30 +01003407 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003408
Lars Ellenberge9ef7bb2010-10-07 15:55:39 +02003409 /* peer says his disk is uptodate, while we think it is inconsistent,
3410 * and this happens while we think we have a sync going on. */
3411 if (os.pdsk == D_INCONSISTENT && real_peer_disk == D_UP_TO_DATE &&
3412 os.conn > C_CONNECTED && os.disk == D_UP_TO_DATE) {
3413 /* If we are (becoming) SyncSource, but peer is still in sync
3414 * preparation, ignore its uptodate-ness to avoid flapping, it
3415 * will change to inconsistent once the peer reaches active
3416 * syncing states.
3417 * It may have changed syncer-paused flags, however, so we
3418 * cannot ignore this completely. */
3419 if (peer_state.conn > C_CONNECTED &&
3420 peer_state.conn < C_SYNC_SOURCE)
3421 real_peer_disk = D_INCONSISTENT;
3422
3423 /* if peer_state changes to connected at the same time,
3424 * it explicitly notifies us that it finished resync.
3425 * Maybe we should finish it up, too? */
3426 else if (os.conn >= C_SYNC_SOURCE &&
3427 peer_state.conn == C_CONNECTED) {
3428 if (drbd_bm_total_weight(mdev) <= mdev->rs_failed)
3429 drbd_resync_finished(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003430 return 0;
Lars Ellenberge9ef7bb2010-10-07 15:55:39 +02003431 }
3432 }
3433
3434 /* peer says his disk is inconsistent, while we think it is uptodate,
3435 * and this happens while the peer still thinks we have a sync going on,
3436 * but we think we are already done with the sync.
3437 * We ignore this to avoid flapping pdsk.
3438 * This should not happen, if the peer is a recent version of drbd. */
3439 if (os.pdsk == D_UP_TO_DATE && real_peer_disk == D_INCONSISTENT &&
3440 os.conn == C_CONNECTED && peer_state.conn > C_SYNC_SOURCE)
3441 real_peer_disk = D_UP_TO_DATE;
3442
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003443 if (ns.conn == C_WF_REPORT_PARAMS)
3444 ns.conn = C_CONNECTED;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003445
Philipp Reisner67531712010-10-27 12:21:30 +02003446 if (peer_state.conn == C_AHEAD)
3447 ns.conn = C_BEHIND;
3448
Philipp Reisnerb411b362009-09-25 16:07:19 -07003449 if (mdev->p_uuid && peer_state.disk >= D_NEGOTIATING &&
3450 get_ldev_if_state(mdev, D_NEGOTIATING)) {
3451 int cr; /* consider resync */
3452
3453 /* if we established a new connection */
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003454 cr = (os.conn < C_CONNECTED);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003455 /* if we had an established connection
3456 * and one of the nodes newly attaches a disk */
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003457 cr |= (os.conn == C_CONNECTED &&
Philipp Reisnerb411b362009-09-25 16:07:19 -07003458 (peer_state.disk == D_NEGOTIATING ||
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003459 os.disk == D_NEGOTIATING));
Philipp Reisnerb411b362009-09-25 16:07:19 -07003460 /* if we have both been inconsistent, and the peer has been
3461 * forced to be UpToDate with --overwrite-data */
3462 cr |= test_bit(CONSIDER_RESYNC, &mdev->flags);
3463 /* if we had been plain connected, and the admin requested to
3464 * start a sync by "invalidate" or "invalidate-remote" */
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003465 cr |= (os.conn == C_CONNECTED &&
Philipp Reisnerb411b362009-09-25 16:07:19 -07003466 (peer_state.conn >= C_STARTING_SYNC_S &&
3467 peer_state.conn <= C_WF_BITMAP_T));
3468
3469 if (cr)
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003470 ns.conn = drbd_sync_handshake(mdev, peer_state.role, real_peer_disk);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003471
3472 put_ldev(mdev);
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003473 if (ns.conn == C_MASK) {
3474 ns.conn = C_CONNECTED;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003475 if (mdev->state.disk == D_NEGOTIATING) {
Lars Ellenberg82f59cc2010-10-16 12:13:47 +02003476 drbd_force_state(mdev, NS(disk, D_FAILED));
Philipp Reisnerb411b362009-09-25 16:07:19 -07003477 } else if (peer_state.disk == D_NEGOTIATING) {
3478 dev_err(DEV, "Disk attach process on the peer node was aborted.\n");
3479 peer_state.disk = D_DISKLESS;
Lars Ellenberg580b9762010-02-26 23:15:23 +01003480 real_peer_disk = D_DISKLESS;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003481 } else {
Philipp Reisner8169e412011-03-15 18:40:27 +01003482 if (test_and_clear_bit(CONN_DRY_RUN, &mdev->tconn->flags))
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003483 return -EIO;
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003484 D_ASSERT(os.conn == C_WF_REPORT_PARAMS);
Philipp Reisner38fa9982011-03-15 18:24:49 +01003485 conn_request_state(mdev->tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003486 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003487 }
3488 }
3489 }
3490
Philipp Reisner87eeee42011-01-19 14:16:30 +01003491 spin_lock_irq(&mdev->tconn->req_lock);
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003492 if (mdev->state.i != os.i)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003493 goto retry;
3494 clear_bit(CONSIDER_RESYNC, &mdev->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003495 ns.peer = peer_state.role;
3496 ns.pdsk = real_peer_disk;
3497 ns.peer_isp = (peer_state.aftr_isp | peer_state.user_isp);
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003498 if ((ns.conn == C_CONNECTED || ns.conn == C_WF_BITMAP_S) && ns.disk == D_NEGOTIATING)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003499 ns.disk = mdev->new_state_tmp.disk;
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003500 cs_flags = CS_VERBOSE + (os.conn < C_CONNECTED && ns.conn >= C_CONNECTED ? 0 : CS_HARD);
3501 if (ns.pdsk == D_CONSISTENT && is_susp(ns) && ns.conn == C_CONNECTED && os.conn < C_CONNECTED &&
Philipp Reisner481c6f52010-06-22 14:03:27 +02003502 test_bit(NEW_CUR_UUID, &mdev->flags)) {
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01003503 /* Do not allow tl_restart(RESEND) for a rebooted peer. We can only allow this
Philipp Reisner481c6f52010-06-22 14:03:27 +02003504 for temporal network outages! */
Philipp Reisner87eeee42011-01-19 14:16:30 +01003505 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisner481c6f52010-06-22 14:03:27 +02003506 dev_err(DEV, "Aborting Connect, can not thaw IO with an only Consistent peer\n");
Philipp Reisner2f5cdd02011-02-21 14:29:27 +01003507 tl_clear(mdev->tconn);
Philipp Reisner481c6f52010-06-22 14:03:27 +02003508 drbd_uuid_new_current(mdev);
3509 clear_bit(NEW_CUR_UUID, &mdev->flags);
Philipp Reisner38fa9982011-03-15 18:24:49 +01003510 conn_request_state(mdev->tconn, NS2(conn, C_PROTOCOL_ERROR, susp, 0), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003511 return -EIO;
Philipp Reisner481c6f52010-06-22 14:03:27 +02003512 }
Philipp Reisner65d922c2010-06-16 16:18:09 +02003513 rv = _drbd_set_state(mdev, ns, cs_flags, NULL);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003514 ns = mdev->state;
Philipp Reisner87eeee42011-01-19 14:16:30 +01003515 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003516
3517 if (rv < SS_SUCCESS) {
Philipp Reisner38fa9982011-03-15 18:24:49 +01003518 conn_request_state(mdev->tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003519 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003520 }
3521
Lars Ellenberg4ac4aad2010-07-22 17:39:26 +02003522 if (os.conn > C_WF_REPORT_PARAMS) {
3523 if (ns.conn > C_CONNECTED && peer_state.conn <= C_CONNECTED &&
Philipp Reisnerb411b362009-09-25 16:07:19 -07003524 peer_state.disk != D_NEGOTIATING ) {
3525 /* we want resync, peer has not yet decided to sync... */
3526 /* Nowadays only used when forcing a node into primary role and
3527 setting its disk to UpToDate with that */
3528 drbd_send_uuids(mdev);
3529 drbd_send_state(mdev);
3530 }
3531 }
3532
Philipp Reisner89e58e72011-01-19 13:12:45 +01003533 mdev->tconn->net_conf->want_lose = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003534
3535 drbd_md_sync(mdev); /* update connected indicator, la_size, ... */
3536
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003537 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003538}
3539
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003540static int receive_sync_uuid(struct drbd_conf *mdev, enum drbd_packet cmd,
3541 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003542{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003543 struct p_rs_uuid *p = mdev->tconn->data.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003544
3545 wait_event(mdev->misc_wait,
3546 mdev->state.conn == C_WF_SYNC_UUID ||
Philipp Reisnerc4752ef2010-10-27 17:32:36 +02003547 mdev->state.conn == C_BEHIND ||
Philipp Reisnerb411b362009-09-25 16:07:19 -07003548 mdev->state.conn < C_CONNECTED ||
3549 mdev->state.disk < D_NEGOTIATING);
3550
3551 /* D_ASSERT( mdev->state.conn == C_WF_SYNC_UUID ); */
3552
Philipp Reisnerb411b362009-09-25 16:07:19 -07003553 /* Here the _drbd_uuid_ functions are right, current should
3554 _not_ be rotated into the history */
3555 if (get_ldev_if_state(mdev, D_NEGOTIATING)) {
3556 _drbd_uuid_set(mdev, UI_CURRENT, be64_to_cpu(p->uuid));
3557 _drbd_uuid_set(mdev, UI_BITMAP, 0UL);
3558
Lars Ellenberg62b0da32011-01-20 13:25:21 +01003559 drbd_print_uuids(mdev, "updated sync uuid");
Philipp Reisnerb411b362009-09-25 16:07:19 -07003560 drbd_start_resync(mdev, C_SYNC_TARGET);
3561
3562 put_ldev(mdev);
3563 } else
3564 dev_err(DEV, "Ignoring SyncUUID packet!\n");
3565
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003566 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003567}
3568
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003569/**
3570 * receive_bitmap_plain
3571 *
3572 * Return 0 when done, 1 when another iteration is needed, and a negative error
3573 * code upon failure.
3574 */
3575static int
Philipp Reisner02918be2010-08-20 14:35:10 +02003576receive_bitmap_plain(struct drbd_conf *mdev, unsigned int data_size,
Andreas Gruenbacherfc568152011-03-24 21:23:50 +01003577 struct p_header *h, struct bm_xfer_ctx *c)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003578{
Andreas Gruenbacherfc568152011-03-24 21:23:50 +01003579 unsigned long *buffer = (unsigned long *)h->payload;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003580 unsigned num_words = min_t(size_t, BM_PACKET_WORDS, c->bm_words - c->word_offset);
3581 unsigned want = num_words * sizeof(long);
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003582 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003583
Philipp Reisner02918be2010-08-20 14:35:10 +02003584 if (want != data_size) {
3585 dev_err(DEV, "%s:want (%u) != data_size (%u)\n", __func__, want, data_size);
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003586 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003587 }
3588 if (want == 0)
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003589 return 0;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003590 err = drbd_recv_all(mdev->tconn, buffer, want);
3591 if (err)
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003592 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003593
3594 drbd_bm_merge_lel(mdev, c->word_offset, num_words, buffer);
3595
3596 c->word_offset += num_words;
3597 c->bit_offset = c->word_offset * BITS_PER_LONG;
3598 if (c->bit_offset > c->bm_bits)
3599 c->bit_offset = c->bm_bits;
3600
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003601 return 1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003602}
3603
Andreas Gruenbachera02d1242011-03-22 17:20:45 +01003604static enum drbd_bitmap_code dcbp_get_code(struct p_compressed_bm *p)
3605{
3606 return (enum drbd_bitmap_code)(p->encoding & 0x0f);
3607}
3608
3609static int dcbp_get_start(struct p_compressed_bm *p)
3610{
3611 return (p->encoding & 0x80) != 0;
3612}
3613
3614static int dcbp_get_pad_bits(struct p_compressed_bm *p)
3615{
3616 return (p->encoding >> 4) & 0x7;
3617}
3618
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003619/**
3620 * recv_bm_rle_bits
3621 *
3622 * Return 0 when done, 1 when another iteration is needed, and a negative error
3623 * code upon failure.
3624 */
3625static int
Philipp Reisnerb411b362009-09-25 16:07:19 -07003626recv_bm_rle_bits(struct drbd_conf *mdev,
3627 struct p_compressed_bm *p,
Philipp Reisnerc6d25cf2011-01-19 16:13:06 +01003628 struct bm_xfer_ctx *c,
3629 unsigned int len)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003630{
3631 struct bitstream bs;
3632 u64 look_ahead;
3633 u64 rl;
3634 u64 tmp;
3635 unsigned long s = c->bit_offset;
3636 unsigned long e;
Andreas Gruenbachera02d1242011-03-22 17:20:45 +01003637 int toggle = dcbp_get_start(p);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003638 int have;
3639 int bits;
3640
Andreas Gruenbachera02d1242011-03-22 17:20:45 +01003641 bitstream_init(&bs, p->code, len, dcbp_get_pad_bits(p));
Philipp Reisnerb411b362009-09-25 16:07:19 -07003642
3643 bits = bitstream_get_bits(&bs, &look_ahead, 64);
3644 if (bits < 0)
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003645 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003646
3647 for (have = bits; have > 0; s += rl, toggle = !toggle) {
3648 bits = vli_decode_bits(&rl, look_ahead);
3649 if (bits <= 0)
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003650 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003651
3652 if (toggle) {
3653 e = s + rl -1;
3654 if (e >= c->bm_bits) {
3655 dev_err(DEV, "bitmap overflow (e:%lu) while decoding bm RLE packet\n", e);
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003656 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003657 }
3658 _drbd_bm_set_bits(mdev, s, e);
3659 }
3660
3661 if (have < bits) {
3662 dev_err(DEV, "bitmap decoding error: h:%d b:%d la:0x%08llx l:%u/%u\n",
3663 have, bits, look_ahead,
3664 (unsigned int)(bs.cur.b - p->code),
3665 (unsigned int)bs.buf_len);
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003666 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003667 }
3668 look_ahead >>= bits;
3669 have -= bits;
3670
3671 bits = bitstream_get_bits(&bs, &tmp, 64 - have);
3672 if (bits < 0)
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003673 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003674 look_ahead |= tmp << have;
3675 have += bits;
3676 }
3677
3678 c->bit_offset = s;
3679 bm_xfer_ctx_bit_to_word_offset(c);
3680
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003681 return (s != c->bm_bits);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003682}
3683
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003684/**
3685 * decode_bitmap_c
3686 *
3687 * Return 0 when done, 1 when another iteration is needed, and a negative error
3688 * code upon failure.
3689 */
3690static int
Philipp Reisnerb411b362009-09-25 16:07:19 -07003691decode_bitmap_c(struct drbd_conf *mdev,
3692 struct p_compressed_bm *p,
Philipp Reisnerc6d25cf2011-01-19 16:13:06 +01003693 struct bm_xfer_ctx *c,
3694 unsigned int len)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003695{
Andreas Gruenbachera02d1242011-03-22 17:20:45 +01003696 if (dcbp_get_code(p) == RLE_VLI_Bits)
Philipp Reisnerc6d25cf2011-01-19 16:13:06 +01003697 return recv_bm_rle_bits(mdev, p, c, len);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003698
3699 /* other variants had been implemented for evaluation,
3700 * but have been dropped as this one turned out to be "best"
3701 * during all our tests. */
3702
3703 dev_err(DEV, "receive_bitmap_c: unknown encoding %u\n", p->encoding);
Philipp Reisner38fa9982011-03-15 18:24:49 +01003704 conn_request_state(mdev->tconn, NS(conn, C_PROTOCOL_ERROR), CS_HARD);
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003705 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003706}
3707
3708void INFO_bm_xfer_stats(struct drbd_conf *mdev,
3709 const char *direction, struct bm_xfer_ctx *c)
3710{
3711 /* what would it take to transfer it "plaintext" */
Philipp Reisnerc0129492011-01-19 16:58:16 +01003712 unsigned plain = sizeof(struct p_header) *
Philipp Reisnerb411b362009-09-25 16:07:19 -07003713 ((c->bm_words+BM_PACKET_WORDS-1)/BM_PACKET_WORDS+1)
3714 + c->bm_words * sizeof(long);
3715 unsigned total = c->bytes[0] + c->bytes[1];
3716 unsigned r;
3717
3718 /* total can not be zero. but just in case: */
3719 if (total == 0)
3720 return;
3721
3722 /* don't report if not compressed */
3723 if (total >= plain)
3724 return;
3725
3726 /* total < plain. check for overflow, still */
3727 r = (total > UINT_MAX/1000) ? (total / (plain/1000))
3728 : (1000 * total / plain);
3729
3730 if (r > 1000)
3731 r = 1000;
3732
3733 r = 1000 - r;
3734 dev_info(DEV, "%s bitmap stats [Bytes(packets)]: plain %u(%u), RLE %u(%u), "
3735 "total %u; compression: %u.%u%%\n",
3736 direction,
3737 c->bytes[1], c->packets[1],
3738 c->bytes[0], c->packets[0],
3739 total, r/10, r % 10);
3740}
3741
3742/* Since we are processing the bitfield from lower addresses to higher,
3743 it does not matter if the process it in 32 bit chunks or 64 bit
3744 chunks as long as it is little endian. (Understand it as byte stream,
3745 beginning with the lowest byte...) If we would use big endian
3746 we would need to process it from the highest address to the lowest,
3747 in order to be agnostic to the 32 vs 64 bits issue.
3748
3749 returns 0 on failure, 1 if we successfully received it. */
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003750static int receive_bitmap(struct drbd_conf *mdev, enum drbd_packet cmd,
3751 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003752{
3753 struct bm_xfer_ctx c;
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003754 int err;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003755 struct p_header *h = mdev->tconn->data.rbuf;
Philipp Reisner77351055b2011-02-07 17:24:26 +01003756 struct packet_info pi;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003757
Lars Ellenberg20ceb2b2011-01-21 10:56:44 +01003758 drbd_bm_lock(mdev, "receive bitmap", BM_LOCKED_SET_ALLOWED);
3759 /* you are supposed to send additional out-of-sync information
3760 * if you actually set bits during this phase */
Philipp Reisnerb411b362009-09-25 16:07:19 -07003761
Philipp Reisnerb411b362009-09-25 16:07:19 -07003762 c = (struct bm_xfer_ctx) {
3763 .bm_bits = drbd_bm_bits(mdev),
3764 .bm_words = drbd_bm_words(mdev),
3765 };
3766
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003767 for(;;) {
Philipp Reisner02918be2010-08-20 14:35:10 +02003768 if (cmd == P_BITMAP) {
Andreas Gruenbacherfc568152011-03-24 21:23:50 +01003769 err = receive_bitmap_plain(mdev, data_size, h, &c);
Philipp Reisner02918be2010-08-20 14:35:10 +02003770 } else if (cmd == P_COMPRESSED_BITMAP) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07003771 /* MAYBE: sanity check that we speak proto >= 90,
3772 * and the feature is enabled! */
3773 struct p_compressed_bm *p;
3774
Philipp Reisner02918be2010-08-20 14:35:10 +02003775 if (data_size > BM_PACKET_PAYLOAD_BYTES) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07003776 dev_err(DEV, "ReportCBitmap packet too large\n");
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003777 err = -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003778 goto out;
3779 }
Andreas Gruenbacherfc568152011-03-24 21:23:50 +01003780
3781 p = mdev->tconn->data.rbuf;
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003782 err = drbd_recv_all(mdev->tconn, p->head.payload, data_size);
3783 if (err)
3784 goto out;
Lars Ellenberg004352f2010-10-05 20:13:58 +02003785 if (data_size <= (sizeof(*p) - sizeof(p->head))) {
3786 dev_err(DEV, "ReportCBitmap packet too small (l:%u)\n", data_size);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003787 err = -EIO;
Andreas Gruenbacher78fcbda2010-12-10 22:18:27 +01003788 goto out;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003789 }
Philipp Reisnerc6d25cf2011-01-19 16:13:06 +01003790 err = decode_bitmap_c(mdev, p, &c, data_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003791 } else {
Philipp Reisner02918be2010-08-20 14:35:10 +02003792 dev_warn(DEV, "receive_bitmap: cmd neither ReportBitMap nor ReportCBitMap (is 0x%x)", cmd);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003793 err = -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003794 goto out;
3795 }
3796
Philipp Reisner02918be2010-08-20 14:35:10 +02003797 c.packets[cmd == P_BITMAP]++;
Philipp Reisner257d0af2011-01-26 12:15:29 +01003798 c.bytes[cmd == P_BITMAP] += sizeof(struct p_header) + data_size;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003799
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003800 if (err <= 0) {
3801 if (err < 0)
3802 goto out;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003803 break;
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003804 }
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003805 err = drbd_recv_header(mdev->tconn, &pi);
3806 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003807 goto out;
Philipp Reisner77351055b2011-02-07 17:24:26 +01003808 cmd = pi.cmd;
3809 data_size = pi.size;
Andreas Gruenbacher2c464072010-12-11 21:53:12 +01003810 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003811
3812 INFO_bm_xfer_stats(mdev, "receive", &c);
3813
3814 if (mdev->state.conn == C_WF_BITMAP_T) {
Andreas Gruenbacherde1f8e42010-12-10 21:04:00 +01003815 enum drbd_state_rv rv;
3816
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003817 err = drbd_send_bitmap(mdev);
3818 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003819 goto out;
3820 /* Omit CS_ORDERED with this state transition to avoid deadlocks. */
Andreas Gruenbacherde1f8e42010-12-10 21:04:00 +01003821 rv = _drbd_request_state(mdev, NS(conn, C_WF_SYNC_UUID), CS_VERBOSE);
3822 D_ASSERT(rv == SS_SUCCESS);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003823 } else if (mdev->state.conn != C_WF_BITMAP_S) {
3824 /* admin may have requested C_DISCONNECTING,
3825 * other threads may have noticed network errors */
3826 dev_info(DEV, "unexpected cstate (%s) in receive_bitmap\n",
3827 drbd_conn_str(mdev->state.conn));
3828 }
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003829 err = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003830
Philipp Reisnerb411b362009-09-25 16:07:19 -07003831 out:
Lars Ellenberg20ceb2b2011-01-21 10:56:44 +01003832 drbd_bm_unlock(mdev);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003833 if (!err && mdev->state.conn == C_WF_BITMAP_S)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003834 drbd_start_resync(mdev, C_SYNC_SOURCE);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003835 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003836}
3837
Philipp Reisner2de876e2011-03-15 14:38:01 +01003838static int _tconn_receive_skip(struct drbd_tconn *tconn, unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003839{
3840 /* TODO zero copy sink :) */
3841 static char sink[128];
3842 int size, want, r;
3843
Philipp Reisner02918be2010-08-20 14:35:10 +02003844 size = data_size;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003845 while (size > 0) {
3846 want = min_t(int, size, sizeof(sink));
Philipp Reisner2de876e2011-03-15 14:38:01 +01003847 r = drbd_recv(tconn, sink, want);
3848 if (r <= 0)
Andreas Gruenbacher841ce242010-12-15 19:31:20 +01003849 break;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003850 size -= r;
3851 }
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003852 return size ? -EIO : 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003853}
3854
Philipp Reisner2de876e2011-03-15 14:38:01 +01003855static int receive_skip(struct drbd_conf *mdev, enum drbd_packet cmd,
3856 unsigned int data_size)
3857{
3858 dev_warn(DEV, "skipping unknown optional packet type %d, l: %d!\n",
3859 cmd, data_size);
3860
3861 return _tconn_receive_skip(mdev->tconn, data_size);
3862}
3863
3864static int tconn_receive_skip(struct drbd_tconn *tconn, enum drbd_packet cmd, unsigned int data_size)
3865{
3866 conn_warn(tconn, "skipping packet for non existing volume type %d, l: %d!\n",
3867 cmd, data_size);
3868
3869 return _tconn_receive_skip(tconn, data_size);
3870}
3871
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003872static int receive_UnplugRemote(struct drbd_conf *mdev, enum drbd_packet cmd,
3873 unsigned int data_size)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003874{
Philipp Reisnerb411b362009-09-25 16:07:19 -07003875 /* Make sure we've acked all the TCP data associated
3876 * with the data requests being unplugged */
Philipp Reisnere42325a2011-01-19 13:55:45 +01003877 drbd_tcp_quickack(mdev->tconn->data.socket);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003878
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003879 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003880}
3881
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01003882static int receive_out_of_sync(struct drbd_conf *mdev, enum drbd_packet cmd,
3883 unsigned int data_size)
Philipp Reisner73a01a12010-10-27 14:33:00 +02003884{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003885 struct p_block_desc *p = mdev->tconn->data.rbuf;
Philipp Reisner73a01a12010-10-27 14:33:00 +02003886
Lars Ellenbergf735e3632010-12-17 21:06:18 +01003887 switch (mdev->state.conn) {
3888 case C_WF_SYNC_UUID:
3889 case C_WF_BITMAP_T:
3890 case C_BEHIND:
3891 break;
3892 default:
3893 dev_err(DEV, "ASSERT FAILED cstate = %s, expected: WFSyncUUID|WFBitMapT|Behind\n",
3894 drbd_conn_str(mdev->state.conn));
3895 }
3896
Philipp Reisner73a01a12010-10-27 14:33:00 +02003897 drbd_set_out_of_sync(mdev, be64_to_cpu(p->sector), be32_to_cpu(p->blksize));
3898
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003899 return 0;
Philipp Reisner73a01a12010-10-27 14:33:00 +02003900}
3901
Philipp Reisner02918be2010-08-20 14:35:10 +02003902struct data_cmd {
3903 int expect_payload;
3904 size_t pkt_size;
Philipp Reisnera4fbda82011-03-16 11:13:17 +01003905 enum mdev_or_conn fa_type; /* first argument's type */
Philipp Reisnerd9ae84e2011-03-15 18:50:22 +01003906 union {
3907 int (*mdev_fn)(struct drbd_conf *, enum drbd_packet cmd,
3908 unsigned int to_receive);
3909 int (*conn_fn)(struct drbd_tconn *, enum drbd_packet cmd,
3910 unsigned int to_receive);
3911 };
Philipp Reisnerb411b362009-09-25 16:07:19 -07003912};
3913
Philipp Reisner02918be2010-08-20 14:35:10 +02003914static struct data_cmd drbd_cmd_handler[] = {
Philipp Reisnerd9ae84e2011-03-15 18:50:22 +01003915 [P_DATA] = { 1, sizeof(struct p_data), MDEV, { receive_Data } },
3916 [P_DATA_REPLY] = { 1, sizeof(struct p_data), MDEV, { receive_DataReply } },
3917 [P_RS_DATA_REPLY] = { 1, sizeof(struct p_data), MDEV, { receive_RSDataReply } } ,
3918 [P_BARRIER] = { 0, sizeof(struct p_barrier), MDEV, { receive_Barrier } } ,
3919 [P_BITMAP] = { 1, sizeof(struct p_header), MDEV, { receive_bitmap } } ,
3920 [P_COMPRESSED_BITMAP] = { 1, sizeof(struct p_header), MDEV, { receive_bitmap } } ,
3921 [P_UNPLUG_REMOTE] = { 0, sizeof(struct p_header), MDEV, { receive_UnplugRemote } },
3922 [P_DATA_REQUEST] = { 0, sizeof(struct p_block_req), MDEV, { receive_DataRequest } },
3923 [P_RS_DATA_REQUEST] = { 0, sizeof(struct p_block_req), MDEV, { receive_DataRequest } },
3924 [P_SYNC_PARAM] = { 1, sizeof(struct p_header), MDEV, { receive_SyncParam } },
3925 [P_SYNC_PARAM89] = { 1, sizeof(struct p_header), MDEV, { receive_SyncParam } },
Philipp Reisner7204624c2011-03-15 18:51:47 +01003926 [P_PROTOCOL] = { 1, sizeof(struct p_protocol), CONN, { .conn_fn = receive_protocol } },
Philipp Reisnerd9ae84e2011-03-15 18:50:22 +01003927 [P_UUIDS] = { 0, sizeof(struct p_uuids), MDEV, { receive_uuids } },
3928 [P_SIZES] = { 0, sizeof(struct p_sizes), MDEV, { receive_sizes } },
3929 [P_STATE] = { 0, sizeof(struct p_state), MDEV, { receive_state } },
3930 [P_STATE_CHG_REQ] = { 0, sizeof(struct p_req_state), MDEV, { receive_req_state } },
3931 [P_SYNC_UUID] = { 0, sizeof(struct p_rs_uuid), MDEV, { receive_sync_uuid } },
3932 [P_OV_REQUEST] = { 0, sizeof(struct p_block_req), MDEV, { receive_DataRequest } },
3933 [P_OV_REPLY] = { 1, sizeof(struct p_block_req), MDEV, { receive_DataRequest } },
3934 [P_CSUM_RS_REQUEST] = { 1, sizeof(struct p_block_req), MDEV, { receive_DataRequest } },
3935 [P_DELAY_PROBE] = { 0, sizeof(struct p_delay_probe93), MDEV, { receive_skip } },
3936 [P_OUT_OF_SYNC] = { 0, sizeof(struct p_block_desc), MDEV, { receive_out_of_sync } },
Philipp Reisnerdfafcc82011-03-16 10:55:07 +01003937 [P_CONN_ST_CHG_REQ] = { 0, sizeof(struct p_req_state), CONN, { .conn_fn = receive_req_conn_state } },
Philipp Reisner02918be2010-08-20 14:35:10 +02003938};
3939
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003940static void drbdd(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003941{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01003942 struct p_header *header = tconn->data.rbuf;
Philipp Reisner77351055b2011-02-07 17:24:26 +01003943 struct packet_info pi;
Philipp Reisner02918be2010-08-20 14:35:10 +02003944 size_t shs; /* sub header size */
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003945 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003946
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003947 while (get_t_state(&tconn->receiver) == RUNNING) {
Andreas Gruenbacherdeebe192011-03-25 00:01:04 +01003948 struct data_cmd *cmd;
3949
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003950 drbd_thread_current_set_cpu(&tconn->receiver);
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01003951 if (drbd_recv_header(tconn, &pi))
Philipp Reisner02918be2010-08-20 14:35:10 +02003952 goto err_out;
3953
Andreas Gruenbacherdeebe192011-03-25 00:01:04 +01003954 cmd = &drbd_cmd_handler[pi.cmd];
3955 if (unlikely(pi.cmd >= ARRAY_SIZE(drbd_cmd_handler) || !cmd->mdev_fn)) {
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003956 conn_err(tconn, "unknown packet type %d, l: %d!\n", pi.cmd, pi.size);
Philipp Reisner02918be2010-08-20 14:35:10 +02003957 goto err_out;
Lars Ellenberg0b33a912009-11-16 15:58:04 +01003958 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003959
Andreas Gruenbacherdeebe192011-03-25 00:01:04 +01003960 shs = cmd->pkt_size - sizeof(struct p_header);
3961 if (pi.size - shs > 0 && !cmd->expect_payload) {
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003962 conn_err(tconn, "No payload expected %s l:%d\n", cmdname(pi.cmd), pi.size);
Philipp Reisner02918be2010-08-20 14:35:10 +02003963 goto err_out;
3964 }
3965
Lars Ellenbergc13f7e12010-10-29 23:32:01 +02003966 if (shs) {
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01003967 err = drbd_recv_all_warn(tconn, &header->payload, shs);
3968 if (err)
Lars Ellenbergc13f7e12010-10-29 23:32:01 +02003969 goto err_out;
Lars Ellenbergc13f7e12010-10-29 23:32:01 +02003970 }
3971
Andreas Gruenbacherdeebe192011-03-25 00:01:04 +01003972 if (cmd->fa_type == CONN) {
3973 err = cmd->conn_fn(tconn, pi.cmd, pi.size - shs);
Philipp Reisnerd9ae84e2011-03-15 18:50:22 +01003974 } else {
3975 struct drbd_conf *mdev = vnr_to_mdev(tconn, pi.vnr);
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003976 err = mdev ?
Andreas Gruenbacherdeebe192011-03-25 00:01:04 +01003977 cmd->mdev_fn(mdev, pi.cmd, pi.size - shs) :
Philipp Reisnerd9ae84e2011-03-15 18:50:22 +01003978 tconn_receive_skip(tconn, pi.cmd, pi.size - shs);
3979 }
Philipp Reisner02918be2010-08-20 14:35:10 +02003980
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003981 if (unlikely(err)) {
Philipp Reisnereefc2f72011-02-08 12:55:24 +01003982 conn_err(tconn, "error receiving %s, l: %d!\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01003983 cmdname(pi.cmd), pi.size);
Philipp Reisner02918be2010-08-20 14:35:10 +02003984 goto err_out;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003985 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07003986 }
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003987 return;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003988
Andreas Gruenbacher82bc0192011-03-17 12:10:19 +01003989 err_out:
3990 conn_request_state(tconn, NS(conn, C_PROTOCOL_ERROR), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -07003991}
3992
Philipp Reisner0e29d162011-02-18 14:23:11 +01003993void conn_flush_workqueue(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07003994{
3995 struct drbd_wq_barrier barr;
3996
3997 barr.w.cb = w_prev_work_done;
Philipp Reisner0e29d162011-02-18 14:23:11 +01003998 barr.w.tconn = tconn;
Philipp Reisnerb411b362009-09-25 16:07:19 -07003999 init_completion(&barr.done);
Philipp Reisner0e29d162011-02-18 14:23:11 +01004000 drbd_queue_work(&tconn->data.work, &barr.w);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004001 wait_for_completion(&barr.done);
4002}
4003
Philipp Reisner360cc742011-02-08 14:29:53 +01004004static void drbd_disconnect(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004005{
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004006 enum drbd_conns oc;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004007 int rv = SS_UNKNOWN_ERROR;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004008
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004009 if (tconn->cstate == C_STANDALONE)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004010 return;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004011
4012 /* asender does not clean up anything. it must not interfere, either */
Philipp Reisner360cc742011-02-08 14:29:53 +01004013 drbd_thread_stop(&tconn->asender);
4014 drbd_free_sock(tconn);
4015
4016 idr_for_each(&tconn->volumes, drbd_disconnected, tconn);
Philipp Reisner360cc742011-02-08 14:29:53 +01004017 conn_info(tconn, "Connection closed\n");
4018
Philipp Reisnercb703452011-03-24 11:03:07 +01004019 if (conn_highest_role(tconn) == R_PRIMARY && conn_highest_pdsk(tconn) >= D_UNKNOWN)
4020 conn_try_outdate_peer_async(tconn);
4021
Philipp Reisner360cc742011-02-08 14:29:53 +01004022 spin_lock_irq(&tconn->req_lock);
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004023 oc = tconn->cstate;
4024 if (oc >= C_UNCONNECTED)
4025 rv = _conn_request_state(tconn, NS(conn, C_UNCONNECTED), CS_VERBOSE);
4026
Philipp Reisner360cc742011-02-08 14:29:53 +01004027 spin_unlock_irq(&tconn->req_lock);
4028
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004029 if (oc == C_DISCONNECTING) {
Philipp Reisner360cc742011-02-08 14:29:53 +01004030 wait_event(tconn->net_cnt_wait, atomic_read(&tconn->net_cnt) == 0);
4031
4032 crypto_free_hash(tconn->cram_hmac_tfm);
4033 tconn->cram_hmac_tfm = NULL;
4034
4035 kfree(tconn->net_conf);
4036 tconn->net_conf = NULL;
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004037 conn_request_state(tconn, NS(conn, C_STANDALONE), CS_VERBOSE);
Philipp Reisner360cc742011-02-08 14:29:53 +01004038 }
4039}
4040
4041static int drbd_disconnected(int vnr, void *p, void *data)
4042{
4043 struct drbd_conf *mdev = (struct drbd_conf *)p;
4044 enum drbd_fencing_p fp;
4045 unsigned int i;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004046
Philipp Reisner85719572010-07-21 10:20:17 +02004047 /* wait for current activity to cease. */
Philipp Reisner87eeee42011-01-19 14:16:30 +01004048 spin_lock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004049 _drbd_wait_ee_list_empty(mdev, &mdev->active_ee);
4050 _drbd_wait_ee_list_empty(mdev, &mdev->sync_ee);
4051 _drbd_wait_ee_list_empty(mdev, &mdev->read_ee);
Philipp Reisner87eeee42011-01-19 14:16:30 +01004052 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004053
4054 /* We do not have data structures that would allow us to
4055 * get the rs_pending_cnt down to 0 again.
4056 * * On C_SYNC_TARGET we do not have any data structures describing
4057 * the pending RSDataRequest's we have sent.
4058 * * On C_SYNC_SOURCE there is no data structure that tracks
4059 * the P_RS_DATA_REPLY blocks that we sent to the SyncTarget.
4060 * And no, it is not the sum of the reference counts in the
4061 * resync_LRU. The resync_LRU tracks the whole operation including
4062 * the disk-IO, while the rs_pending_cnt only tracks the blocks
4063 * on the fly. */
4064 drbd_rs_cancel_all(mdev);
4065 mdev->rs_total = 0;
4066 mdev->rs_failed = 0;
4067 atomic_set(&mdev->rs_pending_cnt, 0);
4068 wake_up(&mdev->misc_wait);
4069
Philipp Reisner7fde2be2011-03-01 11:08:28 +01004070 del_timer(&mdev->request_timer);
4071
Philipp Reisnerb411b362009-09-25 16:07:19 -07004072 del_timer_sync(&mdev->resync_timer);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004073 resync_timer_fn((unsigned long)mdev);
4074
Philipp Reisnerb411b362009-09-25 16:07:19 -07004075 /* wait for all w_e_end_data_req, w_e_end_rsdata_req, w_send_barrier,
4076 * w_make_resync_request etc. which may still be on the worker queue
4077 * to be "canceled" */
Philipp Reisnera21e9292011-02-08 15:08:49 +01004078 drbd_flush_workqueue(mdev);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004079
4080 /* This also does reclaim_net_ee(). If we do this too early, we might
4081 * miss some resync ee and pages.*/
4082 drbd_process_done_ee(mdev);
4083
4084 kfree(mdev->p_uuid);
4085 mdev->p_uuid = NULL;
4086
Philipp Reisnerfb22c402010-09-08 23:20:21 +02004087 if (!is_susp(mdev->state))
Philipp Reisner2f5cdd02011-02-21 14:29:27 +01004088 tl_clear(mdev->tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004089
Philipp Reisnerb411b362009-09-25 16:07:19 -07004090 drbd_md_sync(mdev);
4091
4092 fp = FP_DONT_CARE;
4093 if (get_ldev(mdev)) {
4094 fp = mdev->ldev->dc.fencing;
4095 put_ldev(mdev);
4096 }
4097
Lars Ellenberg20ceb2b2011-01-21 10:56:44 +01004098 /* serialize with bitmap writeout triggered by the state change,
4099 * if any. */
4100 wait_event(mdev->misc_wait, !test_bit(BITMAP_IO, &mdev->flags));
4101
Philipp Reisnerb411b362009-09-25 16:07:19 -07004102 /* tcp_close and release of sendpage pages can be deferred. I don't
4103 * want to use SO_LINGER, because apparently it can be deferred for
4104 * more than 20 seconds (longest time I checked).
4105 *
4106 * Actually we don't care for exactly when the network stack does its
4107 * put_page(), but release our reference on these pages right here.
4108 */
4109 i = drbd_release_ee(mdev, &mdev->net_ee);
4110 if (i)
4111 dev_info(DEV, "net_ee not empty, killed %u entries\n", i);
Lars Ellenberg435f0742010-09-06 12:30:25 +02004112 i = atomic_read(&mdev->pp_in_use_by_net);
4113 if (i)
4114 dev_info(DEV, "pp_in_use_by_net = %d, expected 0\n", i);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004115 i = atomic_read(&mdev->pp_in_use);
4116 if (i)
Lars Ellenberg45bb9122010-05-14 17:10:48 +02004117 dev_info(DEV, "pp_in_use = %d, expected 0\n", i);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004118
4119 D_ASSERT(list_empty(&mdev->read_ee));
4120 D_ASSERT(list_empty(&mdev->active_ee));
4121 D_ASSERT(list_empty(&mdev->sync_ee));
4122 D_ASSERT(list_empty(&mdev->done_ee));
4123
4124 /* ok, no more ee's on the fly, it is safe to reset the epoch_size */
4125 atomic_set(&mdev->current_epoch->epoch_size, 0);
4126 D_ASSERT(list_empty(&mdev->current_epoch->list));
Philipp Reisner360cc742011-02-08 14:29:53 +01004127
4128 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004129}
4130
4131/*
4132 * We support PRO_VERSION_MIN to PRO_VERSION_MAX. The protocol version
4133 * we can agree on is stored in agreed_pro_version.
4134 *
4135 * feature flags and the reserved array should be enough room for future
4136 * enhancements of the handshake protocol, and possible plugins...
4137 *
4138 * for now, they are expected to be zero, but ignored.
4139 */
Philipp Reisner8a22ccc2011-02-07 16:47:12 +01004140static int drbd_send_handshake(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004141{
Philipp Reisnere6b3ea82011-01-19 14:02:01 +01004142 /* ASSERT current == mdev->tconn->receiver ... */
Andreas Gruenbacher5a87d922011-03-24 21:17:52 +01004143 struct p_handshake *p = tconn->data.sbuf;
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004144 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004145
Philipp Reisner8a22ccc2011-02-07 16:47:12 +01004146 if (mutex_lock_interruptible(&tconn->data.mutex)) {
4147 conn_err(tconn, "interrupted during initial handshake\n");
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004148 return -EINTR;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004149 }
4150
Philipp Reisner8a22ccc2011-02-07 16:47:12 +01004151 if (tconn->data.socket == NULL) {
4152 mutex_unlock(&tconn->data.mutex);
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004153 return -EIO;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004154 }
4155
4156 memset(p, 0, sizeof(*p));
4157 p->protocol_min = cpu_to_be32(PRO_VERSION_MIN);
4158 p->protocol_max = cpu_to_be32(PRO_VERSION_MAX);
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004159 err = _conn_send_cmd(tconn, 0, tconn->data.socket, P_HAND_SHAKE,
Andreas Gruenbacherecf23632011-03-15 23:48:25 +01004160 &p->head, sizeof(*p), 0);
Philipp Reisner8a22ccc2011-02-07 16:47:12 +01004161 mutex_unlock(&tconn->data.mutex);
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004162 return err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004163}
4164
4165/*
4166 * return values:
4167 * 1 yes, we have a valid connection
4168 * 0 oops, did not work out, please try again
4169 * -1 peer talks different language,
4170 * no point in trying again, please go standalone.
4171 */
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004172static int drbd_do_handshake(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004173{
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004174 /* ASSERT current == tconn->receiver ... */
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004175 struct p_handshake *p = tconn->data.rbuf;
Philipp Reisner02918be2010-08-20 14:35:10 +02004176 const int expect = sizeof(struct p_handshake) - sizeof(struct p_header80);
Philipp Reisner77351055b2011-02-07 17:24:26 +01004177 struct packet_info pi;
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01004178 int err;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004179
Andreas Gruenbachere8d17b02011-03-16 00:54:19 +01004180 err = drbd_send_handshake(tconn);
4181 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004182 return 0;
4183
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004184 err = drbd_recv_header(tconn, &pi);
4185 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004186 return 0;
4187
Philipp Reisner77351055b2011-02-07 17:24:26 +01004188 if (pi.cmd != P_HAND_SHAKE) {
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004189 conn_err(tconn, "expected HandShake packet, received: %s (0x%04x)\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004190 cmdname(pi.cmd), pi.cmd);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004191 return -1;
4192 }
4193
Philipp Reisner77351055b2011-02-07 17:24:26 +01004194 if (pi.size != expect) {
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004195 conn_err(tconn, "expected HandShake length: %u, received: %u\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004196 expect, pi.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004197 return -1;
4198 }
4199
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01004200 err = drbd_recv_all_warn(tconn, &p->head.payload, expect);
4201 if (err)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004202 return 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004203
Philipp Reisnerb411b362009-09-25 16:07:19 -07004204 p->protocol_min = be32_to_cpu(p->protocol_min);
4205 p->protocol_max = be32_to_cpu(p->protocol_max);
4206 if (p->protocol_max == 0)
4207 p->protocol_max = p->protocol_min;
4208
4209 if (PRO_VERSION_MAX < p->protocol_min ||
4210 PRO_VERSION_MIN > p->protocol_max)
4211 goto incompat;
4212
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004213 tconn->agreed_pro_version = min_t(int, PRO_VERSION_MAX, p->protocol_max);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004214
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004215 conn_info(tconn, "Handshake successful: "
4216 "Agreed network protocol version %d\n", tconn->agreed_pro_version);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004217
4218 return 1;
4219
4220 incompat:
Philipp Reisner65d11ed2011-02-07 17:35:59 +01004221 conn_err(tconn, "incompatible DRBD dialects: "
Philipp Reisnerb411b362009-09-25 16:07:19 -07004222 "I support %d-%d, peer supports %d-%d\n",
4223 PRO_VERSION_MIN, PRO_VERSION_MAX,
4224 p->protocol_min, p->protocol_max);
4225 return -1;
4226}
4227
4228#if !defined(CONFIG_CRYPTO_HMAC) && !defined(CONFIG_CRYPTO_HMAC_MODULE)
Philipp Reisner13e60372011-02-08 09:54:40 +01004229static int drbd_do_auth(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004230{
4231 dev_err(DEV, "This kernel was build without CONFIG_CRYPTO_HMAC.\n");
4232 dev_err(DEV, "You need to disable 'cram-hmac-alg' in drbd.conf.\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +01004233 return -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004234}
4235#else
4236#define CHALLENGE_LEN 64
Johannes Thomab10d96c2010-01-07 16:02:50 +01004237
4238/* Return value:
4239 1 - auth succeeded,
4240 0 - failed, try again (network error),
4241 -1 - auth failed, don't try again.
4242*/
4243
Philipp Reisner13e60372011-02-08 09:54:40 +01004244static int drbd_do_auth(struct drbd_tconn *tconn)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004245{
4246 char my_challenge[CHALLENGE_LEN]; /* 64 Bytes... */
4247 struct scatterlist sg;
4248 char *response = NULL;
4249 char *right_response = NULL;
4250 char *peers_ch = NULL;
Philipp Reisner13e60372011-02-08 09:54:40 +01004251 unsigned int key_len = strlen(tconn->net_conf->shared_secret);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004252 unsigned int resp_size;
4253 struct hash_desc desc;
Philipp Reisner77351055b2011-02-07 17:24:26 +01004254 struct packet_info pi;
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004255 int err, rv;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004256
Philipp Reisner13e60372011-02-08 09:54:40 +01004257 desc.tfm = tconn->cram_hmac_tfm;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004258 desc.flags = 0;
4259
Philipp Reisner13e60372011-02-08 09:54:40 +01004260 rv = crypto_hash_setkey(tconn->cram_hmac_tfm,
4261 (u8 *)tconn->net_conf->shared_secret, key_len);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004262 if (rv) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004263 conn_err(tconn, "crypto_hash_setkey() failed with %d\n", rv);
Johannes Thomab10d96c2010-01-07 16:02:50 +01004264 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004265 goto fail;
4266 }
4267
4268 get_random_bytes(my_challenge, CHALLENGE_LEN);
4269
Andreas Gruenbacherce9879c2011-03-15 23:34:29 +01004270 rv = !conn_send_cmd2(tconn, P_AUTH_CHALLENGE, my_challenge, CHALLENGE_LEN);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004271 if (!rv)
4272 goto fail;
4273
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004274 err = drbd_recv_header(tconn, &pi);
4275 if (err) {
4276 rv = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004277 goto fail;
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004278 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004279
Philipp Reisner77351055b2011-02-07 17:24:26 +01004280 if (pi.cmd != P_AUTH_CHALLENGE) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004281 conn_err(tconn, "expected AuthChallenge packet, received: %s (0x%04x)\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004282 cmdname(pi.cmd), pi.cmd);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004283 rv = 0;
4284 goto fail;
4285 }
4286
Philipp Reisner77351055b2011-02-07 17:24:26 +01004287 if (pi.size > CHALLENGE_LEN * 2) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004288 conn_err(tconn, "expected AuthChallenge payload too big.\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +01004289 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004290 goto fail;
4291 }
4292
Philipp Reisner77351055b2011-02-07 17:24:26 +01004293 peers_ch = kmalloc(pi.size, GFP_NOIO);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004294 if (peers_ch == NULL) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004295 conn_err(tconn, "kmalloc of peers_ch failed\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +01004296 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004297 goto fail;
4298 }
4299
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01004300 err = drbd_recv_all_warn(tconn, peers_ch, pi.size);
4301 if (err) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004302 rv = 0;
4303 goto fail;
4304 }
4305
Philipp Reisner13e60372011-02-08 09:54:40 +01004306 resp_size = crypto_hash_digestsize(tconn->cram_hmac_tfm);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004307 response = kmalloc(resp_size, GFP_NOIO);
4308 if (response == NULL) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004309 conn_err(tconn, "kmalloc of response failed\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +01004310 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004311 goto fail;
4312 }
4313
4314 sg_init_table(&sg, 1);
Philipp Reisner77351055b2011-02-07 17:24:26 +01004315 sg_set_buf(&sg, peers_ch, pi.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004316
4317 rv = crypto_hash_digest(&desc, &sg, sg.length, response);
4318 if (rv) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004319 conn_err(tconn, "crypto_hash_digest() failed with %d\n", rv);
Johannes Thomab10d96c2010-01-07 16:02:50 +01004320 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004321 goto fail;
4322 }
4323
Andreas Gruenbacherce9879c2011-03-15 23:34:29 +01004324 rv = !conn_send_cmd2(tconn, P_AUTH_RESPONSE, response, resp_size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004325 if (!rv)
4326 goto fail;
4327
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004328 err = drbd_recv_header(tconn, &pi);
4329 if (err) {
4330 rv = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004331 goto fail;
Andreas Gruenbacher69bc7bc2011-03-16 17:31:52 +01004332 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004333
Philipp Reisner77351055b2011-02-07 17:24:26 +01004334 if (pi.cmd != P_AUTH_RESPONSE) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004335 conn_err(tconn, "expected AuthResponse packet, received: %s (0x%04x)\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004336 cmdname(pi.cmd), pi.cmd);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004337 rv = 0;
4338 goto fail;
4339 }
4340
Philipp Reisner77351055b2011-02-07 17:24:26 +01004341 if (pi.size != resp_size) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004342 conn_err(tconn, "expected AuthResponse payload of wrong size\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004343 rv = 0;
4344 goto fail;
4345 }
4346
Andreas Gruenbachera5c31902011-03-24 03:28:04 +01004347 err = drbd_recv_all_warn(tconn, response , resp_size);
4348 if (err) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004349 rv = 0;
4350 goto fail;
4351 }
4352
4353 right_response = kmalloc(resp_size, GFP_NOIO);
Julia Lawall2d1ee872009-12-27 22:27:11 +01004354 if (right_response == NULL) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004355 conn_err(tconn, "kmalloc of right_response failed\n");
Johannes Thomab10d96c2010-01-07 16:02:50 +01004356 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004357 goto fail;
4358 }
4359
4360 sg_set_buf(&sg, my_challenge, CHALLENGE_LEN);
4361
4362 rv = crypto_hash_digest(&desc, &sg, sg.length, right_response);
4363 if (rv) {
Philipp Reisner13e60372011-02-08 09:54:40 +01004364 conn_err(tconn, "crypto_hash_digest() failed with %d\n", rv);
Johannes Thomab10d96c2010-01-07 16:02:50 +01004365 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004366 goto fail;
4367 }
4368
4369 rv = !memcmp(response, right_response, resp_size);
4370
4371 if (rv)
Philipp Reisner13e60372011-02-08 09:54:40 +01004372 conn_info(tconn, "Peer authenticated using %d bytes of '%s' HMAC\n",
4373 resp_size, tconn->net_conf->cram_hmac_alg);
Johannes Thomab10d96c2010-01-07 16:02:50 +01004374 else
4375 rv = -1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004376
4377 fail:
4378 kfree(peers_ch);
4379 kfree(response);
4380 kfree(right_response);
4381
4382 return rv;
4383}
4384#endif
4385
4386int drbdd_init(struct drbd_thread *thi)
4387{
Philipp Reisner392c8802011-02-09 10:33:31 +01004388 struct drbd_tconn *tconn = thi->tconn;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004389 int h;
4390
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004391 conn_info(tconn, "receiver (re)started\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004392
4393 do {
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004394 h = drbd_connect(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004395 if (h == 0) {
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004396 drbd_disconnect(tconn);
Philipp Reisner20ee6392011-01-18 15:28:59 +01004397 schedule_timeout_interruptible(HZ);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004398 }
4399 if (h == -1) {
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004400 conn_warn(tconn, "Discarding network configuration.\n");
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004401 conn_request_state(tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004402 }
4403 } while (h == 0);
4404
4405 if (h > 0) {
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004406 if (get_net_conf(tconn)) {
4407 drbdd(tconn);
4408 put_net_conf(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004409 }
4410 }
4411
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004412 drbd_disconnect(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004413
Philipp Reisner4d641dd2011-02-08 15:40:24 +01004414 conn_info(tconn, "receiver terminated\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004415 return 0;
4416}
4417
4418/* ********* acknowledge sender ******** */
4419
Philipp Reisnere4f78ed2011-03-16 11:27:48 +01004420static int got_conn_RqSReply(struct drbd_tconn *tconn, enum drbd_packet cmd)
4421{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004422 struct p_req_state_reply *p = tconn->meta.rbuf;
Philipp Reisnere4f78ed2011-03-16 11:27:48 +01004423 int retcode = be32_to_cpu(p->retcode);
4424
4425 if (retcode >= SS_SUCCESS) {
4426 set_bit(CONN_WD_ST_CHG_OKAY, &tconn->flags);
4427 } else {
4428 set_bit(CONN_WD_ST_CHG_FAIL, &tconn->flags);
4429 conn_err(tconn, "Requested state change failed by peer: %s (%d)\n",
4430 drbd_set_st_err_str(retcode), retcode);
4431 }
4432 wake_up(&tconn->ping_wait);
4433
4434 return true;
4435}
4436
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004437static int got_RqSReply(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004438{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004439 struct p_req_state_reply *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004440 int retcode = be32_to_cpu(p->retcode);
4441
Philipp Reisnere4f78ed2011-03-16 11:27:48 +01004442 if (retcode >= SS_SUCCESS) {
4443 set_bit(CL_ST_CHG_SUCCESS, &mdev->flags);
4444 } else {
4445 set_bit(CL_ST_CHG_FAIL, &mdev->flags);
4446 dev_err(DEV, "Requested state change failed by peer: %s (%d)\n",
4447 drbd_set_st_err_str(retcode), retcode);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004448 }
Philipp Reisnere4f78ed2011-03-16 11:27:48 +01004449 wake_up(&mdev->state_wait);
4450
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004451 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004452}
4453
Philipp Reisnerf19e4f82011-03-16 11:21:50 +01004454static int got_Ping(struct drbd_tconn *tconn, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004455{
Philipp Reisnerf19e4f82011-03-16 11:21:50 +01004456 return drbd_send_ping_ack(tconn);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004457
4458}
4459
Philipp Reisnerf19e4f82011-03-16 11:21:50 +01004460static int got_PingAck(struct drbd_tconn *tconn, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004461{
4462 /* restore idle timeout */
Philipp Reisner2a67d8b2011-02-09 14:10:32 +01004463 tconn->meta.socket->sk->sk_rcvtimeo = tconn->net_conf->ping_int*HZ;
4464 if (!test_and_set_bit(GOT_PING_ACK, &tconn->flags))
4465 wake_up(&tconn->ping_wait);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004466
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004467 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004468}
4469
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004470static int got_IsInSync(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004471{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004472 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004473 sector_t sector = be64_to_cpu(p->sector);
4474 int blksize = be32_to_cpu(p->blksize);
4475
Philipp Reisner31890f42011-01-19 14:12:51 +01004476 D_ASSERT(mdev->tconn->agreed_pro_version >= 89);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004477
4478 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4479
Lars Ellenberg1d53f092010-09-05 01:13:24 +02004480 if (get_ldev(mdev)) {
4481 drbd_rs_complete_io(mdev, sector);
4482 drbd_set_in_sync(mdev, sector, blksize);
4483 /* rs_same_csums is supposed to count in units of BM_BLOCK_SIZE */
4484 mdev->rs_same_csum += (blksize >> BM_BLOCK_SHIFT);
4485 put_ldev(mdev);
4486 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004487 dec_rs_pending(mdev);
Philipp Reisner778f2712010-07-06 11:14:00 +02004488 atomic_add(blksize >> 9, &mdev->rs_sect_in);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004489
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004490 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004491}
4492
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01004493static int
4494validate_req_change_req_state(struct drbd_conf *mdev, u64 id, sector_t sector,
4495 struct rb_root *root, const char *func,
4496 enum drbd_req_event what, bool missing_ok)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004497{
4498 struct drbd_request *req;
4499 struct bio_and_error m;
4500
Philipp Reisner87eeee42011-01-19 14:16:30 +01004501 spin_lock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01004502 req = find_request(mdev, root, id, sector, missing_ok, func);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004503 if (unlikely(!req)) {
Philipp Reisner87eeee42011-01-19 14:16:30 +01004504 spin_unlock_irq(&mdev->tconn->req_lock);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004505 return false;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004506 }
4507 __req_mod(req, what, &m);
Philipp Reisner87eeee42011-01-19 14:16:30 +01004508 spin_unlock_irq(&mdev->tconn->req_lock);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004509
4510 if (m.bio)
4511 complete_master_bio(mdev, &m);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004512 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004513}
4514
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004515static int got_BlockAck(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004516{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004517 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004518 sector_t sector = be64_to_cpu(p->sector);
4519 int blksize = be32_to_cpu(p->blksize);
4520 enum drbd_req_event what;
4521
4522 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4523
Andreas Gruenbacher579b57e2011-01-13 18:40:57 +01004524 if (p->block_id == ID_SYNCER) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004525 drbd_set_in_sync(mdev, sector, blksize);
4526 dec_rs_pending(mdev);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004527 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004528 }
Philipp Reisner257d0af2011-01-26 12:15:29 +01004529 switch (cmd) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004530 case P_RS_WRITE_ACK:
Philipp Reisner89e58e72011-01-19 13:12:45 +01004531 D_ASSERT(mdev->tconn->net_conf->wire_protocol == DRBD_PROT_C);
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01004532 what = WRITE_ACKED_BY_PEER_AND_SIS;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004533 break;
4534 case P_WRITE_ACK:
Philipp Reisner89e58e72011-01-19 13:12:45 +01004535 D_ASSERT(mdev->tconn->net_conf->wire_protocol == DRBD_PROT_C);
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01004536 what = WRITE_ACKED_BY_PEER;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004537 break;
4538 case P_RECV_ACK:
Philipp Reisner89e58e72011-01-19 13:12:45 +01004539 D_ASSERT(mdev->tconn->net_conf->wire_protocol == DRBD_PROT_B);
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01004540 what = RECV_ACKED_BY_PEER;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004541 break;
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01004542 case P_DISCARD_WRITE:
Philipp Reisner89e58e72011-01-19 13:12:45 +01004543 D_ASSERT(mdev->tconn->net_conf->wire_protocol == DRBD_PROT_C);
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01004544 what = DISCARD_WRITE;
4545 break;
4546 case P_RETRY_WRITE:
4547 D_ASSERT(mdev->tconn->net_conf->wire_protocol == DRBD_PROT_C);
4548 what = POSTPONE_WRITE;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004549 break;
4550 default:
4551 D_ASSERT(0);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004552 return false;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004553 }
4554
4555 return validate_req_change_req_state(mdev, p->block_id, sector,
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01004556 &mdev->write_requests, __func__,
4557 what, false);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004558}
4559
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004560static int got_NegAck(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004561{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004562 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004563 sector_t sector = be64_to_cpu(p->sector);
Philipp Reisner2deb8332011-01-17 18:39:18 +01004564 int size = be32_to_cpu(p->blksize);
Philipp Reisner89e58e72011-01-19 13:12:45 +01004565 bool missing_ok = mdev->tconn->net_conf->wire_protocol == DRBD_PROT_A ||
4566 mdev->tconn->net_conf->wire_protocol == DRBD_PROT_B;
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01004567 bool found;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004568
4569 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4570
Andreas Gruenbacher579b57e2011-01-13 18:40:57 +01004571 if (p->block_id == ID_SYNCER) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004572 dec_rs_pending(mdev);
4573 drbd_rs_failed_io(mdev, sector, size);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004574 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004575 }
Philipp Reisner2deb8332011-01-17 18:39:18 +01004576
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01004577 found = validate_req_change_req_state(mdev, p->block_id, sector,
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01004578 &mdev->write_requests, __func__,
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01004579 NEG_ACKED, missing_ok);
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01004580 if (!found) {
4581 /* Protocol A has no P_WRITE_ACKs, but has P_NEG_ACKs.
4582 The master bio might already be completed, therefore the
4583 request is no longer in the collision hash. */
4584 /* In Protocol B we might already have got a P_RECV_ACK
4585 but then get a P_NEG_ACK afterwards. */
4586 if (!missing_ok)
Philipp Reisner2deb8332011-01-17 18:39:18 +01004587 return false;
Andreas Gruenbacherc3afd8f2011-01-20 22:25:40 +01004588 drbd_set_out_of_sync(mdev, sector, size);
Philipp Reisner2deb8332011-01-17 18:39:18 +01004589 }
Philipp Reisner2deb8332011-01-17 18:39:18 +01004590 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004591}
4592
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004593static int got_NegDReply(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004594{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004595 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004596 sector_t sector = be64_to_cpu(p->sector);
4597
4598 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
Andreas Gruenbacher7be8da02011-02-22 02:15:32 +01004599
Philipp Reisnerb411b362009-09-25 16:07:19 -07004600 dev_err(DEV, "Got NegDReply; Sector %llus, len %u; Fail original request.\n",
4601 (unsigned long long)sector, be32_to_cpu(p->blksize));
4602
4603 return validate_req_change_req_state(mdev, p->block_id, sector,
Andreas Gruenbacherbc9c5c42011-01-21 18:00:55 +01004604 &mdev->read_requests, __func__,
Andreas Gruenbacher8554df12011-01-25 15:37:43 +01004605 NEG_ACKED, false);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004606}
4607
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004608static int got_NegRSDReply(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004609{
4610 sector_t sector;
4611 int size;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004612 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004613
4614 sector = be64_to_cpu(p->sector);
4615 size = be32_to_cpu(p->blksize);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004616
4617 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4618
4619 dec_rs_pending(mdev);
4620
4621 if (get_ldev_if_state(mdev, D_FAILED)) {
4622 drbd_rs_complete_io(mdev, sector);
Philipp Reisner257d0af2011-01-26 12:15:29 +01004623 switch (cmd) {
Philipp Reisnerd612d302010-12-27 10:53:28 +01004624 case P_NEG_RS_DREPLY:
4625 drbd_rs_failed_io(mdev, sector, size);
4626 case P_RS_CANCEL:
4627 break;
4628 default:
4629 D_ASSERT(0);
4630 put_ldev(mdev);
4631 return false;
4632 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004633 put_ldev(mdev);
4634 }
4635
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004636 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004637}
4638
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004639static int got_BarrierAck(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004640{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004641 struct p_barrier_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004642
Philipp Reisner2f5cdd02011-02-21 14:29:27 +01004643 tl_release(mdev->tconn, p->barrier, be32_to_cpu(p->set_size));
Philipp Reisnerb411b362009-09-25 16:07:19 -07004644
Philipp Reisnerc4752ef2010-10-27 17:32:36 +02004645 if (mdev->state.conn == C_AHEAD &&
4646 atomic_read(&mdev->ap_in_flight) == 0 &&
Philipp Reisner370a43e2011-01-14 16:03:11 +01004647 !test_and_set_bit(AHEAD_TO_SYNC_SOURCE, &mdev->current_epoch->flags)) {
4648 mdev->start_resync_timer.expires = jiffies + HZ;
4649 add_timer(&mdev->start_resync_timer);
Philipp Reisnerc4752ef2010-10-27 17:32:36 +02004650 }
4651
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004652 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004653}
4654
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004655static int got_OVResult(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004656{
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004657 struct p_block_ack *p = mdev->tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004658 struct drbd_work *w;
4659 sector_t sector;
4660 int size;
4661
4662 sector = be64_to_cpu(p->sector);
4663 size = be32_to_cpu(p->blksize);
4664
4665 update_peer_seq(mdev, be32_to_cpu(p->seq_num));
4666
4667 if (be64_to_cpu(p->block_id) == ID_OUT_OF_SYNC)
Andreas Gruenbacher8f7bed72010-12-19 23:53:14 +01004668 drbd_ov_out_of_sync_found(mdev, sector, size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004669 else
Andreas Gruenbacher8f7bed72010-12-19 23:53:14 +01004670 ov_out_of_sync_print(mdev);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004671
Lars Ellenberg1d53f092010-09-05 01:13:24 +02004672 if (!get_ldev(mdev))
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004673 return true;
Lars Ellenberg1d53f092010-09-05 01:13:24 +02004674
Philipp Reisnerb411b362009-09-25 16:07:19 -07004675 drbd_rs_complete_io(mdev, sector);
4676 dec_rs_pending(mdev);
4677
Lars Ellenbergea5442a2010-11-05 09:48:01 +01004678 --mdev->ov_left;
4679
4680 /* let's advance progress step marks only for every other megabyte */
4681 if ((mdev->ov_left & 0x200) == 0x200)
4682 drbd_advance_rs_marks(mdev, mdev->ov_left);
4683
4684 if (mdev->ov_left == 0) {
Philipp Reisnerb411b362009-09-25 16:07:19 -07004685 w = kmalloc(sizeof(*w), GFP_NOIO);
4686 if (w) {
4687 w->cb = w_ov_finished;
Philipp Reisnera21e9292011-02-08 15:08:49 +01004688 w->mdev = mdev;
Philipp Reisnere42325a2011-01-19 13:55:45 +01004689 drbd_queue_work_front(&mdev->tconn->data.work, w);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004690 } else {
4691 dev_err(DEV, "kmalloc(w) failed.");
Andreas Gruenbacher8f7bed72010-12-19 23:53:14 +01004692 ov_out_of_sync_print(mdev);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004693 drbd_resync_finished(mdev);
4694 }
4695 }
Lars Ellenberg1d53f092010-09-05 01:13:24 +02004696 put_ldev(mdev);
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004697 return true;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004698}
4699
Andreas Gruenbacherd8763022011-01-26 17:39:41 +01004700static int got_skip(struct drbd_conf *mdev, enum drbd_packet cmd)
Philipp Reisner0ced55a2010-04-30 15:26:20 +02004701{
Andreas Gruenbacher81e84652010-12-09 15:03:57 +01004702 return true;
Philipp Reisner0ced55a2010-04-30 15:26:20 +02004703}
4704
Philipp Reisner32862ec2011-02-08 16:41:01 +01004705static int tconn_process_done_ee(struct drbd_tconn *tconn)
4706{
Philipp Reisner082a3432011-03-15 16:05:42 +01004707 struct drbd_conf *mdev;
4708 int i, not_empty = 0;
Philipp Reisner32862ec2011-02-08 16:41:01 +01004709
4710 do {
4711 clear_bit(SIGNAL_ASENDER, &tconn->flags);
4712 flush_signals(current);
Philipp Reisner082a3432011-03-15 16:05:42 +01004713 idr_for_each_entry(&tconn->volumes, mdev, i) {
Andreas Gruenbachere2b30322011-03-16 17:16:12 +01004714 if (drbd_process_done_ee(mdev))
Philipp Reisner082a3432011-03-15 16:05:42 +01004715 return 1; /* error */
4716 }
Philipp Reisner32862ec2011-02-08 16:41:01 +01004717 set_bit(SIGNAL_ASENDER, &tconn->flags);
Philipp Reisner082a3432011-03-15 16:05:42 +01004718
4719 spin_lock_irq(&tconn->req_lock);
4720 idr_for_each_entry(&tconn->volumes, mdev, i) {
4721 not_empty = !list_empty(&mdev->done_ee);
4722 if (not_empty)
4723 break;
4724 }
4725 spin_unlock_irq(&tconn->req_lock);
Philipp Reisner32862ec2011-02-08 16:41:01 +01004726 } while (not_empty);
4727
4728 return 0;
4729}
4730
Andreas Gruenbacher7201b972011-03-14 18:23:00 +01004731struct asender_cmd {
4732 size_t pkt_size;
Philipp Reisnera4fbda82011-03-16 11:13:17 +01004733 enum mdev_or_conn fa_type; /* first argument's type */
4734 union {
4735 int (*mdev_fn)(struct drbd_conf *mdev, enum drbd_packet cmd);
4736 int (*conn_fn)(struct drbd_tconn *tconn, enum drbd_packet cmd);
4737 };
Andreas Gruenbacher7201b972011-03-14 18:23:00 +01004738};
4739
4740static struct asender_cmd asender_tbl[] = {
Philipp Reisnerf19e4f82011-03-16 11:21:50 +01004741 [P_PING] = { sizeof(struct p_header), CONN, { .conn_fn = got_Ping } },
4742 [P_PING_ACK] = { sizeof(struct p_header), CONN, { .conn_fn = got_PingAck } },
Philipp Reisnera4fbda82011-03-16 11:13:17 +01004743 [P_RECV_ACK] = { sizeof(struct p_block_ack), MDEV, { got_BlockAck } },
4744 [P_WRITE_ACK] = { sizeof(struct p_block_ack), MDEV, { got_BlockAck } },
4745 [P_RS_WRITE_ACK] = { sizeof(struct p_block_ack), MDEV, { got_BlockAck } },
4746 [P_DISCARD_WRITE] = { sizeof(struct p_block_ack), MDEV, { got_BlockAck } },
4747 [P_NEG_ACK] = { sizeof(struct p_block_ack), MDEV, { got_NegAck } },
4748 [P_NEG_DREPLY] = { sizeof(struct p_block_ack), MDEV, { got_NegDReply } },
4749 [P_NEG_RS_DREPLY] = { sizeof(struct p_block_ack), MDEV, { got_NegRSDReply } },
4750 [P_OV_RESULT] = { sizeof(struct p_block_ack), MDEV, { got_OVResult } },
4751 [P_BARRIER_ACK] = { sizeof(struct p_barrier_ack), MDEV, { got_BarrierAck } },
4752 [P_STATE_CHG_REPLY] = { sizeof(struct p_req_state_reply), MDEV, { got_RqSReply } },
4753 [P_RS_IS_IN_SYNC] = { sizeof(struct p_block_ack), MDEV, { got_IsInSync } },
4754 [P_DELAY_PROBE] = { sizeof(struct p_delay_probe93), MDEV, { got_skip } },
4755 [P_RS_CANCEL] = { sizeof(struct p_block_ack), MDEV, { got_NegRSDReply } },
Philipp Reisnere4f78ed2011-03-16 11:27:48 +01004756 [P_CONN_ST_CHG_REPLY]={ sizeof(struct p_req_state_reply), CONN, {.conn_fn = got_conn_RqSReply}},
Philipp Reisnera4fbda82011-03-16 11:13:17 +01004757 [P_RETRY_WRITE] = { sizeof(struct p_block_ack), MDEV, { got_BlockAck } },
Andreas Gruenbacher7201b972011-03-14 18:23:00 +01004758};
4759
Philipp Reisnerb411b362009-09-25 16:07:19 -07004760int drbd_asender(struct drbd_thread *thi)
4761{
Philipp Reisner392c8802011-02-09 10:33:31 +01004762 struct drbd_tconn *tconn = thi->tconn;
Andreas Gruenbachere6ef8a52011-03-24 18:07:54 +01004763 struct p_header *h = tconn->meta.rbuf;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004764 struct asender_cmd *cmd = NULL;
Philipp Reisner77351055b2011-02-07 17:24:26 +01004765 struct packet_info pi;
Philipp Reisner257d0af2011-01-26 12:15:29 +01004766 int rv;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004767 void *buf = h;
4768 int received = 0;
Philipp Reisner257d0af2011-01-26 12:15:29 +01004769 int expect = sizeof(struct p_header);
Lars Ellenbergf36af182011-03-09 22:44:55 +01004770 int ping_timeout_active = 0;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004771
Philipp Reisnerb411b362009-09-25 16:07:19 -07004772 current->policy = SCHED_RR; /* Make this a realtime task! */
4773 current->rt_priority = 2; /* more important than all other tasks */
4774
Andreas Gruenbachere77a0a52011-01-25 15:43:39 +01004775 while (get_t_state(thi) == RUNNING) {
Philipp Reisner80822282011-02-08 12:46:30 +01004776 drbd_thread_current_set_cpu(thi);
Philipp Reisner32862ec2011-02-08 16:41:01 +01004777 if (test_and_clear_bit(SEND_PING, &tconn->flags)) {
Philipp Reisner2a67d8b2011-02-09 14:10:32 +01004778 if (!drbd_send_ping(tconn)) {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004779 conn_err(tconn, "drbd_send_ping has failed\n");
Andreas Gruenbacher841ce242010-12-15 19:31:20 +01004780 goto reconnect;
4781 }
Philipp Reisner32862ec2011-02-08 16:41:01 +01004782 tconn->meta.socket->sk->sk_rcvtimeo =
4783 tconn->net_conf->ping_timeo*HZ/10;
Lars Ellenbergf36af182011-03-09 22:44:55 +01004784 ping_timeout_active = 1;
Philipp Reisnerb411b362009-09-25 16:07:19 -07004785 }
4786
Philipp Reisner32862ec2011-02-08 16:41:01 +01004787 /* TODO: conditionally cork; it may hurt latency if we cork without
4788 much to send */
4789 if (!tconn->net_conf->no_cork)
4790 drbd_tcp_cork(tconn->meta.socket);
Philipp Reisner082a3432011-03-15 16:05:42 +01004791 if (tconn_process_done_ee(tconn)) {
4792 conn_err(tconn, "tconn_process_done_ee() failed\n");
Philipp Reisner32862ec2011-02-08 16:41:01 +01004793 goto reconnect;
Philipp Reisner082a3432011-03-15 16:05:42 +01004794 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004795 /* but unconditionally uncork unless disabled */
Philipp Reisner32862ec2011-02-08 16:41:01 +01004796 if (!tconn->net_conf->no_cork)
4797 drbd_tcp_uncork(tconn->meta.socket);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004798
4799 /* short circuit, recv_msg would return EINTR anyways. */
4800 if (signal_pending(current))
4801 continue;
4802
Philipp Reisner32862ec2011-02-08 16:41:01 +01004803 rv = drbd_recv_short(tconn->meta.socket, buf, expect-received, 0);
4804 clear_bit(SIGNAL_ASENDER, &tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004805
4806 flush_signals(current);
4807
4808 /* Note:
4809 * -EINTR (on meta) we got a signal
4810 * -EAGAIN (on meta) rcvtimeo expired
4811 * -ECONNRESET other side closed the connection
4812 * -ERESTARTSYS (on data) we got a signal
4813 * rv < 0 other than above: unexpected error!
4814 * rv == expected: full header or command
4815 * rv < expected: "woken" by signal during receive
4816 * rv == 0 : "connection shut down by peer"
4817 */
4818 if (likely(rv > 0)) {
4819 received += rv;
4820 buf += rv;
4821 } else if (rv == 0) {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004822 conn_err(tconn, "meta connection shut down by peer.\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004823 goto reconnect;
4824 } else if (rv == -EAGAIN) {
Lars Ellenbergcb6518c2011-06-20 14:44:45 +02004825 /* If the data socket received something meanwhile,
4826 * that is good enough: peer is still alive. */
Philipp Reisner32862ec2011-02-08 16:41:01 +01004827 if (time_after(tconn->last_received,
4828 jiffies - tconn->meta.socket->sk->sk_rcvtimeo))
Lars Ellenbergcb6518c2011-06-20 14:44:45 +02004829 continue;
Lars Ellenbergf36af182011-03-09 22:44:55 +01004830 if (ping_timeout_active) {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004831 conn_err(tconn, "PingAck did not arrive in time.\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004832 goto reconnect;
4833 }
Philipp Reisner32862ec2011-02-08 16:41:01 +01004834 set_bit(SEND_PING, &tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004835 continue;
4836 } else if (rv == -EINTR) {
4837 continue;
4838 } else {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004839 conn_err(tconn, "sock_recvmsg returned %d\n", rv);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004840 goto reconnect;
4841 }
4842
4843 if (received == expect && cmd == NULL) {
Andreas Gruenbacher8172f3e2011-03-16 17:22:39 +01004844 if (decode_header(tconn, h, &pi))
Philipp Reisnerb411b362009-09-25 16:07:19 -07004845 goto reconnect;
Andreas Gruenbacher7201b972011-03-14 18:23:00 +01004846 cmd = &asender_tbl[pi.cmd];
4847 if (pi.cmd >= ARRAY_SIZE(asender_tbl) || !cmd) {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004848 conn_err(tconn, "unknown command %d on meta (l: %d)\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004849 pi.cmd, pi.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004850 goto disconnect;
4851 }
4852 expect = cmd->pkt_size;
Philipp Reisner77351055b2011-02-07 17:24:26 +01004853 if (pi.size != expect - sizeof(struct p_header)) {
Philipp Reisner32862ec2011-02-08 16:41:01 +01004854 conn_err(tconn, "Wrong packet size on meta (c: %d, l: %d)\n",
Philipp Reisner77351055b2011-02-07 17:24:26 +01004855 pi.cmd, pi.size);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004856 goto reconnect;
Philipp Reisner257d0af2011-01-26 12:15:29 +01004857 }
Philipp Reisnerb411b362009-09-25 16:07:19 -07004858 }
4859 if (received == expect) {
Philipp Reisnera4fbda82011-03-16 11:13:17 +01004860 bool rv;
4861
4862 if (cmd->fa_type == CONN) {
4863 rv = cmd->conn_fn(tconn, pi.cmd);
4864 } else {
4865 struct drbd_conf *mdev = vnr_to_mdev(tconn, pi.vnr);
4866 rv = cmd->mdev_fn(mdev, pi.cmd);
4867 }
4868
4869 if (!rv)
Philipp Reisnerb411b362009-09-25 16:07:19 -07004870 goto reconnect;
4871
Philipp Reisnera4fbda82011-03-16 11:13:17 +01004872 tconn->last_received = jiffies;
4873
Lars Ellenbergf36af182011-03-09 22:44:55 +01004874 /* the idle_timeout (ping-int)
4875 * has been restored in got_PingAck() */
Andreas Gruenbacher7201b972011-03-14 18:23:00 +01004876 if (cmd == &asender_tbl[P_PING_ACK])
Lars Ellenbergf36af182011-03-09 22:44:55 +01004877 ping_timeout_active = 0;
4878
Philipp Reisnerb411b362009-09-25 16:07:19 -07004879 buf = h;
4880 received = 0;
Philipp Reisner257d0af2011-01-26 12:15:29 +01004881 expect = sizeof(struct p_header);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004882 cmd = NULL;
4883 }
4884 }
4885
4886 if (0) {
4887reconnect:
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004888 conn_request_state(tconn, NS(conn, C_NETWORK_FAILURE), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004889 }
4890 if (0) {
4891disconnect:
Philipp Reisnerbbeb6412011-02-10 13:45:46 +01004892 conn_request_state(tconn, NS(conn, C_DISCONNECTING), CS_HARD);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004893 }
Philipp Reisner32862ec2011-02-08 16:41:01 +01004894 clear_bit(SIGNAL_ASENDER, &tconn->flags);
Philipp Reisnerb411b362009-09-25 16:07:19 -07004895
Philipp Reisner32862ec2011-02-08 16:41:01 +01004896 conn_info(tconn, "asender terminated\n");
Philipp Reisnerb411b362009-09-25 16:07:19 -07004897
4898 return 0;
4899}