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/Modules/codegen-extern-template.h =================================================================== --- /dev/null +++ clang/test/Modules/codegen-extern-template.h @@ -0,0 +1,12 @@ +// header for codegen-extern-template.cpp +#ifndef CODEGEN_EXTERN_TEMPLATE_H +#define CODEGEN_EXTERN_TEMPLATE_H + +template +inline T foo() { return 10; } + +extern template int foo(); + +inline int bar() { return foo(); } + +#endif Index: clang/test/Modules/codegen-extern-template.cpp =================================================================== --- /dev/null +++ clang/test/Modules/codegen-extern-template.cpp @@ -0,0 +1,9 @@ +// RUN: %clang_cc1 -triple=x86_64-linux-gnu -fmodules -fmodules-codegen -emit-module -fmodule-name=foo %S/codegen-extern-template.modulemap -x c++ -o %t.pcm +// RUN: %clang_cc1 -triple x86_64-linux-gnu -fmodules -fmodule-file=%t.pcm %s -emit-llvm -o - | FileCheck %s +// expected-no-diagnostics + +#include "codegen-extern-template.h" + +template int foo(); + +// CHECK: define weak_odr i32 @_Z3fooIiET_v Index: clang/test/Modules/codegen-extern-template.modulemap =================================================================== --- /dev/null +++ clang/test/Modules/codegen-extern-template.modulemap @@ -0,0 +1 @@ +module foo { header "codegen-extern-template.h" }