mirror of
git://sourceware.org/git/lvm2.git
synced 2025-10-25 03:33:16 +03:00
The new label_scan() function reads a large buffer of data from the start of the disk, and saves it so that multiple structs can be read from it. Previously, only the label_header was read from this buffer, and the code which needed data structures that immediately followed the label_header would read those from disk separately. This created a large number of small, unnecessary disk reads. In each place that the two read paths (label_scan and vg_read) need to read data from disk, first check if that data is already available from the label_read_data buffer, and if so just copy it from the buffer instead of reading from disk. Code changes ------------ - passing the label_read_data struct down through both read paths to make it available. - before every disk read, first check if the location and size of the desired piece of data exists fully in the label_read_data buffer, and if so copy it from there. Otherwise, use the existing code to read the data from disk. - adding some log_error messages on existing error paths that were already being updated for the reasons above. - using similar naming for parallel functions on the two parallel read paths that are being updated above. label_scan path calls: read_metadata_location_summary, text_read_metadata_summary vg_read path calls: read_metadata_location_vg, text_read_metadata_file Previously, those functions were named: label_scan path calls: vgname_from_mda, text_vgsummary_import vg_read path calls: _find_vg_rlocn, text_vg_import_fd I/O changes ----------- In the label_scan path, the following data is either copied from label_read_data or read from disk for each PV: - label_header and pv_header - mda_header (in _raw_read_mda_header) - vg metadata name (in read_metadata_location_summary) - vg metadata (in config_file_read_fd) Total of 4 reads per PV in the label_scan path. In the vg_read path, the following data is either copied from label_read_data or read from disk for each PV: - mda_header (in _raw_read_mda_header) - vg metadata name (in read_metadata_location_vg) - vg metadata (in config_file_read_fd) Total of 3 reads per PV in the vg_read path. For a common read/reporting command, each PV will be: - read by the command's initial lvmcache_label_scan() - read by lvmcache_label_rescan_vg() at the start of vg_read() - read by vg_read() Previously, this would cause 11 synchronous disk reads per PV: 4 from lvmcache_label_scan(), 4 from lvmcache_label_rescan_vg() and 3 from vg_read(). With this commit's optimization, there are now 2 async disk reads per PV: 1 from lvmcache_label_scan() and 1 from lvmcache_label_rescan_vg(). When a second mda is used on a PV, it is located at the end of the PV. This second mda and copy of metadata will not be found in the label_read_data buffer, and will always require separate disk reads.
113 lines
3.3 KiB
C
113 lines
3.3 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2006 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#ifndef _LVM_TEXT_LAYOUT_H
|
|
#define _LVM_TEXT_LAYOUT_H
|
|
|
|
#include "config.h"
|
|
#include "metadata.h"
|
|
#include "lvmcache.h"
|
|
#include "uuid.h"
|
|
|
|
/* disk_locn and data_area_list are defined in format-text.h */
|
|
|
|
/*
|
|
* PV header extension versions:
|
|
* - version 1: bootloader area support
|
|
* - version 2: PV_EXT_USED flag support
|
|
*/
|
|
#define PV_HEADER_EXTENSION_VSN 2
|
|
|
|
struct pv_header_extension {
|
|
uint32_t version;
|
|
uint32_t flags;
|
|
/* NULL-terminated list of bootloader areas */
|
|
struct disk_locn bootloader_areas_xl[0];
|
|
} __attribute__ ((packed));
|
|
|
|
/* Fields with the suffix _xl should be xlate'd wherever they appear */
|
|
/* On disk */
|
|
struct pv_header {
|
|
int8_t pv_uuid[ID_LEN];
|
|
|
|
/* This size can be overridden if PV belongs to a VG */
|
|
uint64_t device_size_xl; /* Bytes */
|
|
|
|
/* NULL-terminated list of data areas followed by */
|
|
/* NULL-terminated list of metadata area headers */
|
|
struct disk_locn disk_areas_xl[0]; /* Two lists */
|
|
} __attribute__ ((packed));
|
|
|
|
/*
|
|
* Ignore this raw location. This allows us to
|
|
* ignored metadata areas easily, and thus balance
|
|
* metadata across VGs with many PVs.
|
|
*/
|
|
#define RAW_LOCN_IGNORED 0x00000001
|
|
|
|
/* On disk */
|
|
struct raw_locn {
|
|
uint64_t offset; /* Offset in bytes to start sector */
|
|
uint64_t size; /* Bytes */
|
|
uint32_t checksum;
|
|
uint32_t flags;
|
|
} __attribute__ ((packed));
|
|
|
|
int rlocn_is_ignored(const struct raw_locn *rlocn);
|
|
void rlocn_set_ignored(struct raw_locn *rlocn, unsigned mda_ignored);
|
|
|
|
/* On disk */
|
|
/* Structure size limited to one sector */
|
|
struct mda_header {
|
|
uint32_t checksum_xl; /* Checksum of rest of mda_header */
|
|
int8_t magic[16]; /* To aid scans for metadata */
|
|
uint32_t version;
|
|
uint64_t start; /* Absolute start byte of mda_header */
|
|
uint64_t size; /* Size of metadata area */
|
|
|
|
struct raw_locn raw_locns[0]; /* NULL-terminated list */
|
|
} __attribute__ ((packed));
|
|
|
|
struct mda_header *raw_read_mda_header(const struct format_type *fmt,
|
|
struct device_area *dev_area,
|
|
struct label_read_data *ld);
|
|
|
|
struct mda_lists {
|
|
struct dm_list dirs;
|
|
struct dm_list raws;
|
|
struct metadata_area_ops *file_ops;
|
|
struct metadata_area_ops *raw_ops;
|
|
};
|
|
|
|
struct mda_context {
|
|
struct device_area area;
|
|
uint64_t free_sectors;
|
|
struct raw_locn rlocn; /* Store inbetween write and commit */
|
|
};
|
|
|
|
/* FIXME Convert this at runtime */
|
|
#define FMTT_MAGIC "\040\114\126\115\062\040\170\133\065\101\045\162\060\116\052\076"
|
|
#define FMTT_VERSION 1
|
|
#define MDA_HEADER_SIZE 512
|
|
#define LVM2_LABEL "LVM2 001"
|
|
#define MDA_SIZE_MIN (8 * (unsigned) lvm_getpagesize())
|
|
|
|
int read_metadata_location_summary(const struct format_type *fmt, struct mda_header *mdah,
|
|
struct label_read_data *ld,
|
|
struct device_area *dev_area, struct lvmcache_vgsummary *vgsummary,
|
|
uint64_t *mda_free_sectors);
|
|
|
|
#endif
|