blob: 2a3517b4d8151c6595e9c668ab3ca8a0f6070dc0 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*/
/*!
* \file src/ir/replace_global_vars.cc
* \brief IRModule transform to replace GlobalVar instances across any IR type.
*/
#include <tvm/ffi/container/variant.h>
#include <tvm/ffi/reflection/registry.h>
#include <tvm/ir/replace_global_vars.h>
#include <vector>
namespace tvm {
namespace transform {
IRModule ReplaceGlobalVars(IRModule mod, ffi::Map<GlobalVar, GlobalVar> replacements) {
if (replacements.empty()) {
return mod;
}
std::vector<GlobalVar> to_remove;
IRModule updates;
const auto& vtable = GlobalVarReplacer::vtable();
for (const auto& [old_gvar, old_func] : mod->functions) {
auto new_gvar = replacements.Get(old_gvar).value_or(old_gvar);
auto new_func = vtable(old_func, replacements);
if (!new_gvar.same_as(old_gvar)) {
to_remove.push_back(old_gvar);
}
if (!old_gvar.same_as(new_gvar) || !old_func.same_as(new_func)) {
updates->Add(new_gvar, new_func);
}
}
if (to_remove.size() || updates->functions.size()) {
auto write_ptr = mod.CopyOnWrite();
for (const auto& old_gvar : to_remove) {
write_ptr->Remove(old_gvar);
}
write_ptr->Update(updates);
}
return mod;
}
TVM_FFI_STATIC_INIT_BLOCK() {
namespace refl = tvm::ffi::reflection;
refl::GlobalDef().def("transform.ReplaceGlobalVars", ReplaceGlobalVars);
}
IRModule ModuleReplaceGlobalVars(
IRModule mod,
ffi::Map<ffi::Variant<ffi::String, GlobalVar>, ffi::Variant<ffi::String, GlobalVar>>
replacements) {
ffi::Map<GlobalVar, GlobalVar> gvar_replacements;
for (const auto& [before, after] : replacements) {
GlobalVar gvar_before;
if (auto gvar = before.as<GlobalVar>()) {
gvar_before = gvar.value();
} else if (auto str = before.as<ffi::String>()) {
gvar_before = mod->GetGlobalVar(str.value());
} else {
TVM_FFI_THROW(InternalError)
<< "ffi::Variant<ffi::String,GlobalVar> must contain either ffi::String or GlobalVar";
}
GlobalVar gvar_after;
if (auto gvar = after.as<GlobalVar>()) {
gvar_after = gvar.value();
} else if (auto str = after.as<ffi::String>()) {
gvar_after = gvar_before;
gvar_after.CopyOnWrite()->name_hint = str.value();
} else {
TVM_FFI_THROW(InternalError)
<< "ffi::Variant<ffi::String,GlobalVar> must contain either ffi::String or GlobalVar";
}
gvar_replacements.Set(gvar_before, gvar_after);
}
return ReplaceGlobalVars(mod, gvar_replacements);
}
TVM_FFI_STATIC_INIT_BLOCK() {
namespace refl = tvm::ffi::reflection;
refl::GlobalDef().def("ir.Module_ReplaceGlobalVars", ModuleReplaceGlobalVars);
}
} // namespace transform
} // namespace tvm