mirror of
https://github.com/hpcaitech/ColossalAI.git
synced 2025-09-06 11:32:10 +00:00
[fx] add more op patches for profiler and error message for unsupported ops. (#1495)
* [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] merge development into main (#1) * [fx] activation checkpointing using Chen strategies. * [fx] add test for ckpt_solver_chen * [fx] add vanilla activation checkpoint search with test on resnet and densenet * [fx] add a namespace code for solver_chen. * [fx] fix the false interpretation of algorithm 3 in https://arxiv.org/abs/1604.06174. * [fx] fix lowercase naming conventions. * [fx] simplify test for ckpt. * [fx] add rules to linearize computation graphs for searching. (#2) * [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] modify the calculation of node_size in MetaInfoProp for activation checkpointing usages * [fx] merge development into main (#1) * [fx] activation checkpointing using Chen strategies. * [fx] add test for ckpt_solver_chen * [fx] add vanilla activation checkpoint search with test on resnet and densenet * [fx] add a namespace code for solver_chen. * [fx] fix the false interpretation of algorithm 3 in https://arxiv.org/abs/1604.06174. * [fx] fix lowercase naming conventions. * [fx] simplify test for ckpt. * [fx] fix test and algorithm bugs in activation checkpointing. * [fx] polish ckpt_test. * [fx] add rules to linearize computation graphs for searching. * [fx] remove chen_sqrt for sake of simplicity * [fx] remove chen_sqrt for sake of simplicity * [fx] remove chen_sqrt for sake of simplicity * [fx] remove chen_sqrt for sake of simplicity * [fx] fix inconsistencies. * [fx] fix MetaInfoProp. * [fx] fix MetaInfoProp. * [fx] consider MetaInfoProp for inplace operands. * [fx] consider MetaInfoProp for inplace operands. * [fx] consider MetaInfoProp for inplace operands. * [fx] consider MetaInfoProp for inplace operands. * [fx] consider MetaInfoProp for inplace operands. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] add profiler for fx nodes. * [fx] fix error in tests. * [fx] unfix bug. * [fx] unfix bug. * [fx] patch more modules and functions. * [fx] change name of utils.py to profiler.py * [fx] add profiler for rnn. * [fx] add profiler for rnn. * [fx] polish and add more patch for profiler. * [fx] polish and add more patch for profiler.
This commit is contained in:
236
colossalai/fx/profiler/profiler.py
Normal file
236
colossalai/fx/profiler/profiler.py
Normal file
@@ -0,0 +1,236 @@
|
||||
from functools import partial
|
||||
from operator import add, floordiv, getitem, mul, neg, setitem, sub, pos
|
||||
from typing import Callable, List, NamedTuple, Any, Dict, Tuple, Union
|
||||
import torch
|
||||
from torch.fx.node import Argument, Target, map_aggregate
|
||||
from torch.fx._compatibility import compatibility
|
||||
from colossalai.fx.tracer.meta_patch import meta_patched_function, meta_patched_module
|
||||
from . import meta_profiler_function, meta_profiler_module
|
||||
|
||||
__all__ = [
|
||||
'MetaProfile', 'profile_function', 'profile_module', 'profile_method', 'calculate_activation_size',
|
||||
'calculate_param_size'
|
||||
]
|
||||
|
||||
CALL_FUNCTION_MSG = \
|
||||
"""
|
||||
Colossal-AI hasn't supported profiling for {}, you might manually patch it with the following code.\n
|
||||
from colossalai.fx.profiler import meta_profiler_function
|
||||
|
||||
@meta_profiler_function.register(YOUR_FUNCTION)
|
||||
def profile_YOUR_FUNCTION(input: torch.Tensor, *args) -> Tuple[int, int]:
|
||||
flops = ...
|
||||
macs = ...
|
||||
return flops, macs
|
||||
"""
|
||||
CALL_METHOD_MSG = 'Please check if {} is an inplace method. If so, add target to INPLACE_METHOD={}. Otherwise, add target to NON_INPLACE_METHOD={}'
|
||||
CALL_MODULE_MSG = \
|
||||
"""
|
||||
Colossal-AI hasn't supported profiling for {}, you might manually patch it with the following code.\n
|
||||
from colossalai.fx.profiler import meta_profiler_module
|
||||
|
||||
@meta_profiler_module.register(YOUR_MODULE)
|
||||
def profile_YOUR_MODULE(self: torch.nn.Module, input: torch.Tensor) -> Tuple[int, int]:
|
||||
flops = ...
|
||||
macs = ...
|
||||
return flops, macs
|
||||
"""
|
||||
|
||||
# TODO fill out the inplace ops
|
||||
INPLACE_OPS = [
|
||||
add,
|
||||
sub,
|
||||
mul,
|
||||
floordiv,
|
||||
neg,
|
||||
pos,
|
||||
getitem,
|
||||
setitem,
|
||||
getattr,
|
||||
torch.Tensor.cpu,
|
||||
]
|
||||
|
||||
# TODO: list all call_methods that are inplace here
|
||||
INPLACE_METHOD = [
|
||||
'transpose',
|
||||
'permute',
|
||||
# TODO: reshape may return a copy of the data if the data is not contiguous
|
||||
'reshape',
|
||||
'dim',
|
||||
'flatten',
|
||||
]
|
||||
|
||||
# TODO: list all call_methods that are not inplace here
|
||||
NON_INPLACE_METHOD = [
|
||||
'expand',
|
||||
'mean',
|
||||
]
|
||||
|
||||
|
||||
@compatibility(is_backward_compatible=True)
|
||||
class MetaProfile(NamedTuple):
|
||||
|
||||
# MetaProfile is a structure containing pertinent information
|
||||
# about a node within a torch.fx GraphModule.
|
||||
|
||||
param: int
|
||||
activation: int
|
||||
flops: int
|
||||
macs: int
|
||||
|
||||
|
||||
def calculate_activation_size(activation: Union[torch.Tensor, Dict, List, Tuple, int]) -> int:
|
||||
"""Calculate activation size of a node.
|
||||
|
||||
Args:
|
||||
activation (Union[torch.Tensor, Dict, List, Tuple, int]): The activation of a `torch.nn.Module` or `torch.nn.functional`
|
||||
|
||||
Returns:
|
||||
int: The activation size
|
||||
"""
|
||||
activation_size = 0
|
||||
if isinstance(activation, torch.Tensor):
|
||||
activation_size += activation.numel() * torch.tensor([], dtype=activation.dtype).element_size()
|
||||
elif isinstance(activation, dict):
|
||||
value_list = [v for _, v in activation.items()]
|
||||
activation_size += calculate_activation_size(value_list)
|
||||
elif isinstance(activation, tuple) or isinstance(activation, list):
|
||||
for element in activation:
|
||||
activation_size += calculate_activation_size(element)
|
||||
return activation_size
|
||||
|
||||
|
||||
def calculate_param_size(mod: torch.nn.Module) -> int:
|
||||
"""Calculate param size of a node.
|
||||
|
||||
Args:
|
||||
mod (torch.nn.Module): The target `torch.nn.Module`
|
||||
|
||||
Returns:
|
||||
int: The param size
|
||||
"""
|
||||
param_size = 0
|
||||
for param in mod.parameters():
|
||||
param_size += param.numel() * torch.tensor([], dtype=param.dtype).element_size()
|
||||
return param_size
|
||||
|
||||
|
||||
def profile_function(target: 'Target') -> Callable:
|
||||
"""
|
||||
Wrap a `call_function` node or `torch.nn.functional` in order to
|
||||
record the memory cost and FLOPs of the execution.
|
||||
|
||||
Warnings:
|
||||
You may only use tensors with `device=meta` for this wrapped function.
|
||||
Only original `torch.nn.functional` are available.
|
||||
|
||||
Examples:
|
||||
>> input = torch.rand(100, 100, 100, 100, device='meta')
|
||||
>> func = torch.nn.functional.relu
|
||||
>> output, profile = profile_function(func)(input, inplace=False)
|
||||
>> print(f"Profiling function {func},")
|
||||
>> print(f"Param size: {profile.param / 1024**2:.3f} MB, Activation size: {profile.activation / 1024**2:.3f} MB, {profile.flops} FLOPs, {profile.macs} MACs")
|
||||
Profiling function <function relu at 0x7fcdd0258d30>,
|
||||
Param size: 0.000 MB, Activation size: 381.470 MB, 100000000 FLOPs, 0 MACs
|
||||
"""
|
||||
|
||||
def f(*args: Tuple[Argument, ...], **kwargs: Dict[str, Any]) -> Any:
|
||||
assert meta_profiler_function.has(target) or meta_profiler_function.has(
|
||||
target.__name__), CALL_FUNCTION_MSG.format(target)
|
||||
# ensure all arguments satisfy `device='meta'`
|
||||
args, kwargs = map_aggregate([args, kwargs], lambda a: a.to('meta') if isinstance(a, torch.Tensor) else a)
|
||||
|
||||
# call_function has no parameters
|
||||
param_size = 0
|
||||
activation_size = 0
|
||||
result = func(*args, **kwargs)
|
||||
if target not in INPLACE_OPS and not kwargs.get('inplace', False):
|
||||
activation_size += calculate_activation_size(result)
|
||||
if meta_profiler_function.has(target):
|
||||
profiler = meta_profiler_function.get(target)
|
||||
else:
|
||||
profiler = meta_profiler_function.get(target.__name__)
|
||||
flops, macs = profiler(*args, **kwargs)
|
||||
return result, MetaProfile(param_size, activation_size, flops, macs)
|
||||
|
||||
f.__name__ = target.__name__
|
||||
# fetch patched function
|
||||
if meta_patched_function.has(target):
|
||||
func = meta_patched_function.get(target)
|
||||
elif meta_patched_function.has(target.__name__):
|
||||
func = meta_patched_function.get(target.__name__)
|
||||
else:
|
||||
func = target
|
||||
return f
|
||||
|
||||
|
||||
def profile_method(target: 'Target') -> Callable:
|
||||
"""
|
||||
Wrap a `call_method` node
|
||||
record the memory cost and FLOPs of the execution.
|
||||
|
||||
Warnings:
|
||||
This is not fully implemented and you may follow the error message to debug.
|
||||
"""
|
||||
|
||||
def f(*args: Tuple[Argument, ...], **kwargs: Dict[str, Any]) -> Any:
|
||||
# args[0] is the `self` object for this method call
|
||||
self_obj, *args_tail = args
|
||||
|
||||
# execute the method and return the result
|
||||
assert isinstance(target, str), f'{target} instance is not str.'
|
||||
|
||||
# ensure all arguments satisfy `device='meta'`
|
||||
map_aggregate([args, kwargs], lambda a: a.to('meta') if isinstance(a, torch.Tensor) else a)
|
||||
result = getattr(self_obj, target)(*args_tail, **kwargs)
|
||||
assert target in INPLACE_METHOD + NON_INPLACE_METHOD, CALL_METHOD_MSG.format(
|
||||
target, INPLACE_METHOD, NON_INPLACE_METHOD)
|
||||
# call_method has no parameters and are MOSTLY(?) inplace, and has no FLOPs or MACs.
|
||||
param_size = 0
|
||||
activation_size = 0 if target in INPLACE_METHOD else calculate_activation_size(result)
|
||||
flops = 0
|
||||
macs = 0
|
||||
return result, MetaProfile(param_size, activation_size, flops, macs)
|
||||
|
||||
return f
|
||||
|
||||
|
||||
def profile_module(module: torch.nn.Module) -> Callable:
|
||||
"""
|
||||
Wrap a `call_module` node or `torch.nn` in order to
|
||||
record the memory cost and FLOPs of the execution.
|
||||
|
||||
Warnings:
|
||||
You may only use tensors with `device=meta` for this wrapped function.
|
||||
Only original `torch.nn` are available.
|
||||
|
||||
Example:
|
||||
>> input = torch.rand(4, 3, 224, 224, device='meta')
|
||||
>> mod = torch.nn.Conv2d(3, 128, 3)
|
||||
>> output, profile = profile_module(mod)(input)
|
||||
>> print(f"Profiling function {mod},")
|
||||
>> print(f"Param size: {profile.param / 1024**2:.3f} MB, Activation size: {profile.activation / 1024**2:.3f} MB, {profile.flops} FLOPs, {profile.macs} MACs")
|
||||
Profiling function Conv2d(3, 128, kernel_size=(3, 3), stride=(1, 1)),
|
||||
Param size: 0.014 MB, Activation size: 96.258 MB, 1387837440 FLOPs, 681302016 MACs
|
||||
"""
|
||||
|
||||
def f(*args: Tuple[Argument, ...], **kwargs: Dict[str, Any]) -> Any:
|
||||
assert meta_profiler_module.has(type(module)), CALL_MODULE_MSG.format(type(module))
|
||||
# ensure all arguments satisfy `device='meta'`
|
||||
map_aggregate([args, kwargs], lambda a: a.to('meta') if isinstance(a, torch.Tensor) else a)
|
||||
param_size = calculate_param_size(module)
|
||||
activation_size = 0
|
||||
result = func(*args, **kwargs)
|
||||
if not getattr(module, 'inplace', False):
|
||||
activation_size += calculate_activation_size(result)
|
||||
profiler = meta_profiler_module.get(type(module))
|
||||
flops, macs = profiler(module, *args, **kwargs)
|
||||
return result, MetaProfile(param_size, activation_size, flops, macs)
|
||||
|
||||
f.__name__ = module.__class__.__name__
|
||||
# fetch patched module
|
||||
if meta_patched_module.has(type(module)):
|
||||
func = partial(meta_patched_module.get(type(module)), module)
|
||||
else:
|
||||
func = module.forward
|
||||
return f
|
Reference in New Issue
Block a user