This PR is 2nd part of [P1857R3](https://github.com/llvm/llvm-project/pull/107168) implementation, and mainly implement the restriction `A module directive may only appear as the first preprocessing tokens in a file (excluding the global module fragment.)`: [cpp.pre](https://eel.is/c++draft/cpp.pre): ``` module-file: pp-global-module-fragment[opt] pp-module group[opt] pp-private-module-fragment[opt] ``` We also refine tests use `split-file` instead of conditional macro. Signed-off-by: yronglin <yronglin777@gmail.com>
110 lines
3.5 KiB
C++
110 lines
3.5 KiB
C++
// RUN: rm -rf %t
|
|
// RUN: split-file %s %t
|
|
// RUN: %clang_cc1 -std=c++2a -I%t -emit-module-interface %t/interface.cppm -o %t.pcm
|
|
// RUN: %clang_cc1 -std=c++2a -I%t -fmodule-file=A=%t.pcm %t/implA.cppm -verify -fno-modules-error-recovery
|
|
// RUN: %clang_cc1 -std=c++2a -I%t -fmodule-file=A=%t.pcm %t/implB.cppm -verify -fno-modules-error-recovery
|
|
|
|
//--- foo.h
|
|
#ifndef FOO_H
|
|
#define FOO_H
|
|
extern int in_header;
|
|
#endif
|
|
|
|
//--- interface.cppm
|
|
module;
|
|
#include "foo.h"
|
|
// FIXME: The following need to be moved to a header file. The global module
|
|
// fragment is only permitted to contain preprocessor directives.
|
|
int global_module_fragment;
|
|
export module A;
|
|
export int exported;
|
|
int not_exported;
|
|
static int internal;
|
|
|
|
module :private;
|
|
int not_exported_private;
|
|
static int internal_private;
|
|
|
|
//--- implA.cppm
|
|
module;
|
|
|
|
void test_early() {
|
|
in_header = 1; // expected-error {{use of undeclared identifier 'in_header'}}
|
|
// expected-note@* {{not visible}}
|
|
|
|
global_module_fragment = 1; // expected-error {{use of undeclared identifier 'global_module_fragment'}}
|
|
|
|
exported = 1; // expected-error {{use of undeclared identifier 'exported'}}
|
|
|
|
not_exported = 1; // expected-error {{use of undeclared identifier 'not_exported'}}
|
|
|
|
// FIXME: We need better diagnostic message for static variable.
|
|
internal = 1; // expected-error {{use of undeclared identifier 'internal'}}
|
|
|
|
not_exported_private = 1; // expected-error {{undeclared identifier}}
|
|
|
|
internal_private = 1; // expected-error {{undeclared identifier}}
|
|
}
|
|
|
|
module A;
|
|
|
|
void test_late() {
|
|
in_header = 1; // expected-error {{missing '#include "foo.h"'; 'in_header' must be declared before it is used}}
|
|
// expected-note@* {{not visible}}
|
|
|
|
global_module_fragment = 1; // expected-error {{missing '#include'; 'global_module_fragment' must be declared before it is used}}
|
|
|
|
exported = 1;
|
|
|
|
not_exported = 1;
|
|
|
|
internal = 1; // expected-error {{use of undeclared identifier 'internal'}}
|
|
|
|
not_exported_private = 1;
|
|
|
|
internal_private = 1; // expected-error {{use of undeclared identifier 'internal_private'}}
|
|
}
|
|
|
|
//--- implB.cppm
|
|
module;
|
|
|
|
void test_early() {
|
|
in_header = 1; // expected-error {{use of undeclared identifier 'in_header'}}
|
|
// expected-note@* {{not visible}}
|
|
|
|
global_module_fragment = 1; // expected-error {{use of undeclared identifier 'global_module_fragment'}}
|
|
|
|
exported = 1; // expected-error {{use of undeclared identifier 'exported'}}
|
|
|
|
not_exported = 1; // expected-error {{use of undeclared identifier 'not_exported'}}
|
|
|
|
// FIXME: We need better diagnostic message for static variable.
|
|
internal = 1; // expected-error {{use of undeclared identifier 'internal'}}
|
|
|
|
not_exported_private = 1; // expected-error {{undeclared identifier}}
|
|
|
|
internal_private = 1; // expected-error {{undeclared identifier}}
|
|
}
|
|
|
|
export module B;
|
|
import A;
|
|
|
|
void test_late() {
|
|
in_header = 1; // expected-error {{missing '#include "foo.h"'; 'in_header' must be declared before it is used}}
|
|
// expected-note@* {{not visible}}
|
|
|
|
global_module_fragment = 1; // expected-error {{missing '#include'; 'global_module_fragment' must be declared before it is used}}
|
|
|
|
exported = 1;
|
|
|
|
not_exported = 1; // expected-error {{use of undeclared identifier 'not_exported'; did you mean 'exported'?}}
|
|
// expected-note@* {{'exported' declared here}}
|
|
|
|
internal = 1; // expected-error {{use of undeclared identifier 'internal'}}
|
|
|
|
not_exported_private = 1;
|
|
// FIXME: should not be visible here
|
|
// expected-error@-2 {{undeclared identifier}}
|
|
|
|
internal_private = 1; // expected-error {{use of undeclared identifier 'internal_private'}}
|
|
} |