diff options
author | Bartosz Golaszewski <bartekgola@gmail.com> | 2014-06-17 17:09:17 +0200 |
---|---|---|
committer | Denys Vlasenko <vda.linux@googlemail.com> | 2014-06-17 17:09:41 +0200 |
commit | 14158b4127dba30466c50147b868a6a89702960b (patch) | |
tree | f8720082f530cad83054b02c246319999959e722 | |
parent | 85090c162b322a4ffe53d251e59bbfc212a829ee (diff) | |
download | busybox-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.c | 156 |
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;)) | |||
375 | IF_FEATURE_FIND_PAREN( ACTS(paren, action ***subexpr;)) | 392 | IF_FEATURE_FIND_PAREN( ACTS(paren, action ***subexpr;)) |
376 | IF_FEATURE_FIND_PRUNE( ACTS(prune)) | 393 | IF_FEATURE_FIND_PRUNE( ACTS(prune)) |
377 | IF_FEATURE_FIND_DELETE( ACTS(delete)) | 394 | IF_FEATURE_FIND_DELETE( ACTS(delete)) |
378 | IF_FEATURE_FIND_EXEC( ACTS(exec, char **exec_argv; unsigned *subst_count; int exec_argc;)) | 395 | IF_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 | )) | ||
379 | IF_FEATURE_FIND_GROUP( ACTS(group, gid_t gid;)) | 409 | IF_FEATURE_FIND_GROUP( ACTS(group, gid_t gid;)) |
380 | IF_FEATURE_FIND_LINKS( ACTS(links, char links_char; int links_count;)) | 410 | IF_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 |
457 | static char *strcpy_upcase(char *dst, const char *src) | 486 | static 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 |
579 | ACTF(exec) | 608 | static 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 | } |
668 | ACTF(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 | ||
687 | static 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 |
602 | ACTF(user) | 712 | ACTF(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();) | ||
1466 | out: | ||
1341 | return status; | 1467 | return status; |
1342 | } | 1468 | } |