diff --git a/src/plugins/wp/Changelog b/src/plugins/wp/Changelog
index 7c0fa5e2fee5169f4987a60e46cd19f1fd243733..2a4bb7b50269cf021fe54e608ab12e5f5953938d 100644
--- a/src/plugins/wp/Changelog
+++ b/src/plugins/wp/Changelog
@@ -27,9 +27,9 @@ Plugin WP <next-release>
 - WP          [2021-10-25] Removes -wp-overflows option (unsound)
 - WP          [2021-06-11] Adds an experimental support of "terminates" clauses.
                            Adds the options -wp-declaration-terminate and
-                          -wp-frama-c-stdlib to claims that external functions
+                           -wp-frama-c-stdlib to claim that external functions
                            terminates.
-                           Adds -wp-declarations-terminate option to claims that
+                           Adds -wp-definitions-terminate option to claim that
                            defined function terminates.
                            Adds -wp-variant-with-terminates to verify loop
                            variants under the termination hypothesis.
diff --git a/src/plugins/wp/cfgAnnot.ml b/src/plugins/wp/cfgAnnot.ml
index 207a87b2c820ab47af0b85bd12d531e7dc0749a9..930f3e439067597fe0408e5a2fd5c98396a940b2 100644
--- a/src/plugins/wp/cfgAnnot.ml
+++ b/src/plugins/wp/cfgAnnot.ml
@@ -202,18 +202,25 @@ let get_terminates_clause kf =
    *      - then handled in a Some case (as user defined) and returns Defined *)
   let defined p =
     Defined (WpPropId.mk_terminates_id kf Kglobal p, normalize_terminates p) in
-  let populate_true () =
+  let populate_true ?(silence=false) () =
     let p = Logic_const.new_predicate @@ Logic_const.ptrue in
-    Wp_parameters.warning
-      ~source:(fst @@ Kernel_function.get_location kf) ~once:true
-      "Missing terminates clause for %a, populates 'terminates \\true'"
-      Kernel_function.pretty kf ;
+    if not silence then
+      Wp_parameters.warning
+        ~source:(fst @@ Kernel_function.get_location kf) ~once:true
+        "Missing terminates clause for %a, populates 'terminates \\true'"
+        Kernel_function.pretty kf ;
     Annotations.add_terminates wp_populate_terminates kf p ;
     defined p
   in
+  let kf_vi = Kernel_function.get_vi kf in
+  let kf_name = Kernel_function.get_name kf in
   match Annotations.terminates kf with
+  | None
+    when Cil_builtins.is_builtin kf_vi
+      || Cil_builtins.is_special_builtin kf_name ->
+      populate_true ~silence:true ()
   | None when Kernel_function.is_in_libc kf ->
-      if not @@ Wp_parameters.TerminatesFCDeclarations.get ()
+      if not @@ Wp_parameters.TerminatesStdlibDeclarations.get ()
       then Assumed Logic_const.pfalse
       else populate_true ()
   | None when Kernel_function.is_definition kf ->
@@ -237,6 +244,12 @@ let get_terminates_hyp kf =
   | Defined (_, p) -> false, p
   | Assumed p -> true, p
 
+let get_decreases_goal kf =
+  let defined t = WpPropId.mk_decrease_id kf Kglobal t, t in
+  match Annotations.decreases ~populate:false kf with
+  | None -> None
+  | Some v -> Some (defined v)
+
 (* -------------------------------------------------------------------------- *)
 (* --- Contracts                                                          --- *)
 (* -------------------------------------------------------------------------- *)
diff --git a/src/plugins/wp/cfgAnnot.mli b/src/plugins/wp/cfgAnnot.mli
index a6c4c5bcb36dbe60abecf8e23d8980b204212cbf..294b030675af76c28dd493b4728c972334280dd6 100644
--- a/src/plugins/wp/cfgAnnot.mli
+++ b/src/plugins/wp/cfgAnnot.mli
@@ -57,6 +57,7 @@ val get_complete_behaviors : kernel_function -> pred_info list
 val get_disjoint_behaviors : kernel_function -> pred_info list
 
 val get_terminates_goal : kernel_function -> pred_info option
+val get_decreases_goal : kernel_function -> variant_info option
 
 (* -------------------------------------------------------------------------- *)
 (* --- Property Accessors : Assertions                                    --- *)
diff --git a/src/plugins/wp/cfgCalculus.ml b/src/plugins/wp/cfgCalculus.ml
index 41c62992ab860b55b9ef1a7228869fe9e3b34071..e867511bf0c67442688eab50df6a8127497d13e3 100644
--- a/src/plugins/wp/cfgCalculus.ml
+++ b/src/plugins/wp/cfgCalculus.ml
@@ -140,6 +140,7 @@ struct
     succ: Cfg.vertex -> Cfg.G.edge list;
     dead: stmt -> bool ;
     terminates: WpPropId.pred_info option ;
+    decreases: WpPropId.variant_info option ;
     we: W.t_env;
     wp: W.t_prop option Vhash.t; (* None is used for non-dag detection *)
     mutable wk: W.t_prop; (* end point *)
@@ -187,11 +188,18 @@ struct
   let prove_property env (p : WpPropId.pred_info) w =
     if is_selected ~goal:true env p then W.add_goal env.we p w else w
 
-  let prove_subproperty env (p : WpPropId.pred_info) prop stmt source w =
+  let prove_subproperty env (p : WpPropId.pred_info) ?deps prop stmt source w =
     if is_selected ~goal:true env p
-    then W.add_subgoal env.we p prop stmt source w
+    then W.add_subgoal env.we ?deps p prop stmt source w
     else w
 
+  let on_selected_terminates env f =
+    match env.terminates with
+    | Some t when is_default_bhv env.mode && is_selected ~goal:true env t ->
+        f env t
+    | _ ->
+        Extlib.id
+
   (* --- Decomposition of WP Rules --- *)
 
   let rec wp (env:env) (a:vertex) : W.t_prop =
@@ -355,6 +363,33 @@ struct
       List.fold_right (prove_property env) complete @@
       List.fold_right (prove_property env) disjoint w
 
+  let do_terminates_deps env w =
+    match env.body with
+    | None -> w
+    | Some _ ->
+        let deps = CfgInfos.terminates_deps env.mode.infos in
+        let return = Kernel_function.find_return env.mode.kf in
+        let prove goal env t w =
+          prove_subproperty env t ~deps goal return FromReturn w
+        in
+        if CfgInfos.is_recursive env.mode.kf then
+          (* there is a dependency on terminates or decreases is missing *)
+          let goal =
+            if None <> env.decreases then Logic_const.ptrue
+            else begin
+              WpLog.warning ~once:true
+                "No 'decreases' clause on recursive function '%a', \
+                 cannot prove termination"
+                Kernel_function.pretty env.mode.kf ;
+              Logic_const.pfalse
+            end
+          in
+          on_selected_terminates env (prove goal) w
+        else
+        if not @@ Property.Set.is_empty deps then
+          on_selected_terminates env (prove Logic_const.ptrue) w
+        else w
+
   let do_global_init env w =
     I.process_global_init env.we env.mode.kf @@
     W.scope env.we [] SC_Global w
@@ -378,7 +413,6 @@ struct
     (* frame-in *)
     W.scope env.we formals SC_Frame_in w
 
-
   let do_post env ~formals (b : CfgAnnot.behavior) w =
     W.label env.we None Clabels.post @@
     W.scope env.we formals SC_Frame_out @@
@@ -415,9 +449,10 @@ struct
          WpLog.SmokeDeadcode.get ()
       then CfgInfos.smoking infos else (fun _ -> false) in
     let terminates = CfgAnnot.get_terminates_goal kf in
