search_space_autogen.py 4.53 KB
Newer Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
# SPDX-FileCopyrightText: Copyright (c) 2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
# SPDX-License-Identifier: Apache-2.0

import argparse
import logging
import math
import os

import yaml

from benchmarks.profiler.utils.config_modifiers import CONFIG_MODIFIERS
from benchmarks.profiler.utils.model_info import get_model_info
from deploy.utils.gpu_inventory import get_gpu_summary

logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)
console_handler = logging.StreamHandler()
console_handler.setLevel(logging.INFO)
formatter = logging.Formatter(
    "%(asctime)s - %(name)s - %(levelname)s - %(message)s", "%Y-%m-%d %H:%M:%S"
)
console_handler.setFormatter(formatter)
logger.addHandler(console_handler)

MODEL_GPU_MEM_FRAC_MAX = 0.9
MOE_MODEL_MAX_NUM_GPUS = 32


def auto_generate_search_space(args: argparse.Namespace) -> None:
    config_modifier = CONFIG_MODIFIERS[
        args.backend
    ]  # args.backend is already validated in argparse

    # first check if config file exists
35
    if args.model:
36
37
38
39
40
41
42
43
44
45
46
        if not args.config:
            # modify config file from default config file
            logger.info("DGD config file not provided, using default config file")
            config = config_modifier.load_default_config()

        else:
            with open(args.config, "r") as f:
                config = yaml.safe_load(f)

        logger.info(f"Updating model in DGD config file to {args.model}")
        config = config_modifier.update_model(config, args.model)
47
48
49
50
        if args.dgd_image:
            logger.info(f"Updating DGD image to {args.dgd_image}")
            config = config_modifier.update_image(config, args.dgd_image)

51
52
53
54
55
56
57
58
        config_fn = f"{args.output_dir}/disagg_config.yaml"
        logger.info(f"Saving generated disagg DGD config for profiling to {config_fn}")
        os.makedirs(args.output_dir, exist_ok=True)
        with open(config_fn, "w") as f:
            yaml.dump(config, f)
        args.config = config_fn

    # now determine the search space
59
60
61
    model_info = None
    if args.model:
        logger.info(f"Getting model info for {args.model}...")
62
63
        model_info = get_model_info(args.model)

64
65
66
67
68
        num_experts_str = (
            f", num_experts={model_info['num_experts']}"
            if model_info.get("num_experts")
            else ""
        )
69
        logger.info(
70
            f"Model {args.model} has size {model_info['model_size']}, is_moe={model_info['is_moe']}, and max_context_length={model_info['max_context_length']}{num_experts_str}"
71
        )
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
        args.is_moe_model = model_info["is_moe"]  # type: ignore[assignment]
        args.max_context_length = model_info["max_context_length"]  # type: ignore[assignment]

    if (
        args.min_num_gpus_per_engine == 0
        or args.max_num_gpus_per_engine == 0
        or args.num_gpus_per_node == 0
    ):
        if not args.model:
            # TODO: get model info provided DGD config
            error_msg = "No model provided, cannot auto-generate GPU search space. Please provide `--model` or GPU info"
            logger.error(error_msg)
            raise RuntimeError(error_msg)

        logger.info("Getting GPU info from k8s cluster...")
        gpu_info = get_gpu_summary()
88
89
90
91
        logger.info(
            f"Cluster has {gpu_info['gpus_per_node']}x{gpu_info['model']} GPUs per node with {gpu_info['vram']} VRAM"
        )

92
93
94
        # model_info should be set by now (checked above), but mypy needs explicit verification
        assert model_info is not None, "model_info must be set when model is provided"

95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
        min_gpu = math.ceil(
            model_info["model_size"] / MODEL_GPU_MEM_FRAC_MAX / gpu_info["vram"]  # type: ignore[operator]
        )
        max_gpu = (
            gpu_info["gpus_per_node"]  # type: ignore[misc]
            if not model_info["is_moe"]
            else MOE_MODEL_MAX_NUM_GPUS
        )
        if min_gpu > max_gpu:
            error_msg = f"No valid GPU configuration found for model {args.model} on the cluster with {gpu_info['gpus_per_node']}x{gpu_info['model']} GPUs per node"
            logger.error(error_msg)
            raise RuntimeError(error_msg)

        logger.info(
            f"Auto-generated search space for model {args.model} on the cluster with {gpu_info['gpus_per_node']}x{gpu_info['model']} GPUs per node: {min_gpu} to {max_gpu}"
        )
        args.min_num_gpus_per_engine = min_gpu
        args.max_num_gpus_per_engine = max_gpu
        args.num_gpus_per_node = gpu_info["gpus_per_node"]  # type: ignore[assignment]
114
        args.num_experts = model_info.get("num_experts")  # type: ignore[assignment]
115
116

    return