Skip to content

Commit

Permalink
add data flow doc
Browse files Browse the repository at this point in the history
  • Loading branch information
sallyjunjun committed Aug 21, 2024
1 parent 81fb735 commit 4e71314
Show file tree
Hide file tree
Showing 2 changed files with 285 additions and 10 deletions.
261 changes: 261 additions & 0 deletions doc/data_flow.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,261 @@
## 数据格式

### Dataloader加载数据

数据的准备过程参考[使用教程](./usage.md)

这里详细介绍一下在Dataloader中对数据进行pack的过程。

`config.py`中设置`use_packed_dataset``True`时,通过`build_pack`函数构建pack data,构建逻辑举例如下:

假设`micro_bsz`为2,`SEQ_LEN`为8,输入数据格式如下:
```bash
[2323, 442, 252, 341]
[233, 3442, 322, 31, 2514, 49731, 51]
[4326, 427, 465, 22, 314, 9725, 346, 1343]
[24, 2562, 5, 25, 356]
```

`packed_length`的值为`micro_bsz * SEQ_LEN`,即16,对于上述输入数据,将每一条输入pack为长度为16的数据,其中,单个子句拼接超出`packed_length`的部分,截断处理,后半部分作为下一条输入的开始。全部文本pack完成之后,对于最后不足`packed_length`的部分,用`0`填充。pack之后的数据格式如下:
```bash
[2323, 442, 252, 341, 233, 3442, 322, 31, 2514, 49731, 51, 4326, 427, 465, 22, 314]
[9725, 346, 1343, 24, 2562, 5, 25, 356, 0, 0, 0, 0, 0, 0, 0, 0]
```

`label`的值取data每条输入的第二个值到最后一个值,并在最后一位填充`-100`。上述数据pack之后,对应的label值如下:
```bash
[442, 252, 341, -100, 3442, 322, 31, 2514, 49731, 51, -100, 427, 465, 22, 314, 9725]
[346, 1343, -100, 2562, 5, 25, 356, -100, -100, -100, -100, -100, -100, -100, -100, -100]
```

`config.py`中设置`use_packed_dataset``False`时,通过`build_unpack`函数构建pack data,构建逻辑举例如下:

假设`micro_bsz`为2,`SEQ_LEN`为8,输入数据格式如下:
```bash
[2323, 442, 252, 341]
[233, 3442, 322, 31, 2514, 49731, 51]
[4326, 427, 465, 22, 314, 9725, 346, 1343]
[24, 2562, 5, 25, 356, 3145, 246, 25, 1451, 67, 73, 541, 265]
[4524, 2465, 562, 67, 26, 265, 21, 256, 145, 1345]
[34, 14]
```

这里pack的过程遵循三个条件:

1. pack子句的个数不能超过micro_bsz,即便pack之后的总长度小于`micro_bsz * SEQ_LEN`,也不能将下一个子句的内容进行pack,长度不够的部分补`0`

2. 单条子句的长度不能超过SEQ_LEN,超过部分直接截断丢弃,之后再与下一个子句进行pack。

3. 如果子句pack之后的长度超过了`micro_bsz * SEQ_LEN`,超过部分截断丢弃。

按照上述规则,pack之后的数据格式如下:
```bash
[2323, 442, 252, 341, 233, 3442, 322, 31, 2514, 49731, 51, 0, 0, 0, 0, 0]
[4326, 427, 465, 22, 314, 9725, 346, 1343, 24, 2562, 5, 25, 356, 3145, 246, 25]
[4524, 2465, 562, 67, 26, 265, 21, 256, 34, 14, 0, 0, 0, 0, 0, 0]
```

`label`的值如下:
```bash
[442, 252, 341, 233, 3442, 322, 31, 2514, 49731, 51, -100, -100, -100, -100, -100, -100]
[427, 465, 22, 314, 9725, 346, 1343, 24, 2562, 5, 25, 356, 3145, 246, 25, -100]
[2465, 562, 67, 26, 265, 21, 256, 34, 14, -100, -100, -100, -100, -100, -100, -100]
```

注:`use_packed_dataset`不设置时,默认为`True`。一般使用`use_packed_dataset``True`的模式训练,以提升训练效率和精度。

### 获取Dataloader数据
#### 从Dataloader中取出数据
```bash
batch_data, actual_batch_size = engine.load_batch(data_iter)
```
这里`batch_data`的类型为`list`,其中包含两个元素,第一个元素为`dict`类型的数据`data`,第二个元素为`torch.Tensor`类型的标签`label`

