在FreeBSD冻结系统下用c / c ++复制大文件

时间:2012-07-25 20:10:16

标签: c++ freebsd

此代码似乎在Windows(具有意外结果)和Ubuntu下工作。但是当我在FreeBSD 9.0 AMD 64下运行它时会导致系统冻结。我得到这样的错误信息:
ahcich0:插槽28端口0超时 有人知道问题是什么吗? 感谢。

#include <cmath>
#include <cstdlib>
#include <sys/time.h>
#include <iostream>
#include <fstream>
#include <string>
using namespace std;

int main(int argc, char *argv[])
{
    const string FILENAME = "testfile";
    const string COPYNAME = "copy";
    const int FILES = 5;
    const int SIZE_MULTIPLIER = 6;
    const int BUFFER_SIZE = pow(2.0, 16);

    time_t times[2][FILES];

    srand (time(NULL));

    // create test files
    for (int i = 1; i < FILES + 1; i++){
        ofstream os;
        string filename(FILENAME);
        filename += (char)i + 48;
        os.open(filename.c_str(), ios::binary);
        if (os.is_open()){
            cout << "Writing file " << i << " of " << FILES;
            long filesize =pow(2.0, i * SIZE_MULTIPLIER);
            cout << " (" << filesize << " bytes)" <<  endl;

            while(filesize--){
                os << (char)(rand() % 256);
            }
            cout << os.tellp() << " bytes written.\n";
            os.close();
        }else{
            cerr << "Could not create file " << filename;
            cerr << endl;
        }
    }

    // copy the files
    timeval tv;
    time_t start;
    char buffer[BUFFER_SIZE];
    char ci;
    for (int i = 0; i < FILES; i++){
        ci = (char)i + 49;
        string filename(FILENAME);
        filename += ci;
        string copyname("c");
        copyname += COPYNAME;
        copyname += ci;

        cout << "Copying file " << filename.c_str() << endl;

        cout << "the c way: "; 
        cout.flush();

        start = time(NULL);

        FILE *pFile = fopen(filename.c_str(), "rb");
        FILE *pCopy = fopen(copyname.c_str(), "wb");
        if (!(pFile == NULL || pCopy == NULL)){
            do{
                int bytesRead = fread(
                    buffer, 1, BUFFER_SIZE, pFile);

                fwrite(buffer, 1, bytesRead, pCopy);
            }while(!feof(pFile));
            fclose(pFile);
            fclose(pCopy);

            cout << " Done.\n";
        }else{
            cerr << "Could not open either " << filename;
            cerr << " or " << copyname << endl;
        }

        times[0][i] = time(NULL) - start;
        remove(copyname.c_str());

        copyname = "cpp";
        copyname += COPYNAME;
        copyname += ci;

        cout << "the c++ way: ";
        cout.flush();

        start = time(NULL);

        ifstream in;
        in.open(filename.c_str(), ios::binary);
        in.rdbuf()->pubsetbuf(buffer, BUFFER_SIZE);
        ofstream out;
        out.open(copyname.c_str(), ios::binary);
        char copyBuffer[BUFFER_SIZE];
        out.rdbuf()->pubsetbuf(copyBuffer, BUFFER_SIZE);

        if (in.is_open() && out.is_open()){
            out << in.rdbuf();
            in.close();
            out.close();
            cout << " Done.\n";
        }else{
            cerr << "Could not open either " << filename;
            cerr << " or " << copyname << endl;
        }

        times[1][i] = time(NULL) - start ;
        remove(copyname.c_str());
    }

    cout << "Summary:\n";
    cout << "\tc\tc++\n";
    for (int i = 0; i < FILES; i++){
        ci = (char)i + 49;
        cout << "copy" << ci << "\t" << times[0][i];
        cout << "\t" << times[1][i] << endl;
    }

    return 0;
}

2 个答案:

答案 0 :(得分:1)

将FILES更改为4(因为它需要很长时间),你的程序在这里运行得很好:

Writing file 1 of 4 (64 bytes)
64 bytes written.
Writing file 2 of 4 (4096 bytes)
4096 bytes written.
Writing file 3 of 4 (262144 bytes)
262144 bytes written.
Writing file 4 of 4 (16777216 bytes)
16777216 bytes written.
Copying file testfile1
the c way:  Done.
the c++ way:  Done.
Copying file testfile2
the c way:  Done.
the c++ way:  Done.
Copying file testfile3
the c way:  Done.
the c++ way:  Done.
Copying file testfile4
the c way:  Done.
the c++ way:  Done.
Summary:
        c       c++
copy1   0       0
copy2   0       0
copy3   0       0
copy4   0       0

(FreeBSD 9.0-RELEASE-p3 amd64,用clang ++编译)

答案 1 :(得分:1)

在9.0中的achi-driver中可能存在一个错误,该错误出现在负载下。或者,它可能是一个错误的控制器,在相同的负载下失败 - 并且在其他操作系统下没有失败,因为它们不会对它征税太多。

这仍然是FreeBSD 9.2的问题吗?

对于您的程序,您不仅要检查feof(),还要检查读/写循环中的ferror()。此外,在我看来,这样的读/写循环是过去的事情。现在,当size_toffset_t具有相同的宽度(64位平台)时,您应该只需mmap()您的源文件,并fwrite将其放入目标中走。看,妈,没有循环!