5

我正在开发一个 FUSE 驱动程序,当我将它作为守护进程运行时(没有 -f 或 -d 标志),所有通过 libcurl 发出的 https 请求都失败。我能够通过发出 https 请求、分叉并返回父进程,然后从新进程发出第二个请求来重现该错误。如果我删除fork呼叫或发出 http 请求,则没有错误。

我现在正在制作官方错误报告,但有人知道我怎样才能让它工作吗?

这是我的代码和(日志文件)输出:

注意:如果你运行我的程序,管道到 /dev/null 因为 libcurl 默认将接收到的缓冲区发送到标准输出。

#include <curl/curl.h>
#include <string>
#include <unistd.h>
#include <iostream>

using namespace std;

void log(string str)
{   //TODO: remove
    static const char logfile[] = "/home/austin/megalog";
    FILE *f = fopen(logfile, "a");
    fwrite(str.data(), str.size(), 1, f);
    fclose(f);
    cout << str;
}

int main(int argc, char *argv[])
{
    string url = "https://www.google.com/";
    char errBuf[1024];
    CURLcode err;

    curl_global_init(CURL_GLOBAL_DEFAULT);
    CURL *handle = curl_easy_init();
    curl_easy_setopt(handle, CURLOPT_URL, url.c_str());
    curl_easy_setopt(handle, CURLOPT_ERRORBUFFER, errBuf);

    if ((err = curl_easy_perform(handle)))
    {
        log("first request failed\n");
        return 1;
    }
    curl_easy_cleanup(handle);

    if(fork())
        return 0;

    handle = curl_easy_init();
    curl_easy_setopt(handle, CURLOPT_URL, url.c_str());
    curl_easy_setopt(handle, CURLOPT_ERRORBUFFER, errBuf);

    if ((err = curl_easy_perform(handle)))
    {
        log(string("curl error while sending: (") + to_string(err) + ") " + curl_easy_strerror(err) + "\n");
        log(errBuf);
    }
    else
        log("no error\n");

    return 0;
}

...和输出:

$ g++ -std=c++11 main.cpp -lcurl
$ rm -f log
$ ./a.out > /dev/null
$ cat log
curl error while sending: (35) SSL connect error
A PKCS #11 module returned CKR_DEVICE_ERROR, indicating that a problem has occurred with the token or slot.

我正在使用(最新的)libcurl 版本 7.29.0,(最新的)openssl 版本 1.0.1e,并且我正在运行内核版本为 3.7.4 的 Fedora 18。

4

1 回答 1

4

要完成这项工作,您需要在curl_global_cleanup之前forkcurl_global_init之后调用fork。libcurl 邮件列表中有人提到,这是fork在初始化需要初始化的库后调用时的常见问题。

于 2013-03-18T00:22:53.650 回答