Files
swift-mirror/test/Interop/Cxx/class/access/using-non-public-module-interface.swift
John Hui 66c2e2c52b [cxx-interop] Import non-public inherited members (#79348)
This patch is follow-up work from #78942 and imports non-public members,
which were previously not being imported. Those members can be accessed
in a Swift file blessed by the SWIFT_PRIVATE_FILEID annotation.

As a consequence of this patch, we are also now importing inherited members
that are inaccessible from the derived classes, because they were declared
private, or because they were inherited via nested private inheritance. We
import them anyway but mark them unavailable, for better diagnostics and to
(somewhat) simplify the import logic for inheritance.

Because non-public base class members are now imported too, this patch
inflames an existing issue where a 'using' declaration on an inherited member
with a synthesized name (e.g., operators) produces duplicate members, leading
to miscompilation (resulting in a runtime crash). This was not previously noticed
because a 'using' declaration on a public inherited member is not usually
necessary, but is a common way to expose otherwise non-public members.
This patch puts in a workaround to prevent this from affecting the behavior
of MSVC's std::optional implementation, which uses this pattern of 'using'
a private inherited member. That will be fixed in a follow-up patch.

Follow-up work is also needed to correctly diagnose ambiguous overloads
in cases of multiple inheritance, and to account for virtual inheritance.

rdar://137764620
2025-02-25 01:03:16 -08:00

54 lines
2.5 KiB
Swift

// RUN: %target-swift-ide-test -print-module -module-to-print=UsingNonPublic -print-access -I %S/Inputs -source-filename=x -cxx-interoperability-mode=default | %FileCheck %s
// CHECK: public struct PublUser {
// CHECK-NEXT: public init()
// CHECK-NEXT: public func publUsingProt() -> Return
// CHECK-NEXT: public func publUsingPubl() -> Return
// CHECK-NEXT: private func protUsingProt() -> Return
// CHECK-NEXT: private func protUsingPubl() -> Return
// CHECK-NEXT: private func omitUsingProt() -> Return
// CHECK-NEXT: public func omitUsingPubl() -> Return
// CHECK-NEXT: }
// CHECK: public struct ProtUser {
// CHECK-NEXT: public init()
// CHECK-NEXT: public func publUsingProt() -> Return
// CHECK-NEXT: public func publUsingPubl() -> Return
// CHECK-NEXT: private func protUsingProt() -> Return
// CHECK-NEXT: private func protUsingPubl() -> Return
// CHECK-NEXT: private func omitUsingProt() -> Return
// CHECK-NEXT: private func omitUsingPubl() -> Return
// CHECK-NEXT: }
// CHECK: public struct PrivUser {
// CHECK-NEXT: public init()
// CHECK-NEXT: public func publUsingProt() -> Return
// CHECK-NEXT: public func publUsingPubl() -> Return
// CHECK-NEXT: private func protUsingProt() -> Return
// CHECK-NEXT: private func protUsingPubl() -> Return
// CHECK-NEXT: private func omitUsingProt() -> Return
// CHECK-NEXT: private func omitUsingPubl() -> Return
// CHECK-NEXT: }
// CHECK: public struct PublPrivUser {
// CHECK-NEXT: public init()
// CHECK-NEXT: public func publUsingProt() -> Return
// CHECK-NEXT: public func publUsingPubl() -> Return
// CHECK-NEXT: private func protUsingProt() -> Return
// CHECK-NEXT: private func protUsingPubl() -> Return
// CHECK-NEXT: @available(*, unavailable, message: "this base member is not accessible because of private inheritance")
// CHECK-NEXT: private func omitUsingProt() -> Return
// CHECK-NEXT: @available(*, unavailable, message: "this base member is not accessible because of private inheritance")
// CHECK-NEXT: private func omitUsingPubl() -> Return
// CHECK-NEXT: }
// CHECK: public struct PrivUserPubl {
// CHECK-NEXT: public init()
// CHECK-NEXT: public func publUsingProt() -> Return
// CHECK-NEXT: public func publUsingPubl() -> Return
// CHECK-NEXT: private func protUsingProt() -> Return
// CHECK-NEXT: private func protUsingPubl() -> Return
// CHECK-NEXT: private func omitUsingProt() -> Return
// CHECK-NEXT: private func omitUsingPubl() -> Return
// CHECK-NEXT: }