8xH100 训练 Qwen3-4B#

环境准备#

拉取 inferactinc/public:vime-latest 镜像后,用如下方式初始化镜像环境:

cd /root/
git clone https://github.com/vllm-project/vime.git
cd vime/
pip install -e . --no-deps

下载模型与数据:

# hf checkpoint
hf download Qwen/Qwen3-4B --local-dir /root/Qwen3-4B

# train data
hf download --repo-type dataset zhuzilin/dapo-math-17k \
  --local-dir /root/dapo-math-17k

# eval data
hf download --repo-type dataset zhuzilin/aime-2024 \
  --local-dir /root/aime-2024

将 huggingface checkpoint 转换成 megatron 可以加载的 huggingface checkpoint:

# mcore checkpoint
cd /root/vime
source scripts/models/qwen3-4B.sh
PYTHONPATH=/root/Megatron-LM python tools/convert_hf_to_torch_dist.py \
    ${MODEL_ARGS[@]} \
    --hf-checkpoint /root/Qwen3-4B \
    --save /root/Qwen3-4B_torch_dist

执行训练#

执行训练:

cd /root/vime
bash scripts/run-qwen3-4B.sh

参数简介#

这里我们简单介绍一下脚本 run-qwen3-4B.sh 中的各个组成部分:

MODEL_ARGS#

SCRIPT_DIR="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")" &>/dev/null && pwd)"
source "${SCRIPT_DIR}/models/qwen3-4B.sh"

scripts/models/qwen3-4B.sh 读取模型的 config。这些 config 都是 megatron 的参数。在使用 megatron 进行训练的时候,megatron 无法从 ckpt 中读取模型 config,需要我们自行配置。我们在 scripts/models 中提供了一些样例。

⚠️ 注意检查模型文件中的 --rotary-base 等配置是否对应你当前训练模型的配置,因为同一个模型结构的不同模型可能有不同的取值。在这种情况下,你可以在导入模型参数后在脚本里进行覆盖,例如:

source "${SCRIPT_DIR}/models/qwen3-4B.sh"

MODEL_ARGS += ( --rotary-base 10000 )

CKPT_ARGS#

CKPT_ARGS=(
   # vLLM 需要的 hf ckpt,我们也会从这里读 tokenizer
   --hf-checkpoint /root/Qwen3-4B
   # reference model 的 ckp
   --ref-load /root/Qwen3-4B_torch_dist
   # actor 的 load dir,如果是空的,会从 `ref_load` 里面读
   --load /root/Qwen3-4B_vime/
   --save /root/Qwen3-4B_vime/
   --save-interval 20
)

ROLLOUT_ARGS#

ROLLOUT_ARGS=(
   # prompt 数据集,每行是个 json
   --prompt-data /root/dapo-math-17k/dapo-math-17k.jsonl
   --input-key prompt
   --label-key label
   # 如果 prompt 的 `input_key` 中是 openai message,
   # 会进行 tokenizer.apply_chat_template(...)
   --apply-chat-template
   # 是否 shuffle 数据
   --rollout-shuffle

   # reward model 类型,
   # vime 提供了很多类型以及用于自定义的 --custom-rm-path
   --rm-type deepscaler

   # 一共要训练多少 rollout
   --num-rollout 3000
   # 一个 rollout 有多少 prompt
   --rollout-batch-size 32
   # 每个 prompt 采多少回复
   # 一个 rollout 会有 rollout_batch_size * n_samples_per_prompt 条
   --n-samples-per-prompt 8
   # rollout sampling param
   --rollout-max-response-len 8192
   --rollout-temperature 0.8

   # 一次 rollout 对应几个训练步
   --num-steps-per-rollout 1
   # 是否在训练时 balance data,可能对速度有好处
   --balance-data
)

EVAL_ARGS#

eval 的时候基本上是会继承所有 rollout 的参数,但是我们提供了一些可以 rollout 配置覆盖的参数,从而实现训练和 eval 用不同的采样策略。

