Skip to content

Commit 7ad2120

Browse files
committed
Add module to deal with 'front-end' built-in attributes.
1 parent 7fc4265 commit 7ad2120

15 files changed

+274
-240
lines changed

Makefile.shared

+2-3
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,8 @@ PARSING=parsing/location.cmo parsing/longident.cmo \
4646
parsing/syntaxerr.cmo parsing/parser.cmo \
4747
parsing/lexer.cmo parsing/parse.cmo parsing/printast.cmo \
4848
parsing/pprintast.cmo \
49-
parsing/ast_mapper.cmo parsing/attr_helper.cmo
49+
parsing/ast_mapper.cmo parsing/attr_helper.cmo \
50+
parsing/builtin_attributes.cmo
5051

5152
TYPING=typing/ident.cmo typing/path.cmo \
5253
typing/primitive.cmo typing/types.cmo \
@@ -132,5 +133,3 @@ NATTOPOBJS=$(UTILS) $(PARSING) $(TYPING) $(COMP) $(ASMCOMP) \
132133
toplevel/opttopmain.cmo toplevel/opttopstart.cmo
133134

134135
PERVASIVES=$(STDLIB_MODULES) outcometree topdirs toploop
135-
136-

debugger/Makefile.shared

+1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ OTHEROBJS=\
3737
../parsing/location.cmo ../parsing/longident.cmo ../parsing/docstrings.cmo \
3838
../parsing/ast_helper.cmo ../parsing/ast_mapper.cmo \
3939
../parsing/attr_helper.cmo \
40+
../parsing/builtin_attributes.cmo \
4041
../typing/ident.cmo ../typing/path.cmo ../typing/types.cmo \
4142
../typing/btype.cmo ../typing/primitive.cmo ../typing/typedtree.cmo \
4243
../typing/subst.cmo ../typing/predef.cmo \

driver/compile.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ let interface ppf sourcefile outputprefix =
4242
Typecore.force_delayed_checks ();
4343
Warnings.check_fatal ();
4444
if not !Clflags.print_types then begin
45-
let deprecated = Typetexp.deprecated_of_sig ast in
45+
let deprecated = Builtin_attributes.deprecated_of_sig ast in
4646
let sg = Env.save_signature ~deprecated sg modulename (outputprefix ^ ".cmi") in
4747
Typemod.save_signature modulename tsg outputprefix sourcefile
4848
initial_env sg ;

driver/optcompile.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ let interface ppf sourcefile outputprefix =
4343
Typecore.force_delayed_checks ();
4444
Warnings.check_fatal ();
4545
if not !Clflags.print_types then begin
46-
let deprecated = Typetexp.deprecated_of_sig ast in
46+
let deprecated = Builtin_attributes.deprecated_of_sig ast in
4747
let sg = Env.save_signature ~deprecated sg modulename (outputprefix ^ ".cmi") in
4848
Typemod.save_signature modulename tsg outputprefix sourcefile
4949
initial_env sg ;

ocamldoc/Makefile.nt

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ CAMLYACC ?= ../boot/ocamlyacc
1818
##########################
1919
ROOTDIR = ..
2020
OCAMLC = $(CAMLRUN) $(ROOTDIR)/ocamlc -nostdlib -I $(ROOTDIR)/stdlib
21-
OCAMLOPT = $(CAMLRUN) $(ROOTDIR)/ocamlopt -nostdlib -I $(ROOTDIR)/stdlib
21+
OCAMLOPT = $(CAMLRUN) $(ROOTDIR)/ocamlopt -nostdlib -I $(ROOTDIR)/stdlib -inline 0
2222
OCAMLDEP = $(CAMLRUN) $(ROOTDIR)/tools/ocamldep
2323
OCAMLLEX = $(CAMLRUN) $(ROOTDIR)/boot/ocamllex
2424
OCAMLLIB = $(LIBDIR)

otherlibs/dynlink/Makefile

