| /* |
| * Copyright 2010, 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_context.h" |
| |
| #include <string> |
| |
| #include "clang/AST/ASTContext.h" |
| #include "clang/AST/Decl.h" |
| #include "clang/AST/DeclBase.h" |
| #include "clang/AST/Type.h" |
| |
| #include "clang/Basic/Linkage.h" |
| #include "clang/Basic/TargetInfo.h" |
| |
| #include "clang/Index/ASTLocation.h" |
| |
| #include "llvm/LLVMContext.h" |
| |
| #include "llvm/Target/TargetData.h" |
| |
| #include "slang.h" |
| #include "slang_rs_export_func.h" |
| #include "slang_rs_export_type.h" |
| #include "slang_rs_export_var.h" |
| #include "slang_rs_exportable.h" |
| #include "slang_rs_pragma_handler.h" |
| #include "slang_rs_reflection.h" |
| |
| namespace slang { |
| |
| RSContext::RSContext(clang::Preprocessor &PP, |
| clang::ASTContext &Ctx, |
| const clang::TargetInfo &Target) |
| : mPP(PP), |
| mCtx(Ctx), |
| mTarget(Target), |
| mTargetData(NULL), |
| mLLVMContext(llvm::getGlobalContext()), |
| mLicenseNote(NULL), |
| version(0) { |
| // For #pragma rs export_type |
| PP.AddPragmaHandler( |
| "rs", RSPragmaHandler::CreatePragmaExportTypeHandler(this)); |
| |
| // For #pragma rs java_package_name |
| PP.AddPragmaHandler( |
| "rs", RSPragmaHandler::CreatePragmaJavaPackageNameHandler(this)); |
| |
| // For #pragma rs set_reflect_license |
| PP.AddPragmaHandler( |
| "rs", RSPragmaHandler::CreatePragmaReflectLicenseHandler(this)); |
| |
| // For #pragma version |
| PP.AddPragmaHandler(RSPragmaHandler::CreatePragmaVersionHandler(this)); |
| |
| // Prepare target data |
| mTargetData = new llvm::TargetData(Target.getTargetDescription()); |
| |
| return; |
| } |
| |
| bool RSContext::processExportVar(const clang::VarDecl *VD) { |
| assert(!VD->getName().empty() && "Variable name should not be empty"); |
| |
| // TODO(zonr): some check on variable |
| |
| RSExportType *ET = RSExportType::CreateFromDecl(this, VD); |
| if (!ET) |
| return false; |
| |
| RSExportVar *EV = new RSExportVar(this, VD, ET); |
| if (EV == NULL) |
| return false; |
| else |
| mExportVars.push_back(EV); |
| |
| return true; |
| } |
| |
| static bool isSpecialRSFunc(const llvm::StringRef& Name) { |
| static llvm::StringRef FuncInit("init"); |
| static llvm::StringRef FuncRoot("root"); |
| return Name.equals(FuncInit) || Name.equals(FuncRoot); |
| } |
| |
| bool RSContext::processExportFunc(const clang::FunctionDecl *FD) { |
| assert(!FD->getName().empty() && "Function name should not be empty"); |
| |
| if (!FD->isThisDeclarationADefinition()) { |
| return true; |
| } |
| |
| if (FD->getStorageClass() != clang::SC_None) { |
| fprintf(stderr, "RSContext::processExportFunc : cannot export extern or " |
| "static function '%s'\n", FD->getName().str().c_str()); |
| return false; |
| } |
| |
| // Do not reflect specialized RS functions like init/root. |
| if (isSpecialRSFunc(FD->getName())) { |
| return true; |
| } |
| |
| RSExportFunc *EF = RSExportFunc::Create(this, FD); |
| if (EF == NULL) |
| return false; |
| else |
| mExportFuncs.push_back(EF); |
| |
| return true; |
| } |
| |
| |
| bool RSContext::processExportType(const llvm::StringRef &Name) { |
| clang::TranslationUnitDecl *TUDecl = mCtx.getTranslationUnitDecl(); |
| |
| assert(TUDecl != NULL && "Translation unit declaration (top-level " |
| "declaration) is null object"); |
| |
| const clang::IdentifierInfo *II = mPP.getIdentifierInfo(Name); |
| if (II == NULL) |
| // TODO(zonr): alert identifier @Name mark as an exportable type cannot be |
| // found |
| return false; |
| |
| clang::DeclContext::lookup_const_result R = TUDecl->lookup(II); |
| RSExportType *ET = NULL; |
| |
| for (clang::DeclContext::lookup_const_iterator I = R.first, E = R.second; |
| I != E; |
| I++) { |
| clang::NamedDecl *const ND = *I; |
| const clang::Type *T = NULL; |
| |
| switch (ND->getKind()) { |
| case clang::Decl::Typedef: { |
| T = static_cast<const clang::TypedefDecl*>( |
| ND)->getCanonicalDecl()->getUnderlyingType().getTypePtr(); |
| break; |
| } |
| case clang::Decl::Record: { |
| T = static_cast<const clang::RecordDecl*>(ND)->getTypeForDecl(); |
| break; |
| } |
| default: { |
| // unsupported, skip |
| break; |
| } |
| } |
| |
| if (T != NULL) |
| ET = RSExportType::Create(this, T); |
| } |
| |
| return (ET != NULL); |
| } |
| |
| bool RSContext::processExport() { |
| // Export variable |
| clang::TranslationUnitDecl *TUDecl = mCtx.getTranslationUnitDecl(); |
| for (clang::DeclContext::decl_iterator DI = TUDecl->decls_begin(), |
| DE = TUDecl->decls_end(); |
| DI != DE; |
| DI++) { |
| if (DI->getKind() == clang::Decl::Var) { |
| clang::VarDecl *VD = (clang::VarDecl*) (*DI); |
| if (VD->getLinkage() == clang::ExternalLinkage) { |
| if (!processExportVar(VD)) { |
| fprintf(stderr, "RSContext::processExport : failed to export var " |
| "'%s'\n", VD->getNameAsString().c_str()); |
| return false; |
| } |
| } |
| } else if (DI->getKind() == clang::Decl::Function) { |
| // Export functions |
| clang::FunctionDecl *FD = (clang::FunctionDecl*) (*DI); |
| if (FD->getLinkage() == clang::ExternalLinkage) { |
| if (!processExportFunc(FD)) { |
| fprintf(stderr, "RSContext::processExport : failed to export func " |
| "'%s'\n", FD->getNameAsString().c_str()); |
| return false; |
| } |
| } |
| } |
| } |
| |
| // Finally, export type forcely set to be exported by user |
| for (NeedExportTypeSet::const_iterator EI = mNeedExportTypes.begin(), |
| EE = mNeedExportTypes.end(); |
| EI != EE; |
| EI++) { |
| if (!processExportType(EI->getKey())) { |
| fprintf(stderr, "RSContext::processExport : failed to export type " |
| "'%s'\n", EI->getKey().str().c_str()); |
| return false; |
| } |
| } |
| |
| return true; |
| } |
| |
| bool RSContext::insertExportType(const llvm::StringRef &TypeName, |
| RSExportType *ET) { |
| ExportTypeMap::value_type *NewItem = |
| ExportTypeMap::value_type::Create(TypeName.begin(), |
| TypeName.end(), |
| mExportTypes.getAllocator(), |
| ET); |
| |
| if (mExportTypes.insert(NewItem)) { |
| return true; |
| } else { |
| free(NewItem); |
| return false; |
| } |
| } |
| |
| bool RSContext::reflectToJava(const std::string &OutputPathBase, |
| const std::string &OutputPackageName, |
| const std::string &InputFileName, |
| const std::string &OutputBCFileName, |
| std::string *RealPackageName) { |
| if (RealPackageName != NULL) |
| RealPackageName->clear(); |
| |
| const std::string &PackageName = |
| ((OutputPackageName.empty()) ? mReflectJavaPackageName : |
| OutputPackageName); |
| if (PackageName.empty()) { |
| std::cerr << "Error: Missing \"#pragma rs " |
| << "java_package_name(com.foo.bar)\" in " |
| << InputFileName << std::endl; |
| return false; |
| } |
| |
| // Copy back the really applied package name |
| RealPackageName->assign(PackageName); |
| |
| RSReflection *R = new RSReflection(this); |
| bool ret = R->reflect(OutputPathBase, PackageName, |
| InputFileName, OutputBCFileName); |
| if (!ret) |
| fprintf(stderr, "RSContext::reflectToJava : failed to do reflection " |
| "(%s)\n", R->getLastError()); |
| delete R; |
| return ret; |
| } |
| |
| RSContext::~RSContext() { |
| delete mLicenseNote; |
| delete mTargetData; |
| for (ExportableList::iterator I = mExportables.begin(), |
| E = mExportables.end(); |
| I != E; |
| I++) { |
| if (!(*I)->isKeep()) |
| delete *I; |
| } |
| } |
| |
| } // namespace slang |