其中,第一个元素`data`包含`input_ids``cu_seqlens``indexes`三个字段,其类型及形状分别为:
```bash
batch_data[0]['input_ids'] -> torch.Size([micro_num, micro_bsz * SEQ_LEN]),保存输入语句经过tokenize之后的id值
batch_data[0]['cu_seqlens'] -> list类型,大小为micro_num, 其中每个元素类型为torch.Tensor,保存pack到micro_bsz * SEQ_LEN长度的每个拼接字句的索引
batch_data[0]['indexes'] -> torch.Size([micro_num, micro_bsz * SEQ_LEN]),保存每个input_ids的索引值,从0开始递增
```

第二个元素`label`的形状为:
```bash
batch_data[1] -> torch.Size([micro_num, micro_bsz * SEQ_LEN])
```
`micro_num``config.py`配置文件中设置,为梯度累计的大小,即经过`micro_num``forward` + `backward`之后,进行梯度更新。
`micro_bsz * SEQ_LEN``pack data`的长度,即将多条输入拼接为`micro_bsz * SEQ_LEN`长度的单条输入,以提高训练效率。

举例:
假设`micro_num`为2,`micro_bsz`为2,`SEQ_LEN`为8
```bash
batch_data[0]['input_ids']:
tensor([[ 410, 5141, 789, 233, 300, 335, 353, 34, 3414, 545, 76, 4632, 3677, 63, 6363, 872],
[ 328, 300, 313, 435, 530, 288, 54941, 1314, 245, 143, 4557, 248, 211, 2626, 351, 4144]])
```

假设其中第一个batch由长度分别为4,3,2,7的子句拼接而成,第二个batch由长度分别为5,6,5的子句拼接而成,则:
```bash
batch_data[0]['cu_seqlens']:
tensor([[ 0, 4, 7, 9, 16],
[ 0, 5, 11, 16]])
```
其中,每相邻两个数字的差值,为当前子句的长度。

```bash
batch_data[0]['indexes']:
tensor([[ 0, 1, 2, 3, 0, 1, 2, 0, 1, 0, 1, 2, 3, 4, 5, 6],
[ 0, 1, 2, 3, 4, 0, 1, 2, 3, 4, 5, 0, 1, 2, 3, 4]])
```
其中,每一个数字表示了token在当前子句中的位置。

```bash
batch_data[1]:
tensor([[ 5141, 789, 233, 300, 335, 353, 34, 3414, 545, 76, 4632, 3677, 63, 6363, 872, -100],
[ 300, 313, 435, 530, 288, 54941, 1314, 245, 143, 4557, 248, 211, 2626, 351, 4144, -100]])
```

#### 处理数据
```bash
_data, _label = self._load_accum_batch(data, label)
```
首先,通过`_load_micro_batch`函数,将`data``label`中数据的第一个维度`micro_num`转化为1,并通过更新`offset`的值,依次获取每个微批次的数据。

其次,通过注册`data_process_func`对数据做进一步处理。

`config.py`中设置`use_packed_dataset``True`时,`data_process_func`中的流程如下:

通过`packed_data_normalizer`函数,对`data['indexes']``data['cu_seqlens']`做降维处理,去掉size为1的第一维,并通过`data['cu_seqlens']`中的值,计算出单个字句的最大长度,记录在`data['max_seqlen']`中。

按照上述举例,假设加载第一个批次的数据,经过`_load_accum_batch`处理后的`data``label`如下:
```bash
data['input_ids']:
tensor([[ 410, 5141, 789, 233, 300, 335, 353, 34, 3414, 545, 76, 4632, 3677, 63, 6363, 872]])
data['cu_seqlens']:
tensor([ 0, 4, 7, 9, 16])
data['indexes']:
tensor([ 0, 1, 2, 3, 0, 1, 2, 0, 1, 0, 1, 2, 3, 4, 5, 6])
data['max_seqlen']:
7

label:
tensor([[ 5141, 789, 233, 300, 335, 353, 34, 3414, 545, 76, 4632, 3677, 63, 6363, 872, -100]])
```

如果`tp`并行模式为`isp`,且tp size(即sequence parallel size)大于1,则会在`data_process_func`中注册`split_data_sequence_parallel`函数,对数据的`sequence`维度进行切分。

假设tp size为2,则对上述数据`data['input_ids']``data['indexes']``label`切分之后的结果如下:

