/* * 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_export_var.h" #include "clang/AST/ASTContext.h" #include "clang/AST/Type.h" #include "llvm/ADT/APSInt.h" #include "slang_rs_context.h" #include "slang_rs_export_type.h" namespace slang { namespace { static clang::DiagnosticBuilder ReportVarError(RSContext *Context, const clang::SourceLocation Loc, const char *Message) { clang::DiagnosticsEngine *DiagEngine = Context->getDiagnostics(); const clang::SourceManager *SM = Context->getSourceManager(); return DiagEngine->Report(clang::FullSourceLoc(Loc, *SM), DiagEngine->getCustomDiagID(clang::DiagnosticsEngine::Error, Message)); } } // namespace RSExportVar::RSExportVar(RSContext *Context, const clang::VarDecl *VD, const RSExportType *ET) : RSExportable(Context, RSExportable::EX_VAR), mName(VD->getName().data(), VD->getName().size()), mET(ET), mIsConst(false), mIsUnsigned(false), mArraySize(0), mNumInits(0) { // mInit - Evaluate initializer expression const clang::Expr *Initializer = VD->getAnyInitializer(); if (Initializer != NULL) { switch (ET->getClass()) { case RSExportType::ExportClassPrimitive: case RSExportType::ExportClassVector: { Initializer->EvaluateAsRValue(mInit, Context->getASTContext()); break; } case RSExportType::ExportClassPointer: { if (Initializer->isNullPointerConstant(Context->getASTContext(), clang::Expr::NPC_ValueDependentIsNotNull)) { mInit.Val = clang::APValue(llvm::APSInt(1)); } else { if (!Initializer->EvaluateAsRValue(mInit, Context->getASTContext())) { ReportVarError(Context, Initializer->getExprLoc(), "initializer is not an R-value"); } } break; } case RSExportType::ExportClassConstantArray: { const clang::InitListExpr *IList = static_cast(Initializer); if (!IList) { ReportVarError(Context, VD->getLocation(), "Unable to find initializer list"); break; } const RSExportConstantArrayType *ECAT = static_cast(ET); mArraySize = ECAT->getSize(); mNumInits = IList->getNumInits(); for (unsigned int i = 0; i < mNumInits; i++) { clang::Expr::EvalResult tempInit; if (!IList->getInit(i)->EvaluateAsRValue(tempInit, Context->getASTContext())) { ReportVarError(Context, IList->getInit(i)->getExprLoc(), "initializer is not an R-value"); } mInitArray.push_back(tempInit); } break; } case RSExportType::ExportClassMatrix: case RSExportType::ExportClassRecord: { ReportVarError(Context, VD->getLocation(), "Reflection of initializer to variable '%0' (of type " "'%1') is unsupported currently.") << mName << ET->getName(); break; } default: { slangAssert(false && "Unknown class of type"); } } } clang::QualType QT = VD->getTypeSourceInfo()->getType(); if (!QT.isNull()) { mIsConst = QT.isConstQualified(); mIsUnsigned = QT->hasUnsignedIntegerRepresentation(); if (QT == Context->getASTContext().BoolTy) { mIsUnsigned = false; } } return; } } // namespace slang