(* * Copyright 2010, INRIA, University of Copenhagen * Julia Lawall, Rene Rydhof Hansen, Gilles Muller, Nicolas Palix * Copyright 2005-2009, Ecole des Mines de Nantes, University of Copenhagen * Yoann Padioleau, Julia Lawall, Rene Rydhof Hansen, Henrik Stuart, Gilles Muller, Nicolas Palix * This file is part of Coccinelle. * * Coccinelle is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, according to version 2 of the License. * * Coccinelle is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Coccinelle. If not, see . * * The authors reserve the right to distribute this or future versions of * Coccinelle under other licenses. *) module Ast = Ast_cocci module V = Visitor_ast let disjmult2 e1 e2 k = List.concat (List.map (function e1 -> List.map (function e2 -> k e1 e2) e2) e1) let disjmult3 e1 e2 e3 k = List.concat (List.map (function e1 -> List.concat (List.map (function e2 -> List.map (function e3 -> k e1 e2 e3) e3) e2)) e1) let rec disjmult f = function [] -> [[]] | x::xs -> let cur = f x in let rest = disjmult f xs in disjmult2 cur rest (function cur -> function rest -> cur :: rest) let disjoption f = function None -> [None] | Some x -> List.map (function x -> Some x) (f x) let disjdots f d = match Ast.unwrap d with Ast.DOTS(l) -> List.map (function l -> Ast.rewrap d (Ast.DOTS(l))) (disjmult f l) | Ast.CIRCLES(l) -> List.map (function l -> Ast.rewrap d (Ast.CIRCLES(l))) (disjmult f l) | Ast.STARS(l) -> List.map (function l -> Ast.rewrap d (Ast.STARS(l))) (disjmult f l) let rec disjty ft = match Ast.unwrap ft with Ast.Type(cv,ty) -> let ty = disjtypeC ty in List.map (function ty -> Ast.rewrap ft (Ast.Type(cv,ty))) ty | Ast.DisjType(types) -> List.concat (List.map disjty types) | Ast.OptType(ty) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap ft (Ast.OptType(ty))) ty | Ast.UniqueType(ty) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap ft (Ast.UniqueType(ty))) ty and disjtypeC bty = match Ast.unwrap bty with Ast.BaseType(_) | Ast.SignedT(_,_) -> [bty] | Ast.Pointer(ty,star) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap bty (Ast.Pointer(ty,star))) ty | Ast.FunctionPointer(ty,lp1,star,rp1,lp2,params,rp2) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap bty (Ast.FunctionPointer(ty,lp1,star,rp1,lp2,params,rp2))) ty | Ast.FunctionType (s,ty,lp1,params,rp1) -> let ty = disjoption disjty ty in List.map (function ty -> Ast.rewrap bty (Ast.FunctionType (s,ty,lp1,params,rp1))) ty | Ast.Array(ty,lb,size,rb) -> disjmult2 (disjty ty) (disjoption disjexp size) (function ty -> function size -> Ast.rewrap bty (Ast.Array(ty,lb,size,rb))) | Ast.EnumName(_,_) | Ast.StructUnionName(_,_) -> [bty] | Ast.EnumDef(ty,lb,ids,rb) -> disjmult2 (disjty ty) (disjdots disjexp ids) (function ty -> function ids -> Ast.rewrap bty (Ast.EnumDef(ty,lb,ids,rb))) | Ast.StructUnionDef(ty,lb,decls,rb) -> disjmult2 (disjty ty) (disjdots disjdecl decls) (function ty -> function decls -> Ast.rewrap bty (Ast.StructUnionDef(ty,lb,decls,rb))) | Ast.TypeName(_) | Ast.MetaType(_,_,_) -> [bty] and disjident e = match Ast.unwrap e with Ast.DisjId(id_list) -> List.concat (List.map disjident id_list) | Ast.OptIdent(id) -> let id = disjident id in List.map (function id -> Ast.rewrap e (Ast.OptIdent(id))) id | Ast.UniqueIdent(id) -> let id = disjident id in List.map (function id -> Ast.rewrap e (Ast.UniqueIdent(id))) id | _ -> [e] and disjexp e = match Ast.unwrap e with Ast.Ident(_) | Ast.Constant(_) -> [e] (* even Ident can't contain disj *) | Ast.FunCall(fn,lp,args,rp) -> disjmult2 (disjexp fn) (disjdots disjexp args) (function fn -> function args -> Ast.rewrap e (Ast.FunCall(fn,lp,args,rp))) | Ast.Assignment(left,op,right,simple) -> disjmult2 (disjexp left) (disjexp right) (function left -> function right -> Ast.rewrap e (Ast.Assignment(left,op,right,simple))) | Ast.CondExpr(exp1,why,Some exp2,colon,exp3) -> let res = disjmult disjexp [exp1;exp2;exp3] in List.map (function [exp1;exp2;exp3] -> Ast.rewrap e (Ast.CondExpr(exp1,why,Some exp2,colon,exp3)) | _ -> failwith "not possible") res | Ast.CondExpr(exp1,why,None,colon,exp3) -> disjmult2 (disjexp exp1) (disjexp exp3) (function exp1 -> function exp3 -> Ast.rewrap e (Ast.CondExpr(exp1,why,None,colon,exp3))) | Ast.Postfix(exp,op) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.Postfix(exp,op))) exp | Ast.Infix(exp,op) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.Infix(exp,op))) exp | Ast.Unary(exp,op) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.Unary(exp,op))) exp | Ast.Binary(left,op,right) -> disjmult2 (disjexp left) (disjexp right) (function left -> function right -> Ast.rewrap e (Ast.Binary(left,op,right))) | Ast.Nested(exp,op,right) -> (* disj not possible in right *) let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.Nested(exp,op,right))) exp | Ast.Paren(lp,exp,rp) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.Paren(lp,exp,rp))) exp | Ast.ArrayAccess(exp1,lb,exp2,rb) -> disjmult2 (disjexp exp1) (disjexp exp2) (function exp1 -> function exp2 -> Ast.rewrap e (Ast.ArrayAccess(exp1,lb,exp2,rb))) | Ast.RecordAccess(exp,pt,field) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.RecordAccess(exp,pt,field))) exp | Ast.RecordPtAccess(exp,ar,field) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.RecordPtAccess(exp,ar,field))) exp | Ast.Cast(lp,ty,rp,exp) -> disjmult2 (disjty ty) (disjexp exp) (function ty -> function exp -> Ast.rewrap e (Ast.Cast(lp,ty,rp,exp))) | Ast.SizeOfExpr(szf,exp) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.SizeOfExpr(szf,exp))) exp | Ast.SizeOfType(szf,lp,ty,rp) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap e (Ast.SizeOfType(szf,lp,ty,rp))) ty | Ast.TypeExp(ty) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap e (Ast.TypeExp(ty))) ty | Ast.MetaErr(_,_,_,_) | Ast.MetaExpr(_,_,_,_,_,_) | Ast.MetaExprList(_,_,_,_) | Ast.EComma(_) -> [e] | Ast.DisjExpr(exp_list) -> List.concat (List.map disjexp exp_list) | Ast.NestExpr(starter,expr_dots,ender,whencode,multi) -> (* not sure what to do here, so ambiguities still possible *) [e] | Ast.Edots(dots,_) | Ast.Ecircles(dots,_) | Ast.Estars(dots,_) -> [e] | Ast.OptExp(exp) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.OptExp(exp))) exp | Ast.UniqueExp(exp) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap e (Ast.UniqueExp(exp))) exp and disjparam p = match Ast.unwrap p with Ast.VoidParam(ty) -> [p] (* void is the only possible value *) | Ast.Param(ty,id) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap p (Ast.Param(ty,id))) ty | Ast.MetaParam(_,_,_) | Ast.MetaParamList(_,_,_,_) | Ast.PComma(_) -> [p] | Ast.Pdots(dots) | Ast.Pcircles(dots) -> [p] | Ast.OptParam(param) -> let param = disjparam param in List.map (function param -> Ast.rewrap p (Ast.OptParam(param))) param | Ast.UniqueParam(param) -> let param = disjparam param in List.map (function param -> Ast.rewrap p (Ast.UniqueParam(param))) param and disjini i = match Ast.unwrap i with Ast.MetaInit(_,_,_) -> [i] | Ast.InitExpr(exp) -> let exp = disjexp exp in List.map (function exp -> Ast.rewrap i (Ast.InitExpr(exp))) exp | Ast.ArInitList(lb,initlist,rb) -> List.map (function initlist -> Ast.rewrap i (Ast.ArInitList(lb,initlist,rb))) (disjdots disjini initlist) | Ast.StrInitList(allminus,lb,initlist,rb,whencode) -> List.map (function initlist -> Ast.rewrap i (Ast.StrInitList(allminus,lb,initlist,rb,whencode))) (disjmult disjini initlist) | Ast.InitGccExt(designators,eq,ini) -> let designators = disjmult designator designators in let ini = disjini ini in disjmult2 designators ini (function designators -> function ini -> Ast.rewrap i (Ast.InitGccExt(designators,eq,ini))) | Ast.InitGccName(name,eq,ini) -> let ini = disjini ini in List.map (function ini -> Ast.rewrap i (Ast.InitGccName(name,eq,ini))) ini | Ast.IComma(comma) -> [i] | Ast.Idots(dots,_) -> [i] | Ast.OptIni(ini) -> let ini = disjini ini in List.map (function ini -> Ast.rewrap i (Ast.OptIni(ini))) ini | Ast.UniqueIni(ini) -> let ini = disjini ini in List.map (function ini -> Ast.rewrap i (Ast.UniqueIni(ini))) ini and designator = function Ast.DesignatorField(dot,id) -> [Ast.DesignatorField(dot,id)] | Ast.DesignatorIndex(lb,exp,rb) -> let exp = disjexp exp in List.map (function exp -> Ast.DesignatorIndex(lb,exp,rb)) exp | Ast.DesignatorRange(lb,min,dots,max,rb) -> disjmult2 (disjexp min) (disjexp max) (function min -> function max -> Ast.DesignatorRange(lb,min,dots,max,rb)) and disjdecl d = match Ast.unwrap d with Ast.MetaDecl(_,_,_) | Ast.MetaField(_,_,_) | Ast.MetaFieldList(_,_,_,_) -> [d] | Ast.Init(stg,ty,id,eq,ini,sem) -> disjmult2 (disjty ty) (disjini ini) (function ty -> function ini -> Ast.rewrap d (Ast.Init(stg,ty,id,eq,ini,sem))) | Ast.UnInit(stg,ty,id,sem) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap d (Ast.UnInit(stg,ty,id,sem))) ty | Ast.MacroDecl(name,lp,args,rp,sem) -> List.map (function args -> Ast.rewrap d (Ast.MacroDecl(name,lp,args,rp,sem))) (disjdots disjexp args) | Ast.TyDecl(ty,sem) -> let ty = disjty ty in List.map (function ty -> Ast.rewrap d (Ast.TyDecl(ty,sem))) ty | Ast.Typedef(stg,ty,id,sem) -> let ty = disjty ty in (* disj not allowed in id *) List.map (function ty -> Ast.rewrap d (Ast.Typedef(stg,ty,id,sem))) ty | Ast.DisjDecl(decls) -> List.concat (List.map disjdecl decls) | Ast.Ddots(_,_) -> [d] | Ast.OptDecl(decl) -> let decl = disjdecl decl in List.map (function decl -> Ast.rewrap d (Ast.OptDecl(decl))) decl | Ast.UniqueDecl(decl) -> let decl = disjdecl decl in List.map (function decl -> Ast.rewrap d (Ast.UniqueDecl(decl))) decl let generic_orify_rule_elem f re exp rebuild = match f exp with [exp] -> re | orexps -> Ast.rewrap re (Ast.DisjRuleElem (List.map rebuild orexps)) let orify_rule_elem re exp rebuild = generic_orify_rule_elem disjexp re exp rebuild let orify_rule_elem_ty = generic_orify_rule_elem disjty let orify_rule_elem_param = generic_orify_rule_elem disjparam let orify_rule_elem_decl = generic_orify_rule_elem disjdecl let orify_rule_elem_ini = generic_orify_rule_elem disjini let rec disj_rule_elem r k re = match Ast.unwrap re with Ast.FunHeader(bef,allminus,fninfo,name,lp,params,rp) -> generic_orify_rule_elem (disjdots disjparam) re params (function params -> Ast.rewrap re (Ast.FunHeader(bef,allminus,fninfo,name,lp,params,rp))) | Ast.Decl(bef,allminus,decl) -> orify_rule_elem_decl re decl (function decl -> Ast.rewrap re (Ast.Decl(bef,allminus,decl))) | Ast.SeqStart(brace) -> re | Ast.SeqEnd(brace) -> re | Ast.ExprStatement(exp,sem) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.ExprStatement(exp,sem))) | Ast.IfHeader(iff,lp,exp,rp) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.IfHeader(iff,lp,exp,rp))) | Ast.Else(els) -> re | Ast.WhileHeader(whl,lp,exp,rp) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.WhileHeader(whl,lp,exp,rp))) | Ast.DoHeader(d) -> re | Ast.WhileTail(whl,lp,exp,rp,sem) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.WhileTail(whl,lp,exp,rp,sem))) | Ast.ForHeader(fr,lp,e1,sem1,e2,sem2,e3,rp) -> generic_orify_rule_elem (disjmult (disjoption disjexp)) re [e1;e2;e3] (function [exp1;exp2;exp3] -> Ast.rewrap re (Ast.ForHeader(fr,lp,exp1,sem1,exp2,sem2,exp3,rp)) | _ -> failwith "not possible") | Ast.IteratorHeader(whl,lp,args,rp) -> generic_orify_rule_elem (disjdots disjexp) re args (function args -> Ast.rewrap re (Ast.IteratorHeader(whl,lp,args,rp))) | Ast.SwitchHeader(switch,lp,exp,rp) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.SwitchHeader(switch,lp,exp,rp))) | Ast.Break(_,_) | Ast.Continue(_,_) | Ast.Label(_,_) | Ast.Goto(_,_,_) | Ast.Return(_,_) -> re | Ast.ReturnExpr(ret,exp,sem) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.ReturnExpr(ret,exp,sem))) | Ast.MetaRuleElem(_,_,_) | Ast.MetaStmt(_,_,_,_) | Ast.MetaStmtList(_,_,_) -> re | Ast.Exp(exp) -> orify_rule_elem re exp (function exp -> Ast.rewrap exp (Ast.Exp(exp))) | Ast.TopExp(exp) -> orify_rule_elem re exp (function exp -> Ast.rewrap exp (Ast.TopExp(exp))) | Ast.Ty(ty) -> orify_rule_elem_ty re ty (function ty -> Ast.rewrap ty (Ast.Ty(ty))) | Ast.TopInit(init) -> orify_rule_elem_ini re init (function init -> Ast.rewrap init (Ast.TopInit(init))) | Ast.Include(inc,s) -> re | Ast.Undef(def,id) -> re | Ast.DefineHeader(def,id,params) -> re | Ast.Default(def,colon) -> re | Ast.Case(case,exp,colon) -> orify_rule_elem re exp (function exp -> Ast.rewrap re (Ast.Case(case,exp,colon))) | Ast.DisjRuleElem(l) -> (* only case lines *) Ast.rewrap re(Ast.DisjRuleElem(List.map (disj_rule_elem r k) l)) let disj_all = let mcode x = x in let donothing r k e = k e in V.rebuilder mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing disj_rule_elem donothing donothing donothing donothing (* ----------------------------------------------------------------------- *) (* collect iso information at the rule_elem level *) let collect_all_isos = let bind = (@) in let option_default = [] in let mcode r x = [] in let donothing r k e = Common.union_set (Ast.get_isos e) (k e) in let doanything r k e = k e in V.combiner bind option_default mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing doanything let collect_iso_info = let mcode x = x in let donothing r k e = k e in let rule_elem r k e = match Ast.unwrap e with Ast.DisjRuleElem(l) -> k e | _ -> let isos = collect_all_isos.V.combiner_rule_elem e in Ast.set_isos e isos in V.rebuilder mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode mcode donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing donothing rule_elem donothing donothing donothing donothing (* ----------------------------------------------------------------------- *) let disj rules = List.map (function (mv,r) -> match r with Ast.ScriptRule _ | Ast.InitialScriptRule _ | Ast.FinalScriptRule _ -> (mv, r) | Ast.CocciRule (nm, rule_info, r, isexp, ruletype) -> let res = List.map (function x -> let res = disj_all.V.rebuilder_top_level x in if !Flag.track_iso_usage then collect_iso_info.V.rebuilder_top_level res else res) r in (mv, Ast.CocciRule (nm,rule_info,res,isexp,ruletype))) rules