2018-04-11 05:44:28 +03:00
|
|
|
// Copyright (c) 2016 Google Inc.
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
#include "source/opt/build_module.h"
|
2018-04-11 05:44:28 +03:00
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
#include <utility>
|
|
|
|
#include <vector>
|
2018-04-11 05:44:28 +03:00
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
#include "source/opt/ir_context.h"
|
|
|
|
#include "source/opt/ir_loader.h"
|
|
|
|
#include "source/table.h"
|
|
|
|
#include "source/util/make_unique.h"
|
2018-04-11 05:44:28 +03:00
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
namespace spvtools {
|
2018-04-11 05:44:28 +03:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
// Sets the module header for IrLoader. Meets the interface requirement of
|
|
|
|
// spvBinaryParse().
|
|
|
|
spv_result_t SetSpvHeader(void* builder, spv_endianness_t, uint32_t magic,
|
|
|
|
uint32_t version, uint32_t generator,
|
|
|
|
uint32_t id_bound, uint32_t reserved) {
|
2018-09-03 07:14:20 +03:00
|
|
|
reinterpret_cast<opt::IrLoader*>(builder)->SetModuleHeader(
|
2018-04-11 05:44:28 +03:00
|
|
|
magic, version, generator, id_bound, reserved);
|
|
|
|
return SPV_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Processes a parsed instruction for IrLoader. Meets the interface requirement
|
|
|
|
// of spvBinaryParse().
|
|
|
|
spv_result_t SetSpvInst(void* builder, const spv_parsed_instruction_t* inst) {
|
2018-09-03 07:14:20 +03:00
|
|
|
if (reinterpret_cast<opt::IrLoader*>(builder)->AddInstruction(inst)) {
|
2018-04-11 05:44:28 +03:00
|
|
|
return SPV_SUCCESS;
|
|
|
|
}
|
|
|
|
return SPV_ERROR_INVALID_BINARY;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
std::unique_ptr<opt::IRContext> BuildModule(spv_target_env env,
|
|
|
|
MessageConsumer consumer,
|
|
|
|
const uint32_t* binary,
|
|
|
|
const size_t size) {
|
2020-11-30 05:54:40 +03:00
|
|
|
return BuildModule(env, consumer, binary, size, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<opt::IRContext> BuildModule(spv_target_env env,
|
|
|
|
MessageConsumer consumer,
|
|
|
|
const uint32_t* binary,
|
|
|
|
const size_t size,
|
|
|
|
bool extra_line_tracking) {
|
2018-04-11 05:44:28 +03:00
|
|
|
auto context = spvContextCreate(env);
|
2018-09-03 07:14:20 +03:00
|
|
|
SetContextMessageConsumer(context, consumer);
|
2018-04-11 05:44:28 +03:00
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
auto irContext = MakeUnique<opt::IRContext>(env, consumer);
|
|
|
|
opt::IrLoader loader(consumer, irContext->module());
|
2020-11-30 05:54:40 +03:00
|
|
|
loader.SetExtraLineTracking(extra_line_tracking);
|
2018-04-11 05:44:28 +03:00
|
|
|
|
|
|
|
spv_result_t status = spvBinaryParse(context, &loader, binary, size,
|
|
|
|
SetSpvHeader, SetSpvInst, nullptr);
|
|
|
|
loader.EndModule();
|
|
|
|
|
|
|
|
spvContextDestroy(context);
|
|
|
|
|
|
|
|
return status == SPV_SUCCESS ? std::move(irContext) : nullptr;
|
|
|
|
}
|
|
|
|
|
2018-09-03 07:14:20 +03:00
|
|
|
std::unique_ptr<opt::IRContext> BuildModule(spv_target_env env,
|
|
|
|
MessageConsumer consumer,
|
|
|
|
const std::string& text,
|
|
|
|
uint32_t assemble_options) {
|
2018-04-11 05:44:28 +03:00
|
|
|
SpirvTools t(env);
|
|
|
|
t.SetMessageConsumer(consumer);
|
|
|
|
std::vector<uint32_t> binary;
|
|
|
|
if (!t.Assemble(text, &binary, assemble_options)) return nullptr;
|
|
|
|
return BuildModule(env, consumer, binary.data(), binary.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace spvtools
|