Source code for helion.exc

from __future__ import annotations

from typing import TYPE_CHECKING

if TYPE_CHECKING:
    from ._compiler.source_location import SourceLocation


[docs] class Base(RuntimeError):
[docs] def report(self) -> str: raise NotImplementedError
class _FixedMessage(Base): message = "" location_suffix = "\nWhile processing:\n{location}" def __init__(self, *args: object, **kwargs: object) -> None: from ._compiler.source_location import current_location self.location: SourceLocation = current_location() msg = self.__class__.message.format(*args, **kwargs) self.base_msg_len: int = len(msg) if self.location and "Original traceback:" not in msg: msg += self.location_suffix.format(location=self.location.format()) super().__init__(msg)
[docs] class BaseError(_FixedMessage): message = "An error occurred."
[docs] def report(self) -> str: return f"ERROR[{type(self).__name__}]: {self!s}"
[docs] class NotInsideKernel(BaseError): message = ( "Functions found in helion.language.* must be called from inside a kernel. " "Did you forget the @helion.kernel decorator?" )
[docs] class NamingConflict(BaseError): message = "The variable name {} is reserved in Helion and cannot be used."
[docs] class ClosuresNotSupported(BaseError): message = "A closure ({0!r}) was found in the kernel. Closures are not supported."
[docs] class ClosureMutation(BaseError): message = "Closure mutation (of {0}) is not allowed in a function arg."
[docs] class GlobalMutation(BaseError): message = "Global mutation (of {0}) is not allowed in a function arg."
[docs] class LoopFunctionNotInFor(BaseError): message = "{0} must be called from a for loop, e.g. `for ... in {0}(...):"
[docs] class NestedDeviceLoopsConflict(BaseError): message = "Nested device loops must have distinct block sizes."
[docs] class DeviceLoopElseBlock(BaseError): message = "for...else block is not allowed in a {0} device loop."
[docs] class LoopDependencyError(BaseError): message = "Loop dependency detected: '{0}' was written in a previous loop."
[docs] class TopLevelStatementBetweenLoops(BaseError): message = "Statements cannot appear between top level loops."
[docs] class NestedGridLoop(BaseError): message = "Grid loops must be at the top level of a function."
[docs] class RankMismatch(BaseError): message = "Expected ndim={expected_ndim}, but got ndim={actual_ndim}{shape_part}. You have {direction}."
[docs] def __init__( self, expected_ndim: int, actual_ndim: int, shape_info: str = "" ) -> None: if actual_ndim > expected_ndim: direction = "too many indices" elif actual_ndim < expected_ndim: direction = "too few indices" else: direction = "indices that don't match expected structure" shape_part = f" ({shape_info})" if shape_info else "" super().__init__( expected_ndim=expected_ndim, actual_ndim=actual_ndim, shape_part=shape_part, direction=direction, )
[docs] class InvalidIndexingType(BaseError): message = "Expected tile/int/None/tensor/etc in tensor[...], got {0!s}."
[docs] class RequiresTensorInAssignment(BaseError): message = "Expected tensor in right-hand side of assignment, got {0!s}."
[docs] class NotAllowedOnDevice(BaseError): message = "The statement {} is not allowed inside the `hl.tile` or `hl.grid` loop."
[docs] class HostTensorDirectUsage(BaseError): message = ( "Direct use of host tensor '{0}' in op '{1}' not allowed inside the `hl.tile` or `hl.grid` loop. " "First load it using {0}[...] or hl.load({0}, ...)." )
[docs] class ShapeSpecializingCall(BaseError): message = "Call would force shape specialization, try `hl.specialize(x)` or `hl.constexpr`."
[docs] class ShapeSpecializingAllocation(BaseError): message = "Using a tensor size in a device allocation requires specialization. Use `hl.specialize` or `hl.constexpr` to specialize the size."
[docs] class SpecializeOnDevice(BaseError): message = "hl.specialize() must be called outside the `hl.tile` or `hl.grid` loop."
[docs] class SpecializeArgType(BaseError): message = "hl.specialize() must be called on a size from an input tensor, got: {}"
[docs] class FailedToUnpackTupleAssign(BaseError): message = "Failed to unpack values in tuple assignment. Expected a sequence of size {0}, got type: {1!s}."
[docs] class RegisterTunableArgTypes(BaseError): message = "Expected string literal and ConfigSpecFragment literal, got {0} and {1}."
[docs] class TunableTypeNotSupported(BaseError): message = "hl.register_tunable() only supports integer, float, and boolean types, got {0!s}."
[docs] class TunableNameConflict(BaseError): message = ( "Tunable parameter with name {0!s} already exists. Please use a different name." )
[docs] class ConfigSpecFragmentWithSymInt(BaseError): message = "ConfigSpecFragment with SymInt arg is not supported. hl.constexpr or hl.specialize may be used to specialize the SymInt value."
[docs] class FailedToUnpackTile(BaseError): message = ( "Failed to unpack a tile into a tuple assignment. " "Expected a sequence, but got a single tile. " "Did you mix up `hl.tile(x)` and `hl.tile([x])`?" )
[docs] class OverpackedTile(BaseError): message = ( "Got a tile wrapped inside a container when indexing a tensor: {0!s}\n" "Did you mix up `hl.tile([x])` and `hl.tile(x)`?" )
[docs] class AssignmentMultipleTargets(NotAllowedOnDevice): message = "Assignment with multiple targets (a=b=1) is not allowed inside the `hl.tile` or `hl.grid` loop."
[docs] class InvalidAssignment(NotAllowedOnDevice): message = "Assignment target must be Name or Subscript inside the `hl.tile` or `hl.grid` loop."
[docs] class NonTensorSubscriptAssign(BaseError): message = "Expected tensor in subscript assignment, got {0!s} and {1!s}."
[docs] class ShapeMismatch(BaseError): message = "Shape mismatch between {0!s} and {1!s}."
[docs] class DeviceAPIOnHost(BaseError): message = "{} is only allowed inside the `hl.tile` or `hl.grid` loop."
[docs] class StatementNotSupported(BaseError): message = "The statement {} is not supported."
[docs] class CantReadOnDevice(BaseError): message = "Cannot read {0!s} inside the `hl.tile` or `hl.grid` loop."
[docs] class UndefinedVariable(BaseError): message = "{} is not defined."
[docs] class InvalidDeviceForLoop(BaseError): message = "For loops on device must use `hl.tile` or `hl.grid`, got {0!s}."
[docs] class StarredArgsNotSupportedOnDevice(BaseError): message = "*/** args are not supported inside the `hl.tile` or `hl.grid` loop."
[docs] class IncorrectTileUsage(BaseError): message = "Tiles can only be used in tensor indexing (`x[tile]`) or in `hl.*` ops (e.g. `hl.zeros(tile)`), used in {}"
[docs] class TracedArgNotSupported(BaseError): message = "{!s} is not supported as an arg to traced functions."
[docs] class NotEnoughConfigs(BaseError): message = "FiniteSearch requires at least two configs, but got {0}."
[docs] class CantCombineTypesInControlFlow(BaseError): message = "Cannot combine types for {0!r} in control flow: {1} and {2}"
[docs] class ErrorCompilingKernel(BaseError): message = "{0} errors and {1} warnings occurred (see above)"
[docs] class NoTensorArgs(BaseError): message = "Kernel took no tensor or device args, unclear what device to use."
class _WrapException(BaseError): message = "{name}: {msg}" def __init__(self, e: Exception) -> None: super().__init__(name=type(e).__name__, msg=str(e))
[docs] class InvalidConfig(BaseError): message = "{}"
[docs] class InductorLoweringError(BaseError): message = "{}"
[docs] class DecoratorAfterHelionKernelDecorator(BaseError): message = "Decorators after helion kernel decorator are not allowed."
[docs] class InternalError(_WrapException): pass
[docs] class TorchOpTracingError(_WrapException): pass
[docs] class TritonError(BaseError): message = "Error running generated Triton program:\n{1}\n{0}"
[docs] class BaseWarning(_FixedMessage): message = "A warning occurred."
[docs] def report(self) -> str: return f"WARNING[{type(self).__name__}]: {self!s}"
[docs] class TensorOperationInWrapper(BaseWarning): message = ( "A tensor operation outside of the `hl.tile` or `hl.grid` loop will not be fused " "in the generated kernel.\n" "Use @helion.kernel(ignore_warnings=[helion.exc.TensorOperationInWrapper]) to suppress this warning.\n" "If this is not a tensor operation, please report this as a bug." )
[docs] class TensorOperationsInHostCall(TensorOperationInWrapper): message = ( "A tensor operation outside of the `hl.tile` or `hl.grid` loop will not be fused " "in the generated kernel: {}" )
[docs] class WrongDevice(BaseWarning): message = "Operation {0} returned a tensor on {1} device, but the kernel is on {2} device."
[docs] class AutotuningDisallowedInEnvironment(BaseError): message = "Autotuning is disabled {0}, please provide a config to @helion.kernel via the config= argument."
[docs] class UnsupportedPythonType(BaseError): message = "{0} is not supported in Helion kernels"
[docs] class TypeInferenceError(BaseError): message = "{0}"
[docs] class NotAllowedInHelperFunction(BaseError): message = "This operation is not allowed inside helper functions. It requires kernel context."
[docs] class CannotModifyHostVariableOnDevice(BaseError): message = "Cannot modify host variable '{0}' inside `hl.tile` or `hl.grid` loop without subscript assignment. Use '{0}[tile] = ...' instead."
[docs] class CannotReadDeviceVariableOnHost(BaseError): message = "Cannot read variable '{0}' defined inside `hl.tile` or `hl.grid` loop from host code."
[docs] class DeviceTensorSubscriptAssignmentNotAllowed(BaseError): message = "Cannot assign to subscript of device tensor '{0}'."
class InvalidSequenceSubscription(BaseError): message = "Cannot subscript a sequence with non constant indices. Got '{0!s}'. "
[docs] class InvalidAPIUsage(BaseError): message = "Invalid usage of Helion API: {0}"