xref: /illumos-gate/usr/src/tools/smatch/src/ast-inspect.c (revision 1f5207b7604fb44407eb4342aff613f7c4508508)
1*1f5207b7SJohn Levon 
2*1f5207b7SJohn Levon #include "token.h"
3*1f5207b7SJohn Levon #include "parse.h"
4*1f5207b7SJohn Levon #include "symbol.h"
5*1f5207b7SJohn Levon #include "ast-inspect.h"
6*1f5207b7SJohn Levon #include "expression.h"
7*1f5207b7SJohn Levon 
inspect_ptr_list(AstNode * node,const char * name,void (* inspect)(AstNode *))8*1f5207b7SJohn Levon static inline void inspect_ptr_list(AstNode *node, const char *name, void (*inspect)(AstNode *))
9*1f5207b7SJohn Levon {
10*1f5207b7SJohn Levon 	struct ptr_list *ptrlist = node->ptr;
11*1f5207b7SJohn Levon 	void *ptr;
12*1f5207b7SJohn Levon 	int i = 0;
13*1f5207b7SJohn Levon 
14*1f5207b7SJohn Levon 	node->text = g_strdup_printf("%s %s:", node->text, name);
15*1f5207b7SJohn Levon 	FOR_EACH_PTR(ptrlist, ptr) {
16*1f5207b7SJohn Levon 		char *index = g_strdup_printf("%d: ", i++);
17*1f5207b7SJohn Levon 		ast_append_child(node, index, ptr, inspect);
18*1f5207b7SJohn Levon 	} END_FOR_EACH_PTR(ptr);
19*1f5207b7SJohn Levon }
20*1f5207b7SJohn Levon 
21*1f5207b7SJohn Levon 
statement_type_name(enum statement_type type)22*1f5207b7SJohn Levon static const char *statement_type_name(enum statement_type type)
23*1f5207b7SJohn Levon {
24*1f5207b7SJohn Levon 	static const char *statement_type_name[] = {
25*1f5207b7SJohn Levon 		[STMT_NONE] = "STMT_NONE",
26*1f5207b7SJohn Levon 		[STMT_DECLARATION] = "STMT_DECLARATION",
27*1f5207b7SJohn Levon 		[STMT_EXPRESSION] = "STMT_EXPRESSION",
28*1f5207b7SJohn Levon 		[STMT_COMPOUND] = "STMT_COMPOUND",
29*1f5207b7SJohn Levon 		[STMT_IF] = "STMT_IF",
30*1f5207b7SJohn Levon 		[STMT_RETURN] = "STMT_RETURN",
31*1f5207b7SJohn Levon 		[STMT_CASE] = "STMT_CASE",
32*1f5207b7SJohn Levon 		[STMT_SWITCH] = "STMT_SWITCH",
33*1f5207b7SJohn Levon 		[STMT_ITERATOR] = "STMT_ITERATOR",
34*1f5207b7SJohn Levon 		[STMT_LABEL] = "STMT_LABEL",
35*1f5207b7SJohn Levon 		[STMT_GOTO] = "STMT_GOTO",
36*1f5207b7SJohn Levon 		[STMT_ASM] = "STMT_ASM",
37*1f5207b7SJohn Levon 		[STMT_CONTEXT] = "STMT_CONTEXT",
38*1f5207b7SJohn Levon 		[STMT_RANGE] = "STMT_RANGE",
39*1f5207b7SJohn Levon 	};
40*1f5207b7SJohn Levon 	return statement_type_name[type] ?: "UNKNOWN_STATEMENT_TYPE";
41*1f5207b7SJohn Levon }
42*1f5207b7SJohn Levon 
inspect_statement(AstNode * node)43*1f5207b7SJohn Levon void inspect_statement(AstNode *node)
44*1f5207b7SJohn Levon {
45*1f5207b7SJohn Levon 	struct statement *stmt = node->ptr;
46*1f5207b7SJohn Levon 	node->text = g_strdup_printf("%s %s:", node->text, statement_type_name(stmt->type));
47*1f5207b7SJohn Levon 	switch (stmt->type) {
48*1f5207b7SJohn Levon 		case STMT_COMPOUND:
49*1f5207b7SJohn Levon 			ast_append_child(node, "stmts:", stmt->stmts, inspect_statement_list);
50*1f5207b7SJohn Levon 			break;
51*1f5207b7SJohn Levon 		case STMT_EXPRESSION:
52*1f5207b7SJohn Levon 			ast_append_child(node, "expression:", stmt->expression, inspect_expression);
53*1f5207b7SJohn Levon 			break;
54*1f5207b7SJohn Levon 		case STMT_IF:
55*1f5207b7SJohn Levon 			ast_append_child(node, "conditional:", stmt->if_conditional, inspect_expression);
56*1f5207b7SJohn Levon 			ast_append_child(node, "if_true:", stmt->if_true, inspect_statement);
57*1f5207b7SJohn Levon 			ast_append_child(node, "if_false:", stmt->if_false, inspect_statement);
58*1f5207b7SJohn Levon 			break;
59*1f5207b7SJohn Levon 		case STMT_ITERATOR:
60*1f5207b7SJohn Levon 			ast_append_child(node, "break:", stmt->iterator_break, inspect_symbol);
61*1f5207b7SJohn Levon 			ast_append_child(node, "continue:", stmt->iterator_continue, inspect_symbol);
62*1f5207b7SJohn Levon 			ast_append_child(node, "pre_statement:", stmt->iterator_pre_statement,
63*1f5207b7SJohn Levon 					 inspect_statement);
64*1f5207b7SJohn Levon 			ast_append_child(node, "statement:", stmt->iterator_statement,
65*1f5207b7SJohn Levon 					 inspect_statement);
66*1f5207b7SJohn Levon 			ast_append_child(node, "post_statement:", stmt->iterator_post_statement,
67*1f5207b7SJohn Levon 					 inspect_statement);
68*1f5207b7SJohn Levon 			break;
69*1f5207b7SJohn Levon 
70*1f5207b7SJohn Levon 		case STMT_SWITCH:
71*1f5207b7SJohn Levon 			ast_append_child(node, "switch_expression:", stmt->switch_expression, inspect_expression);
72*1f5207b7SJohn Levon 			ast_append_child(node, "switch_statement:", stmt->switch_statement, inspect_statement);
73*1f5207b7SJohn Levon 			ast_append_child(node, "switch_break:", stmt->switch_break, inspect_symbol);
74*1f5207b7SJohn Levon 			ast_append_child(node, "switch_case:", stmt->switch_case, inspect_symbol);
75*1f5207b7SJohn Levon 			break;
76*1f5207b7SJohn Levon 		case STMT_CASE:
77*1f5207b7SJohn Levon 			ast_append_child(node, "case_expression:", stmt->case_expression, inspect_expression);
78*1f5207b7SJohn Levon 			ast_append_child(node, "case_to:", stmt->case_to, inspect_expression);
79*1f5207b7SJohn Levon 			ast_append_child(node, "case_statement:", stmt->case_statement, inspect_statement);
80*1f5207b7SJohn Levon 			ast_append_child(node, "case_label:", stmt->case_label, inspect_symbol);
81*1f5207b7SJohn Levon 			break;
82*1f5207b7SJohn Levon 		case STMT_RETURN:
83*1f5207b7SJohn Levon 			ast_append_child(node, "ret_value:", stmt->ret_value, inspect_expression);
84*1f5207b7SJohn Levon 			ast_append_child(node, "ret_target:", stmt->ret_target, inspect_symbol);
85*1f5207b7SJohn Levon 			break;
86*1f5207b7SJohn Levon 
87*1f5207b7SJohn Levon 		default:
88*1f5207b7SJohn Levon 			break;
89*1f5207b7SJohn Levon 	}
90*1f5207b7SJohn Levon }
91*1f5207b7SJohn Levon 
92*1f5207b7SJohn Levon 
inspect_statement_list(AstNode * node)93*1f5207b7SJohn Levon void inspect_statement_list(AstNode *node)
94*1f5207b7SJohn Levon {
95*1f5207b7SJohn Levon 	inspect_ptr_list(node, "statement_list", inspect_statement);
96*1f5207b7SJohn Levon }
97*1f5207b7SJohn Levon 
98*1f5207b7SJohn Levon 
symbol_type_name(enum type type)99*1f5207b7SJohn Levon static const char *symbol_type_name(enum type type)
100*1f5207b7SJohn Levon {
101*1f5207b7SJohn Levon 	static const char *type_name[] = {
102*1f5207b7SJohn Levon 		[SYM_UNINITIALIZED] = "SYM_UNINITIALIZED",
103*1f5207b7SJohn Levon 		[SYM_PREPROCESSOR] = "SYM_PREPROCESSOR",
104*1f5207b7SJohn Levon 		[SYM_BASETYPE] = "SYM_BASETYPE",
105*1f5207b7SJohn Levon 		[SYM_NODE] = "SYM_NODE",
106*1f5207b7SJohn Levon 		[SYM_PTR] = "SYM_PTR",
107*1f5207b7SJohn Levon 		[SYM_FN] = "SYM_FN",
108*1f5207b7SJohn Levon 		[SYM_ARRAY] = "SYM_ARRAY",
109*1f5207b7SJohn Levon 		[SYM_STRUCT] = "SYM_STRUCT",
110*1f5207b7SJohn Levon 		[SYM_UNION] = "SYM_UNION",
111*1f5207b7SJohn Levon 		[SYM_ENUM] = "SYM_ENUM",
112*1f5207b7SJohn Levon 		[SYM_TYPEDEF] = "SYM_TYPEDEF",
113*1f5207b7SJohn Levon 		[SYM_TYPEOF] = "SYM_TYPEOF",
114*1f5207b7SJohn Levon 		[SYM_MEMBER] = "SYM_MEMBER",
115*1f5207b7SJohn Levon 		[SYM_BITFIELD] = "SYM_BITFIELD",
116*1f5207b7SJohn Levon 		[SYM_LABEL] = "SYM_LABEL",
117*1f5207b7SJohn Levon 		[SYM_RESTRICT] = "SYM_RESTRICT",
118*1f5207b7SJohn Levon 		[SYM_FOULED] = "SYM_FOULED",
119*1f5207b7SJohn Levon 		[SYM_KEYWORD] = "SYM_KEYWORD",
120*1f5207b7SJohn Levon 		[SYM_BAD] = "SYM_BAD",
121*1f5207b7SJohn Levon 	};
122*1f5207b7SJohn Levon 	return type_name[type] ?: "UNKNOWN_TYPE";
123*1f5207b7SJohn Levon }
124*1f5207b7SJohn Levon 
125*1f5207b7SJohn Levon 
inspect_symbol(AstNode * node)126*1f5207b7SJohn Levon void inspect_symbol(AstNode *node)
127*1f5207b7SJohn Levon {
128*1f5207b7SJohn Levon 	struct symbol *sym = node->ptr;
129*1f5207b7SJohn Levon 	node->text = g_strdup_printf("%s %s: %s", node->text, symbol_type_name(sym->type),
130*1f5207b7SJohn Levon 				      builtin_typename(sym) ?: show_ident(sym->ident));
131*1f5207b7SJohn Levon 	ast_append_child(node, "ctype.base_type:", sym->ctype.base_type,inspect_symbol);
132*1f5207b7SJohn Levon 
133*1f5207b7SJohn Levon 	switch (sym->namespace) {
134*1f5207b7SJohn Levon 		case NS_PREPROCESSOR:
135*1f5207b7SJohn Levon 			break;
136*1f5207b7SJohn Levon 		default:
137*1f5207b7SJohn Levon 			ast_append_child(node, "arguments:", sym->arguments, inspect_symbol_list);
138*1f5207b7SJohn Levon 			ast_append_child(node, "symbol_list:", sym->symbol_list, inspect_symbol_list);
139*1f5207b7SJohn Levon 			ast_append_child(node, "stmt:", sym->stmt, inspect_statement);
140*1f5207b7SJohn Levon 			break;
141*1f5207b7SJohn Levon 	}
142*1f5207b7SJohn Levon }
143*1f5207b7SJohn Levon 
144*1f5207b7SJohn Levon 
inspect_symbol_list(AstNode * node)145*1f5207b7SJohn Levon void inspect_symbol_list(AstNode *node)
146*1f5207b7SJohn Levon {
147*1f5207b7SJohn Levon 	inspect_ptr_list(node, "symbol_list", inspect_symbol);
148*1f5207b7SJohn Levon }
149*1f5207b7SJohn Levon 
150*1f5207b7SJohn Levon 
expression_type_name(enum expression_type type)151*1f5207b7SJohn Levon static const char *expression_type_name(enum expression_type type)
152*1f5207b7SJohn Levon {
153*1f5207b7SJohn Levon 	static const char *expression_type_name[] = {
154*1f5207b7SJohn Levon 		[EXPR_VALUE] = "EXPR_VALUE",
155*1f5207b7SJohn Levon 		[EXPR_STRING] = "EXPR_STRING",
156*1f5207b7SJohn Levon 		[EXPR_SYMBOL] = "EXPR_SYMBOL",
157*1f5207b7SJohn Levon 		[EXPR_TYPE] = "EXPR_TYPE",
158*1f5207b7SJohn Levon 		[EXPR_BINOP] = "EXPR_BINOP",
159*1f5207b7SJohn Levon 		[EXPR_ASSIGNMENT] = "EXPR_ASSIGNMENT",
160*1f5207b7SJohn Levon 		[EXPR_LOGICAL] = "EXPR_LOGICAL",
161*1f5207b7SJohn Levon 		[EXPR_DEREF] = "EXPR_DEREF",
162*1f5207b7SJohn Levon 		[EXPR_PREOP] = "EXPR_PREOP",
163*1f5207b7SJohn Levon 		[EXPR_POSTOP] = "EXPR_POSTOP",
164*1f5207b7SJohn Levon 		[EXPR_CAST] = "EXPR_CAST",
165*1f5207b7SJohn Levon 		[EXPR_FORCE_CAST] = "EXPR_FORCE_CAST",
166*1f5207b7SJohn Levon 		[EXPR_IMPLIED_CAST] = "EXPR_IMPLIED_CAST",
167*1f5207b7SJohn Levon 		[EXPR_SIZEOF] = "EXPR_SIZEOF",
168*1f5207b7SJohn Levon 		[EXPR_ALIGNOF] = "EXPR_ALIGNOF",
169*1f5207b7SJohn Levon 		[EXPR_PTRSIZEOF] = "EXPR_PTRSIZEOF",
170*1f5207b7SJohn Levon 		[EXPR_CONDITIONAL] = "EXPR_CONDITIONAL",
171*1f5207b7SJohn Levon 		[EXPR_SELECT] = "EXPR_SELECT",
172*1f5207b7SJohn Levon 		[EXPR_STATEMENT] = "EXPR_STATEMENT",
173*1f5207b7SJohn Levon 		[EXPR_CALL] = "EXPR_CALL",
174*1f5207b7SJohn Levon 		[EXPR_COMMA] = "EXPR_COMMA",
175*1f5207b7SJohn Levon 		[EXPR_COMPARE] = "EXPR_COMPARE",
176*1f5207b7SJohn Levon 		[EXPR_LABEL] = "EXPR_LABEL",
177*1f5207b7SJohn Levon 		[EXPR_INITIALIZER] = "EXPR_INITIALIZER",
178*1f5207b7SJohn Levon 		[EXPR_IDENTIFIER] = "EXPR_IDENTIFIER",
179*1f5207b7SJohn Levon 		[EXPR_INDEX] = "EXPR_INDEX",
180*1f5207b7SJohn Levon 		[EXPR_POS] = "EXPR_POS",
181*1f5207b7SJohn Levon 		[EXPR_FVALUE] = "EXPR_FVALUE",
182*1f5207b7SJohn Levon 		[EXPR_SLICE] = "EXPR_SLICE",
183*1f5207b7SJohn Levon 		[EXPR_OFFSETOF] = "EXPR_OFFSETOF",
184*1f5207b7SJohn Levon 	};
185*1f5207b7SJohn Levon 	return expression_type_name[type] ?: "UNKNOWN_EXPRESSION_TYPE";
186*1f5207b7SJohn Levon }
187*1f5207b7SJohn Levon 
inspect_expression(AstNode * node)188*1f5207b7SJohn Levon void inspect_expression(AstNode *node)
189*1f5207b7SJohn Levon {
190*1f5207b7SJohn Levon 	struct expression *expr = node->ptr;
191*1f5207b7SJohn Levon 	node->text = g_strdup_printf("%s %s", node->text, expression_type_name(expr->type));
192*1f5207b7SJohn Levon 	switch (expr->type) {
193*1f5207b7SJohn Levon 		case EXPR_STATEMENT:
194*1f5207b7SJohn Levon 			ast_append_child(node, "statement:", expr->statement, inspect_statement);
195*1f5207b7SJohn Levon 			break;
196*1f5207b7SJohn Levon 		case EXPR_BINOP:
197*1f5207b7SJohn Levon 		case EXPR_COMMA:
198*1f5207b7SJohn Levon 		case EXPR_COMPARE:
199*1f5207b7SJohn Levon 		case EXPR_LOGICAL:
200*1f5207b7SJohn Levon 		case EXPR_ASSIGNMENT:
201*1f5207b7SJohn Levon 			ast_append_child(node, "left:", expr->left, inspect_expression);
202*1f5207b7SJohn Levon 			ast_append_child(node, "right:", expr->right, inspect_expression);
203*1f5207b7SJohn Levon 			break;
204*1f5207b7SJohn Levon 
205*1f5207b7SJohn Levon 		case EXPR_CAST:
206*1f5207b7SJohn Levon 		case EXPR_FORCE_CAST:
207*1f5207b7SJohn Levon 		case EXPR_IMPLIED_CAST:
208*1f5207b7SJohn Levon 			ast_append_child(node, "cast_type:", expr->cast_type, inspect_symbol);
209*1f5207b7SJohn Levon 			ast_append_child(node, "cast_expression:", expr->cast_expression, inspect_expression);
210*1f5207b7SJohn Levon 			break;
211*1f5207b7SJohn Levon 
212*1f5207b7SJohn Levon 		case EXPR_PREOP:
213*1f5207b7SJohn Levon 			ast_append_child(node, "unop:", expr->unop, inspect_expression);
214*1f5207b7SJohn Levon 			break;
215*1f5207b7SJohn Levon 
216*1f5207b7SJohn Levon 		default:
217*1f5207b7SJohn Levon 			break;
218*1f5207b7SJohn Levon 	}
219*1f5207b7SJohn Levon }
220*1f5207b7SJohn Levon 
221*1f5207b7SJohn Levon 
222*1f5207b7SJohn Levon 
223