|
15 | 15 | //
|
16 | 16 | //===----------------------------------------------------------------------===//
|
17 | 17 |
|
| 18 | +#include "swift/Strings.h" |
18 | 19 | #include "swift/AST/Decl.h"
|
19 | 20 | #include "swift/AST/ASTContext.h"
|
20 | 21 | #include "swift/AST/ASTMangler.h"
|
@@ -5348,9 +5349,39 @@ int TypeDecl::compare(const TypeDecl *type1, const TypeDecl *type2) {
|
5348 | 5349 |
|
5349 | 5350 | // Prefer module names earlier in the alphabet.
|
5350 | 5351 | if (dc1->isModuleScopeContext() && dc2->isModuleScopeContext()) {
|
5351 |
| - auto module1 = dc1->getParentModule(); |
5352 |
| - auto module2 = dc2->getParentModule(); |
5353 |
| - if (int result = module1->getName().str().compare(module2->getName().str())) |
| 5352 | + // For protocol declarations specifically, the order here is part |
| 5353 | + // of the ABI, and so we must take care to get the correct module |
| 5354 | + // name for the comparison. |
| 5355 | + auto getModuleNameForOrder = [&](const TypeDecl *decl) -> StringRef { |
| 5356 | + // This used to just call getName(), which caused accidental ABI breaks |
| 5357 | + // when a module is imported under different aliases. |
| 5358 | + // |
| 5359 | + // The new behavior matches the behavior in the mangler: |
| 5360 | + // |
| 5361 | + // - getName() returns the name of the module as its imported by the |
| 5362 | + // client, which might be an alias different from the real name. |
| 5363 | + // - getRealName() returns the real name after desugaring aliases. |
| 5364 | + // - getABIName() returns the name set with the -module-abi-name flag, or |
| 5365 | + // **getName()** if the flag is not set. |
| 5366 | + // |
| 5367 | + // Thus, the correct module name to use for mangling and ordering is |
| 5368 | + // getRealName(), unless getABIName() is distinct from getName(), in |
| 5369 | + // which case we use getABIName(). |
| 5370 | + // |
| 5371 | + // FIXME: This should be fixed after auditing callers of getABIName(). |
| 5372 | + // |
| 5373 | + // However, to maintain ABI compatibility, we ignore the ABI name for the |
| 5374 | + // _Concurrency module, which is "Swift", and still use the real name in |
| 5375 | + // this case. |
| 5376 | + auto *module = decl->getDeclContext()->getParentModule(); |
| 5377 | + if (module->getRealName().str() == SWIFT_CONCURRENCY_NAME || |
| 5378 | + module->getName() == module->getABIName()) { |
| 5379 | + return module->getRealName().str(); |
| 5380 | + } |
| 5381 | + return module->getABIName().str(); |
| 5382 | + }; |
| 5383 | + |
| 5384 | + if (int result = getModuleNameForOrder(type1).compare(getModuleNameForOrder(type2))) |
5354 | 5385 | return result;
|
5355 | 5386 | }
|
5356 | 5387 |
|
|
0 commit comments