получить фактическое соглашение о вызовах для функции, использующей libclang - PullRequest
0 голосов
/ 14 января 2019

Я пытаюсь извлечь объявление функции из определения функции, для этого мне нужно фактическое заданное соглашение о вызове функции из исходного определения функции, в настоящее время я использую что-то вроде этого:

 string Convert(const CXString& s)
{
    string result = clang_getCString(s);
    clang_disposeString(s);
    return result;
}

void print_function_prototype(CXCursor cursor) {
  auto type = clang_getCursorType(cursor);

  auto function_name = Convert(clang_getCursorSpelling(cursor));
  auto return_type = Convert(clang_getTypeSpelling(clang_getResultType(type)));
  auto calling_convention = std::to_string(clang_getFunctionTypeCallingConv(type));

  std::cout << return_type << " " << calling_convention << " " << function_name
            << "(";

  int num_args = clang_Cursor_getNumArguments(cursor);
  for (int i = 0; i < num_args; ++i) {
    auto arg_cursor = clang_Cursor_getArgument(cursor, i);
    auto arg_name = Convert(clang_getCursorSpelling(arg_cursor));
    auto arg_data_type =
        Convert(clang_getTypeSpelling(clang_getArgType(type, i)));
    std::cout << (i > 0 ? "," : "") << arg_data_type << " " << arg_name;
  }

  std::cout << ");\n";
}

int main() {
CXIndex index = clang_createIndex(0, 0);
  const char *CompilerArgs[] = {"-IE:/Cpp/Projects/Headers", "-fms-extensions",
                                "-fms-compatibility"};
  auto CompilerArgsCount = sizeof(CompilerArgs) / sizeof(CompilerArgs[0]);
  CXTranslationUnit unit = clang_parseTranslationUnit(
      index, "test.cpp", CompilerArgs,
      CompilerArgsCount, nullptr, 0, CXTranslationUnit_None);
  if (unit == nullptr) {
    cerr << "Unable to parse translation unit. Quitting." << endl;
    exit(-1);
  }

  CXCursor cursor = clang_getTranslationUnitCursor(unit);
  clang_visitChildren(
      cursor,
      [](CXCursor c, CXCursor parent, CXClientData client_data) {
        if (clang_Location_isFromMainFile(clang_getCursorLocation(c)))
          if (clang_getCursorKind(c) == CXCursorKind::CXCursor_FunctionDecl) {
            print_function_prototype(c);
          }
        return CXChildVisit_Recurse;
      },
      nullptr);

  clang_disposeTranslationUnit(unit);
  clang_disposeIndex(index);
}

Теперь соглашение о вызовах, такое как __stdcall, будет игнорироваться в x64 (clang_getFunctionTypeCallingConv (type) == 1), но я хочу получить его независимо от архитектуры.

...