## Generate Text with MLX and :hugs: Hugging Face This an example of large language model text generation that can pull models from the Hugging Face Hub. For more information on this example, see the [README](../README.md) in the parent directory. This package also supports fine tuning with LoRA or QLoRA. For more information see the [LoRA documentation](LORA.md). ## Reinforcement Learning from Human Feedback (RLHF) with Direct Preference Optimization (DPO) This package now includes an example of Reinforcement Learning from Human Feedback (RLHF) using the Direct Preference Optimization (DPO) method. ### Paper [Direct Preference Optimization: Your Language Model is Secretly a Reward Model](https://arxiv.org/abs/2305.18290) ### Notes [Direct Preference Optimization (DPO): A Simplified Explanation by João Lages](https://medium.com/@joaolages/direct-preference-optimization-dpo-622fc1f18707) ![](https://miro.medium.com/v2/resize:fit:1400/format:webp/1*AqKOT0pxzi5kOgiobb-Fvg.png) ### Implementation examples - [huggingface/trl: TRL - Transformer Reinforcement Learning](https://github.com/huggingface/trl) - [eric-mitchell/direct-preference-optimization: Direct Preference Optimization](https://github.com/eric-mitchell/direct-preference-optimization) ### Possible MLX implementation Policy and reference log probabilities: ```python def get_batched_logps(model, inputs, targets): logits, _ = model(inputs) logits = logits.astype(mx.float32) loss_mask = targets != 0 per_token_logps = mx.take_along_axis(nn.log_softmax(logits), targets[..., None], axis=2).squeeze(2) return tuple((per_token_logps * loss_mask).sum(-1).split(2)) ``` Loss: ```python def dpo_loss(model, beta, label_smoothing, reference_chosen_logps, reference_rejected_logps, inputs, targets): chosen_logps, rejected_logps = get_batched_logps(model, inputs, targets) pi_logratios = chosen_logps - rejected_logps reference_logratios = reference_chosen_logps - reference_rejected_logps logits = pi_logratios - reference_logratios losses = -nn.log_sigmoid(beta * logits) * (1.0 - label_smoothing) - nn.log_sigmoid(-beta * logits) * label_smoothing chosen_rewards = beta * (chosen_logps - reference_chosen_logps) rejected_rewards = beta * (rejected_logps - reference_rejected_logps) reward_accuracies = (chosen_rewards > rejected_rewards).astype(mx.float32) reward_margins = chosen_rewards - rejected_rewards ntoks = (inputs != 0).sum() return ( losses.mean(), chosen_rewards.mean(), rejected_rewards.mean(), reward_accuracies.mean(), reward_margins.mean(), ntoks, ) ``` Beta: The temperature parameter for the DPO loss is typically set in the range of 0.1 to 0.5. The reference model is ignored when `beta` equals 0. Label smoothing: This parameter represents the conservativeness for DPO loss, assuming that preferences are noisy and can be flipped with a probability of `label_smoothing`. > **Note** `label_smoothing > 0` defines the [Conservative DPO](https://ericmitchell.ai/cdpo.pdf) loss. ### Usage Instructions To use the Direct Preference Optimization (DPO) method in your training, follow these steps: 1. **Add Configuration Options**: Update your configuration file (e.g., `llms/mlx_lm/examples/lora_config.yaml`) to include the DPO-specific options: ```yaml loss_type: "dpo" beta: 0.1 label_smoothing: 0.0 ``` 2. **Implement DPO Functions**: Ensure that the `get_batched_logps` and `dpo_loss` functions are implemented in your `llms/mlx_lm/utils.py` file. 3. **Update Training Logic**: Modify your training script (e.g., `llms/mlx_lm/tuner/trainer.py`) to include DPO-specific training logic. This involves updating the `train` function to check for the DPO loss type and apply the DPO loss calculation accordingly. 4. **Run Training**: Execute your training script with the updated configuration and logic to train your model using the DPO method. By following these steps, you can leverage the Direct Preference Optimization (DPO) method for Reinforcement Learning from Human Feedback (RLHF) in your MLX training pipeline.