1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-03 05:18:29 +03:00

Detect partition table signature.

This commit is contained in:
Alasdair Kergon 2004-11-23 11:44:04 +00:00
parent 82416639c4
commit 985ef4617f
5 changed files with 92 additions and 50 deletions

View File

@ -1,5 +1,6 @@
Version 2.00.26 - Version 2.00.26 -
===================================== =====================================
Detect partition table signature.
pvcreate wipes md superblocks. (With --uuid or --restorefile it prompts.) pvcreate wipes md superblocks. (With --uuid or --restorefile it prompts.)
Separate out md superblock detection code. Separate out md superblock detection code.
Prevent snapshot origin resizing. Prevent snapshot origin resizing.

View File

@ -13,6 +13,54 @@
* Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/ */
#include "lib.h"
#include "lvm-types.h"
#include "device.h"
#include "metadata.h"
#include "filter.h"
#define PART_MAGIC 0xAA55
static int _is_whole_disk(struct device *dev)
{
int parts = max_partitions(MINOR(dev->dev));
if (!parts || !(MINOR(dev->dev) % parts))
return 1;
return 0;
}
static int _has_partition_table(struct device *dev)
{
int ret = 0;
uint32_t part_magic;
uint64_t part_offset;
if (!dev_open(dev)) {
stack;
return -1;
}
part_offset = sizeof(unsigned short) * 255;
if (dev_read(dev, part_offset, sizeof(part_magic), &part_magic) &&
(part_magic == PART_MAGIC))
ret = 1;
if (!dev_close(dev))
stack;
return ret;
}
int is_partitioned_dev(struct device *dev)
{
if (_is_whole_disk(dev))
return 0;
return _has_partition_table(dev);
}
#if 0 #if 0
#include <sys/stat.h> #include <sys/stat.h>
#include <sys/mman.h> #include <sys/mman.h>
@ -27,24 +75,13 @@
#include <linux/major.h> #include <linux/major.h>
#include <linux/genhd.h> #include <linux/genhd.h>
#include "dbg_malloc.h"
#include "log.h"
#include "dev-cache.h"
#include "metadata.h"
#include "device.h"
int _get_partition_type(struct dev_filter *filter, struct device *d); int _get_partition_type(struct dev_filter *filter, struct device *d);
#define MINOR_PART(dm, d) (MINOR((d)->dev) % dev_max_partitions(dm, (d)->dev)) #define MINOR_PART(dev) (MINOR((dev)->dev) % max_partitions(MINOR((dev)->dev)))
int is_whole_disk(struct dev_filter *filter, struct device *d) int is_extended_partition(struct device *d)
{ {
return (MINOR_PART(dm, d)) ? 0 : 1; return (MINOR_PART(d) > 4) ? 1 : 0;
}
int is_extended_partition(struct dev_mgr *dm, struct device *d)
{
return (MINOR_PART(dm, d) > 4) ? 1 : 0;
} }
struct device *dev_primary(struct dev_mgr *dm, struct device *d) struct device *dev_primary(struct dev_mgr *dm, struct device *d)

View File

@ -95,12 +95,8 @@ int dev_is_md(struct device *dev, uint64_t *sb);
/* FIXME Check partition type if appropriate */ /* FIXME Check partition type if appropriate */
#define is_lvm_partition(a) 1 #define is_lvm_partition(a) 1
/* int is_lvm_partition(const char *name); */
/* int is_partitioned_dev(struct device *dev);
static inline int is_lvm_partition(const char *name)
{
return 1;
}
*/
#endif #endif

View File

