[PATCH] D55676: [Modules] Fix decl order for DeclsInPrototype
Bruno Cardoso Lopes via Phabricator via cfe-commits
cfe-commits at lists.llvm.org
Thu Dec 13 14:48:30 PST 2018
bruno created this revision.
bruno added reviewers: rsmith, aprantl, v.g.vassilev.
Herald added subscribers: dexonsmith, mgrang, jkorous.
When a declarator is constructed for a function prototype in
`Parser::ParseFunctionDeclarator`, it calls getCurScope()->decls()
in order to populate DeclsInPrototype.
getCurScope()->decls() return a range from a llvm::SmallPtrSet, which
doesn't guarantee any order. Later on, DeclsInPrototype are used to
populate decl contexts that end up being serialized when PCH or modules
are used. This causes non-determinism in module serialization, which
is bad for lots of reason, including that when a PCH used Modules, it
can't rebuild the module if there's a signature mismatch.
This patch fixes that by sorting out the result getCurScope()->decls()
before populating DeclsInPrototype. I don't believe we can change this
data structure because it can be used for any type of scope (not only
function prototypes).
There are no testcases for this change, since you have to run the
problematic non-reducible testcase multiple times to get a mismatch.
However, one can try to reproduce it on macOS using:
echo '@import Foundation;' > test.m
clang -fmodules test.m -o test.o -c \
-fmodules-cache-path=/tmp/cache \
-isysroot
/Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.14.sdk
mv /tmp/cache /tmp/cache1
clang -fmodules test.m -o test.o -c \
-fmodules-cache-path=/tmp/cache \
-isysroot
/Applications/Xcode.app/Contents/Developer/Platforms/MacOSX.platform/Developer/SDKs/MacOSX10.14.sdk
mv /tmp/cache /tmp/cache2
HASH=`ls /tmp/cache1`
for i in `find /tmp/cache1 -type f -iname "*.pcm"`; do
F=`basename $i`;
diff /tmp/cache1/$HASH/$F /tmp/cache2/$HASH/$F;
done
rdar://problem/43442957
https://reviews.llvm.org/D55676
Files:
lib/Parse/ParseDecl.cpp
Index: lib/Parse/ParseDecl.cpp
===================================================================
--- lib/Parse/ParseDecl.cpp
+++ lib/Parse/ParseDecl.cpp
@@ -6227,7 +6227,16 @@
SmallVector<NamedDecl *, 0> DeclsInPrototype;
if (getCurScope()->getFlags() & Scope::FunctionDeclarationScope &&
!getLangOpts().CPlusPlus) {
- for (Decl *D : getCurScope()->decls()) {
+ // The decls in the scope are in arbitrary order. Add them in sorted order
+ // now and allow for the declarator chunk to always contain the decls in
+ // deterministic order. This is necessary because ActOnFunctionDeclarator
+ // copies the declarator chunk as is when populating the decl context,
+ // which could be later serialized for modules or PCHs.
+ SmallVector<Decl *, 8> SortedDecls(getCurScope()->decls());
+ llvm::sort(SortedDecls, [](const Decl *L, const Decl *R) {
+ return L->getID() < R->getID();
+ });
+ for (Decl *D : SortedDecls) {
NamedDecl *ND = dyn_cast<NamedDecl>(D);
if (!ND || isa<ParmVarDecl>(ND))
continue;
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D55676.178141.patch
Type: text/x-patch
Size: 1070 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/cfe-commits/attachments/20181213/0412b223/attachment-0001.bin>
More information about the cfe-commits
mailing list