diff options
author | Christian Speckner <christian.speckner@mayflower.de> | 2024-07-16 01:04:37 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-07-15 16:04:37 -0700 |
commit | fd8b2bd43d73cf1976426e60c22c5261fa343510 (patch) | |
tree | 79d1fbc91d65ba0f8a4a6b3db0ca070b0628ebda /src/tools/optimization-options.h | |
parent | aec516f1259c3fec92982db92dc0e4e67ab2251a (diff) | |
download | binaryen-fd8b2bd43d73cf1976426e60c22c5261fa343510.tar.gz binaryen-fd8b2bd43d73cf1976426e60c22c5261fa343510.tar.bz2 binaryen-fd8b2bd43d73cf1976426e60c22c5261fa343510.zip |
Allow different arguments for multiple instances of a pass (#6687)
Each pass instance can now store an argument for it, which can be different.
This may be a breaking change for the corner case of running a pass multiple
times and setting the pass's argument multiple times as well (before, the last
pass argument affected them all; now, it affects the last instance only). This
only affects arguments with the name of a pass; others remain global, as
before (and multiple passes can read them, in fact). See the CHANGELOG for
details.
Fixes #6646
Diffstat (limited to 'src/tools/optimization-options.h')
-rw-r--r-- | src/tools/optimization-options.h | 44 |
1 files changed, 38 insertions, 6 deletions
diff --git a/src/tools/optimization-options.h b/src/tools/optimization-options.h index af468769d..f04b73211 100644 --- a/src/tools/optimization-options.h +++ b/src/tools/optimization-options.h @@ -51,6 +51,9 @@ struct OptimizationOptions : public ToolOptions { // The name of the pass to run. std::string name; + // The main argument of the pass, if applicable. + std::optional<std::string> argument; + // The optimize and shrink levels to run the pass with, if specified. If not // specified then the defaults are used. std::optional<int> optimizeLevel; @@ -330,18 +333,47 @@ struct OptimizationOptions : public ToolOptions { // --foo --pass-arg=foo@ARG Options::Arguments::Optional, [this, p](Options*, const std::string& arg) { + PassInfo info(p); if (!arg.empty()) { - if (passOptions.arguments.count(p)) { - Fatal() << "Cannot pass multiple pass arguments to " << p; - } - passOptions.arguments[p] = arg; + info.argument = arg; } - passes.push_back(p); + + passes.push_back(info); }, PassRegistry::get()->isPassHidden(p)); } } + // Pass arguments with the same name as the pass are stored per-instance on + // PassInfo, while all other arguments are stored globally on + // passOptions.arguments (which is what the overriden method on ToolOptions + // does). + void addPassArg(const std::string& key, const std::string& value) override { + // Scan the current pass list for the last defined instance of a pass named + // like the argument under consideration. + for (auto i = passes.rbegin(); i != passes.rend(); i++) { + if (i->name != key) { + continue; + } + + if (i->argument.has_value()) { + Fatal() << i->name << " already set to " << *(i->argument); + } + + // Found? Store the argument value there and return. + i->argument = value; + return; + } + + // Not found? Store it globally if there is no pass with the same name. + if (!PassRegistry::get()->containsPass(key)) { + return ToolOptions::addPassArg(key, value); + } + + // Not found, but we have a pass with the same name? Bail out. + Fatal() << "can't set " << key << ": pass not enabled"; + } + bool runningDefaultOptimizationPasses() { for (auto& pass : passes) { if (pass.name == DEFAULT_OPT_PASSES) { @@ -393,7 +425,7 @@ struct OptimizationOptions : public ToolOptions { passRunner.options.shrinkLevel = passOptions.shrinkLevel; } else { // This is a normal pass. Add it to the queue for execution. - passRunner.add(pass.name); + passRunner.add(pass.name, pass.argument); // Normal passes do not set their own optimize/shrinkLevels. assert(!pass.optimizeLevel); |