3 * Copyright (C) 2010, University of Copenhagen DIKU and INRIA.
4 * Copyright (C) 2007, 2008, 2009 Ecole des Mines de Nantes
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License (GPL)
8 * version 2 as published by the Free Software Foundation.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * file license.txt for more details.
17 (*****************************************************************************)
19 (*****************************************************************************)
20 let pr2, pr2_once
= Common.mk_pr2_wrappers
Flag_parsing_c.verbose_parsing
22 (*****************************************************************************)
24 (*****************************************************************************)
26 (* todo?: al_expr doit enlever les infos de type ? et doit remettre en
29 No! Keeping the type information is important to ensuring that variables
30 of different type and declared in different places do not seem to match
31 each other. On the other hand, we don't want to keep around the
32 information about whether the expression is a test expression, because a
33 term that is a test expression should match one that is not. The test
34 information is only useful for matching to the CTL.
38 (* drop all info information *)
40 let strip_info_visitor _
=
43 ref (ty
,Ast_c.NotTest
) in
45 { Visitor_c.default_visitor_c_s
with
47 (* traversal should be deterministic... *)
50 function i
-> ctr := !ctr + 1; Ast_c.al_info_cpp
!ctr i
));
52 Visitor_c.kexpr_s
= (fun (k
,_
) e
->
53 let (e'
, ty
), ii'
= k e
in
54 (e'
, drop_test ty
), ii'
(* keep type - jll *)
58 Visitor_c.ktype_s = (fun (k,_) ft ->
60 match Ast_c.unwrap_typeC ft' with
61 | Ast_c.TypeName (s,_typ) ->
62 Ast_c.TypeName (s, Ast_c.noTypedefDef()) +> Ast_c.rewrap_typeC ft'
70 let al_expr x
= Visitor_c.vk_expr_s
(strip_info_visitor()) x
71 let al_declaration x
= Visitor_c.vk_decl_s
(strip_info_visitor()) x
72 let al_field x
= Visitor_c.vk_struct_field_s
(strip_info_visitor()) x
73 let al_statement x
= Visitor_c.vk_statement_s
(strip_info_visitor()) x
74 let al_type x
= Visitor_c.vk_type_s
(strip_info_visitor()) x
75 let al_init x
= Visitor_c.vk_ini_s
(strip_info_visitor()) x
76 let al_inits x
= Visitor_c.vk_inis_s
(strip_info_visitor()) x
77 let al_param x
= Visitor_c.vk_param_s
(strip_info_visitor()) x
78 let al_params x
= Visitor_c.vk_params_s
(strip_info_visitor()) x
79 let al_arguments x
= Visitor_c.vk_arguments_s
(strip_info_visitor()) x
80 let al_fields x
= Visitor_c.vk_struct_fields_s
(strip_info_visitor()) x
81 let al_name x
= Visitor_c.vk_name_s
(strip_info_visitor()) x
83 let al_node x
= Visitor_c.vk_node_s
(strip_info_visitor()) x
85 let al_program x
= List.map
(Visitor_c.vk_toplevel_s
(strip_info_visitor())) x
86 let al_ii x
= Visitor_c.vk_ii_s
(strip_info_visitor()) x
91 let strip_inh_info_visitor _
= (* for inherited metavariables *)
97 | Some
(ty,_
) -> Some
(ty,Ast_c.NotLocalVar
) in
98 ref (ty,Ast_c.NotTest
) in
100 { Visitor_c.default_visitor_c_s
with
102 (* traversal should be deterministic... *)
105 function i
-> ctr := !ctr + 1; Ast_c.al_info_cpp
!ctr i
));
107 Visitor_c.kexpr_s
= (fun (k
,_
) e
->
108 let (e'
, ty), ii'
= k e
in
109 (e'
, drop_test_lv ty), ii'
(* keep type - jll *)
113 Visitor_c.ktype_s = (fun (k,_) ft ->
115 match Ast_c.unwrap_typeC ft' with
116 | Ast_c.TypeName (s,_typ) ->
117 Ast_c.TypeName (s, Ast_c.noTypedefDef()) +> Ast_c.rewrap_typeC ft'
125 let al_inh_expr x
= Visitor_c.vk_expr_s
(strip_inh_info_visitor()) x
126 let al_inh_declaration x
= Visitor_c.vk_decl_s
(strip_inh_info_visitor()) x
127 let al_inh_field x
= Visitor_c.vk_struct_field_s
(strip_inh_info_visitor()) x
128 let al_inh_field_list x
=
129 Visitor_c.vk_struct_fields_s
(strip_inh_info_visitor()) x
130 let al_inh_statement x
= Visitor_c.vk_statement_s
(strip_inh_info_visitor()) x
131 let al_inh_type x
= Visitor_c.vk_type_s
(strip_inh_info_visitor()) x
132 let al_inh_init x
= Visitor_c.vk_ini_s
(strip_inh_info_visitor()) x
133 let al_inh_inits x
= Visitor_c.vk_inis_s
(strip_inh_info_visitor()) x
134 let al_inh_arguments x
= Visitor_c.vk_arguments_s
(strip_inh_info_visitor()) x
138 let semi_strip_info_visitor = (* keep position information *)
141 ref (ty,Ast_c.NotTest
) in
143 { Visitor_c.default_visitor_c_s
with
144 Visitor_c.kinfo_s
= (fun (k
,_
) i
-> Ast_c.semi_al_info_cpp i
);
146 Visitor_c.kexpr_s
= (fun (k
,_
) e
->
147 let (e'
, ty),ii'
= k e
in
148 (e'
, drop_test ty), ii'
(* keep type - jll *)
153 let semi_al_expr = Visitor_c.vk_expr_s
semi_strip_info_visitor
154 let semi_al_declaration = Visitor_c.vk_decl_s
semi_strip_info_visitor
155 let semi_al_field = Visitor_c.vk_struct_field_s
semi_strip_info_visitor
156 let semi_al_fields = Visitor_c.vk_struct_fields_s
semi_strip_info_visitor
157 let semi_al_statement = Visitor_c.vk_statement_s
semi_strip_info_visitor
158 let semi_al_type = Visitor_c.vk_type_s
semi_strip_info_visitor
159 let semi_al_init = Visitor_c.vk_ini_s
semi_strip_info_visitor
160 let semi_al_inits = Visitor_c.vk_inis_s
semi_strip_info_visitor
161 let semi_al_param = Visitor_c.vk_param_s
semi_strip_info_visitor
162 let semi_al_params = Visitor_c.vk_params_s
semi_strip_info_visitor
163 let semi_al_arguments = Visitor_c.vk_arguments_s
semi_strip_info_visitor
165 let semi_al_program =
166 List.map
(Visitor_c.vk_toplevel_s
semi_strip_info_visitor)
171 (* really strip, do not keep position nor anything specificities, true
172 * abstracted form. This is used outside coccinelle in Yacfe and aComment *)
173 let real_strip_info_visitor _
=
174 { Visitor_c.default_visitor_c_s
with
175 Visitor_c.kinfo_s
= (fun (k
,_
) i
->
176 Ast_c.real_al_info_cpp i
179 Visitor_c.kexpr_s
= (fun (k
,_
) e
->
180 let (e'
, ty),ii'
= k e
in
181 (e'
, Ast_c.noType
()), ii'
185 Visitor_c.ktype_s = (fun (k,_) ft ->
187 match Ast_c.unwrap_typeC ft' with
188 | Ast_c.TypeName (s,_typ) ->
189 Ast_c.TypeName (s, Ast_c.noTypedefDef()) +> Ast_c.rewrap_typeC ft'
197 let real_al_expr x
= Visitor_c.vk_expr_s
(real_strip_info_visitor()) x
198 let real_al_node x
= Visitor_c.vk_node_s
(real_strip_info_visitor()) x
199 let real_al_type x
= Visitor_c.vk_type_s
(real_strip_info_visitor()) x
200 let real_al_statement x
= Visitor_c.vk_statement_s
(real_strip_info_visitor()) x
201 let real_al_def x
= Visitor_c.vk_toplevel_s
(real_strip_info_visitor()) x
203 (*****************************************************************************)
205 (*****************************************************************************)
207 let extract_info_visitor recursor x
=
208 let globals = ref [] in
211 Visitor_c.default_visitor_c
with
212 Visitor_c.kinfo
= (fun (k
, _
) i
-> Common.push2 i
globals)
219 let ii_of_decl = extract_info_visitor Visitor_c.vk_decl
220 let ii_of_field = extract_info_visitor Visitor_c.vk_struct_field
221 let ii_of_node = extract_info_visitor Visitor_c.vk_node
222 let ii_of_expr = extract_info_visitor Visitor_c.vk_expr
223 let ii_of_stmt = extract_info_visitor Visitor_c.vk_statement
224 let ii_of_args = extract_info_visitor Visitor_c.vk_args_splitted
225 let ii_of_type = extract_info_visitor Visitor_c.vk_type
226 let ii_of_ini = extract_info_visitor Visitor_c.vk_ini
227 let ii_of_inis = extract_info_visitor Visitor_c.vk_inis_splitted
228 let ii_of_param = extract_info_visitor Visitor_c.vk_param
229 let ii_of_params = extract_info_visitor Visitor_c.vk_params_splitted
230 let ii_of_enum_fields = extract_info_visitor Visitor_c.vk_enum_fields_splitted
231 let ii_of_struct_fields = extract_info_visitor Visitor_c.vk_struct_fields
232 (*let ii_of_struct_field = extract_info_visitor Visitor_c.vk_struct_field*)
233 let ii_of_struct_fieldkinds = extract_info_visitor Visitor_c.vk_struct_fieldkinds
234 let ii_of_cst = extract_info_visitor Visitor_c.vk_cst
235 let ii_of_define_params =
236 extract_info_visitor Visitor_c.vk_define_params_splitted
237 let ii_of_toplevel = extract_info_visitor Visitor_c.vk_toplevel
239 (*****************************************************************************)
241 (*****************************************************************************)
242 let max_min_ii_by_pos xs
=
244 | [] -> failwith
"empty list, max_min_ii_by_pos"
247 let pos_leq p1 p2
= (Ast_c.compare_pos p1 p2
) =|= (-1) in
248 xs
+> List.fold_left
(fun (maxii
,minii
) e
->
249 let maxii'
= if pos_leq maxii e
then e
else maxii in
250 let minii'
= if pos_leq e
minii then e
else minii in
254 let info_to_fixpos ii
=
255 match Ast_c.pinfo_of_info ii
with
256 Ast_c.OriginTok pi
-> Ast_cocci.Real pi
.Common.charpos
257 | Ast_c.ExpandedTok
(_
,(pi
,offset
)) ->
258 Ast_cocci.Virt
(pi
.Common.charpos
,offset
)
259 | Ast_c.FakeTok
(_
,(pi
,offset
)) ->
260 Ast_cocci.Virt
(pi
.Common.charpos
,offset
)
261 | Ast_c.AbstractLineTok pi
-> failwith
"unexpected abstract"
263 let max_min_by_pos xs
=
264 let (i1
, i2
) = max_min_ii_by_pos xs
in
265 (info_to_fixpos i1
, info_to_fixpos i2
)
267 let lin_col_by_pos xs
=
268 (* put min before max; no idea why they are backwards above *)
269 let non_fake = List.filter
(function ii
-> not
(Ast_c.is_fake ii
)) xs
in
270 let (i2
, i1
) = max_min_ii_by_pos non_fake in
271 let posf x
= Ast_c.col_of_info x
in
272 let mposf x
= Ast_c.col_of_info x
+ String.length
(Ast_c.str_of_info x
) in
273 (Ast_c.file_of_info i1
,!Flag.current_element
,
274 (Ast_c.line_of_info i1
, posf i1
), (Ast_c.line_of_info i2
, mposf i2
))
280 let min_pinfo_of_node node
=
281 let ii = ii_of_node node
in
282 let (maxii, minii) = max_min_ii_by_pos ii in
283 Ast_c.parse_info_of_info
minii
286 let (range_of_origin_ii
: Ast_c.info list
-> (int * int) option) =
288 let ii = List.filter
Ast_c.is_origintok
ii in
290 let (max
, min
) = max_min_ii_by_pos ii in
291 assert(Ast_c.is_origintok max
);
292 assert(Ast_c.is_origintok min
);
293 let strmax = Ast_c.str_of_info max
in
295 (Ast_c.pos_of_info min
, Ast_c.pos_of_info max
+ String.length
strmax)
300 (*****************************************************************************)
302 (*****************************************************************************)
304 let names_of_parameters_in_def def
=
305 match def
.Ast_c.f_old_c_style
with
307 pr2_once
"names_of_parameters_in_def: f_old_c_style not handled";
310 let ftyp = def
.Ast_c.f_type
in
311 let (ret
, (params
, bwrap
)) = ftyp in
312 params
+> Common.map_filter
(fun (param
,ii) ->
313 Ast_c.name_of_parameter param
316 let names_of_parameters_in_macro xs
=
317 xs
+> List.map
(fun (xx
, ii) ->
324 (* only used in ast_to_flow, so move it ? *)
325 let rec stmt_elems_of_sequencable xs
=
326 xs
+> Common.map
(fun x
->
328 | Ast_c.StmtElem e
-> [e
]
329 | Ast_c.CppDirectiveStmt _
332 pr2_once
("stmt_elems_of_sequencable: filter a directive");
334 | Ast_c.IfdefStmt2
(_ifdef
, xxs
) ->
335 pr2 ("stmt_elems_of_sequencable: IfdefStm2 TODO?");
336 xxs
+> List.map
(fun xs
->
337 let xs'
= stmt_elems_of_sequencable xs in