• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2015-2016 The Khronos Group Inc.
2 //
3 // 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
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // 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.
14 
15 #include "source/diagnostic.h"
16 
17 #include <cassert>
18 #include <cstring>
19 #include <iostream>
20 #include <sstream>
21 #include <utility>
22 
23 #include "source/table.h"
24 
25 // Diagnostic API
26 
spvDiagnosticCreate(const spv_position position,const char * message)27 spv_diagnostic spvDiagnosticCreate(const spv_position position,
28                                    const char* message) {
29   spv_diagnostic diagnostic = new spv_diagnostic_t;
30   if (!diagnostic) return nullptr;
31   size_t length = strlen(message) + 1;
32   diagnostic->error = new char[length];
33   if (!diagnostic->error) {
34     delete diagnostic;
35     return nullptr;
36   }
37   diagnostic->position = *position;
38   diagnostic->isTextSource = false;
39   memset(diagnostic->error, 0, length);
40   strcpy(diagnostic->error, message);
41   return diagnostic;
42 }
43 
spvDiagnosticDestroy(spv_diagnostic diagnostic)44 void spvDiagnosticDestroy(spv_diagnostic diagnostic) {
45   if (!diagnostic) return;
46   delete[] diagnostic->error;
47   delete diagnostic;
48 }
49 
spvDiagnosticPrint(const spv_diagnostic diagnostic)50 spv_result_t spvDiagnosticPrint(const spv_diagnostic diagnostic) {
51   if (!diagnostic) return SPV_ERROR_INVALID_DIAGNOSTIC;
52 
53   if (diagnostic->isTextSource) {
54     // 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;
61   }
62 
63   // NOTE: Assume this is a binary position
64   std::cerr << "error: ";
65   if (diagnostic->position.index > 0)
66     std::cerr << diagnostic->position.index << ": ";
67   std::cerr << diagnostic->error << "\n";
68   return SPV_SUCCESS;
69 }
70 
71 namespace spvtools {
72 
DiagnosticStream(DiagnosticStream && other)73 DiagnosticStream::DiagnosticStream(DiagnosticStream&& other)
74     : stream_(),
75       position_(other.position_),
76       consumer_(other.consumer_),
77       disassembled_instruction_(std::move(other.disassembled_instruction_)),
78       error_(other.error_) {
79   // Prevent the other object from emitting output during destruction.
80   other.error_ = SPV_FAILED_MATCH;
81   // Some platforms are missing support for std::ostringstream functionality,
82   // including:  move constructor, swap method.  Either would have been a
83   // better choice than copying the string.
84   stream_ << other.stream_.str();
85 }
86 
~DiagnosticStream()87 DiagnosticStream::~DiagnosticStream() {
88   if (error_ != SPV_FAILED_MATCH && consumer_ != nullptr) {
89     auto level = SPV_MSG_ERROR;
90     switch (error_) {
91       case SPV_SUCCESS:
92       case SPV_REQUESTED_TERMINATION:  // Essentially success.
93         level = SPV_MSG_INFO;
94         break;
95       case SPV_WARNING:
96         level = SPV_MSG_WARNING;
97         break;
98       case SPV_UNSUPPORTED:
99       case SPV_ERROR_INTERNAL:
100       case SPV_ERROR_INVALID_TABLE:
101         level = SPV_MSG_INTERNAL_ERROR;
102         break;
103       case SPV_ERROR_OUT_OF_MEMORY:
104         level = SPV_MSG_FATAL;
105         break;
106       default:
107         break;
108     }
109     if (disassembled_instruction_.size() > 0)
110       stream_ << std::endl << "  " << disassembled_instruction_ << std::endl;
111 
112     consumer_(level, "input", position_, stream_.str().c_str());
113   }
114 }
115 
UseDiagnosticAsMessageConsumer(spv_context context,spv_diagnostic * diagnostic)116 void UseDiagnosticAsMessageConsumer(spv_context context,
117                                     spv_diagnostic* diagnostic) {
118   assert(diagnostic && *diagnostic == nullptr);
119 
120   auto create_diagnostic = [diagnostic](spv_message_level_t, const char*,
121                                         const spv_position_t& position,
122                                         const char* message) {
123     auto p = position;
124     spvDiagnosticDestroy(*diagnostic);  // Avoid memory leak.
125     *diagnostic = spvDiagnosticCreate(&p, message);
126   };
127   SetContextMessageConsumer(context, std::move(create_diagnostic));
128 }
129 
spvResultToString(spv_result_t res)130 std::string spvResultToString(spv_result_t res) {
131   std::string out;
132   switch (res) {
133     case SPV_SUCCESS:
134       out = "SPV_SUCCESS";
135       break;
136     case SPV_UNSUPPORTED:
137       out = "SPV_UNSUPPORTED";
138       break;
139     case SPV_END_OF_STREAM:
140       out = "SPV_END_OF_STREAM";
141       break;
142     case SPV_WARNING:
143       out = "SPV_WARNING";
144       break;
145     case SPV_FAILED_MATCH:
146       out = "SPV_FAILED_MATCH";
147       break;
148     case SPV_REQUESTED_TERMINATION:
149       out = "SPV_REQUESTED_TERMINATION";
150       break;
151     case SPV_ERROR_INTERNAL:
152       out = "SPV_ERROR_INTERNAL";
153       break;
154     case SPV_ERROR_OUT_OF_MEMORY:
155       out = "SPV_ERROR_OUT_OF_MEMORY";
156       break;
157     case SPV_ERROR_INVALID_POINTER:
158       out = "SPV_ERROR_INVALID_POINTER";
159       break;
160     case SPV_ERROR_INVALID_BINARY:
161       out = "SPV_ERROR_INVALID_BINARY";
162       break;
163     case SPV_ERROR_INVALID_TEXT:
164       out = "SPV_ERROR_INVALID_TEXT";
165       break;
166     case SPV_ERROR_INVALID_TABLE:
167       out = "SPV_ERROR_INVALID_TABLE";
168       break;
169     case SPV_ERROR_INVALID_VALUE:
170       out = "SPV_ERROR_INVALID_VALUE";
171       break;
172     case SPV_ERROR_INVALID_DIAGNOSTIC:
173       out = "SPV_ERROR_INVALID_DIAGNOSTIC";
174       break;
175     case SPV_ERROR_INVALID_LOOKUP:
176       out = "SPV_ERROR_INVALID_LOOKUP";
177       break;
178     case SPV_ERROR_INVALID_ID:
179       out = "SPV_ERROR_INVALID_ID";
180       break;
181     case SPV_ERROR_INVALID_CFG:
182       out = "SPV_ERROR_INVALID_CFG";
183       break;
184     case SPV_ERROR_INVALID_LAYOUT:
185       out = "SPV_ERROR_INVALID_LAYOUT";
186       break;
187     default:
188       out = "Unknown Error";
189   }
190   return out;
191 }
192 
193 }  // namespace spvtools
194