123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- //===--- OpenCLOptions.cpp---------------------------------------*- C++ -*-===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #include "clang/Basic/OpenCLOptions.h"
- #include "clang/Basic/Diagnostic.h"
- #include "clang/Basic/TargetInfo.h"
- namespace clang {
- // First feature in a pair requires the second one to be supported.
- static const std::pair<StringRef, StringRef> DependentFeaturesList[] = {
- {"__opencl_c_read_write_images", "__opencl_c_images"},
- {"__opencl_c_3d_image_writes", "__opencl_c_images"},
- {"__opencl_c_pipes", "__opencl_c_generic_address_space"},
- {"__opencl_c_device_enqueue", "__opencl_c_generic_address_space"},
- {"__opencl_c_device_enqueue", "__opencl_c_program_scope_global_variables"}};
- // Extensions and equivalent feature pairs.
- static const std::pair<StringRef, StringRef> FeatureExtensionMap[] = {
- {"cl_khr_fp64", "__opencl_c_fp64"},
- {"cl_khr_3d_image_writes", "__opencl_c_3d_image_writes"}};
- bool OpenCLOptions::isKnown(llvm::StringRef Ext) const {
- return OptMap.find(Ext) != OptMap.end();
- }
- bool OpenCLOptions::isAvailableOption(llvm::StringRef Ext,
- const LangOptions &LO) const {
- if (!isKnown(Ext))
- return false;
- auto &OptInfo = OptMap.find(Ext)->getValue();
- if (OptInfo.isCoreIn(LO) || OptInfo.isOptionalCoreIn(LO))
- return isSupported(Ext, LO);
- return isEnabled(Ext);
- }
- bool OpenCLOptions::isEnabled(llvm::StringRef Ext) const {
- auto I = OptMap.find(Ext);
- return I != OptMap.end() && I->getValue().Enabled;
- }
- bool OpenCLOptions::isWithPragma(llvm::StringRef Ext) const {
- auto E = OptMap.find(Ext);
- return E != OptMap.end() && E->second.WithPragma;
- }
- bool OpenCLOptions::isSupported(llvm::StringRef Ext,
- const LangOptions &LO) const {
- auto I = OptMap.find(Ext);
- return I != OptMap.end() && I->getValue().Supported &&
- I->getValue().isAvailableIn(LO);
- }
- bool OpenCLOptions::isSupportedCore(llvm::StringRef Ext,
- const LangOptions &LO) const {
- auto I = OptMap.find(Ext);
- return I != OptMap.end() && I->getValue().Supported &&
- I->getValue().isCoreIn(LO);
- }
- bool OpenCLOptions::isSupportedOptionalCore(llvm::StringRef Ext,
- const LangOptions &LO) const {
- auto I = OptMap.find(Ext);
- return I != OptMap.end() && I->getValue().Supported &&
- I->getValue().isOptionalCoreIn(LO);
- }
- bool OpenCLOptions::isSupportedCoreOrOptionalCore(llvm::StringRef Ext,
- const LangOptions &LO) const {
- return isSupportedCore(Ext, LO) || isSupportedOptionalCore(Ext, LO);
- }
- bool OpenCLOptions::isSupportedExtension(llvm::StringRef Ext,
- const LangOptions &LO) const {
- auto I = OptMap.find(Ext);
- return I != OptMap.end() && I->getValue().Supported &&
- I->getValue().isAvailableIn(LO) &&
- !isSupportedCoreOrOptionalCore(Ext, LO);
- }
- void OpenCLOptions::enable(llvm::StringRef Ext, bool V) {
- OptMap[Ext].Enabled = V;
- }
- void OpenCLOptions::acceptsPragma(llvm::StringRef Ext, bool V) {
- OptMap[Ext].WithPragma = V;
- }
- void OpenCLOptions::support(llvm::StringRef Ext, bool V) {
- assert(!Ext.empty() && "Extension is empty.");
- assert(Ext[0] != '+' && Ext[0] != '-');
- OptMap[Ext].Supported = V;
- }
- OpenCLOptions::OpenCLOptions() {
- #define OPENCL_GENERIC_EXTENSION(Ext, ...) \
- OptMap.insert_or_assign(#Ext, OpenCLOptionInfo{__VA_ARGS__});
- #include "clang/Basic/OpenCLExtensions.def"
- }
- void OpenCLOptions::addSupport(const llvm::StringMap<bool> &FeaturesMap,
- const LangOptions &Opts) {
- for (const auto &F : FeaturesMap) {
- const auto &Name = F.getKey();
- if (F.getValue() && isKnown(Name) && OptMap[Name].isAvailableIn(Opts))
- support(Name);
- }
- }
- void OpenCLOptions::disableAll() {
- for (auto &Opt : OptMap)
- Opt.getValue().Enabled = false;
- }
- bool OpenCLOptions::diagnoseUnsupportedFeatureDependencies(
- const TargetInfo &TI, DiagnosticsEngine &Diags) {
- auto OpenCLFeaturesMap = TI.getSupportedOpenCLOpts();
- bool IsValid = true;
- for (auto &FeaturePair : DependentFeaturesList) {
- auto Feature = FeaturePair.first;
- auto Dep = FeaturePair.second;
- if (TI.hasFeatureEnabled(OpenCLFeaturesMap, Feature) &&
- !TI.hasFeatureEnabled(OpenCLFeaturesMap, Dep)) {
- IsValid = false;
- Diags.Report(diag::err_opencl_feature_requires) << Feature << Dep;
- }
- }
- return IsValid;
- }
- bool OpenCLOptions::diagnoseFeatureExtensionDifferences(
- const TargetInfo &TI, DiagnosticsEngine &Diags) {
- auto OpenCLFeaturesMap = TI.getSupportedOpenCLOpts();
- bool IsValid = true;
- for (auto &ExtAndFeat : FeatureExtensionMap)
- if (TI.hasFeatureEnabled(OpenCLFeaturesMap, ExtAndFeat.first) !=
- TI.hasFeatureEnabled(OpenCLFeaturesMap, ExtAndFeat.second)) {
- IsValid = false;
- Diags.Report(diag::err_opencl_extension_and_feature_differs)
- << ExtAndFeat.first << ExtAndFeat.second;
- }
- return IsValid;
- }
- } // end namespace clang
|