/* * Copyright 2010-2012, 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. */ #include "slang_rs.h" #include #include #include #include #include #include #include "clang/Basic/SourceLocation.h" #include "clang/Frontend/FrontendDiagnostic.h" #include "clang/Sema/SemaDiagnostic.h" #include "llvm/Support/Path.h" #include "os_sep.h" #include "rs_cc_options.h" #include "slang_rs_backend.h" #include "slang_rs_context.h" #include "slang_rs_export_type.h" #include "slang_rs_reflection.h" #include "slang_rs_reflection_cpp.h" namespace slang { #define FS_SUFFIX "fs" #define RS_HEADER_SUFFIX "rsh" /* RS_HEADER_ENTRY(name) */ #define ENUM_RS_HEADER() \ RS_HEADER_ENTRY(rs_allocation) \ RS_HEADER_ENTRY(rs_atomic) \ RS_HEADER_ENTRY(rs_cl) \ RS_HEADER_ENTRY(rs_core) \ RS_HEADER_ENTRY(rs_core_math) \ RS_HEADER_ENTRY(rs_debug) \ RS_HEADER_ENTRY(rs_element) \ RS_HEADER_ENTRY(rs_graphics) \ RS_HEADER_ENTRY(rs_math) \ RS_HEADER_ENTRY(rs_mesh) \ RS_HEADER_ENTRY(rs_matrix) \ RS_HEADER_ENTRY(rs_object) \ RS_HEADER_ENTRY(rs_program) \ RS_HEADER_ENTRY(rs_quaternion) \ RS_HEADER_ENTRY(rs_sampler) \ RS_HEADER_ENTRY(rs_time) \ RS_HEADER_ENTRY(rs_types) \ // Returns true if \p Filename ends in ".fs". bool SlangRS::isFilterscript(const char *Filename) { const char *c = strrchr(Filename, '.'); if (c && !strncmp(FS_SUFFIX, c + 1, strlen(FS_SUFFIX) + 1)) { return true; } else { return false; } } bool SlangRS::generateJavaBitcodeAccessor(const std::string &OutputPathBase, const std::string &PackageName, const std::string *LicenseNote) { RSSlangReflectUtils::BitCodeAccessorContext BCAccessorContext; BCAccessorContext.rsFileName = getInputFileName().c_str(); BCAccessorContext.bc32FileName = getOutput32FileName().c_str(); BCAccessorContext.bc64FileName = getOutputFileName().c_str(); BCAccessorContext.reflectPath = OutputPathBase.c_str(); BCAccessorContext.packageName = PackageName.c_str(); BCAccessorContext.licenseNote = LicenseNote; BCAccessorContext.bcStorage = BCST_JAVA_CODE; // Must be BCST_JAVA_CODE BCAccessorContext.verbose = false; return RSSlangReflectUtils::GenerateJavaBitCodeAccessor(BCAccessorContext); } bool SlangRS::checkODR(const char *CurInputFile) { for (RSContext::ExportableList::iterator I = mRSContext->exportable_begin(), E = mRSContext->exportable_end(); I != E; I++) { RSExportable *RSE = *I; if (RSE->getKind() != RSExportable::EX_TYPE) continue; RSExportType *ET = static_cast(RSE); if (ET->getClass() != RSExportType::ExportClassRecord) continue; RSExportRecordType *ERT = static_cast(ET); // Artificial record types (create by us not by user in the source) always // conforms the ODR. if (ERT->isArtificial()) continue; // Key to lookup ERT in ReflectedDefinitions llvm::StringRef RDKey(ERT->getName()); ReflectedDefinitionListTy::const_iterator RD = ReflectedDefinitions.find(RDKey); if (RD != ReflectedDefinitions.end()) { const RSExportRecordType *Reflected = RD->getValue().first; // There's a record (struct) with the same name reflected before. Enforce // ODR checking - the Reflected must hold *exactly* the same "definition" // as the one defined previously. We say two record types A and B have the // same definition iff: // // struct A { struct B { // Type(a1) a1, Type(b1) b1, // Type(a2) a2, Type(b1) b2, // ... ... // Type(aN) aN Type(b3) b3, // }; } // Cond. #1. They have same number of fields, i.e., N = M; // Cond. #2. for (i := 1 to N) // Type(ai) = Type(bi) must hold; // Cond. #3. for (i := 1 to N) // Name(ai) = Name(bi) must hold; // // where, // Type(F) = the type of field F and // Name(F) = the field name. bool PassODR = false; // Cond. #1 and Cond. #2 if (Reflected->equals(ERT)) { // Cond #3. RSExportRecordType::const_field_iterator AI = Reflected->fields_begin(), BI = ERT->fields_begin(); for (unsigned i = 0, e = Reflected->getFields().size(); i != e; i++) { if ((*AI)->getName() != (*BI)->getName()) break; AI++; BI++; } PassODR = (AI == (Reflected->fields_end())); } if (!PassODR) { getDiagnostics().Report(mDiagErrorODR) << Reflected->getName() << getInputFileName() << RD->getValue().second; return false; } } else { llvm::StringMapEntry *ME = llvm::StringMapEntry::Create(RDKey); ME->setValue(std::make_pair(ERT, CurInputFile)); if (!ReflectedDefinitions.insert(ME)) delete ME; // Take the ownership of ERT such that it won't be freed in ~RSContext(). ERT->keep(); } } return true; } void SlangRS::initDiagnostic() { clang::DiagnosticsEngine &DiagEngine = getDiagnostics(); if (DiagEngine.setSeverityForGroup("implicit-function-declaration", clang::diag::Severity::Error)) DiagEngine.Report(clang::diag::warn_unknown_warning_option) << "implicit-function-declaration"; DiagEngine.setSeverity( clang::diag::ext_typecheck_convert_discards_qualifiers, clang::diag::Severity::Error, clang::SourceLocation()); mDiagErrorInvalidOutputDepParameter = DiagEngine.getCustomDiagID( clang::DiagnosticsEngine::Error, "invalid parameter for output dependencies files."); mDiagErrorODR = DiagEngine.getCustomDiagID( clang::DiagnosticsEngine::Error, "type '%0' in different translation unit (%1 v.s. %2) " "has incompatible type definition"); mDiagErrorTargetAPIRange = DiagEngine.getCustomDiagID( clang::DiagnosticsEngine::Error, "target API level '%0' is out of range ('%1' - '%2')"); } void SlangRS::initPreprocessor() { clang::Preprocessor &PP = getPreprocessor(); std::stringstream RSH; RSH << PP.getPredefines(); RSH << "#define RS_VERSION " << mTargetAPI << "\n"; RSH << "#include \"rs_core." RS_HEADER_SUFFIX "\"\n"; PP.setPredefines(RSH.str()); } void SlangRS::initASTContext() { mRSContext = new RSContext(getPreprocessor(), getASTContext(), getTargetInfo(), &mPragmas, mTargetAPI, mVerbose); } clang::ASTConsumer *SlangRS::createBackend(const clang::CodeGenOptions& CodeGenOpts, llvm::raw_ostream *OS, Slang::OutputType OT) { return new RSBackend(mRSContext, &getDiagnostics(), CodeGenOpts, getTargetOptions(), &mPragmas, OS, OT, getSourceManager(), mAllowRSPrefix, mIsFilterscript); } bool SlangRS::IsRSHeaderFile(const char *File) { #define RS_HEADER_ENTRY(name) \ if (::strcmp(File, #name "." RS_HEADER_SUFFIX) == 0) \ return true; ENUM_RS_HEADER() #undef RS_HEADER_ENTRY return false; } bool SlangRS::IsLocInRSHeaderFile(const clang::SourceLocation &Loc, const clang::SourceManager &SourceMgr) { clang::FullSourceLoc FSL(Loc, SourceMgr); clang::PresumedLoc PLoc = SourceMgr.getPresumedLoc(FSL); const char *Filename = PLoc.getFilename(); if (!Filename) { return false; } else { return IsRSHeaderFile(llvm::sys::path::filename(Filename).data()); } } SlangRS::SlangRS() : Slang(), mRSContext(NULL), mAllowRSPrefix(false), mTargetAPI(0), mVerbose(false), mIsFilterscript(false) { } bool SlangRS::compile( const std::list > &IOFiles64, const std::list > &IOFiles32, const std::list > &DepFiles, const RSCCOptions &Opts) { if (IOFiles32.empty()) return true; if (Opts.mEmitDependency && (DepFiles.size() != IOFiles32.size())) { getDiagnostics().Report(mDiagErrorInvalidOutputDepParameter); return false; } if (Opts.mEmit3264 && (IOFiles64.size() != IOFiles32.size())) { slangAssert(false && "Should have equal number of 32/64-bit files"); return false; } std::string RealPackageName; const char *InputFile, *Output64File, *Output32File, *BCOutputFile, *DepOutputFile; std::list >::const_iterator IOFile64Iter = IOFiles64.begin(), IOFile32Iter = IOFiles32.begin(), DepFileIter = DepFiles.begin(); setIncludePaths(Opts.mIncludePaths); setOutputType(Opts.mOutputType); if (Opts.mEmitDependency) { setAdditionalDepTargets(Opts.mAdditionalDepTargets); } setDebugMetadataEmission(Opts.mDebugEmission); setOptimizationLevel(Opts.mOptimizationLevel); mAllowRSPrefix = Opts.mAllowRSPrefix; mTargetAPI = Opts.mTargetAPI; if (mTargetAPI != SLANG_DEVELOPMENT_TARGET_API && (mTargetAPI < SLANG_MINIMUM_TARGET_API || mTargetAPI > SLANG_MAXIMUM_TARGET_API)) { getDiagnostics().Report(mDiagErrorTargetAPIRange) << mTargetAPI << SLANG_MINIMUM_TARGET_API << SLANG_MAXIMUM_TARGET_API; return false; } mVerbose = Opts.mVerbose; // Skip generation of warnings a second time if we are doing more than just // a single pass over the input file. bool SuppressAllWarnings = (Opts.mOutputType != Slang::OT_Dependency); bool CompileSecondTimeFor64Bit = Opts.mEmit3264 && Opts.mBitWidth == 64; for (unsigned i = 0, e = IOFiles32.size(); i != e; i++) { InputFile = IOFile64Iter->first; Output64File = IOFile64Iter->second; Output32File = IOFile32Iter->second; // We suppress warnings (via reset) if we are doing a second compilation. reset(CompileSecondTimeFor64Bit); if (!setInputSource(InputFile)) return false; if (!setOutput(Output64File)) return false; setOutput32(Output32File); mIsFilterscript = isFilterscript(InputFile); if (Slang::compile() > 0) return false; if (!Opts.mJavaReflectionPackageName.empty()) { mRSContext->setReflectJavaPackageName(Opts.mJavaReflectionPackageName); } const std::string &RealPackageName = mRSContext->getReflectJavaPackageName(); bool doReflection = true; if (Opts.mEmit3264 && (Opts.mBitWidth == 32)) { // Skip reflection on the 32-bit path if we are going to emit it on the // 64-bit path. doReflection = false; } if (Opts.mOutputType != Slang::OT_Dependency && doReflection) { if (Opts.mBitcodeStorage == BCST_CPP_CODE) { const std::string &outputFileName = (Opts.mBitWidth == 64) ? getOutputFileName() : getOutput32FileName(); RSReflectionCpp R(mRSContext, Opts.mJavaReflectionPathBase, getInputFileName(), outputFileName); if (!R.reflect()) { return false; } } else { if (!Opts.mRSPackageName.empty()) { mRSContext->setRSPackageName(Opts.mRSPackageName); } RSReflectionJava R(mRSContext, &mGeneratedFileNames, Opts.mJavaReflectionPathBase, getInputFileName(), getOutputFileName(), Opts.mBitcodeStorage == BCST_JAVA_CODE); if (!R.reflect()) { // TODO Is this needed or will the error message have been printed // already? and why not for the C++ case? fprintf(stderr, "RSContext::reflectToJava : failed to do reflection " "(%s)\n", R.getLastError()); return false; } for (std::vector::const_iterator I = mGeneratedFileNames.begin(), E = mGeneratedFileNames.end(); I != E; I++) { std::string ReflectedName = RSSlangReflectUtils::ComputePackagedPath( Opts.mJavaReflectionPathBase.c_str(), (RealPackageName + OS_PATH_SEPARATOR_STR + *I).c_str()); appendGeneratedFileName(ReflectedName + ".java"); } if ((Opts.mOutputType == Slang::OT_Bitcode) && (Opts.mBitcodeStorage == BCST_JAVA_CODE) && !generateJavaBitcodeAccessor(Opts.mJavaReflectionPathBase, RealPackageName.c_str(), mRSContext->getLicenseNote())) { return false; } } } if (Opts.mEmitDependency) { BCOutputFile = DepFileIter->first; DepOutputFile = DepFileIter->second; setDepTargetBC(BCOutputFile); if (!setDepOutput(DepOutputFile)) return false; if (SuppressAllWarnings) { getDiagnostics().setSuppressAllDiagnostics(true); } if (generateDepFile() > 0) return false; if (SuppressAllWarnings) { getDiagnostics().setSuppressAllDiagnostics(false); } DepFileIter++; } if (!checkODR(InputFile)) return false; IOFile64Iter++; IOFile32Iter++; } return true; } void SlangRS::reset(bool SuppressWarnings) { delete mRSContext; mRSContext = NULL; mGeneratedFileNames.clear(); Slang::reset(SuppressWarnings); } SlangRS::~SlangRS() { delete mRSContext; for (ReflectedDefinitionListTy::iterator I = ReflectedDefinitions.begin(), E = ReflectedDefinitions.end(); I != E; I++) { delete I->getValue().first; } } } // namespace slang