Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit b5bc8e08 authored by Lars Ellenberg's avatar Lars Ellenberg Committed by Jens Axboe
Browse files

drbd: split drbd_al_begin_io into fastpath, prepare, and commit

parent 113fef9e
Loading
Loading
Loading
Loading
+71 −33
Original line number Original line Diff line number Diff line
@@ -104,7 +104,6 @@ struct update_al_work {
	int err;
	int err;
};
};


static int al_write_transaction(struct drbd_conf *mdev, bool delegate);


void *drbd_md_get_buffer(struct drbd_conf *mdev)
void *drbd_md_get_buffer(struct drbd_conf *mdev)
{
{
@@ -246,30 +245,37 @@ static struct lc_element *_al_get(struct drbd_conf *mdev, unsigned int enr)
	return al_ext;
	return al_ext;
}
}


/*
bool drbd_al_begin_io_fastpath(struct drbd_conf *mdev, struct drbd_interval *i)
 * @delegate:   delegate activity log I/O to the worker thread
 */
void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool delegate)
{
{
	/* for bios crossing activity log extent boundaries,
	/* for bios crossing activity log extent boundaries,
	 * we may need to activate two extents in one go */
	 * we may need to activate two extents in one go */
	unsigned first = i->sector >> (AL_EXTENT_SHIFT-9);
	unsigned first = i->sector >> (AL_EXTENT_SHIFT-9);
	unsigned last = i->size == 0 ? first : (i->sector + (i->size >> 9) - 1) >> (AL_EXTENT_SHIFT-9);
	unsigned last = i->size == 0 ? first : (i->sector + (i->size >> 9) - 1) >> (AL_EXTENT_SHIFT-9);
	unsigned enr;
	bool fastpath_ok = true;
	bool need_transaction = false;
	bool locked = false;


	/* When called through generic_make_request(), we must delegate
	D_ASSERT((unsigned)(last - first) <= 1);
	 * activity log I/O to the worker thread: a further request
	D_ASSERT(atomic_read(&mdev->local_cnt) > 0);
	 * submitted via generic_make_request() within the same task
	 * would be queued on current->bio_list, and would only start
	 * after this function returns (see generic_make_request()).
	 *
	 * However, if we *are* the worker, we must not delegate to ourselves.
	 */


	if (delegate)
	/* FIXME figure out a fast path for bios crossing AL extent boundaries */
		BUG_ON(current == mdev->tconn->worker.task);
	if (first != last)
		return false;

	spin_lock_irq(&mdev->al_lock);
	fastpath_ok =
		lc_find(mdev->resync, first/AL_EXT_PER_BM_SECT) == NULL &&
		lc_try_get(mdev->act_log, first) != NULL;
	spin_unlock_irq(&mdev->al_lock);
	return fastpath_ok;
}

bool drbd_al_begin_io_prepare(struct drbd_conf *mdev, struct drbd_interval *i)
{
	/* for bios crossing activity log extent boundaries,
	 * we may need to activate two extents in one go */
	unsigned first = i->sector >> (AL_EXTENT_SHIFT-9);
	unsigned last = i->size == 0 ? first : (i->sector + (i->size >> 9) - 1) >> (AL_EXTENT_SHIFT-9);
	unsigned enr;
	bool need_transaction = false;


	D_ASSERT(first <= last);
	D_ASSERT(first <= last);
	D_ASSERT(atomic_read(&mdev->local_cnt) > 0);
	D_ASSERT(atomic_read(&mdev->local_cnt) > 0);
@@ -280,11 +286,28 @@ void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool dele
		if (al_ext->lc_number != enr)
		if (al_ext->lc_number != enr)
			need_transaction = true;
			need_transaction = true;
	}
	}
	return need_transaction;
}


	/* If *this* request was to an already active extent,
static int al_write_transaction(struct drbd_conf *mdev, bool delegate);
	 * we're done, even if there are pending changes. */

	if (!need_transaction)
/* When called through generic_make_request(), we must delegate
		return;
 * activity log I/O to the worker thread: a further request
 * submitted via generic_make_request() within the same task
 * would be queued on current->bio_list, and would only start
 * after this function returns (see generic_make_request()).
 *
 * However, if we *are* the worker, we must not delegate to ourselves.
 */