tp rank0 中的数据:
```bash
data['input_ids']:
tensor([[ 410, 5141, 789, 233, 300, 335, 353, 34]])
data['indexes']:
tensor([ 0, 1, 2, 3, 0, 1, 2, 0])
label:
tensor([[ 5141, 789, 233, 300, 335, 353, 34, 3414]])
```

tp rank1 中的数据:
```bash
data['input_ids']:
tensor([[ 3414, 545, 76, 4632, 3677, 63, 6363, 872]])
data['indexes']:
tensor([ 1, 0, 1, 2, 3, 4, 5, 6])
label:
tensor([[ 545, 76, 4632, 3677, 63, 6363, 872, -100]])
```

`config.py`中设置`use_packed_dataset``False`时,`data_process_func`中的流程如下:

通过`unpack_data`函数对数据做unpack处理,将`data["input_ids"]``label`的数据恢复到unpack的格式,并从data中去除掉"cu_seqlens"和"indexes"字段。unpack之后`data["input_ids"]``label`的形状为`torch.Size([micro_bsz, SEQ_LEN])`

按照上述数据举例:

假设`micro_bsz`为2,`SEQ_LEN`为8,输入数据格式如下:
```bash
[2323, 442, 252, 341]
[233, 3442, 322, 31, 2514, 49731, 51]
[4326, 427, 465, 22, 314, 9725, 346, 1343]
[24, 2562, 5, 25, 356, 3145, 246, 25, 1451, 67, 73, 541, 265]
[4524, 2465, 562, 67, 26, 265, 21, 256, 145, 1345]
[34, 14]
```

pack之后的数据格式如下:
```bash
[2323, 442, 252, 341, 233, 3442, 322, 31, 2514, 49731, 51, 0, 0, 0, 0, 0]
[4326, 427, 465, 22, 314, 9725, 346, 1343, 24, 2562, 5, 25, 356, 3145, 246, 25]
[4524, 2465, 562, 67, 26, 265, 21, 256, 34, 14, 0, 0, 0, 0, 0, 0]
```

`label`的值如下:
```bash
[442, 252, 341, 233, 3442, 322, 31, 2514, 49731, 51, -100, -100, -100, -100, -100, -100]
[427, 465, 22, 314, 9725, 346, 1343, 24, 2562, 5, 25, 356, 3145, 246, 25, -100]
[2465, 562, 67, 26, 265, 21, 256, 34, 14, -100, -100, -100, -100, -100, -100, -100]
```

经过`unpack_data`处理之后,`data["input_ids"]``label`分别如下:
```bash
data["input_ids"]:
tensor([[2323, 442, 252, 341, 0, 0, 0, 0],
[233, 3442, 322, 31, 2514, 49731, 51, 0]])
tensor([[4326, 427, 465, 22, 314, 9725, 346, 1343],
[24, 2562, 5, 25, 356, 3145, 246, 25]])
tensor([[4524, 2465, 562, 67, 26, 265, 21, 256],
[34, 14, 0, 0, 0, 0, 0, 0]])

label:
tensor([[442, 252, 341, 233, -100, -100, -100, -100],
[3442, 322, 31, 2514, 49731, 51, -100, -100]])
tensor([[427, 465, 22, 314, 9725, 346, 1343, 24],
[2562, 5, 25, 356, 3145, 246, 25, -100]])
tensor([[2465, 562, 67, 26, 265, 21, 256, 34],
[14, -100, -100, -100, -100, -100, -100, -100]])
```

如果`tp`并行模式为`isp`,且tp size(即sequence parallel size)大于1,则会在`data_process_func`中注册`split_data_sequence_parallel`函数,对数据的`sequence`维度进行切分。

假设tp size为2,则对上述数据`data['input_ids']``label`切分之后的结果如下:

tp rank0 中的数据:
```bash
data["input_ids"]:
tensor([[2323, 442, 252, 341],
[233, 3442, 322, 31]])
tensor([[4326, 427, 465, 22],
[24, 2562, 5, 25]])
tensor([[4524, 2465, 562, 67],
[34, 14, 0, 0]])

label:
tensor([[442, 252, 341, 233],
[3442, 322, 31, 2514]])
tensor([[427, 465, 22, 314],
[2562, 5, 25, 356]])
tensor([[2465, 562, 67, 26],
[14, -100, -100, -100]])
```

tp rank1 中的数据:
```bash
data["input_ids"]:
tensor([[0, 0, 0, 0],
[2514, 49731, 51, 0]])
tensor([[314, 9725, 346, 1343],
[356, 3145, 246, 25]])
tensor([[26, 265, 21, 256],
[0, 0, 0, 0]])

label:
tensor([[-100, -100, -100, -100],
[49731, 51, -100, -100]])
tensor([[9725, 346, 1343, 24],
[3145, 246, 25, -100]])
tensor([[265, 21, 256, 34],
[-100, -100, -100, -100]])
```

