aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBartosz Golaszewski <bartekgola@gmail.com>2014-06-17 17:09:17 +0200
committerDenys Vlasenko <vda.linux@googlemail.com>2014-06-17 17:09:41 +0200
commit14158b4127dba30466c50147b868a6a89702960b (patch)
treef8720082f530cad83054b02c246319999959e722
parent85090c162b322a4ffe53d251e59bbfc212a829ee (diff)
downloadbusybox-w32-14158b4127dba30466c50147b868a6a89702960b.tar.gz
busybox-w32-14158b4127dba30466c50147b868a6a89702960b.tar.bz2
busybox-w32-14158b4127dba30466c50147b868a6a89702960b.zip
find: add optional support for '-exec ... {} +'
function old new delta do_exec - 309 +309 parse_params 1416 1487 +71 find_main 342 406 +64 packed_usage 29958 30014 +56 func_exec 138 127 -11 ------------------------------------------------------------------------------ (add/remove: 1/0 grow/shrink: 3/1 up/down: 500/-11) Total: 489 bytes Signed-off-by: Bartosz Golaszewski <bartekgola@gmail.com> Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
-rw-r--r--findutils/find.c156
1 files changed, 141 insertions, 15 deletions
diff --git a/findutils/find.c b/findutils/find.c
index 6d34f4d68..8ac3da7a0 100644
--- a/findutils/find.c
+++ b/findutils/find.c
@@ -137,6 +137,16 @@
137//config: Support the 'find -exec' option for executing commands based upon 137//config: Support the 'find -exec' option for executing commands based upon
138//config: the files matched. 138//config: the files matched.
139//config: 139//config:
140//config:config FEATURE_FIND_EXEC_PLUS
141//config: bool "Enable -exec ... {} +"
142//config: default y
143//config: depends on FEATURE_FIND_EXEC
144//config: help
145//config: Support the 'find -exec ... {} +' option for executing commands
146//config: for all matched files at once.
147//config: Without this option, -exec + is a synonym for -exec ;
148//config: (IOW: it works correctly, but without expected speedup)
149//config:
140//config:config FEATURE_FIND_USER 150//config:config FEATURE_FIND_USER
141//config: bool "Enable -user: username/uid matching" 151//config: bool "Enable -user: username/uid matching"
142//config: default y 152//config: default y
@@ -319,6 +329,9 @@
319//usage: "\n -exec CMD ARG ; Run CMD with all instances of {} replaced by" 329//usage: "\n -exec CMD ARG ; Run CMD with all instances of {} replaced by"
320//usage: "\n file name. Fails if CMD exits with nonzero" 330//usage: "\n file name. Fails if CMD exits with nonzero"
321//usage: ) 331//usage: )
332//usage: IF_FEATURE_FIND_EXEC_PLUS(
333//usage: "\n -exec CMD ARG + Run CMD with {} replaced by list of file names"
334//usage: )
322//usage: IF_FEATURE_FIND_DELETE( 335//usage: IF_FEATURE_FIND_DELETE(
323//usage: "\n -delete Delete current file/directory. Turns on -depth option" 336//usage: "\n -delete Delete current file/directory. Turns on -depth option"
324//usage: ) 337//usage: )
@@ -337,8 +350,12 @@
337# define FNM_CASEFOLD 0 350# define FNM_CASEFOLD 0
338#endif 351#endif
339 352
340#define dbg(...) ((void)0) 353#if 1
341/* #define dbg(...) bb_error_msg(__VA_ARGS__) */ 354# define dbg(...) ((void)0)
355#else
356# define dbg(...) bb_error_msg(__VA_ARGS__)
357#endif
358
342 359
343/* This is a NOEXEC applet. Be very careful! */ 360/* This is a NOEXEC applet. Be very careful! */
344 361
@@ -375,7 +392,20 @@ IF_FEATURE_FIND_CONTEXT(ACTS(context, security_context_t context;))
375IF_FEATURE_FIND_PAREN( ACTS(paren, action ***subexpr;)) 392IF_FEATURE_FIND_PAREN( ACTS(paren, action ***subexpr;))
376IF_FEATURE_FIND_PRUNE( ACTS(prune)) 393IF_FEATURE_FIND_PRUNE( ACTS(prune))
377IF_FEATURE_FIND_DELETE( ACTS(delete)) 394IF_FEATURE_FIND_DELETE( ACTS(delete))
378IF_FEATURE_FIND_EXEC( ACTS(exec, char **exec_argv; unsigned *subst_count; int exec_argc;)) 395IF_FEATURE_FIND_EXEC( ACTS(exec,
396 char **exec_argv; /* -exec ARGS */
397 unsigned *subst_count;
398 int exec_argc; /* count of ARGS */
399 IF_FEATURE_FIND_EXEC_PLUS(
400 /*
401 * filelist is NULL if "exec ;"
402 * non-NULL if "exec +"
403 */
404 char **filelist;
405 int filelist_idx;
406 int file_len;
407 )
408 ))
379IF_FEATURE_FIND_GROUP( ACTS(group, gid_t gid;)) 409IF_FEATURE_FIND_GROUP( ACTS(group, gid_t gid;))
380IF_FEATURE_FIND_LINKS( ACTS(links, char links_char; int links_count;)) 410IF_FEATURE_FIND_LINKS( ACTS(links, char links_char; int links_count;))
381 411
@@ -452,7 +482,6 @@ static int exec_actions(action ***appp, const char *fileName, const struct stat
452 return rc ^ TRUE; /* restore TRUE bit */ 482 return rc ^ TRUE; /* restore TRUE bit */
453} 483}
454 484
455
456#if !FNM_CASEFOLD 485#if !FNM_CASEFOLD
457static char *strcpy_upcase(char *dst, const char *src) 486static char *strcpy_upcase(char *dst, const char *src)
458{ 487{
@@ -576,17 +605,56 @@ ACTF(inum)
576} 605}
577#endif 606#endif
578#if ENABLE_FEATURE_FIND_EXEC 607#if ENABLE_FEATURE_FIND_EXEC
579ACTF(exec) 608static int do_exec(action_exec *ap, const char *fileName)
580{ 609{
581 int i, rc; 610 int i, rc;
582#if ENABLE_USE_PORTABLE_CODE 611# if ENABLE_FEATURE_FIND_EXEC_PLUS
583 char **argv = alloca(sizeof(char*) * (ap->exec_argc + 1)); 612 int size = ap->exec_argc + ap->filelist_idx + 1;
584#else /* gcc 4.3.1 generates smaller code: */ 613# else
585 char *argv[ap->exec_argc + 1]; 614 int size = ap->exec_argc + 1;
586#endif 615# endif
587 for (i = 0; i < ap->exec_argc; i++) 616# if ENABLE_USE_PORTABLE_CODE
588 argv[i] = xmalloc_substitute_string(ap->exec_argv[i], ap->subst_count[i], "{}", fileName); 617 char **argv = alloca(sizeof(char*) * size);
589 argv[i] = NULL; /* terminate the list */ 618# else /* gcc 4.3.1 generates smaller code: */
619 char *argv[size];
620# endif
621 char **pp = argv;
622
623 for (i = 0; i < ap->exec_argc; i++) {
624 const char *arg = ap->exec_argv[i];
625
626# if ENABLE_FEATURE_FIND_EXEC_PLUS
627 if (ap->filelist) {
628 /* Handling "-exec +"
629 * Only one exec_argv[i] has substitution in it.
630 * Expand that one exec_argv[i] into file list.
631 */
632 if (ap->subst_count[i] == 0) {
633 *pp++ = xstrdup(arg);
634 } else {
635 int j = 0;
636 while (ap->filelist[j]) {
637 *pp++ = xmalloc_substitute_string(arg, 1, "{}", ap->filelist[j]);
638 free(ap->filelist[j]);
639 j++;
640 }
641 }
642 } else
643# endif
644 {
645 /* Handling "-exec ;" */
646 *pp++ = xmalloc_substitute_string(arg, ap->subst_count[i], "{}", fileName);
647 }
648 }
649 *pp = NULL; /* terminate the list */
650
651# if ENABLE_FEATURE_FIND_EXEC_PLUS
652 if (ap->filelist) {
653 ap->filelist[0] = NULL;
654 ap->filelist_idx = 0;
655 ap->file_len = 0;
656 }
657# endif
590 658
591 rc = spawn_and_wait(argv); 659 rc = spawn_and_wait(argv);
592 if (rc < 0) 660 if (rc < 0)
@@ -597,6 +665,48 @@ ACTF(exec)
597 free(argv[i++]); 665 free(argv[i++]);
598 return rc == 0; /* return 1 if exitcode 0 */ 666 return rc == 0; /* return 1 if exitcode 0 */
599} 667}
668ACTF(exec)
669{
670# if ENABLE_FEATURE_FIND_EXEC_PLUS
671 if (ap->filelist) {
672 int rc = 0;
673
674 /* If we have lots of files already, exec the command */
675 if (ap->file_len >= 32*1024)
676 rc = do_exec(ap, NULL);
677
678 ap->file_len += strlen(fileName) + sizeof(char*) + 1;
679 ap->filelist = xrealloc_vector(ap->filelist, 8, ap->filelist_idx);
680 ap->filelist[ap->filelist_idx++] = xstrdup(fileName);
681 return rc == 0; /* return 1 if exitcode 0 */
682 }
683# endif
684 return do_exec(ap, fileName);
685}
686# if ENABLE_FEATURE_FIND_EXEC_PLUS
687static int flush_exec_plus(void)
688{
689 action *ap;
690 action **app;
691 action ***appp = G.actions;
692 while ((app = *appp++) != NULL) {
693 while ((ap = *app++) != NULL) {
694 if (ap->f == (action_fp)func_exec) {
695 action_exec *ae = (void*)ap;
696 if (ae->filelist_idx != 0) {
697 int rc = do_exec(ae, NULL);
698# if ENABLE_FEATURE_FIND_NOT
699 if (ap->invert) rc = !rc;
700# endif
701 if (rc)
702 return rc;
703 }
704 }
705 }
706 }
707 return 0;
708}
709# endif
600#endif 710#endif
601#if ENABLE_FEATURE_FIND_USER 711#if ENABLE_FEATURE_FIND_USER
602ACTF(user) 712ACTF(user)
@@ -1037,6 +1147,7 @@ static action*** parse_params(char **argv)
1037 else if (parm == PARM_exec) { 1147 else if (parm == PARM_exec) {
1038 int i; 1148 int i;
1039 action_exec *ap; 1149 action_exec *ap;
1150 IF_FEATURE_FIND_EXEC_PLUS(int all_subst = 0;)
1040 dbg("%d", __LINE__); 1151 dbg("%d", __LINE__);
1041 G.need_print = 0; 1152 G.need_print = 0;
1042 ap = ALLOC_ACTION(exec); 1153 ap = ALLOC_ACTION(exec);
@@ -1049,10 +1160,13 @@ static action*** parse_params(char **argv)
1049 // executes "echo Foo >FILENAME<", 1160 // executes "echo Foo >FILENAME<",
1050 // find -exec echo Foo ">{}<" "+" 1161 // find -exec echo Foo ">{}<" "+"
1051 // executes "echo Foo FILENAME1 FILENAME2 FILENAME3...". 1162 // executes "echo Foo FILENAME1 FILENAME2 FILENAME3...".
1052 // TODO (so far we treat "+" just like ";")
1053 if ((argv[0][0] == ';' || argv[0][0] == '+') 1163 if ((argv[0][0] == ';' || argv[0][0] == '+')
1054 && argv[0][1] == '\0' 1164 && argv[0][1] == '\0'
1055 ) { 1165 ) {
1166# if ENABLE_FEATURE_FIND_EXEC_PLUS
1167 if (argv[0][0] == '+')
1168 ap->filelist = xzalloc(sizeof(ap->filelist[0]));
1169# endif
1056 break; 1170 break;
1057 } 1171 }
1058 argv++; 1172 argv++;
@@ -1062,8 +1176,17 @@ static action*** parse_params(char **argv)
1062 bb_error_msg_and_die(bb_msg_requires_arg, arg); 1176 bb_error_msg_and_die(bb_msg_requires_arg, arg);
1063 ap->subst_count = xmalloc(ap->exec_argc * sizeof(int)); 1177 ap->subst_count = xmalloc(ap->exec_argc * sizeof(int));
1064 i = ap->exec_argc; 1178 i = ap->exec_argc;
1065 while (i--) 1179 while (i--) {
1066 ap->subst_count[i] = count_strstr(ap->exec_argv[i], "{}"); 1180 ap->subst_count[i] = count_strstr(ap->exec_argv[i], "{}");
1181 IF_FEATURE_FIND_EXEC_PLUS(all_subst += ap->subst_count[i];)
1182 }
1183# if ENABLE_FEATURE_FIND_EXEC_PLUS
1184 /*
1185 * coreutils expects {} to appear only once in "-exec +"
1186 */
1187 if (all_subst != 1 && ap->filelist)
1188 bb_error_msg_and_die("only one '{}' allowed for -exec +");
1189# endif
1067 } 1190 }
1068#endif 1191#endif
1069#if ENABLE_FEATURE_FIND_PAREN 1192#if ENABLE_FEATURE_FIND_PAREN
@@ -1335,8 +1458,11 @@ int find_main(int argc UNUSED_PARAM, char **argv)
1335 0) /* depth */ 1458 0) /* depth */
1336 ) { 1459 ) {
1337 status = EXIT_FAILURE; 1460 status = EXIT_FAILURE;
1461 goto out;
1338 } 1462 }
1339 } 1463 }
1340 1464
1465 IF_FEATURE_FIND_EXEC_PLUS(status = flush_exec_plus();)
1466out:
1341 return status; 1467 return status;
1342} 1468}