utils.c 11.9 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>
30
#include <pwd.h>
Chris Allegretta's avatar
Chris Allegretta committed
31
#include <ctype.h>
32
#include <errno.h>
Chris Allegretta's avatar
Chris Allegretta committed
33
#include <assert.h>
Chris Allegretta's avatar
Chris Allegretta committed
34
35
#include "proto.h"

36
#ifdef HAVE_REGEX_H
37
#ifdef BROKEN_REGEXEC
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
38
/* Work around a potential segfault in glibc 2.2.3's regexec(). */
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
39
int safe_regexec(const regex_t *preg, const char *string, size_t nmatch,
40
41
42
43
	regmatch_t pmatch[], int eflags)
{
    if (string != NULL && *string != '\0')
	return regexec(preg, string, nmatch, pmatch, eflags);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
44

45
46
    return REG_NOMATCH;
}
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
47
#endif
48
49
50

int regexp_bol_or_eol(const regex_t *preg, const char *string)
{
51
52
    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
53
	REG_NOMATCH);
54
55
}
#endif /* HAVE_REGEX_H */
56

57
int digits(size_t n)
58
59
60
61
62
63
64
65
66
67
68
{
    int i = 1;

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

    return i;
}

69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/* Return the user's home directory.  We use $HOME, and if that fails,
 * we fall back on getpwuid(). */
void get_homedir(void)
{
    if (homedir == NULL) {
	const char *homenv = getenv("HOME");

	if (homenv == NULL) {
	    const struct passwd *userage = getpwuid(geteuid());

	    if (userage != NULL)
		homenv = userage->pw_dir;
	}
	homedir = mallocstrcpy(NULL, homenv);
    }
}

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
86
87
88
/* 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. */
89
bool parse_num(const char *str, ssize_t *val)
90
91
92
93
94
{
    char *first_error;
    ssize_t j;

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

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

98
    if (errno == ERANGE || *str == '\0' || *first_error != '\0')
99
	return FALSE;
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
100

101
102
    if (val != NULL)
	*val = j;
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
103

104
    return TRUE;
105
106
}

Chris Allegretta's avatar
Chris Allegretta committed
107
108
/* Fix the memory allocation for a string. */
void align(char **strp)
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
109
{
David Lawrence Ramsey's avatar
   
David Lawrence Ramsey committed
110
    assert(strp != NULL);
111

David Lawrence Ramsey's avatar
   
David Lawrence Ramsey committed
112
113
    if (*strp != NULL)
	*strp = charealloc(*strp, strlen(*strp) + 1);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
114
115
}

Chris Allegretta's avatar
Chris Allegretta committed
116
117
/* 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
118
{
Chris Allegretta's avatar
Chris Allegretta committed
119
    assert(data != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
120

Chris Allegretta's avatar
Chris Allegretta committed
121
    *data = charealloc(*data, index + 1);
Chris Allegretta's avatar
Chris Allegretta committed
122
    (*data)[index] = '\0';
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
123
124
}

Chris Allegretta's avatar
Chris Allegretta committed
125
126
127
/* 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
128
{
Chris Allegretta's avatar
Chris Allegretta committed
129
    assert(str != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
130

131
    for (; true_len > 0; true_len--, str++) {
Chris Allegretta's avatar
Chris Allegretta committed
132
133
	if (*str == '\0')
	    *str = '\n';
134
    }
Chris Allegretta's avatar
Chris Allegretta committed
135
}
Chris Allegretta's avatar
Chris Allegretta committed
136

Chris Allegretta's avatar
Chris Allegretta committed
137
138
139
140
141
/* 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);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
142

143
    for (; *str != '\0'; str++) {
Chris Allegretta's avatar
Chris Allegretta committed
144
145
	if (*str == '\n')
	    *str = '\0';
146
    }
Chris Allegretta's avatar
Chris Allegretta committed
147
148
}

149
#if !defined(NANO_SMALL) && defined(ENABLE_NANORC)
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
#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) {
173
174
	*lineptr = charalloc(MAX_BUF_SIZE);
	*n = MAX_BUF_SIZE;
175
176
177
178
179
    }

    while ((c = getc(stream)) != EOF) {
	/* Check if more memory is needed. */
	if (indx >= *n) {
180
181
	    *lineptr = charealloc(*lineptr, *n + MAX_BUF_SIZE);
	    *n += MAX_BUF_SIZE;
182
183
184
185
186
187
188
189
190
191
192
193
	}

	/* 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) {
194
195
	*lineptr = charealloc(*lineptr, *n + MAX_BUF_SIZE);
	*n += MAX_BUF_SIZE;
196
197
198
    }

    /* Null terminate the buffer. */