+    let decreases = CfgAnnot.get_decreases_goal kf in
     let env = {
       mode ; props ; body ;
-      succ ; dead ; terminates ;
+      succ ; dead ; terminates ; decreases ;
       we = W.new_env kf ;
       wp = Vhash.create 32 ;
       wk = W.empty ;
@@ -428,6 +463,7 @@ struct
     let bhv = CfgAnnot.get_behavior_goals kf ~smoking ~exits mode.bhv in
     begin
       W.close env.we @@
+      do_terminates_deps env @@
       do_global_init env @@
       do_preconditions env ~formals bhv @@
       do_complete_disjoint env @@
diff --git a/src/plugins/wp/cfgDump.ml b/src/plugins/wp/cfgDump.ml
index a7592df0bfb6c2f3d939956d2358c1b833dad9da..c15ac2a858f54f23f3ab69d5e3649e0dbf8c49ab 100644
--- a/src/plugins/wp/cfgDump.ml
+++ b/src/plugins/wp/cfgDump.ml
@@ -106,7 +106,8 @@ let add_goal env (pid,pred) k =
   Format.fprintf !out "  %a -> %a [ style=dotted ] ;@." pretty u pretty k ;
   merge env u k
 
-let add_subgoal env (pid, t_pred) prop stmt _source k =
+let add_subgoal env (pid, t_pred) ?deps prop stmt _source k =
+  ignore deps ;
   let u = node () in
   if Wp_parameters.debug_atleast 1 then
     Format.fprintf !out "  %a [ color=red , label=\"Prove %a @@ %a (%a)\" ] ;@."
diff --git a/src/plugins/wp/cfgInfos.ml b/src/plugins/wp/cfgInfos.ml
index 2fc2e6b4a4fab68b201fe30d18f9b3fbc6d8fecb..839c3cf03d55e305fa4125b35e9a0559b1f47124 100644
--- a/src/plugins/wp/cfgInfos.ml
+++ b/src/plugins/wp/cfgInfos.ml
@@ -23,6 +23,7 @@
 module Cfg = Interpreted_automata
 module Fset = Kernel_function.Set
 module Sset = Cil_datatype.Stmt.Set
+module Pset = Property.Set
 module Shash = Cil_datatype.Stmt.Hashtbl
 
 (* -------------------------------------------------------------------------- *)
@@ -39,6 +40,7 @@ type t = {
   mutable doomed : WpPropId.prop_id Bag.t;
   mutable calls : Kernel_function.Set.t;
   mutable no_variant_loops : Sset.t;
+  mutable terminates_deps : Pset.t;
 }
 
 (* -------------------------------------------------------------------------- *)
@@ -61,6 +63,8 @@ let unreachable infos v =
       not @@ CheckPath.check_path checkpath cfg.entry_point v
   | _ -> true
 
+let terminates_deps infos = infos.terminates_deps
+
 (* -------------------------------------------------------------------------- *)
 (* --- Selected Properties                                                --- *)
 (* -------------------------------------------------------------------------- *)
@@ -119,6 +123,14 @@ let selected_terminates ~prop kf =
       let tp_names = WpPropId.user_pred_names ip.Cil_types.ip_content in
       WpPropId.are_selected_names prop (tk_name :: tp_names)
 
+let selected_decreases ~prop kf =
+  match Annotations.decreases kf with
+  | None -> false
+  | Some (it, _) ->
+      let tk_name = "@decreases" in
+      let tp_names = WpPropId.ident_names it.term_name in
+      WpPropId.are_selected_names prop (tk_name :: tp_names)
+
 let selected_disjoint_complete kf ~bhv ~prop =
   selected_default ~bhv &&
   ( selected_clause ~prop "@complete_behaviors" Annotations.complete kf ||
@@ -140,7 +152,7 @@ let selected_main_bhv ~bhv ~prop (b : Cil_types.funbehavior) =
 (* --- Calls                                                              --- *)
 (* -------------------------------------------------------------------------- *)
 
-let collect_calls ~bhv kf stmt =
+let collect_calls ~bhv ?(on_missing_calls=fun _ -> ()) kf stmt =
   let open Cil_types in
   match stmt.skind with
   | Instr(Call(_,fct,_,_)) ->
@@ -152,11 +164,16 @@ let collect_calls ~bhv kf stmt =
               if bhv = []
               then List.map (fun b -> b.b_name) (Annotations.behaviors kf)
               else bhv in
-            List.fold_left
-              (fun fs bhv -> match Dyncall.get ~bhv stmt with
-                 | None -> fs
-                 | Some(_,kfs) -> List.fold_right Fset.add kfs fs
-              ) Fset.empty bhvs
+            let calls =
+              List.fold_left
+                (fun fs bhv -> match Dyncall.get ~bhv stmt with
+                   | None -> fs
+                   | Some(_,kfs) -> List.fold_right Fset.add kfs fs
+                ) Fset.empty bhvs
+            in
+            if Fset.is_empty calls then
+              on_missing_calls stmt ;
+            calls
       end
   | Instr(Local_init(x,ConsInit(vf, args, kind), loc)) ->
       Cil.treat_constructor_as_func
@@ -167,21 +184,138 @@ let collect_calls ~bhv kf stmt =
         x vf args kind loc
   | _ -> Fset.empty
 
+(* -------------------------------------------------------------------------- *)
+(* --- Recursion                                                          --- *)
+(* -------------------------------------------------------------------------- *)
+
+module Callees = WpContext.StaticGenerator(Kernel_function)
+    (struct
+      type key = Kernel_function.t
+      type data = Fset.t * Cil_types.stmt list
+      (** functions + unspecified function pointer calls *)
+      let name = "Wp.CfgInfos.SCallees"
+      let compile = function
+        | { Cil_types.fundec = Definition(fd, _ ) } as kf ->
+            let stmts = ref [] in
+            let on_missing_calls s = stmts := s :: !stmts in
+            let fold e s =
+              Fset.union e (collect_calls ~on_missing_calls ~bhv:[] kf s)
+            in
+            let kfs = List.fold_left fold Fset.empty fd.sallstmts in
+            kfs, !stmts
+        | _ -> Fset.empty, []
+    end)
+
+module RecursiveClusters : sig
+  val is_recursive : Kernel_function.t -> bool
+  val get_cluster : Kernel_function.t -> Kernel_function.Set.t option
+end =
+struct
+  (* Tarjan's algorithm, adapted from:
+       http://pauillac.inria.fr/~levy/why3/graph/abs/scct/2/scc.html
+     (proved in Why3)
+  *)
+  let successors kf = fst @@ Callees.get kf
+
+  module HT = Cil_datatype.Kf.Hashtbl
+
+  type env = {
+    mutable stack: Fset.elt list;
+    mutable id: int;
+    table: (Fset.elt, int) Hashtbl.t ;
+    clusters: Fset.t option HT.t ;
+  }
+
+  let rec unstack_to x ?(cluster=[]) = function
+    | [] -> cluster, []
+    | y :: s' when Kernel_function.equal y x -> x :: cluster, s'
+    | y :: s' -> unstack_to x ~cluster:(y :: cluster) s'
+
+  let rec dfs roots env =
+    try
+      let v = Fset.choose roots in
+      let vn =
+        try Hashtbl.find env.table v
+        with Not_found -> visit_node v env
+      in
+      let others_n = dfs (Fset.remove v roots) env in
+      min vn others_n
+    with Not_found -> max_int
+  and visit_node x env =
+    let n = env.id in
+    Hashtbl.replace env.table x n ;
+    env.id <- n + 1;
+    env.stack <- x :: env.stack;
+    let base = dfs (successors x) env in
+    if base < n then base
+    else begin
+      let (cluster, stack) = unstack_to x env.stack in
+      List.iter (fun v -> Hashtbl.replace env.table v max_int) cluster ;
+      env.stack <- stack;
+      begin match cluster with
+        | [ x ] when base = max_int ->
+            HT.add env.clusters x None
+        | cluster ->
+            let set = Some (Fset.of_list cluster) in
+            List.iter (fun kf -> HT.add env.clusters kf set) cluster
+      end ;
+      max_int
+    end
+
+  let make_clusters s =
+    let e = {
+      stack = []; id = 0; table = Hashtbl.create 37; clusters = HT.create 37
+    } in
+    ignore (dfs s e);
+    e.clusters
+
+  module RTable =
+    State_builder.Option_ref(HT.Make(Datatype.Option(Fset)))
+      (struct
+        let name = "Wp.CfgInfo.RecursiveClusters"
+        let dependencies = [ Ast.self ]
+      end)
+
+  let create () =
+    let kfs = ref Kernel_function.Set.empty in
+    Globals.Functions.iter(fun kf -> kfs := Fset.add kf !kfs) ;
+    make_clusters !kfs
+
+  let table () = RTable.memo create
+
+  let get_cluster kf = HT.find (table ()) kf
+  let is_recursive kf =
+    (* in a recursive cluster or contains unspecified function pointer call *)
+    None <> get_cluster kf || [] <> snd @@ Callees.get kf
+end
+
+let is_recursive = RecursiveClusters.is_recursive
+let get_cluster = RecursiveClusters.get_cluster
+
 (* -------------------------------------------------------------------------- *)
 (* --- No variant loops                                                   --- *)
 (* -------------------------------------------------------------------------- *)
 
-let collect_loops_no_variant stmt =
+let collect_loops_no_variant kf stmt =
   let open Cil_types in
   let fold_no_variant _ = function
-    | { annot_content = AVariant _ } -> (&&) false
-    | _ -> (&&) true
+    | { annot_content = AVariant v } as ca -> fun _ -> Some (ca, v)
+    | _ -> Extlib.id
+  in
+  let props_of_v ca v =
+    let (d, _), (p, _) = WpStrategy.mk_variant_properties kf stmt ca v in
+    Pset.union
+      (Pset.singleton @@ WpPropId.property_of_id d)
+      (Pset.singleton @@ WpPropId.property_of_id p)
   in
   match stmt.skind with
-  | Loop _ when Annotations.fold_code_annot fold_no_variant stmt true ->
-      Sset.singleton stmt
+  | Loop _ ->
+      begin match Annotations.fold_code_annot fold_no_variant stmt None with
+        | None -> Sset.singleton stmt, Pset.empty
+        | Some (ca, v) -> Sset.empty, props_of_v ca (fst v)
+      end
   | _ ->
-      Sset.empty
+      Sset.empty, Pset.empty
 
 (* -------------------------------------------------------------------------- *)
 (* --- Memoization Key                                                    --- *)
@@ -265,7 +399,8 @@ let compile Key.{ kf ; smoking ; bhv ; prop } =
     annots = false ;
     doomed = Bag.empty ;
     calls = Fset.empty ;
-    no_variant_loops = Sset.empty
+    no_variant_loops = Sset.empty ;
+    terminates_deps = Pset.empty ;
   } in
   let behaviors = Annotations.behaviors kf in
   (* Inits *)
@@ -275,6 +410,10 @@ let compile Key.{ kf ; smoking ; bhv ; prop } =
   Option.iter
     begin fun (cfg : Cfg.automaton) ->
       (* Spec Iteration *)
+      if selected_decreases ~prop kf then
+        Wp_parameters.warning
+          "Unsupported clause @decreases in '%a' (ignored)"
+          Kernel_function.pretty kf ;
       if selected_terminates ~prop kf ||
          selected_disjoint_complete kf ~bhv ~prop ||
          (List.exists (selected_bhv ~smoking ~bhv ~prop) behaviors)
@@ -283,7 +422,7 @@ let compile Key.{ kf ; smoking ; bhv ; prop } =
       Shash.iter
         (fun stmt (src,_) ->
            let fs = collect_calls ~bhv kf stmt in
-           let nv_loops = collect_loops_no_variant stmt in
+           let nv_loops, term_deps = collect_loops_no_variant kf stmt in
            let dead = unreachable infos src in
            let ca = CfgAnnot.get_code_assertions kf stmt in
            let ca_pids = List.map fst ca.code_verified in
@@ -306,6 +445,8 @@ let compile Key.{ kf ; smoking ; bhv ; prop } =
                infos.calls <- Fset.union fs infos.calls ;
                infos.no_variant_loops <-
                  Sset.union nv_loops infos.no_variant_loops ;
+               infos.terminates_deps <-
+                 Pset.union term_deps infos.terminates_deps
              end
         ) cfg.stmt_table ;
       (* Dead Post Conditions *)
@@ -327,16 +468,44 @@ let compile Key.{ kf ; smoking ; bhv ; prop } =
   Bag.iter
     (fun p -> if WpPropId.filter_status p then WpAnnot.set_unreachable p)
     infos.doomed ;
-  (* Trivial terminates *)
-  if Kernel_function.is_definition kf then
-    begin match CfgAnnot.get_terminates_goal kf with
-      | Some (id, _t)
-        when selected_terminates ~prop kf
-          && infos.calls = Fset.empty
-          && infos.no_variant_loops = Sset.empty ->
-          WpAnnot.set_trivially_terminates id
-      | _ -> ()
-    end ;
+  (* Termination *)
+  let infos =
+    if Kernel_function.is_definition kf then
+      match CfgAnnot.get_terminates_goal kf with
+      | Some (id, _) when selected_terminates ~prop kf ->
+          let warning_locs =
+            List.map Cil_datatype.Stmt.loc @@ snd @@ Callees.get kf
+          in
+          if warning_locs <> [] then
+            Wp_parameters.warning ~once:true
+              "In '%a', no 'calls' specification for statement(s) on \
+               line(s): %a, @\nAssuming that they can call '%a'"
+              Kernel_function.pretty kf
+              (Pretty_utils.pp_list ~sep:", " Cil_datatype.Location.pretty_line)
+              warning_locs
+              Kernel_function.pretty kf ;
+          if is_recursive kf then
+            (* Notes:
+               - a recursive function never trivially terminates,
+               - in absence of decreases, CfgCalculus will warn *)
+            begin match CfgAnnot.get_decreases_goal kf with
+              | None -> infos
+              | Some (id, _) ->
+                  let deps =
+                    Pset.add (WpPropId.property_of_id id) infos.terminates_deps
+                  in
+                  { infos with terminates_deps = deps }
+            end
+          else if infos.calls = Fset.empty
+               && infos.no_variant_loops = Sset.empty then begin
+            WpAnnot.set_trivially_terminates id infos.terminates_deps ;
+            (* Drop dependencies for this terminates, we've used it. *)
+            { infos with terminates_deps = Pset.empty }
+          end
+          else infos
+      | _ -> infos
+    else infos
+  in
   (* Collected Infos *)
   infos
 
diff --git a/src/plugins/wp/cfgInfos.mli b/src/plugins/wp/cfgInfos.mli
index 472a321933a16e7853b7fd20727ceb0136b27d0f..45172d3653075c7ba3dd971dd286ab2022ac7bec 100644
--- a/src/plugins/wp/cfgInfos.mli
+++ b/src/plugins/wp/cfgInfos.mli
@@ -38,5 +38,9 @@ val doomed : t -> WpPropId.prop_id Bag.t
 val calls : t -> Kernel_function.Set.t
 val smoking : t -> Cil_types.stmt -> bool
 val unreachable : t -> Cfg.vertex -> bool
+val terminates_deps : t -> Property.Set.t
+
+val is_recursive : Kernel_function.t -> bool
+val get_cluster : Kernel_function.t -> Kernel_function.Set.t option
 
 (**************************************************************************)
diff --git a/src/plugins/wp/cfgWP.ml b/src/plugins/wp/cfgWP.ml
index 869a3f3915845f8bf290cc1f736ed76c918ad304..7c5edf076e143af3de5c65afa06401a88a5fa86b 100644
--- a/src/plugins/wp/cfgWP.ml
+++ b/src/plugins/wp/cfgWP.ml
@@ -438,10 +438,11 @@ struct
         (F.varsp goal) hs
     in xs , hs , goal
 
-  let add_vc target ?(warn=Warning.Set.empty) pred vcs =
+  let add_vc
+      target ?(warn=Warning.Set.empty) ?(deps=Property.Set.empty) pred vcs =
     let xs , hs , goal = introduction pred in
     let hyps = Conditions.intros hs Conditions.nil in
-    let vc = { empty_vc with goal=goal ; vars=xs ; hyps=hyps ; warn=warn } in
+    let vc = { empty_vc with goal ; vars=xs ; hyps ; warn ; deps } in
     Gmap.add target (Splitter.singleton vc) vcs
 
   (* ------------------------------------------------------------------------ *)
@@ -507,7 +508,7 @@ struct
          let vcs = add_vc (Gprop gpid) ~warn goal wp.vcs in
          { wp with vcs = vcs })
 
-  let add_subgoal wenv (gpid, _) predicate stmt source wp = in_wenv wenv wp
+  let add_subgoal wenv (gpid,_) ?deps predicate stmt src wp = in_wenv wenv wp
       (fun env wp ->
          let outcome = Warning.catch
              ~severe:true ~effect:"Degenerated goal"
@@ -516,7 +517,7 @@ struct
            | Warning.Result(warn,goal) -> warn,goal
            | Warning.Failed warn -> warn,F.p_false
          in
-         let vcs = add_vc (Gsource(gpid, stmt, source)) ~warn goal wp.vcs in
+         let vcs = add_vc (Gsource(gpid, stmt, src)) ~warn ?deps goal wp.vcs in
          { wp with vcs = vcs })
 
   let add_assigns wenv (gpid,ainfo) wp = in_wenv wenv wp
diff --git a/src/plugins/wp/doc/manual/wp_plugin.tex b/src/plugins/wp/doc/manual/wp_plugin.tex
index 937fcb63221b3deac4e2cec20d30eb37727a3547..bd11618cab799e9a36358a40b27274bc1eaad2db 100644
--- a/src/plugins/wp/doc/manual/wp_plugin.tex
+++ b/src/plugins/wp/doc/manual/wp_plugin.tex
@@ -797,7 +797,8 @@ interface of the programmatic API.
     Recognized categories are: \texttt{@lemma}, \texttt{@requires}, \texttt{@assigns},
     \texttt{@ensures}, \texttt{@exits}, \texttt{@assert}, \texttt{@check},
     \texttt{@invariant}, \texttt{@variant}, \texttt{@breaks},
-    \texttt{@continues}, \texttt{@returns}, \\
+    \texttt{@continues}, \texttt{@returns}, \texttt{@terminates},\\
+    \texttt{@decreases},
     \texttt{\mbox{@complete\_behaviors}}, \texttt{\mbox{@disjoint\_behaviors}}.
     \\
     Properties can be prefixed with a minus sign to \emph{skip} the associated annotations.
@@ -986,7 +987,7 @@ weakest precondition calculus.
   \texttt{terminates \textbackslash{}false}, any loop variant proof is trivial)
   (default is: \texttt{no}).
 \item[\tt -wp-(no)-dynamic] handles calls \textit{via} function pointers
