|
| 1 | + |
| 2 | +(* This file is free software, part of dolmen. See file "LICENSE" for more information *) |
| 3 | + |
| 4 | +(* Printing of identifiers *) |
| 5 | +(* ************************************************************************* *) |
| 6 | + |
| 7 | +module L = Lexer |
| 8 | + |
| 9 | +exception Cannot_print of string |
| 10 | + |
| 11 | +let _cannot_print format = |
| 12 | + Format.kasprintf (fun msg -> raise (Cannot_print msg)) format |
| 13 | + |
| 14 | +(* lexical definitions taken from the smtlib specification *) |
| 15 | + |
| 16 | +let[@inline] is_whitespace c = |
| 17 | + let c = Char.code c in |
| 18 | + c = 9 (* tab *) || c = 10 (* line feed *) || |
| 19 | + c = 13 (* cariage return *) || c = 32 (* space *) |
| 20 | + |
| 21 | +let[@inline] is_printable c = |
| 22 | + let c = Char.code c in |
| 23 | + (32 <= c && c <= 126) || 128 <= c |
| 24 | + |
| 25 | +let is_quoted_symbol_char c = |
| 26 | + (is_whitespace c || is_printable c) && |
| 27 | + (c <> '|' && c <> '\\') |
| 28 | + |
| 29 | +let[@inline] is_letter = function |
| 30 | + | 'a'..'z' | 'A'..'Z' -> true |
| 31 | + | _ -> false |
| 32 | + |
| 33 | +let[@inline] is_digit = function |
| 34 | + | '0'..'9' -> true |
| 35 | + | _ -> false |
| 36 | + |
| 37 | +let[@inline] is_other_simple_symbol_chars = function |
| 38 | + | '~' | '!' | '@' | '$' | '%' | '^' | '&' | '*' | '_' |
| 39 | + | '-' | '+' | '=' | '<' | '>' | '.' | '?' | '/' -> true |
| 40 | + | _ -> false |
| 41 | + |
| 42 | +let is_simple_symbol_char c = |
| 43 | + is_letter c || is_digit c || is_other_simple_symbol_chars c |
| 44 | + |
| 45 | +(* symbol categorization *) |
| 46 | + |
| 47 | +type symbol = |
| 48 | + | Simple |
| 49 | + | Quoted |
| 50 | + | Unprintable |
| 51 | + |
| 52 | +let categorize_symbol s = |
| 53 | + match s with |
| 54 | + | "" -> Unprintable |
| 55 | + | "_" | "!" | "as" | "let" |
| 56 | + | "exists" | "forall" |
| 57 | + | "match" | "par" |
| 58 | + | "assert" |
| 59 | + | "check-sat" |
| 60 | + | "check-sat-assuming" |
| 61 | + | "declare-const" |
| 62 | + | "declare-datatype" |
| 63 | + | "declare-datatypes" |
| 64 | + | "declare-fun" |
| 65 | + | "declare-sort" |
| 66 | + | "define-fun" |
| 67 | + | "define-fun-rec" |
| 68 | + | "define-funs-rec" |
| 69 | + | "define-sort" |
| 70 | + | "echo" |
| 71 | + | "exit" |
| 72 | + | "get-assertions" |
| 73 | + | "get-assignment" |
| 74 | + | "get-info" |
| 75 | + | "get-model" |
| 76 | + | "get-option" |
| 77 | + | "get-proof" |
| 78 | + | "get-unsat-assumptions" |
| 79 | + | "get-unsat-core" |
| 80 | + | "get-value" |
| 81 | + | "pop" |
| 82 | + | "push" |
| 83 | + | "reset" |
| 84 | + | "reset-assertions" |
| 85 | + | "set-info" |
| 86 | + | "set-logic" |
| 87 | + | "set-option" -> Quoted |
| 88 | + | _ -> |
| 89 | + (* we are guaranteed that `s` is not the empty string *) |
| 90 | + if not (is_digit s.[0]) && (String.for_all is_simple_symbol_char s) then |
| 91 | + Simple |
| 92 | + else if String.for_all is_quoted_symbol_char s then |
| 93 | + Quoted |
| 94 | + else |
| 95 | + Unprintable |
| 96 | + |
| 97 | +let id fmt name = |
| 98 | + let aux fmt s = |
| 99 | + (* TODO: expose/add a cache to not redo the `categorize_symbol` computation each time *) |
| 100 | + match categorize_symbol s with |
| 101 | + | Simple -> Format.pp_print_string fmt s |
| 102 | + | Quoted -> Format.fprintf fmt "|%s|" s |
| 103 | + | Unprintable -> |
| 104 | + _cannot_print "symbol \"%s\" cannot be printed due to lexical constraints" s |
| 105 | + in |
| 106 | + match (name : Dolmen_std.Name.t) with |
| 107 | + | Simple s -> aux fmt s |
| 108 | + | Indexed { basename = _; indexes = [] } -> |
| 109 | + _cannot_print "indexed id with no indexes: %a" Dolmen_std.Name.print name |
| 110 | + | Indexed { basename; indexes; } -> |
| 111 | + let pp_sep fmt () = Format.fprintf fmt " " in |
| 112 | + Format.fprintf fmt "(_ %a %a)" |
| 113 | + aux basename (Format.pp_print_list ~pp_sep aux) indexes |
| 114 | + | Qualified _ -> |
| 115 | + _cannot_print "qualified identifier: %a" Dolmen_std.Name.print name |
| 116 | + |
0 commit comments