199
    null_at(lineptr, indx++);
200
    *n = indx;
201
202
203
204
205
206

    /* 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
207
#endif /* !NANO_SMALL && ENABLE_NANORC */
208

209
210
211
212
213
214
/* 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
215
const char *strstrwrapper(const char *haystack, const char *needle,
216
	const char *start)
Chris Allegretta's avatar
Chris Allegretta committed
217
{
218
    /* start can be 1 character before the start or after the end of the
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
219
     * line.  In either case, we just say no match was found. */
220
221
    if ((start > haystack && *(start - 1) == '\0') || start < haystack)
	return NULL;
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
222

223
    assert(haystack != NULL && needle != NULL && start != NULL);
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
224

225
#ifdef HAVE_REGEX_H
226
    if (ISSET(USE_REGEXP)) {
227
#ifndef NANO_SMALL
228
	if (ISSET(REVERSE_SEARCH)) {
229
230
	    if (regexec(&search_regexp, haystack, 1, regmatches, 0) == 0
		&& haystack + regmatches[0].rm_so <= start) {
231
232
		const char *retval = haystack + regmatches[0].rm_so;

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
233
		/* Search forward until there are no more matches. */
234
		while (regexec(&search_regexp, retval + 1, 1, regmatches,
235
236
			REG_NOTBOL) == 0 && retval + 1 +
			regmatches[0].rm_so <= start)
237
238
239
240
241
242
		    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
243
	    }
244
245
	} else
#endif /* !NANO_SMALL */
246
	if (regexec(&search_regexp, start, 10, regmatches,
247
		(start > haystack) ? REG_NOTBOL : 0) == 0) {
248
	    const char *retval = start + regmatches[0].rm_so;
249
250
251

	    regexec(&search_regexp, retval, 10, regmatches, 0);
	    return retval;
252
	}
253
	return NULL;
254
    }
255
#endif /* HAVE_REGEX_H */
256
#if !defined(NANO_SMALL) || !defined(DISABLE_SPELLER)
257
    if (ISSET(CASE_SENSITIVE)) {
258
#ifndef NANO_SMALL
259
	if (ISSET(REVERSE_SEARCH))
260
	    return revstrstr(haystack, needle, start);
261
	else
262
#endif
263
	    return strstr(start, needle);
264
265
266
267
    }
#endif /* !DISABLE_SPELLER || !NANO_SMALL */
#ifndef NANO_SMALL
    else if (ISSET(REVERSE_SEARCH))
268
	return mbrevstrcasestr(haystack, needle, start);
269
#endif
270
    return mbstrcasestr(start, needle);
Chris Allegretta's avatar
Chris Allegretta committed
271
}
Chris Allegretta's avatar
Chris Allegretta committed
272

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
273
/* This is a wrapper for the perror() function.  The wrapper takes care
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
274
275
276
 * of curses, calls perror() (which writes to stderr), and then
 * refreshes the screen.  Note that nperror() causes the window to
 * flicker once. */
277
278
void nperror(const char *s)
{
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
279
    /* Leave curses mode and go to the terminal. */
280
    if (endwin() != ERR) {
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
281
282
	perror(s);		/* Print the error. */
	total_refresh();	/* Return to curses and refresh. */
283
284
285
    }
}