-  thanks to the dedicated \verb+@calls f1,...,fn+ code annotation (default is: \texttt{yes}).
+  thanks to the dedicated code annotation \verb+@calls f1,...,fn+ (default is: \texttt{yes}).
 \end{description}
 
 \subsubsection{ACSL extension \texttt{@calls}}
@@ -1018,6 +1019,99 @@ These annotations split post-conditions to the dynamic call into two sub-goals:
 a second for the call to \verb+f2+. A last goal is generated at the call
 site: one must prove that \verb+fp+ is either \verb+f1+ or \verb+f2+.
 
+\subsubsection{Proof of \texttt{@terminates}}
+
+In ACSL, a function can be specified to terminate when some condition \verb+P+ holds
+in pre-condition.
+
+\begin{lstlisting}[language=c, alsolanguage=acsl]
+/*@ terminates P ; */
+void function(void){
+  ...
+}
+\end{lstlisting}
+
+If terminates verification is active (depending on \verb+-wp+ options) WP collects all function calls and loops
+and generate verification conditions (VCs) to prove them, as detailed below.
+
+\paragraph{Loop variants.} When a loop has a variant, no particular verification
+is generated for termination. However, if \verb+-wp-variant-with-terminates+ is
+active, the variant is verified only when \verb+P+ holds in pre-condition.
+Intuitively, this option means that variant has to decrease only if the function has to terminate. For example:
+
+\begin{lstlisting}[language=c, alsolanguage=acsl]
+//@ terminates i >= 0 ;
+void positive(int i){
+  /*@ loop invariant \at(i, Pre) >= 0 ==> 0 <= i <= \at(i, Pre) ;
+      loop assigns i ;
+      loop variant i ; // verified when i >= 0 at pre
+  */
+  while(i) --i;
+}
+\end{lstlisting}
+
+Note that when \verb+terminates \false+ is specified, any loop
+variant would trivially holds.
+
+When a loop does not have a variant, WP checks that the loop is unreachable
+when \verb+P+ holds in precondition, which means that a loop is allowed to not
+terminate when the function does not have to terminate. For example:
+
+\begin{lstlisting}[language=c, alsolanguage=acsl]
+//@ terminates i >= 0 ;
+void function(int i){
+  if(i < 0){
+    // note i >= 0 && i < 0 ==> \false
+    while(1);
+  }
+}
+\end{lstlisting}
+
+\paragraph{Function call.} When a function \verb+f+ is required to terminate on
+\verb+P+, and calling function \verb+g+ only terminates when \verb+Q+, WP tries to
+prove that \verb+P@pre ==> Q@call-point+. Indeed, for \verb+f+ to terminates while calling
+\verb+g+, then \verb+g+ shall terminate. Note that it
+means that if the function under verification shall terminate, any call to a
+non-terminating function shall be unreachable. For example:
+
+\begin{lstlisting}[language=c, alsolanguage=acsl]
+//@ terminates i <= 0 ;
+void f(int i) ;
+
+//@ terminates i >  0 ;
+void g(int i) ;
+
+/*@ requires i <= 0 ;
+    terminates \true ; */
+void caller(int i){
+  f(i); // succeeds
+  g(i); // fails, thus global termination is not proved
+}
+\end{lstlisting}
+
+When a called function does not have a terminates specification, WP assumes one
+as specified by the options:
+
+\begin{itemize}
+\item \verb+-wp-frama-c-stdlib-terminate+,
+\item \verb+-wp-declarations-terminate+,
+\item \verb+-wp-definitions-terminate+.
+\end{itemize}
+
+\paragraph{(Mutually) recursive function.} When a function is involved in a
+recursive cluster (as defined in the ACSL manual) and shall terminate, WP checks that a
+\verb+decreases+ clause is available for the function. Proving that existing
+recursive calls might be unreachable is delegated to the proof of the
+\verb+decreases+ clause. Note however that, currenlty, the proof of \verb+decreases+
+clauses is not supported yet.
+
+\paragraph{Function pointers. } On a call via a function pointer, WP checks that
+a \verb+@calls+ is specified for this function pointer. If so, WP just behaves
+as explained before for function calls. If such a specification is not available,
+WP considers that the pointed function might call the function under
+verification and thus that it is part of a recursive cluster and shall provide
+a \verb+decreases+ clause.
+
 \subsection{Smoke Tests}
 
 During modular deductive verification, inconsistencies in function requirements
