Home | History | Annotate | Download | only in driver
      1 /*
      2  * Copyright (C) 2015 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include "compiler_options.h"
     18 
     19 #include <fstream>
     20 
     21 namespace art {
     22 
     23 CompilerOptions::CompilerOptions()
     24     : compiler_filter_(CompilerFilter::kDefaultCompilerFilter),
     25       huge_method_threshold_(kDefaultHugeMethodThreshold),
     26       large_method_threshold_(kDefaultLargeMethodThreshold),
     27       small_method_threshold_(kDefaultSmallMethodThreshold),
     28       tiny_method_threshold_(kDefaultTinyMethodThreshold),
     29       num_dex_methods_threshold_(kDefaultNumDexMethodsThreshold),
     30       inline_max_code_units_(kUnsetInlineMaxCodeUnits),
     31       no_inline_from_(nullptr),
     32       boot_image_(false),
     33       app_image_(false),
     34       top_k_profile_threshold_(kDefaultTopKProfileThreshold),
     35       debuggable_(false),
     36       generate_debug_info_(kDefaultGenerateDebugInfo),
     37       generate_mini_debug_info_(kDefaultGenerateMiniDebugInfo),
     38       generate_build_id_(false),
     39       implicit_null_checks_(true),
     40       implicit_so_checks_(true),
     41       implicit_suspend_checks_(false),
     42       compile_pic_(false),
     43       verbose_methods_(),
     44       abort_on_hard_verifier_failure_(false),
     45       init_failure_output_(nullptr),
     46       dump_cfg_file_name_(""),
     47       dump_cfg_append_(false),
     48       force_determinism_(false),
     49       register_allocation_strategy_(RegisterAllocator::kRegisterAllocatorDefault),
     50       passes_to_run_(nullptr) {
     51 }
     52 
     53 CompilerOptions::~CompilerOptions() {
     54   // The destructor looks empty but it destroys a PassManagerOptions object. We keep it here
     55   // because we don't want to include the PassManagerOptions definition from the header file.
     56 }
     57 
     58 void CompilerOptions::ParseHugeMethodMax(const StringPiece& option, UsageFn Usage) {
     59   ParseUintOption(option, "--huge-method-max", &huge_method_threshold_, Usage);
     60 }
     61 
     62 void CompilerOptions::ParseLargeMethodMax(const StringPiece& option, UsageFn Usage) {
     63   ParseUintOption(option, "--large-method-max", &large_method_threshold_, Usage);
     64 }
     65 
     66 void CompilerOptions::ParseSmallMethodMax(const StringPiece& option, UsageFn Usage) {
     67   ParseUintOption(option, "--small-method-max", &small_method_threshold_, Usage);
     68 }
     69 
     70 void CompilerOptions::ParseTinyMethodMax(const StringPiece& option, UsageFn Usage) {
     71   ParseUintOption(option, "--tiny-method-max", &tiny_method_threshold_, Usage);
     72 }
     73 
     74 void CompilerOptions::ParseNumDexMethods(const StringPiece& option, UsageFn Usage) {
     75   ParseUintOption(option, "--num-dex-methods", &num_dex_methods_threshold_, Usage);
     76 }
     77 
     78 void CompilerOptions::ParseInlineMaxCodeUnits(const StringPiece& option, UsageFn Usage) {
     79   ParseUintOption(option, "--inline-max-code-units", &inline_max_code_units_, Usage);
     80 }
     81 
     82 void CompilerOptions::ParseDumpInitFailures(const StringPiece& option,
     83                                             UsageFn Usage ATTRIBUTE_UNUSED) {
     84   DCHECK(option.starts_with("--dump-init-failures="));
     85   std::string file_name = option.substr(strlen("--dump-init-failures=")).data();
     86   init_failure_output_.reset(new std::ofstream(file_name));
     87   if (init_failure_output_.get() == nullptr) {
     88     LOG(ERROR) << "Failed to allocate ofstream";
     89   } else if (init_failure_output_->fail()) {
     90     LOG(ERROR) << "Failed to open " << file_name << " for writing the initialization "
     91                << "failures.";
     92     init_failure_output_.reset();
     93   }
     94 }
     95 
     96 void CompilerOptions::ParseRegisterAllocationStrategy(const StringPiece& option,
     97                                                       UsageFn Usage) {
     98   DCHECK(option.starts_with("--register-allocation-strategy="));
     99   StringPiece choice = option.substr(strlen("--register-allocation-strategy=")).data();
    100   if (choice == "linear-scan") {
    101     register_allocation_strategy_ = RegisterAllocator::Strategy::kRegisterAllocatorLinearScan;
    102   } else if (choice == "graph-color") {
    103     register_allocation_strategy_ = RegisterAllocator::Strategy::kRegisterAllocatorGraphColor;
    104   } else {
    105     Usage("Unrecognized register allocation strategy. Try linear-scan, or graph-color.");
    106   }
    107 }
    108 
    109 bool CompilerOptions::ParseCompilerOption(const StringPiece& option, UsageFn Usage) {
    110   if (option.starts_with("--compiler-filter=")) {
    111     const char* compiler_filter_string = option.substr(strlen("--compiler-filter=")).data();
    112     if (!CompilerFilter::ParseCompilerFilter(compiler_filter_string, &compiler_filter_)) {
    113       Usage("Unknown --compiler-filter value %s", compiler_filter_string);
    114     }
    115   } else if (option == "--compile-pic") {
    116     compile_pic_ = true;
    117   } else if (option.starts_with("--huge-method-max=")) {
    118     ParseHugeMethodMax(option, Usage);
    119   } else if (option.starts_with("--large-method-max=")) {
    120     ParseLargeMethodMax(option, Usage);
    121   } else if (option.starts_with("--small-method-max=")) {
    122     ParseSmallMethodMax(option, Usage);
    123   } else if (option.starts_with("--tiny-method-max=")) {
    124     ParseTinyMethodMax(option, Usage);
    125   } else if (option.starts_with("--num-dex-methods=")) {
    126     ParseNumDexMethods(option, Usage);
    127   } else if (option.starts_with("--inline-max-code-units=")) {
    128     ParseInlineMaxCodeUnits(option, Usage);
    129   } else if (option == "--generate-debug-info" || option == "-g") {
    130     generate_debug_info_ = true;
    131   } else if (option == "--no-generate-debug-info") {
    132     generate_debug_info_ = false;
    133   } else if (option == "--generate-mini-debug-info") {
    134     generate_mini_debug_info_ = true;
    135   } else if (option == "--no-generate-mini-debug-info") {
    136     generate_mini_debug_info_ = false;
    137   } else if (option == "--generate-build-id") {
    138     generate_build_id_ = true;
    139   } else if (option == "--no-generate-build-id") {
    140     generate_build_id_ = false;
    141   } else if (option == "--debuggable") {
    142     debuggable_ = true;
    143   } else if (option.starts_with("--top-k-profile-threshold=")) {
    144     ParseDouble(option.data(), '=', 0.0, 100.0, &top_k_profile_threshold_, Usage);
    145   } else if (option == "--abort-on-hard-verifier-error") {
    146     abort_on_hard_verifier_failure_ = true;
    147   } else if (option.starts_with("--dump-init-failures=")) {
    148     ParseDumpInitFailures(option, Usage);
    149   } else if (option.starts_with("--dump-cfg=")) {
    150     dump_cfg_file_name_ = option.substr(strlen("--dump-cfg=")).data();
    151   } else if (option == "--dump-cfg-append") {
    152     dump_cfg_append_ = true;
    153   } else if (option.starts_with("--register-allocation-strategy=")) {
    154     ParseRegisterAllocationStrategy(option, Usage);
    155   } else if (option.starts_with("--verbose-methods=")) {
    156     // TODO: rather than switch off compiler logging, make all VLOG(compiler) messages
    157     //       conditional on having verbose methods.
    158     gLogVerbosity.compiler = false;
    159     Split(option.substr(strlen("--verbose-methods=")).ToString(), ',', &verbose_methods_);
    160   } else {
    161     // Option not recognized.
    162     return false;
    163   }
    164   return true;
    165 }
    166 
    167 }  // namespace art
    168