1
0
mirror of git://sourceware.org/git/lvm2.git synced 2024-12-21 13:34:40 +03:00

Fix several incorrect comparisons in parallel area avoidance code.

Fix segment lengths when flattening existing parallel areas.
Log existing parallel areas prior to allocation.
Fix mirror log creation when activation disabled.
This commit is contained in:
Alasdair Kergon 2006-09-11 14:24:58 +00:00
parent fe748f0092
commit d9c6bbab80
3 changed files with 71 additions and 14 deletions

View File

@ -1,5 +1,9 @@
Version 2.02.10 - Version 2.02.10 -
================================== ==================================
Fix several incorrect comparisons in parallel area avoidance code.
Fix segment lengths when flattening existing parallel areas.
Log existing parallel areas prior to allocation.
Fix mirror log creation when activation disabled.
Don't attempt automatic recovery without proper locking. Don't attempt automatic recovery without proper locking.
When using local file locking, skip clustered VGs. When using local file locking, skip clustered VGs.
Add fallback_to_clustered_locking and fallback_to_local_locking parameters. Add fallback_to_clustered_locking and fallback_to_local_locking parameters.

View File

@ -490,6 +490,49 @@ void alloc_destroy(struct alloc_handle *ah)
dm_pool_destroy(ah->mem); dm_pool_destroy(ah->mem);
} }
static int _log_parallel_areas(struct dm_pool *mem, struct list *parallel_areas)
{
struct seg_pvs *spvs;
struct pv_list *pvl;
char *pvnames;
if (!parallel_areas)
return 1;
if (!dm_pool_begin_object(mem, 256)) {
log_error("dm_pool_begin_object failed");
return 0;
}
list_iterate_items(spvs, parallel_areas) {
list_iterate_items(pvl, &spvs->pvs) {
if (!dm_pool_grow_object(mem, dev_name(pvl->pv->dev), strlen(dev_name(pvl->pv->dev)))) {
log_error("dm_pool_grow_object failed");
dm_pool_abandon_object(mem);
return 0;
}
if (!dm_pool_grow_object(mem, " ", 1)) {
log_error("dm_pool_grow_object failed");
dm_pool_abandon_object(mem);
return 0;
}
}
if (!dm_pool_grow_object(mem, "\0", 1)) {
log_error("dm_pool_grow_object failed");
dm_pool_abandon_object(mem);
return 0;
}
pvnames = dm_pool_end_object(mem);
log_debug("Parallel PVs at LE %" PRIu32 " length %" PRIu32 ": %s",
spvs->le, spvs->len, pvnames);
dm_pool_free(mem, pvnames);
}
return 1;
}
static int _setup_alloced_segment(struct logical_volume *lv, uint32_t status, static int _setup_alloced_segment(struct logical_volume *lv, uint32_t status,
uint32_t area_count, uint32_t area_count,
uint32_t stripe_size, uint32_t stripe_size,
@ -711,16 +754,17 @@ static int _find_parallel_space(struct alloc_handle *ah, alloc_policy_t alloc,
* the maximum we can allocate in one go accordingly. * the maximum we can allocate in one go accordingly.
*/ */
if (ah->parallel_areas) { if (ah->parallel_areas) {
next_le = (prev_lvseg ? prev_lvseg->le + prev_lvseg->len : 0) + *allocated / ah->area_multiple;
list_iterate_items(spvs, ah->parallel_areas) { list_iterate_items(spvs, ah->parallel_areas) {
next_le = (prev_lvseg ? prev_lvseg->le + prev_lvseg->len : 0) + *allocated; if (next_le >= spvs->le + spvs->len)
if (next_le >= spvs->le) { continue;
if (next_le + max_parallel > spvs->le + spvs->len)
max_parallel = (spvs->le + spvs->len - next_le) * ah->area_multiple; if (max_parallel > (spvs->le + spvs->len) * ah->area_multiple)
max_parallel = (spvs->le + spvs->len) * ah->area_multiple;
parallel_pvs = &spvs->pvs; parallel_pvs = &spvs->pvs;
break; break;
} }
} }
}
/* /*
* Put the smallest area of each PV that is at least the * Put the smallest area of each PV that is at least the
@ -760,7 +804,8 @@ static int _find_parallel_space(struct alloc_handle *ah, alloc_policy_t alloc,
} }
/* Is it big enough on its own? */ /* Is it big enough on its own? */
if ((pva->count < max_parallel - *allocated) && if (pva->count * ah->area_multiple <
max_parallel - *allocated &&
((!can_split && !ah->log_count) || ((!can_split && !ah->log_count) ||
(already_found_one && (already_found_one &&
!(alloc == ALLOC_ANYWHERE)))) !(alloc == ALLOC_ANYWHERE))))
@ -853,6 +898,9 @@ static int _allocate(struct alloc_handle *ah,
return 0; return 0;
} }
if (!_log_parallel_areas(ah->mem, ah->parallel_areas))
stack;
areas_size = list_size(pvms); areas_size = list_size(pvms);
if (areas_size < ah->area_count + ah->log_count) { if (areas_size < ah->area_count + ah->log_count) {
if (ah->alloc != ALLOC_ANYWHERE) { if (ah->alloc != ALLOC_ANYWHERE) {
@ -1345,7 +1393,7 @@ static int _for_each_pv(struct cmd_context *cmd, struct logical_volume *lv,
/* Remaining logical length of segment */ /* Remaining logical length of segment */
remaining_seg_len = seg->len - (le - seg->le); remaining_seg_len = seg->len - (le - seg->le);
if (len > remaining_seg_len) if (remaining_seg_len > len)
remaining_seg_len = len; remaining_seg_len = len;
if (spvs->len > remaining_seg_len) if (spvs->len > remaining_seg_len)
@ -1426,7 +1474,7 @@ struct list *build_parallel_areas_from_lv(struct cmd_context *cmd,
/* Find next segment end */ /* Find next segment end */
/* FIXME Unnecessary nesting! */ /* FIXME Unnecessary nesting! */
if (!_for_each_pv(cmd, lv, current_le, lv->le_count, _add_pvs, spvs)) { if (!_for_each_pv(cmd, lv, current_le, lv->le_count - current_le, _add_pvs, spvs)) {
stack; stack;
return NULL; return NULL;
} }

View File

@ -1204,7 +1204,6 @@ int set_lv(struct cmd_context *cmd, struct logical_volume *lv, int value)
return 1; return 1;
} }
/* /*
* This function writes a new header to the mirror log header to the lv * This function writes a new header to the mirror log header to the lv
* *
@ -1301,6 +1300,12 @@ struct logical_volume *create_mirror_log(struct cmd_context *cmd,
goto error; goto error;
} }
if (!activation() && in_sync) {
log_error("Aborting. Unable to create in-sync mirror log "
"while activation is disabled.");
goto error;
}
if (!activate_lv(cmd, log_lv)) { if (!activate_lv(cmd, log_lv)) {
log_error("Aborting. Failed to activate mirror log. " log_error("Aborting. Failed to activate mirror log. "
"Remove new LVs and retry."); "Remove new LVs and retry.");
@ -1313,7 +1318,7 @@ struct logical_volume *create_mirror_log(struct cmd_context *cmd,
goto error; goto error;
} }
if (!_write_log_header(cmd, log_lv)) { if (activation() && !_write_log_header(cmd, log_lv)) {
log_error("Aborting. Failed to write mirror log header. " log_error("Aborting. Failed to write mirror log header. "
"Remove new LV and retry."); "Remove new LV and retry.");
goto error; goto error;