1 /* 2 * Copyright 2010-2012, 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 #ifndef _FRAMEWORKS_COMPILE_SLANG_SLANG_RS_CONTEXT_H_ // NOLINT 18 #define _FRAMEWORKS_COMPILE_SLANG_SLANG_RS_CONTEXT_H_ 19 20 #include <cstdio> 21 #include <list> 22 #include <map> 23 #include <set> 24 #include <string> 25 #include <unordered_set> 26 #include <vector> 27 28 #include "clang/Lex/Preprocessor.h" 29 #include "clang/AST/Mangle.h" 30 31 #include "llvm/ADT/StringSet.h" 32 #include "llvm/ADT/StringMap.h" 33 34 #include "slang_pragma_list.h" 35 36 namespace llvm { 37 class LLVMContext; 38 class DataLayout; 39 } // namespace llvm 40 41 namespace clang { 42 class VarDecl; 43 class ASTContext; 44 class TargetInfo; 45 class FunctionDecl; 46 class QualType; 47 class SourceManager; 48 class TypeDecl; 49 class FunctionDecl; 50 } // namespace clang 51 52 namespace slang { 53 class Backend; 54 class RSExportable; 55 class RSExportVar; 56 class RSExportFunc; 57 class RSExportForEach; 58 class RSExportReduce; 59 class RSExportType; 60 61 class RSContext { 62 typedef llvm::StringSet<> NeedExportVarSet; 63 typedef llvm::StringSet<> NeedExportFuncSet; 64 typedef llvm::StringSet<> NeedExportTypeSet; 65 66 public: 67 typedef std::list<RSExportable*> ExportableList; 68 typedef std::list<RSExportVar*> ExportVarList; 69 typedef std::list<RSExportFunc*> ExportFuncList; 70 typedef std::vector<RSExportForEach*> ExportForEachVector; 71 typedef std::list<RSExportReduce*> ExportReduceList; 72 73 // WARNING: Sorted by pointer value, resulting in unpredictable order 74 typedef std::unordered_set<RSExportType*> ExportReduceResultTypeSet; 75 76 typedef llvm::StringMap<RSExportType*> ExportTypeMap; 77 78 private: 79 clang::Preprocessor &mPP; 80 clang::ASTContext &mCtx; 81 PragmaList *mPragmas; 82 // Precision specified via pragma, either rs_fp_full or rs_fp_relaxed. If 83 // empty, rs_fp_full is assumed. 84 std::string mPrecision; 85 unsigned int mTargetAPI; 86 bool mVerbose; 87 88 const llvm::DataLayout &mDataLayout; 89 llvm::LLVMContext &mLLVMContext; 90 91 ExportableList mExportables; 92 93 NeedExportTypeSet mNeedExportTypes; 94 95 std::string *mLicenseNote; 96 std::string mReflectJavaPackageName; 97 std::string mReflectJavaPathName; 98 99 std::string mRSPackageName; 100 101 int version; 102 103 std::unique_ptr<clang::MangleContext> mMangleCtx; 104 105 bool mIs64Bit; 106 107 bool processExportVar(const clang::VarDecl *VD); 108 bool processExportFunc(const clang::FunctionDecl *FD); 109 bool processExportType(const llvm::StringRef &Name); 110 111 int getForEachSlotNumber(const clang::StringRef& funcName); 112 unsigned mNextSlot; 113 114 ExportVarList mExportVars; 115 ExportFuncList mExportFuncs; 116 std::map<llvm::StringRef, unsigned> mExportForEachMap; 117 ExportForEachVector mExportForEach; 118 ExportForEachVector::iterator mFirstOldStyleKernel; 119 ExportReduceList mExportReduce; 120 ExportReduceResultTypeSet mExportReduceResultType; 121 ExportTypeMap mExportTypes; 122 123 clang::QualType mAllocationType; 124 clang::QualType mScriptCallType; 125 126 std::set<const clang::FunctionDecl *> mUsedByReducePragmaFns; 127 128 // Populated by markUsedByReducePragma(). 129 // Consumed by processReducePragmas(). 130 std::vector<clang::VarDecl *> mUsedByReducePragmaDummyVars; 131 132 public: 133 RSContext(clang::Preprocessor &PP, 134 clang::ASTContext &Ctx, 135 const clang::TargetInfo &Target, 136 PragmaList *Pragmas, 137 unsigned int TargetAPI, 138 bool Verbose); 139 140 enum CheckName { CheckNameNo, CheckNameYes }; 141 isSyntheticName(const llvm::StringRef Name)142 static bool isSyntheticName(const llvm::StringRef Name) { return Name.startswith(".rs."); } 143 getPreprocessor()144 inline clang::Preprocessor &getPreprocessor() const { return mPP; } getASTContext()145 inline clang::ASTContext &getASTContext() const { return mCtx; } getMangleContext()146 inline clang::MangleContext &getMangleContext() const { 147 return *mMangleCtx; 148 } getDataLayout()149 inline const llvm::DataLayout &getDataLayout() const { return mDataLayout; } getLLVMContext()150 inline llvm::LLVMContext &getLLVMContext() const { return mLLVMContext; } getSourceManager()151 inline const clang::SourceManager *getSourceManager() const { 152 return &mPP.getSourceManager(); 153 } getDiagnostics()154 inline clang::DiagnosticsEngine *getDiagnostics() const { 155 return &mPP.getDiagnostics(); 156 } getTargetAPI()157 inline unsigned int getTargetAPI() const { 158 return mTargetAPI; 159 } 160 getVerbose()161 inline bool getVerbose() const { 162 return mVerbose; 163 } is64Bit()164 inline bool is64Bit() const { 165 return mIs64Bit; 166 } 167 setLicenseNote(const std::string & S)168 inline void setLicenseNote(const std::string &S) { 169 mLicenseNote = new std::string(S); 170 } getLicenseNote()171 inline const std::string *getLicenseNote() const { return mLicenseNote; } 172 addExportType(const std::string & S)173 inline void addExportType(const std::string &S) { 174 mNeedExportTypes.insert(S); 175 } 176 setReflectJavaPackageName(const std::string & S)177 inline void setReflectJavaPackageName(const std::string &S) { 178 mReflectJavaPackageName = S; 179 } getReflectJavaPackageName()180 inline const std::string &getReflectJavaPackageName() const { 181 return mReflectJavaPackageName; 182 } 183 setRSPackageName(const std::string & S)184 inline void setRSPackageName(const std::string &S) { 185 mRSPackageName = S; 186 } 187 getRSPackageName()188 inline const std::string &getRSPackageName() const { return mRSPackageName; } 189 190 void setAllocationType(const clang::TypeDecl* TD); getAllocationType()191 inline const clang::QualType& getAllocationType() const { 192 return mAllocationType; 193 } 194 195 void setScriptCallType(const clang::TypeDecl* TD); getScriptCallType()196 inline const clang::QualType& getScriptCallType() const { 197 return mScriptCallType; 198 } 199 200 bool addForEach(const clang::FunctionDecl* FD); 201 bool processExports(); newExportable(RSExportable * E)202 inline void newExportable(RSExportable *E) { 203 if (E != nullptr) 204 mExportables.push_back(E); 205 } 206 typedef ExportableList::iterator exportable_iterator; exportable_begin()207 exportable_iterator exportable_begin() { 208 return mExportables.begin(); 209 } exportable_end()210 exportable_iterator exportable_end() { 211 return mExportables.end(); 212 } 213 214 typedef ExportVarList::const_iterator const_export_var_iterator; export_vars_begin()215 const_export_var_iterator export_vars_begin() const { 216 return mExportVars.begin(); 217 } export_vars_end()218 const_export_var_iterator export_vars_end() const { 219 return mExportVars.end(); 220 } hasExportVar()221 inline bool hasExportVar() const { 222 return !mExportVars.empty(); 223 } 224 225 typedef ExportFuncList::const_iterator const_export_func_iterator; export_funcs_begin()226 const_export_func_iterator export_funcs_begin() const { 227 return mExportFuncs.begin(); 228 } export_funcs_end()229 const_export_func_iterator export_funcs_end() const { 230 return mExportFuncs.end(); 231 } hasExportFunc()232 inline bool hasExportFunc() const { return !mExportFuncs.empty(); } 233 234 typedef ExportForEachVector::const_iterator const_export_foreach_iterator; export_foreach_begin()235 const_export_foreach_iterator export_foreach_begin() const { 236 return mExportForEach.begin(); 237 } export_foreach_end()238 const_export_foreach_iterator export_foreach_end() const { 239 return mExportForEach.end(); 240 } hasExportForEach()241 inline bool hasExportForEach() const { return !mExportForEach.empty(); } 242 int getForEachSlotNumber(const clang::FunctionDecl* FD); 243 244 typedef ExportReduceList::const_iterator const_export_reduce_iterator; export_reduce_begin()245 const_export_reduce_iterator export_reduce_begin() const { 246 return mExportReduce.begin(); 247 } export_reduce_end()248 const_export_reduce_iterator export_reduce_end() const { 249 return mExportReduce.end(); 250 } hasExportReduce()251 inline bool hasExportReduce() const { return !mExportReduce.empty(); } addExportReduce(RSExportReduce * Reduce)252 void addExportReduce(RSExportReduce *Reduce) { 253 mExportReduce.push_back(Reduce); 254 } 255 bool processReducePragmas(Backend *BE); 256 void markUsedByReducePragma(clang::FunctionDecl *FD, CheckName Check); 257 258 // If the type has already been inserted, has no effect. insertExportReduceResultType(RSExportType * Type)259 void insertExportReduceResultType(RSExportType *Type) { mExportReduceResultType.insert(Type); } 260 261 template <class FilterIn, class Compare> getReduceResultTypes(FilterIn Filt,Compare Comp)262 std::vector<RSExportType *> getReduceResultTypes(FilterIn Filt, Compare Comp) const { 263 std::vector<RSExportType *> Return; 264 std::copy_if(mExportReduceResultType.begin(), mExportReduceResultType.end(), std::back_inserter(Return), Filt); 265 std::sort(Return.begin(), Return.end(), Comp); 266 auto ReturnNewEndIter = std::unique(Return.begin(), Return.end(), 267 [Comp](const RSExportType *a, const RSExportType *b) { 268 return !Comp(a, b) && !Comp(b, a); 269 }); 270 Return.erase(ReturnNewEndIter, Return.end()); 271 return Return; 272 } 273 274 typedef ExportTypeMap::iterator export_type_iterator; 275 typedef ExportTypeMap::const_iterator const_export_type_iterator; export_types_begin()276 export_type_iterator export_types_begin() { return mExportTypes.begin(); } export_types_end()277 export_type_iterator export_types_end() { return mExportTypes.end(); } export_types_begin()278 const_export_type_iterator export_types_begin() const { 279 return mExportTypes.begin(); 280 } export_types_end()281 const_export_type_iterator export_types_end() const { 282 return mExportTypes.end(); 283 } hasExportType()284 inline bool hasExportType() const { return !mExportTypes.empty(); } findExportType(const llvm::StringRef & TypeName)285 export_type_iterator findExportType(const llvm::StringRef &TypeName) { 286 return mExportTypes.find(TypeName); 287 } findExportType(const llvm::StringRef & TypeName)288 const_export_type_iterator findExportType(const llvm::StringRef &TypeName) 289 const { 290 return mExportTypes.find(TypeName); 291 } 292 293 // Insert the specified Typename/Type pair into the map. If the key already 294 // exists in the map, return false and ignore the request, otherwise insert it 295 // and return true. 296 bool insertExportType(const llvm::StringRef &TypeName, RSExportType *Type); 297 getVersion()298 int getVersion() const { return version; } setVersion(int v)299 void setVersion(int v) { 300 version = v; 301 } 302 isCompatLib()303 bool isCompatLib() const { 304 // If we are not targeting the actual Android Renderscript classes, 305 // we should reflect code that works with the compatibility library. 306 return (mRSPackageName.compare("android.renderscript") != 0); 307 } 308 addPragma(const std::string & T,const std::string & V)309 void addPragma(const std::string &T, const std::string &V) { 310 mPragmas->push_back(make_pair(T, V)); 311 } setPrecision(const std::string & P)312 void setPrecision(const std::string &P) { mPrecision = P; } getPrecision()313 std::string getPrecision() { return mPrecision; } 314 315 // Report an error or a warning to the user. 316 template <unsigned N> Report(clang::DiagnosticsEngine::Level Level,const char (& Message)[N])317 clang::DiagnosticBuilder Report(clang::DiagnosticsEngine::Level Level, 318 const char (&Message)[N]) { 319 clang::DiagnosticsEngine *DiagEngine = getDiagnostics(); 320 return DiagEngine->Report(DiagEngine->getCustomDiagID(Level, Message)); 321 } 322 323 template <unsigned N> Report(clang::DiagnosticsEngine::Level Level,const clang::SourceLocation Loc,const char (& Message)[N])324 clang::DiagnosticBuilder Report(clang::DiagnosticsEngine::Level Level, 325 const clang::SourceLocation Loc, 326 const char (&Message)[N]) { 327 clang::DiagnosticsEngine *DiagEngine = getDiagnostics(); 328 const clang::SourceManager *SM = getSourceManager(); 329 return DiagEngine->Report(clang::FullSourceLoc(Loc, *SM), 330 DiagEngine->getCustomDiagID(Level, Message)); 331 } 332 333 // Utility functions to report errors and warnings to make the calling code 334 // easier to read. 335 template <unsigned N> ReportError(const char (& Message)[N])336 clang::DiagnosticBuilder ReportError(const char (&Message)[N]) { 337 return Report<N>(clang::DiagnosticsEngine::Error, Message); 338 } 339 340 template <unsigned N> ReportError(const clang::SourceLocation Loc,const char (& Message)[N])341 clang::DiagnosticBuilder ReportError(const clang::SourceLocation Loc, 342 const char (&Message)[N]) { 343 return Report<N>(clang::DiagnosticsEngine::Error, Loc, Message); 344 } 345 346 template <unsigned N> ReportWarning(const char (& Message)[N])347 clang::DiagnosticBuilder ReportWarning(const char (&Message)[N]) { 348 return Report<N>(clang::DiagnosticsEngine::Warning, Message); 349 } 350 351 template <unsigned N> ReportWarning(const clang::SourceLocation Loc,const char (& Message)[N])352 clang::DiagnosticBuilder ReportWarning(const clang::SourceLocation Loc, 353 const char (&Message)[N]) { 354 return Report<N>(clang::DiagnosticsEngine::Warning, Loc, Message); 355 } 356 357 ~RSContext(); 358 }; 359 360 } // namespace slang 361 362 #endif // _FRAMEWORKS_COMPILE_SLANG_SLANG_RS_CONTEXT_H_ NOLINT 363