mirror of
https://github.com/ceph/ceph
synced 2025-01-08 20:21:33 +00:00
3ce41446a6
We end up needing _GNU_SOURCE in a bunch of places-- to get direct i/o, pipe2, and some other Linux-specific interfaces. Signed-off-by: Colin McCabe <colin.mccabe@dreamhost.com>
187 lines
4.0 KiB
C
187 lines
4.0 KiB
C
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <inttypes.h>
|
|
#include <linux/types.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/ioctl.h>
|
|
|
|
//#include "../client/ioctl.h"
|
|
|
|
#include <linux/ioctl.h>
|
|
#include <linux/types.h>
|
|
#define CEPH_IOCTL_MAGIC 0x97
|
|
#define CEPH_IOC_SYNCIO _IO(CEPH_IOCTL_MAGIC, 5)
|
|
|
|
void write_pattern()
|
|
{
|
|
printf("writing pattern\n");
|
|
|
|
int fd = open("foo", O_CREAT|O_WRONLY, 0644);
|
|
uint64_t i;
|
|
int r;
|
|
|
|
for (i=0; i<1048576 * sizeof(i); i += sizeof(i)) {
|
|
r = write(fd, &i, sizeof(i));
|
|
}
|
|
|
|
close(fd);
|
|
}
|
|
|
|
int verify_pattern(char *buf, size_t len, uint64_t off)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < len; i += sizeof(uint64_t)) {
|
|
uint64_t expected = i + off;
|
|
uint64_t actual = *(uint64_t*)(buf + i);
|
|
if (expected != actual) {
|
|
printf("error: offset %llu had %llu\n", (unsigned long long)expected,
|
|
(unsigned long long)actual);
|
|
exit(1);
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void generate_pattern(void *buf, size_t len, uint64_t offset)
|
|
{
|
|
uint64_t *v = buf;
|
|
size_t i;
|
|
|
|
for (i=0; i<len / sizeof(v); i++)
|
|
v[i] = i * sizeof(v) + offset;
|
|
verify_pattern(buf, len, offset);
|
|
}
|
|
|
|
int read_direct(int buf_align, uint64_t offset, int len)
|
|
{
|
|
printf("read_direct buf_align %d offset %llu len %d\n", buf_align,
|
|
(unsigned long long)offset, len);
|
|
int fd = open("foo", O_RDONLY|O_DIRECT);
|
|
void *rawbuf;
|
|
int r = posix_memalign(&rawbuf, 4096, len + buf_align);
|
|
void *buf = (char *)rawbuf + buf_align;
|
|
memset(buf, 0, len);
|
|
r = pread(fd, buf, len, offset);
|
|
close(fd);
|
|
r = verify_pattern(buf, len, offset);
|
|
free(rawbuf);
|
|
return r;
|
|
}
|
|
|
|
int read_sync(int buf_align, uint64_t offset, int len)
|
|
{
|
|
printf("read_sync buf_align %d offset %llu len %d\n", buf_align,
|
|
(unsigned long long)offset, len);
|
|
int fd = open("foo", O_RDONLY);
|
|
ioctl(fd, CEPH_IOC_SYNCIO);
|
|
void *rawbuf;
|
|
int r = posix_memalign(&rawbuf, 4096, len + buf_align);
|
|
void *buf = (char *)rawbuf + buf_align;
|
|
memset(buf, 0, len);
|
|
r = pread(fd, buf, len, offset);
|
|
close(fd);
|
|
r = verify_pattern(buf, len, offset);
|
|
free(rawbuf);
|
|
return r;
|
|
}
|
|
|
|
int write_direct(int buf_align, uint64_t offset, int len)
|
|
{
|
|
printf("write_direct buf_align %d offset %llu len %d\n", buf_align,
|
|
(unsigned long long)offset, len);
|
|
int fd = open("foo", O_WRONLY|O_DIRECT|O_CREAT, 0644);
|
|
void *rawbuf;
|
|
posix_memalign(&rawbuf, 4096, len + buf_align);
|
|
void *buf = (char *)rawbuf + buf_align;
|
|
int r;
|
|
|
|
generate_pattern(buf, len, offset);
|
|
|
|
r = pwrite(fd, buf, len, offset);
|
|
close(fd);
|
|
|
|
fd = open("foo", O_RDONLY);
|
|
void *buf2 = malloc(len);
|
|
memset(buf2, 0, len);
|
|
r = pread(fd, buf2, len, offset);
|
|
close(fd);
|
|
|
|
r = verify_pattern(buf2, len, offset);
|
|
|
|
unlink("foo");
|
|
free(rawbuf);
|
|
free(buf2);
|
|
return r;
|
|
}
|
|
|
|
int write_sync(int buf_align, uint64_t offset, int len)
|
|
{
|
|
printf("write_sync buf_align %d offset %llu len %d\n", buf_align,
|
|
(unsigned long long)offset, len);
|
|
int fd = open("foo", O_WRONLY|O_CREAT, 0644);
|
|
ioctl(fd, CEPH_IOC_SYNCIO);
|
|
void *rawbuf;
|
|
int r = posix_memalign(&rawbuf, 4096, len + buf_align);
|
|
void *buf = (char *)rawbuf + buf_align;
|
|
|
|
generate_pattern(buf, len, offset);
|
|
|
|
r = pwrite(fd, buf, len, offset);
|
|
close(fd);
|
|
|
|
fd = open("foo", O_RDONLY);
|
|
void *buf2 = malloc(len);
|
|
memset(buf2, 0, len);
|
|
r = pread(fd, buf2, len, offset);
|
|
close(fd);
|
|
|
|
r = verify_pattern(buf2, len, offset);
|
|
|
|
unlink("foo");
|
|
free(buf2);
|
|
free(rawbuf);
|
|
return r;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
uint64_t i, j, k;
|
|
int read = 1;
|
|
int write = 1;
|
|
|
|
if (argc >= 2 && strcmp(argv[1], "read") == 0)
|
|
write = 0;
|
|
if (argc >= 2 && strcmp(argv[1], "write") == 0)
|
|
read = 0;
|
|
|
|
if (read) {
|
|
write_pattern();
|
|
|
|
for (i = 0; i < 4096; i += 512)
|
|
for (j = 4*1024*1024 - 4096; j < 4*1024*1024 + 4096; j += 512)
|
|
for (k = 1024; k <= 16384; k *= 2) {
|
|
read_direct(i, j, k);
|
|
read_sync(i, j, k);
|
|
}
|
|
|
|
}
|
|
unlink("foo");
|
|
if (write) {
|
|
for (i = 0; i < 4096; i += 512)
|
|
for (j = 4*1024*1024 - 4096 + 512; j < 4*1024*1024 + 4096; j += 512)
|
|
for (k = 1024; k <= 16384; k *= 2) {
|
|
write_direct(i, j, k);
|
|
write_sync(i, j, k);
|
|
}
|
|
}
|
|
|
|
|
|
return 0;
|
|
}
|