(* camlp4r *) (****************************************************************************) (* *) (* OCaml *) (* *) (* INRIA Rocquencourt *) (* *) (* Copyright 2006 Institut National de Recherche en Informatique et *) (* en Automatique. All rights reserved. This file is distributed under *) (* the terms of the GNU Library General Public License, with the special *) (* exception on linking described in LICENSE at the top of the Camlp4 *) (* source tree. *) (* *) (****************************************************************************) (* Authors: * - Nicolas Pouillard: initial version *) module Make (Ast : Sig.Camlp4Ast) = struct module S = Set.Make String; class c_fold_pattern_vars ['accu] f init = object inherit Ast.fold as super; value acc = init; method acc : 'accu = acc; method patt = fun [ <:patt< $lid:s$ >> | <:patt< ~ $s$ >> | <:patt< ? $s$ >> -> {< acc = f s acc >} | p -> super#patt p ]; end; value fold_pattern_vars f p init = ((new c_fold_pattern_vars f init)#patt p)#acc; value rec fold_binding_vars f bi acc = match bi with [ <:binding< $bi1$ and $bi2$ >> -> fold_binding_vars f bi1 (fold_binding_vars f bi2 acc) | <:binding< $p$ = $_$ >> -> fold_pattern_vars f p acc | <:binding<>> -> acc | <:binding< $anti:_$ >> -> assert False ]; class fold_free_vars ['accu] (f : string -> 'accu -> 'accu) ?(env_init = S.empty) free_init = object (o) inherit Ast.fold as super; value free : 'accu = free_init; value env : S.t = env_init; method free = free; method set_env env = {< env = env >}; method add_atom s = {< env = S.add s env >}; method add_patt p = {< env = fold_pattern_vars S.add p env >}; method add_binding bi = {< env = fold_binding_vars S.add bi env >}; method expr = fun [ <:expr< $lid:s$ >> | <:expr< ~ $s$ >> | <:expr< ? $s$ >> -> if S.mem s env then o else {< free = f s free >} | <:expr< let $bi$ in $e$ >> -> (((o#add_binding bi)#expr e)#set_env env)#binding bi | <:expr< let rec $bi$ in $e$ >> -> (((o#add_binding bi)#expr e)#binding bi)#set_env env | <:expr< for $p$ = $e1$ $to:_$ $e2$ do { $e3$ } >> -> ((((o#expr e1)#expr e2)#patt p)#expr e3)#set_env env | <:expr< $id:_$ >> | <:expr< new $_$ >> -> o | <:expr< object ($p$) $cst$ end >> -> ((o#add_patt p)#class_str_item cst)#set_env env | e -> super#expr e ]; method match_case = fun [ <:match_case< $p$ when $e1$ -> $e2$ >> -> (((o#add_patt p)#expr e1)#expr e2)#set_env env | m -> super#match_case m ]; method str_item = fun [ <:str_item< external $s$ : $t$ = $_$ >> -> (o#ctyp t)#add_atom s | <:str_item< value $bi$ >> -> (o#binding bi)#add_binding bi | <:str_item< value rec $bi$ >> -> (o#add_binding bi)#binding bi | st -> super#str_item st ]; method class_expr = fun [ <:class_expr< fun $p$ -> $ce$ >> -> ((o#add_patt p)#class_expr ce)#set_env env | <:class_expr< let $bi$ in $ce$ >> -> (((o#binding bi)#add_binding bi)#class_expr ce)#set_env env | <:class_expr< let rec $bi$ in $ce$ >> -> (((o#add_binding bi)#binding bi)#class_expr ce)#set_env env | <:class_expr< object ($p$) $cst$ end >> -> ((o#add_patt p)#class_str_item cst)#set_env env | ce -> super#class_expr ce ]; method class_str_item = fun [ <:class_str_item< inherit $override:_$ $_$ >> as cst -> super#class_str_item cst | <:class_str_item< inherit $override:_$ $ce$ as $s$ >> -> (o#class_expr ce)#add_atom s | <:class_str_item< value $override:_$ $mutable:_$ $s$ = $e$ >> -> (o#expr e)#add_atom s | <:class_str_item< value virtual $mutable:_$ $s$ : $t$ >> -> (o#ctyp t)#add_atom s | cst -> super#class_str_item cst ]; method module_expr = fun [ <:module_expr< struct $st$ end >> -> (o#str_item st)#set_env env | me -> super#module_expr me ]; end; value free_vars env_init e = let fold = new fold_free_vars S.add ~env_init S.empty in (fold#expr e)#free; end;