@ -27,6 +27,9 @@
#define NUMBER_OF_MAJORS 4096 #define NUMBER_OF_MAJORS 4096
/* FIXME Make this sparse */
static int _max_partitions_by_major[NUMBER_OF_MAJORS];
typedef struct { typedef struct {
const char *name; const char *name;
const int max_partitions; const int max_partitions;
@ -62,29 +65,34 @@ static const device_info_t device_info[] = {
static int _passes_lvm_type_device_filter(struct dev_filter *f, static int _passes_lvm_type_device_filter(struct dev_filter *f,
struct device *dev) struct device *dev)
{ {
int fd;
const char *name = dev_name(dev); const char *name = dev_name(dev);
int ret = 1;
/* Is this a recognised device type? */ /* Is this a recognised device type? */
if (!(((int *) f->private)[MAJOR(dev->dev)])) { if (!_max_partitions_by_major[MAJOR(dev->dev)]) {
log_debug("%s: Skipping: Unrecognised LVM device type %" log_debug("%s: Skipping: Unrecognised LVM device type %"
PRIu64, name, (uint64_t) MAJOR(dev->dev)); PRIu64, name, (uint64_t) MAJOR(dev->dev));
return 0; return 0;
} }
/* Check it's accessible */ /* Check it's accessible */
if ((fd = open(name, O_RDONLY)) < 0) { if (!dev_open_flags(dev, O_RDONLY, 0, 0)) {
log_debug("%s: Skipping: open failed: %s", name, log_debug("%s: Skipping: open failed", name);
strerror(errno));
return 0; return 0;
} }
close(fd); if (is_partitioned_dev(dev)) {
log_debug("%s: Skipping: partition table signature found",
name);
ret = 0;
}
return 1; dev_close(dev);
return ret;
} }
static int *_scan_proc_dev(const char *proc, const struct config_node *cn) static int _scan_proc_dev(const char *proc, const struct config_node *cn)
{ {
char line[80]; char line[80];
char proc_devices[PATH_MAX]; char proc_devices[PATH_MAX];
@ -94,36 +102,29 @@ static int *_scan_proc_dev(const char *proc, const struct config_node *cn)
int blocksection = 0; int blocksection = 0;
size_t dev_len = 0; size_t dev_len = 0;
struct config_value *cv; struct config_value *cv;
int *max_partitions_by_major;
char *name; char *name;
/* FIXME Make this sparse */
if (!(max_partitions_by_major =
dbg_malloc(sizeof(int) * NUMBER_OF_MAJORS))) {
log_error("Filter failed to allocate max_partitions_by_major");
return NULL;
}
if (!*proc) { if (!*proc) {
log_verbose("No proc filesystem found: using all block device " log_verbose("No proc filesystem found: using all block device "
"types"); "types");
for (i = 0; i < NUMBER_OF_MAJORS; i++) for (i = 0; i < NUMBER_OF_MAJORS; i++)
max_partitions_by_major[i] = 1; _max_partitions_by_major[i] = 1;
return max_partitions_by_major; return 1;
} }
memset(_max_partitions_by_major, 0, sizeof(int) * NUMBER_OF_MAJORS);
if (lvm_snprintf(proc_devices, sizeof(proc_devices), if (lvm_snprintf(proc_devices, sizeof(proc_devices),
"%s/devices", proc) < 0) { "%s/devices", proc) < 0) {
log_error("Failed to create /proc/devices string"); log_error("Failed to create /proc/devices string");
return NULL; return 0;
} }
if (!(pd = fopen(proc_devices, "r"))) { if (!(pd = fopen(proc_devices, "r"))) {
log_sys_error("fopen", proc_devices); log_sys_error("fopen", proc_devices);
return NULL; return 0;
} }
memset(max_partitions_by_major, 0, sizeof(int) * NUMBER_OF_MAJORS);
while (fgets(line, 80, pd) != NULL) { while (fgets(line, 80, pd) != NULL) {
i = 0; i = 0;
while (line[i] == ' ' && line[i] != '\0') while (line[i] == ' ' && line[i] != '\0')
@ -158,13 +159,13 @@ static int *_scan_proc_dev(const char *proc, const struct config_node *cn)
if (dev_len <= strlen(line + i) && if (dev_len <= strlen(line + i) &&
!strncmp(device_info[j].name, line + i, dev_len) && !strncmp(device_info[j].name, line + i, dev_len) &&
(line_maj < NUMBER_OF_MAJORS)) { (line_maj < NUMBER_OF_MAJORS)) {
max_partitions_by_major[line_maj] = _max_partitions_by_major[line_maj] =
device_info[j].max_partitions; device_info[j].max_partitions;
break; break;
} }
} }
if (max_partitions_by_major[line_maj] || !cn) if (_max_partitions_by_major[line_maj] || !cn)
continue; continue;
/* Check devices/types for local variations */ /* Check devices/types for local variations */
@ -172,7 +173,7 @@ static int *_scan_proc_dev(const char *proc, const struct config_node *cn)
if (cv->type != CFG_STRING) { if (cv->type != CFG_STRING) {
log_error("Expecting string in devices/types " log_error("Expecting string in devices/types "
"in config file"); "in config file");
return NULL; return 0;
} }
dev_len = strlen(cv->v.str); dev_len = strlen(cv->v.str);
name = cv->v.str; name = cv->v.str;
@ -181,24 +182,29 @@ static int *_scan_proc_dev(const char *proc, const struct config_node *cn)
log_error("Max partition count missing for %s " log_error("Max partition count missing for %s "
"in devices/types in config file", "in devices/types in config file",
name); name);
return NULL; return 0;
} }
if (!cv->v.i) { if (!cv->v.i) {
log_error("Zero partition count invalid for " log_error("Zero partition count invalid for "
"%s in devices/types in config file", "%s in devices/types in config file",
name); name);
return NULL; return 0;
} }
if (dev_len <= strlen(line + i) && if (dev_len <= strlen(line + i) &&
!strncmp(name, line + i, dev_len) && !strncmp(name, line + i, dev_len) &&
(line_maj < NUMBER_OF_MAJORS)) { (line_maj < NUMBER_OF_MAJORS)) {
max_partitions_by_major[line_maj] = cv->v.i; _max_partitions_by_major[line_maj] = cv->v.i;
break; break;
} }
} }
} }
fclose(pd); fclose(pd);
return max_partitions_by_major; return 1;
}
int max_partitions(int major)
{
return _max_partitions_by_major[major];
} }
struct dev_filter *lvm_type_filter_create(const char *proc, struct dev_filter *lvm_type_filter_create(const char *proc,
@ -213,8 +219,9 @@ struct dev_filter *lvm_type_filter_create(const char *proc,
f->passes_filter = _passes_lvm_type_device_filter; f->passes_filter = _passes_lvm_type_device_filter;
f->destroy = lvm_type_filter_destroy; f->destroy = lvm_type_filter_destroy;
f->private = NULL;
if (!(f->private = _scan_proc_dev(proc, cn))) { if (!_scan_proc_dev(proc, cn)) {
stack; stack;
return NULL; return NULL;
} }
@ -224,7 +231,6 @@ struct dev_filter *lvm_type_filter_create(const char *proc,
void lvm_type_filter_destroy(struct dev_filter *f) void lvm_type_filter_destroy(struct dev_filter *f)
{ {
dbg_free(f->private);
dbg_free(f); dbg_free(f);
return; return;
} }

View File

@ -37,4 +37,6 @@ void lvm_type_filter_destroy(struct dev_filter *f);
int md_major(void); int md_major(void);
int max_partitions(int major);
#endif #endif