Skip to content

Commit 201ee17

Browse files
Octachrondamiendoligez
authored andcommitted
Extended indexing operators (ocaml#1064)
* Extended indexing operators This commit adds extended indexing operators to the parser. For instance, ``` let (.%()) p (x,y) = p.( x ).( y ) ;; p.%(0,0) ``` Extended indexing operators name starts with a leading dot "." followed by an operator symbol, except "." or "<" and a sequence of any operators and must be closes with a couple of enclosing parentheses ( i.e. "()", "[]", "{}" ) and then an optional assignment operator "<-": * '.' dotsymbolchar symbolchar* '(' ')' ['<-'] * '.' dotsymbolchar symbolchar* '[' ']' ['<-'] * '.' dotsymbolchar symbolchar* '{' '}' ['<-'] Similarly, expressions of the form * expr_1 '.' dotsymbolchar symbolchar* '(' expr_2 ')' * expr_1 '.' dotsymbolchar symbolchar* '[' expr_2 ']' * expr_1 '.' dotsymbolchar symbolchar* '{' expr_2 '}' and * expr_1 '.' dotsymbolchar symbolchar* '(' expr_2 ')' '<-' expr_3 * expr_1 '.' dotsymbolchar symbolchar* '[' expr_2 ']' '<-' expr_3 * expr_1 '.' dotsymbolchar symbolchar* '{' expr_2 '}' '<-' expr_3 are desugared to * ('.' dotsymbolchar symbolchar* '(' ')' ) expr_1 expr_2 * ('.' dotsymbolchar symbolchar* '[' ']' ) expr_1 expr_2 * ('.' dotsymbolchar symbolchar* '{' '}' ) expr_1 expr_2 and * ('.' dotsymbolchar symbolchar* '(' ')' '<-' ) expr_1 expr_2 expr_3 * ('.' dotsymbolchar symbolchar* '[' ']' ) expr_1 expr_2 expr_3 * ('.' dotsymbolchar symbolchar* '{' '}' '<-' ) expr_1 expr_2 expr_3
1 parent 2066800 commit 201ee17

File tree

11 files changed

+329
-20
lines changed

11 files changed

+329
-20
lines changed

Changes

+11-4
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,12 @@ Working version
44
(Changes that can break existing programs are marked with a "*")
55

66
### Language features:
7+
8+
- GPR#792: fix limitations of destructive substitutions, by
9+
allowing "S with type t := type-expr",
10+
"S with type M.t := type-expr", "S with module M.N := path"
11+
(Valentin Gatien-Baron, review by Jacques Garrigue and Leo White)
12+
713
- GPR#1118: Support inherited field in object type expression
814
(Runhang Li, reivew by Jeremy Yallop, Leo White, Jacques Garrigue,
915
and Florian Angeletti)
@@ -12,10 +18,11 @@ Working version
1218
in class expressions and class type expressions.
1319
(Alain Frisch, reviews by Thomas Refis and Jacques Garrigue)
1420

15-
- GPR#792: fix limitations of destructive substitutions, by
16-
allowing "S with type t := type-expr",
17-
"S with type M.t := type-expr", "S with module M.N := path"
18-
(Valentin Gatien-Baron, review by Jacques Garrigue and Leo White)
21+
- GPR#1064: extended indexing operators, add a new class of
22+
user-defined indexing operators, obtained by adding at least
23+
one operator character after the dot symbol to the standard indexing
24+
operators: e,g ".%()", ".?[]", ".@{}<-"
25+
(Florian Angeletti, review by Damien Doligez and Gabriel Radanne)
1926

2027
- GPR#1142: Mark assertions nonexpansive, so that 'assert false'
2128
can be used as a placeholder for a polymorphic function.

manual/manual/refman/expr.etex

+3-2
Original file line numberDiff line numberDiff line change
@@ -118,8 +118,9 @@ See also the following language extensions:
118118
\hyperref[s:explicit-overriding-open]{overriding in open statements},
119119
\hyperref[s:bigarray-access]{syntax for Bigarray access},
120120
\hyperref[s:attributes]{attributes},
121-
\hyperref[s:extension-nodes]{extension nodes} and
122-
\hyperref[s:local-exceptions]{local exceptions}.
121+
\hyperref[s:extension-nodes]{extension nodes},
122+
\hyperref[s:local-exceptions]{local exceptions}
123+
\hyperref[s:index-operators]{extended indexing operators}.
123124

124125
The table below shows the relative precedences and associativity of
125126
operators and non-closed constructions. The constructions with higher

manual/manual/refman/exten.etex

+42
Original file line numberDiff line numberDiff line change
@@ -2351,3 +2351,45 @@ will be converted directly to
23512351
type t = T of string
23522352
[@@ocaml.doc " Attaches to t "]
23532353
\end{verbatim}
2354+
2355+
\section{Extendend indexing operators \label{s:index-operators} }
2356+
(Introduced in 4.06)
2357+
2358+
\begin{syntax}
2359+
2360+
dot-ext:
2361+
| ('!'||'$'||'%'||'&'||'*'||'+'||'-'||'/'||':'||'='||'>'||'?'||'@'||'^'||'|'||'~') { operator-char }
2362+
;
2363+
expr:
2364+
...
2365+
| expr '.' dot-ext ( '(' expr ')' || '[' expr ']' || '{' expr '}' ) [ '<-' expr ]
2366+
;
2367+
operator-name:
2368+
...
2369+
| '.' dot-ext ('()' || '[]' || '{}') ['<-']
2370+
;
2371+
\end{syntax}
2372+
2373+
2374+
This extension provides syntactic sugar for getting and setting elements
2375+
for user-defined indexed types. For instance, we can define python-like
2376+
dictionaries with
2377+
\begin{caml-example}
2378+
module Dict = struct
2379+
2380+
include Hashtbl
2381+
let ( .%{} ) tabl index = find tabl index
2382+
let ( .%{}<- ) tabl index value = add tabl index value
2383+
end;;
2384+
let dict =
2385+
let open Dict in
2386+
let dict = create 10 in
2387+
let () =
2388+
dict.%{"one"} <- 1;
2389+
dict.%{"two"} <- 2 in
2390+
dict
2391+
;;
2392+
let () =
2393+
let open Dict in
2394+
assert( dict.%{"one"} = 1 );;
2395+
\end{caml-example}

manual/manual/refman/lex.etex

+3-2
Original file line numberDiff line numberDiff line change
@@ -218,8 +218,9 @@ operator-char:
218218
'/' || ':' || '<' || '=' || '>' || '?' || '@' ||
219219
'^' || '|' || '~'
220220
\end{syntax}
221-
See also the following language extension:
222-
\hyperref[s:ext-ops]{extension operators}.
221+
See also the following language extensions:
222+
\hyperref[s:ext-ops]{extension operators} and
223+
\hyperref[s:index-operators]{extended indexing operators}.
223224

224225
Sequences of ``operator characters'', such as "<=>" or "!!",
225226
are read as a single token from the @infix-symbol@ or @prefix-symbol@

manual/manual/refman/names.etex

+3
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,9 @@ inst-var-name:
7373
method-name:
7474
lowercase-ident
7575
\end{syntax}
76+
See also the following language extension:
77+
\hyperref[s:index-operators]{extended indexing operators}.
78+
7679
As shown above, prefix and infix symbols as well as some keywords can
7780
be used as value names, provided they are written between parentheses.
7881
The capitalization rules are summarized in the table below.

parsing/lexer.mll

+3
Original file line numberDiff line numberDiff line change
@@ -292,6 +292,8 @@ let identchar_latin1 =
292292
['A'-'Z' 'a'-'z' '_' '\192'-'\214' '\216'-'\246' '\248'-'\255' '\'' '0'-'9']
293293
let symbolchar =
294294
['!' '$' '%' '&' '*' '+' '-' '.' '/' ':' '<' '=' '>' '?' '@' '^' '|' '~']
295+
let dotsymbolchar =
296+
['!' '$' '%' '&' '*' '+' '-' '/' ':' '=' '>' '?' '@' '^' '|' '~']
295297
let decimal_literal =
296298
['0'-'9'] ['0'-'9' '_']*
297299
let hex_digit =
@@ -467,6 +469,7 @@ rule token = parse
467469
| "->" { MINUSGREATER }
468470
| "." { DOT }
469471
| ".." { DOTDOT }
472+
| "." (dotsymbolchar symbolchar* as s) { DOTOP s }
470473
| ":" { COLON }
471474
| "::" { COLONCOLON }
472475
| ":=" { COLONEQUAL }

parsing/parser.mly

+32-1
Original file line numberDiff line numberDiff line change
@@ -465,6 +465,7 @@ let package_type_of_module_type pmty =
465465
%token <string> INFIXOP2
466466
%token <string> INFIXOP3
467467
%token <string> INFIXOP4
468+
%token <string> DOTOP
468469
%token INHERIT
469470
%token INITIALIZER
470471
%token <string * char option> INT
@@ -1411,6 +1412,30 @@ expr:
14111412
[Nolabel,$1; Nolabel,$4; Nolabel,$7])) }
14121413
| simple_expr DOT LBRACE expr RBRACE LESSMINUS expr
14131414
{ bigarray_set $1 $4 $7 }
1415+
| simple_expr DOTOP LBRACKET expr RBRACKET LESSMINUS expr
1416+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "[]<-")) in
1417+
mkexp @@ Pexp_apply(id , [Nolabel, $1; Nolabel, $4; Nolabel, $7]) }
1418+
| simple_expr DOTOP LBRACKET expr RBRACKET
1419+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "[]")) in
1420+
mkexp @@ Pexp_apply(id, [Nolabel, $1; Nolabel, $4]) }
1421+
| simple_expr DOTOP LBRACKET expr error
1422+
{ unclosed "[" 3 "]" 5 }
1423+
| simple_expr DOTOP LPAREN expr RPAREN LESSMINUS expr
1424+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "()<-")) in
1425+
mkexp @@ Pexp_apply(id , [Nolabel, $1; Nolabel, $4; Nolabel, $7]) }
1426+
| simple_expr DOTOP LPAREN expr RPAREN
1427+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "()")) in
1428+
mkexp @@ Pexp_apply(id, [Nolabel, $1; Nolabel, $4]) }
1429+
| simple_expr DOTOP LPAREN expr error
1430+
{ unclosed "(" 3 ")" 5 }
1431+
| simple_expr DOTOP LBRACE expr RBRACE LESSMINUS expr
1432+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "{}<-")) in
1433+
mkexp @@ Pexp_apply(id , [Nolabel, $1; Nolabel, $4; Nolabel, $7]) }
1434+
| simple_expr DOTOP LBRACE expr RBRACE
1435+
{ let id = mkexp @@ Pexp_ident( ghloc @@ Lident ("." ^ $2 ^ "{}")) in
1436+
mkexp @@ Pexp_apply(id, [Nolabel, $1; Nolabel, $4]) }
1437+
| simple_expr DOTOP LBRACE expr error
1438+
{ unclosed "{" 3 "}" 5 }
14141439
| label LESSMINUS expr
14151440
{ mkexp(Pexp_setinstvar(mkrhs $1 1, $3)) }
14161441
| ASSERT ext_attributes simple_expr %prec below_HASH
@@ -2364,7 +2389,13 @@ operator:
23642389
| INFIXOP2 { $1 }
23652390
| INFIXOP3 { $1 }
23662391
| INFIXOP4 { $1 }
2367-
| HASHOP { $1 }
2392+
| DOTOP LPAREN RPAREN { "."^ $1 ^"()" }
2393+
| DOTOP LPAREN RPAREN LESSMINUS { "."^ $1 ^ "()<-" }
2394+
| DOTOP LBRACKET RBRACKET { "."^ $1 ^"[]" }
2395+
| DOTOP LBRACKET RBRACKET LESSMINUS { "."^ $1 ^ "[]<-" }
2396+
| DOTOP LBRACE RBRACE { "."^ $1 ^"{}" }
2397+
| DOTOP LBRACE RBRACE LESSMINUS { "."^ $1 ^ "{}<-" }
2398+
| HASHOP { $1 }
23682399
| BANG { "!" }
23692400
| PLUS { "+" }
23702401
| PLUSDOT { "+." }

