Как получить имена аргументов указателя функции с помощью clang LibTooling?

Скажем, я анализирую такой код:

struct Foo
{
    void(*setParam)(const char* name, int value);
};

Я использую clang LibTooling и получаю FieldDecl на setParam.

Я подумал, что могу получить такие типы аргументов:

auto ft = fieldDecl->getFunctionType()->getAs<FunctionProtoType>();
for (size_t i = 0; i < fpt->getNumParams(); i++)
{
    QualType paramType = fpt->getParamType(i);
    ....
}

Но как мне получить имена аргументов? («имя» и «значение» в этом случае) Это вообще возможно или мне нужно вручную заглянуть в источник (с SourceManager)?


person Shens    schedule 24.12.2018    source источник


Ответы (1)


Я не думаю, что возможно получить имена параметров непосредственно из типа, поскольку они не являются частью информации о типе.

Но вашу задачу можно решить еще одним обращением к объявлению указателя функции:

class ParmVisitor
    : public RecursiveASTVisitor<ParmVisitor>
{
public:
    bool VisitParmVarDecl(ParmVarDecl *d) {
        if (d->getFunctionScopeDepth() != 0) return true;

        names.push_back(d->getName().str());
        return true;
    }

    std::vector<std::string> names;
};

Тогда вызывающий сайт:

bool VisitFieldDecl(Decl *d) {
    if (!d->getFunctionType()) {
        // not a function pointer
        return true;
    }
    ParmVisitor pv;
    pv.TraverseDecl(d);
    auto names = std::move(pv.names);

    // now you get the parameter names...

    return true;
}

Обратите внимание на часть getFunctionScopeDepth(), она необходима, потому что параметр функции может быть указателем на функцию, например:

void(*setParam)(const char* name, int value, void(*evil)(int evil_name, int evil_value));

getFunctionScopeDepth() значение 0 гарантирует, что этот параметр не находится во вложенном контексте.

person llllllllll    schedule 24.12.2018