diff --git a/src/plugins/wp/mcfg.mli b/src/plugins/wp/mcfg.mli
index 15905e4eab2aecfcec837c07a2690c903bf33cab..3ecd5a5f9a2ae8fe6549d4ba40fd9eaec575d988 100644
--- a/src/plugins/wp/mcfg.mli
+++ b/src/plugins/wp/mcfg.mli
@@ -67,7 +67,7 @@ module type S = sig
   val add_axiom : WpPropId.prop_id -> LogicUsage.logic_lemma -> unit
   val add_hyp  : t_env -> WpPropId.pred_info -> t_prop -> t_prop
   val add_goal : t_env -> WpPropId.pred_info -> t_prop -> t_prop
-  val add_subgoal : t_env -> WpPropId.pred_info ->
+  val add_subgoal : t_env -> WpPropId.pred_info -> ?deps:Property.Set.t ->
     predicate -> stmt -> WpPropId.effect_source -> t_prop -> t_prop
 
   val add_assigns : t_env -> WpPropId.assigns_info -> t_prop -> t_prop
diff --git a/src/plugins/wp/tests/wp_acsl/clusters.i b/src/plugins/wp/tests/wp_acsl/clusters.i
new file mode 100644
index 0000000000000000000000000000000000000000..c1a2e71653c867c4a4cb93eba854d6f5f55f69fa
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/clusters.i
@@ -0,0 +1,64 @@
+/* run.config
+   OPT: -wp-definitions-terminate -wp-declarations-terminate
+*/
+/* run.config_qualif
+   OPT: -wp-definitions-terminate -wp-declarations-terminate
+*/
+
+///// Simple recursion
+
+//@ decreases n ;
+void simpl_r(unsigned n){ // prove termination, warns unsupported decreases
+  if(n) simpl_r(n-1);
+}
+
+void simpl_rf(unsigned n){ // fails termination
+  if(n) simpl_rf(n-1);
+}
+
+///// Mutual recursion
+
+// asks for decreases on both functions "mutual_"
+void mutual_1(unsigned);
+void mutual_2(unsigned);
+// not part of the cluster, termination must succeed
+void callee_no_r(void);
+
+// fails termination (no decreases)
+void mutual_1(unsigned n){
+  if(n) mutual_2(n-1);
+  callee_no_r();
+}
+
+// succeeds termination
+//@ decreases n ;
+void mutual_2(unsigned n){
+  if(n) mutual_1(n-1);
+  simpl_rf(n); // this does not prevent termination proof
+}
+
+void callee_no_r(void){
+
+}
+
+// calls a cluster but not in cluster, termination must succeed
+void caller_no_cluster(void){
+  mutual_1(42);
+}
+
+///// Mutual recursion with function pointer
+
+void fp(void (*)(unsigned), unsigned);
+
+//@ decreases n ;
+void function(unsigned n){
+  if(n) fp(&function, n-1);
+}
+
+// termination fails : recursion without decreases
+//@ requires pf == &function ;
+void fp(void (*pf)(unsigned), unsigned n){
+  if (n)
+    //@ calls function ;
+    (*pf)(n-1) ;
+}
diff --git a/src/plugins/wp/tests/wp_acsl/oracle/clusters.res.oracle b/src/plugins/wp/tests/wp_acsl/oracle/clusters.res.oracle
new file mode 100644
index 0000000000000000000000000000000000000000..8b18d6554a35709cd335f477d3d044182a349fc0
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/oracle/clusters.res.oracle
@@ -0,0 +1,118 @@
+# frama-c -wp [...]
+[kernel] Parsing tests/wp_acsl/clusters.i (no preprocessing)
+[wp] Running WP plugin...
+[wp] tests/wp_acsl/clusters.i:40: Warning: 
+  Missing terminates clause for callee_no_r, populates 'terminates \true'
+[wp] [CFG] Goal callee_no_r_terminates : Valid (Trivial)
+[wp] Warning: Missing RTE guards
+[wp] tests/wp_acsl/clusters.i:45: Warning: 
+  Missing terminates clause for caller_no_cluster, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Missing terminates clause for fp, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'function' (ignored)
+[wp] tests/wp_acsl/clusters.i:54: Warning: 
+  Missing terminates clause for function, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:28: Warning: 
+  Missing terminates clause for mutual_1, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'mutual_2' (ignored)
+[wp] tests/wp_acsl/clusters.i:35: Warning: 
+  Missing terminates clause for mutual_2, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'simpl_r' (ignored)
+[wp] tests/wp_acsl/clusters.i:11: Warning: 
+  Missing terminates clause for simpl_r, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:15: Warning: 
+  Missing terminates clause for simpl_rf, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'simpl_rf', cannot prove termination
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'mutual_1', cannot prove termination
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'fp', cannot prove termination
+------------------------------------------------------------
+  Function caller_no_cluster
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'caller_no_cluster':
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function fp
+------------------------------------------------------------
+
+Goal Call point function in 'fp' at instruction (file tests/wp_acsl/clusters.i, line 63):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'fp' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'fp' (2/2):
+Call Result at line 64
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function function
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'function':
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Instance of 'Pre-condition (file tests/wp_acsl/clusters.i, line 59) in 'fp'' in 'function' at call 'fp' (file tests/wp_acsl/clusters.i, line 55)
+:
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function mutual_1
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'mutual_1' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'mutual_1' (2/2):
+Call Result at line 31
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function mutual_2
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'mutual_2':
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function simpl_r
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'simpl_r':
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function simpl_rf
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'simpl_rf' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'simpl_rf' (2/2):
+Call Result at line 17
+Prove: false.
+
+------------------------------------------------------------
diff --git a/src/plugins/wp/tests/wp_acsl/oracle/terminates_fp.res.oracle b/src/plugins/wp/tests/wp_acsl/oracle/terminates_fp.res.oracle
new file mode 100644
index 0000000000000000000000000000000000000000..b4d58200d98eed31ee23fd13bb87396a010c8761
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/oracle/terminates_fp.res.oracle
@@ -0,0 +1,252 @@
+# frama-c -wp [...]
+[kernel] Parsing tests/wp_acsl/terminates_fp.i (no preprocessing)
+[wp] Running WP plugin...
+[wp] tests/wp_acsl/terminates_fp.i:42: Warning: 
+  Missing terminates clause for no_t_spec, populates 'terminates \true'
+[wp] Warning: Missing RTE guards
+[wp] tests/wp_acsl/terminates_fp.i:62: Warning: 
+  Missing terminates clause for no_t_spec_cond, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:70: Warning: 
+  Missing terminates clause for no_t_spec_cond_m, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:56: Warning: 
+  Missing terminates clause for no_t_spec_in_bhv, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:22: Warning: 
+  Missing terminates clause for t_spec, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:36: Warning: 
+  Missing terminates clause for t_spec_in_bhv, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  Missing terminates clause for warns_recursive, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  In 'warns_recursive', no 'calls' specification for statement(s) on line(s): 
+  line 79, 
+  Assuming that they can call 'warns_recursive'
+[wp] tests/wp_acsl/terminates_fp.i:9: Warning: 
+  Missing terminates clause for gt, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:11: Warning: 
+  Missing terminates clause for ht, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:79: Warning: 
+  Missing 'calls' for default behavior
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  No 'decreases' clause on recursive function 'warns_recursive', cannot prove termination
+------------------------------------------------------------
+  Function no_t_spec
+------------------------------------------------------------
+
+Goal Call point gnt hnt in 'no_t_spec' at instruction (file tests/wp_acsl/terminates_fp.i, line 44):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec' (2/2):
+Call Effect at line 44
+Tags: Call gnt.
+Assume { (* Heap *) Type: region(G_gnt_32) <= 0. }
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function no_t_spec_cond
+------------------------------------------------------------
+
+Goal Call point gt hnt in 'no_t_spec_cond' at instruction (file tests/wp_acsl/terminates_fp.i, line 64):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_cond' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_cond' (2/2):
+Call Effect at line 64
+Tags: Call hnt.
+Assume { (* Heap *) Type: region(G_hnt_37) <= 0. }
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function no_t_spec_cond_m
+------------------------------------------------------------
+
+Goal Call point gt ht in 'no_t_spec_cond_m' at instruction (file tests/wp_acsl/terminates_fp.i, line 72):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Call point hnt in 'no_t_spec_cond_m' at instruction (file tests/wp_acsl/terminates_fp.i, line 75) (1/2):
+Tags: Call gt.
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Call point hnt in 'no_t_spec_cond_m' at instruction (file tests/wp_acsl/terminates_fp.i, line 75) (2/2):
+Tags: Call ht.
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_cond_m' (1/3):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_cond_m' (2/3):
+Call Effect at line 75
+Tags: Call gt Call hnt.
+Assume { (* Heap *) Type: (region(G_gt_22) <= 0) /\ (region(G_hnt_37) <= 0).
+}
+Prove: false.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_cond_m' (3/3):
+Call Effect at line 75
+Tags: Call ht Call hnt.
+Assume { (* Heap *) Type: (region(G_hnt_37) <= 0) /\ (region(G_ht_27) <= 0).
+}
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function no_t_spec_in_bhv
+------------------------------------------------------------
+
+Goal Call point gnt hnt in 'no_t_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 58):
+Let a = global(G_hnt_37).
+Let a_1 = global(G_gnt_32).
+Assume {
+  Type: is_sint32(x).
+  (* Heap *)
+  Type: region(f.base) <= 0.
+  (* Pre-condition for 'B1' *)
+  Have: ((x = 0) -> (a_1 = f)).
+  (* Pre-condition for 'B2' *)
+  Have: ((x != 0) -> (a = f)).
+}
+Prove: (a_1 = f) \/ (a = f).
+
+------------------------------------------------------------
+
+Goal Complete behaviors 'B1', 'B2':
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Disjoint behaviors 'B1', 'B2':
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_in_bhv' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'no_t_spec_in_bhv' (2/2):
+Call Effect at line 58
+Tags: Call gnt.
+Assume { (* Heap *) Type: region(G_gnt_32) <= 0. }
+Prove: false.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function no_t_spec_in_bhv with behavior B1
+------------------------------------------------------------
+
+Goal Call point gnt hnt in 'no_t_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 58):
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function no_t_spec_in_bhv with behavior B2
+------------------------------------------------------------
+
+Goal Call point gnt hnt in 'no_t_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 58):
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function t_spec
+------------------------------------------------------------
+
+Goal Call point gt ht in 't_spec' at instruction (file tests/wp_acsl/terminates_fp.i, line 24):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 't_spec':
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function t_spec_in_bhv
+------------------------------------------------------------
+
+Goal Call point gt hnt in 't_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 38):
+Let a = global(G_gt_22).
+Let a_1 = global(G_hnt_37).
+Assume {
+  Type: is_sint32(x).
+  (* Heap *)
+  Type: region(f.base) <= 0.
+  (* Pre-condition for 'B1' *)
+  Have: ((x = 0) -> (a_1 = f)).
+  (* Pre-condition for 'B2' *)
+  Have: ((x != 0) -> (a = f)).
+}
+Prove: (a = f) \/ (a_1 = f).
+
+------------------------------------------------------------
+
+Goal Complete behaviors 'B1', 'B2':
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Disjoint behaviors 'B1', 'B2':
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 't_spec_in_bhv' (1/2):
+Prove: true.
+
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 't_spec_in_bhv' (2/2):
+Call Effect at line 38
+Tags: Call hnt.
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function t_spec_in_bhv with behavior B1
+------------------------------------------------------------
+
+Goal Call point gt hnt in 't_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 38):
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function t_spec_in_bhv with behavior B2
+------------------------------------------------------------
+
+Goal Call point gt hnt in 't_spec_in_bhv' at instruction (file tests/wp_acsl/terminates_fp.i, line 38):
+Prove: true.
+
+------------------------------------------------------------
+------------------------------------------------------------
+  Function warns_recursive
+------------------------------------------------------------
+
+Goal Termination-condition (generated) in 'warns_recursive':
+Call Result at line 80
+Prove: false.
+
+------------------------------------------------------------
diff --git a/src/plugins/wp/tests/wp_acsl/oracle_qualif/clusters.res.oracle b/src/plugins/wp/tests/wp_acsl/oracle_qualif/clusters.res.oracle
new file mode 100644
index 0000000000000000000000000000000000000000..3ce7b7d03b5435e08f555a555d00ab5c0d1b9641
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/oracle_qualif/clusters.res.oracle
@@ -0,0 +1,59 @@
+# frama-c -wp [...]
+[kernel] Parsing tests/wp_acsl/clusters.i (no preprocessing)
+[wp] Running WP plugin...
+[wp] tests/wp_acsl/clusters.i:40: Warning: 
+  Missing terminates clause for callee_no_r, populates 'terminates \true'
+[wp] [CFG] Goal callee_no_r_terminates : Valid (Trivial)
+[wp] Warning: Missing RTE guards
+[wp] tests/wp_acsl/clusters.i:45: Warning: 
+  Missing terminates clause for caller_no_cluster, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Missing terminates clause for fp, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'function' (ignored)
+[wp] tests/wp_acsl/clusters.i:54: Warning: 
+  Missing terminates clause for function, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:28: Warning: 
+  Missing terminates clause for mutual_1, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'mutual_2' (ignored)
+[wp] tests/wp_acsl/clusters.i:35: Warning: 
+  Missing terminates clause for mutual_2, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  Unsupported clause @decreases in 'simpl_r' (ignored)
+[wp] tests/wp_acsl/clusters.i:11: Warning: 
+  Missing terminates clause for simpl_r, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:15: Warning: 
+  Missing terminates clause for simpl_rf, populates 'terminates \true'
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'simpl_rf', cannot prove termination
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'mutual_1', cannot prove termination
+[wp] tests/wp_acsl/clusters.i:60: Warning: 
+  No 'decreases' clause on recursive function 'fp', cannot prove termination
+[wp] 12 goals scheduled
+[wp] [Qed] Goal typed_simpl_r_terminates : Valid
+[wp] [Qed] Goal typed_simpl_rf_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_simpl_rf_terminates_part2 : Unsuccess
+[wp] [Qed] Goal typed_mutual_1_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_mutual_1_terminates_part2 : Unsuccess
+[wp] [Qed] Goal typed_mutual_2_terminates : Valid
+[wp] [Qed] Goal typed_caller_no_cluster_terminates : Valid
+[wp] [Qed] Goal typed_fp_call_point_function_s41 : Valid
+[wp] [Qed] Goal typed_fp_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_fp_terminates_part2 : Unsuccess
+[wp] [Qed] Goal typed_function_terminates : Valid
+[wp] [Qed] Goal typed_function_call_fp_requires : Valid
+[wp] Proved goals:   10 / 13
+  Qed:             9 
+  Alt-Ergo:        0  (unsuccess: 3)
+------------------------------------------------------------
+ Functions                 WP     Alt-Ergo  Total   Success
+  simpl_r                   1        -        1       100%
+  simpl_rf                  1        -        2      50.0%
+  mutual_1                  1        -        2      50.0%
+  mutual_2                  1        -        1       100%
+  caller_no_cluster         1        -        1       100%
+  fp                        2        -        3      66.7%
+  function                  2        -        2       100%
+------------------------------------------------------------
diff --git a/src/plugins/wp/tests/wp_acsl/oracle_qualif/terminates_fp.res.oracle b/src/plugins/wp/tests/wp_acsl/oracle_qualif/terminates_fp.res.oracle
new file mode 100644
index 0000000000000000000000000000000000000000..080a16135ea4b4f841d9a85a5bad19597e487f6d
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/oracle_qualif/terminates_fp.res.oracle
@@ -0,0 +1,73 @@
+# frama-c -wp [...]
+[kernel] Parsing tests/wp_acsl/terminates_fp.i (no preprocessing)
+[wp] Running WP plugin...
+[wp] tests/wp_acsl/terminates_fp.i:42: Warning: 
+  Missing terminates clause for no_t_spec, populates 'terminates \true'
+[wp] Warning: Missing RTE guards
+[wp] tests/wp_acsl/terminates_fp.i:62: Warning: 
+  Missing terminates clause for no_t_spec_cond, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:70: Warning: 
+  Missing terminates clause for no_t_spec_cond_m, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:56: Warning: 
+  Missing terminates clause for no_t_spec_in_bhv, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:22: Warning: 
+  Missing terminates clause for t_spec, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:36: Warning: 
+  Missing terminates clause for t_spec_in_bhv, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  Missing terminates clause for warns_recursive, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  In 'warns_recursive', no 'calls' specification for statement(s) on line(s): 
+  line 79, 
+  Assuming that they can call 'warns_recursive'
+[wp] tests/wp_acsl/terminates_fp.i:9: Warning: 
+  Missing terminates clause for gt, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:11: Warning: 
+  Missing terminates clause for ht, populates 'terminates \true'
+[wp] tests/wp_acsl/terminates_fp.i:79: Warning: 
+  Missing 'calls' for default behavior
+[wp] tests/wp_acsl/terminates_fp.i:78: Warning: 
+  No 'decreases' clause on recursive function 'warns_recursive', cannot prove termination
+[wp] 29 goals scheduled
+[wp] [Qed] Goal typed_t_spec_call_point_gt_ht_s2 : Valid
+[wp] [Qed] Goal typed_t_spec_terminates : Valid
+[wp] [Alt-Ergo] Goal typed_t_spec_in_bhv_call_point_gt_hnt_s6 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_complete_B1_B2 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_disjoint_B1_B2 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_terminates_part1 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_terminates_part2 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_call_point_gt_hnt_s6 : Valid
+[wp] [Qed] Goal typed_t_spec_in_bhv_call_point_gt_hnt_s6 : Valid
+[wp] [Qed] Goal typed_no_t_spec_call_point_gnt_hnt_s10 : Valid
+[wp] [Qed] Goal typed_no_t_spec_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_no_t_spec_terminates_part2 : Unsuccess
+[wp] [Alt-Ergo] Goal typed_no_t_spec_in_bhv_call_point_gnt_hnt_s14 : Valid
+[wp] [Qed] Goal typed_no_t_spec_in_bhv_complete_B1_B2 : Valid
+[wp] [Qed] Goal typed_no_t_spec_in_bhv_disjoint_B1_B2 : Valid
+[wp] [Qed] Goal typed_no_t_spec_in_bhv_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_no_t_spec_in_bhv_terminates_part2 : Unsuccess
+[wp] [Qed] Goal typed_no_t_spec_in_bhv_call_point_gnt_hnt_s14 : Valid
+[wp] [Qed] Goal typed_no_t_spec_in_bhv_call_point_gnt_hnt_s14 : Valid
+[wp] [Qed] Goal typed_no_t_spec_cond_call_point_gt_hnt_s18 : Valid
+[wp] [Qed] Goal typed_no_t_spec_cond_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_no_t_spec_cond_terminates_part2 : Unsuccess
+[wp] [Qed] Goal typed_no_t_spec_cond_m_call_point_gt_ht_s22 : Valid
+[wp] [Qed] Goal typed_no_t_spec_cond_m_call_point_hnt_s24_part1 : Valid
+[wp] [Qed] Goal typed_no_t_spec_cond_m_call_point_hnt_s24_part2 : Valid
+[wp] [Qed] Goal typed_no_t_spec_cond_m_terminates_part1 : Valid
+[wp] [Alt-Ergo] Goal typed_no_t_spec_cond_m_terminates_part2 : Unsuccess
+[wp] [Alt-Ergo] Goal typed_no_t_spec_cond_m_terminates_part3 : Unsuccess
+[wp] [Alt-Ergo] Goal typed_warns_recursive_terminates : Unsuccess
+[wp] Proved goals:   23 / 29
+  Qed:            21 
+  Alt-Ergo:        2  (unsuccess: 6)
+------------------------------------------------------------
+ Functions                 WP     Alt-Ergo  Total   Success
+  t_spec                    2        -        2       100%
+  t_spec_in_bhv             6        1        7       100%
+  no_t_spec                 2        -        3      66.7%
+  no_t_spec_in_bhv          5        1        7      85.7%
+  no_t_spec_cond            2        -        3      66.7%
+  no_t_spec_cond_m          4        -        6      66.7%
+  warns_recursive           -        -        1       0.0%
+------------------------------------------------------------
diff --git a/src/plugins/wp/tests/wp_acsl/terminates_fp.i b/src/plugins/wp/tests/wp_acsl/terminates_fp.i
new file mode 100644
index 0000000000000000000000000000000000000000..533ea26d3c1ae4d84888bd0cedf9b0b8d066d0b1
--- /dev/null
+++ b/src/plugins/wp/tests/wp_acsl/terminates_fp.i
@@ -0,0 +1,80 @@
+/* run.config
+   OPT: -wp-definitions-terminate -wp-declarations-terminate
+*/
+/* run.config_qualif
+   OPT: -wp-definitions-terminate -wp-declarations-terminate
+*/
+
+//@ assigns \nothing ;
+void gt(int);
+//@ assigns \nothing ;
+void ht(int);
+
+/*@ terminates \false ;
+    assigns \nothing ; */
+void gnt(int i);
+
+/*@ terminates i == 0 ;
+    assigns \nothing ; */
+void hnt(int i);
+
+//@ requires f == gt || f == ht ;
+void t_spec(void (*f) (int)){
+  //@ calls gt, ht ;
+  f(0);
+}
+
+/*@ behavior B1:
+      assumes x == 0 ;
+      requires f == hnt ;
+    behavior B2:
+      assumes x != 0 ;
+      requires f == gt ;
+    complete behaviors ;
+    disjoint behaviors ;
+*/
+void t_spec_in_bhv(void (*f) (int), int x){
+  //@ calls gt, hnt ;
+  f(x);
+}
+
+//@ requires f == gnt || f == hnt ;
+void no_t_spec(void (*f) (int)){
+  //@ calls gnt, hnt ;
+  f(0);
+}
+
+/*@ behavior B1:
+      assumes x == 0 ;
+      requires f == gnt ;
+    behavior B2:
+      assumes x != 0 ;
+      requires f == hnt ;
+    complete behaviors ;
+    disjoint behaviors ;
+*/
+void no_t_spec_in_bhv(void (*f) (int), int x){
+  //@ calls gnt, hnt ;
+  f(0);
+}
+
+//@ requires f == gt || f == hnt ;
+void no_t_spec_cond(void (*f) (int)){
+  //@ calls gt, hnt ;
+  f(1);
+}
+
+/*@ requires f == gt || f == ht ;
+    requires g == hnt ;
+*/
+void no_t_spec_cond_m(void (*f) (int), void (*g) (int)){
+  //@ calls gt, ht ;
+  f(0);
+
+  //@ calls hnt ;
+  g(1);
+}
+
+void warns_recursive(void (*f) (int)){
+  f(42);
+}
diff --git a/src/plugins/wp/wpAnnot.ml b/src/plugins/wp/wpAnnot.ml
index 7f6b34561c7a477d185f523f4a2062febb460a50..7a0dc8f397154c76cfcfc13f8aa029173ab82de8 100644
--- a/src/plugins/wp/wpAnnot.ml
+++ b/src/plugins/wp/wpAnnot.ml
@@ -89,11 +89,12 @@ let wp_trivially_terminates =
     ~correctness:[] (* TBC *)
     ~tuning:[] (* TBC *)
 
