122 lines
2.9 KiB
C++
122 lines
2.9 KiB
C++
|
// Clear and create directories
|
||
|
// RUN: rm -rf %t
|
||
|
// RUN: mkdir %t
|
||
|
// RUN: mkdir %t/cache
|
||
|
// RUN: mkdir %t/Inputs
|
||
|
|
||
|
// Build first header file
|
||
|
// RUN: echo "#define FIRST" >> %t/Inputs/first.h
|
||
|
// RUN: cat %s >> %t/Inputs/first.h
|
||
|
|
||
|
// Build second header file
|
||
|
// RUN: echo "#define SECOND" >> %t/Inputs/second.h
|
||
|
// RUN: cat %s >> %t/Inputs/second.h
|
||
|
|
||
|
// Test that each header can compile
|
||
|
// RUN: %clang_cc1 -fsyntax-only -x c++ -std=c++11 -fblocks %t/Inputs/first.h
|
||
|
// RUN: %clang_cc1 -fsyntax-only -x c++ -std=c++11 -fblocks %t/Inputs/second.h
|
||
|
|
||
|
// Build module map file
|
||
|
// RUN: echo "module FirstModule {" >> %t/Inputs/module.map
|
||
|
// RUN: echo " header \"first.h\"" >> %t/Inputs/module.map
|
||
|
// RUN: echo "}" >> %t/Inputs/module.map
|
||
|
// RUN: echo "module SecondModule {" >> %t/Inputs/module.map
|
||
|
// RUN: echo " header \"second.h\"" >> %t/Inputs/module.map
|
||
|
// RUN: echo "}" >> %t/Inputs/module.map
|
||
|
|
||
|
// Run test
|
||
|
// RUN: %clang_cc1 -fmodules -fimplicit-module-maps \
|
||
|
// RUN: -fmodules-cache-path=%t/cache -x c++ -I%t/Inputs \
|
||
|
// RUN: -verify %s -std=c++11 -fblocks
|
||
|
|
||
|
#if !defined(FIRST) && !defined(SECOND)
|
||
|
#include "first.h"
|
||
|
#include "second.h"
|
||
|
#endif
|
||
|
|
||
|
// Used for testing
|
||
|
#if defined(FIRST)
|
||
|
#define ACCESS public:
|
||
|
#elif defined(SECOND)
|
||
|
#define ACCESS private:
|
||
|
#endif
|
||
|
|
||
|
// TODO: S1 and S2 should generate errors.
|
||
|
namespace Blocks {
|
||
|
#if defined(FIRST)
|
||
|
struct S1 {
|
||
|
void (^block)(int x) = ^(int x) { };
|
||
|
};
|
||
|
#elif defined(SECOND)
|
||
|
struct S1 {
|
||
|
void (^block)(int x) = ^(int y) { };
|
||
|
};
|
||
|
#else
|
||
|
S1 s1;
|
||
|
#endif
|
||
|
|
||
|
#if defined(FIRST)
|
||
|
struct S2 {
|
||
|
int (^block)(int x) = ^(int x) { return x + 1; };
|
||
|
};
|
||
|
#elif defined(SECOND)
|
||
|
struct S2 {
|
||
|
int (^block)(int x) = ^(int x) { return x; };
|
||
|
};
|
||
|
#else
|
||
|
S2 s2;
|
||
|
#endif
|
||
|
|
||
|
#if defined(FIRST)
|
||
|
struct S3 {
|
||
|
void run(int (^block)(int x));
|
||
|
};
|
||
|
#elif defined(SECOND)
|
||
|
struct S3 {
|
||
|
void run(int (^block)(int x, int y));
|
||
|
};
|
||
|
#else
|
||
|
S3 s3;
|
||
|
// expected-error@first.h:* {{'Blocks::S3::run' from module 'FirstModule' is not present in definition of 'Blocks::S3' in module 'SecondModule'}}
|
||
|
// expected-note@second.h:* {{declaration of 'run' does not match}}
|
||
|
#endif
|
||
|
|
||
|
#define DECLS \
|
||
|
int (^block)(int x) = ^(int x) { return x + x; }; \
|
||
|
void run(int (^block)(int x, int y));
|
||
|
|
||
|
#if defined(FIRST) || defined(SECOND)
|
||
|
struct Valid1 {
|
||
|
DECLS
|
||
|
};
|
||
|
#else
|
||
|
Valid1 v1;
|
||
|
#endif
|
||
|
|
||
|
#if defined(FIRST) || defined(SECOND)
|
||
|
struct Invalid1 {
|
||
|
DECLS
|
||
|
ACCESS
|
||
|
};
|
||
|
#else
|
||
|
Invalid1 i1;
|
||
|
// expected-error@second.h:* {{'Blocks::Invalid1' has different definitions in different modules; first difference is definition in module 'SecondModule' found private access specifier}}
|
||
|
// expected-note@first.h:* {{but in 'FirstModule' found public access specifier}}
|
||
|
#endif
|
||
|
|
||
|
#undef DECLS
|
||
|
}
|
||
|
|
||
|
// Keep macros contained to one file.
|
||
|
#ifdef FIRST
|
||
|
#undef FIRST
|
||
|
#endif
|
||
|
|
||
|
#ifdef SECOND
|
||
|
#undef SECOND
|
||
|
#endif
|
||
|
|
||
|
#ifdef ACCESS
|
||
|
#undef ACCESS
|
||
|
#endif
|