forked from OSchip/llvm-project
				
			
		
			
				
	
	
		
			192 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			192 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C++
		
	
	
	
| //===-- ASTStructExtractor.cpp ----------------------------------*- C++ -*-===//
 | |
| //
 | |
| //                     The LLVM Compiler Infrastructure
 | |
| //
 | |
| // This file is distributed under the University of Illinois Open Source
 | |
| // License. See LICENSE.TXT for details.
 | |
| //
 | |
| //===----------------------------------------------------------------------===//
 | |
| 
 | |
| #include "stdlib.h"
 | |
| #include "clang/AST/ASTContext.h"
 | |
| #include "clang/AST/Decl.h"
 | |
| #include "clang/AST/DeclCXX.h"
 | |
| #include "clang/AST/DeclGroup.h"
 | |
| #include "clang/AST/Expr.h"
 | |
| #include "clang/AST/RecordLayout.h"
 | |
| #include "clang/AST/Stmt.h"
 | |
| #include "clang/Parse/Parser.h"
 | |
| #include "clang/Sema/Sema.h"
 | |
| #include "llvm/Support/Casting.h"
 | |
| #include "llvm/Support/raw_ostream.h"
 | |
| #include "lldb/Core/Log.h"
 | |
| #include "lldb/Expression/ASTStructExtractor.h"
 | |
| 
 | |
| using namespace llvm;
 | |
| using namespace clang;
 | |
| using namespace lldb_private;
 | |
| 
 | |
| ASTStructExtractor::ASTStructExtractor(ASTConsumer *passthrough,
 | |
|                                        const char *struct_name,
 | |
|                                        ClangFunction &function) :
 | |
|     m_ast_context (NULL),
 | |
|     m_passthrough (passthrough),
 | |
|     m_passthrough_sema (NULL),
 | |
|     m_sema (NULL),
 | |
|     m_action (NULL),
 | |
|     m_function (function),
 | |
|     m_struct_name (struct_name)
 | |
| {
 | |
|     if (!m_passthrough)
 | |
|         return;
 | |
|     
 | |
|     m_passthrough_sema = dyn_cast<SemaConsumer>(passthrough);
 | |
| }
 | |
| 
 | |
| ASTStructExtractor::~ASTStructExtractor()
 | |
| {
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::Initialize(ASTContext &Context) 
 | |
| {
 | |
|     m_ast_context = &Context;
 | |
|     
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->Initialize(Context);
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::ExtractFromFunctionDecl(FunctionDecl *F)
 | |
| {
 | |
|     DeclarationName struct_name(&m_ast_context->Idents.get(m_struct_name.c_str()));
 | |
|     RecordDecl::lookup_result struct_lookup = F->lookup(struct_name);
 | |
|     
 | |
|     if (struct_lookup.first == struct_lookup.second)
 | |
|         return;
 | |
|     
 | |
|     RecordDecl *struct_decl = dyn_cast<RecordDecl>(*(struct_lookup.first));
 | |
|     
 | |
|     if (!struct_decl)
 | |
|         return;
 | |
|     
 | |
|     const ASTRecordLayout* struct_layout(&m_ast_context->getASTRecordLayout (struct_decl));
 | |
|     
 | |
|     if (!struct_layout)
 | |
|         return;
 | |
|     
 | |
|     m_function.m_struct_size = struct_layout->getSize().getQuantity(); // TODO Store m_struct_size as CharUnits   
 | |
|     m_function.m_return_offset = struct_layout->getFieldOffset(struct_layout->getFieldCount() - 1) / 8;
 | |
|     m_function.m_return_size = struct_layout->getDataSize().getQuantity() - m_function.m_return_offset;
 | |
|     
 | |
|     for (unsigned field_index = 0, num_fields = struct_layout->getFieldCount();
 | |
|          field_index < num_fields;
 | |
|          ++field_index)
 | |
|     {
 | |
|         m_function.m_member_offsets.push_back(struct_layout->getFieldOffset(field_index) / 8);
 | |
|     }
 | |
|     
 | |
|     m_function.m_struct_valid = true;
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::ExtractFromTopLevelDecl(Decl* D)
 | |
| {
 | |
|     LinkageSpecDecl *linkage_spec_decl = dyn_cast<LinkageSpecDecl>(D);
 | |
|     
 | |
|     if (linkage_spec_decl)
 | |
|     {
 | |
|         RecordDecl::decl_iterator decl_iterator;
 | |
|         
 | |
|         for (decl_iterator = linkage_spec_decl->decls_begin();
 | |
|              decl_iterator != linkage_spec_decl->decls_end();
 | |
|              ++decl_iterator)
 | |
|         {
 | |
|             ExtractFromTopLevelDecl(*decl_iterator);
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     FunctionDecl *function_decl = dyn_cast<FunctionDecl>(D);
 | |
|     
 | |
|     if (m_ast_context &&
 | |
|         function_decl &&
 | |
|         !m_function.m_wrapper_function_name.compare(function_decl->getNameAsString().c_str()))
 | |
|     {
 | |
|         ExtractFromFunctionDecl(function_decl);
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool 
 | |
| ASTStructExtractor::HandleTopLevelDecl(DeclGroupRef D)
 | |
| {
 | |
|     DeclGroupRef::iterator decl_iterator;
 | |
|     
 | |
|     for (decl_iterator = D.begin();
 | |
|          decl_iterator != D.end();
 | |
|          ++decl_iterator)
 | |
|     {
 | |
|         Decl *decl = *decl_iterator;
 | |
|         
 | |
|         ExtractFromTopLevelDecl(decl);
 | |
|     }
 | |
|     
 | |
|     if (m_passthrough)
 | |
|         return m_passthrough->HandleTopLevelDecl(D);
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::HandleTranslationUnit(ASTContext &Ctx)
 | |
| {    
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->HandleTranslationUnit(Ctx);
 | |
| }
 | |
| 
 | |
| void 
 | |
| ASTStructExtractor::HandleTagDeclDefinition(TagDecl *D)
 | |
| {
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->HandleTagDeclDefinition(D);
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::CompleteTentativeDefinition(VarDecl *D)
 | |
| {
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->CompleteTentativeDefinition(D);
 | |
| }
 | |
| 
 | |
| void 
 | |
| ASTStructExtractor::HandleVTable(CXXRecordDecl *RD, bool DefinitionRequired) 
 | |
| {
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->HandleVTable(RD, DefinitionRequired);
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::PrintStats() 
 | |
| {
 | |
|     if (m_passthrough)
 | |
|         m_passthrough->PrintStats();
 | |
| }
 | |
| 
 | |
| void
 | |
| ASTStructExtractor::InitializeSema(Sema &S)
 | |
| {
 | |
|     m_sema = &S;
 | |
|     m_action = reinterpret_cast<Action*>(m_sema);
 | |
|     
 | |
|     if (m_passthrough_sema)
 | |
|         m_passthrough_sema->InitializeSema(S);
 | |
| }
 | |
| 
 | |
| void 
 | |
| ASTStructExtractor::ForgetSema() 
 | |
| {
 | |
|     m_sema = NULL;
 | |
|     m_action = NULL;
 | |
|     
 | |
|     if (m_passthrough_sema)
 | |
|         m_passthrough_sema->ForgetSema();
 | |
| }
 |