/*
 * @delegate:   delegate activity log I/O to the worker thread
 */
void drbd_al_begin_io_commit(struct drbd_conf *mdev, bool delegate)
{
	bool locked = false;

	BUG_ON(delegate && current == mdev->tconn->worker.task);


	/* Serialize multiple transactions.
	/* Serialize multiple transactions.
	 * This uses test_and_set_bit, memory barrier is implicit.
	 * This uses test_and_set_bit, memory barrier is implicit.
@@ -303,11 +326,8 @@ void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool dele
			write_al_updates = rcu_dereference(mdev->ldev->disk_conf)->al_updates;
			write_al_updates = rcu_dereference(mdev->ldev->disk_conf)->al_updates;
			rcu_read_unlock();
			rcu_read_unlock();


			if (write_al_updates) {
			if (write_al_updates)
				al_write_transaction(mdev, delegate);
				al_write_transaction(mdev, delegate);
				mdev->al_writ_cnt++;
			}

			spin_lock_irq(&mdev->al_lock);
			spin_lock_irq(&mdev->al_lock);
			/* FIXME
			/* FIXME
			if (err)
			if (err)
@@ -321,6 +341,17 @@ void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool dele
	}
	}
}
}


/*
 * @delegate:   delegate activity log I/O to the worker thread
 */
void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool delegate)
{
	BUG_ON(delegate && current == mdev->tconn->worker.task);

	if (drbd_al_begin_io_prepare(mdev, i))
		drbd_al_begin_io_commit(mdev, delegate);
}

void drbd_al_complete_io(struct drbd_conf *mdev, struct drbd_interval *i)
void drbd_al_complete_io(struct drbd_conf *mdev, struct drbd_interval *i)
{
{
	/* for bios crossing activity log extent boundaries,
	/* for bios crossing activity log extent boundaries,
@@ -478,15 +509,22 @@ _al_write_transaction(struct drbd_conf *mdev)
	crc = crc32c(0, buffer, 4096);
	crc = crc32c(0, buffer, 4096);
	buffer->crc32c = cpu_to_be32(crc);
	buffer->crc32c = cpu_to_be32(crc);


	/* normal execution path goes through all three branches */
	if (drbd_bm_write_hinted(mdev))
	if (drbd_bm_write_hinted(mdev))
		err = -EIO;
		err = -EIO;
		/* drbd_chk_io_error done already */
	else {
	else if (drbd_md_sync_page_io(mdev, mdev->ldev, sector, WRITE)) {
		bool write_al_updates;
		rcu_read_lock();
		write_al_updates = rcu_dereference(mdev->ldev->disk_conf)->al_updates;
		rcu_read_unlock();
		if (write_al_updates) {
			if (drbd_md_sync_page_io(mdev, mdev->ldev, sector, WRITE)) {
				err = -EIO;
				err = -EIO;
				drbd_chk_io_error(mdev, 1, DRBD_META_IO_ERROR);
				drbd_chk_io_error(mdev, 1, DRBD_META_IO_ERROR);
			} else {
			} else {
				mdev->al_tr_number++;
				mdev->al_tr_number++;
				mdev->al_writ_cnt++;
			}
		}
	}
	}


	drbd_md_put_buffer(mdev);
	drbd_md_put_buffer(mdev);
+1 −0
Original line number Original line Diff line number Diff line
@@ -1611,6 +1611,7 @@ extern const char *drbd_conn_str(enum drbd_conns s);
extern const char *drbd_role_str(enum drbd_role s);
extern const char *drbd_role_str(enum drbd_role s);


/* drbd_actlog.c */
/* drbd_actlog.c */
extern bool drbd_al_begin_io_fastpath(struct drbd_conf *mdev, struct drbd_interval *i);
extern void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool delegate);
extern void drbd_al_begin_io(struct drbd_conf *mdev, struct drbd_interval *i, bool delegate);
extern void drbd_al_complete_io(struct drbd_conf *mdev, struct drbd_interval *i);
extern void drbd_al_complete_io(struct drbd_conf *mdev, struct drbd_interval *i);
extern void drbd_rs_complete_io(struct drbd_conf *mdev, sector_t sector);
extern void drbd_rs_complete_io(struct drbd_conf *mdev, sector_t sector);