+1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ COMPILEROBJS=\
3737
../../parsing/location.cmo ../../parsing/longident.cmo \
3838
../../parsing/docstrings.cmo ../../parsing/ast_helper.cmo \
3939
../../parsing/ast_mapper.cmo ../../parsing/attr_helper.cmo \
40+
../../parsing/builtin_attributes.cmo \
4041
../../typing/ident.cmo ../../typing/path.cmo \
4142
../../typing/primitive.cmo ../../typing/types.cmo \
4243
../../typing/btype.cmo ../../typing/subst.cmo ../../typing/predef.cmo \

parsing/builtin_attributes.ml

+158
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,158 @@
1+
(***********************************************************************)
2+
(* *)
3+
(* OCaml *)
4+
(* *)
5+
(* Alain Frisch, LexiFi *)
6+
(* *)
7+
(* Copyright 2012 Institut National de Recherche en Informatique et *)
8+
(* en Automatique. All rights reserved. This file is distributed *)
9+
(* under the terms of the Q Public License version 1.0. *)
10+
(* *)
11+
(***********************************************************************)
12+
13+
open Asttypes
14+
open Parsetree
15+
16+
let string_of_cst = function
17+
| Const_string(s, _) -> Some s
18+
| _ -> None
19+
20+
let string_of_payload = function
21+
| PStr[{pstr_desc=Pstr_eval({pexp_desc=Pexp_constant c},_)}] ->
22+
string_of_cst c
23+
| _ -> None
24+
25+
let rec error_of_extension ext =
26+
match ext with
27+
| ({txt = ("ocaml.error"|"error") as txt; loc}, p) ->
28+
let rec sub_from inner =
29+
match inner with
30+
| {pstr_desc=Pstr_extension (ext, _)} :: rest ->
31+
error_of_extension ext :: sub_from rest
32+
| {pstr_loc} :: rest ->
33+
(Location.errorf ~loc
34+
"Invalid syntax for sub-error of extension '%s'." txt) ::
35+
sub_from rest
36+
| [] -> []
37+
in
38+
begin match p with
39+
| PStr({pstr_desc=Pstr_eval
40+
({pexp_desc=Pexp_constant(Const_string(msg,_))}, _)}::
41+
{pstr_desc=Pstr_eval
42+
({pexp_desc=Pexp_constant(Const_string(if_highlight,_))}, _)}::
43+
inner) ->
44+
Location.error ~loc ~if_highlight ~sub:(sub_from inner) msg
45+
| PStr({pstr_desc=Pstr_eval
46+
({pexp_desc=Pexp_constant(Const_string(msg,_))}, _)}::inner) ->
47+
Location.error ~loc ~sub:(sub_from inner) msg
48+
| _ -> Location.errorf ~loc "Invalid syntax for extension '%s'." txt
49+
end
50+
| ({txt; loc}, _) ->
51+
Location.errorf ~loc "Uninterpreted extension '%s'." txt
52+
53+
let rec deprecated_of_attrs = function
54+
| [] -> None
55+
| ({txt = "ocaml.deprecated"|"deprecated"; _}, p) :: _ ->
56+
begin match string_of_payload p with
57+
| Some txt -> Some txt
58+
| None -> Some ""
59+
end
60+
| _ :: tl -> deprecated_of_attrs tl
61+
62+
let check_deprecated loc attrs s =
63+
match deprecated_of_attrs attrs with
64+
| None -> ()
65+
| Some "" -> Location.prerr_warning loc (Warnings.Deprecated s)
66+
| Some txt -> Location.prerr_warning loc (Warnings.Deprecated (s ^ "\n" ^ txt))
67+
68+
let rec deprecated_of_sig = function
69+
| {psig_desc = Psig_attribute a} :: tl ->
70+
begin match deprecated_of_attrs [a] with
71+
| None -> deprecated_of_sig tl
72+
| Some _ as r -> r
73+
end
74+
| _ -> None
75+
76+
77+
let rec deprecated_of_str = function
78+
| {pstr_desc = Pstr_attribute a} :: tl ->
79+
begin match deprecated_of_attrs [a] with
80+
| None -> deprecated_of_str tl
81+
| Some _ as r -> r
82+
end
83+
| _ -> None
84+
85+
86+
let emit_external_warnings =
87+
(* Note: this is run as a preliminary pass when type-checking an
88+
interface or implementation. This allows to cover all kinds of
89+
attributes, but the drawback is that it doesn't take local
90+
configuration of warnings (with '@@warning'/'@@warnerror'
91+
attributes) into account. We should rather check for
92+
'ppwarning' attributes during the actual type-checking, making
93+
sure to cover all contexts (easier and more ugly alternative:
94+
duplicate here the logic which control warnings locally). *)
95+
let open Ast_mapper in
96+
{
97+
default_mapper with
98+
attribute = (fun _ a ->
99+
begin match a with
100+
| {txt="ocaml.ppwarning"|"ppwarning"},
101+
PStr[{pstr_desc=Pstr_eval({pexp_desc=Pexp_constant
102+
(Const_string (s, _))},_);
103+
pstr_loc}] ->
104+
Location.prerr_warning pstr_loc (Warnings.Preprocessor s)
105+
| _ -> ()
106+
end;
107+
a
108+
)
109+
}
110+
111+
112+
let warning_scope = ref []
113+
114+
let warning_enter_scope () =
115+
warning_scope := (Warnings.backup ()) :: !warning_scope
116+
let warning_leave_scope () =
117+
match !warning_scope with
118+
| [] -> assert false
119+
| hd :: tl ->
120+
Warnings.restore hd;
121+
warning_scope := tl
122+
123+
let warning_attribute attrs =
124+
let process loc txt errflag payload =
125+
match string_of_payload payload with
126+
| Some s ->
127+
begin try Warnings.parse_options errflag s
128+
with Arg.Bad _ ->
129+
Location.prerr_warning loc
130+
(Warnings.Attribute_payload
131+
(txt, "Ill-formed list of warnings"))
132+
end
133+
| None ->
134+
Location.prerr_warning loc
135+
(Warnings.Attribute_payload
136+
(txt, "A single string literal is expected"))
137+
in
138+
List.iter
139+
(function
140+
| ({txt = ("ocaml.warning"|"warning") as txt; loc}, payload) ->
141+
process loc txt false payload
142+
| ({txt = ("ocaml.warnerror"|"warnerror") as txt; loc}, payload) ->
143+
process loc txt true payload
144+
| _ ->
145+
()
146+
)
147+
attrs
148+
149+
let with_warning_attribute attrs f =
150+
try
151+
warning_enter_scope ();
152+
warning_attribute attrs;
153+
let ret = f () in
154+
warning_leave_scope ();
155+
ret
156+
with exn ->
157+
warning_leave_scope ();
158+
raise exn