David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
286
/* Thanks, BG, many people have been asking for this... */
Chris Allegretta's avatar
Chris Allegretta committed
287
void *nmalloc(size_t howmuch)
Chris Allegretta's avatar
Chris Allegretta committed
288
{
Chris Allegretta's avatar
Chris Allegretta committed
289
    void *r = malloc(howmuch);
Chris Allegretta's avatar
Chris Allegretta committed
290

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

Chris Allegretta's avatar
Chris Allegretta committed
294
    return r;
295
296
}

Chris Allegretta's avatar
Chris Allegretta committed
297
void *nrealloc(void *ptr, size_t howmuch)
Chris Allegretta's avatar
Chris Allegretta committed
298
{
Chris Allegretta's avatar
Chris Allegretta committed
299
    void *r = realloc(ptr, howmuch);
Chris Allegretta's avatar
Chris Allegretta committed
300

Chris Allegretta's avatar
Chris Allegretta committed
301
302
    if (r == NULL && howmuch != 0)
	die(_("nano is out of memory!"));
Chris Allegretta's avatar
Chris Allegretta committed
303
304
305

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

307
308
309
310
/* 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)
311
{
312
313
    if (src == NULL)
	src = "";
314

315
    if (src != dest)
316
317
	free(dest);

318
319
    dest = charalloc(n);
    strncpy(dest, src, n);
320
321
322
323

    return dest;
}

324
325
326
327
328
329
330
/* 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);
}

331
332
333
334
335
336
337
338
339
340
/* 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. */
341
342
void new_magicline(void)
{
David Lawrence Ramsey's avatar
   
David Lawrence Ramsey committed
343
    filebot->next = (filestruct *)nmalloc(sizeof(filestruct));
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
344
    filebot->next->data = mallocstrcpy(NULL, "");
Robert Siemborski's avatar
Robert Siemborski committed
345
346
347
348
349
    filebot->next->prev = filebot;
    filebot->next->next = NULL;
    filebot->next->lineno = filebot->lineno + 1;
    filebot = filebot->next;
    totlines++;
350
    totsize++;
Robert Siemborski's avatar
Robert Siemborski committed
351
}
Chris Allegretta's avatar
Chris Allegretta committed
352

353
#ifndef NANO_SMALL
David Lawrence Ramsey's avatar
David Lawrence Ramsey committed
354
355
/* Remove the magicline from filebot, if there is one and it isn't the
 * only line in the file. */
356
357
void remove_magicline(void)
{
358
    if (filebot->data[0] == '\0' && filebot->prev != NULL) {
359
360
361
362
363
364
365
366
	filebot = filebot->prev;
	free_filestruct(filebot->next);
	filebot->next = NULL;
	totlines--;
	totsize--;
    }
}

367
368
369
370
371
372
373
374
375
376
377
/* 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
378
	mark_beginx) || current->lineno > mark_beginbuf->lineno) {
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
	*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

396
397
398
/* 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
399
	*lines, size_t *size)
400
401
402
403
404
405
406
407
408
409
410
{
    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. */
411
412
	if (lines != NULL)
	    (*lines)++;
413
414

	/* Count the number of characters on this line. */
415
	if (size != NULL) {
416
	    *size += mbstrlen(f->data);
417

418
419
420
421
	    /* Count the newline if we have one. */
	    if (f->next != NULL)
		(*size)++;
	}
422
423
424
425
426
    }

    /* Go through the line at end, if we can. */
    if (f != NULL) {
	/* Count this line. */
427
428
	if (lines != NULL)
	    (*lines)++;
429
430

	/* Count the number of characters on this line. */
431
	if (size != NULL) {
432
	    *size += mbstrlen(f->data);
433

434
435
436
437
	    /* Count the newline if we have one. */
	    if (f->next != NULL)
		(*size)++;
	}
438
439
    }
}