-let set_trivially_terminates p =
+let set_trivially_terminates p hyps =
   incr trivial_terminates ;
   Wp_parameters.result "[CFG] Goal %a : Valid (Trivial)" WpPropId.pp_propid p ;
   let pid = WpPropId.property_of_id p in
-  Property_status.emit wp_trivially_terminates ~hyps:[] pid Property_status.True
+  let hyps = Property.Set.elements hyps in
+  Property_status.emit wp_trivially_terminates ~hyps pid Property_status.True
 
 (* -------------------------------------------------------------------------- *)
 (* --- Preconditions at Callsites                                         --- *)
diff --git a/src/plugins/wp/wpAnnot.mli b/src/plugins/wp/wpAnnot.mli
index 171a83e4299dd5991544d77a0d13e102a0a1c3e3..a2b4ebcf85106f1964d12a6273f8296d4ced158e 100644
--- a/src/plugins/wp/wpAnnot.mli
+++ b/src/plugins/wp/wpAnnot.mli
@@ -36,7 +36,7 @@ val unreachable_proved : int ref
 val unreachable_failed : int ref
 val trivial_terminates : int ref
 val set_unreachable : WpPropId.prop_id -> unit
-val set_trivially_terminates : WpPropId.prop_id -> unit
+val set_trivially_terminates : WpPropId.prop_id -> Property.Set.t -> unit
 
 type asked_assigns = NoAssigns | OnlyAssigns | WithAssigns
 
