diff --git a/src/kernel_internals/typing/cabs2cil.ml b/src/kernel_internals/typing/cabs2cil.ml index b83030c8f8a27ffce8d6b5d6bdf72ad0eacd044f..0bdc611ed92bee320148dfab22327c22d55d1de1 100644 --- a/src/kernel_internals/typing/cabs2cil.ml +++ b/src/kernel_internals/typing/cabs2cil.ml @@ -1424,7 +1424,7 @@ struct let empty_stmts l = let rec is_empty_stmt s = match s.skind with - | Instr (Skip _) -> s.labels = [] + | Instr (Skip _) -> s.labels = [] && s.sattr = [] | Block b -> b.battrs = [] && List.for_all is_empty_stmt b.bstmts | UnspecifiedSequence seq -> List.for_all is_empty_stmt (List.map (fun (x,_,_,_,_) -> x) seq) @@ -1865,10 +1865,10 @@ struct let canDrop (c: chunk) = List.for_all (fun (s,_,_,_,_) -> canDropStatement s) c.stmts - let loopChunk ~ghost a (body: chunk) : chunk = + let loopChunk ~ghost ~sattr a (body: chunk) : chunk = (* Make the statement *) let loop = - mkStmt ~ghost ~valid_sid + mkStmt ~ghost ~valid_sid ~sattr (Loop (a,c2block ~ghost body, CurrentLoc.get (), None, None)) in { stmts = [ loop,[],[],[],[] ]; @@ -9485,7 +9485,7 @@ and doStatement local_env (s : A.statement) : chunk = let break_cond = breakChunk ~ghost loc' in exitLoop (); CurrentLoc.set loc'; - loopChunk ~ghost a + loopChunk ~ghost ~sattr:[Attr("while",[])] a ((doCondition local_env false e skipChunk break_cond) @@ (s', ghost)) @@ -9520,7 +9520,7 @@ and doStatement local_env (s : A.statement) : chunk = false e skipChunk (breakChunk ~ghost loc')) in exitLoop (); - loopChunk ~ghost a (s' @@ (s'', ghost)) + loopChunk ~ghost ~sattr:[Attr("dowhile",[])] a (s' @@ (s'', ghost)) | A.FOR(a,fc1,e2,e3,s,loc) -> begin let loc' = convLoc loc in @@ -9545,10 +9545,10 @@ and doStatement local_env (s : A.statement) : chunk = let res = match e2.expr_node with | A.NOTHING -> (* This means true *) - se1 @@ (loopChunk ~ghost a (s' @@ (s'', ghost)), ghost) + se1 @@ (loopChunk ~sattr:[Attr("for",[])] ~ghost a (s' @@ (s'', ghost)), ghost) | _ -> se1 @@ - (loopChunk ~ghost a + (loopChunk ~sattr:[Attr("for",[])] ~ghost a (((doCondition local_env false e2 skipChunk break_cond) @@ (s', ghost)) @@ (s'', ghost)), ghost) diff --git a/src/kernel_internals/typing/unroll_loops.ml b/src/kernel_internals/typing/unroll_loops.ml index 4ed4a1c7ea381ebe0b282db96d256de616148d30..a14e8c949d178ccb6ae85753051397b002291ff6 100644 --- a/src/kernel_internals/typing/unroll_loops.ml +++ b/src/kernel_internals/typing/unroll_loops.ml @@ -322,7 +322,8 @@ let copy_block kf switch_label_action break_continue_must_change bl = succs = []; preds = []; skind = stmt.skind; - ghost = stmt.ghost} + ghost = stmt.ghost; + sattr = []} in let labelled_stmt_tbl = if stmt.labels = [] then diff --git a/src/kernel_services/ast_data/cil_types.mli b/src/kernel_services/ast_data/cil_types.mli index 6aea5a03a1390e04aba9a8614cc008473c8c3c28..d9acbb6b3b8891732a8607cce573208df9d8b005 100644 --- a/src/kernel_services/ast_data/cil_types.mli +++ b/src/kernel_services/ast_data/cil_types.mli @@ -1062,7 +1062,11 @@ and stmt = { mutable preds: stmt list; (** The inverse of the succs function. *) - mutable ghost : bool + mutable ghost : bool; + + mutable sattr : attributes + (** Statement attributes. + @since Frama-C+dev *) } (** Labels *) diff --git a/src/kernel_services/ast_printing/cil_printer.ml b/src/kernel_services/ast_printing/cil_printer.ml index b4c6b964b73062e25b193a279e2a4d32c9f2e4c5..1da7b9aa53ffc9d030056d865ebbe672fce02f9c 100644 --- a/src/kernel_services/ast_printing/cil_printer.ml +++ b/src/kernel_services/ast_printing/cil_printer.ml @@ -495,7 +495,7 @@ class cil_printer () = object (self) method private current_stmt = try Some (Stack.top current_stmt) with Stack.Empty -> None - method private may_be_skipped s = s.labels = [] + method private may_be_skipped s = s.labels = [] && s.sattr = [] method location fmt loc = Format.fprintf fmt "%a" Filepath.pp_pos (fst loc) @@ -1058,7 +1058,7 @@ class cil_printer () = object (self) pp_open_hvbox fmt 0; self#stmt_labels fmt s; (* print the statement. *) - if Cil.is_skip s.skind && not s.ghost then begin + if Cil.is_skip s.skind && not s.ghost && s.sattr = [] then begin if verbose || s.labels <> [] then fprintf fmt ";" end else begin let was_ghost = is_ghost in @@ -1068,7 +1068,7 @@ class cil_printer () = object (self) Format.fprintf fmt "%t %a " (fun fmt -> self#pp_open_annotation fmt) self#pp_acsl_keyword "ghost" end; - self#stmtkind next fmt s.skind ; + self#stmtkind s.sattr next fmt s.skind ; if display_ghost then begin is_ghost <- false; self#pp_close_annotation fmt @@ -1234,7 +1234,7 @@ class cil_printer () = object (self) fprintf fmt "@[@<0>\n@<0>%s@<0> @<0>%d@<0> @<0>%s@]@\n" directive (fst l).Filepath.pos_lnum filename - method stmtkind (next: stmt) fmt = function + method stmtkind sattr (next: stmt) fmt = function | UnspecifiedSequence seq -> let ctxt = match self#current_stmt with None -> Other | Some s -> Stmt_block s @@ -1374,6 +1374,11 @@ class cil_printer () = object (self) Pretty_utils.pp_list ~sep:"@\n" self#code_annotation fmt a; Format.fprintf fmt "@ %t" (fun fmt -> self#pp_close_annotation fmt); end; + let pp_sattr fmt = + if sattr <> [] && Kernel.is_debug_key_enabled Kernel.dkey_print_attrs + then Format.fprintf fmt "@[/*%a */ @]" self#attributes sattr + else Format.ifprintf fmt "" + in ((* Maybe the first thing is a conditional. Turn it into a WHILE *) try let rec skipEmpty = function @@ -1409,15 +1414,17 @@ class cil_printer () = object (self) [{ skind=Block b} as s ] when self#may_be_skipped s -> b | _ -> { b with bstmts = bodystmts } in - Format.fprintf fmt "%a@[<v 2>%a (%a) %a@]" + Format.fprintf fmt "%a@[<v 2>%a (%a) %t%a@]" (fun fmt -> self#line_directive fmt) l self#pp_keyword "while" self#exp term + pp_sattr (self#unboxed_block Other) b; with Not_found -> - Format.fprintf fmt "%a@[<v 2>%a (1) %a@]" + Format.fprintf fmt "%a@[<v 2>%a (1) %t%a@]" (fun fmt -> self#line_directive fmt) l self#pp_keyword "while" + pp_sattr (self#unboxed_block Other) b); Format.pp_close_box fmt () diff --git a/src/kernel_services/ast_printing/printer.ml b/src/kernel_services/ast_printing/printer.ml index 61f6615b15fa27c804ac4048523ac2f2834ff36f..b6ced2eb01a5166366c46d2026e418b1c6779152 100644 --- a/src/kernel_services/ast_printing/printer.ml +++ b/src/kernel_services/ast_printing/printer.ml @@ -244,7 +244,7 @@ class printer_with_annot () = object (self) self#pp_acsl_keyword "ghost"; is_ghost <- true end; - self#stmtkind next fmt s.skind; + self#stmtkind s.sattr next fmt s.skind; if not was_ghost && s.ghost then begin self#pp_close_annotation ~suf:"@,*/@]" fmt; is_ghost <- false; @@ -257,7 +257,7 @@ class printer_with_annot () = object (self) (fun fmt -> self#pp_open_annotation ~block:false fmt) self#code_annotation a (fun fmt -> self#pp_close_annotation ~block:false fmt) - (self#stmtkind next) s.skind; + (self#stmtkind s.sattr next) s.skind; | _ -> let loop_annot, stmt_annot = List.partition Logic_utils.is_loop_annot all_annot @@ -266,11 +266,11 @@ class printer_with_annot () = object (self) self#loop_annotations fmt loop_annot; pGhost fmt s) end else - self#stmtkind next fmt s.skind; + self#stmtkind s.sattr next fmt s.skind; Format.pp_close_box fmt () - method! stmtkind (next: stmt) fmt skind = - super#stmtkind next fmt + method! stmtkind sattr (next: stmt) fmt skind = + super#stmtkind sattr next fmt begin match skind with | Goto({ contents = { skind = (Return _) as return }},_) diff --git a/src/kernel_services/ast_printing/printer_api.mli b/src/kernel_services/ast_printing/printer_api.mli index 493de08bad1d547358b357dd0086f1c03e00f76c..15bfee3925cc10931886343dd1a88b79320cf67a 100644 --- a/src/kernel_services/ast_printing/printer_api.mli +++ b/src/kernel_services/ast_printing/printer_api.mli @@ -190,7 +190,7 @@ class type extensible_printer_type = object last {!Cil_types.stmt} argument. The initial {!Cil_types.stmt} argument records the statement which follows the one being printed. *) - method stmtkind: stmt -> Format.formatter -> stmtkind -> unit + method stmtkind: attributes -> stmt -> Format.formatter -> stmtkind -> unit (** Print a statement kind. The code to be printed is given in the {!Cil_types.stmtkind} argument. The initial {!Cil_types.stmt} argument records the statement which follows the one being printed; diff --git a/src/kernel_services/ast_queries/cil.ml b/src/kernel_services/ast_queries/cil.ml index 9493c45605a07006f03f9a29264b86f09364cfcf..86eb84205298d80638abceaf63fc096dda62aa85 100644 --- a/src/kernel_services/ast_queries/cil.ml +++ b/src/kernel_services/ast_queries/cil.ml @@ -265,7 +265,7 @@ let mkBlock (slst: stmt list) : block = let mkBlockNonScoping l = let b = mkBlock l in b.bscoping <- false; b -let mkStmt ?(ghost=false) ?(valid_sid=false) (sk: stmtkind) : stmt = +let mkStmt ?(ghost=false) ?(valid_sid=false) ?(sattr=[]) (sk: stmtkind) : stmt = { skind = sk; labels = []; (* It is better to create statements with a valid sid, so that they can @@ -274,7 +274,8 @@ let mkStmt ?(ghost=false) ?(valid_sid=false) (sk: stmtkind) : stmt = (e.g. slicing). *) sid = if valid_sid then Sid.next () else -1; succs = []; preds = []; - ghost = ghost} + ghost = ghost; + sattr = sattr;} let stmt_of_instr_list ?(loc=Location.unknown) = function | [] -> Instr (Skip loc) @@ -4212,7 +4213,7 @@ let parseIntExp ~loc repr = let mkStmtCfg ~before ~(new_stmtkind:stmtkind) ~(ref_stmt:stmt) : stmt = let new_ = { skind = new_stmtkind; labels = []; - sid = -1; succs = []; preds = []; ghost = false } + sid = -1; succs = []; preds = []; ghost = false; sattr = [] } in new_.sid <- Sid.next (); if before then begin @@ -4263,10 +4264,11 @@ let parseIntExp ~loc repr = old_preds; n - let mkEmptyStmt ?ghost ?valid_sid ?(loc=Location.unknown) () = - mkStmt ?ghost ?valid_sid (Instr (Skip loc)) + let mkEmptyStmt ?ghost ?valid_sid ?sattr ?(loc=Location.unknown) () = + mkStmt ?ghost ?valid_sid ?sattr (Instr (Skip loc)) - let mkStmtOneInstr ?ghost ?valid_sid i = mkStmt ?ghost ?valid_sid (Instr i) + let mkStmtOneInstr ?ghost ?valid_sid ?sattr i = + mkStmt ?ghost ?valid_sid ?sattr (Instr i) let dummyInstr = Asm([], ["dummy statement!!"], None, Location.unknown) let dummyStmt = mkStmt (Instr dummyInstr) @@ -4310,9 +4312,9 @@ let parseIntExp ~loc repr = let mkString ~loc s = new_exp ~loc (Const(CStr s)) - let mkWhile ~(guard:exp) ~(body: stmt list) : stmt list = + let mkLoop ?(sattr = [Attr("while", [])]) ~(guard:exp) ~(body: stmt list) : stmt list = (* Do it like this so that the pretty printer recognizes it *) - [ mkStmt ~valid_sid:true + [ mkStmt ~valid_sid:true ~sattr (Loop ([], mkBlock (mkStmt ~valid_sid:true @@ -4324,7 +4326,7 @@ let parseIntExp ~loc repr = let mkFor ~(start: stmt list) ~(guard: exp) ~(next: stmt list) ~(body: stmt list) : stmt list = (start @ - (mkWhile guard (body @ next))) + (mkLoop ~sattr:[Attr("For",[])] ~guard ~body:(body @ next))) let mkForIncr ~(iter : varinfo) ~(first: exp) ~(stopat: exp) ~(incr: exp) ~(body: stmt list) : stmt list = diff --git a/src/kernel_services/ast_queries/cil.mli b/src/kernel_services/ast_queries/cil.mli index ec5424a8b4737a8b5d4f2fa0542525ebfb922b8d..799986722ec0afbc0e43e5acc5cffcd8fb59c5a0 100644 --- a/src/kernel_services/ast_queries/cil.mli +++ b/src/kernel_services/ast_queries/cil.mli @@ -1025,7 +1025,8 @@ val appears_in_expr: varinfo -> exp -> bool if [valid_sid] is false (the default), or to a valid sid if [valid_sid] is true, and [labels], [succs] and [preds] to the empty list *) -val mkStmt: ?ghost:bool -> ?valid_sid:bool -> stmtkind -> stmt +val mkStmt: ?ghost:bool -> ?valid_sid:bool -> ?sattr:attributes -> stmtkind -> + stmt (* make the [new_stmtkind] changing the CFG relatively to [ref_stmt] *) val mkStmtCfg: before:bool -> new_stmtkind:stmtkind -> ref_stmt:stmt -> stmt @@ -1048,7 +1049,8 @@ val mkStmtCfgBlock: stmt list -> stmt (** Construct a statement consisting of just one instruction See {!Cil.mkStmt} for the signification of the optional args. *) -val mkStmtOneInstr: ?ghost:bool -> ?valid_sid:bool -> instr -> stmt +val mkStmtOneInstr: ?ghost:bool -> ?valid_sid:bool -> ?sattr:attributes -> + instr -> stmt (** Try to compress statements so as to get maximal basic blocks. * use this instead of List.@ because you get fewer basic blocks *) @@ -1057,7 +1059,8 @@ val mkStmtOneInstr: ?ghost:bool -> ?valid_sid:bool -> instr -> stmt (** Returns an empty statement (of kind [Instr]). See [mkStmt] for [ghost] and [valid_sid] arguments. @modify Neon-20130301 adds the [valid_sid] optional argument. *) -val mkEmptyStmt: ?ghost:bool -> ?valid_sid:bool -> ?loc:location -> unit -> stmt +val mkEmptyStmt: ?ghost:bool -> ?valid_sid:bool -> ?sattr:attributes -> + ?loc:location -> unit -> stmt (** A instr to serve as a placeholder *) val dummyInstr: instr @@ -1089,8 +1092,10 @@ val mkPureExpr: ?ghost:bool -> ?valid_sid:bool -> fundec:fundec -> ?loc:location -> exp -> stmt -(** Make a while loop. Can contain Break or Continue *) -val mkWhile: guard:exp -> body:stmt list -> stmt list +(** Make a loop. Can contain Break or Continue. + The kind of loop (While, For, DoWhile) is given by [sattr]; + it is a While loop if unspecified. *) +val mkLoop: ?sattr:attributes -> guard:exp -> body:stmt list -> stmt list (** Make a for loop for(i=start; i<past; i += incr) \{ ... \}. The body can contain Break but not Continue. Can be used with i a pointer diff --git a/src/kernel_services/ast_queries/cil_datatype.ml b/src/kernel_services/ast_queries/cil_datatype.ml index 11ee585d0ccc150d1b3c9305f0fec6314753e2ec..4603eeb11fa5b9f38eee5d96a67ca73ace3757de 100644 --- a/src/kernel_services/ast_queries/cil_datatype.ml +++ b/src/kernel_services/ast_queries/cil_datatype.ml @@ -271,7 +271,8 @@ module Stmt_Id = struct sid = -1; succs = []; preds = []; - ghost = false } ] + ghost = false; + sattr = [] } ] let compare t1 t2 = Datatype.Int.compare t1.sid t2.sid let hash t1 = t1.sid let equal t1 t2 = t1.sid = t2.sid diff --git a/src/kernel_services/ast_queries/file.ml b/src/kernel_services/ast_queries/file.ml index 8b0f5b522b18bb4dae2d71f3433482d83c7f35d0..0d0b2949cf04b95003c0fd38891e45d4e700d069 100644 --- a/src/kernel_services/ast_queries/file.ml +++ b/src/kernel_services/ast_queries/file.ml @@ -918,7 +918,7 @@ let cleanup file = method! vstmt_aux st = self#remove_lexical_annotations st; let loc = Stmt.loc st in - if Annotations.has_code_annot st || st.labels <> [] then + if Annotations.has_code_annot st || st.labels <> [] || st.sattr <> [] then keep_stmt <- Stmt.Set.add st keep_stmt; match st.skind with | Block b -> diff --git a/src/plugins/gui/pretty_source.ml b/src/plugins/gui/pretty_source.ml index 4b3263d04dec3295a5047a62bc3ab4a6e7f57c5a..9bd7e7586f61a621f3bf1316e0e1b99fa52dc7f4 100644 --- a/src/plugins/gui/pretty_source.ml +++ b/src/plugins/gui/pretty_source.ml @@ -641,11 +641,11 @@ module TagPrinterClassDeferred (X: Printer.PrinterClass) = struct Format.fprintf fmt "@{<%s>%a@}" (Tag.create (PIP ip)) (super#allocation ~isloop) a; - method! stmtkind next fmt sk = + method! stmtkind sattr next fmt sk = (* Special tag denoting the start of the statement, WITHOUT any ACSL assertion/statement contract, etc. *) let s = Extlib.the self#current_stmt in - Format.fprintf fmt "@{<gui:stmt_start%d>%a@}" s.sid (super#stmtkind next) sk + Format.fprintf fmt "@{<gui:stmt_start%d>%a@}" s.sid (super#stmtkind sattr next) sk initializer force_brace <- true diff --git a/src/plugins/slicing/printSlice.ml b/src/plugins/slicing/printSlice.ml index 1f0109af6dd177363c41988512b1aa0221f8e92a..37e5f8a1c6704e77656fca762cf1ad60d1ae412d 100644 --- a/src/plugins/slicing/printSlice.ml +++ b/src/plugins/slicing/printSlice.ml @@ -75,7 +75,7 @@ class printerClass optional_ff = object(self) str_m super#vdecl var - method! stmtkind next fmt kind = + method! stmtkind sattr next fmt kind = let stmt_info fmt stmt = match opt_ff with | None -> Format.fprintf fmt "@[/* %d */@]" stmt.Cil_types.sid | Some ff -> @@ -94,7 +94,7 @@ class printerClass optional_ff = object(self) try Format.fprintf fmt "@[<v>%a@ %a@]" stmt_info s - (fun fmt -> super#stmtkind next fmt) kind + (fun fmt -> super#stmtkind sattr next fmt) kind with Not_found -> (* some sub statements may be visible *) let sub_stmts = find_sub_stmts s in