utils.c 20 KB
Newer Older
Chris Allegretta's avatar
Chris Allegretta committed
1
/* $Id$ */
Chris Allegretta's avatar
Chris Allegretta committed
2
3
4
/**************************************************************************
 *   utils.c                                                              *
 *                                                                        *
5
 *   Copyright (C) 1999-2005 Chris Allegretta                             *
Chris Allegretta's avatar
Chris Allegretta committed
6
7
 *   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 *
8
 *   the Free Software Foundation; either version 2, or (at your option)  *
Chris Allegretta's avatar
Chris Allegretta committed
9
10
11
12
13
14
15
16
17
18
19
20
21
 *   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.            *
 *                                                                        *
 **************************************************************************/

22
23
24
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
25

Chris Allegretta's avatar
Chris Allegretta committed
26
27
#include <stdlib.h>
#include <string.h>
28
29
#include <stdio.h>
#include <unistd.h>
Chris Allegretta's avatar
Chris Allegretta committed
30
#include <ctype.h>
31
#include <errno.h>
Chris Allegretta's avatar
Chris Allegretta committed
32
#include <assert.h>
Chris Allegretta's avatar
Chris Allegretta committed
33
#include "proto.h"
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
34
#include "nano.h"
Chris Allegretta's avatar
Chris Allegretta committed
35

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
36
#if defined(HAVE_WCHAR_H) && defined(NANO_WIDE)
37
38
39
#include <wchar.h>
#endif

40
#ifdef HAVE_REGEX_H
41
42
43
44
45
46
47
48
#ifdef BROKEN_REGEXEC
int regexec_safe(const regex_t *preg, const char *string, size_t nmatch,
	regmatch_t pmatch[], int eflags)
{
    if (string != NULL && *string != '\0')
	return regexec(preg, string, nmatch, pmatch, eflags);
    return REG_NOMATCH;
}
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
49
#endif
50
51
52

int regexp_bol_or_eol(const regex_t *preg, const char *string)
{
53
54
    return (regexec(preg, string, 0, NULL, 0) == 0 &&
	regexec(preg, string, 0, NULL, REG_NOTBOL | REG_NOTEOL) ==
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
55
	REG_NOMATCH);
56
57
}
#endif /* HAVE_REGEX_H */
58

59
60
61
62
#ifndef HAVE_ISBLANK
/* This function is equivalent to isblank(). */
int is_blank_char(int c)
{
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
63
    return isspace(c) && (!is_cntrl_char(c) || c == '\t');
64
65
66
67
68
}
#endif

/* This function is equivalent to iscntrl(), except in that it also
 * handles control characters with their high bits set. */
Chris Allegretta's avatar
Chris Allegretta committed
69
70
int is_cntrl_char(int c)
{
71
    return (-128 <= c && c < -96) || (0 <= c && c < 32) ||
72
	(127 <= c && c < 160);
Chris Allegretta's avatar
Chris Allegretta committed
73
74
}

75
76
77
78
79
80
81
/* Return TRUE if the character c is in byte range, and FALSE
 * otherwise. */
bool is_byte_char(int c)
{
    return (unsigned int)c == (unsigned char)c;
}

82
83
84
85
86
int num_of_digits(int n)
{
    int i = 1;

    if (n < 0)
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
87
	n = -n;
88
89
90
91
92
93
94
95
96

    while (n > 10) {
	n /= 10;
	i++;
    }

    return i;
}

97
98
99
100
101
102
103
104
105
106
107
108
109
/* c is a control character.  It displays as ^@, ^?, or ^[ch] where ch
 * is c + 64.  We return that character. */
unsigned char control_rep(unsigned char c)
{
    /* Treat newlines embedded in a line as encoded nulls. */
    if (c == '\n')
	return '@';
    else if (c == NANO_CONTROL_8)
	return '?';
    else
	return c + 64;
}

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
110
111
112
/* Read a ssize_t from str, and store it in *val (if val is not NULL).
 * On error, we return FALSE and don't change *val.  Otherwise, we
 * return TRUE. */