diff --git a/src/plugins/wp/wpPropId.ml b/src/plugins/wp/wpPropId.ml
index 7d30a86199f42b9353311eee209fd5e65edcfbf8..904d488371713df01e86d8f875623224f47e8378 100644
--- a/src/plugins/wp/wpPropId.ml
+++ b/src/plugins/wp/wpPropId.ml
@@ -150,7 +150,7 @@ let mk_disj_bhv_id (kf,ki,active,disj)  =
   mk_prop PKProp (Property.ip_of_disjoint kf ki active disj)
 let mk_compl_bhv_id (kf,ki,active,comp) =
   mk_prop PKProp (Property.ip_of_complete kf ki active comp)
-let mk_decrease_id (kf, s, x)  =
+let mk_decrease_id kf s x  =
   mk_prop PKProp (Property.ip_of_decreases kf s x)
 
 let mk_lemma_id l = mk_prop PKProp (LogicUsage.ip_lemma l)
@@ -934,6 +934,7 @@ let pp_assigns_desc fmt a = Wp_error.pp_assigns fmt a.a_assigns
 (*----------------------------------------------------------------------------*)
 
 type pred_info = prop_id * Cil_types.predicate
+type variant_info = prop_id * Cil_types.variant
 
 let mk_pred_info id p = (id, p)
 let pred_info_id (id, _) = id
