// Check that tracking of VFSs works with PCH. // RUN: rm -rf %t // RUN: split-file %s %t // RUN: sed -e "s|DIR|%/t|g" %t/build/compile-commands-pch.json.in > %t/build/compile-commands-pch.json // RUN: sed -e "s|DIR|%/t|g" %t/build/compile-commands-tu.json.in > %t/build/compile-commands-tu.json // RUN: sed -e "s|DIR|%/t|g" %t/build/compile-commands-tu-no-vfs-error.json.in > %t/build/compile-commands-tu-no-vfs-error.json // RUN: sed -e "s|DIR|%/t|g" %t/build/compile-commands-tu1.json.in > %t/build/compile-commands-tu1.json // RUN: sed -e "s|DIR|%/t|g" %t/build/pch-overlay.yaml.in > %t/build/pch-overlay.yaml // RUN: clang-scan-deps -compilation-database %t/build/compile-commands-pch.json \ // RUN: -j 1 -format experimental-full --optimize-args=vfs,header-search > %t/pch-deps.db // RUN: %deps-to-rsp %t/pch-deps.db --module-name=A > %t/A.rsp // RUN: %deps-to-rsp %t/pch-deps.db --module-name=B > %t/B.rsp // RUN: %deps-to-rsp %t/pch-deps.db --tu-index=0 > %t/pch.rsp // RUN: %clang @%t/A.rsp // RUN: %clang @%t/B.rsp // RUN: %clang @%t/pch.rsp // RUN: clang-scan-deps -compilation-database %t/build/compile-commands-tu.json \ // RUN: -j 1 -format experimental-full --optimize-args=vfs,header-search > %t/tu-deps.db // RUN: %deps-to-rsp %t/tu-deps.db --module-name=C > %t/C.rsp // RUN: %deps-to-rsp %t/tu-deps.db --tu-index=0 > %t/tu.rsp // RUN: %clang @%t/C.rsp // RUN: %clang @%t/tu.rsp // RUN: not clang-scan-deps -compilation-database %t/build/compile-commands-tu-no-vfs-error.json \ // RUN: -j 1 -format experimental-full --optimize-args=vfs,header-search 2>&1 | FileCheck --check-prefix=CHECK-ERROR %s // CHECK-ERROR: error: PCH was compiled with different VFS overlay files than are currently in use // CHECK-ERROR: note: current translation unit has no VFS overlays // Next test is to verify that a module that doesn't use the VFS, that depends // on the PCH's A, which does use the VFS, still records that it needs the VFS. // This avoids a fatal error when emitting diagnostics. // RUN: clang-scan-deps -compilation-database %t/build/compile-commands-tu1.json \ // RUN: -j 1 -format experimental-full --optimize-args=vfs,header-search > %t/tu1-deps.db // RUN: %deps-to-rsp %t/tu1-deps.db --tu-index=0 > %t/tu1.rsp // Reuse existing B // RUN: %deps-to-rsp %t/tu1-deps.db --module-name=E > %t/E.rsp // RUN: %deps-to-rsp %t/tu1-deps.db --module-name=D > %t/D.rsp // The build of D depends on B which depend on the prebuilt A. D will only build // if it has A's VFS, as it needs to emit a diagnostic showing the content of A. // RUN: %clang @%t/E.rsp // RUN: %clang @%t/D.rsp -verify // RUN: %clang @%t/tu1.rsp // RUN: cat %t/tu1-deps.db | sed 's:\\\\\?:/:g' | FileCheck %s -DPREFIX=%/t // Check that D has the overlay, but E doesn't. // CHECK: { // CHECK-NEXT: "modules": [ // CHECK-NEXT: { // CHECK-NEXT: "clang-module-deps": [ // CHECK-NEXT: { // CHECK-NEXT: "context-hash": "{{.*}}", // CHECK-NEXT: "module-name": "E" // CHECK-NEXT: } // CHECK-NEXT: ], // CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/modules/D/module.modulemap", // CHECK-NEXT: "command-line": [ // CHECK: "-ivfsoverlay" // CHECK-NEXT: "[[PREFIX]]/build/pch-overlay.yaml" // CHECK: ], // CHECK-NEXT: "context-hash": "{{.*}}", // CHECK-NEXT: "file-deps": [ // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: ], // CHECK: "name": "D" // CHECK-NEXT: }, // CHECK-NEXT: { // CHECK-NEXT: "clang-module-deps": [], // CHECK-NEXT: "clang-modulemap-file": "[[PREFIX]]/modules/E/module.modulemap", // CHECK-NEXT: "command-line": [ // CHECK-NOT: "-ivfsoverlay" // CHECK: ], // CHECK-NEXT: "context-hash": "{{.*}}", // CHECK-NEXT: "file-deps": [ // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: "{{.*}}" // CHECK-NEXT: ], // CHECK: "name": "E" // CHECK-NEXT: } //--- build/compile-commands-pch.json.in [ { "directory": "DIR", "command": "clang -x objective-c-header DIR/pch.h -I DIR/modules/A -I DIR/modules/B -fmodules -fimplicit-module-maps -fmodules-cache-path=DIR/cache -o DIR/pch.h.pch -ivfsoverlay DIR/build/pch-overlay.yaml", "file": "DIR/pch.h" } ] //--- build/compile-commands-tu.json.in [ { "directory": "DIR", "command": "clang -fsyntax-only DIR/tu.m -I DIR/modules/A -I DIR/modules/B -I DIR/modules/C -fmodules -fimplicit-module-maps -fmodules-cache-path=DIR/cache -include DIR/pch.h -o DIR/tu.o -ivfsoverlay DIR/build/pch-overlay.yaml", "file": "DIR/tu.m" } ] //--- build/compile-commands-tu-no-vfs-error.json.in [ { "directory": "DIR", "command": "clang -Wpch-vfs-diff -Werror=pch-vfs-diff -fsyntax-only DIR/tu.m -I DIR/modules/A -I DIR/modules/B -I DIR/modules/C -fmodules -fimplicit-module-maps -fmodules-cache-path=DIR/cache -include DIR/pch.h -o DIR/tu.o", "file": "DIR/tu.m" } ] //--- build/compile-commands-tu1.json.in [ { "directory": "DIR", "command": "clang -fsyntax-only DIR/tu1.m -I DIR/modules/B -I DIR/modules/D -I DIR/modules/E -fmodules -fimplicit-module-maps -fmodules-cache-path=DIR/cache -include DIR/pch.h -o DIR/tu1.o -ivfsoverlay DIR/build/pch-overlay.yaml", "file": "DIR/tu1.m" } ] //--- build/pch-overlay.yaml.in { "version":0, "case-sensitive":"false", "roots":[ { "contents":[ { "external-contents":"DIR/build/module.modulemap", "name":"module.modulemap", "type":"file" }, { "external-contents":"DIR/build/A.h", "name":"A.h", "type":"file" } ], "name":"DIR/modules/A", "type":"directory" } ] } //--- pch.h #include //--- build/module.modulemap module A { umbrella header "A.h" } //--- build/A.h typedef int A_t __attribute__((deprecated("yep, it's depr"))); //--- modules/B/module.modulemap module B { umbrella header "B.h" export * } //--- modules/B/B.h #include typedef int B_t; //--- modules/C/module.modulemap module C { umbrella header "C.h" } //--- modules/C/C.h #include typedef int C_t; //--- tu.m #include A_t a = 0; B_t b = 0; C_t c = 0; //--- modules/D/module.modulemap module D { umbrella header "D.h" export * } //--- modules/D/D.h #include #include typedef A_t D_t; // expected-warning{{'A_t' is deprecated}} // expected-note@*:* {{marked deprecated here}} //--- modules/E/module.modulemap module E { umbrella header "E.h" } //--- modules/E/E.h typedef int E_t; //--- tu1.m #include D_t d = 0; E_t e = 0;