blob: ad9d3836ba8a4dcab75ee61ce6b5cca961d5021f [file] [log] [blame]
Dejan Mircevskib6fe02f2016-01-07 13:44:22 -05001// Copyright (c) 2015-2016 The Khronos Group Inc.
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01002//
David Neto9fc86582016-09-01 15:33:59 -04003// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01006//
David Neto9fc86582016-09-01 15:33:59 -04007// http://www.apache.org/licenses/LICENSE-2.0
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01008//
David Neto9fc86582016-09-01 15:33:59 -04009// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010014
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010015#include "diagnostic.h"
16
Lei Zhang620f05e2016-09-16 16:12:04 -040017#include <cassert>
18#include <cstring>
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010019#include <iostream>
David Neto169266e2017-10-01 09:27:00 -040020#include <sstream>
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010021
Lei Zhang755f97f2016-09-02 18:06:18 -040022#include "table.h"
Lei Zhang923f6c12015-11-11 12:45:23 -050023
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010024// Diagnostic API
25
26spv_diagnostic spvDiagnosticCreate(const spv_position position,
Lei Zhang1a0334e2015-11-02 09:41:20 -050027 const char* message) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010028 spv_diagnostic diagnostic = new spv_diagnostic_t;
Lei Zhang40056702015-09-11 14:31:27 -040029 if (!diagnostic) return nullptr;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010030 size_t length = strlen(message) + 1;
31 diagnostic->error = new char[length];
Lei Zhang40056702015-09-11 14:31:27 -040032 if (!diagnostic->error) {
33 delete diagnostic;
34 return nullptr;
35 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010036 diagnostic->position = *position;
David Netoc9786432015-09-01 18:05:14 -040037 diagnostic->isTextSource = false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010038 memset(diagnostic->error, 0, length);
39 strncpy(diagnostic->error, message, length);
40 return diagnostic;
41}
42
43void spvDiagnosticDestroy(spv_diagnostic diagnostic) {
Lei Zhang40056702015-09-11 14:31:27 -040044 if (!diagnostic) return;
Eric Engestromeb6ae972016-02-18 23:41:16 +000045 delete[] diagnostic->error;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010046 delete diagnostic;
47}
48
49spv_result_t spvDiagnosticPrint(const spv_diagnostic diagnostic) {
Lei Zhang40056702015-09-11 14:31:27 -040050 if (!diagnostic) return SPV_ERROR_INVALID_DIAGNOSTIC;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010051
David Netoc9786432015-09-01 18:05:14 -040052 if (diagnostic->isTextSource) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010053 // NOTE: This is a text position
54 // NOTE: add 1 to the line as editors start at line 1, we are counting new
55 // line characters to start at line 0
56 std::cerr << "error: " << diagnostic->position.line + 1 << ": "
57 << diagnostic->position.column + 1 << ": " << diagnostic->error
58 << "\n";
59 return SPV_SUCCESS;
David Netoc9786432015-09-01 18:05:14 -040060 } else {
61 // NOTE: Assume this is a binary position
dan sinclair51091042018-07-03 15:06:54 -040062 std::cerr << "error: ";
63 if (diagnostic->position.index > 0)
64 std::cerr << diagnostic->position.index << ": ";
65 std::cerr << diagnostic->error << "\n";
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010066 return SPV_SUCCESS;
67 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010068}
Andrew Woloszyn71fc0552015-09-24 10:26:51 -040069
dan sinclair3dad1cd2018-07-07 09:38:00 -040070namespace spvtools {
David Neto01656362015-11-20 10:44:41 -050071
David Neto169266e2017-10-01 09:27:00 -040072DiagnosticStream::DiagnosticStream(DiagnosticStream&& other)
73 : stream_(),
74 position_(other.position_),
75 consumer_(other.consumer_),
dan sinclairf80696e2018-06-19 16:02:44 -040076 disassembled_instruction_(std::move(other.disassembled_instruction_)),
David Neto169266e2017-10-01 09:27:00 -040077 error_(other.error_) {
78 // Prevent the other object from emitting output during destruction.
79 other.error_ = SPV_FAILED_MATCH;
80 // Some platforms are missing support for std::ostringstream functionality,
81 // including: move constructor, swap method. Either would have been a
82 // better choice than copying the string.
83 stream_ << other.stream_.str();
84}
85
Andrew Woloszyn71fc0552015-09-24 10:26:51 -040086DiagnosticStream::~DiagnosticStream() {
Lei Zhang755f97f2016-09-02 18:06:18 -040087 if (error_ != SPV_FAILED_MATCH && consumer_ != nullptr) {
Lei Zhang620f05e2016-09-16 16:12:04 -040088 auto level = SPV_MSG_ERROR;
Lei Zhang755f97f2016-09-02 18:06:18 -040089 switch (error_) {
90 case SPV_SUCCESS:
91 case SPV_REQUESTED_TERMINATION: // Essentially success.
Lei Zhang620f05e2016-09-16 16:12:04 -040092 level = SPV_MSG_INFO;
Lei Zhang755f97f2016-09-02 18:06:18 -040093 break;
94 case SPV_WARNING:
David Netoc3caa542017-03-16 15:19:51 -040095 level = SPV_MSG_WARNING;
Lei Zhang755f97f2016-09-02 18:06:18 -040096 break;
97 case SPV_UNSUPPORTED:
98 case SPV_ERROR_INTERNAL:
99 case SPV_ERROR_INVALID_TABLE:
Lei Zhang620f05e2016-09-16 16:12:04 -0400100 level = SPV_MSG_INTERNAL_ERROR;
Lei Zhang755f97f2016-09-02 18:06:18 -0400101 break;
102 case SPV_ERROR_OUT_OF_MEMORY:
Lei Zhang620f05e2016-09-16 16:12:04 -0400103 level = SPV_MSG_FATAL;
Lei Zhang755f97f2016-09-02 18:06:18 -0400104 break;
105 default:
106 break;
107 }
dan sinclairf80696e2018-06-19 16:02:44 -0400108 if (disassembled_instruction_.size() > 0)
109 stream_ << std::endl << " " << disassembled_instruction_ << std::endl;
110
Lei Zhang35902792016-09-16 15:43:41 -0400111 consumer_(level, "input", position_, stream_.str().c_str());
Andrew Woloszyn71fc0552015-09-24 10:26:51 -0400112 }
113}
Lei Zhang755f97f2016-09-02 18:06:18 -0400114
115void UseDiagnosticAsMessageConsumer(spv_context context,
116 spv_diagnostic* diagnostic) {
117 assert(diagnostic && *diagnostic == nullptr);
118
Lei Zhang620f05e2016-09-16 16:12:04 -0400119 auto create_diagnostic = [diagnostic](spv_message_level_t, const char*,
120 const spv_position_t& position,
121 const char* message) {
Lei Zhang755f97f2016-09-02 18:06:18 -0400122 auto p = position;
123 spvDiagnosticDestroy(*diagnostic); // Avoid memory leak.
124 *diagnostic = spvDiagnosticCreate(&p, message);
125 };
dan sinclair3dad1cd2018-07-07 09:38:00 -0400126 SetContextMessageConsumer(context, std::move(create_diagnostic));
Lei Zhang755f97f2016-09-02 18:06:18 -0400127}
128
129std::string spvResultToString(spv_result_t res) {
Umar Arshadc7413852015-12-15 21:44:21 -0500130 std::string out;
131 switch (res) {
132 case SPV_SUCCESS:
133 out = "SPV_SUCCESS";
134 break;
135 case SPV_UNSUPPORTED:
136 out = "SPV_UNSUPPORTED";
137 break;
138 case SPV_END_OF_STREAM:
139 out = "SPV_END_OF_STREAM";
140 break;
141 case SPV_WARNING:
142 out = "SPV_WARNING";
143 break;
144 case SPV_FAILED_MATCH:
145 out = "SPV_FAILED_MATCH";
146 break;
147 case SPV_REQUESTED_TERMINATION:
148 out = "SPV_REQUESTED_TERMINATION";
149 break;
150 case SPV_ERROR_INTERNAL:
151 out = "SPV_ERROR_INTERNAL";
152 break;
153 case SPV_ERROR_OUT_OF_MEMORY:
154 out = "SPV_ERROR_OUT_OF_MEMORY";
155 break;
156 case SPV_ERROR_INVALID_POINTER:
157 out = "SPV_ERROR_INVALID_POINTER";
158 break;
159 case SPV_ERROR_INVALID_BINARY:
160 out = "SPV_ERROR_INVALID_BINARY";
161 break;
162 case SPV_ERROR_INVALID_TEXT:
163 out = "SPV_ERROR_INVALID_TEXT";
164 break;
165 case SPV_ERROR_INVALID_TABLE:
166 out = "SPV_ERROR_INVALID_TABLE";
167 break;
168 case SPV_ERROR_INVALID_VALUE:
169 out = "SPV_ERROR_INVALID_VALUE";
170 break;
171 case SPV_ERROR_INVALID_DIAGNOSTIC:
172 out = "SPV_ERROR_INVALID_DIAGNOSTIC";
173 break;
174 case SPV_ERROR_INVALID_LOOKUP:
175 out = "SPV_ERROR_INVALID_LOOKUP";
176 break;
177 case SPV_ERROR_INVALID_ID:
178 out = "SPV_ERROR_INVALID_ID";
179 break;
180 case SPV_ERROR_INVALID_CFG:
181 out = "SPV_ERROR_INVALID_CFG";
182 break;
183 case SPV_ERROR_INVALID_LAYOUT:
184 out = "SPV_ERROR_INVALID_LAYOUT";
185 break;
186 default:
187 out = "Unknown Error";
188 }
189 return out;
190}
David Neto01656362015-11-20 10:44:41 -0500191
dan sinclair3dad1cd2018-07-07 09:38:00 -0400192} // namespace spvtools