EVAL_ARGS=(
   --eval-interval 5
   --eval-prompt-data /root/aime-2024/aime-2024.jsonl
   --n-samples-per-eval-prompt 16
   --eval-max-response-len 16384
   --eval-top-p 0.7
)

PERF_ARGS#

一堆 megatron 的并行参数,只有 --use-dynamic-batch-size--max-tokens-per-gpu 是 vime 添加的。

max_tokens_per_gpu 是指每张卡最多跑多少 token,在开启 use_dynamic_batch_size 之后,会尽可能将一个 batch 内部长短不一的数据拼到 max_tokens_per_gpu,从而组成动态的 micro batch size,如果有一条数据长度超过了 max_tokens_per_gpu,则自成一条,不会对数据进行截断。在开启 context parallel (CP) 时,会让 CP 张卡去上的数据去共享总长为 CP * max_tokens_per_gpu 的 token。

在开启 dynamic_batch_size,会忽略传统的 micro_batch_size

⚠️ vime 总是会通过 data packing 的方法训练模型,并且严格保证 per sample loss 或 per token loss,也就是开启 dynamic batch size 不会对 loss 计算有影响,推荐开启。

PERF_ARGS=(
   --tensor-model-parallel-size 2
   --sequence-parallel
   --pipeline-model-parallel-size 1
   --context-parallel-size 1
   --expert-model-parallel-size 1
   --expert-tensor-parallel-size 1

   --recompute-granularity full
   --recompute-method uniform
   --recompute-num-layers 1

   # --micro-batch-size 1
   --use-dynamic-batch-size
   --max-tokens-per-gpu 9216
)

GRPO_ARGS#

目前 vime 这是一些 grpo 相关的参数:

GRPO_ARGS=(
   --advantage-estimator grpo
   --use-kl-loss
   --kl-loss-coef 0.00
   --kl-loss-type low_var_kl
   --entropy-coef 0.00
   --eps-clip 0.2
   --eps-clip-high 0.28
)

OPTIMIZER_ARGS#

OPTIMIZER_ARGS=(
   --optimizer adam
   --lr 1e-6
   --lr-decay-style constant
   --weight-decay 0.1
   --adam-beta1 0.9
   --adam-beta2 0.98
)

VLLM_ARGS#

vLLM 推理所需的参数。vime 默认使用 vLLM 作为 rollout 后端(rollout.py 启动 VLLMEngine,默认 rollout 函数为 vime.rollout.vllm_rollout.generate_rollout),无需额外指定 backend。--rollout-num-gpus-per-engine 对应每个 vLLM engine 的 tensor_parallel_size;除此之外的 vLLM 参数均通过添加 --vllm- 前缀传给 vime(例如 --vllm-max-model-len)。

VLLM_ARGS=(
   --rollout-num-gpus-per-engine 2
   --vllm-gpu-memory-utilization 0.7
)

rollout 并发较高时,还可以通过 --vllm- 前缀调节 vLLM scheduler,例如 --vllm-max-num-seqs--vllm-max-num-batched-tokens;调试或规避 CUDA graph 相关限制时可加 --vllm-enforce-eager

⚠️ vime 会用 vLLM router 来调度多个 vLLM server。训推一体(--colocate)时,训练与推理权重经 CUDA IPC 同步;训推分离时,训练侧经 NCCL 与 vLLM engine 同步权重。

dynamic sampling#

vime 支持了更复杂的 sampling 方案,例如 DAPO 中的 dynamic sampling。如果要开启 dynamic sampling,需要配置:

   --over-sampling-batch-size ${OVER_SAMPLING_BS} \
   --dynamic-sampling-filter-path \
     vime.rollout.filter_hub.dynamic_sampling_filters.check_reward_nonzero_std \

