package goblint-cil
A front-end for the C programming language that facilitates program analysis and transformation
Install
Dune Dependency
Authors
Maintainers
Sources
goblint-cil-2.0.5.tbz
sha256=bef0769f2322d03248f6e23b447642baec63c72c430c5affd04239a063d5d601
sha512=3d0d4a161515cb9ee51c32eca967fa3b2fbb43b647aab61da2f5cb15fec975eb088ed00a5b9961f1e0b7780773c369509a177639c6630b48d222deb56a795760
doc/src/goblint-cil.dataslicing/dataslicing.ml.html
Source file dataslicing.ml
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 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456
(* Copyright (c) 2004, Jeremy Condit <jcondit@cs.berkeley.edu> George C. Necula <necula@cs.berkeley.edu> All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. The names of the contributors may not be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *) open GoblintCil open Feature module E = Errormsg let debug = false let numRegions : int = 2 let newGlobals : global list ref = ref [] let curFundec : fundec ref = ref dummyFunDec let curLocation : location ref = ref locUnknown let applyOption (fn : 'a -> 'b) (ao : 'a option) : 'b option = match ao with | Some a -> Some (fn a) | None -> None let getRegion (attrs : attributes) : int = try match List.hd (filterAttributes "region" attrs) with | Attr (_, [AInt i]) -> i | _ -> E.s (bug "bad region attribute") with Failure _ -> 1 let checkRegion (i : int) (attrs : attributes) : bool = (getRegion attrs) = i let regionField (i : int) : string = "r" ^ (string_of_int i) let regionStruct (i : int) (name : string) : string = name ^ "_r" ^ (string_of_int i) let foldRegions (fn : int -> 'a -> 'a) (base : 'a) : 'a = let rec helper (i : int) : 'a = if i <= numRegions then fn i (helper (i + 1)) else base in helper 1 let rec getTypeName (t : typ) : string = match t with | TVoid _ -> "void" | TInt _ -> "int" | TFloat _ -> "float" | TComp (cinfo, _) -> "comp_" ^ cinfo.cname | TNamed (tinfo, _) -> "td_" ^ tinfo.tname | TPtr (bt, _) -> "ptr_" ^ (getTypeName bt) | TArray (bt, _, _) -> "array_" ^ (getTypeName bt) | TFun _ -> "fn" | _ -> E.s (unimp "typename") let isAllocFunction (fn : exp) : bool = match fn with | Lval (Var vinfo, NoOffset) when vinfo.vname = "malloc" -> true | _ -> false let isExternalFunction (fn : exp) : bool = match fn with | Lval (Var vinfo, NoOffset) when vinfo.vstorage = Extern -> true | _ -> false let types : (int * typsig, typ) Hashtbl.t = Hashtbl.create 113 let typeInfos : (int * string, typeinfo) Hashtbl.t = Hashtbl.create 113 let compInfos : (int * int, compinfo) Hashtbl.t = Hashtbl.create 113 let varTypes : (typsig, typ) Hashtbl.t = Hashtbl.create 113 let varCompInfos : (typsig, compinfo) Hashtbl.t = Hashtbl.create 113 let rec sliceCompInfo (i : int) (cinfo : compinfo) : compinfo = try Hashtbl.find compInfos (i, cinfo.ckey) with Not_found -> mkCompInfo cinfo.cstruct (regionStruct i cinfo.cname) (fun cinfo' -> Hashtbl.add compInfos (i, cinfo.ckey) cinfo'; List.fold_right (fun finfo rest -> let t = sliceType i finfo.ftype in if not (isVoidType t) then (finfo.fname, t, finfo.fbitfield, finfo.fattr, finfo.floc) :: rest else rest) cinfo.cfields []) cinfo.cattr and sliceTypeInfo (i : int) (tinfo : typeinfo) : typeinfo = try Hashtbl.find typeInfos (i, tinfo.tname) with Not_found -> let result = { tinfo with tname = regionStruct i tinfo.tname; ttype = sliceType i tinfo.ttype; } in Hashtbl.add typeInfos (i, tinfo.tname) result; result and sliceType (i : int) (t : typ) : typ = let ts = typeSig t in try Hashtbl.find types (i, ts) with Not_found -> let result = match t with | TVoid _ -> t | TInt (_, attrs) -> if checkRegion i attrs then t else TVoid [] | TFloat (_, attrs) -> if checkRegion i attrs then t else TVoid [] | TComp (cinfo, attrs) -> TComp (sliceCompInfo i cinfo, attrs) | TNamed (tinfo, attrs) -> TNamed (sliceTypeInfo i tinfo, attrs) | TPtr (TVoid _, _) -> t (* Avoid discarding void*. *) | TPtr (bt, attrs) -> let bt' = sliceType i bt in if not (isVoidType bt') then TPtr (bt', attrs) else TVoid [] | TArray (bt, eo, attrs) -> TArray (sliceType i bt, applyOption (sliceExp 1) eo, attrs) | TFun (ret, args, va, attrs) -> if checkRegion i attrs then TFun (sliceTypeAll ret, applyOption (Util.list_map (fun (aname, atype, aattrs) -> (aname, sliceTypeAll atype, aattrs))) args, va, attrs) else TVoid [] | TBuiltin_va_list _ -> t | _ -> E.s (unimp "type %a" d_type t) in Hashtbl.add types (i, ts) result; result and sliceTypeAll (t : typ) : typ = begin match t with | TComp (cinfo, _) when hasAttribute "var_type_sliced" cinfo.cattr -> E.s (bug "tried to slice twice") | _ -> () end; let ts = typeSig t in try Hashtbl.find varTypes ts with Not_found -> let cinfo = let name = ("var_" ^ (getTypeName t)) in if debug then ignore (E.log "creating %s\n" name); try Hashtbl.find varCompInfos ts with Not_found -> mkCompInfo true name (fun cinfo -> Hashtbl.add varCompInfos ts cinfo; foldRegions (fun i rest -> let t' = sliceType i t in if not (isVoidType t') then (regionField i, t', None, [], !curLocation) :: rest else rest) []) [Attr ("var_type_sliced", [])] in let t' = if List.length cinfo.cfields > 1 then begin newGlobals := GCompTag (cinfo, !curLocation) :: !newGlobals; TComp (cinfo, []) end else t in Hashtbl.add varTypes ts t'; t' and sliceLval (i : int) (lv : lval) : lval = if debug then ignore (E.log "lval %a\n" d_lval lv); let lh, offset = lv in match lh with | Var vinfo -> let t = sliceTypeAll vinfo.vtype in let offset' = match t with | TComp (cinfo, _) when hasAttribute "var_type_sliced" cinfo.cattr -> Field (getCompField cinfo (regionField i), offset) | _ -> offset in Var vinfo, offset' | Mem e -> Mem (sliceExp i e), offset and sliceExp (i : int) (e : exp) : exp = if debug then ignore (E.log "exp %a\n" d_exp e); match e with | Const c -> Const c | Lval lv -> Lval (sliceLval i lv) | UnOp (op, e1, t) -> UnOp (op, sliceExp i e1, sliceType i t) | BinOp (op, e1, e2, t) -> BinOp (op, sliceExp i e1, sliceExp i e2, sliceType i t) | CastE (t, e) -> sliceCast i t e | AddrOf lv -> AddrOf (sliceLval i lv) | StartOf lv -> StartOf (sliceLval i lv) | SizeOf t -> SizeOf (sliceTypeAll t) | _ -> E.s (unimp "exp %a" d_exp e) and sliceCast (i : int) (t : typ) (e : exp) : exp = let te = typeOf e in match t, te with | TInt (k1, _), TInt (k2, attrs2) when k1 = k2 -> (* Note: We strip off integer cast operations. *) sliceExp (getRegion attrs2) e | TInt (k1, _), TPtr _ -> (* Note: We strip off integer cast operations. *) sliceExp i e | TPtr _, _ when isZero e -> CastE (sliceType i t, sliceExp i e) | TPtr (bt1, _), TPtr (bt2, _) when (typeSig bt1) = (typeSig bt2) -> CastE (sliceType i t, sliceExp i e) | _ -> E.s (unimp "sketchy cast (%a) -> (%a)\n" d_type te d_type t) and sliceExpAll (e : exp) (l : location) : instr list * exp = let t = typeOf e in let t' = sliceTypeAll t in match t' with | TComp (cinfo, _) when hasAttribute "var_type_sliced" cinfo.cattr -> let vinfo = makeTempVar !curFundec t in let instrs = foldRegions (fun i rest -> try let finfo = getCompField cinfo (regionField i) in if not (isVoidType finfo.ftype) then Set ((Var vinfo, Field (finfo, NoOffset)), sliceExp i e, l, locUnknown) :: rest (* TODO: better eloc? *) else rest with Not_found -> rest) [] in instrs, Lval (var vinfo) | _ -> [], sliceExp 1 e let sliceVar (vinfo : varinfo) : unit = if hasAttribute "var_sliced" vinfo.vattr then E.s (bug "tried to slice a var twice"); let t = sliceTypeAll vinfo.vtype in if debug then ignore (E.log "setting %s type to %a\n" vinfo.vname d_type t); vinfo.vattr <- addAttribute (Attr ("var_sliced", [])) vinfo.vattr; vinfo.vtype <- t let sliceInstr (inst : instr) : instr list = match inst with | Set (lv, e, loc, eloc) -> if debug then ignore (E.log "set %a %a\n" d_lval lv d_exp e); let t = typeOfLval lv in foldRegions (fun i rest -> if not (isVoidType (sliceType i t)) then Set (sliceLval i lv, sliceExp i e, loc, eloc) :: rest else rest) [] | Call (ret, fn, args, l, eloc) when isAllocFunction fn -> let lv = match ret with | Some lv -> lv | None -> E.s (bug "malloc call has no return lval") in let t = typeOfLval lv in foldRegions (fun i rest -> if not (isVoidType (sliceType i t)) then Call (Some (sliceLval i lv), sliceExp 1 fn, Util.list_map (sliceExp i) args, l, eloc) :: rest else rest) [] | Call (ret, fn, args, l, el) when isExternalFunction fn -> [Call (applyOption (sliceLval 1) ret, sliceExp 1 fn, Util.list_map (sliceExp 1) args, l, el)] | Call (ret, fn, args, l, el) -> let ret', set = match ret with | Some lv -> let vinfo = makeTempVar !curFundec (typeOfLval lv) in Some (var vinfo), [Set (lv, Lval (var vinfo), l, el)] | None -> None, [] in let instrs, args' = List.fold_right (fun arg (restInstrs, restArgs) -> let instrs, arg' = sliceExpAll arg l in instrs @ restInstrs, (arg' :: restArgs)) args ([], []) in instrs @ (Call (ret', sliceExp 1 fn, args', l, el) :: set) | _ -> E.s (unimp "inst %a" d_instr inst) let sliceReturnExp (eo : exp option) (l : location) (el : location) : stmtkind = match eo with | Some e -> begin match sliceExpAll e l with | [], e' -> Return (Some e', l, el) | instrs, e' -> Block (mkBlock [mkStmt (Instr instrs); mkStmt (Return (Some e', l, el))]) end | None -> Return (None, l, el) let rec sliceStmtKind (sk : stmtkind) : stmtkind = match sk with | Instr instrs -> Instr (List.flatten (Util.list_map sliceInstr instrs)) | Block b -> Block (sliceBlock b) | If (e, b1, b2, l, el) -> If (sliceExp 1 e, sliceBlock b1, sliceBlock b2, l, el) | Break l -> Break l | Continue l -> Continue l | Return (eo, l, el) -> sliceReturnExp eo l el | Switch (e, b, sl, l, el) -> Switch (sliceExp 1 e, sliceBlock b, Util.list_map sliceStmt sl, l, el) | Loop (b, l, el, so1, so2) -> Loop (sliceBlock b, l, el, applyOption sliceStmt so1, applyOption sliceStmt so2) | Goto _ -> sk | _ -> E.s (unimp "statement") and sliceStmt (s : stmt) : stmt = (* Note: We update statements destructively so that goto/switch work. *) s.skind <- sliceStmtKind s.skind; s and sliceBlock (b : block) : block = ignore (Util.list_map sliceStmt b.bstmts); b let sliceFundec (fd : fundec) (l : location) : unit = curFundec := fd; curLocation := l; ignore (sliceBlock fd.sbody); curFundec := dummyFunDec; curLocation := locUnknown let sliceGlobal (g : global) : unit = match g with | GType (tinfo, l) -> newGlobals := foldRegions (fun i rest -> GType (sliceTypeInfo i tinfo, l) :: rest) !newGlobals | GCompTag (cinfo, l) -> newGlobals := foldRegions (fun i rest -> GCompTag (sliceCompInfo i cinfo, l) :: rest) !newGlobals | GCompTagDecl (cinfo, l) -> newGlobals := foldRegions (fun i rest -> GCompTagDecl (sliceCompInfo i cinfo, l) :: rest) !newGlobals | GFun (fd, l) -> sliceFundec fd l; newGlobals := GFun (fd, l) :: !newGlobals | GVarDecl _ | GVar _ -> (* Defer processing of vars until end. *) newGlobals := g :: !newGlobals | _ -> E.s (unimp "global %a\n" d_global g) let sliceGlobalVars (g : global) : unit = match g with | GFun (fd, l) -> curFundec := fd; curLocation := l; List.iter sliceVar fd.slocals; List.iter sliceVar fd.sformals; setFunctionType fd (sliceType 1 fd.svar.vtype); curFundec := dummyFunDec; curLocation := locUnknown; | GVar (vinfo, _, l) -> curLocation := l; sliceVar vinfo; curLocation := locUnknown | _ -> () class dropAttrsVisitor = object inherit nopCilVisitor method! vvrbl (vinfo : varinfo) = vinfo.vattr <- dropAttribute "var_sliced" vinfo.vattr; DoChildren method! vglob (g : global) = begin match g with | GCompTag (cinfo, _) -> cinfo.cattr <- dropAttribute "var_type_sliced" cinfo.cattr; | _ -> () end; DoChildren end let sliceFile (f : file) : unit = newGlobals := []; List.iter sliceGlobal f.globals; List.iter sliceGlobalVars f.globals; f.globals <- List.rev !newGlobals; visitCilFile (new dropAttrsVisitor) f let feature = { fd_name = "DataSlicing"; fd_enabled = false; fd_description = "data slicing"; fd_extraopt = []; fd_doit = sliceFile; fd_post_check = true; } let () = Feature.register feature
sectionYPositions = computeSectionYPositions($el), 10)"
x-init="setTimeout(() => sectionYPositions = computeSectionYPositions($el), 10)"
>