blob: 0c467d5666aa5c046802fed14c9f05e7bddffd5b [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
17#include <assert.h>
18#include <string.h>
19
20#include <iostream>
21
David Neto5a703352016-02-17 14:44:00 -050022#include "spirv-tools/libspirv.h"
Lei Zhang755f97f2016-09-02 18:06:18 -040023#include "table.h"
Lei Zhang923f6c12015-11-11 12:45:23 -050024
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010025// Diagnostic API
26
27spv_diagnostic spvDiagnosticCreate(const spv_position position,
Lei Zhang1a0334e2015-11-02 09:41:20 -050028 const char* message) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010029 spv_diagnostic diagnostic = new spv_diagnostic_t;
Lei Zhang40056702015-09-11 14:31:27 -040030 if (!diagnostic) return nullptr;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010031 size_t length = strlen(message) + 1;
32 diagnostic->error = new char[length];
Lei Zhang40056702015-09-11 14:31:27 -040033 if (!diagnostic->error) {
34 delete diagnostic;
35 return nullptr;
36 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010037 diagnostic->position = *position;
David Netoc9786432015-09-01 18:05:14 -040038 diagnostic->isTextSource = false;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010039 memset(diagnostic->error, 0, length);
40 strncpy(diagnostic->error, message, length);
41 return diagnostic;
42}
43
44void spvDiagnosticDestroy(spv_diagnostic diagnostic) {
Lei Zhang40056702015-09-11 14:31:27 -040045 if (!diagnostic) return;
Eric Engestromeb6ae972016-02-18 23:41:16 +000046 delete[] diagnostic->error;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010047 delete diagnostic;
48}
49
50spv_result_t spvDiagnosticPrint(const spv_diagnostic diagnostic) {
Lei Zhang40056702015-09-11 14:31:27 -040051 if (!diagnostic) return SPV_ERROR_INVALID_DIAGNOSTIC;
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010052
David Netoc9786432015-09-01 18:05:14 -040053 if (diagnostic->isTextSource) {
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010054 // NOTE: This is a text position
55 // NOTE: add 1 to the line as editors start at line 1, we are counting new
56 // line characters to start at line 0
57 std::cerr << "error: " << diagnostic->position.line + 1 << ": "
58 << diagnostic->position.column + 1 << ": " << diagnostic->error
59 << "\n";
60 return SPV_SUCCESS;
David Netoc9786432015-09-01 18:05:14 -040061 } else {
62 // NOTE: Assume this is a binary position
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010063 std::cerr << "error: " << diagnostic->position.index << ": "
64 << diagnostic->error << "\n";
65 return SPV_SUCCESS;
66 }
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010067}
Andrew Woloszyn71fc0552015-09-24 10:26:51 -040068
David Neto01656362015-11-20 10:44:41 -050069namespace libspirv {
70
Andrew Woloszyn71fc0552015-09-24 10:26:51 -040071DiagnosticStream::~DiagnosticStream() {
Lei Zhang755f97f2016-09-02 18:06:18 -040072 if (error_ != SPV_FAILED_MATCH && consumer_ != nullptr) {
Lei Zhang6849a3c2016-09-21 12:44:37 -040073 auto level = spvtools::SPV_MSG_ERROR;
Lei Zhang755f97f2016-09-02 18:06:18 -040074 switch (error_) {
75 case SPV_SUCCESS:
76 case SPV_REQUESTED_TERMINATION: // Essentially success.
Lei Zhang6849a3c2016-09-21 12:44:37 -040077 level = spvtools::SPV_MSG_INFO;
Lei Zhang755f97f2016-09-02 18:06:18 -040078 break;
79 case SPV_WARNING:
Lei Zhang6849a3c2016-09-21 12:44:37 -040080 level = spvtools::SPV_MSG_WARNINING;
Lei Zhang755f97f2016-09-02 18:06:18 -040081 break;
82 case SPV_UNSUPPORTED:
83 case SPV_ERROR_INTERNAL:
84 case SPV_ERROR_INVALID_TABLE:
Lei Zhang6849a3c2016-09-21 12:44:37 -040085 level = spvtools::SPV_MSG_INTERNAL_ERROR;
Lei Zhang755f97f2016-09-02 18:06:18 -040086 break;
87 case SPV_ERROR_OUT_OF_MEMORY:
Lei Zhang6849a3c2016-09-21 12:44:37 -040088 level = spvtools::SPV_MSG_FATAL;
Lei Zhang755f97f2016-09-02 18:06:18 -040089 break;
90 default:
91 break;
92 }
Lei Zhang35902792016-09-16 15:43:41 -040093 consumer_(level, "input", position_, stream_.str().c_str());
Andrew Woloszyn71fc0552015-09-24 10:26:51 -040094 }
95}
Lei Zhang755f97f2016-09-02 18:06:18 -040096
97void UseDiagnosticAsMessageConsumer(spv_context context,
98 spv_diagnostic* diagnostic) {
99 assert(diagnostic && *diagnostic == nullptr);
100
Lei Zhang6849a3c2016-09-21 12:44:37 -0400101 auto create_diagnostic = [diagnostic](
102 spvtools::spv_message_level_t, const char*,
103 const spv_position_t& position, const char* message) {
Lei Zhang755f97f2016-09-02 18:06:18 -0400104 auto p = position;
105 spvDiagnosticDestroy(*diagnostic); // Avoid memory leak.
106 *diagnostic = spvDiagnosticCreate(&p, message);
107 };
108 SetContextMessageConsumer(context, std::move(create_diagnostic));
109}
110
111std::string spvResultToString(spv_result_t res) {
Umar Arshadc7413852015-12-15 21:44:21 -0500112 std::string out;
113 switch (res) {
114 case SPV_SUCCESS:
115 out = "SPV_SUCCESS";
116 break;
117 case SPV_UNSUPPORTED:
118 out = "SPV_UNSUPPORTED";
119 break;
120 case SPV_END_OF_STREAM:
121 out = "SPV_END_OF_STREAM";
122 break;
123 case SPV_WARNING:
124 out = "SPV_WARNING";
125 break;
126 case SPV_FAILED_MATCH:
127 out = "SPV_FAILED_MATCH";
128 break;
129 case SPV_REQUESTED_TERMINATION:
130 out = "SPV_REQUESTED_TERMINATION";
131 break;
132 case SPV_ERROR_INTERNAL:
133 out = "SPV_ERROR_INTERNAL";
134 break;
135 case SPV_ERROR_OUT_OF_MEMORY:
136 out = "SPV_ERROR_OUT_OF_MEMORY";
137 break;
138 case SPV_ERROR_INVALID_POINTER:
139 out = "SPV_ERROR_INVALID_POINTER";
140 break;
141 case SPV_ERROR_INVALID_BINARY:
142 out = "SPV_ERROR_INVALID_BINARY";
143 break;
144 case SPV_ERROR_INVALID_TEXT:
145 out = "SPV_ERROR_INVALID_TEXT";
146 break;
147 case SPV_ERROR_INVALID_TABLE:
148 out = "SPV_ERROR_INVALID_TABLE";
149 break;
150 case SPV_ERROR_INVALID_VALUE:
151 out = "SPV_ERROR_INVALID_VALUE";
152 break;
153 case SPV_ERROR_INVALID_DIAGNOSTIC:
154 out = "SPV_ERROR_INVALID_DIAGNOSTIC";
155 break;
156 case SPV_ERROR_INVALID_LOOKUP:
157 out = "SPV_ERROR_INVALID_LOOKUP";
158 break;
159 case SPV_ERROR_INVALID_ID:
160 out = "SPV_ERROR_INVALID_ID";
161 break;
162 case SPV_ERROR_INVALID_CFG:
163 out = "SPV_ERROR_INVALID_CFG";
164 break;
165 case SPV_ERROR_INVALID_LAYOUT:
166 out = "SPV_ERROR_INVALID_LAYOUT";
167 break;
168 default:
169 out = "Unknown Error";
170 }
171 return out;
172}
David Neto01656362015-11-20 10:44:41 -0500173
174} // namespace libspirv