summaryrefslogtreecommitdiff
path: root/mnv/src/mnv9generics.c
diff options
context:
space:
mode:
Diffstat (limited to 'mnv/src/mnv9generics.c')
-rw-r--r--mnv/src/mnv9generics.c1312
1 files changed, 1312 insertions, 0 deletions
diff --git a/mnv/src/mnv9generics.c b/mnv/src/mnv9generics.c
new file mode 100644
index 0000000000..adc56aca16
--- /dev/null
+++ b/mnv/src/mnv9generics.c
@@ -0,0 +1,1312 @@
+/* vi:set ts=8 sts=4 sw=4 noet:
+ *
+ * MNV - MNV is not Vim by Bram Moolenaar
+ *
+ * Do ":help uganda" in MNV to read copying and usage conditions.
+ * Do ":help credits" in MNV to see a list of people who contributed.
+ * See README.txt for an overview of the MNV source code.
+ */
+
+/*
+ * mnv9generics.c: MNV9 script generics support
+ */
+
+#include "mnv.h"
+
+#if defined(FEAT_EVAL)
+
+
+/*
+ * A hash table is used to lookup a generic function with specific types.
+ * The specific type names are used as the key.
+ */
+typedef struct gfitem_S gfitem_T;
+struct gfitem_S
+{
+ ufunc_T *gfi_ufunc;
+ char_u gfi_name[1]; // actually longer
+};
+#define GFITEM_KEY_OFF offsetof(gfitem_T, gfi_name)
+#define HI2GFITEM(hi) ((gfitem_T *)((hi)->hi_key - GFITEM_KEY_OFF))
+
+static type_T *find_generic_type_in_cctx(char_u *gt_name, size_t len, cctx_T *cctx);
+
+/*
+ * Returns a pointer to the first '<' character in "name" that starts the
+ * generic type argument list, skipping an initial <SNR> or <lambda> prefix if
+ * present. The prefix is only skipped if "name" starts with '<'.
+ *
+ * Returns NULL if no '<' is found before a '(' or the end of the string.
+ * The returned pointer refers to the original string.
+ *
+ * Examples:
+ * "<SNR>123_Fn<number>" -> returns pointer to '<'
+ * "<lambda>123_Fn<number>" -> returns pointer to '<'
+ * "Func<number>" -> returns pointer to '<'
+ * "Func()" -> returns NULL
+ */
+ char_u *
+generic_func_find_open_bracket(char_u *name)
+{
+ char_u *p = name;
+
+ if (name[0] == '<')
+ {
+ // Skip the <SNR> or <lambda> at the start of the name
+ if (STRNCMP(name + 1, "SNR>", 4) == 0)
+ p += 5;
+ else if (STRNCMP(name + 1, "lambda>", 7) == 0)
+ p += 8;
+ }
+
+ while (*p && *p != '(' && *p != '<')
+ p++;
+
+ if (*p == '<')
+ return p;
+
+ return NULL;
+}
+
+/*
+ * Finds the matching '>' character for a generic function type parameter or
+ * argument list, starting from the opening '<'.
+ *
+ * Enforces correct syntax for a flat, comma-separated list of types:
+ * - No whitespace before or after type names or commas
+ * - Each type must be non-empty and separated by a comma and whitespace
+ * - At least one type must be present
+ *
+ * Arguments:
+ * start - pointer to the opening '<'
+ *
+ * Returns:
+ * Pointer to the matching '>' character if found and syntax is valid,
+ * or NULL if not found, invalid syntax, or on error.
+ */
+ static char_u *
+generic_func_find_close_bracket(char_u *start)
+{
+ char_u *p = start + 1;
+ int type_count = 0;
+
+ while (*p && *p != '>')
+ {
+ char_u *typename = p;
+
+ if (MNV_ISWHITE(*p))
+ {
+ char tmpstr[2];
+ tmpstr[0] = *(p - 1); tmpstr[1] = NUL;
+ semsg(_(e_no_white_space_allowed_after_str_str), tmpstr, start);
+ return NULL;
+ }
+
+ p = skip_type(p, FALSE);
+ if (p == typename)
+ {
+ char_u cc = *p;
+ *p = NUL;
+ semsg(_(e_missing_type_after_str), start);
+ *p = cc;
+ return NULL;
+ }
+ type_count++;
+
+ if (*p == '>' || *p == NUL)
+ break;
+
+ if (MNV_ISWHITE(*p))
+ {
+ char_u cc = *p;
+ *p = NUL;
+ semsg(_(e_no_white_space_allowed_after_str_str), typename, start);
+ *p = cc;
+ return NULL;
+ }
+
+ if (*p != ',')
+ {
+ semsg(_(e_missing_comma_in_generic_function_str), start);
+ return NULL;
+ }
+ p++;
+
+ if (*p == NUL)
+ break;
+
+ if (!MNV_ISWHITE(*p))
+ {
+ semsg(_(e_white_space_required_after_str_str), ",", start);
+ return NULL;
+ }
+ p = skipwhite(p);
+ }
+
+ if (*p != '>')
+ {
+ semsg(_(e_missing_closing_angle_bracket_in_generic_function_str), start);
+ return NULL;
+ }
+
+ if (MNV_ISWHITE(*(p + 1)) && *skipwhite(p + 1) == '(')
+ {
+ // white space not allowed between '>' and '('
+ semsg(_(e_no_white_space_allowed_after_str_str), ">", start);
+ return NULL;
+ }
+
+
+ if (type_count == 0)
+ {
+ semsg(_(e_empty_type_list_for_generic_function_str), start);
+ return NULL;
+ }
+
+ return p;
+}
+
+/*
+ * Advances the argument pointer past a generic function's type argument list.
+ *
+ * On entry, "*argp" must point to the opening '<' of a generic type argument
+ * list. This function finds the matching closing '>' (validating the syntax
+ * via generic_func_find_close_bracket), and if successful, advances "*argp" to
+ * the character immediately after the closing '>'.
+ *
+ * Returns OK on success, or FAIL if the type argument list is invalid or no
+ * matching '>' is found. On failure, "*argp" is not modified.
+ */
+ int
+skip_generic_func_type_args(char_u **argp)
+{
+ char_u *p = generic_func_find_close_bracket(*argp);
+ if (p == NULL)
+ return FAIL;
+
+ *argp = p + 1; // skip '>'
+
+ return OK;
+}
+
+/*
+ * Appends the generic function type arguments, starting at "*argp", to the
+ * function name "funcname" (of length "namelen") and returns a newly allocated
+ * string containing the result.
+ *
+ * On entry, "*argp" must point to the opening '<' of the generic type argument
+ * list. If the type argument list is valid, the substring from "*argp" up to
+ * and including the matching '>' is appended to "funcname". On success,
+ * "*argp" is updated to point to the character after the closing '>'.
+ *
+ * Returns:
+ * A newly allocated string with the combined function name and type
+ * arguments, or NULL if there is a syntax error in the generic type
+ * arguments.
+ *
+ * The caller is responsible for freeing the returned string.
+ */
+ char_u *
+append_generic_func_type_args(
+ char_u *funcname,
+ size_t namelen,
+ char_u **argp)
+{
+ char_u *p = generic_func_find_close_bracket(*argp);
+
+ if (p == NULL)
+ return NULL;
+
+ mnv_strncpy(IObuff, funcname, namelen);
+ STRNCAT(IObuff, *argp, p - *argp + 1);
+
+ *argp = p + 1;
+
+ return mnv_strsave(IObuff);
+}
+
+/*
+ * Returns a newly allocated string containing the function name from "fp" with
+ * the generic type arguments from "*argp" appended.
+ *
+ * On entry, "*argp" must point to the opening '<' of the generic type argument
+ * list. On success, "*argp" is advanced to the character after the closing
+ * '>'.
+ *
+ * Returns:
+ * A newly allocated string with the combined function name and type
+ * arguments, or NULL if "fp" is not a generic function, if there is a
+ * parsing error, or on memory allocation failure.
+ *
+ * The caller is responsible for freeing the returned string.
+ */
+ char_u *
+get_generic_func_name(ufunc_T *fp, char_u **argp)
+{
+ if (!IS_GENERIC_FUNC(fp))
+ {
+ emsg_funcname(e_not_a_generic_function_str, fp->uf_name);
+ return NULL;
+ }
+
+ return append_generic_func_type_args(fp->uf_name, fp->uf_namelen, argp);
+}
+
+/*
+ * Parses the concrete type arguments provided in a generic function call,
+ * starting at the opening '<' character and ending at the matching '>'.
+ *
+ * On entry, "start" must point to the opening '<' character.
+ * On success, returns a pointer to the character after the closing '>'.
+ * On failure, returns NULL and reports an error message.
+ *
+ * Arguments:
+ * func_name - the name of the function being called (used for error
+ * messages)
+ * namelen - length of the function name
+ * start - pointer to the opening '<' character in the call
+ * gfatab - args table to allocate new type objects and to store parsed
+ * type argument names and their types.
+ * cctx - compile context for type resolution (may be NULL)
+ *
+ * This function enforces correct syntax for generic type argument lists,
+ * including whitespace rules, comma separation, and non-empty argument lists.
+ */
+ char_u *
+parse_generic_func_type_args(
+ char_u *func_name,
+ size_t namelen,
+ char_u *start,
+ gfargs_tab_T *gfatab,
+ cctx_T *cctx)
+{
+ generic_T *generic_arg;
+ type_T *type_arg;
+ char_u *p = start;
+
+ // White spaces not allowed after '<'
+ if (MNV_ISWHITE(*(p + 1)))
+ {
+ semsg(_(e_no_white_space_allowed_after_str_str), "<", p);
+ return NULL;
+ }
+
+ ++p; // skip the '<'
+
+ // parse each type argument until '>' or end of string
+ while (*p && *p != '>')
+ {
+ p = skipwhite(p);
+
+ if (!ASCII_ISALNUM(*p))
+ {
+ semsg(_(e_missing_type_after_str), start);
+ return NULL;
+ }
+
+ // parse the type
+ type_arg = parse_type(&p, &gfatab->gfat_arg_types, NULL, cctx, TRUE);
+ if (type_arg == NULL || !valid_declaration_type(type_arg))
+ return NULL;
+
+ char *ret_free = NULL;
+ char *ret_name = type_name(type_arg, &ret_free);
+
+ // create space for the name and the new type
+ if (ga_grow(&gfatab->gfat_args, 1) == FAIL)
+ {
+ mnv_free(ret_free);
+ return NULL;
+ }
+ generic_arg = (generic_T *)gfatab->gfat_args.ga_data +
+ gfatab->gfat_args.ga_len;
+ gfatab->gfat_args.ga_len++;
+
+ // copy the type name
+ generic_arg->gt_name = alloc(STRLEN(ret_name) + 1);
+ if (generic_arg->gt_name == NULL)
+ return NULL;
+ STRCPY(generic_arg->gt_name, ret_name);
+ mnv_free(ret_free);
+
+ // add the new type
+ generic_arg->gt_type = type_arg;
+
+ p = skipwhite(p);
+
+ if (*p == NUL || *p == '>')
+ break;
+
+ // after a type, expect ',' or '>'
+ if (*p != ',')
+ {
+ semsg(_(e_missing_comma_in_generic_function_str), start);
+ return NULL;
+ }
+
+ if (*(p + 1) == NUL)
+ break;
+
+ // Require whitespace after a comma and skip it
+ if (!MNV_ISWHITE(*(p + 1)))
+ {
+ semsg(_(e_white_space_required_after_str_str), ",", p);
+ return NULL;
+ }
+ p++;
+ }
+
+ // ensure the list of types ends in a closing '>'
+ if (*p != '>')
+ {
+ semsg(_(e_missing_closing_angle_bracket_in_generic_function_str),
+ func_name);
+ return NULL;
+ }
+
+ // no whitespace allowed before '>'
+ if (MNV_ISWHITE(*(p - 1)))
+ {
+ semsg(_(e_no_white_space_allowed_before_str_str), ">", p);
+ return NULL;
+ }
+
+ // at least one type argument is required
+ if (generic_func_args_table_size(gfatab) == 0)
+ {
+ char_u cc = func_name[namelen];
+ func_name[namelen] = NUL;
+ semsg(_(e_empty_type_list_for_generic_function_str), func_name);
+ func_name[namelen] = cc;
+ return NULL;
+ }
+ ++p; // skip the '>'
+
+ return p;
+}
+
+/*
+ * Checks if a generic type name already exists in the current context.
+ *
+ * This function verifies that the given generic type name "name" does not
+ * conflict with an imported variable, an existing generic type in the provided
+ * growarray "gt_gap", or a generic type in the current or outer compile
+ * context "cctx". If a conflict is found, an appropriate error message is
+ * reported.
+ *
+ * Arguments:
+ * name - the generic type name to check
+ * gfatab - args table to allocate new type objects and to store parsed
+ * type argument names and their types.
+ * cctx - current compile context, used to check for outer generic types
+ * (may be NULL)
+ *
+ * Returns:
+ * TRUE if the name already exists or conflicts, FALSE otherwise.
+ */
+ static int
+generic_name_exists(
+ char_u *gt_name,
+ size_t name_len,
+ gfargs_tab_T *gfatab,
+ cctx_T *cctx)
+{
+ typval_T tv;
+
+ tv.v_type = VAR_UNKNOWN;
+
+ if (eval_variable_import(gt_name, &tv) == OK)
+ {
+ semsg(_(e_redefining_script_item_str), gt_name);
+ clear_tv(&tv);
+ return TRUE;
+ }
+
+ for (int i = 0; i < gfatab->gfat_args.ga_len; i++)
+ {
+ generic_T *generic = &((generic_T *)gfatab->gfat_args.ga_data)[i];
+
+ if (STRNCMP(gt_name, generic->gt_name, name_len) == 0)
+ {
+ semsg(_(e_duplicate_type_var_name_str), gt_name);
+ return TRUE;
+ }
+ }
+
+ if (cctx != NULL &&
+ find_generic_type_in_cctx(gt_name, name_len, cctx) != NULL)
+ {
+ semsg(_(e_duplicate_type_var_name_str), gt_name);
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+/*
+ * Parses the type parameters specified when defining a new generic function,
+ * starting at the opening '<' character and ending at the matching '>'.
+ *
+ * On entry, "p" must point to the opening '<' character.
+ * On success, returns a pointer to the character after the closing '>'.
+ * On failure, returns NULL and reports an error message.
+ *
+ * Arguments:
+ * func_name - the name of the function being defined (for error messages)
+ * p - pointer to the opening '<' character in the definition
+ * gfatab - args table to allocate new type objects and to store parsed
+ * type argument names and their types.
+ * cctx - current compile context, used to check for duplicate names in
+ * outer scopes (may be NULL)
+ *
+ * This function enforces correct syntax for generic type parameter lists:
+ * - No whitespace before or after the opening '<'
+ * - Parameters must be separated by a comma and whitespace
+ * - No whitespace after a parameter name
+ * - The list must not be empty
+ */
+ char_u *
+parse_generic_func_type_params(
+ char_u *func_name,
+ char_u *p,
+ gfargs_tab_T *gfatab,
+ cctx_T *cctx)
+{
+ // No white space allowed before the '<'
+ if (MNV_ISWHITE(*(p - 1)))
+ {
+ semsg(_(e_no_white_space_allowed_before_str_str), "<", p);
+ return NULL;
+ }
+
+ if (MNV_ISWHITE(*(p + 1)))
+ {
+ semsg(_(e_no_white_space_allowed_after_str_str), "<", p);
+ return NULL;
+ }
+
+ char_u *start = ++p;
+
+ while (*p && *p != '>')
+ {
+ p = skipwhite(p);
+
+ if (*p == NUL || *p == '>')
+ {
+ semsg(_(e_missing_type_after_str), p - 1);
+ return NULL;
+ }
+
+ if (!ASCII_ISUPPER(*p))
+ {
+ if (ASCII_ISLOWER(*p))
+ semsg(_(e_type_var_name_must_start_with_uppercase_letter_str), p);
+ else
+ semsg(_(e_missing_type_after_str), p - 1);
+ return NULL;
+ }
+
+ char_u *name_start = p;
+ char_u *name_end = NULL;
+ char_u cc;
+ size_t name_len = 0;
+
+ p++;
+ while (ASCII_ISALNUM(*p) || *p == '_')
+ p++;
+ name_end = p;
+
+ name_len = name_end - name_start;
+ cc = *name_end;
+ *name_end = NUL;
+
+ int name_exists = generic_name_exists(name_start, name_len, gfatab,
+ cctx);
+ *name_end = cc;
+ if (name_exists)
+ return NULL;
+
+ if (ga_grow(&gfatab->gfat_args, 1) == FAIL)
+ return NULL;
+ generic_T *generic =
+ &((generic_T *)gfatab->gfat_args.ga_data)[gfatab->gfat_args.ga_len];
+ gfatab->gfat_args.ga_len++;
+
+ generic->gt_name = alloc(name_len + 1);
+ if (generic->gt_name == NULL)
+ return NULL;
+ mnv_strncpy(generic->gt_name, name_start, name_len);
+ generic->gt_type = NULL;
+
+ if (MNV_ISWHITE(*p))
+ {
+ semsg(_(e_no_white_space_allowed_after_str_str), generic->gt_name,
+ name_start);
+ return NULL;
+ }
+
+ if (*p != ',' && *p != '>')
+ {
+ semsg(_(e_missing_comma_in_generic_function_str), start);
+ return NULL;
+ }
+ if (*p == ',')
+ {
+ if (!MNV_ISWHITE(*(p + 1)))
+ {
+ semsg(_(e_white_space_required_after_str_str), ",", p);
+ return NULL;
+ }
+ p++;
+ }
+ }
+ if (*p != '>')
+ return NULL;
+ p++;
+
+ int gfat_sz = generic_func_args_table_size(gfatab);
+
+ if (gfat_sz == 0)
+ {
+ emsg_funcname(e_empty_type_list_for_generic_function_str, func_name);
+ return NULL;
+ }
+
+ // set the generic params to VAR_ANY type
+ if (ga_grow(&gfatab->gfat_param_types, gfat_sz) == FAIL)
+ return NULL;
+
+ gfatab->gfat_param_types.ga_len = gfat_sz;
+ for (int i = 0; i < generic_func_args_table_size(gfatab); i++)
+ {
+ type_T *gt = &((type_T *)gfatab->gfat_param_types.ga_data)[i];
+
+ CLEAR_POINTER(gt);
+ gt->tt_type = VAR_ANY;
+ gt->tt_flags = TTFLAG_GENERIC;
+
+ generic_T *generic = &((generic_T *)gfatab->gfat_args.ga_data)[i];
+ generic->gt_type = gt;
+ }
+
+ return p;
+}
+
+/*
+ * Initialize a new generic function "fp" using the list of generic types and
+ * generic arguments in "gfatab".
+ *
+ * This function:
+ * - Marks the function as generic.
+ * - Sets the generic argument count and stores the type and argument lists.
+ * - Transfers ownership of the arrays from the growarrays to the function.
+ * - Initializes the generic function's lookup table.
+ */
+ void
+generic_func_init(ufunc_T *fp, gfargs_tab_T *gfatab)
+{
+ fp->uf_flags |= FC_GENERIC;
+ fp->uf_generic_argcount = gfatab->gfat_args.ga_len;
+ fp->uf_generic_args = (generic_T *)gfatab->gfat_args.ga_data;
+ ga_init(&gfatab->gfat_args); // remove the reference to the args
+ fp->uf_generic_param_types = (type_T *)gfatab->gfat_param_types.ga_data;
+ ga_init(&gfatab->gfat_param_types); // remove the reference to the types
+ ga_init(&fp->uf_generic_arg_types);
+ hash_init(&fp->uf_generic_functab);
+}
+
+/*
+ * Initialize the generic function args table
+ */
+ void
+generic_func_args_table_init(gfargs_tab_T *gfatab)
+{
+ ga_init2(&gfatab->gfat_args, sizeof(generic_T), 10);
+ ga_init2(&gfatab->gfat_param_types, sizeof(type_T), 10);
+ ga_init2(&gfatab->gfat_arg_types, sizeof(type_T), 10);
+}
+
+/*
+ * Return the number of entries in the generic function args table
+ */
+ int
+generic_func_args_table_size(gfargs_tab_T *gfatab)
+{
+ return gfatab->gfat_args.ga_len;
+}
+
+/*
+ * Free all the generic function args table items
+ */
+ void
+generic_func_args_table_clear(gfargs_tab_T *gfatab)
+{
+ clear_type_list(&gfatab->gfat_param_types);
+ clear_type_list(&gfatab->gfat_arg_types);
+ for (int i = 0; i < gfatab->gfat_args.ga_len; i++)
+ {
+ generic_T *generic = &((generic_T *)gfatab->gfat_args.ga_data)[i];
+ MNV_CLEAR(generic->gt_name);
+ }
+ ga_clear(&gfatab->gfat_args);
+}
+
+/*
+ * When a cloning a function "fp" to "new_fp", copy the generic function
+ * related information.
+ */
+ void
+copy_generic_function(ufunc_T *fp, ufunc_T *new_fp)
+{
+ int i;
+ int sz;
+
+ if (!IS_GENERIC_FUNC(fp))
+ return;
+
+ sz = fp->uf_generic_argcount * sizeof(type_T);
+ new_fp->uf_generic_param_types = alloc_clear(sz);
+ if (new_fp->uf_generic_param_types == NULL)
+ return;
+
+ memcpy(new_fp->uf_generic_param_types, fp->uf_generic_param_types, sz);
+
+ sz = fp->uf_generic_argcount * sizeof(generic_T);
+ new_fp->uf_generic_args = alloc_clear(sz);
+ if (new_fp->uf_generic_args == NULL)
+ {
+ MNV_CLEAR(new_fp->uf_generic_param_types);
+ return;
+ }
+ memcpy(new_fp->uf_generic_args, fp->uf_generic_args, sz);
+
+ for (i = 0; i < fp->uf_generic_argcount; i++)
+ new_fp->uf_generic_args[i].gt_name =
+ mnv_strsave(fp->uf_generic_args[i].gt_name);
+
+ for (i = 0; i < fp->uf_generic_argcount; i++)
+ new_fp->uf_generic_args[i].gt_type =
+ &new_fp->uf_generic_param_types[i];
+
+ ga_init(&new_fp->uf_generic_arg_types);
+ hash_init(&new_fp->uf_generic_functab);
+}
+
+/*
+ * Returns the index of the generic type pointer "t" in the generic type list
+ * of the function "fp".
+ *
+ * Arguments:
+ * fp - pointer to the generic function (ufunc_T)
+ * t - pointer to the type_T to search for in the function's generic type
+ * list
+ *
+ * Returns:
+ * The zero-based index of "t" in fp->uf_generic_param_types if found,
+ * or -1 if not found.
+ */
+ static int
+get_generic_type_index(ufunc_T *fp, type_T *t)
+{
+ for (int i = 0; i < fp->uf_generic_argcount; i++)
+ {
+ if (&fp->uf_generic_param_types[i] == t)
+ return i;
+ }
+ return -1;
+}
+
+/*
+ * Evaluates the type arguments for a generic function call and looks up the
+ * corresponding concrete function.
+ *
+ * Arguments:
+ * ufunc - the original (possibly generic) function to evaluate
+ * name - the function name (used for error messages and lookup)
+ * argp - pointer to a pointer to the argument string; on entry, "*argp"
+ * should point to the character after the function name (possibly
+ * '<')
+ *
+ * Returns:
+ * The concrete function corresponding to the given type arguments,
+ * or NULL on error (with an error message reported).
+ *
+ * Behavior:
+ * - If "ufunc" is a generic function and "*argp" points to '<', attempts to
+ * find or instantiate the concrete function with the specified type
+ * arguments. On success, advances "*argp" past the type argument list.
+ * - If "ufunc" is generic but "*argp" does not point to '<', reports a
+ * missing type argument error.
+ * - If "ufunc" is not generic but "*argp" points to '<', reports an error
+ * that the function is not generic.
+ * - Otherwise, returns the original function.
+ */
+ ufunc_T *
+eval_generic_func(
+ ufunc_T *ufunc,
+ char_u *name,
+ char_u **argp)
+{
+ if (IS_GENERIC_FUNC(ufunc))
+ {
+ if (**argp == '<')
+ ufunc = find_generic_func(ufunc, name, argp);
+ else
+ {
+ emsg_funcname(e_generic_func_missing_type_args_str, name);
+ return NULL;
+ }
+ }
+ else if (**argp == '<')
+ {
+ emsg_funcname(e_not_a_generic_function_str, name);
+ return NULL;
+ }
+
+ return ufunc;
+}
+
+/*
+ * Checks if the string at "*argp" represents a generic function call with type
+ * arguments, i.e., if it starts with a '<', contains a valid type argument
+ * list, a closing '>', and is immediately followed by '('.
+ *
+ * On entry, "*argp" should point to the '<' character.
+ * If the pattern matches, advances "*argp" to point to the '(' and returns
+ * TRUE. If not, leaves "*argp" unchanged and returns FALSE.
+ *
+ * Example:
+ * "<number, string>("
+ */
+ int
+generic_func_call(char_u **argp)
+{
+ char_u *p = *argp;
+
+ if (*p != '<')
+ return FALSE;
+
+ if (skip_generic_func_type_args(&p) == FAIL)
+ return FALSE;
+
+ if (*p != '(')
+ return FALSE;
+
+ *argp = p;
+ return TRUE;
+}
+
+/*
+ * Recursively replaces all occurrences of the generic type "generic_type" in a
+ * type structure with the corresponding concrete type from "new_ufunc", based
+ * on the mapping from the original generic function "ufunc".
+ *
+ * This is used when instantiating a new function "new_ufunc" from a generic
+ * function "ufunc" with specific type arguments. The function updates all
+ * relevant type pointers in place, including nested types (such as lists,
+ * dictionaries, and tuples).
+ *
+ * Arguments:
+ * ufunc - the original generic function
+ * new_ufunc - the new function being created with concrete types
+ * generic_type - the generic type to be replaced (may be a nested type)
+ * specific_type - pointer to the location where the concrete type should be
+ * set
+ * func_type - pointer to the function type to update (may be NULL)
+ */
+ static void
+update_generic_type(
+ ufunc_T *ufunc,
+ ufunc_T *new_ufunc,
+ type_T *generic_type,
+ type_T **specific_type,
+ type_T **func_type)
+{
+ int idx;
+
+ switch (generic_type->tt_type)
+ {
+ case VAR_ANY:
+ idx = get_generic_type_index(ufunc, generic_type);
+ if (idx != -1)
+ {
+ *specific_type = new_ufunc->uf_generic_args[idx].gt_type;
+ if (func_type != NULL)
+ *func_type = new_ufunc->uf_generic_args[idx].gt_type;
+ }
+ break;
+ case VAR_LIST:
+ case VAR_DICT:
+ update_generic_type(ufunc, new_ufunc, generic_type->tt_member,
+ &(*specific_type)->tt_member,
+ func_type != NULL ? &(*func_type)->tt_member : NULL);
+ break;
+ case VAR_TUPLE:
+ for (int i = 0; i < generic_type->tt_argcount; i++)
+ update_generic_type(ufunc, new_ufunc,
+ generic_type->tt_args[i],
+ &(*specific_type)->tt_args[i],
+ func_type != NULL ? &(*func_type)->tt_args[i] : NULL);
+ break;
+ case VAR_FUNC:
+ for (int i = 0; i < generic_type->tt_argcount; i++)
+ update_generic_type(ufunc, new_ufunc,
+ generic_type->tt_args[i],
+ &(*specific_type)->tt_args[i],
+ func_type != NULL ? &(*func_type)->tt_args[i] : NULL);
+ update_generic_type(ufunc, new_ufunc,
+ generic_type->tt_member,
+ &(*specific_type)->tt_member,
+ func_type != NULL ? &(*func_type)->tt_member : NULL);
+ break;
+ default:
+ break;
+ }
+}
+
+/*
+ * Adds a new concrete instance of a generic function for a specific set of
+ * type arguments.
+ *
+ * Arguments:
+ * fp - the original generic function to instantiate
+ * key - a string key representing the specific type arguments (used for
+ * lookup)
+ * gfatab - generic function args table containing the parsed type
+ * arguments and their names
+ *
+ * Returns:
+ * Pointer to the new ufunc_T representing the instantiated function,
+ * or NULL if the function already exists or on allocation failure.
+ *
+ * This function:
+ * - Checks if a function with the given type arguments already exists.
+ * - Allocates and initializes a new function instance with the specific
+ * types.
+ * - Updates the function's name and expanded name to include the type
+ * arguments.
+ * - Copies and updates all relevant type information (argument types, return
+ * type, vararg type, function type), replacing generic types with the
+ * actual types.
+ * - Sets the new function's status to UF_TO_BE_COMPILED.
+ * - Registers the new function in the generic function's lookup table.
+ */
+ static ufunc_T *
+generic_func_add(ufunc_T *fp, char_u *key, gfargs_tab_T *gfatab)
+{
+ hashtab_T *ht = &fp->uf_generic_functab;
+ long_u hash;
+ hashitem_T *hi;
+ int i;
+
+ hash = hash_hash(key);
+ hi = hash_lookup(ht, key, hash);
+ if (!HASHITEM_EMPTY(hi))
+ return NULL;
+
+ size_t keylen = STRLEN(key);
+ gfitem_T *gfitem = alloc(sizeof(gfitem_T) + keylen);
+ if (gfitem == NULL)
+ return NULL;
+
+ STRCPY(gfitem->gfi_name, key);
+
+ ufunc_T *new_fp = copy_function(fp, (int)(keylen + 2));
+ if (new_fp == NULL)
+ {
+ mnv_free(gfitem);
+ return NULL;
+ }
+
+ new_fp->uf_generic_arg_types = gfatab->gfat_arg_types;
+ // now that the type arguments is copied, remove the reference to the type
+ // arguments
+ ga_init(&gfatab->gfat_arg_types);
+
+ if (fp->uf_class != NULL)
+ new_fp->uf_class = fp->uf_class;
+
+ // Create a new name for the function: name<type1, type2...>
+ new_fp->uf_name[new_fp->uf_namelen] = '<';
+ STRCPY(new_fp->uf_name + new_fp->uf_namelen + 1, key);
+ new_fp->uf_name[new_fp->uf_namelen + keylen + 1] = '>';
+ new_fp->uf_namelen += keylen + 2;
+
+ if (new_fp->uf_name_exp != NULL)
+ {
+ char_u *new_name_exp = alloc(STRLEN(new_fp->uf_name_exp) + keylen + 3);
+ if (new_name_exp != NULL)
+ {
+ STRCPY(new_name_exp, new_fp->uf_name_exp);
+ STRCAT(new_name_exp, "<");
+ STRCAT(new_name_exp, key);
+ STRCAT(new_name_exp, ">");
+ mnv_free(new_fp->uf_name_exp);
+ new_fp->uf_name_exp = new_name_exp;
+ }
+ }
+
+ gfitem->gfi_ufunc = new_fp;
+ gfitem->gfi_ufunc->uf_def_status = UF_TO_BE_COMPILED;
+
+ // create a copy of
+ // - all the argument types
+ // - return type
+ // - vararg type
+ // - function type
+ // if any generic type is used, it will be replaced below).
+ for (i = 0; i < fp->uf_args.ga_len; i++)
+ new_fp->uf_arg_types[i] = copy_type_deep(fp->uf_arg_types[i],
+ &new_fp->uf_type_list);
+
+ if (fp->uf_ret_type != NULL)
+ new_fp->uf_ret_type = copy_type_deep(fp->uf_ret_type,
+ &new_fp->uf_type_list);
+
+ if (fp->uf_va_type != NULL)
+ new_fp->uf_va_type = copy_type_deep(fp->uf_va_type,
+ &new_fp->uf_type_list);
+
+ if (fp->uf_func_type != NULL)
+ new_fp->uf_func_type = copy_type_deep(fp->uf_func_type,
+ &new_fp->uf_type_list);
+
+ // Replace the t_any generic types with the actual types
+ for (i = 0; i < fp->uf_generic_argcount; i++)
+ {
+ generic_T *generic_arg;
+ generic_arg = (generic_T *)gfatab->gfat_args.ga_data + i;
+ generic_T *gt = &new_fp->uf_generic_args[i];
+ gt->gt_type = generic_arg->gt_type;
+ }
+
+ // Update any generic types in the function arguments
+ for (i = 0; i < fp->uf_args.ga_len; i++)
+ update_generic_type(fp, new_fp, fp->uf_arg_types[i],
+ &new_fp->uf_arg_types[i],
+ &new_fp->uf_func_type->tt_args[i]);
+
+ // Update the vararg type if it uses generic types
+ if (fp->uf_va_type != NULL)
+ update_generic_type(fp, new_fp, fp->uf_va_type, &new_fp->uf_va_type,
+ NULL);
+
+ // Update the return type if it is a generic type
+ if (fp->uf_ret_type != NULL)
+ update_generic_type(fp, new_fp, fp->uf_ret_type, &new_fp->uf_ret_type,
+ &new_fp->uf_func_type->tt_member);
+
+ hash_add_item(ht, hi, gfitem->gfi_name, hash);
+
+ return new_fp;
+}
+
+/*
+ * Looks up a concrete instance of a generic function "fp" using the type
+ * arguments specified in "gfatab".
+ *
+ * The lookup key is constructed by concatenating the type argument names from
+ * "gfatab", separated by ", ", and stored in the provided growarray
+ * "gfkey_gap". The contents of "gfkey_gap" will be overwritten.
+ *
+ * Arguments:
+ * fp - the generic function to search in
+ * gfatab - generic function args table containing the parsed type
+ * arguments and their names
+ * gfkey_gap - growarray used to build and store the lookup key string
+ *
+ * Returns:
+ * Pointer to the ufunc_T representing the concrete function if found, or
+ * NULL if no matching function exists.
+ */
+ static ufunc_T *
+generic_lookup_func(ufunc_T *fp, gfargs_tab_T *gfatab, garray_T *gfkey_gap)
+{
+ hashtab_T *ht = &fp->uf_generic_functab;
+ hashitem_T *hi;
+
+ for (int i = 0; i < gfatab->gfat_args.ga_len; i++)
+ {
+ generic_T *generic_arg;
+
+ generic_arg = (generic_T *)gfatab->gfat_args.ga_data + i;
+ ga_concat(gfkey_gap, generic_arg->gt_name);
+
+ if (i != gfatab->gfat_args.ga_len - 1)
+ {
+ ga_append(gfkey_gap, ',');
+ ga_append(gfkey_gap, ' ');
+ }
+ }
+ ga_append(gfkey_gap, NUL);
+
+ char_u *key = ((char_u *)gfkey_gap->ga_data);
+
+ hi = hash_find(ht, key);
+
+ if (HASHITEM_EMPTY(hi))
+ return NULL;
+
+ gfitem_T *gfitem = HI2GFITEM(hi);
+ return gfitem->gfi_ufunc;
+}
+
+/*
+ * Returns a concrete instance of the generic function "fp" using the type
+ * arguments specified in "gfatab". If such an instance does not exist,
+ * it is created and registered.
+ *
+ * Arguments:
+ * fp - the generic function to instantiate
+ * gfatab - generic function args table containing the parsed type
+ * arguments and their names
+ *
+ * Returns:
+ * Pointer to the ufunc_T representing the concrete function instance,
+ * or NULL if the type arguments are invalid or on allocation failure.
+ *
+ * Behavior:
+ * - If "fp" is not a generic function and no type arguments are given,
+ * returns "fp" as-is.
+ * - If "fp" is not generic but type arguments are given, reports an error
+ * and returns NULL.
+ * - Validates the number of type arguments, reporting errors for missing,
+ * too few, or too many.
+ * - Looks up an existing function instance with the given types.
+ * - If not found, creates and registers a new function instance.
+ */
+ ufunc_T *
+generic_func_get(ufunc_T *fp, gfargs_tab_T *gfatab)
+{
+ char *emsg = NULL;
+
+ if (!IS_GENERIC_FUNC(fp))
+ {
+ if (gfatab && generic_func_args_table_size(gfatab) > 0)
+ {
+ emsg_funcname(e_not_a_generic_function_str, fp->uf_name);
+ return NULL;
+ }
+ return fp;
+ }
+
+ if (gfatab == NULL || gfatab->gfat_args.ga_len == 0)
+ emsg = e_generic_func_missing_type_args_str;
+ else if (gfatab->gfat_args.ga_len < fp->uf_generic_argcount)
+ emsg = e_not_enough_types_for_generic_function_str;
+ else if (gfatab->gfat_args.ga_len > fp->uf_generic_argcount)
+ emsg = e_too_many_types_for_generic_function_str;
+
+ if (emsg != NULL)
+ {
+ emsg_funcname(emsg, printable_func_name(fp));
+ return NULL;
+ }
+
+ // generic function call
+ garray_T gfkey_ga;
+
+ ga_init2(&gfkey_ga, 1, 80);
+
+ // Look up the function with specific types
+ ufunc_T *generic_fp = generic_lookup_func(fp, gfatab, &gfkey_ga);
+ if (generic_fp == NULL)
+ // generic function with these type arguments doesn't exist.
+ // Create a new one.
+ generic_fp = generic_func_add(fp, (char_u *)gfkey_ga.ga_data, gfatab);
+ ga_clear(&gfkey_ga);
+
+ return generic_fp;
+}
+
+/*
+ * Looks up or creates a concrete instance of a generic function "ufunc" using
+ * the type arguments specified after the function name in "name".
+ *
+ * On entry, "name" points to the function name, and "*argp" points to the
+ * opening '<' of the type argument list (i.e., name + namelen).
+ *
+ * Arguments:
+ * ufunc - the generic function to instantiate or look up
+ * name - the function name, followed by the type argument list
+ * argp - pointer to a pointer to the type argument list (should point to
+ * '<'); on success, advanced to the character after the closing '>'
+ *
+ * Returns:
+ * Pointer to the ufunc_T representing the concrete function instance if
+ * successful, or NULL if parsing fails or the instance cannot be created.
+ *
+ * This function:
+ * - Parses the type arguments from the string after the function name.
+ * - Looks up an existing function instance with those type arguments.
+ * - If not found, creates and registers a new function instance.
+ * - Advances "*argp" to after the type argument list on success.
+ */
+ ufunc_T *
+find_generic_func(ufunc_T *ufunc, char_u *name, char_u **argp)
+{
+ gfargs_tab_T gfatab;
+ char_u *p;
+ ufunc_T *new_ufunc = NULL;
+
+ generic_func_args_table_init(&gfatab);
+
+ // Get the list of types following the name
+ p = parse_generic_func_type_args(name, *argp - name, *argp, &gfatab, NULL);
+ if (p != NULL)
+ {
+ new_ufunc = generic_func_get(ufunc, &gfatab);
+ *argp = p;
+ }
+
+ generic_func_args_table_clear(&gfatab);
+
+ return new_ufunc;
+}
+
+/*
+ * Searches for a generic type with the given name "gt_name" in the generic
+ * function "ufunc".
+ *
+ * Arguments:
+ * gt_name - the name of the generic type to search for
+ * ufunc - the generic function in which to search for the type
+ *
+ * Returns:
+ * Pointer to the type_T representing the found generic type,
+ * or NULL if the type is not found or if "ufunc" is not a generic function.
+ */
+ static type_T *
+find_generic_type_in_ufunc(char_u *gt_name, size_t name_len, ufunc_T *ufunc)
+{
+ if (!IS_GENERIC_FUNC(ufunc))
+ return NULL;
+
+ for (int i = 0; i < ufunc->uf_generic_argcount; i++)
+ {
+ generic_T *generic;
+
+ generic = ((generic_T *)ufunc->uf_generic_args) + i;
+ if (STRNCMP(generic->gt_name, gt_name, name_len) == 0)
+ {
+ type_T *type = generic->gt_type;
+ return type;
+ }
+ }
+
+ return NULL;
+}
+
+/*
+ * Searches for a generic type with the given name "gt_name" in the current
+ * function context "cctx" and its outer (enclosing) contexts, if necessary.
+ *
+ * Arguments:
+ * gt_name - the name of the generic type to search for
+ * cctx - the current compile context, which may be nested
+ *
+ * Returns:
+ * Pointer to the type_T representing the found generic type,
+ * or NULL if the type is not found in the current or any outer context.
+ */
+ static type_T *
+find_generic_type_in_cctx(char_u *gt_name, size_t name_len, cctx_T *cctx)
+{
+ type_T *type;
+
+ type = find_generic_type_in_ufunc(gt_name, name_len, cctx->ctx_ufunc);
+ if (type != NULL)
+ return type;
+
+ if (cctx->ctx_outer != NULL)
+ return find_generic_type_in_cctx(gt_name, name_len, cctx->ctx_outer);
+
+ return NULL;
+}
+
+/*
+ * Looks up a generic type with the given name "gt_name" in the generic
+ * function "ufunc". If not found, searches in the enclosing compile context
+ * "cctx" (for nested functions).
+ *
+ * Arguments:
+ * gt_name - the name of the generic type to search for
+ * ufunc - the generic function to search in first (may be NULL)
+ * cctx - the compile context to search in outer functions if not found
+ * in "ufunc" (may be NULL)
+ *
+ * Returns:
+ * Pointer to the type_T representing the found generic type, or NULL if the
+ * type is not found in the given function or any outer context.
+ */
+ type_T *
+find_generic_type(
+ char_u *gt_name,
+ size_t name_len,
+ ufunc_T *ufunc,
+ cctx_T *cctx)
+{
+ if (ufunc != NULL)
+ {
+ type_T *type = find_generic_type_in_ufunc(gt_name, name_len, ufunc);
+ if (type != NULL)
+ return type;
+ }
+
+ if (cctx != NULL && ufunc != cctx->ctx_ufunc)
+ return find_generic_type_in_cctx(gt_name, name_len, cctx);
+
+ return NULL;
+}
+
+/*
+ * Frees all concrete function instances stored in the generic function table
+ * of "fp". This includes freeing each instantiated function and its
+ * associated gfitem_T structure, and clearing the hash table.
+ *
+ * Arguments:
+ * fp - the generic function whose function table should be freed
+ */
+ static void
+free_generic_functab(ufunc_T *fp)
+{
+ hashtab_T *ht = &fp->uf_generic_functab;
+ long todo;
+ hashitem_T *hi;
+
+ todo = (long)ht->ht_used;
+ FOR_ALL_HASHTAB_ITEMS(ht, hi, todo)
+ {
+ if (!HASHITEM_EMPTY(hi))
+ {
+ gfitem_T *gfitem = HI2GFITEM(hi);
+
+ func_clear_free(gfitem->gfi_ufunc, FALSE);
+ mnv_free(gfitem);
+ --todo;
+ }
+ }
+ hash_clear(ht);
+}
+
+/*
+ * Frees all memory and state associated with a generic function "fp".
+ * This includes the generic type list, generic argument list, and all
+ * concrete function instances in the generic function table.
+ *
+ * Arguments:
+ * fp - the generic function to clear
+ */
+ void
+generic_func_clear_items(ufunc_T *fp)
+{
+ MNV_CLEAR(fp->uf_generic_param_types);
+ clear_type_list(&fp->uf_generic_arg_types);
+ for (int i = 0; i < fp->uf_generic_argcount; i++)
+ MNV_CLEAR(fp->uf_generic_args[i].gt_name);
+ MNV_CLEAR(fp->uf_generic_args);
+ free_generic_functab(fp);
+ fp->uf_flags &= ~FC_GENERIC;
+}
+
+#endif // FEAT_EVAL