2019-04-08 07:14:02 +03:00
|
|
|
# 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.
|
2018-09-20 01:55:12 +03:00
|
|
|
# pylint: disable=no-else-return, unidiomatic-typecheck, undefined-variable, wildcard-import
|
2018-11-03 00:27:34 +03:00
|
|
|
"""A global module storing everything needed to interpret or compile a Relay program."""
|
2018-10-18 23:06:48 +03:00
|
|
|
from .base import register_relay_node, RelayNode
|
2018-10-20 07:56:09 +03:00
|
|
|
from .._ffi import base as _base
|
2018-09-20 01:55:12 +03:00
|
|
|
from . import _make
|
2018-11-03 00:27:34 +03:00
|
|
|
from . import _module
|
2018-10-20 07:56:09 +03:00
|
|
|
from . import expr as _expr
|
2019-02-16 09:37:49 +03:00
|
|
|
from . import ty as _ty
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
@register_relay_node
|
2018-11-03 00:27:34 +03:00
|
|
|
class Module(RelayNode):
|
|
|
|
"""The global Relay module containing collection of functions.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
Each global function is identified by an unique tvm.relay.GlobalVar.
|
2018-11-03 00:27:34 +03:00
|
|
|
tvm.relay.GlobalVar and Module is necessary in order to enable
|
2018-10-20 07:56:09 +03:00
|
|
|
recursions in function to avoid cyclic reference in the function.x
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
Parameters
|
|
|
|
----------
|
2019-07-02 05:29:56 +03:00
|
|
|
functions: Optional[dict].
|
2018-10-20 07:56:09 +03:00
|
|
|
Map of global var to Function
|
|
|
|
"""
|
2019-02-16 09:37:49 +03:00
|
|
|
def __init__(self, functions=None, type_definitions=None):
|
2018-10-20 07:56:09 +03:00
|
|
|
if functions is None:
|
|
|
|
functions = {}
|
|
|
|
elif isinstance(functions, dict):
|
|
|
|
mapped_funcs = {}
|
|
|
|
for k, v in functions.items():
|
|
|
|
if isinstance(k, _base.string_types):
|
|
|
|
k = _expr.GlobalVar(k)
|
|
|
|
if not isinstance(k, _expr.GlobalVar):
|
|
|
|
raise TypeError("Expect functions to be Dict[GlobalVar, Function]")
|
|
|
|
mapped_funcs[k] = v
|
|
|
|
functions = mapped_funcs
|
2019-02-16 09:37:49 +03:00
|
|
|
if type_definitions is None:
|
|
|
|
type_definitions = {}
|
|
|
|
elif isinstance(type_definitions, dict):
|
|
|
|
mapped_type_defs = {}
|
|
|
|
for k, v in type_definitions.items():
|
|
|
|
if isinstance(k, _base.string_types):
|
|
|
|
k = _ty.GlobalTypeVar(k)
|
|
|
|
if not isinstance(k, _ty.GlobalTypeVar):
|
|
|
|
raise TypeError("Expect type_definitions to be Dict[GlobalTypeVar, Type]")
|
|
|
|
mapped_type_defs[k] = v
|
|
|
|
type_definitions = mapped_type_defs
|
|
|
|
self.__init_handle_by_constructor__(_make.Module, functions, type_definitions)
|
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
|
2019-02-16 09:37:49 +03:00
|
|
|
def __setitem__(self, var, val):
|
|
|
|
"""Add a mapping to the module.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
---------
|
|
|
|
var: GlobalVar
|
2019-02-16 09:37:49 +03:00
|
|
|
The global variable.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2019-02-16 09:37:49 +03:00
|
|
|
val: Union[Function, Type]
|
|
|
|
The value.
|
2018-09-20 01:55:12 +03:00
|
|
|
"""
|
2019-02-16 09:37:49 +03:00
|
|
|
return self._add(var, val)
|
2018-10-31 01:29:36 +03:00
|
|
|
|
2019-02-16 09:37:49 +03:00
|
|
|
def _add(self, var, val, update=False):
|
2019-05-09 09:09:15 +03:00
|
|
|
if isinstance(val, _expr.Expr):
|
2019-02-16 09:37:49 +03:00
|
|
|
if isinstance(var, _base.string_types):
|
2019-07-06 07:23:27 +03:00
|
|
|
if _module.Module_ContainGlobalVar(self, var):
|
|
|
|
var = _module.Module_GetGlobalVar(self, var)
|
|
|
|
else:
|
|
|
|
var = _expr.GlobalVar(var)
|
|
|
|
_module.Module_Add(self, var, val, update)
|
2019-02-16 09:37:49 +03:00
|
|
|
else:
|
|
|
|
assert isinstance(val, _ty.Type)
|
|
|
|
if isinstance(var, _base.string_types):
|
|
|
|
var = _ty.GlobalTypeVar(var)
|
|
|
|
_module.Module_AddDef(self, var, val)
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
def __getitem__(self, var):
|
2019-02-16 09:37:49 +03:00
|
|
|
"""Lookup a global definition by name or by variable.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
2019-07-02 05:29:56 +03:00
|
|
|
var: Union[String, GlobalVar, GlobalTypeVar]
|
2018-10-20 07:56:09 +03:00
|
|
|
The name or global variable.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
2019-02-16 09:37:49 +03:00
|
|
|
val: Union[Function, Type]
|
|
|
|
The definition referenced by :code:`var` (either a function or type).
|
2018-09-20 01:55:12 +03:00
|
|
|
"""
|
2018-10-20 07:56:09 +03:00
|
|
|
if isinstance(var, _base.string_types):
|
2018-11-03 00:27:34 +03:00
|
|
|
return _module.Module_Lookup_str(self, var)
|
2019-02-16 09:37:49 +03:00
|
|
|
elif isinstance(var, _expr.GlobalVar):
|
2018-11-03 00:27:34 +03:00
|
|
|
return _module.Module_Lookup(self, var)
|
2019-02-16 09:37:49 +03:00
|
|
|
else:
|
|
|
|
return _module.Module_LookupDef(self, var)
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
def update(self, other):
|
2018-11-03 00:27:34 +03:00
|
|
|
"""Insert functions in another Module to current one.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
2018-11-03 00:27:34 +03:00
|
|
|
other: Module
|
|
|
|
The module to merge into the current Module.
|
2018-09-20 01:55:12 +03:00
|
|
|
"""
|
2018-10-20 07:56:09 +03:00
|
|
|
if isinstance(other, dict):
|
2018-11-03 00:27:34 +03:00
|
|
|
other = Module(other)
|
|
|
|
return _module.Module_Update(self, other)
|
2018-09-20 01:55:12 +03:00
|
|
|
|
2018-10-20 07:56:09 +03:00
|
|
|
def get_global_var(self, name):
|
|
|
|
"""Get a global variable in the function by name.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
2018-10-20 07:56:09 +03:00
|
|
|
name: str
|
|
|
|
The name of the global variable.
|
2018-09-20 01:55:12 +03:00
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
2018-10-20 07:56:09 +03:00
|
|
|
global_var: GlobalVar
|
|
|
|
The global variable mapped to :code:`name`.
|
|
|
|
|
|
|
|
Raises
|
|
|
|
------
|
|
|
|
tvm.TVMError if we cannot find corresponding global var.
|
2018-09-20 01:55:12 +03:00
|
|
|
"""
|
2018-11-03 00:27:34 +03:00
|
|
|
return _module.Module_GetGlobalVar(self, name)
|
2019-02-16 09:37:49 +03:00
|
|
|
|
|
|
|
def get_global_type_var(self, name):
|
|
|
|
"""Get a global type variable in the function by name.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name: str
|
|
|
|
The name of the global type variable.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
global_type_var: GlobalTypeVar
|
|
|
|
The global variable mapped to :code:`name`.
|
|
|
|
|
|
|
|
Raises
|
|
|
|
------
|
|
|
|
tvm.TVMError if we cannot find corresponding global type var.
|
|
|
|
"""
|
|
|
|
return _module.Module_GetGlobalTypeVar(self, name)
|
2019-05-09 09:09:15 +03:00
|
|
|
|
2019-07-05 23:58:16 +03:00
|
|
|
def get_constructor(self, tag):
|
|
|
|
"""Look up an ADT constructor by tag.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
tag: int
|
|
|
|
The tag for a constructor.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
constructor: Constructor
|
|
|
|
The constructor associated with the given tag,
|
|
|
|
|
|
|
|
Raises
|
|
|
|
------
|
|
|
|
tvm.TVMError if the corresponding constructor cannot be found.
|
|
|
|
"""
|
|
|
|
return _module.Module_LookupTag(self, tag)
|
|
|
|
|
2019-05-09 09:09:15 +03:00
|
|
|
@staticmethod
|
2019-07-10 04:39:14 +03:00
|
|
|
def from_expr(expr, functions=None, type_defs=None):
|
|
|
|
"""Construct a module from a standalone expression.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
expr: Expr
|
|
|
|
The starting expression
|
|
|
|
global_funcs: Optional[dict]
|
|
|
|
Map of global vars to function definitions
|
|
|
|
type_defs: Optional[dict]
|
|
|
|
Map of global type vars to type definitions
|
|
|
|
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mod: Module
|
|
|
|
A module containing the passed definitions,
|
|
|
|
where expr is set as the entry point
|
|
|
|
(wrapped in a function if necessary)
|
|
|
|
"""
|
|
|
|
funcs = functions if functions is not None else {}
|
|
|
|
defs = type_defs if type_defs is not None else {}
|
|
|
|
return _module.Module_FromExpr(expr, funcs, defs)
|