parsing/pprintast.ml

+40-11
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@ open Ast_helper
3131
let prefix_symbols = [ '!'; '?'; '~' ] ;;
3232
let infix_symbols = [ '='; '<'; '>'; '@'; '^'; '|'; '&'; '+'; '-'; '*'; '/';
3333
'$'; '%'; '#' ]
34+
3435
(* type fixity = Infix| Prefix *)
3536
let special_infix_strings =
3637
["asr"; "land"; "lor"; "lsl"; "lsr"; "lxor"; "mod"; "or"; ":="; "!="; "::" ]
@@ -44,6 +45,7 @@ let fixity_of_string = function
4445
| s when List.mem s special_infix_strings -> `Infix s
4546
| s when List.mem s.[0] infix_symbols -> `Infix s
4647
| s when List.mem s.[0] prefix_symbols -> `Prefix s
48+
| s when s.[0] = '.' -> `Mixfix s
4749
| _ -> `Normal
4850

4951
let view_fixity_of_exp = function
@@ -52,10 +54,13 @@ let view_fixity_of_exp = function
5254
| _ -> `Normal
5355

5456
let is_infix = function | `Infix _ -> true | _ -> false
57+
let is_mixfix = function `Mixfix _ -> true | _ -> false
5558

5659
(* which identifiers are in fact operators needing parentheses *)
5760
let needs_parens txt =
58-
is_infix (fixity_of_string txt)
61+
let fix = fixity_of_string txt in
62+
is_infix fix
63+
|| is_mixfix fix
5964
|| List.mem txt.[0] prefix_symbols
6065

