libcurl - C: указатель, являющийся realloc'd, не был выделен - PullRequest
0 голосов
/ 18 апреля 2019

Я пишу программу для доступа к Google Drive REST API через HTTP.

Я следую Учебник Google и Пример libcurl

Шаг: запрос устройства и кодов пользователя

curl -d "client_id=client_id&scope=https://www.googleapis.com/auth/drive.file" \
     https://accounts.google.com/o/oauth2/device/code

Шаг: опрос сервера авторизации Google

curl -d "client_id=<client_id>&client_secret=<client_secret>& \
         code=<device_code>&grant_type=http://oauth.net/grant_type/device/1.0" \
         -H "Content-Type: application/x-www-form-urlencoded" \
         https://www.googleapis.com/oauth2/v4/token

Я запускаю свой код несколько раз, иногда он работал хорошо, но в большинстве случаев он терпел неудачу, меня это расстраивало.

Мой код

struct MemoryStruct {
    char *memory;
    size_t size;
};

static size_t WriteMemoryCallback(void *contents, size_t size, size_t nmemb, void *userp)
{
    size_t realsize = size * nmemb;
    struct MemoryStruct *mem = (struct MemoryStruct *)userp;
    char *ptr = realloc(mem->memory, mem->size + realsize + 1);
    if(ptr == NULL) {
        /* out of memory! */
        printf("not enough memory (realloc returned NULL)\n");
        return 0;
    }
    mem->memory = ptr;
    memcpy(&(mem->memory[mem->size]), contents, realsize);
    mem->size += realsize;
    mem->memory[mem->size] = 0;
    return realsize;
}

int main(void)
{
    CURL *curl_handle, *curl_handle2;
    CURLcode res, res2;

    struct MemoryStruct chunk, chunk2;

    chunk.memory = malloc(1);
    chunk.size = 0;

    curl_global_init(CURL_GLOBAL_ALL);

Запрос устройства и кодов пользователя

    curl_handle = curl_easy_init();

    curl_easy_setopt(curl_handle, CURLOPT_URL, "https://accounts.google.com/o/oauth2/device/code");

    curl_easy_setopt(curl_handle, CURLOPT_POSTFIELDS, "client_id=<my_client_id>.apps.googleusercontent.com&scope=https://www.googleapis.com/auth/drive.file");
    curl_easy_setopt(curl_handle, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
    curl_easy_setopt(curl_handle, CURLOPT_WRITEDATA, (void *)&chunk);
    curl_easy_setopt(curl_handle, CURLOPT_USERAGENT, "libcurl-agent/1.0");

    res = curl_easy_perform(curl_handle);

    if(res != CURLE_OK) {
        fprintf(stderr, "curl_easy_perform() failed: %s\n",
                curl_easy_strerror(res));
    }
    else {
        //skip some codes of extracting the user_code here
    }
    curl_easy_cleanup(curl_handle);

    free(chunk.memory);

Опрос сервера авторизации Google

    printf("Enter below code on 'https://www.google.com/device' :\n");
    printf("%s\n", user_code);
    printf("Press enter when finish\n");
    while( getchar() != '\n' );

    curl_handle2 = curl_easy_init();
    curl_easy_setopt(curl_handle2, CURLOPT_URL, "https://www.googleapis.com/oauth2/v4/token");

    char url[300];
    strcat(url, "client_id=<my_client_id>.apps.googleusercontent.com&client_secret=<my_client_secret>&code=");
    strcat(url, device_code);
    strcat(url, "&grant_type=http://oauth.net/grant_type/device/1.0");

    curl_easy_setopt(curl_handle2, CURLOPT_POSTFIELDS, url);
    curl_easy_setopt(curl_handle2, CURLOPT_WRITEFUNCTION, WriteMemoryCallback);
    curl_easy_setopt(curl_handle2, CURLOPT_WRITEDATA, (void *)&chunk2);
    curl_easy_setopt(curl_handle2, CURLOPT_USERAGENT, "libcurl-agent/1.0");

    res2 = curl_easy_perform(curl_handle2);

    if(res2 != CURLE_OK) {
        fprintf(stderr, "curl_easy_perform() failed: %s\n",
                curl_easy_strerror(res2));
    }
    else {
        //do something here
    }
    curl_easy_cleanup(curl_handle2);
    free(chunk2.memory);

В этой строке появилась ошибка res = curl_easy_perform(curl_handle);

malloc: *** error for object 0x10068a400: pointer being realloc'd was not allocated
test(3989,0x1000ab5c0) malloc: *** set a breakpoint in malloc_error_break to debug

Я включил Address Sanitizer в Схеме запуска, и она показала:

ERROR: AddressSanitizer: attempting double-free on 0x6110000205c0 in thread T0:
2019-04-18 10:17:28.969346+0800 atos[3802:117177] examining /Users/USER/Library/Developer/Xcode/DerivedData/test-dbydzdrhbxzzjcgcmnrcjvlrthal/Build/Products/Debug/test [3789]
    #0 0x1001333d7 in wrap_realloc (libclang_rt.asan_osx_dynamic.dylib:x86_64h+0x5c3d7)
    #1 0x1000030f9 in WriteMemoryCallback main.c:30
    #2 0x7fff796da5e4 in Curl_client_chop_write (libcurl.4.dylib:x86_64+0xb5e4)
    #3 0x7fff796f652a in Curl_httpchunk_read (libcurl.4.dylib:x86_64+0x2752a)
    #4 0x7fff796f134b in Curl_readwrite (libcurl.4.dylib:x86_64+0x2234b)
    #5 0x7fff796fa6b9 in multi_runsingle (libcurl.4.dylib:x86_64+0x2b6b9)
    #6 0x7fff796f9b83 in curl_multi_perform (libcurl.4.dylib:x86_64+0x2ab83)
    #7 0x7fff796f2db6 in curl_easy_perform (libcurl.4.dylib:x86_64+0x23db6)
    #8 0x100002363 in main main.c:120
    #9 0x7fff7bcb63d4 in start (libdyld.dylib:x86_64+0x163d4)

0x6110000205c0 is located 0 bytes inside of 246-byte region [0x6110000205c0,0x6110000206b6)
freed by thread T0 here:
    #0 0x10013320d in wrap_free (libclang_rt.asan_osx_dynamic.dylib:x86_64h+0x5c20d)
    #1 0x1000021cd in main main.c:98
    #2 0x7fff7bcb63d4 in start (libdyld.dylib:x86_64+0x163d4)
...