"""
Author: PointNeXt
for profiling the parameters, flops, speed of a model

usage example: 

1. profile pointnext-s on scanobjectnn using 128 * 1024 points as input
CUDA_VISIBLE_DEVICES=0 python examples/profile.py --cfg cfgs/scanobjectnn/pointnext-s.yaml batch_size=128 num_points=1024 timing=True

2. profile all models for scanobjectnn classification using 128 * 1024 points as input
CUDA_VISIBLE_DEVICES=0 python examples/profile.py --cfg cfgs/scanobjectnn batch_size=128 num_points=1024 timing=True
"""
import os, sys, argparse, time, warnings
import torch

sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '../')))
from openpoints_student.utils import EasyConfig, cal_model_parm_nums
from openpoints_student.models_student import build_model_from_cfg
from deepspeed.profiling.flops_profiler import get_model_profile


def profile_model(model, cfg):
    model.eval()
    # for classification, num_points is 128 * 1024
    # for s3dis, num_points 16 * 15000 
    B, N, C = 1, cfg.num_points, 3
    if cfg.variable:
        points = torch.randn(B * N, 3).cuda().contiguous()
        features = torch.randn(B * N, C).cuda().contiguous()
        offset = []
        count = 0
        for i in range(B):
            count += N
            offset.append(count)
        offsets = torch.IntTensor(offset).cuda()
        args = [points, features, offsets]
    else:
        points = torch.randn(B, N, 3).cuda().contiguous()
        if cfg.model.get('feature_last_dim', False):
            features = torch.randn(B, N, C).cuda().contiguous()
        else:
            features = torch.randn(B, C, N).cuda().contiguous()
            cls = torch.zeros(B, 16).long().cuda()
        if cfg.dataset.common.NAME == 'ShapeNetPartNormal':
            args = [points, features, cls]
            # args = {'pos': points, 'x': features, 'cls': cls}
        else:
            args = [{'pos': points, 'x': features}]
            # args = points 
    print(f'test input size: ({points.shape, features.shape})')
    
    if cfg.get('flops', False):
        from deepspeed.profiling.flops_profiler import get_model_profile
        detailed = False
        flops, macs, params = get_model_profile(
            model=model,
            args=args,
            print_profile=detailed,  # prints the model graph with the measured profile attached to each module
            detailed=detailed,  # print the detailed profile
            warm_up=10,  # the number of warm-ups before measuring the time of each module
            as_string=False,  # print raw numbers (e.g. 1000) or as human-readable strings (e.g. 1k)
            output_file=None,  # path to the output file. If None, the profiler prints to stdout.
            ignore_modules=None)  # the list of modules to ignore in the profiling
        print(f'Batches\tnpoints\tParams.(M)\tGFLOPs')
        print(f'{cfg.batch_size}\t{N}\t{params / 1e6: .3f}\t{flops / (float(B) * 1e9): .2f}')
    else:
        warnings.warn('set flops=True to calculate flops')
        
    if cfg.get('timing', False):
        B = cfg.batch_size
        if cfg.variable:
            points = torch.randn(B * N, 3).cuda().contiguous()
            features = torch.randn(B * N, C).cuda().contiguous()
            offset = []
            count = 0
            for i in range(B):
                count += N
                offset.append(count)
            offsets = torch.IntTensor(offset).cuda().contiguous()
            args = [points, features, offsets]
        else:
            points = torch.randn(B, N, 3).cuda().contiguous()
            features = torch.randn(B, C, N).cuda().contiguous()
            cls = torch.zeros(B, 16).long().cuda().contiguous()
            if cfg.dataset.common.NAME == 'ShapeNetPartNormal':
                args = [points, features, cls]
            else:
                # points = torch.cat((points, features.transpose(1, 2)), dim=2)
                args = [features.transpose(1, 2).contiguous()]
        model = build_model_from_cfg(cfg.model).cuda()
        n_runs = cfg.get('nruns', 200)
        with torch.no_grad():
            for _ in range(10):  # warm up.
                model(*args)
            start_time = time.time()
            for _ in range(n_runs):
                model(*args)
                torch.cuda.synchronize()
            time_taken = time.time() - start_time
        n_batches = n_runs * B
        print(f'Throughput (ins./s): {float(n_batches) / float(time_taken)}')

    else:
        warnings.warn('set timing=True to calculate inference time')


@torch.no_grad()
def profile_grouper(cfg):
    from openpoints.models.layers.group import ball_query
    B, N = cfg.batch_size, cfg.num_points

    points = torch.randn(B, N, 3).cuda() * 10
    radius = cfg.get('radius', 0.1)
    nsample = cfg.get('nsample', 32)

    n_runs = cfg.get('nruns', 200)
    for _ in range(10):  # warm up.
        ball_query(radius, nsample, points, points)
        torch.cuda.synchronize()
    start_time = time.time()
    for _ in range(n_runs):
        ball_query(radius, nsample, points, points)
        torch.cuda.synchronize()
    time_taken = time.time() - start_time
    n_batches = n_runs * B
    print(f'Throughput (ins./s): {float(n_batches) / float(time_taken)}')


if __name__ == "__main__":
    parser = argparse.ArgumentParser('S3DIS scene segmentation training')
    parser.add_argument('--cfg', type=str, required=True, help='config file')
    parser.add_argument('--detailed', action='store_true')
    parser.add_argument('--function', type=str, default='profile_model',
                        help='which function to use, which to profile. profile_model, profile_groper')
    args, opts = parser.parse_known_args()

    file_list = []
    for root, dirs, files in os.walk(args.cfg):
        for file in files:
            if file.endswith('yaml') and not file.startswith('default'):
                if 'para' in file or 'opt' in file or 'aug' in file:
                    warnings.warn(f'skip file {file}')
                else:
                    file = os.path.join(root, file)
                    file_list.append(file)

    if len(file_list) == 0 and args.cfg.endswith('yaml'):
        file_list.append(args.cfg)

    if 'model' in args.function:
        for file in file_list:
            cstr = '-' * 12
            print(f'{cstr}\n===>loading from {file}')
            cfg = EasyConfig()
            cfg.load(file, recursive=True)
            cfg.update(opts)

            cfg.variable = 'variable' in file
            if cfg.model.get('encoder_args', None) is not None:
                cfg.model.encoder_args.in_channels = 3
            cfg.model.in_channels = 3
            model = build_model_from_cfg(cfg.model).cuda()
            model.eval()
            model_size = cal_model_parm_nums(model)
            print('Number of params: %.4f M' % (model_size / 1e6))
            profile_model(model, cfg)

    else:
        cfg = EasyConfig()
        print(file_list)
        cfg.load(file_list[0], recursive=True)
        cfg.update(opts)
        eval(cfg.function)(cfg)
