123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171(**************************************************************************)(* *)(* This file is part of Frama-C. *)(* *)(* Copyright (C) 2007-2024 *)(* CEA (Commissariat à l'énergie atomique et aux énergies *)(* alternatives) *)(* *)(* you can redistribute it and/or modify it under the terms of the GNU *)(* Lesser General Public License as published by the Free Software *)(* Foundation, version 2.1. *)(* *)(* It 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 Lesser General Public License for more details. *)(* *)(* See the GNU Lesser General Public License version 2.1 *)(* for more details (enclosed in the file licenses/LGPLv2.1). *)(* *)(**************************************************************************)openCil_typesmoduleEdgeLabel=Abstract_state.EdgeLabel(** Points-to graphs datastructure. *)moduleG=Abstract_state.Gtypev=G.V.tletvid=Abstract_state.vidmoduleLSet=Abstract_state.LSetmoduleVarSet=Abstract_state.VarSetmoduleAbstract_state=Abstract_stateletcheck_computed()=ifnot(Analysis.is_computed())thenOptions.abort"Static analysis must be called before any function of the API can be called"letget_of_stmt~stmtempty(get:Abstract_state.t->'a)=check_computed();matchAnalysis.get_state_before_stmtstmtwith|None->empty|Somestate->getstateletlset~stmtget_set=get_of_stmt~stmtLSet.emptyget_setletvars~stmtget_set=get_of_stmt~stmtVarSet.emptyget_setletget_list~stmtget=get_of_stmt~stmt[]getmoduleStatement=structletpoints_to_vars~stmtlv=vars~stmt(Abstract_state.points_to_varslv)letpoints_to_lvals~stmtlv=lset~stmt(Abstract_state.points_to_lvalslv)letalias_sets_vars~stmt=get_list~stmtAbstract_state.alias_sets_varsletalias_sets_lvals~stmt=get_list~stmtAbstract_state.alias_sets_lvalsletalias_vars~stmtlv=vars~stmt(Abstract_state.alias_varslv)letalias_lvals~stmtlv=lset~stmt(Abstract_state.alias_lvalslv)letaliases=alias_lvals(* deprecated *)letnew_aliases_lvals~stmtlv=letget_setstate=letnew_state=Analysis.do_stmtstatestmtinAbstract_state.alias_lvalslvnew_stateinlset~stmt:stmtget_setletnew_aliases_vars~stmtlv=letget_setstate=letnew_state=Analysis.do_stmtstatestmtinAbstract_state.alias_varslvnew_stateinvars~stmt:stmtget_setletare_aliased~stmt(lv1:lval)(lv2:lval):bool=(* TODO: more efficient algorithm: do they share a successor? *)LSet.memlv2@@aliases~stmtlv1endletpoints_to_set_stmt_kfstmt=Statement.points_to_lvals~stmtletaliases_stmt_kfstmt=Statement.aliases~stmtmoduleFunction=structletreturn_stmtkf=ifKernel_function.has_definitionkfthenKernel_function.find_returnkfelseOptions.abort"function %a has no definition"Kernel_function.prettykfletpoints_to_vars~kf=Statement.points_to_vars~stmt:(return_stmtkf)letpoints_to_lvals~kf=Statement.points_to_lvals~stmt:(return_stmtkf)letalias_sets_vars~kf=Statement.alias_sets_vars~stmt:(return_stmtkf)letalias_sets_lvals~kf=Statement.alias_sets_lvals~stmt:(return_stmtkf)letalias_vars~kf=Statement.alias_vars~stmt:(return_stmtkf)letalias_lvals~kf=Statement.alias_lvals~stmt:(return_stmtkf)letaliases=alias_lvals(* deprecated *)letare_aliased~kf=Statement.are_aliased~stmt:(return_stmtkf)letfundec_stmts~kflv=ifKernel_function.has_definitionkfthenList.map(funstmt->stmt,Statement.new_aliases_lvals~stmtlv)(Kernel_function.get_definitionkf).sallstmtselseOptions.abort"fundec_stmts: function %a has no definition"Kernel_function.prettykfendletpoints_to_set_kfkf=Function.points_to_lvals~kfletaliases_kfkf=Function.aliases~kfletfundec_stmtskf=Function.fundec_stmts~kfletfold_points_to_setf_foldacckfslv=LSet.fold(funea->f_foldae)(points_to_set_stmtkfslv)accletfold_aliases_stmtf_foldacckfslv=LSet.fold(funea->f_foldae)(aliases_stmtkfslv)accletfold_new_aliases_stmtf_foldacc_kfslv=LSet.fold(funea->f_foldae)(Statement.new_aliases_lvals~stmt:slv)accletfold_points_to_set_kf(f_fold:'a->lval->'a)(acc:'a)(kf:kernel_function)(lv:lval):'a=LSet.fold(funea->f_foldae)(points_to_set_kfkflv)accletfold_aliases_kf(f_fold:'a->lval->'a)(acc:'a)kflv:'a=LSet.fold(funea->f_foldae)(aliases_kfkflv)accletfold_fundec_stmts(f_fold:'a->stmt->lval->'a)(acc:'a)(kf:kernel_function)(lv:lval):'a=List.fold_left(funacc(s,set)->LSet.fold(funlva->f_foldaslv)setacc)acc(fundec_stmtskflv)letare_aliased(_kf:kernel_function)stmt=Statement.are_aliased~stmtletfold_vertex(f_fold:'a->G.V.t->lval->'a)(acc:'a)(_kf:kernel_function)(s:stmt)(lv:lval):'a=check_computed();matchAnalysis.get_state_before_stmtswithNone->acc|Somestate->letv:G.V.t=Abstract_state.find_vertexlvstateinletset_aliases=Abstract_state.find_synonymslvstateinLSet.fold(funlva->f_foldavlv)set_aliasesaccletfold_vertex_closure(f_fold:'a->G.V.t->lval->'a)(acc:'a)(_kf:kernel_function)(s:stmt)(lv:lval):'a=check_computed();matchAnalysis.get_state_before_stmtswithNone->acc|Somestate->letlist_closure:(G.V.t*LSet.t)list=Abstract_state.find_transitive_closurelvstateinList.fold_left(funacc(i,s)->LSet.fold(funlva->f_foldailv)sacc)acclist_closureletget_state_before_stmt_kf=Analysis.get_state_before_stmtletcall_functionafresargs=matchAnalysis.get_summaryfwithNone->None|Somesu->Some(Abstract_state.callaresargssu)letsimplify_lval=Simplified.Lval.simplify