From 0e354088cbcced9484ba7ad5c7566e1dcde11eaf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Stig-Arne=20Gr=C3=B6nroos?= Date: Mon, 9 Dec 2024 13:57:51 +0200 Subject: [PATCH] Training tips/best practices --- docs/source/quickstart.md | 2 + docs/source/training_tips.md | 76 ++++++++++++++++++++++++++++++++++++ 2 files changed, 78 insertions(+) create mode 100644 docs/source/training_tips.md diff --git a/docs/source/quickstart.md b/docs/source/quickstart.md index ce88fbe8..08081dbe 100644 --- a/docs/source/quickstart.md +++ b/docs/source/quickstart.md @@ -185,6 +185,8 @@ Congratulations! You've successfully translated text using your Mammoth model. A ### Further reading +[Best practices for training](training_tips.md). + Reference documentation for the `config_config` tool can be found at [The config_config tool](config_config.md). A complete example for configuring different parameter sharing schemes is available at [MAMMOTH sharing schemes](examples/sharing_schemes.md). diff --git a/docs/source/training_tips.md b/docs/source/training_tips.md new file mode 100644 index 00000000..b0a386b2 --- /dev/null +++ b/docs/source/training_tips.md @@ -0,0 +1,76 @@ +# Tips and tricks for training + +## Speed up training with token-based minibatches and gradient accumulation/maxibatching + +Waiting for synchronization is a common reason for inefficiency in distributed training across multiple devices. +Each device performs its forward and backward pass computation independently, +but there is a synchronization point in communicating the gradients before the optimizer can be stepped. +If just one device is still computing its contribution to the gradient, +all other devices have to wait idly even though they have already finished their work. +There are two approaches to maximize throughput: + +1. Balance the workload between devices by using token-based minibatch size `batch_type: tokens` +2. Perform more work before communicating the gradient. + - Set `accum_count: 10` to accumulate the gradient over 10 minibatches before communicating. + - Set `lookahead_minibatches: 10` to the same value as `accum_count` to make the dataloader read in one maxibatch at a time, + and locally sort the contents by length. This minimizes padding waste. + + +## Use `decay_method: linear_warmup` for learning rate scheduling + +Select a decay method before tuning the learning rate. +The recommended method is `linear_warmup`, +which ramps up learning rate linearly for `warmup_steps`, then decays it linearly until `train_steps`. + + +Note that the OpenNMT legacy decay methods have inconsistent scaling of the maximum learning rate. +Changing the decay method also rescales the learning rate in unintuitive ways. + +The `rsqrt` and `exponential_decay` methods don't apply warmup, making them unsuitable for Transformers with SGD or Adam. + + +## Don't rely on `max_grad_norm` to save you from too high learning rate + +The norm of the gradient of each distributed component is clipped, if it exceeds `max_grad_norm`. +Don't rely on max_grad_norm to save you from too high learning rate: +as each component is clipped individually, renormalization does NOT preserve the direction of the global gradient. + +Keep an eye on the logged number of times that gradient clipping has been applied: `n_clips`. +A few clips are likely to be ok, but repeated clipping indicates a need to tune the hyperparameters. + + +## Recommended minimal opts for x-transformers + +You can pass through opts to the x-transformers library in the `x_transformers_opts` dict. + +```yaml +x_transformers_opts: + # Use flash attention + attn_flash: True + # The number of attention heads + heads: 16 + # Use rotary positional embeddings. + rotary_pos_emb: True + # Tie the input and output embeddings of the decoder + tie_embedding: True +``` + +Note in particular the rotary positional embeddings `rotary_pos_emb`. +This seems to be the only type of positional embedding that works properly in Mammoth. + + +## Save storage and speed up config-config by using transforms instead of external preprocessing + +There are two approaches to preprocessing (e.g. subword tokenization, prefixing, autoencoder noise, ...) + +1. Pre-apply the transforms using an external tool. Write the results to disk. Point Mammoth at the transformed files. +2. Apply the transforms at run time using Mammoth. Point Mammoth at the raw original files. + +There are multiple benefits to the latter approach of using Mammoth transforms: + +- The transformation is applied online, and the result is not saved to disk. + This saves storage, which is especially relevant when using very large corpora and sampling different variations for each minibatch. +- Config-config uses the cached line counts of the original files. The tool runs faster when it doesn't need to recount the lines. +- It is easy to apply sampling of different variations for each minibatch, e.g. subword regularization or denoising autoencoder. +- It is easy to use the same corpus files symmetrically (e.g. the same files for English->Finnish and Finnish->English) + even though prefixing the source data with language selection tokens.