#endif
;
-#if 0
-/* Nuke from here { */
-
-
-/* get_line_from_file() - This function reads an entire line from a text file
- * * up to a newline. It returns a malloc'ed char * which must be stored and
- * * free'ed by the caller. */
-extern char *get_line_from_file(FILE *file)
-{
- static const int GROWBY = 80; /* how large we will grow strings by */
-
- int ch;
- int idx = 0;
- char *linebuf = NULL;
- int linebufsz = 0;
-
- while (1) {
- ch = fgetc(file);
- if (ch == EOF)
- break;
- /* grow the line buffer as necessary */
- if (idx > linebufsz-2)
- linebuf = realloc(linebuf, linebufsz += GROWBY);
- linebuf[idx++] = (char)ch;
- if ((char)ch == '\n')
- break;
- }
-
- if (idx == 0)
- return NULL;
-
- linebuf[idx] = 0;
- return linebuf;
-}
-
-static void usage(const char *string)
-{
- printf("usage: %s\n", string);
- exit(0);
-}
-
-/* } to here when we integrate this into busybox */
-#endif
-
static void destroy_cmd_strs()
{
if (sed_cmds == NULL)
idx++;
}
else if (my_str[idx] == '/') {
- int ret;
idx = index_of_next_unescaped_slash(idx, my_str);
- if (idx == -1) {
- free(my_str);
+ if (idx == -1)
exit_sed(1, "sed: unterminated match expression\n");
- }
- my_str[idx] = 0; /* shave off the trailing '/' */
- my_str++; /* shave off the leading '/' */
- *regex = (regex_t *)malloc(sizeof(regex_t));
- if ((ret = regcomp(*regex, my_str, 0)) != 0) {
- /* error handling if regular expression couldn't be compiled */
- int errmsgsz = regerror(ret, *regex, NULL, 0);
- char *errmsg = malloc(errmsgsz);
- if (errmsg == NULL) {
- exit_sed(1, "sed: memory error\n");
- }
- regerror(ret, *regex, errmsg, errmsgsz);
- fprintf(stderr, "sed: %s\n", errmsg);
- free(errmsg);
+ my_str[idx] = '\0';
+ *regex = (regex_t *)xmalloc(sizeof(regex_t));
+ if (bb_regcomp(*regex, my_str+1, REG_NEWLINE) != 0) {
+ free(my_str);
exit_sed(1, NULL);
}
- my_str--; /* move my_str back so free() (below) won't barf */
- idx++; /* advance idx one past the end of the /match/ */
}
else {
fprintf(stderr, "sed.c:get_address: no address found in string\n");
return idx;
}
+static char *strdup_substr(const char *str, int start, int end)
+{
+ int size = end - start + 1;
+ char *newstr = xmalloc(size);
+ memcpy(newstr, str+start, size-1);
+ newstr[size-1] = '\0';
+ return newstr;
+}
+
static void parse_cmd_str(struct sed_cmd *sed_cmd, const char *cmdstr)
{
int idx = 0;
sed_cmd->cmd = cmdstr[idx];
/* special-case handling for 's' */
if (sed_cmd->cmd == 's') {
- int oldidx;
+ int oldidx, cflags = REG_NEWLINE;
+ char *match;
/* format for substitution is:
- * s/match/replace/g
- * | |
+ * s/match/replace/gI
+ * | ||
* mandatory optional
*/
if (cmdstr[++idx] != '/')
exit_sed(1, "sed: bad format in substitution expression\n");
- /* get the substitution part */
- idx += get_address(&cmdstr[idx], NULL, &sed_cmd->sub_match);
+ /* save the match string */
+ oldidx = idx+1;
+ idx = index_of_next_unescaped_slash(idx, cmdstr);
+ if (idx == -1)
+ exit_sed(1, "sed: bad format in substitution expression\n");
+ match = strdup_substr(cmdstr, oldidx, idx);
- /* get the replacement part */
- oldidx = idx;
+ /* save the replacement string */
+ oldidx = idx+1;
idx = index_of_next_unescaped_slash(idx, cmdstr);
- sed_cmd->replace = (char *)malloc(idx - oldidx + 1);
- strncpy(sed_cmd->replace, &cmdstr[oldidx], idx - oldidx);
- sed_cmd->replace[idx - oldidx] = 0;
+ if (idx == -1)
+ exit_sed(1, "sed: bad format in substitution expression\n");
+ sed_cmd->replace = strdup_substr(cmdstr, oldidx, idx);
- /* store the 'g' if present */
- if (cmdstr[++idx] == 'g')
- sed_cmd->sub_g = 1;
+ /* process the flags */
+ while (cmdstr[++idx]) {
+ switch (cmdstr[idx]) {
+ case 'g':
+ sed_cmd->sub_g = 1;
+ break;
+ case 'I':
+ cflags |= REG_ICASE;
+ break;
+ default:
+ exit_sed(1, "sed: bad option in substitution expression\n");
+ }
+ }
+
+ /* compile the regex */
+ sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t));
+ if (bb_regcomp(sed_cmd->sub_match, match, cflags) != 0) {
+ free(match);
+ exit_sed(1, NULL);
+ }
+ free(match);
}
}
/* not reached */
return 0;
}
-
-#ifdef TEST_SED
-int main(int argc, char **argv)
-{
- return sed_main(argc, argv);
-}
-#endif
{
int opt;
int reflags;
- int ret;
/* do special-case option parsing */
if (argv[1] && (strcmp(argv[1], "--help") == 0))
reflags = REG_NOSUB | REG_NEWLINE;
if (ignore_case)
reflags |= REG_ICASE;
- if ((ret = regcomp(®ex, argv[optind], reflags)) != 0) {
- int errmsgsz = regerror(ret, ®ex, NULL, 0);
- char *errmsg = malloc(errmsgsz);
- if (errmsg == NULL) {
- fprintf(stderr, "grep: memory error\n");
- regfree(®ex);
- exit(1);
- }
- regerror(ret, ®ex, errmsg, errmsgsz);
- fprintf(stderr, "grep: %s\n", errmsg);
- free(errmsg);
- regfree(®ex);
+ if (bb_regcomp(®ex, argv[optind], reflags) != 0)
exit(1);
- }
/* argv[(optind+1)..(argc-1)] should be names of file to grep through. If
* there is more than one file to grep, we will print the filenames */
{
int opt;
int reflags;
- int ret;
/* do special-case option parsing */
if (argv[1] && (strcmp(argv[1], "--help") == 0))
reflags = REG_NOSUB | REG_NEWLINE;
if (ignore_case)
reflags |= REG_ICASE;
- if ((ret = regcomp(®ex, argv[optind], reflags)) != 0) {
- int errmsgsz = regerror(ret, ®ex, NULL, 0);
- char *errmsg = malloc(errmsgsz);
- if (errmsg == NULL) {
- fprintf(stderr, "grep: memory error\n");
- regfree(®ex);
- exit(1);
- }
- regerror(ret, ®ex, errmsg, errmsgsz);
- fprintf(stderr, "grep: %s\n", errmsg);
- free(errmsg);
- regfree(®ex);
+ if (bb_regcomp(®ex, argv[optind], reflags) != 0)
exit(1);
- }
/* argv[(optind+1)..(argc-1)] should be names of file to grep through. If
* there is more than one file to grep, we will print the filenames */
#include <sys/stat.h>
#include <sys/param.h>
#include <mntent.h>
-
+#include <regex.h>
/* Some useful definitions */
#define FALSE ((int) 1)
extern char *get_line_from_file(FILE *file);
extern char process_escape_sequence(char **ptr);
extern char *get_last_path_component(char *path);
+extern int bb_regcomp(regex_t *preg, const char *regex, int cflags);
extern void *xmalloc (size_t size);
extern char *xstrdup (const char *s);
#endif
;
-#if 0
-/* Nuke from here { */
-
-
-/* get_line_from_file() - This function reads an entire line from a text file
- * * up to a newline. It returns a malloc'ed char * which must be stored and
- * * free'ed by the caller. */
-extern char *get_line_from_file(FILE *file)
-{
- static const int GROWBY = 80; /* how large we will grow strings by */
-
- int ch;
- int idx = 0;
- char *linebuf = NULL;
- int linebufsz = 0;
-
- while (1) {
- ch = fgetc(file);
- if (ch == EOF)
- break;
- /* grow the line buffer as necessary */
- if (idx > linebufsz-2)
- linebuf = realloc(linebuf, linebufsz += GROWBY);
- linebuf[idx++] = (char)ch;
- if ((char)ch == '\n')
- break;
- }
-
- if (idx == 0)
- return NULL;
-
- linebuf[idx] = 0;
- return linebuf;
-}
-
-static void usage(const char *string)
-{
- printf("usage: %s\n", string);
- exit(0);
-}
-
-/* } to here when we integrate this into busybox */
-#endif
-
static void destroy_cmd_strs()
{
if (sed_cmds == NULL)
idx++;
}
else if (my_str[idx] == '/') {
- int ret;
idx = index_of_next_unescaped_slash(idx, my_str);
- if (idx == -1) {
- free(my_str);
+ if (idx == -1)
exit_sed(1, "sed: unterminated match expression\n");
- }
- my_str[idx] = 0; /* shave off the trailing '/' */
- my_str++; /* shave off the leading '/' */
- *regex = (regex_t *)malloc(sizeof(regex_t));
- if ((ret = regcomp(*regex, my_str, 0)) != 0) {
- /* error handling if regular expression couldn't be compiled */
- int errmsgsz = regerror(ret, *regex, NULL, 0);
- char *errmsg = malloc(errmsgsz);
- if (errmsg == NULL) {
- exit_sed(1, "sed: memory error\n");
- }
- regerror(ret, *regex, errmsg, errmsgsz);
- fprintf(stderr, "sed: %s\n", errmsg);
- free(errmsg);
+ my_str[idx] = '\0';
+ *regex = (regex_t *)xmalloc(sizeof(regex_t));
+ if (bb_regcomp(*regex, my_str+1, REG_NEWLINE) != 0) {
+ free(my_str);
exit_sed(1, NULL);
}
- my_str--; /* move my_str back so free() (below) won't barf */
- idx++; /* advance idx one past the end of the /match/ */
}
else {
fprintf(stderr, "sed.c:get_address: no address found in string\n");
return idx;
}
+static char *strdup_substr(const char *str, int start, int end)
+{
+ int size = end - start + 1;
+ char *newstr = xmalloc(size);
+ memcpy(newstr, str+start, size-1);
+ newstr[size-1] = '\0';
+ return newstr;
+}
+
static void parse_cmd_str(struct sed_cmd *sed_cmd, const char *cmdstr)
{
int idx = 0;
sed_cmd->cmd = cmdstr[idx];
/* special-case handling for 's' */
if (sed_cmd->cmd == 's') {
- int oldidx;
+ int oldidx, cflags = REG_NEWLINE;
+ char *match;
/* format for substitution is:
- * s/match/replace/g
- * | |
+ * s/match/replace/gI
+ * | ||
* mandatory optional
*/
if (cmdstr[++idx] != '/')
exit_sed(1, "sed: bad format in substitution expression\n");
- /* get the substitution part */
- idx += get_address(&cmdstr[idx], NULL, &sed_cmd->sub_match);
+ /* save the match string */
+ oldidx = idx+1;
+ idx = index_of_next_unescaped_slash(idx, cmdstr);
+ if (idx == -1)
+ exit_sed(1, "sed: bad format in substitution expression\n");
+ match = strdup_substr(cmdstr, oldidx, idx);
- /* get the replacement part */
- oldidx = idx;
+ /* save the replacement string */
+ oldidx = idx+1;
idx = index_of_next_unescaped_slash(idx, cmdstr);
- sed_cmd->replace = (char *)malloc(idx - oldidx + 1);
- strncpy(sed_cmd->replace, &cmdstr[oldidx], idx - oldidx);
- sed_cmd->replace[idx - oldidx] = 0;
+ if (idx == -1)
+ exit_sed(1, "sed: bad format in substitution expression\n");
+ sed_cmd->replace = strdup_substr(cmdstr, oldidx, idx);
- /* store the 'g' if present */
- if (cmdstr[++idx] == 'g')
- sed_cmd->sub_g = 1;
+ /* process the flags */
+ while (cmdstr[++idx]) {
+ switch (cmdstr[idx]) {
+ case 'g':
+ sed_cmd->sub_g = 1;
+ break;
+ case 'I':
+ cflags |= REG_ICASE;
+ break;
+ default:
+ exit_sed(1, "sed: bad option in substitution expression\n");
+ }
+ }
+
+ /* compile the regex */
+ sed_cmd->sub_match = (regex_t *)xmalloc(sizeof(regex_t));
+ if (bb_regcomp(sed_cmd->sub_match, match, cflags) != 0) {
+ free(match);
+ exit_sed(1, NULL);
+ }
+ free(match);
}
}
/* not reached */
return 0;
}
-
-#ifdef TEST_SED
-int main(int argc, char **argv)
-{
- return sed_main(argc, argv);
-}
-#endif
}
#endif
+#if defined BB_GREP || defined BB_SED
+int bb_regcomp(regex_t *preg, const char *regex, int cflags)
+{
+ int ret;
+ if ((ret = regcomp(preg, regex, cflags)) != 0) {
+ int errmsgsz = regerror(ret, preg, NULL, 0);
+ char *errmsg = xmalloc(errmsgsz);
+ regerror(ret, preg, errmsg, errmsgsz);
+ errorMsg("bb_regcomp: %s\n", errmsg);
+ free(errmsg);
+ regfree(preg);
+ }
+ return ret;
+}
+#endif
+
/* END CODE */
/*
Local Variables: