Импортированная функция DLL выдает ошибку "Термин не относится к функции, принимающей 1 аргумент"

Я пытаюсь использовать реализацию MIT Kerberos (используя krb5_32.dll из k4w-4.0.1 и связанный файл заголовка), чтобы получить билет TGT и Service.

Я загрузил функцию krb5_init_context, которая, согласно заголовочному файлу google и SO, принимает только 1 аргумент (структура krb5_context) и заполняет его.

#include "stdafx.h"
#include "windows.h"
#include "krb5.h"

typedef int krb5_int32;
typedef krb5_int32 krb5_error_code;

int _tmain(int argc, _TCHAR* argv[])
{    

    HMODULE kerberos = LoadLibrary(L"krb5_32.dll");
    HANDLE krb5_init_context = NULL;

    if(kerberos == NULL)
    {
        printf("Failed to load library!\n");
        printf("%lu", GetLastError());
        return -1;
    }
    else
    {
        printf("Library krb5_32.dll loaded successfully!\n");
    }

    if((krb5_init_context = GetProcAddress(kerberos, "krb5_init_context")) == NULL)
    {
        printf("GetProcAddress for krb5_init_context failed!\n");   
        return -1;
    }
    else
    {
        printf("Function krb5_init_context loaded successfully!\n");
    }

    krb5_context context = NULL;
    krb5_ccache cache = NULL;
    krb5_principal client_princ = NULL;
    char* name = NULL;
    krb5_keytab keytab = 0;
    krb5_creds creds;
    krb5_get_init_creds_opt *options = NULL;
    krb5_error_code error_code = 0; //error_status_ok;

    error_code = (*krb5_init_context)(&context);

    printf("Error Code: " + error_code);

    while(true);


    return 0;
}

1 ответ

Решение

Чтобы вызвать функцию с помощью указателя, вы должны объявить указатель на функцию. В общем, объявление указателя функции (статический член, глобальная или статическая функция) выглядит так:

typedef return_type (* псевдоним_имя)(argtype_1, argtype_2,...argtype_n);

где return_type тип возвращаемого значения, alias_name полученное имя, которое вы будете использовать для объявления переменной указателя на функцию, а arg1type_1, argtype_2, и т.д. - это типы аргументов, которые принимает функция.

Согласно вашему посту, krb5_init_context должен быть объявлен как это (используя typedef упростить вещи):

typedef krb5_int32 (*context_fn)(krb5_context*);  // pointer to function type
contextfn krb5_init_context;  // declare it
//...
krb5_init_context = (context_fn)GetProcAddress(...);  // get the address
//..
krb5_context context; 
krb5_init_context(&context);  // now function can be called

После этих изменений убедитесь, что вы также объявляете указатель функции с соответствующим соглашением о вызовах в качестве экспортируемой функции. Если функция __stdcall, то вам нужно указать, что в typedef, Если вы этого не сделаете, ваша функция потерпит крах.

Чтобы добавить соглашение о вызовах (в данном случае это __stdcall):

typedef krb5_int32 (__stdcall *context_fn)(krb5_context*);  
Другие вопросы по тегам