utils.py 2.17 KB
Newer Older
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
1
2
3
4
5
6
from typing import Optional, Union

import torch
import torch.nn.functional as F


7
8
9
def _compute_approx_kl(
    log_probs: torch.Tensor, log_probs_base: torch.Tensor, action_mask: Optional[torch.Tensor] = None
) -> torch.Tensor:
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
10
11
12
13
14
15
16
17
18
19
    """
    Compute the approximate KL divergence between two distributions.
    Schulman blog: http://joschu.net/blog/kl-approx.html

    Args:
        log_probs: Log probabilities of the new distribution.
        log_probs_base: Log probabilities of the base distribution.
        action_mask: Mask for actions.
    """

20
    log_ratio = log_probs_base - log_probs
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
21
22
23
24
25
26
27
28
    approx_kl = (log_ratio.exp() - 1) - log_ratio
    if action_mask is not None:
        approx_kl = masked_mean(approx_kl, action_mask, dim=1)
        return approx_kl
    approx_kl = approx_kl.mean(dim=1)
    return approx_kl


29
30
31
32
33
34
35
def compute_reward(
    r: Union[torch.Tensor, float],
    kl_coef: float,
    log_probs: torch.Tensor,
    log_probs_base: torch.Tensor,
    action_mask: Optional[torch.Tensor] = None,
) -> torch.Tensor:
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
36
37
    if kl_coef <= 0.0:
        return r
38
    kl = _compute_approx_kl(log_probs, log_probs_base, action_mask=action_mask)
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
39
40
41
42
    reward = r - kl_coef * kl
    return reward


43
def _log_probs_from_logits(logits: torch.Tensor, labels: torch.Tensor) -> torch.Tensor:
Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
44
45
46
47
48
    log_probs = F.log_softmax(logits, dim=-1)
    log_probs_labels = log_probs.gather(dim=-1, index=labels.unsqueeze(-1))
    return log_probs_labels.squeeze(-1)


49
def calc_action_log_probs(output: torch.Tensor, sequences: torch.LongTensor, num_actions: int) -> torch.Tensor:
50
51
52
53
54
55
56
57
58
59
    """Calculate action log probs.

    Args:
        output (torch.Tensor): Output tensor of Actor.forward.
        sequences (torch.LongTensor): Input sequences.
        num_actions (int): Number of actions.

    Returns:
        torch.Tensor: Action log probs.
    """
60
    logits = output["logits"]
61
    log_probs = _log_probs_from_logits(logits[:, :-1, :], sequences[:, 1:])
62
63
64
    return log_probs[:, -num_actions:]


Fazzie-Maqianli's avatar
Fazzie-Maqianli committed
65
66
67
68
69
70
def masked_mean(tensor: torch.Tensor, mask: torch.Tensor, dim: int = 1) -> torch.Tensor:
    tensor = tensor * mask
    tensor = tensor.sum(dim=dim)
    mask_sum = mask.sum(dim=dim)
    mean = tensor / (mask_sum + 1e-8)
    return mean