-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCParser.lua
132 lines (107 loc) · 2.99 KB
/
CParser.lua
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
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
local TS = vim.treesitter
local util = require('util')
TS.set_query('c', 'injections', '') -- https://github.com/neovim/neovim/issues/21275
local Parser = { __index = {} }
function Parser.new(path)
local text = (function()
local file = assert(io.open(path, 'rb'))
local text = file:read('*a')
file:close()
return text
end)()
local tree = TS.get_string_parser(text, 'c'):parse()[1]:root()
local parser = setmetatable({
file = path,
text = text,
tree = tree,
result = {},
_comments = nil,
_comment_line = nil,
}, Parser)
return parser
end
--- Get node string
function Parser.__index:to_str(node)
local _, _, s = node:start()
local _, _, e = node:end_()
return self.text:sub(s + 1, e)
end
function Parser.__index:parse_arg(node)
return self:to_str(node)
end
function Parser.__index:parse_attr(node)
return self:to_str(node)
end
function Parser.__index:parse_function(node)
assert(node:type() == 'function_definition', 'invalid node type')
local return_type = node:field('type')[1]
if not return_type then return end
local declarator = node:field('declarator')[1]
if not declarator then return end
local name = declarator:field('declarator')[1]
if not name then return end
name = self:to_str(name)
if not name:match('^nvim_') then
return
end
local params = declarator:field('parameters')[1]
if not params then return end
local args = {}
for child in params:iter_children() do
if child:type() == 'parameter_declaration' then
table.insert(args, self:parse_arg(child))
end
end
local attrs = {}
for child in declarator:iter_children() do
if child:type() == 'nvim_attribute_specifier' then
table.insert(attrs, self:parse_attr(child))
end
end
local desc = {}
if self._comments ~= nil and node:start() == self._comment_line + 1 then
desc, self._comments = util.dedent(self._comments), nil
end
self.result[name] = {
name = name,
args = args,
ret = self:to_str(return_type),
attrs = attrs,
desc = desc,
}
end
function Parser.__index:parse_comment(node)
assert(node:type() == 'comment', 'invalid node type')
local comment = self:to_str(node):match('^///(.-)%s*$')
if not comment then return false end
local prev_line = self._comment_line
self._comment_line = node:start()
if self._comments == nil or self._comment_line ~= prev_line + 1 then
self._comments = { comment }
else
table.insert(self._comments, comment)
end
end
function Parser.__index:parse()
local function reset_comment()
self._comments, self._comment_line = nil, nil
end
for node in self.tree:iter_children() do
if node:type() == 'comment' then
if self:parse_comment(node) == false then
reset_comment()
end
elseif node:type() == 'function_definition' then
self:parse_function(node)
reset_comment()
else
reset_comment()
end
end
return self.result
end
return {
parse = function(file)
return Parser.new(file):parse()
end,
}