|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | #include <string.h> | 
|  | #include <stdlib.h> | 
|  | #include "util/string2.h" | 
|  |  | 
|  | #include "demangle-ocaml.h" | 
|  |  | 
|  | #include <linux/ctype.h> | 
|  |  | 
|  | static const char *caml_prefix = "caml"; | 
|  | static const size_t caml_prefix_len = 4; | 
|  |  | 
|  | /* mangled OCaml symbols start with "caml" followed by an upper-case letter */ | 
|  | static bool | 
|  | ocaml_is_mangled(const char *sym) | 
|  | { | 
|  | return 0 == strncmp(sym, caml_prefix, caml_prefix_len) | 
|  | && isupper(sym[caml_prefix_len]); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * input: | 
|  | *     sym: a symbol which may have been mangled by the OCaml compiler | 
|  | * return: | 
|  | *     if the input doesn't look like a mangled OCaml symbol, NULL is returned | 
|  | *     otherwise, a newly allocated string containing the demangled symbol is returned | 
|  | */ | 
|  | char * | 
|  | ocaml_demangle_sym(const char *sym) | 
|  | { | 
|  | char *result; | 
|  | int j = 0; | 
|  | int i; | 
|  | int len; | 
|  |  | 
|  | if (!ocaml_is_mangled(sym)) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | len = strlen(sym); | 
|  |  | 
|  | /* the demangled symbol is always smaller than the mangled symbol */ | 
|  | result = malloc(len + 1); | 
|  | if (!result) | 
|  | return NULL; | 
|  |  | 
|  | /* skip "caml" prefix */ | 
|  | i = caml_prefix_len; | 
|  |  | 
|  | while (i < len) { | 
|  | if (sym[i] == '_' && sym[i + 1] == '_') { | 
|  | /* "__" -> "." */ | 
|  | result[j++] = '.'; | 
|  | i += 2; | 
|  | } | 
|  | else if (sym[i] == '$' && isxdigit(sym[i + 1]) && isxdigit(sym[i + 2])) { | 
|  | /* "$xx" is a hex-encoded character */ | 
|  | result[j++] = (hex(sym[i + 1]) << 4) | hex(sym[i + 2]); | 
|  | i += 3; | 
|  | } | 
|  | else { | 
|  | result[j++] = sym[i++]; | 
|  | } | 
|  | } | 
|  | result[j] = '\0'; | 
|  |  | 
|  | return result; | 
|  | } |