fuzz: replace every fuzzer-controlled while loop with a macro #22508
pull apoelstra wants to merge 1 commits into bitcoin:master from apoelstra:2021-07--fuzzer-loops changing 29 files +34 −34-
apoelstra commented at 3:09 pm on July 20, 2021: contributorLimits the number of iterations to 1000 rather than letting the fuzzer do millions or billions of iterations on a single core.
-
apoelstra commented at 3:11 pm on July 20, 2021: contributor
There are several loops in the fuzz test where the number of iterations is directly controlled by the fuzzer, which oftentimes will go for millions of iterations, leading to certain fuzz tests taking forever on a single core. This PR limits these fuzzer-controlled loops to 1000 iterations, an arbitrary high number.
IMHO letting the fuzzer control the number of iterations directly like this, with no restrictions, is a bad idea because the fuzzer will naturally try to hit super high numbers.
I’m looking for a concept ACK (should we limit these) and an approach ACK (is doing this with a macro a reasonable way to go?). Actual ACKs/merges would also be nice :).
-
apoelstra commented at 3:12 pm on July 20, 2021: contributor
On an EPYC 7502 with 256Gb of RAM, just running
time ./test/fuzz/test_runner.py ../../qa-assets/fuzz_seed_corpus/
:before, -j1
0real 67m58.883s 1user 152m7.845s 2sys 4m36.080s
after, -j1
0real 35m57.697s 1user 103m10.803s 2sys 3m52.910s
before, -j32
0real 60m6.374s 1user 153m29.631s 2sys 5m14.991s
after, -j32
0real 28m54.078s 1user 102m52.337s 2sys 4m16.605s
-
MarcoFalke commented at 3:21 pm on July 20, 2021: memberConcept ACK. @DrahtBot please list the conflicts for the alternative solution that has been proposed.
-
DrahtBot commented at 3:22 pm on July 20, 2021: member
The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.
Conflicts
Reviewers, this pull request conflicts with the following ones:
- #22919 (fees: skip pointless fee parameter calculation during IBD by RohitRanjangit)
- #22702 (Add allocator for node based containers by martinus)
If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.
-
MarcoFalke commented at 3:34 pm on July 20, 2021: member
-
apoelstra commented at 4:54 pm on July 20, 2021: contributor@MarcoFalke I added a #define variable and capped the
tx_pool
test to 300 iterations to demonstrate how to use it. -
DrahtBot added the label Tests on Jul 20, 2021
-
in src/test/fuzz/tx_pool.cpp:147 in 5b266249bc outdated
143@@ -142,7 +144,7 @@ FUZZ_TARGET_INIT(tx_pool_standard, initialize_tx_pool) 144 return c.out.nValue; 145 }; 146 147- while (fuzzed_data_provider.ConsumeBool()) { 148+ FUZZER_CONTROLLED_LOOP(fuzzed_data_provider) {
MarcoFalke commented at 6:31 am on July 22, 2021:it would be nice to be able to pass the limit to the loop (in the same line). Otherwise it will get messy if there are nested loops and one forgets to adjust the limit or the loops are moved without moving the limit, which is in a different line.
apoelstra commented at 2:56 pm on July 22, 2021:Sure – replaced the extra #define variable with a new macro argument
apoelstra commented at 2:57 pm on July 22, 2021:Maybe we should just drop the “default 1000” one and always explicitly set the maximum number of iterations?
MarcoFalke commented at 3:02 pm on July 22, 2021:Yeah, I wouldn’t mind that. An additional, 1000
doesn’t seem too verbose.apoelstra force-pushed on Jul 22, 2021in src/test/fuzz/addrman.cpp:47 in 9b7a105fb8 outdated
43@@ -44,7 +44,7 @@ FUZZ_TARGET_INIT(addrman, initialize_addrman) 44 addr_man.m_asmap.clear(); 45 } 46 } 47- while (fuzzed_data_provider.ConsumeBool()) { 48+ FUZZER_CONTROLLED_LOOP(fuzzed_data_provider) {
MarcoFalke commented at 3:01 pm on July 22, 2021:0 LIMITED_WHILE(fuzzed_data_provider.ConsumeBool) {
Since some loops (prefer to?) use
fuzzed_data_provider.remaining_bytes()
, maybe make this configurable as well? Feel free to ignore the naming suggestion.
MarcoFalke commented at 9:33 am on August 2, 2021:@apoelstra Are you still working on this? If not, I am happy to pick up
apoelstra commented at 5:33 pm on September 8, 2021:Hi, sorry, I can pick this up now.DrahtBot added the label Needs rebase on Jul 22, 2021practicalswift commented at 7:43 pm on July 24, 2021: contributorConcept ACK
Nice idea!
apoelstra force-pushed on Sep 8, 2021DrahtBot removed the label Needs rebase on Sep 8, 2021practicalswift commented at 8:13 am on September 9, 2021: contributorThanks for fixing this!
Is the mixing of
LIMITED_WHILE (
andLIMITED_WHILE(
intentional? (Personally I prefer the latter.)ACK modulo spacing consistency :)
laanwj commented at 11:31 am on September 9, 2021: memberThere are several loops in the fuzz test where the number of iterations is directly controlled by the fuzzer, which oftentimes will go for millions of iterations, leading to certain fuzz tests taking forever on a single core.
Concept and code review ACK be655e8b8111f3a7cd7909203c2bac3d16751406
I would slightly prefer this “arbitrary high number” to be a constant as it’s repeated everywhere.
MarcoFalke commented at 11:54 am on September 9, 2021: memberI would slightly prefer this “arbitrary high number” to be a constant as it’s repeated everywhere.
I’d say the number is dependent on the test context. The current ones in master are 30, 300, 3000, and 30000.
I think each number needs to be picked carefully to be in the right 10^x.
Is the mixing of LIMITED_WHILE ( and LIMITED_WHILE( intentional? (Personally I prefer the latter.)
clang-format(-diff) should take care of those in any case.
apoelstra commented at 2:22 pm on September 9, 2021: contributorOops, I will fix the spacing inconsistency. It probably happened when I switched from manual edits to using sed. @MarcoFalke I think we could probably start with an “anbitrary high number”, say 30000 (or 10000), and then when when tests reveal that certain tests are taking too long we can reduce them in individual cases. The current situation effectively has a limit of infinity and things are mostly OK, so I don’ t think this’ll be a big deal.
Once we’re at 5-digit numbers I don’t think there are any situations where we’d want to increase them.
DrahtBot added the label Needs rebase on Oct 5, 2021MarcoFalke commented at 3:11 pm on October 5, 2021: memberSorry, will review soonapoelstra force-pushed on Oct 25, 2021apoelstra commented at 8:03 pm on October 25, 2021: contributorRebased and changed formatting to be consistent.DrahtBot removed the label Needs rebase on Oct 25, 2021in src/test/fuzz/versionbits.cpp:202 in 9116bc17e1 outdated
198@@ -199,7 +199,7 @@ FUZZ_TARGET_INIT(versionbits, initialize) 199 const uint32_t signalling_mask = fuzzed_data_provider.ConsumeIntegral<uint32_t>(); 200 201 // mine prior periods 202- while (fuzzed_data_provider.remaining_bytes() > 0) { 203+ LIMITED_WHILE(fuzzed_data_provider.remaining_bytes() > 0, 10000) {
MarcoFalke commented at 9:37 am on November 3, 2021:Why is this needed? There is already abreak
below, no?
apoelstra commented at 12:21 pm on November 12, 2021:Yep, you’re right, I misread the loop’s exit condition.in src/test/fuzz/rpc.cpp:351 in 9116bc17e1 outdated
347@@ -348,7 +348,7 @@ FUZZ_TARGET_INIT(rpc, initialize_rpc) 348 return; 349 } 350 std::vector<std::string> arguments; 351- while (fuzzed_data_provider.ConsumeBool()) { 352+ LIMITED_WHILE(fuzzed_data_provider.ConsumeBool(), 10000) {
MarcoFalke commented at 9:46 am on November 3, 2021:This seems a bit high?
ConsumeRPCArgument
might callConsumeArrayRPCArgument
, so the limit is 100'000'000. A user shouldn’t be passing in more than 100 items. So any value between 100-1000 seems more appropriate.See also https://bugs.chromium.org/p/oss-fuzz/issues/detail?id=40648
Which is running base58 encoding (expensive) of one rpc arg 6000 times, thus times out. Your limit of 10000 wouldn’t affect that, I think.
apoelstra commented at 12:23 pm on November 12, 2021:ACK. I reduced it to 100.
I worry that reducing this will meaningfully affect test coverage since we presumably want to test what happens when you give gazillions of RPC arguments … but probably a general-purpose fuzz test is the wrong place to try to hit that.
MarcoFalke commented at 3:52 pm on November 12, 2021:Sorry, I also meant in the line above in this file. 100 * 100 is still 10k.MarcoFalke approvedMarcoFalke commented at 9:50 am on November 3, 2021: memberACK, beside commentsapoelstra force-pushed on Nov 12, 2021in src/test/fuzz/versionbits.cpp:202 in 6cfcbdae4e outdated
198@@ -199,7 +199,7 @@ FUZZ_TARGET_INIT(versionbits, initialize) 199 const uint32_t signalling_mask = fuzzed_data_provider.ConsumeIntegral<uint32_t>(); 200 201 // mine prior periods 202- while (fuzzed_data_provider.remaining_bytes() > 0) { 203+ while(fuzzed_data_provider.remaining_bytes() > 0) { // early exit; no need for LIMITED_WHILE
MarcoFalke commented at 3:54 pm on November 12, 2021:0 while (fuzzed_data_provider.remaining_bytes() > 0) { // early exit; no need for LIMITED_WHILE
nit
apoelstra commented at 3:57 pm on November 12, 2021:fixedapoelstra force-pushed on Nov 12, 2021fuzz: replace every fuzzer-controlled loop with a LIMITED_WHILE loop
Blindly chose a cap of 10000 iterations for every loop, except for the two in script_ops.cpp and scriptnum_ops.cpp which appeared to (sometimes) be deserializing individual bytes; capped those to one million to ensure that sometimes we try working with massive scripts. There was also one fuzzer-controlled loop in timedata.cpp which was already capped, so I left that alone. git grep 'while (fuzz' should now run clean except for timedata.cpp
in src/test/fuzz/rpc.cpp:297 in 244eb3c09b outdated
293@@ -294,7 +294,7 @@ std::string ConsumeScalarRPCArgument(FuzzedDataProvider& fuzzed_data_provider) 294 std::string ConsumeArrayRPCArgument(FuzzedDataProvider& fuzzed_data_provider) 295 { 296 std::vector<std::string> scalar_arguments; 297- while (fuzzed_data_provider.ConsumeBool()) { 298+ LIMITED_WHILE(fuzzed_data_provider.ConsumeBool(), 10000) {
MarcoFalke commented at 4:10 pm on November 12, 2021:This one I meant, too ;)
MarcoFalke commented at 4:10 pm on November 12, 2021:apoelstra force-pushed on Nov 12, 2021MarcoFalke commented at 3:50 pm on November 15, 2021: memberSome times:
master | this | relative | name
0 206.2 76.67 0.3718234724 rpc 1 149.06 157.45 1.056286059 process_messages 2 27.21 27.19 0.9992649761 addrman 3 24.19 24.89 1.028937578 script_sign 4 23.93 26.08 1.089845382 process_message 5 23.32 20.44 0.8765008576 coins_view 6 15.22 15.75 1.034822602 policy_estimator 7 14.46 15.32 1.059474412 script_ops 8 13.06 12.85 0.9839203675 bloom_filter 9 12.35 12.41 1.0048583 pow 10 11.93 8.79 0.7367979883 connman 11 10.71 9.38 0.8758169935 net 12 9.22 7.21 0.7819956616 node_eviction 13 7.7 8.23 1.068831169 rbf 14
MarcoFalke commented at 3:51 pm on November 15, 2021: memberSo yeah, this should fix https://bugs.chromium.org/p/oss-fuzz/issues/detail?id=40648MarcoFalke commented at 3:51 pm on November 15, 2021: membercr ACK 214d9055acdd72189a2f415477ce472ca8db4191MarcoFalke merged this on Nov 15, 2021MarcoFalke closed this on Nov 15, 2021
sidhujag referenced this in commit 921b690a0d on Nov 16, 2021apoelstra deleted the branch on Nov 16, 2021DrahtBot locked this on Nov 17, 2022
github-metadata-mirror
This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2025-01-22 06:12 UTC
This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me