### Forward过程数据格式
以internlm2模型为例,详细介绍一下在整个模型运行的过程中数据的流动过程。
[未完待续]
34 changes: 24 additions & 10 deletions doc/usage.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,18 +5,30 @@
### 安装
请参考[安装文档](./install.md)进行安装。

### 数据准备 (使用huggingface数据集)
### 数据准备 (预训练)
#### 使用huggingface格式数据集

如果使用huggingface数据集进行在线加载并且在线tokenize的话,那么以`roneneldan/TinyStories`这个数据为例,数据准备阶段只需要将配置文件做如下改动:
```python
TRAIN_FOLDER = "roneneldan/TinyStories"
如果使用huggingface数据集,需要先将数据集和需要使用的tokenizer下载到本地。

`roneneldan/TinyStories`这个数据为例,数据准备阶段需要通过如下命令将数据集下载到本地:
```bash
huggingface-cli download --repo-type dataset --resume-download "roneneldan/TinyStories" --local-dir "/mnt/petrelfs/hf-TinyStories"
```
其中,"/mnt/petrelfs/hf-TinyStories" 为需要将数据集保存的本地路径。

然后将tokenizer下载到本地,例如,使用internlm2的tokenizer,则将`https://huggingface.co/internlm/internlm2-7b/tree/main`中的`special_tokens_map.json``tokenizer.model``tokenizer_config.json``tokenization_internlm2.py``tokenization_internlm2_fast.py`文件下载到本地路径,如"/mnt/petrelfs/hf-internlm2-tokenizer"中。

将配置文件做如下改动:
```bash
TRAIN_FOLDER = "/mnt/petrelfs/hf-TinyStories"
data = dict(
type="hf",
tokenizer_path="internlm/internlm-7b",
tokenizer_path="/mnt/petrelfs/hf-internlm2-tokenizer",
)
```
type默认为"tokenized",这里需要改为"hf"类型。同时需要指定`tokenizer_path`, 如果使用下述tokenized之后的数据集,则不需要设置该字段。`TRAIN_FOLDER`指定本地数据集路径。

### 数据准备 (预训练)
#### 使用tokenized之后数据集

InternEvo训练任务的数据集包括一系列的`bin``meta`文件。使用`tokenizer`从原始文本文件生成训练用数据集。通过在`tools/tokenizer.py`中指定模型参数路径的方式来导入tokenizer模型。目前提供`V7_sft.model`来生成tokens。若想使用不同的模型,可直接修改`tokernizer.py`中的模型参数路径。

Expand Down Expand Up @@ -321,14 +333,16 @@ data = dict(
数据集的详细内容可参考``数据准备``模块相关的介绍。

同时,也支持huggingface格式的数据集处理。
train_folder设置为huggingface上可以通过load_dataset直接下载的数据集路径,如:"roneneldan/TinyStories"

train_folder设置为从huggingface上下载的本地数据集路径,如:"/mnt/petrelfs/hf-TinyStories"

在data中,需要新增type及tokenizer_path字段,标示数据集是huggingface格式,并指定tokenizer路径,如:
```python
TRAIN_FOLDER = "roneneldan/TinyStories"
TRAIN_FOLDER = "/mnt/petrelfs/hf-TinyStories"
SEQ_LEN = 2048
data = dict(
type="hf",
tokenizer_path="internlm/internlm-7b",
tokenizer_path="/mnt/petrelfs/hf-internlm2-tokenizer",
seq_len=SEQ_LEN, # 数据样本长度,默认值为 2048
micro_num=1, # micro_num 是指在一次模型参数更新中会处理的 micro_batch 的数目,默认值为 1
micro_bsz=1, # packed_length = micro_bsz * SEQ_LEN,为一次处理的 micro_batch 的数据大小,默认值为 1
Expand Down Expand Up @@ -510,4 +524,4 @@ generation = dict(

关于 Dyanmic NTK 的原理,详细请参考
1. [dynamically_scaled_rope_further_increases](https://www.reddit.com/r/LocalLLaMA/comments/14mrgpr/dynamically_scaled_rope_further_increases)
2. [https://kexue.fm/archives/9675](https://kexue.fm/archives/9675)
2. [https://kexue.fm/archives/9675](https://kexue.fm/archives/9675)

0 comments on commit 4e71314

Please sign in to comment.