1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-17 06:04:23 +03:00

Use O_DIRECT for writing to devices.

Doesn't work on HPPA due to a kernel bug but other archs shuld be OK.
This commit is contained in:
Patrick Caulfield 2002-07-22 08:10:54 +00:00
parent 12caf445d4
commit 6b3b4a250b
2 changed files with 104 additions and 39 deletions

View File

@ -1,5 +1,5 @@
/*
* Copyright (C) 2001 Sistina Software
* Copyright (C) 2002 Sistina Software
*
* This file is released under the LGPL.
*/
@ -10,10 +10,22 @@
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <linux/fs.h> // UGH!!! for BLKSSZGET
#include <linux/fs.h> /* UGH!!! for BLKSSZGET */
/* Buffer for O_DIRECT, allocated as
twice _SC_PAGE_SIZE so we can make sure it is
properly aligned.
It is assumed that _SC_PAGE_SIZE is a power of 2
but I think that's fairly safe :)
*/
static char *bigbuf;
static char *aligned_buf;
static long page_size;
int dev_get_size(struct device *dev, uint64_t * size)
{
@ -62,17 +74,22 @@ int dev_get_sectsize(struct device *dev, uint32_t * size)
return 1;
}
static void _flush(int fd)
{
ioctl(fd, BLKFLSBUF, 0);
}
int dev_open(struct device *dev, int flags)
{
struct stat buf;
const char *name = dev_name_confirmed(dev);
/* First time through - allocate & align the buffer */
if (!page_size) {
page_size = sysconf(_SC_PAGESIZE);
bigbuf = malloc(page_size*2);
if (!bigbuf) {
stack;
return 0;
}
aligned_buf = (char *)(((long)bigbuf + page_size) & ~(page_size-1) );
}
if (!name) {
stack;
return 0;
@ -88,7 +105,12 @@ int dev_open(struct device *dev, int flags)
return 0;
}
if ((dev->fd = open(name, flags)) < 0) {
/* If write was requested then add read so we can pre-fill
buffers for aligned writes */
if (flags & O_WRONLY)
flags = (flags & ~O_WRONLY) | O_RDWR;
if ((dev->fd = open(name, flags|O_DIRECT)) < 0) {
log_sys_error("open", name);
return 0;
}
@ -98,7 +120,7 @@ int dev_open(struct device *dev, int flags)
dev_close(dev);
return 0;
}
_flush(dev->fd);
dev->flags = 0;
return 1;
@ -112,9 +134,6 @@ int dev_close(struct device *dev)
return 0;
}
if (dev->flags & DEV_ACCESSED_W)
_flush(dev->fd);
if (close(dev->fd))
log_sys_error("close", dev_name(dev));
@ -151,18 +170,37 @@ int64_t dev_read(struct device * dev, uint64_t offset,
{
const char *name = dev_name(dev);
int fd = dev->fd;
int64_t newlen;
int64_t newoffset;
int64_t offsetdiff;
int ret;
if (fd < 0) {
log_err("Attempt to read an unopened device (%s).", name);
return 0;
}
if (lseek(fd, offset, SEEK_SET) < 0) {
/* Adjust offset to page size */
newoffset = offset & ~(page_size-1);
offsetdiff = offset - newoffset;
if (lseek(fd, newoffset, SEEK_SET) < 0) {
log_sys_error("lseek", name);
return 0;
}
return _read(fd, buffer, len);
/* Copy to aligned buffer & pad to page size */
newlen = (len+offsetdiff) + page_size - ((len+offsetdiff) % page_size);
ret = _read(fd, aligned_buf, newlen);
if (ret < len+offsetdiff)
{
return ret;
}
/* Copy back to users buffer */
memcpy(buffer, aligned_buf+offsetdiff, len);
return len;
}
int _write(int fd, const void *buf, size_t count)
@ -194,26 +232,53 @@ int64_t dev_write(struct device * dev, uint64_t offset,
{
const char *name = dev_name(dev);
int fd = dev->fd;
int64_t newlen;
int64_t newoffset;
int64_t offsetdiff;
int ret;
if (fd < 0) {
log_error("Attempt to write to unopened device %s", name);
return 0;
}
if (lseek(fd, offset, SEEK_SET) < 0) {
/* Adjust offset to page size */
newoffset = offset & ~(page_size-1);
offsetdiff = offset - newoffset;
if (lseek(fd, newoffset, SEEK_SET) < 0) {
log_sys_error("lseek", name);
return 0;
}
dev->flags |= DEV_ACCESSED_W;
return _write(fd, buffer, len);
/* Pad to page size */
newlen = (len+offsetdiff) + page_size - ((len+offsetdiff) % page_size);
/* We read the page in and just overwrite the bits
requested. Needs to fudge O_RDWR in dev_open for this to work. */
if (_read(fd, aligned_buf, newlen) <= 0) {
log_sys_error("pre-read", name);
return 0;
}
if (lseek(fd, newoffset, SEEK_SET) < 0) {
log_sys_error("re-lseek", name);
return 0;
}
memcpy(aligned_buf+offsetdiff, buffer, len);
ret = _write(fd, aligned_buf, newlen);
if (ret == newlen)
return len;
else
return ret;
}
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;
@ -228,10 +293,10 @@ int dev_zero(struct device *dev, uint64_t offset, int64_t len)
return 0;
}
memset(buffer, 0, sizeof(buffer));
memset(aligned_buf, 0, page_size);
while (1) {
s = len > sizeof(buffer) ? sizeof(buffer) : len;
r = _write(fd, buffer, s);
s = len > page_size ? page_size : len;
r = _write(fd, aligned_buf, s);
if (r <= 0)
break;

View File

@ -57,7 +57,7 @@ CFLAGS+=-g -fno-omit-frame-pointer
#CFLAGS+=-pg
#LD_FLAGS=-pg
CFLAGS+=-DDEBUG_MEM -DDEBUG
CFLAGS+=-DDEBUG_MEM -DDEBUG -D_GNU_SOURCE
#CFLAGS+=-DDEBUG_POOL
#CFLAGS+=-DBOUNDS_CHECK