/**************************************************************************** * Copyright 2020-2022,2023 Thomas E. Dickey * * * * Permission is hereby granted, free of charge, to any person obtaining a * * copy of this software and associated documentation files (the * * "Software"), to deal in the Software without restriction, including * * without limitation the rights to use, copy, modify, merge, publish, * * distribute, distribute with modifications, sublicense, and/or sell * * copies of the Software, and to permit persons to whom the Software is * * furnished to do so, subject to the following conditions: * * * * The above copyright notice and this permission notice shall be included * * in all copies or substantial portions of the Software. * * * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS * * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. * * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, * * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR * * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR * * THE USE OR OTHER DEALINGS IN THE SOFTWARE. * * * * Except as contained in this notice, the name(s) of the above copyright * * holders shall not be used in advertising or otherwise to promote the * * sale, use or other dealings in this Software without prior written * * authorization. * ****************************************************************************/ /* * Author: Thomas E. Dickey * * $Id: test_tparm.c,v 1.39 2023/11/11 01:00:03 tom Exp $ * * Exercise tparm/tiparm, either for all possible capabilities with fixed * parameters, or one capability with specific combinations of parameters. */ #define USE_TINFO #include #if NCURSES_XNAMES #if HAVE_TERM_ENTRY_H #include #else #undef NCURSES_XNAMES #define NCURSES_XNAMES 0 #endif #endif #define MAX_PARM 9 #define GrowArray(array,limit,length) \ if (length + 2 >= limit) { \ limit *= 2; \ array = typeRealloc(char *, limit, array); \ if (array == 0) { \ failed("no memory: " #array); \ } \ } static GCC_NORETURN void failed(const char *); static void failed(const char *msg) { fprintf(stderr, "%s\n", msg); ExitProgram(EXIT_FAILURE); } #if HAVE_TIGETSTR static int a_opt; static int p_opt; static int v_opt; #if HAVE_TIPARM static int i_opt; #endif #if HAVE_TIPARM_S static int s_opt; #endif /* * Total tests (and failures): */ static long total_tests; static long total_fails; /* * Total characters formatted for tputs: */ static long total_nulls; static long total_ctrls; static long total_print; static int output_func(int ch) { if (ch == 0) { total_nulls++; } else if (ch < 32 || (ch >= 127 && ch < 160)) { total_ctrls++; } else { total_print++; } return ch; } static int isNumeric(char *source) { char *next = 0; long value = strtol(source, &next, 0); int result = (next == 0 || next == source || *next != '\0') ? 0 : 1; (void) value; return result; } static int relevant(const char *name, const char *value) { int code = 1; if (VALID_STRING(value)) { if (strstr(value, "%p") == 0 && strstr(value, "%d") == 0 && strstr(value, "%s") == 0 && (!p_opt || strstr(value, "$<") == 0)) { if (v_opt > 2) printf("? %s noparams\n", name); code = 0; } } else { if (v_opt > 2) { printf("? %s %s\n", (value == ABSENT_STRING) ? "absent" : "cancel", name); } code = 0; } return code; } static int increment(long *all_parms, int *num_parms, int len_parms, int end_parms) { int rc = 0; int n; if (len_parms > MAX_PARM) len_parms = MAX_PARM; if (end_parms < len_parms) { if (all_parms[end_parms]++ >= num_parms[end_parms]) { all_parms[end_parms] = 0; increment(all_parms, num_parms, len_parms, end_parms + 1); } } for (n = 0; n < len_parms; ++n) { if (all_parms[n] != 0) { rc = 1; break; } } /* return 1 until the vector resets to all 0's */ return rc; } /* parse the format string to determine which positional parameters * are assumed to be strings. */ #if HAVE_TISCAN_S static int analyze_format(const char *format, int *mask, char **p_is_s) { int arg_count; int arg_mask; int n; if (tiscan_s(&arg_count, &arg_mask, format) == OK) { *mask = arg_mask; for (n = 0; n < MAX_PARM; ++n) { static char dummy[1]; p_is_s[n] = (arg_mask & 1) ? dummy : NULL; arg_mask >>= 1; } } else { *mask = 0; arg_count = 0; for (n = 0; n < MAX_PARM; ++n) { p_is_s[n] = NULL; } } return arg_count; } #elif HAVE__NC_TPARM_ANALYZE extern int _nc_tparm_analyze(TERMINAL *, const char *, char **, int *); static int analyze_format(const char *format, int *mask, char **p_is_s) { int popcount = 0; int analyzed = _nc_tparm_analyze(cur_term, format, p_is_s, &popcount); int n; if (analyzed < popcount) { analyzed = popcount; } *mask = 0; for (n = 0; n < MAX_PARM; ++n) { if (p_is_s[n]) *mask |= (1 << n); } return analyzed; } #else /* TODO: make this work without direct use of ncurses internals. */ static int analyze_format(const char *format, int *mask, char **p_is_s) { int n; char *filler = strstr(format, "%s"); *mask = 0; for (n = 0; n < MAX_PARM; ++n) { p_is_s[n] = filler; } return n; } #endif #define NumStr(n) use_strings[n] \ ? (long) (my_intptr_t) (number[n] \ ? string[n] \ : NULL) \ : number[n] #define NS_0(fmt) fmt #define NS_1(fmt) NS_0(fmt), NumStr(0) #define NS_2(fmt) NS_1(fmt), NumStr(1) #define NS_3(fmt) NS_2(fmt), NumStr(2) #define NS_4(fmt) NS_3(fmt), NumStr(3) #define NS_5(fmt) NS_4(fmt), NumStr(4) #define NS_6(fmt) NS_5(fmt), NumStr(5) #define NS_7(fmt) NS_6(fmt), NumStr(6) #define NS_8(fmt) NS_7(fmt), NumStr(7) #define NS_9(fmt) NS_8(fmt), NumStr(8) static void test_tparm(const char *name, const char *format, long *number, char **string) { char *use_strings[MAX_PARM]; char *result = NULL; int nparam; int mask; nparam = analyze_format(format, &mask, use_strings); #if HAVE_TIPARM_S if (s_opt) { switch (nparam) { case 0: result = tiparm_s(0, mask, NS_0(format)); break; case 1: result = tiparm_s(1, mask, NS_1(format)); break; case 2: result = tiparm_s(2, mask, NS_2(format)); break; case 3: result = tiparm_s(3, mask, NS_3(format)); break; case 4: result = tiparm_s(4, mask, NS_4(format)); break; case 5: result = tiparm_s(5, mask, NS_5(format)); break; case 6: result = tiparm_s(6, mask, NS_6(format)); break; case 7: result = tiparm_s(7, mask, NS_7(format)); break; case 8: result = tiparm_s(8, mask, NS_8(format)); break; case 9: result = tiparm_s(9, mask, NS_9(format)); break; } } else #endif #if HAVE_TIPARM if (i_opt) { switch (nparam) { case 0: result = tiparm(NS_0(format)); break; case 1: result = tiparm(NS_1(format)); break; case 2: result = tiparm(NS_2(format)); break; case 3: result = tiparm(NS_3(format)); break; case 4: result = tiparm(NS_4(format)); break; case 5: result = tiparm(NS_5(format)); break; case 6: result = tiparm(NS_6(format)); break; case 7: result = tiparm(NS_7(format)); break; case 8: result = tiparm(NS_8(format)); break; case 9: result = tiparm(NS_9(format)); break; } } else #endif result = tparm(NS_9(format)); total_tests++; if (result != NULL) { tputs(result, 1, output_func); } else { total_fails++; } if (v_opt > 1) { int n; printf(".. %3d =", result != 0 ? (int) strlen(result) : -1); for (n = 0; n < nparam; ++n) { if (use_strings[n]) { if (number[n]) { printf(" \"%s\"", string[n]); } else { printf(" ?"); } } else { printf(" %2ld", number[n]); } } printf(" %s\n", name); } } static void usage(int ok) { static const char *msg[] = { "Usage: test_tparm [options] [capability] [value1 [value2 [...]]]" ,"" ,"Use tparm/tputs for all distinct combinations of given capability." ,"" ,USAGE_COMMON ,"Options:" ," -T TERM override $TERM; this may be a comma-separated list or \"-\"" ," to read a list from standard-input" ," -a test all combinations of parameters" ," [value1...] forms a vector of maximum parameter-values." #if HAVE_TIPARM ," -i test tiparm rather than tparm" #endif ," -p test capabilities with no parameters but having padding" ," -r NUM repeat tests NUM times" #if HAVE_TIPARM_S ," -s test tiparm_s rather than tparm" #endif ," -v show values and results" }; unsigned n; for (n = 0; n < SIZEOF(msg); ++n) { fprintf(stderr, "%s\n", msg[n]); } ExitProgram(ok ? EXIT_SUCCESS : EXIT_FAILURE); } #define PLURAL(n) n, (n != 1) ? "s" : "" #define COLONS(n) (n >= 1) ? ":" : "" #define NUMFORM "%10ld" /* *INDENT-OFF* */ VERSION_COMMON() /* *INDENT-ON* */ int main(int argc, char *argv[]) { int ch; int n; int r_run, t_run, n_run; char *old_term = getenv("TERM"); int r_opt = 1; char *t_opt = 0; int std_caps = 0; /* predefine items in all_caps[] */ int len_caps = 0; /* cur # of items in all_caps[] */ int max_caps = 10; /* max # of items in all_caps[] */ char **all_caps = typeCalloc(char *, max_caps); long all_parms[10]; /* workspace for "-a" option */ int len_terms = 0; /* cur # of items in all_terms[] */ int max_terms = 10; /* max # of items in all_terms[] */ char **all_terms = typeCalloc(char *, max_terms); int use_caps; int max_name = 10; /* max # of items in cap_name[] */ int max_data = 10; /* max # of items in cap_data[] */ char **cap_name; char **cap_data; int len_parms = 0; /* cur # of items in num_parms[], str_parms[] */ int max_parms = argc + 10; /* max # of items in num_parms[], str_parms[] */ int *num_parms = typeCalloc(int, max_parms); char **str_parms = typeCalloc(char *, max_parms); long use_parms = 1; if (all_caps == 0 || all_terms == 0 || num_parms == 0 || str_parms == 0) failed("no memory"); while ((ch = getopt(argc, argv, OPTS_COMMON "T:aipr:sv")) != -1) { switch (ch) { case 'T': t_opt = optarg; break; case 'a': ++a_opt; break; #if HAVE_TIPARM case 'i': ++i_opt; break; #endif case 'p': ++p_opt; break; case 'r': r_opt = atoi(optarg); break; #if HAVE_TIPARM_S case 's': ++s_opt; break; #endif case 'v': ++v_opt; break; case OPTS_VERSION: show_version(argv); ExitProgram(EXIT_SUCCESS); default: usage(ch == OPTS_USAGE); /* NOTREACHED */ } } /* * If there is a nonnumeric parameter after the options, use that as the * capability name. */ if (optind < argc) { if (!isNumeric(argv[optind])) { all_caps[len_caps++] = strdup(argv[optind++]); } } /* * Any remaining arguments must be possible parameter values. If numeric, * and "-a" is not set, use those as the actual values for which the * capabilities are tested. */ while (optind < argc) { if (isNumeric(argv[optind])) { char *dummy = 0; long value = strtol(argv[optind], &dummy, 0); num_parms[len_parms] = (int) value; } str_parms[len_parms] = argv[optind]; ++optind; ++len_parms; } for (n = len_parms; n < max_parms; ++n) { static char dummy[1]; str_parms[n] = dummy; } if (v_opt) { printf("%d parameter%s%s\n", PLURAL(len_parms), COLONS(len_parms)); if (v_opt > 3) { for (n = 0; n < len_parms; ++n) { printf(" %d: %d (%s)\n", n + 1, num_parms[n], str_parms[n]); } } } /* * Make a list of values for $TERM. Accept "-" for standard input to * simplify scripting a check of the whole database. */ old_term = strdup((old_term == 0) ? "unknown" : old_term); if (t_opt != 0) { if (!strcmp(t_opt, "-")) { char buffer[BUFSIZ]; while (fgets(buffer, sizeof(buffer) - 1, stdin) != 0) { char *s = buffer; char *t; while (isspace(UChar(s[0]))) ++s; t = s + strlen(s); while (t != s && isspace(UChar(t[-1]))) *--t = '\0'; s = strdup(s); if (len_terms + 2 >= max_terms) { max_terms *= 2; all_terms = typeRealloc(char *, max_terms, all_terms); if (all_terms == 0) failed("no memory: all_terms"); } all_terms[len_terms++] = s; } } else { char *s = t_opt; char *t; while ((t = strtok(s, ",")) != 0) { s = 0; if (len_terms + 2 >= max_terms) { max_terms *= 2; all_terms = typeRealloc(char *, max_terms, all_terms); if (all_terms == 0) failed("no memory: all_terms"); } all_terms[len_terms++] = strdup(t); } } } else { all_terms[len_terms++] = strdup(old_term); } all_terms[len_terms] = 0; if (v_opt) { printf("%d term%s:\n", PLURAL(len_terms)); if (v_opt > 3) { for (n = 0; n < len_terms; ++n) { printf(" %d: %s\n", n + 1, all_terms[n]); } } } /* * If no capability name was selected, use the predefined list of string * capabilities. * * TODO: To address the "other" systems which do not follow SVr4, * just use the output from infocmp on $TERM. */ if (len_caps == 0) { #if defined(HAVE_CURSES_DATA_BOOLNAMES) || defined(DECL_CURSES_DATA_BOOLNAMES) for (n = 0; strnames[n] != 0; ++n) { GrowArray(all_caps, max_caps, len_caps); all_caps[len_caps++] = strdup(strnames[n]); } #else all_caps[len_caps++] = strdup("cup"); all_caps[len_caps++] = strdup("sgr"); #endif } std_caps = len_caps; all_caps[len_caps] = 0; if (v_opt) { printf("%d name%s%s\n", PLURAL(len_caps), COLONS(len_caps)); if (v_opt > 3) { for (n = 0; n < len_caps; ++n) { printf(" %d: %s\n", n + 1, all_caps[n]); } } } cap_name = typeMalloc(char *, (max_name = 1 + len_caps)); cap_data = typeMalloc(char *, (max_data = 1 + len_caps)); if (r_opt <= 0) r_opt = 1; if (a_opt) { for (n = 0; n < max_parms; ++n) if (num_parms[n]) use_parms *= (num_parms[n] + 1); } for (r_run = 0; r_run < r_opt; ++r_run) { for (t_run = 0; t_run < len_terms; ++t_run) { int errs; if (setupterm(all_terms[t_run], fileno(stdout), &errs) != OK) { printf("** skipping %s (errs:%d)\n", all_terms[t_run], errs); } #if NCURSES_XNAMES len_caps = std_caps; if (cur_term) { TERMTYPE *term = (TERMTYPE *) cur_term; for (n = STRCOUNT; n < NUM_STRINGS(term); ++n) { GrowArray(all_caps, max_caps, len_caps); GrowArray(cap_name, max_name, len_caps); GrowArray(cap_data, max_data, len_caps); all_caps[len_caps++] = strdup(ExtStrname(term, (int) n, strnames)); } } #else (void) std_caps; #endif /* * Most of the capabilities have no parameters, e.g., they are * function-keys or simple operations such as clear-display. * Ignore those, since they do not really exercise tparm. */ use_caps = 0; for (n = 0; n < len_caps; ++n) { char *value = tigetstr(all_caps[n]); if (relevant(all_caps[n], value)) { cap_name[use_caps] = all_caps[n]; cap_data[use_caps] = value; use_caps++; } } if (v_opt) { printf("[%d:%d] %d paramerized cap%s * %ld test-case%s \"%s\"\n", r_run + 1, r_opt, PLURAL(use_caps), PLURAL(use_parms), all_terms[t_run]); } memset(all_parms, 0, sizeof(all_parms)); if (a_opt) { /* for each combination of values */ do { for (n_run = 0; n_run < use_caps; ++n_run) { test_tparm(cap_name[n_run], cap_data[n_run], all_parms, str_parms); } } while (increment(all_parms, num_parms, len_parms, 0)); } else { /* for the given values */ for (n_run = 0; n_run < use_caps; ++n_run) { test_tparm(cap_name[n_run], cap_data[n_run], all_parms, str_parms); } } #if NCURSES_XNAMES for (n = std_caps; n < len_caps; ++n) { free(all_caps[n]); } #endif if (cur_term != 0) { del_curterm(cur_term); } else { printf("? no cur_term\n"); } } } printf("Tests:\n"); printf(NUMFORM " total\n", total_tests); if (total_fails) printf(NUMFORM " failed\n", total_fails); printf("Characters:\n"); printf(NUMFORM " nulls\n", total_nulls); printf(NUMFORM " controls\n", total_ctrls); printf(NUMFORM " printable\n", total_print); printf(NUMFORM " total\n", total_nulls + total_ctrls + total_print); #if NO_LEAKS for (n = 0; n < std_caps; ++n) { free(all_caps[n]); } free(all_caps); free(old_term); for (n = 0; n < len_terms; ++n) { free(all_terms[n]); } free(all_terms); free(num_parms); free(str_parms); free(cap_name); free(cap_data); #endif ExitProgram(EXIT_SUCCESS); } #else /* !HAVE_TIGETSTR */ int main(void) { failed("This program requires the terminfo functions such as tigetstr"); } #endif /* HAVE_TIGETSTR */