diff --git a/src/plugins/wp/wpPropId.mli b/src/plugins/wp/wpPropId.mli
index 643b69d72d03055c3323e9dda7e53b95be818221..4be96caf2049b165a4fe196bc98ffd91f1701e07 100644
--- a/src/plugins/wp/wpPropId.mli
+++ b/src/plugins/wp/wpPropId.mli
@@ -173,7 +173,7 @@ val mk_disj_bhv_id :
 val mk_compl_bhv_id :
   kernel_function * kinstr * string list * string list -> prop_id
 
-val mk_decrease_id : kernel_function * kinstr * variant -> prop_id
+val mk_decrease_id : kernel_function -> kinstr -> variant -> prop_id
 
 (** axiom identification *)
 val mk_lemma_id : logic_lemma -> prop_id
@@ -265,7 +265,8 @@ type axiom_info = prop_id * LogicUsage.logic_lemma
 val mk_axiom_info : LogicUsage.logic_lemma -> axiom_info
 val pp_axiom_info : Format.formatter -> axiom_info -> unit
 
-type pred_info = (prop_id * Cil_types.predicate)
+type pred_info = prop_id * Cil_types.predicate
+type variant_info = prop_id * Cil_types.variant
 
 val mk_pred_info : prop_id -> Cil_types.predicate -> pred_info
 val pred_info_id : pred_info -> prop_id
