8

これらの関数を使用してファイルをコピーしようとすると、プログラムが memcpy 関数にヒットし、バス エラーが発生してプロセスが終了するまで、すべてがうまくいきます。

void copy_mmap(char* in, char* out){

int input_fd, output_fd;

input_fd = open (in, O_RDONLY);
if (input_fd == -1) {
        printf("Error opening input file.\n");
        exit(2);
}

output_fd = open(out, O_RDWR | O_CREAT, S_IWUSR | S_IRUSR);
if(output_fd == -1){
    printf("Error opening output file.\n");
    exit(3);
}

struct stat st;
fstat(input_fd, &st);

char* target;
target=mmap(0, st.st_size+1, PROT_READ, MAP_SHARED, input_fd, 0);
if (target==(void*) -1){
    printf("Error mapping target with errno: %d.\n", errno);
    exit(6);
}


char* destination;
destination=mmap(0, st.st_size+1, PROT_READ | PROT_WRITE, MAP_SHARED, output_fd, 0);
if (destination==(void*) -1){
    printf("Error mapping destination with errno: %d.\n", errno);
    exit(5);
}

memcpy(destination, target, st.st_size);
munmap(destination, st.st_size);



}

「バスエラー」は説明的なエラーメッセージではなく、この問題に関する資料がインターネット上にあまりないため、何が問題なのかを突き止めることができませんでした。

4

3 に答える 3

22

宛先ファイルを新規ファイルとして作成すると、そのサイズは 0 バイトになります。memcpyファイルの終わりを超えてデータを書き込もうとするため、クラッシュします。

これを機能させるには、宛先ファイルをソース ファイルのサイズに合わせて事前にサイズ設定します ( を使用ftruncate()) mmap()

st.st_sizeまた、 2 番目の引数としてmmapではなくを渡す必要がありますst.st_size+1st.st_size+1ファイルのサイズより大きい範囲をマップしようとしますが、これは無効です。

于 2013-01-09T17:02:23.060 に答える
1

off_t fileposition= lseek(output_fd, st.st_size-1, SEEK_SET) に lseek し、output_fd に空の文字を書き込むこともできます。

于 2016-02-17T09:27:43.467 に答える
0

おそらくmemmoveを使用してみてください?同じメモリ/ファイル/デバイスの場所から読み取りと書き込みを行っていませんか?Memcpyはそのような状況では失敗します。

于 2013-01-09T16:58:11.393 に答える