parsing/builtin_attributes.mli

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
(***********************************************************************)
2+
(* *)
3+
(* OCaml *)
4+
(* *)
5+
(* Alain Frisch, LexiFi *)
6+
(* *)
7+
(* Copyright 2012 Institut National de Recherche en Informatique et *)
8+
(* en Automatique. All rights reserved. This file is distributed *)
9+
(* under the terms of the Q Public License version 1.0. *)
10+
(* *)
11+
(***********************************************************************)
12+
13+
(* Support for some of the builtin attributes:
14+
15+
ocaml.deprecated
16+
ocaml.error
17+
ocaml.ppwarning
18+
ocaml.warning
19+
ocaml.warnerror
20+
21+
*)
22+
23+
24+
val check_deprecated: Location.t -> Parsetree.attributes -> string -> unit
25+
val deprecated_of_attrs: Parsetree.attributes -> string option
26+
val deprecated_of_sig: Parsetree.signature -> string option
27+
val deprecated_of_str: Parsetree.structure -> string option
28+
29+
val error_of_extension: Parsetree.extension -> Location.error
30+
31+
val warning_enter_scope: unit -> unit
32+
val warning_leave_scope: unit -> unit
33+
val warning_attribute: Parsetree.attributes -> unit
34+
val with_warning_attribute: Parsetree.attributes -> (unit -> 'a) -> 'a
35+
36+
val emit_external_warnings: Ast_mapper.mapper

typing/env.ml

+6-11
Original file line numberDiff line numberDiff line change
@@ -312,14 +312,6 @@ let strengthen =
312312
ref ((fun env mty path -> assert false) :
313313
t -> module_type -> Path.t -> module_type)
314314

315-
let deprecated_of_attrs_forward = ref (fun _ -> None)
316-
(* to be filled with Typetexp.deprecated_of_attrs.
317-
Note: ocamldebug link with Env (and use its lookup functions)
318-
but not Typetexp. So we return None instead of failing
319-
to avoid breaking the debugger. *)
320-
321-
let deprecated_of_attrs attrs = !deprecated_of_attrs_forward attrs
322-
323315
let md md_type =
324316
{md_type; md_attributes=[]; md_loc=Location.none}
325317

@@ -810,7 +802,8 @@ and lookup_module ~load ?loc lid env : Path.t =
810802
raise Recmodule
811803
| _ -> ()
812804
end;
813-
report_deprecated ?loc p (deprecated_of_attrs md_attributes);
805+
report_deprecated ?loc p
806+
(Builtin_attributes.deprecated_of_attrs md_attributes);
814807
p
815808
with Not_found ->
816809
if s = !current_unit then raise Not_found;
@@ -1382,7 +1375,9 @@ and components_of_module_maker (env, sub, path, mty) =
13821375
let mty' = EnvLazy.create (sub, mty) in
13831376
c.comp_modules <-
13841377
Tbl.add (Ident.name id) (mty', !pos) c.comp_modules;
1385-
let deprecated = deprecated_of_attrs md.md_attributes in
1378+
let deprecated =
1379+
Builtin_attributes.deprecated_of_attrs md.md_attributes
1380+
in
13861381
let comps = components_of_module ~deprecated !env sub path mty in
13871382
c.comp_components <-
13881383
Tbl.add (Ident.name id) (comps, !pos) c.comp_components;
@@ -1546,7 +1541,7 @@ and store_extension ~check slot id path ext env renv =
15461541
summary = Env_extension(env.summary, id, ext) }
15471542

15481543
and store_module slot id path md env renv =
1549-
let deprecated = deprecated_of_attrs md.md_attributes in
1544+
let deprecated = Builtin_attributes.deprecated_of_attrs md.md_attributes in
15501545
{ env with
15511546
modules = EnvTbl.add slot (fun x -> `Module x) id (path, md)
15521547
env.modules renv.modules;

typing/env.mli

-2
Original file line numberDiff line numberDiff line change
@@ -245,8 +245,6 @@ val check_modtype_inclusion:
245245
val add_delayed_check_forward: ((unit -> unit) -> unit) ref
246246
(* Forward declaration to break mutual recursion with Mtype. *)
247247
val strengthen: (t -> module_type -> Path.t -> module_type) ref
248-
(* Forward declaration to break mutual recursion with Typetexp. *)
249-
val deprecated_of_attrs_forward: (Parsetree.attributes -> string option) ref
250248
(* Forward declaration to break mutual recursion with Ctype. *)
251249
val same_constr: (t -> type_expr -> type_expr -> bool) ref
252250

typing/typeclass.ml

+10-10
Original file line numberDiff line numberDiff line change
@@ -417,12 +417,12 @@ let rec class_type_field env self_type meths
417417
val_sig, concr_meths, inher)
418418

419419
| Pctf_attribute x ->
420-
Typetexp.warning_attribute [x];
420+
Builtin_attributes.warning_attribute [x];
421421
(mkctf (Tctf_attribute x) :: fields,
422422
val_sig, concr_meths, inher)
423423

424424
| Pctf_extension ext ->
425-
raise (Error_forward (Typetexp.error_of_extension ext))
425+
raise (Error_forward (Builtin_attributes.error_of_extension ext))
426426

427427
and class_signature env {pcsig_self=sty; pcsig_fields=sign} =
428428
let meths = ref Meths.empty in
@@ -443,13 +443,13 @@ and class_signature env {pcsig_self=sty; pcsig_fields=sign} =
443443
end;
444444

445445
(* Class type fields *)
446-
Typetexp.warning_enter_scope ();
446+
Builtin_attributes.warning_enter_scope ();
447447
let (rev_fields, val_sig, concr_meths, inher) =
448448
List.fold_left (class_type_field env self_type meths)
449449
([], Vars.empty, Concr.empty, [])
450450
sign
451451
in
452-
Typetexp.warning_leave_scope ();
452+
Builtin_attributes.warning_leave_scope ();
453453
let cty = {csig_self = self_type;
454454
csig_vars = val_sig;
455455
csig_concr = concr_meths;
@@ -512,7 +512,7 @@ and class_type env scty =
512512
let typ = Cty_arrow (l, ty, clty.cltyp_type) in
513513
cltyp (Tcty_arrow (l, cty, clty)) typ
514514
| Pcty_extension ext ->
515-
raise (Error_forward (Typetexp.error_of_extension ext))
515+
raise (Error_forward (Builtin_attributes.error_of_extension ext))
516516

517517
let class_type env scty =
518518
delayed_meth_specs := [];
@@ -722,12 +722,12 @@ let rec class_field self_loc cl_num self_type meths vars
722722
(val_env, met_env, par_env, field::fields, concr_meths, warn_vals,
723723
inher, local_meths, local_vals)
724724
| Pcf_attribute x ->
725-
Typetexp.warning_attribute [x];
725+
Builtin_attributes.warning_attribute [x];
726726
(val_env, met_env, par_env,
727727
lazy (mkcf (Tcf_attribute x)) :: fields,
728728
concr_meths, warn_vals, inher, local_meths, local_vals)
729729
| Pcf_extension ext ->
730-
raise (Error_forward (Typetexp.error_of_extension ext))
730+
raise (Error_forward (Builtin_attributes.error_of_extension ext))
731731

732732
and class_structure cl_num final val_env met_env loc
733733
{ pcstr_self = spat; pcstr_fields = str } =
@@ -776,14 +776,14 @@ and class_structure cl_num final val_env met_env loc
776776
end;
777777

778778
(* Typing of class fields *)
779-
Typetexp.warning_enter_scope ();
779+
Builtin_attributes.warning_enter_scope ();
780780
let (_, _, _, fields, concr_meths, _, inher, _local_meths, _local_vals) =
781781
List.fold_left (class_field self_loc cl_num self_type meths vars)
782782
(val_env, meth_env, par_env, [], Concr.empty, Concr.empty, [],
783783
Concr.empty, Concr.empty)
784784
str
785785
in
786-
Typetexp.warning_leave_scope ();
786+
Builtin_attributes.warning_leave_scope ();
787787
Ctype.unify val_env self_type (Ctype.newvar ());
788788
let sign =
789789
{csig_self = public_self;
@@ -1165,7 +1165,7 @@ and class_expr cl_num val_env met_env scl =
11651165
cl_attributes = scl.pcl_attributes;
11661166
}
11671167
| Pcl_extension ext ->
1168-
raise (Error_forward (Typetexp.error_of_extension ext))
1168+
raise (Error_forward (Builtin_attributes.error_of_extension ext))
11691169

11701170
(*******************************)
11711171

0 commit comments

Comments
 (0)