这里 over_sampling_batch_size 需要大于 ``rollout_batch_size`,例如配置为:

   --rollout-batch-size 32 \
   --n-samples-per-prompt 8 \
   --over-sampling-batch-size 64 \

那么 sampling 会直接采样 64 条 prompt,每条 prompt 采样 8 次。因为 vime 内部进行的是异步采样,所以我们会先后获得每个 prompt 的 8 条回复。在收到回复时,会用 dynamic_sampling_filter_path 对应的函数进行筛选,如果通过,则留下这 8 条数据,否则则丢掉。例子中的函数是判断回答是否全对或全错:

def check_reward_nonzero_std(args, samples: list[Sample], **kwargs):
    rewards = [sample.reward for sample in samples]
    return torch.tensor(rewards, dtype=torch.float).std() > 0.0

当我们收到了 32 * 8 条数据时,我们会立刻停止采样,而不会等剩余的数据采样完成。如果删除的数据超过了 32 条 prompt(剩余的小于 32 条 prompt),那么我们会再采样 64 条 prompt。

partial rollout#

在进行 dynamic sampling 的过程中,会提前终止(abort)大量请求,我们可以通过配置 --partial-rollout 参数来将生成到一半的请求保存至 data buffer,在下一个 rollout 中取出来继续进行数据生成,从而进一步优化性能。

可以通过配置 --buffer-filter-path 来自定义如何从 buffer 中取出数据,默认的函数为:

def pop_first(args, rollout_id, buffer: list[list[Sample]], num_samples: int) -> list[list[Sample]]:
    num_to_pop = min(len(buffer), num_samples)
    samples = buffer[:num_to_pop]
    del buffer[:num_to_pop]
    return samples

即每次取出前 num_samples 个 prompt 对应的 num_samples * n_samples_per_prompt 条数据。

⚠️ 每条 partial rollout sample 的 sample.metadata 中存储了第一次进行生成的 rollout id,可以用于数据过滤。

训推分离#

在原始的脚本中,资源配置如下:

ray job submit ... \
   -- python3 train.py \
   --actor-num-nodes 1 \
   --actor-num-gpus-per-node 8 \
   --colocate \
   ...

即开启训推一体(colocate),并且训练部分会使用 1 机 8 卡,推理会和训练共同使用这 8 张卡张卡。

如果想使用训推分离的功能,需要去掉 --colocate 并配置上 --rollout-num-gpus,例如:

ray job submit ... \
   -- python3 train.py \
   --actor-num-nodes 1 \
   --actor-num-gpus-per-node 2 \
   --rollout-num-gpus 6 \
   ...

此时,就会分配 2 张卡给训练,6 张卡给推理。--rollout-num-gpus--actor-num-gpus-per-node 一样,是传给 train.pyRay 资源参数:框架据此创建 placement group,并把前若干 bundle 分给训练 actor、后续 bundle 分给 rollout engine(见 vime/ray/placement_group.py)。共卡模式(--colocate)下该参数会被忽略,并自动设为 actor_num_gpus_per_node * actor_num_nodes。请勿把 --rollout-num-gpus 写在 VLLM_ARGS 中。

训推分离时,VLLM_ARGS 仅需配置推理后端相关参数,例如:

VLLM_ARGS=(
   --rollout-num-gpus-per-engine 2
   --vllm-gpu-memory-utilization 0.9
   --vllm-max-num-seqs 256
   --vllm-max-num-batched-tokens 8192
)

如需调试或规避 CUDA graph 相关限制,可额外加上 --vllm-enforce-eager

⚠️ 在训推一体的训练时,megatron 始终会占据一些显存,需要通过 --vllm-gpu-memory-utilization 来降低 vLLM 占据的显存比例,并配合 --train-memory-margin-bytes 为训练侧预留空间。

异步训练#

当进行训推分离时,你会发现训练和推理的 GPU 总是相互等待着,为了避免这种资源空闲,我们可以开启异步训练。开启的方式即为将启动脚本中的 train.py 改变为 train_async.py。这样 vime 就会在进行当前 rollout 的训练时进行下一个 rollout 的数据生成了。

train.pytrain_async.py 的差别只在于 train loop 的同步逻辑,我们通过 ray 的异步(.remote, ray.get)实现了这点。