1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
| // RUN: rm -rf %t
// RUN: mkdir -p %t
// RUN: %clang_cc1 -std=c++2a -x c++-header %S/Inputs/header.h -emit-header-module -fmodule-name=FIXME -o %t/h.pcm
// RUN: %clang_cc1 -std=c++2a %s -DX_INTERFACE -emit-module-interface -o %t/x.pcm
// RUN: %clang_cc1 -std=c++2a %s -DY_INTERFACE -emit-module-interface -o %t/y.pcm
// RUN: %clang_cc1 -std=c++2a %s -DINTERFACE -fmodule-file=%t/x.pcm -fmodule-file=%t/y.pcm -emit-module-interface -o %t/m.pcm
// RUN: %clang_cc1 -std=c++2a %s -DIMPLEMENTATION -I%S/Inputs -fmodule-file=%t/h.pcm -fmodule-file=%t/m.pcm -verify
// RUN: %clang_cc1 -std=c++2a %s -DUSER -I%S/Inputs -fmodule-file=%t/h.pcm -fmodule-file=%t/m.pcm -verify
#if defined(X_INTERFACE)
export module X;
export int x;
#elif defined(Y_INTERFACE)
export module Y;
export int y;
#elif defined(INTERFACE)
export module p2;
export import X;
import Y; // not exported
namespace A {
int f();
export int g();
int h();
namespace inner {}
}
export namespace B {
namespace inner {}
}
namespace B {
int f();
}
namespace C {}
namespace D { int f(); }
export namespace D {}
#elif defined(IMPLEMENTATION)
module p2;
import "header.h";
// Per [basic.scope.namespace]/2.3, exportedness has no impact on visibility
// within the same module.
//
// expected-no-diagnostics
void use() {
A::f();
A::g();
A::h();
using namespace A::inner;
using namespace B;
using namespace B::inner;
B::f();
f();
using namespace C;
D::f();
}
int use_header() { return foo + bar::baz(); }
#elif defined(USER)
import p2;
import "header.h";
void use() {
// namespace A is implicitly exported by the export of A::g.
A::f(); // expected-error {{no member named 'f' in namespace 'A'}}
A::g();
A::h(); // expected-error {{no member named 'h' in namespace 'A'}}
using namespace A::inner; // expected-error {{expected namespace name}}
// namespace B and B::inner are explicitly exported
using namespace B;
using namespace B::inner;
B::f(); // expected-error {{no member named 'f' in namespace 'B'}}
f(); // expected-error {{undeclared identifier 'f'}}
// namespace C is not exported
using namespace C; // expected-error {{expected namespace name}}
// namespace D is exported, but D::f is not
D::f(); // expected-error {{no member named 'f' in namespace 'D'}}
}
int use_header() { return foo + bar::baz(); }
#else
#error unknown mode
#endif
|