113
bool parse_num(const char *str, ssize_t *val)
114
115
116
117
118
{
    char *first_error;
    ssize_t j;

    assert(str != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
119

120
    j = (ssize_t)strtol(str, &first_error, 10);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
121

122
    if (errno == ERANGE || *str == '\0' || *first_error != '\0')
123
	return FALSE;
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
124

125
126
    if (val != NULL)
	*val = j;
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
127

128
    return TRUE;
129
130
}

131
132
133
134
135
136
137
/* Parse a multibyte character from buf.  Return the number of bytes
 * used.  If chr isn't NULL, store the wide character in it.  If
 * bad_chr isn't NULL, set it to TRUE if we have a null byte or a bad
 * multibyte character.  If col isn't NULL, store the new display width
 * in it.  If *str is '\t', we expect col to have the current display
 * width. */
int parse_char(const char *buf, int *chr
138
#ifdef NANO_WIDE
139
	, bool *bad_chr
140
#endif
141
	, size_t *col)
142
{
143
    int wide_buf, mb_buf_len;
144

145
    assert(buf != NULL);
146
147

#ifdef NANO_WIDE
148
149
    if (bad_chr != NULL)
	*bad_chr = FALSE;
150
151
152
153
154
155

    if (!ISSET(NO_UTF8)) {
	wchar_t tmp;

	/* Get the wide character equivalent of the multibyte
	 * character. */
156
157
158
159
160
161
162
163
164
165
166
	mb_buf_len = mbtowc(&tmp, buf, MB_CUR_MAX);
	wide_buf = (int)tmp;

	/* If buf contains a null byte or an invalid multibyte
	 * character, interpret buf's first byte as a single-byte
	 * sequence and set bad_chr to TRUE. */
	if (mb_buf_len <= 0) {
	    mb_buf_len = 1;
	    wide_buf = (unsigned char)*buf;
	    if (bad_chr != NULL)
		*bad_chr = TRUE;
167
168
169
170
	}

	/* Save the wide character in chr. */
	if (chr != NULL)
171
	    *chr = wide_buf;
172
173
174
175
176

	/* Save the column width of the wide character in col. */
	if (col != NULL) {
	    /* If we have a tab, get its width in columns using the
	     * current value of col. */
177
	    if (wide_buf == '\t')
178
179
180
181
182
		*col += tabsize - *col % tabsize;
	    /* If we have a control character, get its width using one
	     * column for the "^" that will be displayed in front of it,
	     * and the width in columns of its visible equivalent as
	     * returned by control_rep(). */
183
184
	    else if (is_cntrl_char(wide_buf)) {
		char *ctrl_mb_buf = charalloc(MB_CUR_MAX);
185
186

		(*col)++;
187
		wide_buf = control_rep((unsigned char)wide_buf);
188

189
190
		if (wctomb(ctrl_mb_buf, (wchar_t)wide_buf) != -1) {
		    int width = wcwidth((wchar_t)wide_buf);
191
192
193
194

		    if (width != -1)
			*col += width;
		}
195
196
		else
		    (*col)++;
197

198
		free(ctrl_mb_buf);
199
200
	    /* If we have a normal character, get its width in columns
	     * normally. */
201
	    } else {
202
		int width = wcwidth((wchar_t)wide_buf);
203
204
205
206

		if (width != -1)
		    *col += width;
	    }
207
208
209
	}
    } else {
#endif
210
211
212
	/* Interpret buf's first character as a single-byte sequence. */
	mb_buf_len = 1;
	wide_buf = (unsigned char)*buf;
213
214
215
216

	/* Save the single-byte sequence in chr as though it's a wide
	 * character. */
	if (chr != NULL)
217
	    *chr = wide_buf;
218
219
220
221

	if (col != NULL) {
	    /* If we have a tab, get its width in columns using the
	     * current value of col. */
222
	    if (wide_buf == '\t')
223
224
225
226
227
		*col += tabsize - *col % tabsize;
	    /* If we have a control character, it's two columns wide:
	     * one column for the "^" that will be displayed in front of
	     * it, and one column for its visible equivalent as returned
	     * by control_rep(). */
228
	    else if (is_cntrl_char(wide_buf))
229
230
231
232
233
234
235
236
237
		*col += 2;
	    /* If we have a normal character, it's one column wide. */
	    else
		(*col)++;
	}
#ifdef NANO_WIDE
    }
#endif

238
    return mb_buf_len;
239
240
}

241
/* Return the index in buf of the beginning of the character before the
242
 * one at pos. */
243
size_t move_left(const char *buf, size_t pos)
244
245
246
{
    size_t pos_prev = pos;

247
    assert(str != NULL && pos <= strlen(buf));
248
249
250
251

    /* There is no library function to move backward one multibyte
     * character.  Here is the naive, O(pos) way to do it. */
    while (TRUE) {
252
	int mb_buf_len = parse_char(buf + pos - pos_prev, NULL
253
254
255
#ifdef NANO_WIDE
		, NULL
#endif
256
		, NULL);
257

258
	if (pos_prev <= mb_buf_len)
259
260
	    break;

261
	pos_prev -= mb_buf_len;
262
263
264
265
266
    }

    return pos - pos_prev;
}

267
/* Return the index in buf of the beginning of the character after the
268
 * one at pos. */
269
size_t move_right(const char *buf, size_t pos)
270
{
271
    return pos + parse_char(buf + pos, NULL
272
273
274
#ifdef NANO_WIDE
	, NULL
#endif
275
	, NULL);
276
277
}

Chris Allegretta's avatar
Chris Allegretta committed
278
279
/* Fix the memory allocation for a string. */
void align(char **strp)
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
280
{
David Lawrence Ramsey's avatar
   
David Lawrence Ramsey committed
281
282
283
    assert(strp != NULL);
    if (*strp != NULL)
	*strp = charealloc(*strp, strlen(*strp) + 1);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
284
285
}

Chris Allegretta's avatar
Chris Allegretta committed
286
287
/* Null a string at a certain index and align it. */
void null_at(char **data, size_t index)
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
288
{
Chris Allegretta's avatar
Chris Allegretta committed
289
    assert(data != NULL);
Chris Allegretta's avatar
Chris Allegretta committed
290
    *data = charealloc(*data, index + 1);
Chris Allegretta's avatar
Chris Allegretta committed
291
    (*data)[index] = '\0';
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
292
293
}

Chris Allegretta's avatar
Chris Allegretta committed
294
295
296
/* For non-null-terminated lines.  A line, by definition, shouldn't
 * normally have newlines in it, so encode its nulls as newlines. */
void unsunder(char *str, size_t true_len)
Chris Allegretta's avatar
Chris Allegretta committed
297
{
Chris Allegretta's avatar
Chris Allegretta committed
298
    assert(str != NULL);
299
    for (; true_len > 0; true_len--, str++) {
Chris Allegretta's avatar
Chris Allegretta committed
300
301
	if (*str == '\0')
	    *str = '\n';
302
    }
Chris Allegretta's avatar
Chris Allegretta committed
303
}
Chris Allegretta's avatar
Chris Allegretta committed
304

Chris Allegretta's avatar
Chris Allegretta committed
305
306
307
308
309
/* For non-null-terminated lines.  A line, by definition, shouldn't
 * normally have newlines in it, so decode its newlines into nulls. */
void sunder(char *str)
{
    assert(str != NULL);
310
    for (; *str != '\0'; str++) {
Chris Allegretta's avatar
Chris Allegretta committed
311
312
	if (*str == '\n')
	    *str = '\0';
313
    }
Chris Allegretta's avatar
Chris Allegretta committed
314
315
}

316
317
318
319
320
#ifndef HAVE_STRCASECMP
/* This function is equivalent to strcasecmp(). */
int nstricmp(const char *s1, const char *s2)
{
    assert(s1 != NULL && s2 != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
321

322
323
324
325
    for (; *s1 != '\0' && *s2 != '\0'; s1++, s2++) {
	if (tolower(*s1) != tolower(*s2))
	    break;
    }
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
326

327
328
329
330
331
332
333
334
335
    return (tolower(*s1) - tolower(*s2));
}
#endif

#ifndef HAVE_STRNCASECMP
/* This function is equivalent to strncasecmp(). */
int nstrnicmp(const char *s1, const char *s2, size_t n)
{
    assert(s1 != NULL && s2 != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
336

337
338
339
340
    for (; n > 0 && *s1 != '\0' && *s2 != '\0'; n--, s1++, s2++) {
	if (tolower(*s1) != tolower(*s2))
	    break;
    }
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
341

342
343
    if (n > 0)
	return (tolower(*s1) - tolower(*s2));
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
344
    else
345
	return 0;
346
347
348
}
#endif

349
#ifndef HAVE_STRCASESTR
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
/* This function is equivalent to strcasestr().  It was adapted from
 * mutt's mutt_stristr() function. */
const char *nstristr(const char *haystack, const char *needle)
{
    assert(haystack != NULL && needle != NULL);

    for (; *haystack != '\0'; haystack++) {
	const char *p = haystack;
	const char *q = needle;

	for (; tolower(*p) == tolower(*q) && *q != '\0'; p++, q++)
	    ;

	if (*q == '\0')
	    return haystack;
    }

    return NULL;
}
369
#endif
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
370

371
372
/* None of this is needed if we're using NANO_SMALL! */
#ifndef NANO_SMALL
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
373
374
const char *revstrstr(const char *haystack, const char *needle, const
	char *rev_start)
Chris Allegretta's avatar
Chris Allegretta committed
375
{
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
376
    for (; rev_start >= haystack; rev_start--) {
Chris Allegretta's avatar
Chris Allegretta committed
377
	const char *r, *q;
Chris Allegretta's avatar
Chris Allegretta committed
378

Chris Allegretta's avatar
Chris Allegretta committed
379
	for (r = rev_start, q = needle ; *q == *r && *q != '\0'; r++, q++)
Chris Allegretta's avatar
Chris Allegretta committed
380
381
	    ;
	if (*q == '\0')
Chris Allegretta's avatar
Chris Allegretta committed
382
	    return rev_start;
Chris Allegretta's avatar
Chris Allegretta committed
383
    }
Chris Allegretta's avatar
Chris Allegretta committed
384
    return NULL;
Chris Allegretta's avatar
Chris Allegretta committed
385
386
}

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
387
388
const char *revstristr(const char *haystack, const char *needle, const
	char *rev_start)
Chris Allegretta's avatar
Chris Allegretta committed
389
{
Chris Allegretta's avatar
Chris Allegretta committed
390
391
    for (; rev_start >= haystack; rev_start--) {
	const char *r = rev_start, *q = needle;
Chris Allegretta's avatar
Chris Allegretta committed
392

Chris Allegretta's avatar
Chris Allegretta committed
393
	for (; (tolower(*q) == tolower(*r)) && (*q != '\0') ; r++, q++)
Chris Allegretta's avatar
Chris Allegretta committed
394
395
	    ;
	if (*q == '\0')
Chris Allegretta's avatar
Chris Allegretta committed
396
	    return rev_start;
Chris Allegretta's avatar
Chris Allegretta committed
397
    }
Chris Allegretta's avatar
Chris Allegretta committed
398
    return NULL;
Chris Allegretta's avatar
Chris Allegretta committed
399
}
Chris Allegretta's avatar
Chris Allegretta committed
400
#endif /* !NANO_SMALL */
Chris Allegretta's avatar
Chris Allegretta committed
401

402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
#ifndef HAVE_STRNLEN
/* This function is equivalent to strnlen(). */
size_t nstrnlen(const char *s, size_t maxlen)
{
    size_t n = 0;

    assert(s != NULL);

    for (; maxlen > 0 && *s != '\0'; maxlen--, n++, s++)
	;

    return n;
}
#endif

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
417
#if !defined(NANO_SMALL) && defined(ENABLE_NANORC)
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
#ifndef HAVE_GETLINE
/* This function is equivalent to getline().  It was adapted from
 * GNU mailutils' getline() function. */
ssize_t ngetline(char **lineptr, size_t *n, FILE *stream)
{
    return getdelim(lineptr, n, '\n', stream);
}
#endif

#ifndef HAVE_GETDELIM
/* This function is equivalent to getdelim().  It was adapted from
 * GNU mailutils' getdelim() function. */
ssize_t ngetdelim(char **lineptr, size_t *n, int delim, FILE *stream)
{
    size_t indx = 0;
    int c;

    /* Sanity checks. */
    if (lineptr == NULL || n == NULL || stream == NULL)
	return -1;

    /* Allocate the line the first time. */
    if (*lineptr == NULL) {
441
442
	*lineptr = charalloc(128);
	*n = 128;
443
444
445
446
447
    }

    while ((c = getc(stream)) != EOF) {
	/* Check if more memory is needed. */
	if (indx >= *n) {
448
449
	    *lineptr = charealloc(*lineptr, *n + 128);
	    *n += 128;
450
451
452
453
454
455
456
457
458
459
460
461
	}

	/* Push the result in the line. */
	(*lineptr)[indx++] = (char)c;

	/* Bail out. */
	if (c == delim)
	    break;
    }

    /* Make room for the null character. */
    if (indx >= *n) {
462
463
	*lineptr = charealloc(*lineptr, *n + 128);
	*n += 128;
464
465
466
    }

    /* Null terminate the buffer. */
467
    null_at(lineptr, indx++);
468
    *n = indx;
469
470
471
472
473
474

    /* The last line may not have the delimiter, we have to return what
     * we got and the error will be seen on the next iteration. */
    return (c == EOF && (indx - 1) == 0) ? -1 : indx - 1;
}
#endif
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
475
#endif /* !NANO_SMALL && ENABLE_NANORC */
476

477
478
479
480
481
482
/* If we are searching backwards, we will find the last match that
 * starts no later than start.  Otherwise we find the first match
 * starting no earlier than start.  If we are doing a regexp search, we
 * fill in the global variable regmatches with at most 9 subexpression
 * matches.  Also, all .rm_so elements are relative to the start of the
 * whole match, so regmatches[0].rm_so == 0. */
Chris Allegretta's avatar
Chris Allegretta committed
483
const char *strstrwrapper(const char *haystack, const char *needle,
484
	const char *start)
Chris Allegretta's avatar
Chris Allegretta committed
485
{
486
    /* start can be 1 character before the start or after the end of the
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
487
     * line.  In either case, we just say no match was found. */
488
489
490
    if ((start > haystack && *(start - 1) == '\0') || start < haystack)
	return NULL;
    assert(haystack != NULL && needle != NULL && start != NULL);
491
#ifdef HAVE_REGEX_H
492
    if (ISSET(USE_REGEXP)) {
493
#ifndef NANO_SMALL
494
	if (ISSET(REVERSE_SEARCH)) {
495
496
	    if (regexec(&search_regexp, haystack, 1, regmatches, 0) == 0
		&& haystack + regmatches[0].rm_so <= start) {
497
498
		const char *retval = haystack + regmatches[0].rm_so;

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
499
		/* Search forward until there are no more matches. */
500
		while (regexec(&search_regexp, retval + 1, 1, regmatches,
501
502
			REG_NOTBOL) == 0 && retval + 1 +
			regmatches[0].rm_so <= start)
503
504
505
506
507
508
		    retval += 1 + regmatches[0].rm_so;
		/* Finally, put the subexpression matches in global
		 * variable regmatches.  The REG_NOTBOL flag doesn't
		 * matter now. */
		regexec(&search_regexp, retval, 10, regmatches, 0);
		return retval;
Chris Allegretta's avatar
Chris Allegretta committed
509
	    }
510
511
	} else
#endif /* !NANO_SMALL */
512
513
514
	if (regexec(&search_regexp, start, 10, regmatches,
		start > haystack ? REG_NOTBOL : 0) == 0) {
	    const char *retval = start + regmatches[0].rm_so;
515
516
517

	    regexec(&search_regexp, retval, 10, regmatches, 0);
	    return retval;
518
	}
519
	return NULL;
520
    }
521
#endif /* HAVE_REGEX_H */
522
#if !defined(DISABLE_SPELLER) || !defined(NANO_SMALL)
523
    if (ISSET(CASE_SENSITIVE)) {
524
#ifndef NANO_SMALL
525
	if (ISSET(REVERSE_SEARCH))
526
	    return revstrstr(haystack, needle, start);
527
	else
528
#endif
529
	    return strstr(start, needle);
530
531
532
533
534
    }
#endif /* !DISABLE_SPELLER || !NANO_SMALL */
#ifndef NANO_SMALL
    else if (ISSET(REVERSE_SEARCH))
	return revstristr(haystack, needle, start);
535
#endif
536
    return strcasestr(start, needle);
Chris Allegretta's avatar
Chris Allegretta committed
537
}
Chris Allegretta's avatar
Chris Allegretta committed
538

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
539
540
541
/* This is a wrapper for the perror() function.  The wrapper takes care
 * of ncurses, calls perror (which writes to stderr), then refreshes the
 * screen.  Note that nperror() causes the window to flicker once. */
542
543
void nperror(const char *s)
{
544
    /* leave ncurses mode, go to the terminal */
545
546
547
548
549
550
    if (endwin() != ERR) {
	perror(s);		/* print the error */
	total_refresh();	/* return to ncurses and repaint */
    }
}

Chris Allegretta's avatar
Chris Allegretta committed
551
552
/* Thanks BG, many ppl have been asking for this... */
void *nmalloc(size_t howmuch)
Chris Allegretta's avatar
Chris Allegretta committed
553
{
Chris Allegretta's avatar
Chris Allegretta committed
554
    void *r = malloc(howmuch);
Chris Allegretta's avatar
Chris Allegretta committed
555

Chris Allegretta's avatar
Chris Allegretta committed
556
557
    if (r == NULL && howmuch != 0)
	die(_("nano is out of memory!"));
558

Chris Allegretta's avatar
Chris Allegretta committed
559
    return r;
560
561
}

Chris Allegretta's avatar
Chris Allegretta committed
562
void *nrealloc(void *ptr, size_t howmuch)
Chris Allegretta's avatar
Chris Allegretta committed
563
{
Chris Allegretta's avatar
Chris Allegretta committed
564
    void *r = realloc(ptr, howmuch);
Chris Allegretta's avatar
Chris Allegretta committed
565

Chris Allegretta's avatar
Chris Allegretta committed
566
567
    if (r == NULL && howmuch != 0)
	die(_("nano is out of memory!"));
Chris Allegretta's avatar
Chris Allegretta committed
568
569
570

    return r;
}
Robert Siemborski's avatar
Robert Siemborski committed
571

572
573
574
575
/* Copy the first n characters of one malloc()ed string to another
 * pointer.  Should be used as: "dest = mallocstrncpy(dest, src,
 * n);". */
char *mallocstrncpy(char *dest, const char *src, size_t n)
576
{
577
578
    if (src == NULL)
	src = "";
579

580
    if (src != dest)
581
582
	free(dest);

583
584
    dest = charalloc(n);
    strncpy(dest, src, n);
585
586
587
588

    return dest;
}

589
590
591
592
593
594
595
/* Copy one malloc()ed string to another pointer.  Should be used as:
 * "dest = mallocstrcpy(dest, src);". */
char *mallocstrcpy(char *dest, const char *src)
{
    return mallocstrncpy(dest, src, src == NULL ? 1 : strlen(src) + 1);
}

596
597
598
599
600
601
602
603
604
605
/* Free the malloc()ed string at dest and return the malloc()ed string
 * at src.  Should be used as: "answer = mallocstrassn(answer,
 * real_dir_from_tilde(answer));". */
char *mallocstrassn(char *dest, char *src)
{
    free(dest);
    return src;
}

/* Append a new magicline to filebot. */
606
607
void new_magicline(void)
{
David Lawrence Ramsey's avatar
   
David Lawrence Ramsey committed
608
    filebot->next = (filestruct *)nmalloc(sizeof(filestruct));
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
609
    filebot->next->data = mallocstrcpy(NULL, "");
Robert Siemborski's avatar
Robert Siemborski committed
610
611
612
613
614
    filebot->next->prev = filebot;
    filebot->next->next = NULL;
    filebot->next->lineno = filebot->lineno + 1;
    filebot = filebot->next;
    totlines++;
615
    totsize++;
Robert Siemborski's avatar
Robert Siemborski committed
616
}
Chris Allegretta's avatar
Chris Allegretta committed
617

618
#ifndef NANO_SMALL
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
619
620
/* Remove the magicline from filebot, if there is one and it isn't the
 * only line in the file. */
621
622
void remove_magicline(void)
{
623
    if (filebot->data[0] == '\0' && filebot->prev != NULL) {
624
625
626
627
628
629
630
631
	filebot = filebot->prev;
	free_filestruct(filebot->next);
	filebot->next = NULL;
	totlines--;
	totsize--;
    }
}

632
633
634
635
636
637
638
639
640
641
642
/* Set top_x and bot_x to the top and bottom x-coordinates of the mark,
 * respectively, based on the locations of top and bot.  If
 * right_side_up isn't NULL, set it to TRUE If the mark begins with
 * (mark_beginbuf, mark_beginx) and ends with (current, current_x), or
 * FALSE otherwise. */
void mark_order(const filestruct **top, size_t *top_x, const filestruct
	**bot, size_t *bot_x, bool *right_side_up)
{
    assert(top != NULL && top_x != NULL && bot != NULL && bot_x != NULL);

    if ((current->lineno == mark_beginbuf->lineno && current_x >
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
643
	mark_beginx) || current->lineno > mark_beginbuf->lineno) {
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
	*top = mark_beginbuf;
	*top_x = mark_beginx;
	*bot = current;
	*bot_x = current_x;
	if (right_side_up != NULL)
	    *right_side_up = TRUE;
    } else {
	*bot = mark_beginbuf;
	*bot_x = mark_beginx;
	*top = current;
	*top_x = current_x;
	if (right_side_up != NULL)
	    *right_side_up = FALSE;
    }
}
#endif

661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
/* Calculate the number of lines and the number of characters between
 * begin and end, and return them in lines and size, respectively. */
void get_totals(const filestruct *begin, const filestruct *end, int
	*lines, long *size)
{
    const filestruct *f;

    if (lines != NULL)
	*lines = 0;
    if (size != NULL)
	*size = 0;

    /* Go through the lines from begin to end->prev, if we can. */
    for (f = begin; f != NULL && f != end; f = f->next) {
	/* Count this line. */
676
677
	if (lines != NULL)
	    (*lines)++;
678
679

	/* Count the number of characters on this line. */
680
681
	if (size != NULL) {
	    *size += strlen(f->data);
682

683
684
685
686
	    /* Count the newline if we have one. */
	    if (f->next != NULL)
		(*size)++;
	}
687
688
689
690
691
    }

    /* Go through the line at end, if we can. */
    if (f != NULL) {
	/* Count this line. */
692
693
	if (lines != NULL)
	    (*lines)++;
694
695

	/* Count the number of characters on this line. */
696
697
	if (size != NULL) {
	    *size += strlen(f->data);
698

699
700
701
702
	    /* Count the newline if we have one. */
	    if (f->next != NULL)
		(*size)++;
	}
703
704
705
    }
}

706
#ifndef DISABLE_TABCOMP
Chris Allegretta's avatar
Chris Allegretta committed
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
/*
 * 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)
{
Chris Allegretta's avatar
Chris Allegretta committed
723
724
    const char *retrypat;
    const char *retrytext;
Chris Allegretta's avatar
Chris Allegretta committed
725
726
727
728
    int ch;
    int found;
    int len;

Chris Allegretta's avatar
Chris Allegretta committed
729
730
    retrypat = NULL;
    retrytext = NULL;
Chris Allegretta's avatar
Chris Allegretta committed
731

732
    while (*text != '\0' || *pattern != '\0') {
Chris Allegretta's avatar
Chris Allegretta committed
733
734
735
736
	ch = *pattern++;

	switch (ch) {
	case '*':
Chris Allegretta's avatar
Chris Allegretta committed
737
738
	    retrypat = pattern;
	    retrytext = text;
Chris Allegretta's avatar
Chris Allegretta committed
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
	    break;

	case '[':
	    found = FALSE;

	    while ((ch = *pattern++) != ']') {
		if (ch == '\\')
		    ch = *pattern++;

		if (ch == '\0')
		    return FALSE;

		if (*text == ch)
		    found = TRUE;
	    }
	    len = strlen(text);
755
	    if (!found && len != 0) {
Chris Allegretta's avatar
Chris Allegretta committed
756
757
		return FALSE;
	    }
758
	    if (found) {
Chris Allegretta's avatar
Chris Allegretta committed
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
		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) {
786
		if (*text != '\0')
Chris Allegretta's avatar
Chris Allegretta committed
787
788
789
790
		    text++;
		break;
	    }

791
	    if (*text != '\0') {
Chris Allegretta's avatar
Chris Allegretta committed
792
793
		pattern = retrypat;
		text = ++retrytext;
Chris Allegretta's avatar
Chris Allegretta committed
794
795
796
797
798
799
		break;
	    }

	    return FALSE;
	}

800
	if (pattern == NULL)
Chris Allegretta's avatar
Chris Allegretta committed
801
802
803
804
805
	    return FALSE;
    }

    return TRUE;
}
806
#endif