前段时间准备以direct方式写入文件,结果一直写不进去内容,后面发现必须要512字节对齐才可以。使用new或者malloc分配的内存是无法在O_DIRECT选项下工作的,必须使用posix_memalign(或valloc, memalign,这两个函数已经被标记为废弃)。
读写操作的传输数据大小和缓冲区地址都需要按照一定的规则对齐:
Linux下,对于不同的文件系统和内核版本,需要的对齐边界不同,也没有统一的接口可以获取到该边界值。
对于kernel 2.4版本:传输大小和缓冲区地址均需要按照访问文件系统的逻辑块大小对齐,比如文件系统的块大小是4K,buffer地址需要按照4K对齐,需要读写4K倍数的数据
对于kernel 2.6版本:传输大小和缓冲区地址按照目标存储设备的扇区大小(一般512)对齐
在open文件的时候传入O_DIRECT标志,绕过缓冲区高速缓存直接I/O.
但是存在如下限制:
用于传递数据的缓冲区,其内存边界必须对齐为块大小的整数倍。
用于传输数据的缓冲区,其内存边界必须为快大小的整数倍。
待传输数据的长度必须是块大小的整数倍。
如果不遵守上述任一条,将导致EINVAL错误。块大小(block size)指设备的物理块大小,通常为512字节.
可使用memalign (malloc.h)来分配指定地址对齐的资源接口:void *memalign(size_t boundary, size_t size);
#define _GNU_SOURCE
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
#include <string.h>
int main()
{
int fd;
char buf[4096];
char *str = "0123456789\n";
char *content = NULL;
content = memalign(512, 512 * 2); // align by 512
memcpy(content, str, sizeof(str));
fd = open("/tmp/bbbbbbb", O_RDWR | O_CREAT | O_DIRECT, 0644);
write(fd, content, 1024);
write(fd, content, 1024);
close(fd);
}