int uds_make_open_chapter_index(struct open_chapter_index **chapter_index, conststruct index_geometry *geometry, u64 volume_nonce)
{ int result;
size_t memory_size; struct open_chapter_index *index;
result = vdo_allocate(1, struct open_chapter_index, "open chapter index", &index); if (result != VDO_SUCCESS) return result;
/* * The delta index will rebalance delta lists when memory gets tight, * so give the chapter index one extra page.
*/
memory_size = ((geometry->index_pages_per_chapter + 1) * geometry->bytes_per_page);
index->geometry = geometry;
index->volume_nonce = volume_nonce;
result = uds_initialize_delta_index(&index->delta_index, 1,
geometry->delta_lists_per_chapter,
geometry->chapter_mean_delta,
geometry->chapter_payload_bits,
memory_size, 'm'); if (result != UDS_SUCCESS) {
vdo_free(index); return result;
}
/* Re-initialize an open chapter index for a new chapter. */ void uds_empty_open_chapter_index(struct open_chapter_index *chapter_index,
u64 virtual_chapter_number)
{
uds_reset_delta_index(&chapter_index->delta_index);
chapter_index->virtual_chapter_number = virtual_chapter_number;
}
/* Associate a record name with the record page containing its metadata. */ int uds_put_open_chapter_index_record(struct open_chapter_index *chapter_index, conststruct uds_record_name *name,
u32 page_number)
{ int result; struct delta_index_entry entry;
u32 address;
u32 list_number; const u8 *found_name; bool found; conststruct index_geometry *geometry = chapter_index->geometry;
u64 chapter_number = chapter_index->virtual_chapter_number;
u32 record_pages = geometry->record_pages_per_chapter;
result = VDO_ASSERT(page_number < record_pages, "Page number within chapter (%u) exceeds the maximum value %u",
page_number, record_pages); if (result != VDO_SUCCESS) return UDS_INVALID_ARGUMENT;
found = was_entry_found(&entry, address);
result = VDO_ASSERT(!(found && entry.is_collision), "Chunk appears more than once in chapter %llu",
(unsignedlonglong) chapter_number); if (result != VDO_SUCCESS) return UDS_BAD_STATE;
/* * Pack a section of an open chapter index into a chapter index page. A range of delta lists * (starting with a specified list index) is copied from the open chapter index into a memory page. * The number of lists copied onto the page is returned to the caller on success. * * @chapter_index: The open chapter index * @memory: The memory page to use * @first_list: The first delta list number to be copied * @last_page: If true, this is the last page of the chapter index and all the remaining lists must * be packed onto this page * @lists_packed: The number of delta lists that were packed onto this page
*/ int uds_pack_open_chapter_index_page(struct open_chapter_index *chapter_index,
u8 *memory, u32 first_list, bool last_page,
u32 *lists_packed)
{ int result; struct delta_index *delta_index = &chapter_index->delta_index; struct delta_index_stats stats;
u64 nonce = chapter_index->volume_nonce;
u64 chapter_number = chapter_index->virtual_chapter_number; conststruct index_geometry *geometry = chapter_index->geometry;
u32 list_count = geometry->delta_lists_per_chapter; unsignedint removals = 0; struct delta_index_entry entry;
u32 next_list;
s32 list_number;
for (;;) {
result = uds_pack_delta_index_page(delta_index, nonce, memory,
geometry->bytes_per_page,
chapter_number, first_list,
lists_packed); if (result != UDS_SUCCESS) return result;
if ((first_list + *lists_packed) == list_count) { /* All lists are packed. */ break;
} elseif (*lists_packed == 0) { /* * The next delta list does not fit on a page. This delta list will be * removed.
*/
} elseif (last_page) { /* * This is the last page and there are lists left unpacked, but all of the * remaining lists must fit on the page. Find a list that contains entries * and remove the entire list. Try the first list that does not fit. If it * is empty, we will select the last list that already fits and has any * entries.
*/
} else { /* This page is done. */ break;
}
if (removals == 0) {
uds_get_delta_index_stats(delta_index, &stats);
vdo_log_warning("The chapter index for chapter %llu contains %llu entries with %llu collisions",
(unsignedlonglong) chapter_number,
(unsignedlonglong) stats.record_count,
(unsignedlonglong) stats.collision_count);
}
list_number = *lists_packed; do { if (list_number < 0) return UDS_OVERFLOW;
next_list = first_list + list_number--;
result = uds_start_delta_index_search(delta_index, next_list, 0,
&entry); if (result != UDS_SUCCESS) return result;
result = uds_next_delta_index_entry(&entry); if (result != UDS_SUCCESS) return result;
} while (entry.at_end);
do {
result = uds_remove_delta_index_entry(&entry); if (result != UDS_SUCCESS) return result;
removals++;
} while (!entry.at_end);
}
if (removals > 0) {
vdo_log_warning("To avoid chapter index page overflow in chapter %llu, %u entries were removed from the chapter index",
(unsignedlonglong) chapter_number, removals);
}
return UDS_SUCCESS;
}
/* Make a new chapter index page, initializing it with the data from a given index_page buffer. */ int uds_initialize_chapter_index_page(struct delta_index_page *index_page, conststruct index_geometry *geometry,
u8 *page_buffer, u64 volume_nonce)
{ return uds_initialize_delta_index_page(index_page, volume_nonce,
geometry->chapter_mean_delta,
geometry->chapter_payload_bits,
page_buffer, geometry->bytes_per_page);
}
/* Validate a chapter index page read during rebuild. */ int uds_validate_chapter_index_page(conststruct delta_index_page *index_page, conststruct index_geometry *geometry)
{ int result; conststruct delta_index *delta_index = &index_page->delta_index;
u32 first = index_page->lowest_list_number;
u32 last = index_page->highest_list_number;
u32 list_number;
/* We walk every delta list from start to finish. */ for (list_number = first; list_number <= last; list_number++) { struct delta_index_entry entry;
result = uds_start_delta_index_search(delta_index, list_number - first,
0, &entry); if (result != UDS_SUCCESS) return result;
for (;;) {
result = uds_next_delta_index_entry(&entry); if (result != UDS_SUCCESS) { /* * A random bit stream is highly likely to arrive here when we go * past the end of the delta list.
*/ return result;
}
if (entry.at_end) break;
/* Also make sure that the record page field contains a plausible value. */ if (uds_get_delta_entry_value(&entry) >=
geometry->record_pages_per_chapter) { /* * Do not log this as an error. It happens in normal operation when * we are doing a rebuild but haven't written the entire volume * once.
*/ return UDS_CORRUPT_DATA;
}
}
} return UDS_SUCCESS;
}
/* * Search a chapter index page for a record name, returning the record page number that may contain * the name.
*/ int uds_search_chapter_index_page(struct delta_index_page *index_page, conststruct index_geometry *geometry, conststruct uds_record_name *name,
u16 *record_page_ptr)
{ int result; struct delta_index *delta_index = &index_page->delta_index;
u32 address = uds_hash_to_chapter_delta_address(name, geometry);
u32 delta_list_number = uds_hash_to_chapter_delta_list(name, geometry);
u32 sub_list_number = delta_list_number - index_page->lowest_list_number; struct delta_index_entry entry;
result = uds_get_delta_index_entry(delta_index, sub_list_number, address,
name->name, &entry); if (result != UDS_SUCCESS) return result;
if (was_entry_found(&entry, address))
*record_page_ptr = uds_get_delta_entry_value(&entry); else
*record_page_ptr = NO_CHAPTER_INDEX_ENTRY;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.