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:
Adrian Prantl 2019-11-22 09:58:16 -08:00
parent c0eeea5d74
commit 539117616d
7 changed files with 109 additions and 1 deletions

View File

@ -0,0 +1,5 @@
CFLAGS_EXTRAS = -I$(BUILDDIR)
USE_PRIVATE_MODULE_CACHE := YES
OBJC_SOURCES := main.m foo.m
include Makefile.rules

View File

@ -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")

View File

@ -0,0 +1,7 @@
@import Foundation;
@import Foo;
@implementation Foo
+(instancetype)init {
return [super init];
}
@end

View File

@ -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;
}

View File

@ -0,0 +1,4 @@
module Umbrella {
header "umbrella.h"
export *
}

View File

@ -0,0 +1 @@
@import Foo;

View File

@ -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(),