diff --git a/src/plugins/wp/wp_parameters.ml b/src/plugins/wp/wp_parameters.ml
index 528a9b251ff910b2b78a1cf4303b73e8259e9718..d037da922d3448206ffcd2da3aea622730a853c8 100644
--- a/src/plugins/wp/wp_parameters.ml
+++ b/src/plugins/wp/wp_parameters.ml
@@ -113,12 +113,12 @@ module Properties =
       let arg_name = "p,..."
       let help =
         "Select properties based names and category.\n\
-         Use +name or +category to select properties and -name or -category\n\
+         Use +name or +category to select properties and -name or -category \
          to remove them from the selection. The '+' sign can be omitted.\n\
-         Categories are: @lemma, @requires, @assigns, @ensures, @exits,\n\
-         @assert, @invariant, @variant, @breaks, @continues, @returns,\n\
-         @complete_behaviors, @disjoint_behaviors and\n\
-         @check (which includes all check clauses)."
+         Categories are: @lemma, @requires, @assigns, @ensures, @exits, \
+         @assert, @invariant, @variant, @breaks, @continues, @returns, \
+         @complete_behaviors, @disjoint_behaviors, @terminates, \
+         @decreases and @check (which includes all check clauses)."
     end)
 let () = on_reset Properties.clear
 
@@ -487,7 +487,7 @@ module TerminatesDefinitions =
                 considered to terminate when called."
   end)
 
-module TerminatesFCDeclarations =
+module TerminatesStdlibDeclarations =
   False(struct
     let option_name = "-wp-frama-c-stdlib-terminate"
     let help = "Frama-C stdlib functions without terminates specification \
diff --git a/src/plugins/wp/wp_parameters.mli b/src/plugins/wp/wp_parameters.mli
index bcd61322625875596952fe032fdd4899c4000e4e..9c7961f3df2bda0a0235f294a216c67a41372501 100644
--- a/src/plugins/wp/wp_parameters.mli
+++ b/src/plugins/wp/wp_parameters.mli
@@ -102,7 +102,7 @@ module SimplifyType : Parameter_sig.Bool
 module CalleePreCond : Parameter_sig.Bool
 module PrecondWeakening : Parameter_sig.Bool
 module TerminatesExtDeclarations : Parameter_sig.Bool
-module TerminatesFCDeclarations : Parameter_sig.Bool
+module TerminatesStdlibDeclarations : Parameter_sig.Bool
 module TerminatesDefinitions : Parameter_sig.Bool
 module TerminatesVariantHyp : Parameter_sig.Bool