mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-30 17:18:21 +03:00
b849de1d18
This should be a rare occurrence so the aim is to recover if it's straightforward to do so, otherwise just to abort the operation. If people *knowingly* change device names, they should always run vgscan afterwards. A few bytes of memory gets leaked inside a pool each time an alias has to be discarded - it's not worth restructuring the code to reuse it. More of LVM2 needs updating to pass device objects (or uuids) about instead of pathnames so that resolution of pathname->object only happens once per operation. dev_cache_get() should now always return the *current* device at the path given dev_name_confirmed() replaces dev_name() whenever it's important to know that name for the device is still current (ie when opening it). If the cache doesn't know a current name, the function fails. dev_open() guarantees that the file descriptor returned is for the dev_t of the device structure it was passed.
236 lines
4.0 KiB
C
236 lines
4.0 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software
|
|
*
|
|
* This file is released under the LGPL.
|
|
*/
|
|
|
|
#include "device.h"
|
|
#include "lvm-types.h"
|
|
#include "log.h"
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <sys/ioctl.h>
|
|
#include <linux/fs.h> // UGH!!! for BLKSSZGET
|
|
|
|
int dev_get_size(struct device *dev, uint64_t *size)
|
|
{
|
|
int fd;
|
|
long s;
|
|
const char *name = dev_name(dev);
|
|
|
|
log_very_verbose("Getting size of %s", name);
|
|
if ((fd = open(name, O_RDONLY)) < 0) {
|
|
log_sys_error("open", name);
|
|
return 0;
|
|
}
|
|
|
|
/* FIXME: add 64 bit ioctl */
|
|
if (ioctl(fd, BLKGETSIZE, &s) < 0) {
|
|
log_sys_error("ioctl BLKGETSIZE", name);
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
close(fd);
|
|
*size = (uint64_t) s;
|
|
return 1;
|
|
}
|
|
|
|
int dev_get_sectsize(struct device *dev, uint32_t *size)
|
|
{
|
|
int fd;
|
|
int s;
|
|
const char *name = dev_name(dev);
|
|
|
|
log_very_verbose("Getting size of %s", name);
|
|
if ((fd = open(name, O_RDONLY)) < 0) {
|
|
log_sys_error("open", name);
|
|
return 0;
|
|
}
|
|
|
|
if (ioctl(fd, BLKSSZGET, &s) < 0) {
|
|
log_sys_error("ioctl BLKSSZGET", name);
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
close(fd);
|
|
*size = (uint32_t) s;
|
|
return 1;
|
|
}
|
|
|
|
int dev_open(struct device *dev, int flags)
|
|
{
|
|
struct stat buf;
|
|
const char *name = dev_name_confirmed(dev);
|
|
|
|
if (!name) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
if (dev->fd >= 0) {
|
|
log_error("Device '%s' has already been opened", name);
|
|
return 0;
|
|
}
|
|
|
|
if ((stat(name, &buf) < 0) || (buf.st_rdev != dev->dev)) {
|
|
log_error("%s: stat failed: Has device name changed?", name);
|
|
return 0;
|
|
}
|
|
|
|
if ((dev->fd = open(name, flags)) < 0) {
|
|
log_sys_error("open", name);
|
|
return 0;
|
|
}
|
|
|
|
if ((fstat(dev->fd, &buf) < 0) || (buf.st_rdev != dev->dev)) {
|
|
log_error("%s: fstat failed: Has device name changed?", name);
|
|
dev_close(dev);
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
int dev_close(struct device *dev)
|
|
{
|
|
if (dev->fd < 0) {
|
|
log_error("Attempt to close device '%s' "
|
|
"which is not open.", dev_name(dev));
|
|
return 0;
|
|
}
|
|
|
|
if (close(dev->fd))
|
|
log_sys_error("close", dev_name(dev));
|
|
|
|
dev->fd = -1;
|
|
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* FIXME: factor common code out.
|
|
*/
|
|
int _read(int fd, void *buf, size_t count)
|
|
{
|
|
size_t n = 0;
|
|
int tot = 0;
|
|
|
|
while (tot < count) {
|
|
do
|
|
n = read(fd, buf, count - tot);
|
|
while ((n < 0) && ((errno == EINTR) || (errno == EAGAIN)));
|
|
|
|
if (n <= 0)
|
|
return tot ? tot : n;
|
|
|
|
tot += n;
|
|
buf += n;
|
|
}
|
|
|
|
return tot;
|
|
}
|
|
|
|
int64_t dev_read(struct device *dev, uint64_t offset,
|
|
int64_t len, void *buffer)
|
|
{
|
|
const char *name = dev_name(dev);
|
|
int fd = dev->fd;
|
|
|
|
if (fd < 0) {
|
|
log_err("Attempt to read an unopened device (%s).", name);
|
|
return 0;
|
|
}
|
|
|
|
if (lseek(fd, offset, SEEK_SET) < 0) {
|
|
log_sys_error("lseek", name);
|
|
return 0;
|
|
}
|
|
|
|
return _read(fd, buffer, len);
|
|
}
|
|
|
|
int _write(int fd, const void *buf, size_t count)
|
|
{
|
|
size_t n = 0;
|
|
int tot = 0;
|
|
|
|
/* Skip all writes */
|
|
if (test_mode())
|
|
return count;
|
|
|
|
while (tot < count) {
|
|
do
|
|
n = write(fd, buf, count - tot);
|
|
while ((n < 0) && ((errno == EINTR) || (errno == EAGAIN)));
|
|
|
|
if (n <= 0)
|
|
return tot ? tot : n;
|
|
|
|
tot += n;
|
|
buf += n;
|
|
}
|
|
|
|
return tot;
|
|
}
|
|
|
|
int64_t dev_write(struct device *dev, uint64_t offset,
|
|
int64_t len, void *buffer)
|
|
{
|
|
const char *name = dev_name(dev);
|
|
int fd = dev->fd;
|
|
|
|
if (fd < 0) {
|
|
log_error("Attempt to write to unopened device %s", name);
|
|
return 0;
|
|
}
|
|
|
|
if (lseek(fd, offset, SEEK_SET) < 0) {
|
|
log_sys_error("lseek", name);
|
|
return 0;
|
|
}
|
|
|
|
return _write(fd, buffer, len);
|
|
}
|
|
|
|
int dev_zero(struct device *dev, uint64_t offset, int64_t len)
|
|
{
|
|
int64_t r, s;
|
|
char buffer[4096];
|
|
const char *name = dev_name(dev);
|
|
int fd = dev->fd;
|
|
|
|
if (fd < 0) {
|
|
log_error("Attempt to zero part of an unopened device %s",
|
|
name);
|
|
return 0;
|
|
}
|
|
|
|
if (lseek(fd, offset, SEEK_SET) < 0) {
|
|
log_sys_error("lseek", name);
|
|
return 0;
|
|
}
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
|
while (1) {
|
|
s = len > sizeof(buffer) ? sizeof(buffer) : len;
|
|
r = _write(fd, buffer, s);
|
|
|
|
if (r <= 0)
|
|
break;
|
|
|
|
len -= r;
|
|
if (!len) {
|
|
r = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* FIXME: Always display error */
|
|
return (len == 0);
|
|
}
|