Line | Count | Source (jump to first uncovered line) |
1 | | /* $OpenBSD$ */ |
2 | | |
3 | | /* |
4 | | * Copyright (c) 2019 Nicholas Marriott <nicholas.marriott@gmail.com> |
5 | | * |
6 | | * Permission to use, copy, modify, and distribute this software for any |
7 | | * purpose with or without fee is hereby granted, provided that the above |
8 | | * copyright notice and this permission notice appear in all copies. |
9 | | * |
10 | | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
11 | | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
12 | | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR |
13 | | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
14 | | * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER |
15 | | * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING |
16 | | * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
17 | | */ |
18 | | |
19 | | #include <sys/types.h> |
20 | | |
21 | | #include <regex.h> |
22 | | #include <string.h> |
23 | | |
24 | | #include "tmux.h" |
25 | | |
26 | | static void |
27 | | regsub_copy(char **buf, ssize_t *len, const char *text, size_t start, size_t end) |
28 | 0 | { |
29 | 0 | size_t add = end - start; |
30 | |
|
31 | 0 | *buf = xrealloc(*buf, (*len) + add + 1); |
32 | 0 | memcpy((*buf) + *len, text + start, add); |
33 | 0 | (*len) += add; |
34 | 0 | } |
35 | | |
36 | | static void |
37 | | regsub_expand(char **buf, ssize_t *len, const char *with, const char *text, |
38 | | regmatch_t *m, u_int n) |
39 | 0 | { |
40 | 0 | const char *cp; |
41 | 0 | u_int i; |
42 | |
|
43 | 0 | for (cp = with; *cp != '\0'; cp++) { |
44 | 0 | if (*cp == '\\') { |
45 | 0 | cp++; |
46 | 0 | if (*cp >= '0' && *cp <= '9') { |
47 | 0 | i = *cp - '0'; |
48 | 0 | if (i < n && m[i].rm_so != m[i].rm_eo) { |
49 | 0 | regsub_copy(buf, len, text, m[i].rm_so, |
50 | 0 | m[i].rm_eo); |
51 | 0 | continue; |
52 | 0 | } |
53 | 0 | } |
54 | 0 | } |
55 | 0 | *buf = xrealloc(*buf, (*len) + 2); |
56 | 0 | (*buf)[(*len)++] = *cp; |
57 | 0 | } |
58 | 0 | } |
59 | | |
60 | | char * |
61 | | regsub(const char *pattern, const char *with, const char *text, int flags) |
62 | 0 | { |
63 | 0 | regex_t r; |
64 | 0 | regmatch_t m[10]; |
65 | 0 | ssize_t start, end, last, len = 0; |
66 | 0 | int empty = 0; |
67 | 0 | char *buf = NULL; |
68 | |
|
69 | 0 | if (*text == '\0') |
70 | 0 | return (xstrdup("")); |
71 | 0 | if (regcomp(&r, pattern, flags) != 0) |
72 | 0 | return (NULL); |
73 | | |
74 | 0 | start = 0; |
75 | 0 | last = 0; |
76 | 0 | end = strlen(text); |
77 | |
|
78 | 0 | while (start <= end) { |
79 | 0 | if (regexec(&r, text + start, nitems(m), m, 0) != 0) { |
80 | 0 | regsub_copy(&buf, &len, text, start, end); |
81 | 0 | break; |
82 | 0 | } |
83 | | |
84 | | /* |
85 | | * Append any text not part of this match (from the end of the |
86 | | * last match). |
87 | | */ |
88 | 0 | regsub_copy(&buf, &len, text, last, m[0].rm_so + start); |
89 | | |
90 | | /* |
91 | | * If the last match was empty and this one isn't (it is either |
92 | | * later or has matched text), expand this match. If it is |
93 | | * empty, move on one character and try again from there. |
94 | | */ |
95 | 0 | if (empty || |
96 | 0 | start + m[0].rm_so != last || |
97 | 0 | m[0].rm_so != m[0].rm_eo) { |
98 | 0 | regsub_expand(&buf, &len, with, text + start, m, |
99 | 0 | nitems(m)); |
100 | |
|
101 | 0 | last = start + m[0].rm_eo; |
102 | 0 | start += m[0].rm_eo; |
103 | 0 | empty = 0; |
104 | 0 | } else { |
105 | 0 | last = start + m[0].rm_eo; |
106 | 0 | start += m[0].rm_eo + 1; |
107 | 0 | empty = 1; |
108 | 0 | } |
109 | | |
110 | | /* Stop now if anchored to start. */ |
111 | 0 | if (*pattern == '^') { |
112 | 0 | regsub_copy(&buf, &len, text, start, end); |
113 | 0 | break; |
114 | 0 | } |
115 | 0 | } |
116 | 0 | buf[len] = '\0'; |
117 | |
|
118 | 0 | regfree(&r); |
119 | 0 | return (buf); |
120 | 0 | } |