1

这是 LIST ftp 的 cmd 的片段:

count = file_list("./", &files);
if((fp_list = fopen("listfiles.txt", "w")) == NULL){
  perror("Impossibile aprire il file per la scrittura LIST");
  onexit(newsockd, sockd, 0, 2);
}
for(i=0; i < count; i++){
  if(strcmp(files[i], "DIR ..") == 0 || strcmp(files[i], "DIR .") == 0) continue;
  else{
    fprintf(fp_list, "%s\n", files[i]);
  }
}
fclose(fp_list);
if((fpl = open("listfiles.txt", O_RDONLY)) < 0){
  perror("open file with open");
  onexit(newsockd, sockd, 0, 2);
  exit(1);
}
if(fstat(fpl, &fileStat) < 0){
  perror("Errore fstat");
  onexit(newsockd, sockd, fpl, 3);
}
fsize = fileStat.st_size;
if(send(newsockd, &fsize, sizeof(fsize), 0) < 0){
  perror("Errore durante l'invio grande file list");
  onexit(newsockd, sockd, fpl, 3);
}
rc_list = sendfile(newsockd, fpl, &offset_list, fileStat.st_size);
if(rc_list == -1){
  perror("Invio file list non riuscito");
  onexit(newsockd, sockd, fpl, 3);
}
if((uint32_t)rc_list != fsize){
  fprintf(stderr, "Error: transfer incomplete: %d di %d bytes inviati\n", rc_list, (int)fileStat.st_size);
  onexit(newsockd, sockd, fpl, 3);
}
printf("OK\n");
close(fpl);
if(remove( "listfiles.txt" ) == -1 ){
  perror("errore cancellazione file");
  onexit(newsockd, sockd, 0, 2);
}

where&files被声明为char **files,并且该函数list_files是我编写的与我的问题无关的函数。
我的问题:第一次调用 LIST cmd 它工作正常,但如果我再次调用 LIST,它总是给我“错误,传输不完整”我不明白为什么......

4

2 回答 2

7

sendfile函数可能不会在一次调用中发送所有数据,在这种情况下,它将返回一个小于请求的数字。您将此视为错误,但您应该重试。一种方法是使用与此类似的循环:

// Offset into buffer, this is where sendfile starts reading the buffer
off_t offset = 0;

// Loop while there's still some data to send
for (size_t size_to_send = fsize; size_to_send > 0; )
{
    ssize_t sent = sendfile(newsockd, fpl, &offset, size_to_send);

    if (sent <= 0)
    {
        // Error or end of file
        if (sent != 0)
            perror("sendfile");  // Was an error, report it
        break;
    }

    size_to_send -= sent;  // Decrease the length to send by the amount actually sent
}
于 2012-08-08T08:59:39.903 回答
2

我发现了我的问题。
当我sendfile多次调用时,变量off_t offset_list;仍然是“脏的”
如果第一次调用 sendfileoffest_list将具有一个不会在我第二次调用该函数时被删除的值。
所以如果我必须写offset_list = 0;之前rc_list = sendfile(newsockd, fpl, &offset_list, fileStat.st_size);和所有的工作!

于 2012-08-08T09:01:28.547 回答