gpt4 book ai didi

c - linux内核aio功能

转载 作者:IT王子 更新时间:2023-10-29 00:01:05 26 4
gpt4 key购买 nike

我正在测试内核异步 io 函数(不是 posix aio)并试图弄清楚它是如何工作的。下面的代码是一个完整的程序,我只是将一个数组重复写入使用 O_DIRECT 打开的文件。我在回调函数中收到错误消息“write missed bytes expect 1024 got 0”(请参阅​​ work_done() 中的 fprintf 语句)。

对于那些不熟悉内核 aio 的人,下面的代码执行以下操作:

  1. 初始化一些结构
  2. 准备 aio (io_prep_pwrite)
  3. 提交 io 请求 (io_submit)
  4. 检查事件是否完成 (io_getevents)
  5. 调用回调函数以查看是否一切正常。

我在第 5 步遇到错误。如果我不使用 O_DIRECT 打开文件,一切正常,但它超出了异步写入的目的。有人可以告诉我我做错了什么吗?这是内核 aio 的正确用法吗,例如,我对回调的使用是否正确? O_DIRECT 的使用有什么限制吗?

我使用'gcc -Wall test.c -laio'编译

提前致谢。

/* 
* File: myaiocp.c
* Author: kmehta
*
* Created on July 11, 2011, 12:50 PM
*
*
* Testing kernel aio.
* Program creates a 2D matrix and writes it multiple times to create a file of desired size.
* Writes are performed using kernel aio functions (io_prep_pwrite, io_submit, etc.)
*/
#define _GNU_SOURCE
#define _XOPEN_SOURCE 600

#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <pthread.h>
#include <fcntl.h>
#include <string.h>
#include <sys/uio.h>
#include <sys/time.h>
#include <omp.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <errno.h>
#include <libaio.h>

char ** buf;
long seg_size;
int seg_rows;
double total_size;
char * filename;
static int wait_count = 0;

void io_task();
void cleanup();
void allocate_2D_matrix(int[]);
int file_open(char *);
void wr_done(io_context_t ctx, struct iocb* iocb, long res, long res2);

int main(int argc, char **argv) {
total_size = 1048576; //1MB
seg_size = 1024; //1kB
seg_rows = 1024;
filename = "aio.out";

int dims[] = {seg_rows, seg_size};
allocate_2D_matrix(dims); //Creates 2D matrix

io_task();
cleanup();

return 0;
}

/*
* Create a 2D matrix
*/
void allocate_2D_matrix(int dims[2]) {
int i;
char *data;

//create the matrix
data = (char *) calloc(1, dims[0] * dims[1] * sizeof (char));
if (data == NULL) {
printf("\nCould not allocate memory for matrix.\n");
exit(1);
}

buf = (char **) malloc(dims[0] * sizeof (char *));
if (buf == NULL) {
printf("\nCould not allocate memory for matrix.\n");
exit(1);
}

for (i = 0; i < dims[0]; i++) {
buf[i] = &(data[i * dims[1]]);
}
}

static void io_error(const char *func, int rc)
{
if (rc == -ENOSYS)
fprintf(stderr, "AIO not in this kernel\n");
else if (rc < 0)
fprintf(stderr, "%s: %s\n", func, strerror(-rc));
else
fprintf(stderr, "%s: error %d\n", func, rc);

exit(1);
}

/*
* Callback function
*/
static void work_done(io_context_t ctx, struct iocb *iocb, long res, long res2)
{

if (res2 != 0) {
io_error("aio write", res2);
}

if (res != iocb->u.c.nbytes) {
fprintf(stderr, "write missed bytes expect %lu got %ld\n",
iocb->u.c.nbytes, res2);
exit(1);
}
wait_count --;
printf("%d ", wait_count);
}

/*
* Wait routine. Get events and call the callback function work_done()
*/
int io_wait_run(io_context_t ctx, long iter)
{
struct io_event events[iter];
struct io_event *ep;
int ret, n;

/*
* get up to aio_maxio events at a time.
*/
ret = n = io_getevents(ctx, iter, iter, events, NULL);
printf("got %d events\n", n);
/*
* Call the callback functions for each event.
*/
for (ep = events ; n-- > 0 ; ep++) {
io_callback_t cb = (io_callback_t)ep->data ; struct iocb *iocb = ep->obj ; cb(ctx, iocb, ep->res, ep->res2);
}
return ret;
}

void io_task() {
long offset = 0;
int bufIndex = 0;

//Open file
int fd = file_open(filename);

//Initialize structures
long i;
long iter = total_size / seg_size; //No. of iterations to reach desired file size (total_size)
io_context_t myctx;
if(0 != io_queue_init(iter, &myctx))
{
perror("Could not initialize io queue");
exit(EXIT_FAILURE);
}
struct iocb * ioq[iter];

//loop through iter times to reach desired file size
for (i = 0; i < iter; i++) {
struct iocb *io = (struct iocb*) malloc(sizeof (struct iocb));
io_prep_pwrite(io, fd, buf[bufIndex], seg_size, offset);
io_set_callback(io, work_done);
ioq[i] = io;

offset += seg_size;
bufIndex ++;
if (bufIndex > seg_rows - 1) //If entire matrix written, start again from index 0
bufIndex = 0;
}

printf("done preparing. Now submitting..\n");
if(iter != io_submit(myctx, iter, ioq))
{
perror("Failure on submit");
exit(EXIT_FAILURE);
}

printf("now awaiting completion..\n");
wait_count = iter;
int res;

while (wait_count) {
res = io_wait_run(myctx, iter);
if (res < 0)
io_error("io_wait_run", res);
}

close(fd);
}

void cleanup() {
free(buf[0]);
free(buf);
}

int file_open(char *filename) {
int fd;
if (-1 == (fd = open(filename, O_DIRECT | O_CREAT | O_WRONLY | O_TRUNC, 0666))) {
printf("\nError opening file. \n");
exit(-1);
}

return fd;
}

最佳答案

首先,使用 libaio 代替 POSIX aio 做得很好。

Are there any restrictions on the usage of O_DIRECT ?

我不是 100% 确定这是真正的问题,但 O_DIRECT 有一些要求(主要引用自 TLPI):

  • 正在传输的数据缓冲区必须在 block 大小的倍数的内存边界上对齐(使用 posix_memalign)
  • 数据传输开始的文件或设备偏移量必须是 block 大小的倍数
  • 要传输的数据长度必须是 block 大小的倍数

乍一看,我可以看出您没有采取任何预防措施来对齐 allocate_2D_matrix 中的内存。

If I do not open the file using O_DIRECT, things work fine, but it beats the purpose of having async writes.

事实并非如此。异步 I/O 在没有 O_DIRECT 的情况下也能很好地工作(例如,考虑削减的系统调用数量)。

关于c - linux内核aio功能,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6918530/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com