Index: clang/lib/Serialization/ASTWriterDecl.cpp =================================================================== --- clang/lib/Serialization/ASTWriterDecl.cpp +++ clang/lib/Serialization/ASTWriterDecl.cpp @@ -2412,11 +2412,12 @@ } if (Writer->Context->getLangOpts().ModulesCodegen) { // Under -fmodules-codegen, codegen is performed for all non-internal, - // non-always_inline functions. + // non-always_inline functions, unless they are available elsewhere. if (!FD->hasAttr()) { if (!Linkage) Linkage = Writer->Context->GetGVALinkageForFunction(FD); - ModulesCodegen = *Linkage != GVA_Internal; + ModulesCodegen = + *Linkage != GVA_Internal && *Linkage != GVA_AvailableExternally; } } } Index: clang/test/PCH/codegen-extern-template.h =================================================================== --- /dev/null +++ clang/test/PCH/codegen-extern-template.h @@ -0,0 +1,16 @@ +// header for codegen-extern-template.cpp +#pragma once + +template< typename T > +struct A +{ + T foo() { return 10;} +}; + +extern template struct A< int >; + +inline +int bar(A* p) +{ + return p->foo(); +} Index: clang/test/PCH/codegen-extern-template.cpp =================================================================== --- /dev/null +++ clang/test/PCH/codegen-extern-template.cpp @@ -0,0 +1,21 @@ +// Build the PCH with extern template. +// RUN: %clang -x c++-header %S/codegen-extern-template.h -o %t.pch -Xclang -building-pch-with-obj -Xclang -fmodules-codegen +// Build the PCH's object file. +// RUN: %clang -c %s -include-pch %t.pch -Xclang -building-pch-with-obj -Xclang -fmodules-codegen -o %t.1.o +// Build source with explicit template instantiation. +// RUN: %clang -c %s -DMAIN -include-pch %t.pch -o %t.2.o +// There should be no unresolved symbol. +// RUN: %clang %t.1.o %t.2.o +// expected-no-diagnostics + +#include "codegen-extern-template.h" + +#ifdef MAIN +template struct A< int >; +int main() +{ + A< int > a; + return bar(&a); +} + +#endif