2000-08-06 21:13:45 +00:00
|
|
|
/* $Id$ */
|
2000-06-06 05:53:49 +00:00
|
|
|
/**************************************************************************
|
|
|
|
* utils.c *
|
|
|
|
* *
|
2002-01-04 17:57:40 +00:00
|
|
|
* Copyright (C) 1999-2002 Chris Allegretta *
|
2000-06-06 05:53:49 +00:00
|
|
|
* This program is free software; you can redistribute it and/or modify *
|
|
|
|
* it under the terms of the GNU General Public License as published by *
|
2001-10-24 11:33:54 +00:00
|
|
|
* the Free Software Foundation; either version 2, or (at your option) *
|
2000-06-06 05:53:49 +00:00
|
|
|
* any later version. *
|
|
|
|
* *
|
|
|
|
* This program is distributed in the hope that it will be useful, *
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
|
|
|
|
* GNU General Public License for more details. *
|
|
|
|
* *
|
|
|
|
* You should have received a copy of the GNU General Public License *
|
|
|
|
* along with this program; if not, write to the Free Software *
|
|
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. *
|
|
|
|
* *
|
|
|
|
**************************************************************************/
|
|
|
|
|
2001-04-28 18:03:52 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2000-07-07 04:25:00 +00:00
|
|
|
#include <unistd.h>
|
2000-06-06 05:53:49 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include "nano.h"
|
|
|
|
#include "proto.h"
|
|
|
|
|
2000-06-21 03:00:43 +00:00
|
|
|
#ifndef NANO_SMALL
|
2000-06-19 04:22:15 +00:00
|
|
|
#include <libintl.h>
|
|
|
|
#define _(string) gettext(string)
|
|
|
|
#else
|
|
|
|
#define _(string) (string)
|
|
|
|
#endif
|
|
|
|
|
2002-02-22 04:30:50 +00:00
|
|
|
int num_of_digits(int n)
|
|
|
|
{
|
|
|
|
int i = 1;
|
|
|
|
|
|
|
|
if (n < 0)
|
|
|
|
n = 0 - n;
|
|
|
|
|
|
|
|
while (n > 10) {
|
|
|
|
n /= 10;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2000-06-06 05:53:49 +00:00
|
|
|
/* Lower case a string - must be null terminated */
|
|
|
|
void lowercase(char *src)
|
|
|
|
{
|
|
|
|
long i = 0;
|
|
|
|
|
|
|
|
while (src[i] != 0) {
|
|
|
|
src[i] = (char) tolower(src[i]);
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-16 00:48:53 +00:00
|
|
|
/* None of this is needed if we're using NANO_SMALL! */
|
|
|
|
#ifndef NANO_SMALL
|
2001-06-13 02:35:44 +00:00
|
|
|
char *revstrstr(char *haystack, char *needle, char *rev_start)
|
|
|
|
{
|
|
|
|
char *p, *q, *r;
|
|
|
|
|
|
|
|
for(p = rev_start ; p >= haystack ; --p) {
|
|
|
|
for (r = p, q = needle ; (*q == *r) && (*q != '\0') ; r++, q++)
|
|
|
|
;
|
|
|
|
if (*q == '\0')
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-02-28 00:57:18 +00:00
|
|
|
char *revstristr(char *haystack, char *needle, char *rev_start)
|
2001-06-13 02:35:44 +00:00
|
|
|
{
|
|
|
|
char *p, *q, *r;
|
|
|
|
|
|
|
|
for(p = rev_start ; p >= haystack ; --p) {
|
|
|
|
for (r = p, q = needle ; (tolower(*q) == tolower(*r)) && (*q != '\0') ; r++, q++)
|
|
|
|
;
|
|
|
|
if (*q == '\0')
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2002-03-06 15:27:44 +00:00
|
|
|
#endif /* NANO_SMALL */
|
2000-06-06 05:53:49 +00:00
|
|
|
|
2001-05-16 04:20:57 +00:00
|
|
|
/* This is now mutt's version (called mutt_stristr) because it doesn't
|
2001-05-16 04:23:37 +00:00
|
|
|
use memory allocation to do a simple search (yuck). */
|
2002-02-28 00:57:18 +00:00
|
|
|
char *stristr(char *haystack, char *needle)
|
2000-06-06 05:53:49 +00:00
|
|
|
{
|
2001-05-16 04:20:57 +00:00
|
|
|
const char *p, *q;
|
2000-06-06 05:53:49 +00:00
|
|
|
|
2001-05-16 04:20:57 +00:00
|
|
|
if (!haystack)
|
2000-06-06 05:53:49 +00:00
|
|
|
return NULL;
|
2001-05-16 04:20:57 +00:00
|
|
|
if (!needle)
|
|
|
|
return (haystack);
|
|
|
|
|
2001-06-13 02:35:44 +00:00
|
|
|
while (*(p = haystack)) {
|
2001-05-16 04:20:57 +00:00
|
|
|
for (q = needle; *p && *q && tolower (*p) == tolower (*q); p++, q++)
|
|
|
|
;
|
|
|
|
if (!*q)
|
|
|
|
return (haystack);
|
2001-06-13 02:35:44 +00:00
|
|
|
haystack++;
|
2000-06-06 05:53:49 +00:00
|
|
|
}
|
2001-05-16 04:20:57 +00:00
|
|
|
return NULL;
|
2000-06-06 05:53:49 +00:00
|
|
|
}
|
|
|
|
|
2001-06-13 02:35:44 +00:00
|
|
|
char *strstrwrapper(char *haystack, char *needle, char *rev_start)
|
2000-06-06 05:53:49 +00:00
|
|
|
{
|
2001-06-13 02:35:44 +00:00
|
|
|
|
2000-09-06 13:39:17 +00:00
|
|
|
#ifdef HAVE_REGEX_H
|
2001-06-13 02:35:44 +00:00
|
|
|
int result;
|
|
|
|
|
2000-07-07 01:49:52 +00:00
|
|
|
if (ISSET(USE_REGEXP)) {
|
2001-06-13 02:35:44 +00:00
|
|
|
if (!ISSET(REVERSE_SEARCH)) {
|
|
|
|
result = regexec(&search_regexp, haystack, 10, regmatches, 0);
|
|
|
|
if (!result)
|
|
|
|
return haystack + regmatches[0].rm_so;
|
2002-01-28 15:56:15 +00:00
|
|
|
#ifndef NANO_SMALL
|
2001-06-13 02:35:44 +00:00
|
|
|
} else {
|
2002-01-28 15:56:15 +00:00
|
|
|
char *i, *j;
|
|
|
|
|
2002-02-16 20:03:44 +00:00
|
|
|
/* do a quick search forward first */
|
2001-06-13 02:35:44 +00:00
|
|
|
if (!(regexec(&search_regexp, haystack, 10, regmatches, 0))) {
|
2002-02-16 20:03:44 +00:00
|
|
|
/* there's a match somewhere in the line - now search for it backwards, much slower */
|
2001-06-13 02:35:44 +00:00
|
|
|
for(i = rev_start ; i >= haystack ; --i)
|
|
|
|
if (!(result = regexec(&search_regexp, i, 10, regmatches, 0))) {
|
|
|
|
j = i + regmatches[0].rm_so;
|
|
|
|
if (j <= rev_start)
|
|
|
|
return j;
|
|
|
|
}
|
|
|
|
}
|
2001-07-16 00:48:53 +00:00
|
|
|
#endif
|
2002-01-28 15:56:15 +00:00
|
|
|
}
|
2000-10-26 01:44:42 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2000-07-07 04:25:00 +00:00
|
|
|
#endif
|
2001-07-16 00:48:53 +00:00
|
|
|
#ifndef NANO_SMALL
|
2001-10-02 03:54:40 +00:00
|
|
|
if (ISSET(CASE_SENSITIVE)) {
|
2001-07-16 00:48:53 +00:00
|
|
|
if (ISSET(REVERSE_SEARCH))
|
2001-06-13 02:35:44 +00:00
|
|
|
return revstrstr(haystack, needle, rev_start);
|
2001-07-16 00:48:53 +00:00
|
|
|
else
|
|
|
|
return strstr(haystack,needle);
|
|
|
|
|
2001-06-13 02:35:44 +00:00
|
|
|
} else {
|
2001-07-16 00:48:53 +00:00
|
|
|
if (ISSET(REVERSE_SEARCH))
|
2002-02-28 00:57:18 +00:00
|
|
|
return revstristr(haystack, needle, rev_start);
|
2001-07-16 00:48:53 +00:00
|
|
|
else
|
|
|
|
#endif
|
2002-02-28 00:57:18 +00:00
|
|
|
return stristr(haystack, needle);
|
2001-10-02 03:54:40 +00:00
|
|
|
#ifndef NANO_SMALL
|
2001-06-13 02:35:44 +00:00
|
|
|
}
|
2001-10-02 03:54:40 +00:00
|
|
|
#endif
|
2000-06-06 05:53:49 +00:00
|
|
|
}
|
2000-06-19 04:22:15 +00:00
|
|
|
|
|
|
|
/* Thanks BG, many ppl have been asking for this... */
|
|
|
|
void *nmalloc(size_t howmuch)
|
2000-06-21 03:00:43 +00:00
|
|
|
{
|
2000-06-19 04:22:15 +00:00
|
|
|
void *r;
|
2000-06-21 03:00:43 +00:00
|
|
|
|
2000-06-19 04:22:15 +00:00
|
|
|
/* Panic save? */
|
|
|
|
|
|
|
|
if (!(r = malloc(howmuch)))
|
|
|
|
die(_("nano: malloc: out of memory!"));
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
2000-06-21 03:00:43 +00:00
|
|
|
|
2001-05-17 11:35:43 +00:00
|
|
|
/* We're going to need this too - Hopefully this will minimize
|
|
|
|
the transition cost of moving to the apropriate function. */
|
|
|
|
char *charalloc(size_t howmuch)
|
2001-04-18 04:28:54 +00:00
|
|
|
{
|
|
|
|
void *r;
|
|
|
|
|
|
|
|
/* Panic save? */
|
|
|
|
|
2001-05-18 04:53:45 +00:00
|
|
|
if (!(r = calloc(howmuch, sizeof (char))))
|
2001-04-18 04:28:54 +00:00
|
|
|
die(_("nano: calloc: out of memory!"));
|
|
|
|
|
2001-05-17 11:35:43 +00:00
|
|
|
return (char *) r;
|
2001-04-18 04:28:54 +00:00
|
|
|
}
|
|
|
|
|
2000-06-19 04:22:15 +00:00
|
|
|
void *nrealloc(void *ptr, size_t howmuch)
|
2000-06-21 03:00:43 +00:00
|
|
|
{
|
2000-06-19 04:22:15 +00:00
|
|
|
void *r;
|
2000-06-21 03:00:43 +00:00
|
|
|
|
2000-06-19 04:22:15 +00:00
|
|
|
if (!(r = realloc(ptr, howmuch)))
|
2000-11-01 18:43:21 +00:00
|
|
|
die(_("nano: realloc: out of memory!"));
|
2000-06-19 04:22:15 +00:00
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
2000-07-04 22:15:39 +00:00
|
|
|
|
2001-05-05 17:45:54 +00:00
|
|
|
/* Copy one malloc()ed string to another pointer.
|
2000-11-02 04:40:39 +00:00
|
|
|
|
|
|
|
Should be used as dest = mallocstrcpy(dest, src);
|
|
|
|
*/
|
2001-01-05 21:13:14 +00:00
|
|
|
void *mallocstrcpy(char *dest, char *src)
|
2000-11-02 04:40:39 +00:00
|
|
|
{
|
|
|
|
|
2001-01-07 05:50:36 +00:00
|
|
|
|
|
|
|
if (src == dest)
|
|
|
|
return src;
|
|
|
|
|
2000-11-02 04:40:39 +00:00
|
|
|
if (dest != NULL)
|
|
|
|
free(dest);
|
|
|
|
|
2000-11-16 06:01:10 +00:00
|
|
|
if (src == NULL) {
|
|
|
|
dest = NULL;
|
|
|
|
return(dest);
|
|
|
|
}
|
|
|
|
|
2001-05-17 11:35:43 +00:00
|
|
|
dest = charalloc(strlen(src) + 1);
|
2000-11-02 04:40:39 +00:00
|
|
|
strcpy(dest, src);
|
|
|
|
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-07-04 22:15:39 +00:00
|
|
|
/* Append a new magic-line to filebot */
|
2000-10-26 01:44:42 +00:00
|
|
|
void new_magicline(void)
|
|
|
|
{
|
2000-07-04 22:15:39 +00:00
|
|
|
filebot->next = nmalloc(sizeof(filestruct));
|
2001-05-17 11:35:43 +00:00
|
|
|
filebot->next->data = charalloc(1);
|
2000-07-04 22:15:39 +00:00
|
|
|
filebot->next->data[0] = '\0';
|
|
|
|
filebot->next->prev = filebot;
|
|
|
|
filebot->next->next = NULL;
|
|
|
|
filebot->next->lineno = filebot->lineno + 1;
|
|
|
|
filebot = filebot->next;
|
|
|
|
totlines++;
|
2000-12-10 05:54:27 +00:00
|
|
|
totsize++;
|
2000-07-04 22:15:39 +00:00
|
|
|
}
|
2000-11-05 17:54:41 +00:00
|
|
|
|
2001-01-11 05:30:31 +00:00
|
|
|
#ifndef DISABLE_TABCOMP
|
2000-11-05 17:54:41 +00:00
|
|
|
/*
|
|
|
|
* Routine to see if a text string is matched by a wildcard pattern.
|
|
|
|
* Returns TRUE if the text is matched, or FALSE if it is not matched
|
|
|
|
* or if the pattern is invalid.
|
|
|
|
* * matches zero or more characters
|
|
|
|
* ? matches a single character
|
|
|
|
* [abc] matches 'a', 'b' or 'c'
|
|
|
|
* \c quotes character c
|
|
|
|
* Adapted from code written by Ingo Wilken, and
|
|
|
|
* then taken from sash, Copyright (c) 1999 by David I. Bell
|
|
|
|
* Permission is granted to use, distribute, or modify this source,
|
|
|
|
* provided that this copyright notice remains intact.
|
|
|
|
* Permission to distribute this code under the GPL has been granted.
|
|
|
|
*/
|
|
|
|
int check_wildcard_match(const char *text, const char *pattern)
|
|
|
|
{
|
|
|
|
const char *retryPat;
|
|
|
|
const char *retryText;
|
|
|
|
int ch;
|
|
|
|
int found;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
retryPat = NULL;
|
|
|
|
retryText = NULL;
|
|
|
|
|
|
|
|
while (*text || *pattern) {
|
|
|
|
ch = *pattern++;
|
|
|
|
|
|
|
|
switch (ch) {
|
|
|
|
case '*':
|
|
|
|
retryPat = pattern;
|
|
|
|
retryText = text;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '[':
|
|
|
|
found = FALSE;
|
|
|
|
|
|
|
|
while ((ch = *pattern++) != ']') {
|
|
|
|
if (ch == '\\')
|
|
|
|
ch = *pattern++;
|
|
|
|
|
|
|
|
if (ch == '\0')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (*text == ch)
|
|
|
|
found = TRUE;
|
|
|
|
}
|
|
|
|
len = strlen(text);
|
|
|
|
if (found == FALSE && len != 0) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (found == TRUE) {
|
|
|
|
if (strlen(pattern) == 0 && len == 1) {
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
if (len != 0) {
|
|
|
|
text++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fall into next case */
|
|
|
|
|
|
|
|
case '?':
|
|
|
|
if (*text++ == '\0')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '\\':
|
|
|
|
ch = *pattern++;
|
|
|
|
|
|
|
|
if (ch == '\0')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* fall into next case */
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (*text == ch) {
|
|
|
|
if (*text)
|
|
|
|
text++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*text) {
|
|
|
|
pattern = retryPat;
|
|
|
|
text = ++retryText;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pattern == NULL)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
2000-11-30 02:31:13 +00:00
|
|
|
#endif
|