mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-10 14:02:43 +00:00
cfeecda0a9
Before = dpv: <__func__>: posix_spawnp(3): No such file or directory After = dpv: <path/cmd>: No such file or directory Most notably, show the 2nd argument being passed to posix_spawnp(3) so we know what path/cmd failed. Also, we don't need to have "posix_spawnp(3)" in the error message nor the function because that can [a] change and [b] traversed using a debugger if necessary.
640 lines
18 KiB
C
640 lines
18 KiB
C
/*-
|
|
* Copyright (c) 2013-2018 Devin Teske <dteske@FreeBSD.org>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <ctype.h>
|
|
#include <err.h>
|
|
#include <fcntl.h>
|
|
#include <limits.h>
|
|
#include <spawn.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
|
|
#include "dialog_util.h"
|
|
#include "dpv.h"
|
|
#include "dpv_private.h"
|
|
|
|
extern char **environ;
|
|
|
|
#define TTY_DEFAULT_ROWS 24
|
|
#define TTY_DEFAULT_COLS 80
|
|
|
|
/* [X]dialog(1) characteristics */
|
|
uint8_t dialog_test = 0;
|
|
uint8_t use_dialog = 0;
|
|
uint8_t use_libdialog = 1;
|
|
uint8_t use_xdialog = 0;
|
|
uint8_t use_color = 1;
|
|
char dialog[PATH_MAX] = DIALOG;
|
|
|
|
/* [X]dialog(1) functionality */
|
|
char *title = NULL;
|
|
char *backtitle = NULL;
|
|
int dheight = 0;
|
|
int dwidth = 0;
|
|
static char *dargv[64] = { NULL };
|
|
|
|
/* TTY/Screen characteristics */
|
|
static struct winsize *maxsize = NULL;
|
|
|
|
/* Function prototypes */
|
|
static void tty_maxsize_update(void);
|
|
static void x11_maxsize_update(void);
|
|
|
|
/*
|
|
* Update row/column fields of `maxsize' global (used by dialog_maxrows() and
|
|
* dialog_maxcols()). If the `maxsize' pointer is NULL, it will be initialized.
|
|
* The `ws_row' and `ws_col' fields of `maxsize' are updated to hold current
|
|
* maximum height and width (respectively) for a dialog(1) widget based on the
|
|
* active TTY size.
|
|
*
|
|
* This function is called automatically by dialog_maxrows/cols() to reflect
|
|
* changes in terminal size in-between calls.
|
|
*/
|
|
static void
|
|
tty_maxsize_update(void)
|
|
{
|
|
int fd = STDIN_FILENO;
|
|
struct termios t;
|
|
|
|
if (maxsize == NULL) {
|
|
if ((maxsize = malloc(sizeof(struct winsize))) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
memset((void *)maxsize, '\0', sizeof(struct winsize));
|
|
}
|
|
|
|
if (!isatty(fd))
|
|
fd = open("/dev/tty", O_RDONLY);
|
|
if ((tcgetattr(fd, &t) < 0) || (ioctl(fd, TIOCGWINSZ, maxsize) < 0)) {
|
|
maxsize->ws_row = TTY_DEFAULT_ROWS;
|
|
maxsize->ws_col = TTY_DEFAULT_COLS;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Update row/column fields of `maxsize' global (used by dialog_maxrows() and
|
|
* dialog_maxcols()). If the `maxsize' pointer is NULL, it will be initialized.
|
|
* The `ws_row' and `ws_col' fields of `maxsize' are updated to hold current
|
|
* maximum height and width (respectively) for an Xdialog(1) widget based on
|
|
* the active video resolution of the X11 environment.
|
|
*
|
|
* This function is called automatically by dialog_maxrows/cols() to initialize
|
|
* `maxsize'. Since video resolution changes are less common and more obtrusive
|
|
* than changes to terminal size, the dialog_maxrows/cols() functions only call
|
|
* this function when `maxsize' is set to NULL.
|
|
*/
|
|
static void
|
|
x11_maxsize_update(void)
|
|
{
|
|
FILE *f = NULL;
|
|
char *cols;
|
|
char *cp;
|
|
char *rows;
|
|
char cmdbuf[LINE_MAX];
|
|
char rbuf[LINE_MAX];
|
|
|
|
if (maxsize == NULL) {
|
|
if ((maxsize = malloc(sizeof(struct winsize))) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
memset((void *)maxsize, '\0', sizeof(struct winsize));
|
|
}
|
|
|
|
/* Assemble the command necessary to get X11 sizes */
|
|
snprintf(cmdbuf, LINE_MAX, "%s --print-maxsize 2>&1", dialog);
|
|
|
|
fflush(STDIN_FILENO); /* prevent popen(3) from seeking on stdin */
|
|
|
|
if ((f = popen(cmdbuf, "r")) == NULL) {
|
|
if (debug)
|
|
warnx("WARNING! Command `%s' failed", cmdbuf);
|
|
return;
|
|
}
|
|
|
|
/* Read in the line returned from Xdialog(1) */
|
|
if ((fgets(rbuf, LINE_MAX, f) == NULL) || (pclose(f) < 0))
|
|
return;
|
|
|
|
/* Check for X11-related errors */
|
|
if (strncmp(rbuf, "Xdialog: Error", 14) == 0)
|
|
return;
|
|
|
|
/* Parse expected output: MaxSize: YY, XXX */
|
|
if ((rows = strchr(rbuf, ' ')) == NULL)
|
|
return;
|
|
if ((cols = strchr(rows, ',')) != NULL) {
|
|
/* strtonum(3) doesn't like trailing junk */
|
|
*(cols++) = '\0';
|
|
if ((cp = strchr(cols, '\n')) != NULL)
|
|
*cp = '\0';
|
|
}
|
|
|
|
/* Convert to unsigned short */
|
|
maxsize->ws_row = (unsigned short)strtonum(
|
|
rows, 0, USHRT_MAX, (const char **)NULL);
|
|
maxsize->ws_col = (unsigned short)strtonum(
|
|
cols, 0, USHRT_MAX, (const char **)NULL);
|
|
}
|
|
|
|
/*
|
|
* Return the current maximum height (rows) for an [X]dialog(1) widget.
|
|
*/
|
|
int
|
|
dialog_maxrows(void)
|
|
{
|
|
|
|
if (use_xdialog && maxsize == NULL)
|
|
x11_maxsize_update(); /* initialize maxsize for GUI */
|
|
else if (!use_xdialog)
|
|
tty_maxsize_update(); /* update maxsize for TTY */
|
|
return (maxsize->ws_row);
|
|
}
|
|
|
|
/*
|
|
* Return the current maximum width (cols) for an [X]dialog(1) widget.
|
|
*/
|
|
int
|
|
dialog_maxcols(void)
|
|
{
|
|
|
|
if (use_xdialog && maxsize == NULL)
|
|
x11_maxsize_update(); /* initialize maxsize for GUI */
|
|
else if (!use_xdialog)
|
|
tty_maxsize_update(); /* update maxsize for TTY */
|
|
|
|
if (use_dialog || use_libdialog) {
|
|
if (use_shadow)
|
|
return (maxsize->ws_col - 2);
|
|
else
|
|
return (maxsize->ws_col);
|
|
} else
|
|
return (maxsize->ws_col);
|
|
}
|
|
|
|
/*
|
|
* Return the current maximum width (cols) for the terminal.
|
|
*/
|
|
int
|
|
tty_maxcols(void)
|
|
{
|
|
|
|
if (use_xdialog && maxsize == NULL)
|
|
x11_maxsize_update(); /* initialize maxsize for GUI */
|
|
else if (!use_xdialog)
|
|
tty_maxsize_update(); /* update maxsize for TTY */
|
|
|
|
return (maxsize->ws_col);
|
|
}
|
|
|
|
/*
|
|
* Spawn an [X]dialog(1) `--gauge' box with a `--prompt' value of init_prompt.
|
|
* Writes the resulting process ID to the pid_t pointed at by `pid'. Returns a
|
|
* file descriptor (int) suitable for writing data to the [X]dialog(1) instance
|
|
* (data written to the file descriptor is seen as standard-in by the spawned
|
|
* [X]dialog(1) process).
|
|
*/
|
|
int
|
|
dialog_spawn_gauge(char *init_prompt, pid_t *pid)
|
|
{
|
|
char dummy_init[2] = "";
|
|
char *cp;
|
|
int height;
|
|
int width;
|
|
int error;
|
|
posix_spawn_file_actions_t action;
|
|
#if DIALOG_SPAWN_DEBUG
|
|
unsigned int i;
|
|
#endif
|
|
unsigned int n = 0;
|
|
int stdin_pipe[2] = { -1, -1 };
|
|
|
|
/* Override `dialog' with a path from ENV_DIALOG if provided */
|
|
if ((cp = getenv(ENV_DIALOG)) != NULL)
|
|
snprintf(dialog, PATH_MAX, "%s", cp);
|
|
|
|
/* For Xdialog(1), set ENV_XDIALOG_HIGH_DIALOG_COMPAT */
|
|
setenv(ENV_XDIALOG_HIGH_DIALOG_COMPAT, "1", 1);
|
|
|
|
/* Constrain the height/width */
|
|
height = dialog_maxrows();
|
|
if (backtitle != NULL)
|
|
height -= use_shadow ? 5 : 4;
|
|
if (dheight < height)
|
|
height = dheight;
|
|
width = dialog_maxcols();
|
|
if (dwidth < width)
|
|
width = dwidth;
|
|
|
|
/* Populate argument array */
|
|
dargv[n++] = dialog;
|
|
if (title != NULL) {
|
|
if ((dargv[n] = malloc(8)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--title");
|
|
dargv[n++] = title;
|
|
} else {
|
|
if ((dargv[n] = malloc(8)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--title");
|
|
if ((dargv[n] = malloc(1)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
*dargv[n++] = '\0';
|
|
}
|
|
if (backtitle != NULL) {
|
|
if ((dargv[n] = malloc(12)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--backtitle");
|
|
dargv[n++] = backtitle;
|
|
}
|
|
if (use_color) {
|
|
if ((dargv[n] = malloc(11)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--colors");
|
|
}
|
|
if (use_xdialog) {
|
|
if ((dargv[n] = malloc(7)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--left");
|
|
|
|
/*
|
|
* NOTE: Xdialog(1)'s `--wrap' appears to be broken for the
|
|
* `--gauge' widget prompt-updates. Add it anyway (in-case it
|
|
* gets fixed in some later release).
|
|
*/
|
|
if ((dargv[n] = malloc(7)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--wrap");
|
|
}
|
|
if ((dargv[n] = malloc(8)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
sprintf(dargv[n++], "--gauge");
|
|
dargv[n++] = use_xdialog ? dummy_init : init_prompt;
|
|
if ((dargv[n] = malloc(40)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
snprintf(dargv[n++], 40, "%u", height);
|
|
if ((dargv[n] = malloc(40)) == NULL)
|
|
errx(EXIT_FAILURE, "Out of memory?!");
|
|
snprintf(dargv[n++], 40, "%u", width);
|
|
dargv[n] = NULL;
|
|
|
|
/* Open a pipe(2) to communicate with [X]dialog(1) */
|
|
if (pipe(stdin_pipe) < 0)
|
|
err(EXIT_FAILURE, "%s: pipe(2)", __func__);
|
|
|
|
/* Fork [X]dialog(1) process */
|
|
#if DIALOG_SPAWN_DEBUG
|
|
fprintf(stderr, "%s: spawning `", __func__);
|
|
for (i = 0; i < n; i++) {
|
|
if (i == 0)
|
|
fprintf(stderr, "%s", dargv[i]);
|
|
else if (*dargv[i] == '-' && *(dargv[i] + 1) == '-')
|
|
fprintf(stderr, " %s", dargv[i]);
|
|
else
|
|
fprintf(stderr, " \"%s\"", dargv[i]);
|
|
}
|
|
fprintf(stderr, "'\n");
|
|
#endif
|
|
posix_spawn_file_actions_init(&action);
|
|
posix_spawn_file_actions_adddup2(&action, stdin_pipe[0], STDIN_FILENO);
|
|
posix_spawn_file_actions_addclose(&action, stdin_pipe[1]);
|
|
error = posix_spawnp(pid, dialog, &action,
|
|
(const posix_spawnattr_t *)NULL, dargv, environ);
|
|
if (error != 0) err(EXIT_FAILURE, "%s", dialog);
|
|
|
|
/* NB: Do not free(3) *dargv[], else SIGSEGV */
|
|
|
|
return (stdin_pipe[1]);
|
|
}
|
|
|
|
/*
|
|
* Returns the number of lines in buffer pointed to by `prompt'. Takes both
|
|
* newlines and escaped-newlines into account.
|
|
*/
|
|
unsigned int
|
|
dialog_prompt_numlines(const char *prompt, uint8_t nlstate)
|
|
{
|
|
uint8_t nls = nlstate; /* See dialog_prompt_nlstate() */
|
|
const char *cp = prompt;
|
|
unsigned int nlines = 1;
|
|
|
|
if (prompt == NULL || *prompt == '\0')
|
|
return (0);
|
|
|
|
while (*cp != '\0') {
|
|
if (use_dialog) {
|
|
if (strncmp(cp, "\\n", 2) == 0) {
|
|
cp++;
|
|
nlines++;
|
|
nls = TRUE; /* See declaration comment */
|
|
} else if (*cp == '\n') {
|
|
if (!nls)
|
|
nlines++;
|
|
nls = FALSE; /* See declaration comment */
|
|
}
|
|
} else if (use_libdialog) {
|
|
if (*cp == '\n')
|
|
nlines++;
|
|
} else if (strncmp(cp, "\\n", 2) == 0) {
|
|
cp++;
|
|
nlines++;
|
|
}
|
|
cp++;
|
|
}
|
|
|
|
return (nlines);
|
|
}
|
|
|
|
/*
|
|
* Returns the length in bytes of the longest line in buffer pointed to by
|
|
* `prompt'. Takes newlines and escaped newlines into account. Also discounts
|
|
* dialog(1) color escape codes if enabled (via `use_color' global).
|
|
*/
|
|
unsigned int
|
|
dialog_prompt_longestline(const char *prompt, uint8_t nlstate)
|
|
{
|
|
uint8_t backslash = 0;
|
|
uint8_t nls = nlstate; /* See dialog_prompt_nlstate() */
|
|
const char *p = prompt;
|
|
int longest = 0;
|
|
int n = 0;
|
|
|
|
/* `prompt' parameter is required */
|
|
if (prompt == NULL)
|
|
return (0);
|
|
if (*prompt == '\0')
|
|
return (0); /* shortcut */
|
|
|
|
/* Loop until the end of the string */
|
|
while (*p != '\0') {
|
|
/* dialog(1) and dialog(3) will render literal newlines */
|
|
if (use_dialog || use_libdialog) {
|
|
if (*p == '\n') {
|
|
if (!use_libdialog && nls)
|
|
n++;
|
|
else {
|
|
if (n > longest)
|
|
longest = n;
|
|
n = 0;
|
|
}
|
|
nls = FALSE; /* See declaration comment */
|
|
p++;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* Check for backslash character */
|
|
if (*p == '\\') {
|
|
/* If second backslash, count as a single-char */
|
|
if ((backslash ^= 1) == 0)
|
|
n++;
|
|
} else if (backslash) {
|
|
if (*p == 'n' && !use_libdialog) { /* new line */
|
|
/* NB: dialog(3) ignores escaped newlines */
|
|
nls = TRUE; /* See declaration comment */
|
|
if (n > longest)
|
|
longest = n;
|
|
n = 0;
|
|
} else if (use_color && *p == 'Z') {
|
|
if (*++p != '\0')
|
|
p++;
|
|
backslash = 0;
|
|
continue;
|
|
} else /* [X]dialog(1)/dialog(3) only expand those */
|
|
n += 2;
|
|
|
|
backslash = 0;
|
|
} else
|
|
n++;
|
|
p++;
|
|
}
|
|
if (n > longest)
|
|
longest = n;
|
|
|
|
return (longest);
|
|
}
|
|
|
|
/*
|
|
* Returns a pointer to the last line in buffer pointed to by `prompt'. Takes
|
|
* both newlines (if using dialog(1) versus Xdialog(1)) and escaped newlines
|
|
* into account. If no newlines (escaped or otherwise) appear in the buffer,
|
|
* `prompt' is returned. If passed a NULL pointer, returns NULL.
|
|
*/
|
|
char *
|
|
dialog_prompt_lastline(char *prompt, uint8_t nlstate)
|
|
{
|
|
uint8_t nls = nlstate; /* See dialog_prompt_nlstate() */
|
|
char *lastline;
|
|
char *p;
|
|
|
|
if (prompt == NULL)
|
|
return (NULL);
|
|
if (*prompt == '\0')
|
|
return (prompt); /* shortcut */
|
|
|
|
lastline = p = prompt;
|
|
while (*p != '\0') {
|
|
/* dialog(1) and dialog(3) will render literal newlines */
|
|
if (use_dialog || use_libdialog) {
|
|
if (*p == '\n') {
|
|
if (use_libdialog || !nls)
|
|
lastline = p + 1;
|
|
nls = FALSE; /* See declaration comment */
|
|
}
|
|
}
|
|
/* dialog(3) does not expand escaped newlines */
|
|
if (use_libdialog) {
|
|
p++;
|
|
continue;
|
|
}
|
|
if (*p == '\\' && *(p + 1) != '\0' && *(++p) == 'n') {
|
|
nls = TRUE; /* See declaration comment */
|
|
lastline = p + 1;
|
|
}
|
|
p++;
|
|
}
|
|
|
|
return (lastline);
|
|
}
|
|
|
|
/*
|
|
* Returns the number of extra lines generated by wrapping the text in buffer
|
|
* pointed to by `prompt' within `ncols' columns (for prompts, this should be
|
|
* dwidth - 4). Also discounts dialog(1) color escape codes if enabled (via
|
|
* `use_color' global).
|
|
*/
|
|
int
|
|
dialog_prompt_wrappedlines(char *prompt, int ncols, uint8_t nlstate)
|
|
{
|
|
uint8_t backslash = 0;
|
|
uint8_t nls = nlstate; /* See dialog_prompt_nlstate() */
|
|
char *cp;
|
|
char *p = prompt;
|
|
int n = 0;
|
|
int wlines = 0;
|
|
|
|
/* `prompt' parameter is required */
|
|
if (p == NULL)
|
|
return (0);
|
|
if (*p == '\0')
|
|
return (0); /* shortcut */
|
|
|
|
/* Loop until the end of the string */
|
|
while (*p != '\0') {
|
|
/* dialog(1) and dialog(3) will render literal newlines */
|
|
if (use_dialog || use_libdialog) {
|
|
if (*p == '\n') {
|
|
if (use_dialog || !nls)
|
|
n = 0;
|
|
nls = FALSE; /* See declaration comment */
|
|
}
|
|
}
|
|
|
|
/* Check for backslash character */
|
|
if (*p == '\\') {
|
|
/* If second backslash, count as a single-char */
|
|
if ((backslash ^= 1) == 0)
|
|
n++;
|
|
} else if (backslash) {
|
|
if (*p == 'n' && !use_libdialog) { /* new line */
|
|
/* NB: dialog(3) ignores escaped newlines */
|
|
nls = TRUE; /* See declaration comment */
|
|
n = 0;
|
|
} else if (use_color && *p == 'Z') {
|
|
if (*++p != '\0')
|
|
p++;
|
|
backslash = 0;
|
|
continue;
|
|
} else /* [X]dialog(1)/dialog(3) only expand those */
|
|
n += 2;
|
|
|
|
backslash = 0;
|
|
} else
|
|
n++;
|
|
|
|
/* Did we pass the width barrier? */
|
|
if (n > ncols) {
|
|
/*
|
|
* Work backward to find the first whitespace on-which
|
|
* dialog(1) will wrap the line (but don't go before
|
|
* the start of this line).
|
|
*/
|
|
cp = p;
|
|
while (n > 1 && !isspace(*cp)) {
|
|
cp--;
|
|
n--;
|
|
}
|
|
if (n > 0 && isspace(*cp))
|
|
p = cp;
|
|
wlines++;
|
|
n = 1;
|
|
}
|
|
|
|
p++;
|
|
}
|
|
|
|
return (wlines);
|
|
}
|
|
|
|
/*
|
|
* Returns zero if the buffer pointed to by `prompt' contains an escaped
|
|
* newline but only if appearing after any/all literal newlines. This is
|
|
* specific to dialog(1) and does not apply to Xdialog(1).
|
|
*
|
|
* As an attempt to make shell scripts easier to read, dialog(1) will "eat"
|
|
* the first literal newline after an escaped newline. This however has a bug
|
|
* in its implementation in that rather than allowing `\\n\n' to be treated
|
|
* similar to `\\n' or `\n', dialog(1) expands the `\\n' and then translates
|
|
* the following literal newline (with or without characters between [!]) into
|
|
* a single space.
|
|
*
|
|
* If you want to be compatible with Xdialog(1), it is suggested that you not
|
|
* use literal newlines (they aren't supported); but if you have to use them,
|
|
* go right ahead. But be forewarned... if you set $DIALOG in your environment
|
|
* to something other than `cdialog' (our current dialog(1)), then it should
|
|
* do the same thing w/respect to how to handle a literal newline after an
|
|
* escaped newline (you could do no wrong by translating every literal newline
|
|
* into a space but only when you've previously encountered an escaped one;
|
|
* this is what dialog(1) is doing).
|
|
*
|
|
* The ``newline state'' (or nlstate for short; as I'm calling it) is helpful
|
|
* if you plan to combine multiple strings into a single prompt text. In lead-
|
|
* up to this procedure, a common task is to calculate and utilize the widths
|
|
* and heights of each piece of prompt text to later be combined. However, if
|
|
* (for example) the first string ends in a positive newline state (has an
|
|
* escaped newline without trailing literal), the first literal newline in the
|
|
* second string will be mangled.
|
|
*
|
|
* The return value of this function should be used as the `nlstate' argument
|
|
* to dialog_*() functions that require it to allow accurate calculations in
|
|
* the event such information is needed.
|
|
*/
|
|
uint8_t
|
|
dialog_prompt_nlstate(const char *prompt)
|
|
{
|
|
const char *cp;
|
|
|
|
if (prompt == NULL)
|
|
return 0;
|
|
|
|
/*
|
|
* Work our way backward from the end of the string for efficiency.
|
|
*/
|
|
cp = prompt + strlen(prompt);
|
|
while (--cp >= prompt) {
|
|
/*
|
|
* If we get to a literal newline first, this prompt ends in a
|
|
* clean state for rendering with dialog(1). Otherwise, if we
|
|
* get to an escaped newline first, this prompt ends in an un-
|
|
* clean state (following literal will be mangled; see above).
|
|
*/
|
|
if (*cp == '\n')
|
|
return (0);
|
|
else if (*cp == 'n' && --cp > prompt && *cp == '\\')
|
|
return (1);
|
|
}
|
|
|
|
return (0); /* no newlines (escaped or otherwise) */
|
|
}
|
|
|
|
/*
|
|
* Free allocated items initialized by tty_maxsize_update() and
|
|
* x11_maxsize_update()
|
|
*/
|
|
void
|
|
dialog_maxsize_free(void)
|
|
{
|
|
if (maxsize != NULL) {
|
|
free(maxsize);
|
|
maxsize = NULL;
|
|
}
|
|
}
|