forked from OSchip/llvm-project
Complete complete types early when importing types from Clang module DWARF.
This affects -gmodules only.
Under normal operation pcm_type is a shallow forward declaration
that gets completed later. This is necessary to support cyclic
data structures. If, however, pcm_type is already complete (for
example, because it was loaded for a different target before),
the definition needs to be imported right away, too.
Type::ResolveClangType() effectively ignores the ResolveState
inside type_sp and only looks at IsDefined(), so it never calls
ClangASTImporter::ASTImporterDelegate::ImportDefinitionTo(),
which does extra work for Objective-C classes. This would result
in only the forward declaration to be visible.
An alternative implementation would be to sink this into Type::ResolveClangType ( 88235812a7/lldb/source/Symbol/Type.cpp (L5809)
) though it isn't clear to me how to best do this from a layering perspective.
rdar://problem/52134074
Differential Revision: https://reviews.llvm.org/D70415
This commit is contained in:
parent
c0eeea5d74
commit
539117616d
|
@ -0,0 +1,5 @@
|
|||
CFLAGS_EXTRAS = -I$(BUILDDIR)
|
||||
USE_PRIVATE_MODULE_CACHE := YES
|
||||
OBJC_SOURCES := main.m foo.m
|
||||
|
||||
include Makefile.rules
|
|
@ -0,0 +1,61 @@
|
|||
from __future__ import print_function
|
||||
|
||||
import unittest2
|
||||
import os
|
||||
import shutil
|
||||
|
||||
import lldb
|
||||
from lldbsuite.test.decorators import *
|
||||
from lldbsuite.test.lldbtest import *
|
||||
from lldbsuite.test import lldbutil
|
||||
|
||||
|
||||
class TestClangModuleAppUpdate(TestBase):
|
||||
mydir = TestBase.compute_mydir(__file__)
|
||||
|
||||
def setUp(self):
|
||||
TestBase.setUp(self)
|
||||
|
||||
@skipUnlessDarwin
|
||||
@skipIf(debug_info=no_match(["gmodules"]))
|
||||
def test_rebuild_app_modules_untouched(self):
|
||||
with open(self.getBuildArtifact("module.modulemap"), "w") as f:
|
||||
f.write("""
|
||||
module Foo { header "f.h" }
|
||||
""")
|
||||
with open(self.getBuildArtifact("f.h"), "w") as f:
|
||||
f.write("""
|
||||
@import Foundation;
|
||||
@interface Foo : NSObject {
|
||||
int i;
|
||||
}
|
||||
+(instancetype)init;
|
||||
@end
|
||||
""")
|
||||
|
||||
mod_cache = self.getBuildArtifact("private-module-cache")
|
||||
import os
|
||||
if os.path.isdir(mod_cache):
|
||||
shutil.rmtree(mod_cache)
|
||||
self.build()
|
||||
self.assertTrue(os.path.isdir(mod_cache), "module cache exists")
|
||||
|
||||
target, process, _, bkpt = lldbutil.run_to_source_breakpoint(
|
||||
self, "break here", lldb.SBFileSpec("main.m"))
|
||||
bar = target.FindTypes('Bar').GetTypeAtIndex(0)
|
||||
foo = bar.GetDirectBaseClassAtIndex(0).GetType()
|
||||
self.assertEqual(foo.GetNumberOfFields(), 1)
|
||||
self.assertEqual(foo.GetFieldAtIndex(0).GetName(), "i")
|
||||
|
||||
# Rebuild.
|
||||
process.Kill()
|
||||
os.remove(self.getBuildArtifact('main.o'))
|
||||
os.remove(self.getBuildArtifact('a.out'))
|
||||
self.build()
|
||||
|
||||
# Reattach.
|
||||
target, process, _, _ = lldbutil.run_to_breakpoint_do_run(self, target, bkpt)
|
||||
bar = target.FindTypes('Bar').GetTypeAtIndex(0)
|
||||
foo = bar.GetDirectBaseClassAtIndex(0).GetType()
|
||||
self.assertEqual(foo.GetNumberOfFields(), 1)
|
||||
self.assertEqual(foo.GetFieldAtIndex(0).GetName(), "i")
|
|
@ -0,0 +1,7 @@
|
|||
@import Foundation;
|
||||
@import Foo;
|
||||
@implementation Foo
|
||||
+(instancetype)init {
|
||||
return [super init];
|
||||
}
|
||||
@end
|
|
@ -0,0 +1,17 @@
|
|||
@import Umbrella;
|
||||
|
||||
@interface Bar : Foo
|
||||
+(instancetype)init;
|
||||
@end
|
||||
|
||||
@implementation Bar
|
||||
+(instancetype)init {
|
||||
return [super init];
|
||||
}
|
||||
@end
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
id bar = [Bar new];
|
||||
[bar i]; // break here
|
||||
return 0;
|
||||
}
|
|
@ -0,0 +1,4 @@
|
|||
module Umbrella {
|
||||
header "umbrella.h"
|
||||
export *
|
||||
}
|
|
@ -0,0 +1 @@
|
|||
@import Foo;
|
|
@ -187,7 +187,7 @@ TypeSP DWARFASTParserClang::ParseTypeFromClangModule(const SymbolContext &sc,
|
|||
languages.Insert(die.GetCU()->GetLanguageType());
|
||||
llvm::DenseSet<SymbolFile *> searched_symbol_files;
|
||||
clang_module_sp->GetSymbolFile()->FindTypes(decl_context, languages,
|
||||
searched_symbol_files, pcm_types);
|
||||
searched_symbol_files, pcm_types);
|
||||
if (pcm_types.Empty()) {
|
||||
// Since this type is defined in one of the Clang modules imported
|
||||
// by this symbol file, search all of them. Instead of calling
|
||||
|
@ -222,6 +222,19 @@ TypeSP DWARFASTParserClang::ParseTypeFromClangModule(const SymbolContext &sc,
|
|||
if (!type)
|
||||
return TypeSP();
|
||||
|
||||
// Under normal operation pcm_type is a shallow forward declaration
|
||||
// that gets completed later. This is necessary to support cyclic
|
||||
// data structures. If, however, pcm_type is already complete (for
|
||||
// example, because it was loaded for a different target before),
|
||||
// the definition needs to be imported right away, too.
|
||||
// Type::ResolveClangType() effectively ignores the ResolveState
|
||||
// inside type_sp and only looks at IsDefined(), so it never calls
|
||||
// ClangASTImporter::ASTImporterDelegate::ImportDefinitionTo(),
|
||||
// which does extra work for Objective-C classes. This would result
|
||||
// in only the forward declaration to be visible.
|
||||
if (pcm_type.IsDefined())
|
||||
GetClangASTImporter().RequireCompleteType(ClangUtil::GetQualType(type));
|
||||
|
||||
SymbolFileDWARF *dwarf = die.GetDWARF();
|
||||
TypeSP type_sp(new Type(
|
||||
die.GetID(), dwarf, pcm_type_sp->GetName(), pcm_type_sp->GetByteSize(),
|
||||
|
|
Loading…
Reference in New Issue