1 /* utils.c -- various utility routines
3 * This code is Copyright (c) 2006, by the authors of nmh. See the
4 * COPYRIGHT file in the root directory of the nmh distribution for
5 * complete copyright information.
9 #include "read_yes_or_no_if_tty.h"
12 #include "context_foil.h"
13 #include "context_replace.h"
14 #include "context_read.h"
15 #include "context_find.h"
16 #include "print_help.h"
19 #include "h/signals.h"
24 #include "read_line.h"
26 extern char *mhdocdir
;
28 /* plurals gives the letter ess to indicate a plural noun, or an empty
29 * string as plurals+1 for the singular noun. Used by the PLURALS
31 const char plurals
[] = "s";
34 * We allocate space for messages (msgs array)
35 * this number of elements at a time.
39 /* Call malloc(3), exiting on NULL return. */
41 mh_xmalloc(size_t size
)
46 size
= 1; /* Some mallocs don't like 0. */
49 die("malloc failed, size wanted: %zu", size
);
54 /* Call realloc(3), exiting on NULL return. */
56 mh_xrealloc(void *ptr
, size_t size
)
60 /* Copy POSIX behaviour, coping with non-POSIX systems. */
63 return mh_xmalloc(1); /* Get a unique pointer. */
66 return mh_xmalloc(size
);
68 new = realloc(ptr
, size
);
70 die("realloc failed, size wanted: %zu", size
);
75 /* Call calloc(3), exiting on NULL return. */
77 mh_xcalloc(size_t nelem
, size_t elsize
)
81 if (!nelem
|| !elsize
)
82 return mh_xmalloc(1); /* Get a unique pointer. */
84 p
= calloc(nelem
, elsize
);
86 die("calloc failed, size wanted: %zu * %zu", nelem
, elsize
);
91 /* Duplicate a NUL-terminated string, exit on failure. */
93 mh_xstrdup(const char *src
)
98 n
= strlen(src
) + 1; /* Ignore possibility of overflow. */
100 memcpy(dest
, src
, n
);
106 * Return the present working directory, if the current directory does not
107 * exist, or is too long, make / the pwd.
113 static char curwd
[PATH_MAX
];
115 if (!getcwd (curwd
, PATH_MAX
)) {
116 inform("unable to determine working directory, continuing...");
117 if (!mypath
|| !*mypath
118 || (strcpy (curwd
, mypath
), chdir (curwd
)) == -1) {
120 if (chdir (curwd
) < 0) {
121 advise (curwd
, "chdir");
127 if ((cp
= curwd
+ strlen (curwd
) - 1) > curwd
&& *cp
== '/')
133 /* add returns a newly malloc'd string, exiting on failure. The order
134 * of the parameters is unusual. A NULL parameter is treated as an
135 * empty string. s1 is free'd. Use mh_xstrdup(s) rather than add(s,
136 * NULL), with FENDNULL() if s might be NULL.
138 * add(NULL, NULL) -> ""
139 * add(NULL, "foo") -> "foo"
140 * add("bar", NULL) -> "bar"
141 * add("bar", "foo") -> "foobar"
144 add (const char *s2
, char *s1
)
147 size_t len1
= 0, len2
= 0;
154 cp
= mh_xmalloc (len1
+ len2
+ 1);
156 /* Copy s1 and free it */
158 memcpy (cp
, s1
, len1
);
164 memcpy (cp
+ len1
, s2
, len2
);
166 /* Now NULL terminate the string */
167 cp
[len1
+ len2
] = '\0';
174 * Append an item to a comma separated list
177 addlist (char *list
, const char *item
)
180 list
= add (", ", list
);
182 return add (item
, list
);
187 * Check to see if a folder exists.
190 folder_exists(const char *folder
)
194 return stat(folder
, &st
) != -1;
199 * Check to see if a folder exists, if not, prompt the user to create
203 create_folder(char *folder
, int autocreate
, void (*done_callback
)(int))
209 if (stat (folder
, &st
) == -1) {
211 adios (folder
, "error on folder");
212 if (autocreate
== 0) {
213 /* ask before creating folder */
214 cp
= concat ("Create folder \"", folder
, "\"? ", NULL
);
215 if (!read_yes_or_no_if_tty (cp
))
218 } else if (autocreate
== -1) {
219 /* do not create, so exit */
222 if (!makedir (folder
))
223 die("unable to create folder %s", folder
);
229 * Return the number of digits in a nonnegative integer.
238 die("oops, num_digits called with negative value");
252 * Append a message arg to an array of them, resizing it if necessary.
253 * Really a simple vector-of-(char *) maintenance routine.
256 app_msgarg(struct msgs_array
*msgs
, char *cp
)
258 if(msgs
->size
>= msgs
->max
) {
259 msgs
->max
+= MAXMSGS
;
260 msgs
->msgs
= mh_xrealloc(msgs
->msgs
,
261 msgs
->max
* sizeof(*msgs
->msgs
));
263 msgs
->msgs
[msgs
->size
++] = cp
;
267 * Append a message number to an array of them, resizing it if necessary.
268 * Like app_msgarg, but with a vector-of-ints instead.
272 app_msgnum(struct msgnum_array
*msgs
, int msgnum
)
274 if (msgs
->size
>= msgs
->max
) {
275 msgs
->max
+= MAXMSGS
;
276 msgs
->msgnums
= mh_xrealloc(msgs
->msgnums
,
277 msgs
->max
* sizeof(*msgs
->msgnums
));
279 msgs
->msgnums
[msgs
->size
++] = msgnum
;
284 * Finds first occurrence of str in buf. buf is not a C string but a
285 * byte array of length buflen. str is a null-terminated C string.
286 * find_str() does not modify buf but passes back a non-const char *
287 * pointer so that the caller can modify it.
290 find_str (const char buf
[], size_t buflen
, const char *str
)
292 const size_t len
= strlen (str
);
295 for (i
= 0; i
+ len
<= buflen
; ++i
, ++buf
) {
296 if (! memcmp (buf
, str
, len
)) return (char *) buf
;
304 * Finds last occurrence of str in buf. buf is not a C string but a
305 * byte array of length buflen. str is a null-terminated C string.
306 * find_str() does not modify buf but passes back a non-const char *
307 * pointer so that the caller can modify it.
310 rfind_str (const char buf
[], size_t buflen
, const char *str
)
312 const size_t len
= strlen (str
);
315 for (i
= 0, buf
+= buflen
- len
; i
+ len
<= buflen
; ++i
, --buf
) {
316 if (! memcmp (buf
, str
, len
)) return (char *) buf
;
323 /* POSIX doesn't have strcasestr() so emulate it. */
325 nmh_strcasestr (const char *s1
, const char *s2
)
327 const size_t len
= strlen (s2
);
329 if (isupper ((unsigned char) s2
[0]) || islower ((unsigned char)s2
[0])) {
331 first
[0] = (char) toupper ((unsigned char) s2
[0]);
332 first
[1] = (char) tolower ((unsigned char) s2
[0]);
335 for (s1
= strpbrk (s1
, first
); s1
; s1
= strpbrk (++s1
, first
)) {
336 if (! strncasecmp (s1
, s2
, len
)) return (char *) s1
;
339 for (s1
= strchr (s1
, s2
[0]); s1
; s1
= strchr (++s1
, s2
[0])) {
340 if (! strncasecmp (s1
, s2
, len
)) return (char *) s1
;
348 /* truncpy copies at most size - 1 chars from non-NULL src to non-NULL,
349 * non-overlapping, dst, and ensures dst is NUL terminated. If size is
350 * zero then it aborts as dst cannot be NUL terminated.
352 * It's to be used when truncation is intended and correct, e.g.
353 * reporting a possibly very long external string back to the user. One
354 * of its advantages over strncpy(3) is it doesn't pad in the common
355 * case of no truncation. */
357 trunccpy(char *dst
, const char *src
, size_t size
)
360 inform("trunccpy: zero-length destination: \"%.20s\"",
365 if (strnlen(src
, size
) < size
) {
368 memcpy(dst
, src
, size
- 1);
369 dst
[size
- 1] = '\0';
374 /* has_prefix returns true if non-NULL s starts with non-NULL prefix. */
376 has_prefix(const char *s
, const char *prefix
)
378 while (*s
&& *s
== *prefix
) {
383 return *prefix
== '\0';
387 /* has_suffix returns true if non-NULL s ends with non-NULL suffix. */
389 has_suffix(const char *s
, const char *suffix
)
394 lsuf
= strlen(suffix
);
396 return lsuf
<= ls
&& !strcmp(s
+ ls
- lsuf
, suffix
);
400 /* has_suffix_c returns true if non-NULL string s ends with a c before the
401 * terminating NUL. */
403 has_suffix_c(const char *s
, int c
)
405 return *s
&& s
[strlen(s
) - 1] == c
;
409 /* trim_suffix_c deletes c from the end of non-NULL string s if it's
410 * present, shortening s by 1. Only one instance of c is removed. */
412 trim_suffix_c(char *s
, int c
)
423 /* to_lower runs all of s through tolower(3). */
429 for (b
= (unsigned char *)s
; (*b
= tolower(*b
)); b
++)
434 /* to_upper runs all of s through toupper(3). */
440 for (b
= (unsigned char *)s
; (*b
= toupper(*b
)); b
++)
446 nmh_init(const char *argv0
, bool read_context
, bool check_version
)
451 invo_name
= r1bindex ((char *) argv0
, '/');
453 if (setup_signal_handlers()) {
454 admonish("sigaction", "unable to set up signal handlers");
457 /* POSIX atexit() does not define any error conditions. */
458 if (atexit(remove_registered_files_atexit
)) {
459 admonish("atexit", "unable to register atexit function");
462 /* Read context, if supposed to. */
468 bool allow_version_check
= true;
469 bool check_older_version
= false;
470 if (!check_version
||
471 ((cp
= context_find ("Welcome")) && strcasecmp (cp
, "disable") == 0)) {
472 allow_version_check
= false;
473 } else if ((cp
= getenv ("MHCONTEXT")) != NULL
&& *cp
!= '\0') {
474 /* Context file comes from $MHCONTEXT, so only print the message
475 if the context file has an older version. If it does, or if it
476 doesn't have a version at all, update the version. */
477 check_older_version
= true;
480 /* Check to see if the user is running a different (or older, if
481 specified) version of nmh than they had run before, and notify them
483 if (allow_version_check
&& isatty (fileno (stdin
)) &&
484 isatty (fileno (stdout
)) && isatty (fileno (stderr
))) {
485 if (nmh_version_changed (check_older_version
)) {
486 printf ("==================================================="
487 "=====================\n");
488 printf ("Welcome to nmh version %s\n\n", VERSION
);
489 printf ("See the release notes in %s/NEWS\n\n",
491 print_intro (stdout
, 1);
492 printf ("\nThis message will not be repeated until "
493 "nmh is next updated.\n");
494 printf ("==================================================="
495 "=====================\n\n");
497 fputs ("Press enter to continue: ", stdout
);
503 if ((status
= context_foil(NULL
)) != OK
) {
504 advise("", "failed to create minimal profile/context");
508 /* Allow the user to set a locale in their profile. Otherwise, use the
509 "" string to pull it from their environment, see setlocale(3). */
510 if ((locale
= context_find ("locale")) == NULL
) {
514 if (! setlocale (LC_ALL
, locale
)) {
515 inform("setlocale failed, check your LC_ALL, LC_CTYPE, and LANG "
516 "environment variables, continuing...");
524 * Check stored version, and return 1 if out-of-date or non-existent.
525 * Because the output of "mhparam version" is prefixed with "nmh-",
526 * use that prefix here.
529 nmh_version_changed (int older
)
531 const char *const context_version
= context_find("Version");
534 /* Convert the version strings to floats and compare them. This will
535 break for versions with multiple decimal points, etc. */
536 const float current_version
= strtof (VERSION
, NULL
);
537 const float old_version
=
538 context_version
&& has_prefix(context_version
, "nmh-")
539 ? strtof (context_version
+ 4, NULL
)
542 if (context_version
== NULL
|| old_version
< current_version
) {
543 context_replace ("Version", "nmh-" VERSION
);
546 return old_version
< current_version
;
549 if (context_version
== NULL
|| strcmp(context_version
, "nmh-" VERSION
) != 0) {
550 context_replace ("Version", "nmh-" VERSION
);
558 /* contains8bit returns true if any byte from start onwards fails
559 * isascii(3), i.e. is outside [0, 0x7f]. If start is NULL it returns
560 * false. Bytes are examined until a NUL byte, or, if end is not NULL,
561 * whilst start is before end. */
563 contains8bit(const char *start
, const char *end
)
573 while (p
< end
&& (c
= (*p
++)))
574 if (!isascii((unsigned char)c
))
578 if (!isascii((unsigned char)c
))
587 * See if input has any 8-bit bytes.
590 scan_input (int fd
, int *eightbit
)
596 lseek(fd
, 0, SEEK_SET
);
598 while ((state
= read (fd
, buf
, sizeof buf
)) > 0) {
599 if (contains8bit (buf
, buf
+ state
)) {
605 return state
== NOTOK
? NOTOK
: OK
;
610 * Convert an int to a char string.
615 return m_strn(value
, 0);
620 * Convert an int to a char string, of limited width if > 0.
623 /* SIZE(n) includes NUL. n must just be digits, not an equation. */
624 #define SIZE(n) (sizeof STR(n))
627 m_strn(int value
, unsigned int width
)
629 /* Need to include space for negative sign. But don't use INT_MIN
630 because it could be a macro that would fool SIZE(n). */
631 static char buffer
[SIZE(-INT_MAX
)];
632 const int num_chars
= snprintf(buffer
, sizeof buffer
, "%d", value
);
634 return num_chars
> 0 && (width
== 0 || (unsigned int) num_chars
<= width
)