6166
(* some infixes need spaces around parens to avoid clashes with comment
@@ -467,23 +472,24 @@ and sugar_expr ctxt f e =
467472
| Pexp_apply ({ pexp_desc = Pexp_ident {txt = id; _};
468473
pexp_attributes=[]; _}, args)
469474
when List.for_all (fun (lab, _) -> lab = Nolabel) args -> begin
470-
match id, List.map snd args with
471-
| Lident "!", [e] ->
472-
pp f "@[<hov>!%a@]" (simple_expr ctxt) e; true
473-
| Ldot (path, ("get"|"set" as func)), a :: other_args -> begin
474-
let print left right print_index indexes rem_args =
475-
match func, rem_args with
476-
| "get", [] ->
475+
let print a assign left right print_index indexes rem_args =
476+
match assign, rem_args with
477+
| true, [] ->
477478
pp f "@[%a.%s%a%s@]"
478479
(simple_expr ctxt) a
479480
left (list ~sep:"," print_index) indexes right; true
480-
| "set", [v] ->
481+
| false, [v] ->
481482
pp f "@[%a.%s%a%s@ <-@;<1 2>%a@]"
482483
(simple_expr ctxt) a
483484
left (list ~sep:"," print_index) indexes right
484485
(simple_expr ctxt) v; true
485-
| _ -> false
486-
in
486+
| _ -> false in
487+
match id, List.map snd args with
488+
| Lident "!", [e] ->
489+
pp f "@[<hov>!%a@]" (simple_expr ctxt) e; true
490+
| Ldot (path, ("get"|"set" as func)), a :: other_args -> begin
491+
let assign = func = "get" in
492+
let print = print a assign in
487493
match path, other_args with
488494
| Lident "Array", i :: rest ->
489495
print "(" ")" (expression ctxt) [i] rest
@@ -500,6 +506,29 @@ and sugar_expr ctxt f e =
500506
print "{" "}" (simple_expr ctxt) indexes rest
501507
| _ -> false
502508
end
509+
| Lident s, a :: other_args when s.[0] = '.' ->
510+
begin match other_args with
511+
| i :: rest ->
512+
let n = String.length s in
513+
(* extract operator:
514+
assignment operators end with [right_bracket ^ "<-"],
515+
access operators end with [right_bracket] directly
516+
*)
517+
let assign =
518+
s.[n - 1] = '-' in
519+
let kind =
520+
(* extract the right end bracket *)
521+
if assign then s.[n - 3] else s.[n - 1] in
522+
let left, right = match kind with
523+
| ')' -> '(', ")"
524+
| ']' -> '[', "]"
525+
| '}' -> '{', "}"
526+
| _ -> assert false in
527+
let prefix = String.sub s 0 (1+String.index s left) in
528+
print a assign prefix right
529+
(simple_expr ctxt) [i] rest
530+
| _ -> false
531+
end
503532
| _ -> false
504533
end
505534
| _ -> false

testsuite/tests/parsetree/source.ml

+14
Original file line numberDiff line numberDiff line change
@@ -7322,3 +7322,17 @@ module Exotic_list = struct
73227322

73237323
let Inner.(::)(x,y, Inner.[]) = Inner.(::)(1,"one",Inner.[])
73247324
end
7325+
7326+
(** Extended index operators *)
7327+
let ( .%[] ) = Hashtbl.find
7328+
let ( .%[] <- ) = Hashtbl.add
7329+
let ( .%() ) = Hashtbl.find
7330+
let ( .%() <- ) = Hashtbl.add
7331+
let ( .%{} ) = Hashtbl.find
7332+
let ( .%{} <- ) = Hashtbl.add
7333+
7334+
;;
7335+
let h = Hashtbl.create 17 in
7336+
h.%["one"] <- 1;
7337+
h.%("two") <- 2;
7338+
h.%{"three"} <- 3;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
let (.?[]) = Hashtbl.find_opt
2+
let (.@[]) = Hashtbl.find
3+
let ( .@[]<- ) = Hashtbl.add
4+
let (.@{}) = Hashtbl.find
5+
let ( .@{}<- ) = Hashtbl.add
6+
let (.@()) = Hashtbl.find
7+
let ( .@()<- ) = Hashtbl.add
8+
9+
let h = Hashtbl.create 17
10+
11+
;;
12+
h.@("One") <- 1
13+
; assert (h.@{"One"} = 1)
14+
; assert (h.?["Two"] = None)

0 commit comments

Comments
 (0)