509 lines
15 KiB
C++
509 lines
15 KiB
C++
/*
|
|
* Copyright (C) 2014 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#ifndef ART_COMPILER_DRIVER_COMPILER_OPTIONS_H_
|
|
#define ART_COMPILER_DRIVER_COMPILER_OPTIONS_H_
|
|
|
|
#include <memory>
|
|
#include <ostream>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
#include "base/compiler_filter.h"
|
|
#include "base/globals.h"
|
|
#include "base/hash_set.h"
|
|
#include "base/macros.h"
|
|
#include "base/utils.h"
|
|
#include "optimizing/register_allocator.h"
|
|
|
|
namespace art {
|
|
|
|
namespace jit {
|
|
class JitCompiler;
|
|
} // namespace jit
|
|
|
|
namespace verifier {
|
|
class VerifierDepsTest;
|
|
} // namespace verifier
|
|
|
|
namespace linker {
|
|
class Arm64RelativePatcherTest;
|
|
} // namespace linker
|
|
|
|
class DexFile;
|
|
enum class InstructionSet;
|
|
class InstructionSetFeatures;
|
|
class ProfileCompilationInfo;
|
|
class VerificationResults;
|
|
class VerifiedMethod;
|
|
|
|
// Enum for CheckProfileMethodsCompiled. Outside CompilerOptions so it can be forward-declared.
|
|
enum class ProfileMethodsCheck : uint8_t {
|
|
kNone,
|
|
kLog,
|
|
kAbort,
|
|
};
|
|
|
|
class CompilerOptions final {
|
|
public:
|
|
// Guide heuristics to determine whether to compile method if profile data not available.
|
|
static const size_t kDefaultHugeMethodThreshold = 10000;
|
|
static const size_t kDefaultLargeMethodThreshold = 600;
|
|
static const size_t kDefaultNumDexMethodsThreshold = 900;
|
|
static constexpr double kDefaultTopKProfileThreshold = 90.0;
|
|
static const bool kDefaultGenerateDebugInfo = false;
|
|
static const bool kDefaultGenerateMiniDebugInfo = true;
|
|
static const size_t kDefaultInlineMaxCodeUnits = 32;
|
|
static constexpr size_t kUnsetInlineMaxCodeUnits = -1;
|
|
|
|
enum class CompilerType : uint8_t {
|
|
kAotCompiler, // AOT compiler.
|
|
kJitCompiler, // Normal JIT compiler.
|
|
kSharedCodeJitCompiler, // Zygote JIT producing code in the shared region area, putting
|
|
// restrictions on, for example, how literals are being generated.
|
|
};
|
|
|
|
enum class ImageType : uint8_t {
|
|
kNone, // JIT or AOT app compilation producing only an oat file but no image.
|
|
kBootImage, // Creating boot image.
|
|
kBootImageExtension, // Creating boot image extension.
|
|
kAppImage, // Creating app image.
|
|
};
|
|
|
|
CompilerOptions();
|
|
~CompilerOptions();
|
|
|
|
CompilerFilter::Filter GetCompilerFilter() const {
|
|
return compiler_filter_;
|
|
}
|
|
|
|
void SetCompilerFilter(CompilerFilter::Filter compiler_filter) {
|
|
compiler_filter_ = compiler_filter;
|
|
}
|
|
|
|
bool IsAotCompilationEnabled() const {
|
|
return CompilerFilter::IsAotCompilationEnabled(compiler_filter_);
|
|
}
|
|
|
|
bool IsJniCompilationEnabled() const {
|
|
return CompilerFilter::IsJniCompilationEnabled(compiler_filter_);
|
|
}
|
|
|
|
bool IsVerificationEnabled() const {
|
|
return CompilerFilter::IsVerificationEnabled(compiler_filter_);
|
|
}
|
|
|
|
bool AssumeDexFilesAreVerified() const {
|
|
return compiler_filter_ == CompilerFilter::kAssumeVerified;
|
|
}
|
|
|
|
bool AssumeClassesAreVerified() const {
|
|
return compiler_filter_ == CompilerFilter::kAssumeVerified;
|
|
}
|
|
|
|
bool VerifyAtRuntime() const {
|
|
return compiler_filter_ == CompilerFilter::kExtract;
|
|
}
|
|
|
|
bool IsAnyCompilationEnabled() const {
|
|
return CompilerFilter::IsAnyCompilationEnabled(compiler_filter_);
|
|
}
|
|
|
|
size_t GetHugeMethodThreshold() const {
|
|
return huge_method_threshold_;
|
|
}
|
|
|
|
size_t GetLargeMethodThreshold() const {
|
|
return large_method_threshold_;
|
|
}
|
|
|
|
bool IsHugeMethod(size_t num_dalvik_instructions) const {
|
|
return num_dalvik_instructions > huge_method_threshold_;
|
|
}
|
|
|
|
bool IsLargeMethod(size_t num_dalvik_instructions) const {
|
|
return num_dalvik_instructions > large_method_threshold_;
|
|
}
|
|
|
|
size_t GetNumDexMethodsThreshold() const {
|
|
return num_dex_methods_threshold_;
|
|
}
|
|
|
|
size_t GetInlineMaxCodeUnits() const {
|
|
return inline_max_code_units_;
|
|
}
|
|
void SetInlineMaxCodeUnits(size_t units) {
|
|
inline_max_code_units_ = units;
|
|
}
|
|
|
|
double GetTopKProfileThreshold() const {
|
|
return top_k_profile_threshold_;
|
|
}
|
|
|
|
bool GetDebuggable() const {
|
|
return debuggable_;
|
|
}
|
|
|
|
void SetDebuggable(bool value) {
|
|
debuggable_ = value;
|
|
}
|
|
|
|
bool GetNativeDebuggable() const {
|
|
return GetDebuggable() && GetGenerateDebugInfo();
|
|
}
|
|
|
|
// This flag controls whether the compiler collects debugging information.
|
|
// The other flags control how the information is written to disk.
|
|
bool GenerateAnyDebugInfo() const {
|
|
return GetGenerateDebugInfo() || GetGenerateMiniDebugInfo();
|
|
}
|
|
|
|
bool GetGenerateDebugInfo() const {
|
|
return generate_debug_info_;
|
|
}
|
|
|
|
bool GetGenerateMiniDebugInfo() const {
|
|
return generate_mini_debug_info_;
|
|
}
|
|
|
|
// Should run-time checks be emitted in debug mode?
|
|
bool EmitRunTimeChecksInDebugMode() const;
|
|
|
|
bool GetGenerateBuildId() const {
|
|
return generate_build_id_;
|
|
}
|
|
|
|
bool GetImplicitNullChecks() const {
|
|
return implicit_null_checks_;
|
|
}
|
|
|
|
bool GetImplicitStackOverflowChecks() const {
|
|
return implicit_so_checks_;
|
|
}
|
|
|
|
bool IsAotCompiler() const {
|
|
return compiler_type_ == CompilerType::kAotCompiler;
|
|
}
|
|
|
|
bool IsJitCompiler() const {
|
|
return compiler_type_ == CompilerType::kJitCompiler ||
|
|
compiler_type_ == CompilerType::kSharedCodeJitCompiler;
|
|
}
|
|
|
|
bool IsJitCompilerForSharedCode() const {
|
|
return compiler_type_ == CompilerType::kSharedCodeJitCompiler;
|
|
}
|
|
|
|
bool GetImplicitSuspendChecks() const {
|
|
return implicit_suspend_checks_;
|
|
}
|
|
|
|
bool IsGeneratingImage() const {
|
|
return IsBootImage() || IsBootImageExtension() || IsAppImage();
|
|
}
|
|
|
|
// Are we compiling a boot image?
|
|
bool IsBootImage() const {
|
|
return image_type_ == ImageType::kBootImage;
|
|
}
|
|
|
|
// Are we compiling a boot image extension?
|
|
bool IsBootImageExtension() const {
|
|
return image_type_ == ImageType::kBootImageExtension;
|
|
}
|
|
|
|
bool IsBaseline() const {
|
|
return baseline_;
|
|
}
|
|
|
|
// Are we compiling an app image?
|
|
bool IsAppImage() const {
|
|
return image_type_ == ImageType::kAppImage;
|
|
}
|
|
|
|
// Returns whether we are running ART tests.
|
|
// The compiler will use that information for checking invariants.
|
|
bool CompileArtTest() const {
|
|
return compile_art_test_;
|
|
}
|
|
|
|
// Should the code be compiled as position independent?
|
|
bool GetCompilePic() const {
|
|
return compile_pic_;
|
|
}
|
|
|
|
const ProfileCompilationInfo* GetProfileCompilationInfo() const {
|
|
return profile_compilation_info_;
|
|
}
|
|
|
|
bool HasVerboseMethods() const {
|
|
return !verbose_methods_.empty();
|
|
}
|
|
|
|
bool IsVerboseMethod(const std::string& pretty_method) const {
|
|
for (const std::string& cur_method : verbose_methods_) {
|
|
if (pretty_method.find(cur_method) != std::string::npos) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
std::ostream* GetInitFailureOutput() const {
|
|
return init_failure_output_.get();
|
|
}
|
|
|
|
bool AbortOnHardVerifierFailure() const {
|
|
return abort_on_hard_verifier_failure_;
|
|
}
|
|
bool AbortOnSoftVerifierFailure() const {
|
|
return abort_on_soft_verifier_failure_;
|
|
}
|
|
|
|
InstructionSet GetInstructionSet() const {
|
|
return instruction_set_;
|
|
}
|
|
|
|
const InstructionSetFeatures* GetInstructionSetFeatures() const {
|
|
return instruction_set_features_.get();
|
|
}
|
|
|
|
|
|
const std::vector<const DexFile*>& GetNoInlineFromDexFile() const {
|
|
return no_inline_from_;
|
|
}
|
|
|
|
const std::vector<const DexFile*>& GetDexFilesForOatFile() const {
|
|
return dex_files_for_oat_file_;
|
|
}
|
|
|
|
const HashSet<std::string>& GetImageClasses() const {
|
|
return image_classes_;
|
|
}
|
|
|
|
bool IsImageClass(const char* descriptor) const;
|
|
|
|
const VerificationResults* GetVerificationResults() const;
|
|
|
|
const VerifiedMethod* GetVerifiedMethod(const DexFile* dex_file, uint32_t method_idx) const;
|
|
|
|
// Checks if the specified method has been verified without failures. Returns
|
|
// false if the method is not in the verification results (GetVerificationResults).
|
|
bool IsMethodVerifiedWithoutFailures(uint32_t method_idx,
|
|
uint16_t class_def_idx,
|
|
const DexFile& dex_file) const;
|
|
|
|
bool ParseCompilerOptions(const std::vector<std::string>& options,
|
|
bool ignore_unrecognized,
|
|
std::string* error_msg);
|
|
|
|
void SetNonPic() {
|
|
compile_pic_ = false;
|
|
}
|
|
|
|
const std::string& GetDumpCfgFileName() const {
|
|
return dump_cfg_file_name_;
|
|
}
|
|
|
|
bool GetDumpCfgAppend() const {
|
|
return dump_cfg_append_;
|
|
}
|
|
|
|
bool IsForceDeterminism() const {
|
|
return force_determinism_;
|
|
}
|
|
|
|
bool IsCheckLinkageConditions() const {
|
|
return check_linkage_conditions_;
|
|
}
|
|
|
|
bool IsCrashOnLinkageViolation() const {
|
|
return crash_on_linkage_violation_;
|
|
}
|
|
|
|
bool DeduplicateCode() const {
|
|
return deduplicate_code_;
|
|
}
|
|
|
|
RegisterAllocator::Strategy GetRegisterAllocationStrategy() const {
|
|
return register_allocation_strategy_;
|
|
}
|
|
|
|
const std::vector<std::string>* GetPassesToRun() const {
|
|
return passes_to_run_;
|
|
}
|
|
|
|
bool GetDumpTimings() const {
|
|
return dump_timings_;
|
|
}
|
|
|
|
bool GetDumpPassTimings() const {
|
|
return dump_pass_timings_;
|
|
}
|
|
|
|
bool GetDumpStats() const {
|
|
return dump_stats_;
|
|
}
|
|
|
|
bool CountHotnessInCompiledCode() const {
|
|
return count_hotness_in_compiled_code_;
|
|
}
|
|
|
|
bool ResolveStartupConstStrings() const {
|
|
return resolve_startup_const_strings_;
|
|
}
|
|
|
|
ProfileMethodsCheck CheckProfiledMethodsCompiled() const {
|
|
return check_profiled_methods_;
|
|
}
|
|
|
|
uint32_t MaxImageBlockSize() const {
|
|
return max_image_block_size_;
|
|
}
|
|
|
|
void SetMaxImageBlockSize(uint32_t size) {
|
|
max_image_block_size_ = size;
|
|
}
|
|
|
|
bool InitializeAppImageClasses() const {
|
|
return initialize_app_image_classes_;
|
|
}
|
|
|
|
private:
|
|
bool ParseDumpInitFailures(const std::string& option, std::string* error_msg);
|
|
bool ParseRegisterAllocationStrategy(const std::string& option, std::string* error_msg);
|
|
|
|
CompilerFilter::Filter compiler_filter_;
|
|
size_t huge_method_threshold_;
|
|
size_t large_method_threshold_;
|
|
size_t num_dex_methods_threshold_;
|
|
size_t inline_max_code_units_;
|
|
|
|
InstructionSet instruction_set_;
|
|
std::unique_ptr<const InstructionSetFeatures> instruction_set_features_;
|
|
|
|
// Dex files from which we should not inline code. Does not own the dex files.
|
|
// This is usually a very short list (i.e. a single dex file), so we
|
|
// prefer vector<> over a lookup-oriented container, such as set<>.
|
|
std::vector<const DexFile*> no_inline_from_;
|
|
|
|
// List of dex files associated with the oat file, empty for JIT.
|
|
std::vector<const DexFile*> dex_files_for_oat_file_;
|
|
|
|
// Image classes, specifies the classes that will be included in the image if creating an image.
|
|
// Must not be empty for real boot image, only for tests pretending to compile boot image.
|
|
HashSet<std::string> image_classes_;
|
|
|
|
// Results of AOT verification.
|
|
const VerificationResults* verification_results_;
|
|
|
|
CompilerType compiler_type_;
|
|
ImageType image_type_;
|
|
bool compile_art_test_;
|
|
bool baseline_;
|
|
bool debuggable_;
|
|
bool generate_debug_info_;
|
|
bool generate_mini_debug_info_;
|
|
bool generate_build_id_;
|
|
bool implicit_null_checks_;
|
|
bool implicit_so_checks_;
|
|
bool implicit_suspend_checks_;
|
|
bool compile_pic_;
|
|
bool dump_timings_;
|
|
bool dump_pass_timings_;
|
|
bool dump_stats_;
|
|
|
|
// When using a profile file only the top K% of the profiled samples will be compiled.
|
|
double top_k_profile_threshold_;
|
|
|
|
// Info for profile guided compilation.
|
|
const ProfileCompilationInfo* profile_compilation_info_;
|
|
|
|
// Vector of methods to have verbose output enabled for.
|
|
std::vector<std::string> verbose_methods_;
|
|
|
|
// Abort compilation with an error if we find a class that fails verification with a hard
|
|
// failure.
|
|
bool abort_on_hard_verifier_failure_;
|
|
// Same for soft failures.
|
|
bool abort_on_soft_verifier_failure_;
|
|
|
|
// Log initialization of initialization failures to this stream if not null.
|
|
std::unique_ptr<std::ostream> init_failure_output_;
|
|
|
|
std::string dump_cfg_file_name_;
|
|
bool dump_cfg_append_;
|
|
|
|
// Whether the compiler should trade performance for determinism to guarantee exactly reproducible
|
|
// outcomes.
|
|
bool force_determinism_;
|
|
|
|
// Whether the compiler should check for violation of the conditions required to perform AOT
|
|
// "linkage".
|
|
bool check_linkage_conditions_;
|
|
// Whether the compiler should crash when encountering a violation of one of
|
|
// the conditions required to perform AOT "linkage".
|
|
bool crash_on_linkage_violation_;
|
|
|
|
// Whether code should be deduplicated.
|
|
bool deduplicate_code_;
|
|
|
|
// Whether compiled code should increment the hotness count of ArtMethod. Note that the increments
|
|
// won't be atomic for performance reasons, so we accept races, just like in interpreter.
|
|
bool count_hotness_in_compiled_code_;
|
|
|
|
// Whether we eagerly resolve all of the const strings that are loaded from startup methods in the
|
|
// profile.
|
|
bool resolve_startup_const_strings_;
|
|
|
|
// Whether we attempt to run class initializers for app image classes.
|
|
bool initialize_app_image_classes_;
|
|
|
|
// When running profile-guided compilation, check that methods intended to be compiled end
|
|
// up compiled and are not punted.
|
|
ProfileMethodsCheck check_profiled_methods_;
|
|
|
|
// Maximum solid block size in the generated image.
|
|
uint32_t max_image_block_size_;
|
|
|
|
RegisterAllocator::Strategy register_allocation_strategy_;
|
|
|
|
// If not null, specifies optimization passes which will be run instead of defaults.
|
|
// Note that passes_to_run_ is not checked for correctness and providing an incorrect
|
|
// list of passes can lead to unexpected compiler behaviour. This is caused by dependencies
|
|
// between passes. Failing to satisfy them can for example lead to compiler crashes.
|
|
// Passing pass names which are not recognized by the compiler will result in
|
|
// compiler-dependant behavior.
|
|
const std::vector<std::string>* passes_to_run_;
|
|
|
|
friend class Dex2Oat;
|
|
friend class DexToDexDecompilerTest;
|
|
friend class CommonCompilerDriverTest;
|
|
friend class CommonCompilerTestImpl;
|
|
friend class jit::JitCompiler;
|
|
friend class verifier::VerifierDepsTest;
|
|
friend class linker::Arm64RelativePatcherTest;
|
|
|
|
template <class Base>
|
|
friend bool ReadCompilerOptions(Base& map, CompilerOptions* options, std::string* error_msg);
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(CompilerOptions);
|
|
};
|
|
|
|
} // namespace art
|
|
|
|
#endif // ART_COMPILER_DRIVER_COMPILER_OPTIONS_H_
|