1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2024 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/codegen/SkSLWGSLValidator.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLErrorReporter.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLPosition.h"
12*c8dee2aaSAndroid Build Coastguard Worker
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/tint/lang/wgsl/extension.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/tint/lang/wgsl/reader/options.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "tint/tint.h"
16*c8dee2aaSAndroid Build Coastguard Worker
17*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
18*c8dee2aaSAndroid Build Coastguard Worker
validate_wgsl(ErrorReporter & reporter,std::string_view wgsl,bool appendError,std::string * warnings)19*c8dee2aaSAndroid Build Coastguard Worker static bool validate_wgsl(ErrorReporter& reporter,
20*c8dee2aaSAndroid Build Coastguard Worker std::string_view wgsl,
21*c8dee2aaSAndroid Build Coastguard Worker bool appendError,
22*c8dee2aaSAndroid Build Coastguard Worker std::string* warnings) {
23*c8dee2aaSAndroid Build Coastguard Worker // Enable the WGSL optional features that Skia might rely on.
24*c8dee2aaSAndroid Build Coastguard Worker tint::wgsl::reader::Options options;
25*c8dee2aaSAndroid Build Coastguard Worker for (auto extension : {tint::wgsl::Extension::kChromiumExperimentalPixelLocal,
26*c8dee2aaSAndroid Build Coastguard Worker tint::wgsl::Extension::kDualSourceBlending}) {
27*c8dee2aaSAndroid Build Coastguard Worker options.allowed_features.extensions.insert(extension);
28*c8dee2aaSAndroid Build Coastguard Worker }
29*c8dee2aaSAndroid Build Coastguard Worker options.allowed_features.features.insert(
30*c8dee2aaSAndroid Build Coastguard Worker tint::wgsl::LanguageFeature::kUnrestrictedPointerParameters);
31*c8dee2aaSAndroid Build Coastguard Worker
32*c8dee2aaSAndroid Build Coastguard Worker // Verify that the WGSL we produced is valid.
33*c8dee2aaSAndroid Build Coastguard Worker tint::Source::File srcFile("", wgsl);
34*c8dee2aaSAndroid Build Coastguard Worker tint::Program program(tint::wgsl::reader::Parse(&srcFile, options));
35*c8dee2aaSAndroid Build Coastguard Worker
36*c8dee2aaSAndroid Build Coastguard Worker if (program.Diagnostics().ContainsErrors()) {
37*c8dee2aaSAndroid Build Coastguard Worker // The program isn't valid WGSL.
38*c8dee2aaSAndroid Build Coastguard Worker if (appendError) {
39*c8dee2aaSAndroid Build Coastguard Worker // Report the error via SkDEBUGFAIL and append the generated program for
40*c8dee2aaSAndroid Build Coastguard Worker // ease of debugging. We don't do this for our golden test output because
41*c8dee2aaSAndroid Build Coastguard Worker // it can change too often
42*c8dee2aaSAndroid Build Coastguard Worker tint::diag::Formatter diagFormatter;
43*c8dee2aaSAndroid Build Coastguard Worker std::string diagOutput = diagFormatter.Format(program.Diagnostics()).Plain();
44*c8dee2aaSAndroid Build Coastguard Worker diagOutput += "\n";
45*c8dee2aaSAndroid Build Coastguard Worker diagOutput += wgsl;
46*c8dee2aaSAndroid Build Coastguard Worker SkDEBUGFAILF("%s", diagOutput.c_str());
47*c8dee2aaSAndroid Build Coastguard Worker } else {
48*c8dee2aaSAndroid Build Coastguard Worker reporter.error(Position(),
49*c8dee2aaSAndroid Build Coastguard Worker std::string("Tint compilation failed.\n\n") + std::string(wgsl));
50*c8dee2aaSAndroid Build Coastguard Worker }
51*c8dee2aaSAndroid Build Coastguard Worker return false;
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker
54*c8dee2aaSAndroid Build Coastguard Worker if (!program.Diagnostics().empty()) {
55*c8dee2aaSAndroid Build Coastguard Worker // The program contains warnings. Report them as-is.
56*c8dee2aaSAndroid Build Coastguard Worker tint::diag::Formatter diagFormatter;
57*c8dee2aaSAndroid Build Coastguard Worker *warnings = diagFormatter.Format(program.Diagnostics()).Plain();
58*c8dee2aaSAndroid Build Coastguard Worker }
59*c8dee2aaSAndroid Build Coastguard Worker return true;
60*c8dee2aaSAndroid Build Coastguard Worker }
61*c8dee2aaSAndroid Build Coastguard Worker
ValidateWGSL(ErrorReporter & reporter,std::string_view wgsl,std::string * warnings)62*c8dee2aaSAndroid Build Coastguard Worker bool ValidateWGSL(ErrorReporter& reporter, std::string_view wgsl, std::string* warnings) {
63*c8dee2aaSAndroid Build Coastguard Worker return validate_wgsl(reporter, wgsl, false, warnings);
64*c8dee2aaSAndroid Build Coastguard Worker }
65*c8dee2aaSAndroid Build Coastguard Worker
ValidateWGSLVerbose(ErrorReporter & reporter,std::string_view wgsl,std::string * warnings)66*c8dee2aaSAndroid Build Coastguard Worker bool ValidateWGSLVerbose(ErrorReporter& reporter, std::string_view wgsl, std::string* warnings) {
67*c8dee2aaSAndroid Build Coastguard Worker return validate_wgsl(reporter, wgsl, true, warnings);
68*c8dee2aaSAndroid Build Coastguard Worker }
69*c8dee2aaSAndroid Build Coastguard Worker
70*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
71