2016-01-07 18:44:22 +00:00
|
|
|
// Copyright (c) 2015-2016 The Khronos Group Inc.
|
2015-05-22 17:26:19 +00:00
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
2015-05-22 17:26:19 +00:00
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2015-05-22 17:26:19 +00:00
|
|
|
//
|
2016-09-01 19:33:59 +00:00
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
2015-05-22 17:26:19 +00:00
|
|
|
|
|
|
|
#include "diagnostic.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
2016-02-17 19:44:00 +00:00
|
|
|
#include "spirv-tools/libspirv.h"
|
2016-09-02 22:06:18 +00:00
|
|
|
#include "table.h"
|
2015-11-11 17:45:23 +00:00
|
|
|
|
2015-05-22 17:26:19 +00:00
|
|
|
// Diagnostic API
|
|
|
|
|
|
|
|
spv_diagnostic spvDiagnosticCreate(const spv_position position,
|
2015-11-02 14:41:20 +00:00
|
|
|
const char* message) {
|
2015-05-22 17:26:19 +00:00
|
|
|
spv_diagnostic diagnostic = new spv_diagnostic_t;
|
2015-09-11 18:31:27 +00:00
|
|
|
if (!diagnostic) return nullptr;
|
2015-05-22 17:26:19 +00:00
|
|
|
size_t length = strlen(message) + 1;
|
|
|
|
diagnostic->error = new char[length];
|
2015-09-11 18:31:27 +00:00
|
|
|
if (!diagnostic->error) {
|
|
|
|
delete diagnostic;
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-05-22 17:26:19 +00:00
|
|
|
diagnostic->position = *position;
|
2015-09-01 22:05:14 +00:00
|
|
|
diagnostic->isTextSource = false;
|
2015-05-22 17:26:19 +00:00
|
|
|
memset(diagnostic->error, 0, length);
|
|
|
|
strncpy(diagnostic->error, message, length);
|
|
|
|
return diagnostic;
|
|
|
|
}
|
|
|
|
|
|
|
|
void spvDiagnosticDestroy(spv_diagnostic diagnostic) {
|
2015-09-11 18:31:27 +00:00
|
|
|
if (!diagnostic) return;
|
2016-02-18 23:41:16 +00:00
|
|
|
delete[] diagnostic->error;
|
2015-05-22 17:26:19 +00:00
|
|
|
delete diagnostic;
|
|
|
|
}
|
|
|
|
|
|
|
|
spv_result_t spvDiagnosticPrint(const spv_diagnostic diagnostic) {
|
2015-09-11 18:31:27 +00:00
|
|
|
if (!diagnostic) return SPV_ERROR_INVALID_DIAGNOSTIC;
|
2015-05-22 17:26:19 +00:00
|
|
|
|
2015-09-01 22:05:14 +00:00
|
|
|
if (diagnostic->isTextSource) {
|
2015-05-22 17:26:19 +00:00
|
|
|
// NOTE: This is a text position
|
|
|
|
// NOTE: add 1 to the line as editors start at line 1, we are counting new
|
|
|
|
// line characters to start at line 0
|
|
|
|
std::cerr << "error: " << diagnostic->position.line + 1 << ": "
|
|
|
|
<< diagnostic->position.column + 1 << ": " << diagnostic->error
|
|
|
|
<< "\n";
|
|
|
|
return SPV_SUCCESS;
|
2015-09-01 22:05:14 +00:00
|
|
|
} else {
|
|
|
|
// NOTE: Assume this is a binary position
|
2015-05-22 17:26:19 +00:00
|
|
|
std::cerr << "error: " << diagnostic->position.index << ": "
|
|
|
|
<< diagnostic->error << "\n";
|
|
|
|
return SPV_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
2015-09-24 14:26:51 +00:00
|
|
|
|
2015-11-20 15:44:41 +00:00
|
|
|
namespace libspirv {
|
|
|
|
|
2015-09-24 14:26:51 +00:00
|
|
|
DiagnosticStream::~DiagnosticStream() {
|
2016-09-02 22:06:18 +00:00
|
|
|
using spvtools::MessageLevel;
|
|
|
|
if (error_ != SPV_FAILED_MATCH && consumer_ != nullptr) {
|
|
|
|
auto level = MessageLevel::Error;
|
|
|
|
switch (error_) {
|
|
|
|
case SPV_SUCCESS:
|
|
|
|
case SPV_REQUESTED_TERMINATION: // Essentially success.
|
|
|
|
level = MessageLevel::Info;
|
|
|
|
break;
|
|
|
|
case SPV_WARNING:
|
|
|
|
level = MessageLevel::Warning;
|
|
|
|
break;
|
|
|
|
case SPV_UNSUPPORTED:
|
|
|
|
case SPV_ERROR_INTERNAL:
|
|
|
|
case SPV_ERROR_INVALID_TABLE:
|
|
|
|
level = MessageLevel::InternalError;
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_OUT_OF_MEMORY:
|
|
|
|
level = MessageLevel::Fatal;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
consumer_(level, "", position_, stream_.str().c_str());
|
2015-09-24 14:26:51 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-02 22:06:18 +00:00
|
|
|
|
|
|
|
void UseDiagnosticAsMessageConsumer(spv_context context,
|
|
|
|
spv_diagnostic* diagnostic) {
|
|
|
|
assert(diagnostic && *diagnostic == nullptr);
|
|
|
|
|
|
|
|
auto create_diagnostic = [diagnostic](spvtools::MessageLevel, const char*,
|
|
|
|
const spv_position_t& position,
|
|
|
|
const char* message) {
|
|
|
|
auto p = position;
|
|
|
|
spvDiagnosticDestroy(*diagnostic); // Avoid memory leak.
|
|
|
|
*diagnostic = spvDiagnosticCreate(&p, message);
|
|
|
|
};
|
|
|
|
SetContextMessageConsumer(context, std::move(create_diagnostic));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string spvResultToString(spv_result_t res) {
|
2015-12-16 02:44:21 +00:00
|
|
|
std::string out;
|
|
|
|
switch (res) {
|
|
|
|
case SPV_SUCCESS:
|
|
|
|
out = "SPV_SUCCESS";
|
|
|
|
break;
|
|
|
|
case SPV_UNSUPPORTED:
|
|
|
|
out = "SPV_UNSUPPORTED";
|
|
|
|
break;
|
|
|
|
case SPV_END_OF_STREAM:
|
|
|
|
out = "SPV_END_OF_STREAM";
|
|
|
|
break;
|
|
|
|
case SPV_WARNING:
|
|
|
|
out = "SPV_WARNING";
|
|
|
|
break;
|
|
|
|
case SPV_FAILED_MATCH:
|
|
|
|
out = "SPV_FAILED_MATCH";
|
|
|
|
break;
|
|
|
|
case SPV_REQUESTED_TERMINATION:
|
|
|
|
out = "SPV_REQUESTED_TERMINATION";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INTERNAL:
|
|
|
|
out = "SPV_ERROR_INTERNAL";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_OUT_OF_MEMORY:
|
|
|
|
out = "SPV_ERROR_OUT_OF_MEMORY";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_POINTER:
|
|
|
|
out = "SPV_ERROR_INVALID_POINTER";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_BINARY:
|
|
|
|
out = "SPV_ERROR_INVALID_BINARY";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_TEXT:
|
|
|
|
out = "SPV_ERROR_INVALID_TEXT";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_TABLE:
|
|
|
|
out = "SPV_ERROR_INVALID_TABLE";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_VALUE:
|
|
|
|
out = "SPV_ERROR_INVALID_VALUE";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_DIAGNOSTIC:
|
|
|
|
out = "SPV_ERROR_INVALID_DIAGNOSTIC";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_LOOKUP:
|
|
|
|
out = "SPV_ERROR_INVALID_LOOKUP";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_ID:
|
|
|
|
out = "SPV_ERROR_INVALID_ID";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_CFG:
|
|
|
|
out = "SPV_ERROR_INVALID_CFG";
|
|
|
|
break;
|
|
|
|
case SPV_ERROR_INVALID_LAYOUT:
|
|
|
|
out = "SPV_ERROR_INVALID_LAYOUT";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
out = "Unknown Error";
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
2015-11-20 15:44:41 +00:00
|
|
|
|
|
|
|
} // namespace libspirv
|