forked from MaskRay/ccls
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtype_printer.cc
106 lines (97 loc) · 3.84 KB
/
type_printer.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
#include "type_printer.h"
#include <string>
#include "loguru.hpp"
namespace {
int GetNameInsertingPosition(const std::string& type_desc,
const std::string& return_type) {
// Check if type_desc contains an (.
if (type_desc.size() <= return_type.size() ||
type_desc.find("(", 0) == std::string::npos)
return type_desc.size();
// Find the first character where the return_type differs from the
// function_type. In most cases this is the place where the function name
// should be inserted.
int ret = 0;
while (return_type[ret] == type_desc[ret])
ret++;
if (ret == 0) {
// If return type and function type do not overlap at all,
// check if it is `auto (...) ->` trailing return type.
if (type_desc.compare(0, 5, "auto ") == 0 &&
type_desc.find(" -> ") != std::string::npos)
ret = 5;
} else {
// Otherwise return type is just a prefix of the function_type.
// Skip any eventual spaces after the return type.
while (type_desc[ret] == ' ')
ret++;
}
return ret;
}
} // namespace
// Build a detailed function signature, including argument names.
std::string GetFunctionSignature(IndexFile* db,
NamespaceHelper* ns,
const CXIdxDeclInfo* decl) {
int num_args = clang_Cursor_getNumArguments(decl->cursor);
std::string function_name =
ns->QualifiedName(decl->semanticContainer, decl->entityInfo->name);
std::vector<std::pair<int, std::string>> args;
for (int i = 0; i < num_args; i++) {
args.emplace_back(-1, ::ToString(clang_getCursorDisplayName(
clang_Cursor_getArgument(decl->cursor, i))));
}
if (clang_Cursor_isVariadic(decl->cursor)) {
args.emplace_back(-1, "");
num_args++;
}
std::string type_desc = ClangCursor(decl->cursor).get_type_description();
int function_name_offset = GetNameInsertingPosition(
type_desc, ::ToString(clang_getTypeSpelling(
clang_getCursorResultType(decl->cursor))));
if (type_desc[function_name_offset] == '(') {
// Find positions to insert argument names.
// Argument name are before ',' or closing ')'.
num_args = 0;
for (int balance = 0, i = function_name_offset;
i < int(type_desc.size()) && num_args < int(args.size()); i++) {
if (type_desc[i] == '(' || type_desc[i] == '[')
balance++;
else if (type_desc[i] == ')' || type_desc[i] == ']') {
if (--balance <= 0) {
args[num_args].first = i;
break;
}
} else if (type_desc[i] == ',' && balance == 1)
args[num_args++].first = i;
}
// Second pass: insert argument names before each comma and closing paren.
int i = function_name_offset;
std::string type_desc_with_names(type_desc.begin(), type_desc.begin() + i);
type_desc_with_names.append(function_name);
for (auto& arg : args) {
if (arg.first < 0) {
LOG_S(ERROR)
<< "When adding argument names to '" << type_desc
<< "', failed to detect positions to insert argument names";
break;
}
if (arg.second.empty())
continue;
// TODO Use inside-out syntax. Note, clang/lib/AST/TypePrinter.cpp does
// not print arg names.
type_desc_with_names.insert(type_desc_with_names.end(), &type_desc[i],
&type_desc[arg.first]);
i = arg.first;
ConcatTypeAndName(type_desc_with_names, arg.second);
}
type_desc_with_names.insert(type_desc_with_names.end(),
type_desc.begin() + i, type_desc.end());
type_desc = std::move(type_desc_with_names);
} else {
// type_desc is either a typedef, or some complicated type we cannot handle.
// Append the function_name in this case.
ConcatTypeAndName(type_desc, function_name);
}
return type_desc;
}