backup wip

This commit is contained in:
Alexander Soare 2024-04-03 14:21:07 +01:00
parent c7d70a8db9
commit 110ac5ffa1
6 changed files with 182 additions and 191 deletions

View File

@ -191,7 +191,6 @@ class AlohaEnv(AbstractEnv):
{ {
"observation": TensorDict(obs, batch_size=[]), "observation": TensorDict(obs, batch_size=[]),
"reward": torch.tensor([reward], dtype=torch.float32), "reward": torch.tensor([reward], dtype=torch.float32),
# success and done are true when coverage > self.success_threshold in env
"done": torch.tensor([done], dtype=torch.bool), "done": torch.tensor([done], dtype=torch.bool),
"success": torch.tensor([success], dtype=torch.bool), "success": torch.tensor([success], dtype=torch.bool),
}, },

View File

@ -1,18 +1,12 @@
import einops
import numpy as np import numpy as np
import torch import torch
from torch import nn from torch import nn
from torch.autograd import Variable
from .backbone import build_backbone from .backbone import build_backbone
from .transformer import TransformerEncoder, TransformerEncoderLayer, build_transformer from .transformer import TransformerEncoder, TransformerEncoderLayer, build_transformer
def reparametrize(mu, logvar):
std = logvar.div(2).exp()
eps = Variable(std.data.new(std.size()).normal_())
return mu + std * eps
def get_sinusoid_encoding_table(n_position, d_hid): def get_sinusoid_encoding_table(n_position, d_hid):
def get_position_angle_vec(position): def get_position_angle_vec(position):
return [position / np.power(10000, 2 * (hid_j // 2) / d_hid) for hid_j in range(d_hid)] return [position / np.power(10000, 2 * (hid_j // 2) / d_hid) for hid_j in range(d_hid)]
@ -27,7 +21,7 @@ def get_sinusoid_encoding_table(n_position, d_hid):
class ActionChunkingTransformer(nn.Module): class ActionChunkingTransformer(nn.Module):
""" """
Action Chunking Transformer as per Learning Fine-Grained Bimanual Manipulation with Low-Cost Hardware Action Chunking Transformer as per Learning Fine-Grained Bimanual Manipulation with Low-Cost Hardware
(https://arxiv.org/abs/2304.13705). (paper: https://arxiv.org/abs/2304.13705, code: https://github.com/tonyzhaozh/act)
Note: In this code we use the symbols `vae_encoder`, 'encoder', `decoder`. The meanings are as follows. Note: In this code we use the symbols `vae_encoder`, 'encoder', `decoder`. The meanings are as follows.
- The `vae_encoder` is, as per the literature around conditional variational auto-encoders (cVAE), the - The `vae_encoder` is, as per the literature around conditional variational auto-encoders (cVAE), the
@ -49,7 +43,7 @@ class ActionChunkingTransformer(nn.Module):
""" """
def __init__( def __init__(
self, backbones, transformer, vae_encoder, state_dim, action_dim, horizon, camera_names, vae self, backbones, transformer, vae_encoder, state_dim, action_dim, horizon, camera_names, use_vae
): ):
"""Initializes the model. """Initializes the model.
Parameters: Parameters:
@ -63,134 +57,124 @@ class ActionChunkingTransformer(nn.Module):
state_dim: Robot positional state dimension. state_dim: Robot positional state dimension.
action_dim: Action dimension. action_dim: Action dimension.
horizon: The number of actions to generate in one forward pass. horizon: The number of actions to generate in one forward pass.
vae: Whether to use the variational objective. TODO(now): Give more details. use_vae: Whether to use the variational objective. TODO(now): Give more details.
""" """
super().__init__() super().__init__()
self.camera_names = camera_names self.camera_names = camera_names
self.transformer = transformer self.transformer = transformer
self.vae_encoder = vae_encoder self.vae_encoder = vae_encoder
self.vae = vae self.use_vae = use_vae
hidden_dim = transformer.d_model hidden_dim = transformer.d_model
self.action_head = nn.Linear(hidden_dim, action_dim)
self.is_pad_head = nn.Linear(hidden_dim, 1)
# Positional embedding to be used as input to the latent vae_encoder (if applicable) and for the
self.pos_embed = nn.Embedding(horizon, hidden_dim)
if backbones is not None:
self.input_proj = nn.Conv2d(backbones[0].num_channels, hidden_dim, kernel_size=1)
self.backbones = nn.ModuleList(backbones)
self.input_proj_robot_state = nn.Linear(state_dim, hidden_dim)
else:
# input_dim = 14 + 7 # robot_state + env_state
self.input_proj_robot_state = nn.Linear(state_dim, hidden_dim)
# TODO(rcadene): understand what is env_state, and why it needs to be 7
self.input_proj_env_state = nn.Linear(state_dim // 2, hidden_dim)
self.pos = torch.nn.Embedding(2, hidden_dim)
self.backbones = None
# vae_encoder extra parameters # BERT style VAE encoder with input [cls, *joint_space_configuration, *action_sequence].
self.latent_dim = 32 # final size of latent z # TODO tune # The cls token forms parameters of the latent's distribution (like this [*means, *log_variances]).
self.cls_embed = nn.Embedding(1, hidden_dim) # extra cls token embedding if use_vae:
self.vae_encoder_action_proj = nn.Linear(14, hidden_dim) # project action to embedding self.cls_embed = nn.Embedding(1, hidden_dim)
self.vae_encoder_joint_proj = nn.Linear(14, hidden_dim) # project qpos to embedding # Projection layer for joint-space configuration to hidden dimension.
self.latent_proj = nn.Linear( self.vae_encoder_robot_state_input_proj = nn.Linear(state_dim, hidden_dim)
hidden_dim, self.latent_dim * 2 # Projection layer for action (joint-space target) to hidden dimension.
) # project hidden state to latent std, var self.vae_encoder_action_input_proj = nn.Linear(state_dim, hidden_dim)
self.register_buffer( # Final size of latent z. TODO(now): Add to hyperparams.
"pos_table", get_sinusoid_encoding_table(1 + 1 + horizon, hidden_dim) self.latent_dim = 32
) # [CLS], qpos, a_seq # Projection layer from the VAE encoder's output to the latent distribution's parameter space.
self.vae_encoder_latent_output_proj = nn.Linear(hidden_dim, self.latent_dim * 2)
# Fixed sinusoidal positional embedding the whole input to the VAE encoder.
self.register_buffer(
"vae_encoder_pos_enc", get_sinusoid_encoding_table(1 + 1 + horizon, hidden_dim)
)
# decoder extra parameters # Transformer encoder input projections. The tokens will be structured like
self.latent_out_proj = nn.Linear(self.latent_dim, hidden_dim) # project latent sample to embedding # [latent, robot_state, image_feature_map_pixels].
self.backbones = nn.ModuleList(backbones)
self.encoder_img_feat_input_proj = nn.Conv2d(backbones[0].num_channels, hidden_dim, kernel_size=1)
self.encoder_robot_state_input_proj = nn.Linear(state_dim, hidden_dim)
self.encoder_latent_input_proj = nn.Linear(self.latent_dim, hidden_dim)
# TODO(now): Fix this nonsense. One positional embedding is needed. We should extract the image
# feature dimension with a dry run.
self.additional_pos_embed = nn.Embedding( self.additional_pos_embed = nn.Embedding(
2, hidden_dim 2, hidden_dim
) # learned position embedding for proprio and latent ) # learned position embedding for proprio and latent
def forward(self, qpos, image, env_state, actions=None, is_pad=None): # Transformer decoder.
# Learnable positional embedding for the transformer's decoder (in the style of DETR object queries).
self.decoder_pos_embed = nn.Embedding(horizon, hidden_dim)
# Final action regression head on the output of the transformer's decoder.
self.action_head = nn.Linear(hidden_dim, action_dim)
def forward(self, robot_state, image, actions=None):
""" """
qpos: batch, qpos_dim Args:
image: batch, num_cam, channel, height, width robot_state: (B, J) batch of robot joint configurations.
env_state: None image: (B, N, C, H, W) batch of N camera frames.
actions: batch, seq, action_dim actions: (B, S, A) batch of actions from the target dataset which must be provided if the
VAE is enabled and the model is in training mode.
""" """
is_training = actions is not None # train or val if self.use_vae and self.training:
bs, _ = qpos.shape assert (
### Obtain latent z from action sequence actions is not None
if self.vae and is_training: ), "actions must be provided when using the variational objective in training mode."
# project action sequence to embedding dim, and concat with a CLS token
action_embed = self.vae_encoder_action_proj(actions) # (bs, seq, hidden_dim) batch_size, _ = robot_state.shape
qpos_embed = self.vae_encoder_joint_proj(qpos) # (bs, hidden_dim)
qpos_embed = torch.unsqueeze(qpos_embed, axis=1) # (bs, 1, hidden_dim) # Prepare the latent for input to the transformer.
cls_embed = self.cls_embed.weight # (1, hidden_dim) if self.use_vae and actions is not None:
cls_embed = torch.unsqueeze(cls_embed, axis=0).repeat(bs, 1, 1) # (bs, 1, hidden_dim) # Prepare the input to the VAE encoder: [cls, *joint_space_configuration, *action_sequence].
vae_encoder_input = torch.cat( cls_embed = einops.repeat(self.cls_embed.weight, "1 d -> b 1 d", b=batch_size) # (B, 1, D)
[cls_embed, qpos_embed, action_embed], axis=1 robot_state_embed = self.vae_encoder_robot_state_input_proj(robot_state).unsqueeze(1) # (B, 1, D)
) # (bs, seq+1, hidden_dim) action_embed = self.vae_encoder_action_input_proj(actions) # (B, S, D)
vae_encoder_input = vae_encoder_input.permute(1, 0, 2) # (seq+1, bs, hidden_dim) vae_encoder_input = torch.cat([cls_embed, robot_state_embed, action_embed], axis=1) # (B, S+2, D)
# do not mask cls token vae_encoder_input = vae_encoder_input.permute(1, 0, 2) # (S+2, B, D)
# cls_joint_is_pad = torch.full((bs, 2), False).to(qpos.device) # False: not a padding # Note: detach() shouldn't be necessary but leaving it the same as the original code just in case.
# is_pad = torch.cat([cls_joint_is_pad, is_pad], axis=1) # (bs, seq+1) # Prepare fixed positional embedding.
# obtain position embedding pos_embed = self.vae_encoder_pos_enc.clone().detach().permute(1, 0, 2) # (S+2, 1, D)
pos_embed = self.pos_table.clone().detach() # Forward pass through VAE encoder and sample the latent with the reparameterization trick.
pos_embed = pos_embed.permute(1, 0, 2) # (seq+1, 1, hidden_dim)
# query model
vae_encoder_output = self.vae_encoder( vae_encoder_output = self.vae_encoder(
vae_encoder_input, pos=pos_embed vae_encoder_input, pos=pos_embed
) # , src_key_padding_mask=is_pad) ) # , src_key_padding_mask=is_pad) # TODO(now)
vae_encoder_output = vae_encoder_output[0] # take cls output only vae_encoder_output = vae_encoder_output[0] # take cls output only
latent_info = self.latent_proj(vae_encoder_output) latent_pdf_params = self.vae_encoder_latent_output_proj(vae_encoder_output)
mu = latent_info[:, : self.latent_dim] mu = latent_pdf_params[:, : self.latent_dim]
logvar = latent_info[:, self.latent_dim :] logvar = latent_pdf_params[:, self.latent_dim :]
latent_sample = reparametrize(mu, logvar) # Use reparameterization trick to sample from the latent's PDF.
latent_input = self.latent_out_proj(latent_sample) latent_sample = mu + logvar.div(2).exp() * torch.randn_like(mu)
else: else:
# When not using the VAE encoder, we set the latent to be all zeros.
mu = logvar = None mu = logvar = None
latent_sample = torch.zeros([bs, self.latent_dim], dtype=torch.float32).to(qpos.device) latent_sample = torch.zeros([batch_size, self.latent_dim], dtype=robot_state.dtype).to(
latent_input = self.latent_out_proj(latent_sample) robot_state.device
)
if self.backbones is not None: # Prepare all other transformer inputs.
# Image observation features and position embeddings # Image observation features and position embeddings.
all_cam_features = [] all_cam_features = []
all_cam_pos = [] all_cam_pos = []
for cam_id, _ in enumerate(self.camera_names): for cam_id, _ in enumerate(self.camera_names):
features, pos = self.backbones[0](image[:, cam_id]) # HARDCODED # TODO(now): remove the positional embedding from the backbones.
features = features[0] # take the last layer feature features, pos = self.backbones[0](image[:, cam_id]) # HARDCODED
pos = pos[0] features = features[0] # take the last layer feature
all_cam_features.append(self.input_proj(features)) pos = pos[0]
all_cam_pos.append(pos) all_cam_features.append(self.encoder_img_feat_input_proj(features))
# proprioception features all_cam_pos.append(pos)
proprio_input = self.input_proj_robot_state(qpos) # Concatenate image observation feature maps along the width dimension.
# fold camera dimension into width dimension transformer_input = torch.cat(all_cam_features, axis=3)
src = torch.cat(all_cam_features, axis=3) # TODO(now): remove the positional embedding from the backbones.
pos = torch.cat(all_cam_pos, axis=3) pos = torch.cat(all_cam_pos, axis=3)
hs = self.transformer( robot_state_embed = self.encoder_robot_state_input_proj(robot_state)
src, latent_embed = self.encoder_latent_input_proj(latent_sample)
None,
self.pos_embed.weight,
pos,
latent_input,
proprio_input,
self.additional_pos_embed.weight,
)[0]
else:
qpos = self.input_proj_robot_state(qpos)
env_state = self.input_proj_env_state(env_state)
transformer_input = torch.cat([qpos, env_state], axis=1) # seq length = 2
hs = self.transformer(transformer_input, None, self.pos_embed.weight, self.pos.weight)[0]
a_hat = self.action_head(hs)
is_pad_hat = self.is_pad_head(hs)
return a_hat, is_pad_hat, [mu, logvar]
# Run the transformer and project the outputs to the action space.
transformer_output = self.transformer(
transformer_input,
query_embed=self.decoder_pos_embed.weight,
pos_embed=pos,
latent_input=latent_embed,
proprio_input=robot_state_embed,
additional_pos_embed=self.additional_pos_embed.weight,
)
a_hat = self.action_head(transformer_output)
def mlp(input_dim, hidden_dim, output_dim, hidden_depth): return a_hat, [mu, logvar]
if hidden_depth == 0:
mods = [nn.Linear(input_dim, output_dim)]
else:
mods = [nn.Linear(input_dim, hidden_dim), nn.ReLU(inplace=True)]
for _ in range(hidden_depth - 1):
mods += [nn.Linear(hidden_dim, hidden_dim), nn.ReLU(inplace=True)]
mods.append(nn.Linear(hidden_dim, output_dim))
trunk = nn.Sequential(*mods)
return trunk
def build_vae_encoder(args): def build_vae_encoder(args):
@ -231,7 +215,7 @@ def build(args):
action_dim=args.action_dim, action_dim=args.action_dim,
horizon=args.num_queries, horizon=args.num_queries,
camera_names=args.camera_names, camera_names=args.camera_names,
vae=args.vae, use_vae=args.vae,
) )
n_parameters = sum(p.numel() for p in model.parameters() if p.requires_grad) n_parameters = sum(p.numel() for p in model.parameters() if p.requires_grad)

View File

@ -224,8 +224,7 @@ class ActionChunkingTransformerPolicy(AbstractPolicy):
if is_pad is not None: if is_pad is not None:
is_pad = is_pad[:, : self.model.num_queries] is_pad = is_pad[:, : self.model.num_queries]
breakpoint() a_hat, (mu, logvar) = self.model(qpos, image, env_state, actions, is_pad)
a_hat, is_pad_hat, (mu, logvar) = self.model(qpos, image, env_state, actions, is_pad)
all_l1 = F.l1_loss(actions, a_hat, reduction="none") all_l1 = F.l1_loss(actions, a_hat, reduction="none")
l1 = all_l1.mean() if is_pad is None else (all_l1 * ~is_pad.unsqueeze(-1)).mean() l1 = all_l1.mean() if is_pad is None else (all_l1 * ~is_pad.unsqueeze(-1)).mean()
@ -240,5 +239,5 @@ class ActionChunkingTransformerPolicy(AbstractPolicy):
loss_dict["loss"] = loss_dict["l1"] loss_dict["loss"] = loss_dict["l1"]
return loss_dict return loss_dict
else: else:
action, _, (_, _) = self.model(qpos, image, env_state) # no action, sample from prior action, _ = self.model(qpos, image, env_state) # no action, sample from prior
return action return action

View File

@ -26,10 +26,8 @@ class Transformer(nn.Module):
dropout=0.1, dropout=0.1,
activation="relu", activation="relu",
normalize_before=False, normalize_before=False,
return_intermediate_dec=False,
): ):
super().__init__() super().__init__()
encoder_layer = TransformerEncoderLayer( encoder_layer = TransformerEncoderLayer(
d_model, nhead, dim_feedforward, dropout, activation, normalize_before d_model, nhead, dim_feedforward, dropout, activation, normalize_before
) )
@ -40,9 +38,7 @@ class Transformer(nn.Module):
d_model, nhead, dim_feedforward, dropout, activation, normalize_before d_model, nhead, dim_feedforward, dropout, activation, normalize_before
) )
decoder_norm = nn.LayerNorm(d_model) decoder_norm = nn.LayerNorm(d_model)
self.decoder = TransformerDecoder( self.decoder = TransformerDecoder(decoder_layer, num_decoder_layers, decoder_norm)
decoder_layer, num_decoder_layers, decoder_norm, return_intermediate=return_intermediate_dec
)
self._reset_parameters() self._reset_parameters()
@ -57,7 +53,6 @@ class Transformer(nn.Module):
def forward( def forward(
self, self,
src, src,
mask,
query_embed, query_embed,
pos_embed, pos_embed,
latent_input=None, latent_input=None,
@ -68,10 +63,10 @@ class Transformer(nn.Module):
if len(src.shape) == 4: # has H and W if len(src.shape) == 4: # has H and W
# flatten NxCxHxW to HWxNxC # flatten NxCxHxW to HWxNxC
bs, c, h, w = src.shape bs, c, h, w = src.shape
# Each "pixel" on the feature maps will form a token.
src = src.flatten(2).permute(2, 0, 1) src = src.flatten(2).permute(2, 0, 1)
pos_embed = pos_embed.flatten(2).permute(2, 0, 1).repeat(1, bs, 1) pos_embed = pos_embed.flatten(2).permute(2, 0, 1).repeat(1, bs, 1)
query_embed = query_embed.unsqueeze(1).repeat(1, bs, 1) query_embed = query_embed.unsqueeze(1).repeat(1, bs, 1)
# mask = mask.flatten(1)
additional_pos_embed = additional_pos_embed.unsqueeze(1).repeat(1, bs, 1) # seq, bs, dim additional_pos_embed = additional_pos_embed.unsqueeze(1).repeat(1, bs, 1) # seq, bs, dim
pos_embed = torch.cat([additional_pos_embed, pos_embed], axis=0) pos_embed = torch.cat([additional_pos_embed, pos_embed], axis=0)
@ -87,9 +82,9 @@ class Transformer(nn.Module):
query_embed = query_embed.unsqueeze(1).repeat(1, bs, 1) query_embed = query_embed.unsqueeze(1).repeat(1, bs, 1)
tgt = torch.zeros_like(query_embed) tgt = torch.zeros_like(query_embed)
memory = self.encoder(src, src_key_padding_mask=mask, pos=pos_embed) memory = self.encoder(src, pos=pos_embed)
hs = self.decoder(tgt, memory, memory_key_padding_mask=mask, pos=pos_embed, query_pos=query_embed) hs = self.decoder(tgt, memory, pos=pos_embed, query_pos=query_embed)
hs = hs.transpose(1, 2) hs = hs.transpose(0, 1)
return hs return hs
@ -103,14 +98,12 @@ class TransformerEncoder(nn.Module):
def forward( def forward(
self, self,
src, src,
mask: Optional[Tensor] = None,
src_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
): ):
output = src output = src
for layer in self.layers: for layer in self.layers:
output = layer(output, src_mask=mask, src_key_padding_mask=src_key_padding_mask, pos=pos) output = layer(output, pos=pos)
if self.norm is not None: if self.norm is not None:
output = self.norm(output) output = self.norm(output)
@ -119,52 +112,33 @@ class TransformerEncoder(nn.Module):
class TransformerDecoder(nn.Module): class TransformerDecoder(nn.Module):
def __init__(self, decoder_layer, num_layers, norm=None, return_intermediate=False): def __init__(self, decoder_layer, num_layers, norm=None):
super().__init__() super().__init__()
self.layers = _get_clones(decoder_layer, num_layers) self.layers = _get_clones(decoder_layer, num_layers)
self.num_layers = num_layers self.num_layers = num_layers
self.norm = norm self.norm = norm
self.return_intermediate = return_intermediate
def forward( def forward(
self, self,
tgt, tgt,
memory, memory,
tgt_mask: Optional[Tensor] = None,
memory_mask: Optional[Tensor] = None,
tgt_key_padding_mask: Optional[Tensor] = None,
memory_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
query_pos: Optional[Tensor] = None, query_pos: Optional[Tensor] = None,
): ):
output = tgt output = tgt
intermediate = []
for layer in self.layers: for layer in self.layers:
output = layer( output = layer(
output, output,
memory, memory,
tgt_mask=tgt_mask,
memory_mask=memory_mask,
tgt_key_padding_mask=tgt_key_padding_mask,
memory_key_padding_mask=memory_key_padding_mask,
pos=pos, pos=pos,
query_pos=query_pos, query_pos=query_pos,
) )
if self.return_intermediate:
intermediate.append(self.norm(output))
if self.norm is not None: if self.norm is not None:
output = self.norm(output) output = self.norm(output)
if self.return_intermediate:
intermediate.pop()
intermediate.append(output)
if self.return_intermediate: return output
return torch.stack(intermediate)
return output.unsqueeze(0)
class TransformerEncoderLayer(nn.Module): class TransformerEncoderLayer(nn.Module):
@ -192,12 +166,10 @@ class TransformerEncoderLayer(nn.Module):
def forward_post( def forward_post(
self, self,
src, src,
src_mask: Optional[Tensor] = None,
src_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
): ):
q = k = self.with_pos_embed(src, pos) q = k = self.with_pos_embed(src, pos)
src2 = self.self_attn(q, k, value=src, attn_mask=src_mask, key_padding_mask=src_key_padding_mask)[0] src2 = self.self_attn(q, k, value=src)[0]
src = src + self.dropout1(src2) src = src + self.dropout1(src2)
src = self.norm1(src) src = self.norm1(src)
src2 = self.linear2(self.dropout(self.activation(self.linear1(src)))) src2 = self.linear2(self.dropout(self.activation(self.linear1(src))))
@ -208,13 +180,11 @@ class TransformerEncoderLayer(nn.Module):
def forward_pre( def forward_pre(
self, self,
src, src,
src_mask: Optional[Tensor] = None,
src_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
): ):
src2 = self.norm1(src) src2 = self.norm1(src)
q = k = self.with_pos_embed(src2, pos) q = k = self.with_pos_embed(src2, pos)
src2 = self.self_attn(q, k, value=src2, attn_mask=src_mask, key_padding_mask=src_key_padding_mask)[0] src2 = self.self_attn(q, k, value=src2)[0]
src = src + self.dropout1(src2) src = src + self.dropout1(src2)
src2 = self.norm2(src) src2 = self.norm2(src)
src2 = self.linear2(self.dropout(self.activation(self.linear1(src2)))) src2 = self.linear2(self.dropout(self.activation(self.linear1(src2))))
@ -224,13 +194,11 @@ class TransformerEncoderLayer(nn.Module):
def forward( def forward(
self, self,
src, src,
src_mask: Optional[Tensor] = None,
src_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
): ):
if self.normalize_before: if self.normalize_before:
return self.forward_pre(src, src_mask, src_key_padding_mask, pos) return self.forward_pre(src, pos)
return self.forward_post(src, src_mask, src_key_padding_mask, pos) return self.forward_post(src, pos)
class TransformerDecoderLayer(nn.Module): class TransformerDecoderLayer(nn.Module):
@ -262,23 +230,17 @@ class TransformerDecoderLayer(nn.Module):
self, self,
tgt, tgt,
memory, memory,
tgt_mask: Optional[Tensor] = None,
memory_mask: Optional[Tensor] = None,
tgt_key_padding_mask: Optional[Tensor] = None,
memory_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
query_pos: Optional[Tensor] = None, query_pos: Optional[Tensor] = None,
): ):
q = k = self.with_pos_embed(tgt, query_pos) q = k = self.with_pos_embed(tgt, query_pos)
tgt2 = self.self_attn(q, k, value=tgt, attn_mask=tgt_mask, key_padding_mask=tgt_key_padding_mask)[0] tgt2 = self.self_attn(q, k, value=tgt)[0]
tgt = tgt + self.dropout1(tgt2) tgt = tgt + self.dropout1(tgt2)
tgt = self.norm1(tgt) tgt = self.norm1(tgt)
tgt2 = self.multihead_attn( tgt2 = self.multihead_attn(
query=self.with_pos_embed(tgt, query_pos), query=self.with_pos_embed(tgt, query_pos),
key=self.with_pos_embed(memory, pos), key=self.with_pos_embed(memory, pos),
value=memory, value=memory,
attn_mask=memory_mask,
key_padding_mask=memory_key_padding_mask,
)[0] )[0]
tgt = tgt + self.dropout2(tgt2) tgt = tgt + self.dropout2(tgt2)
tgt = self.norm2(tgt) tgt = self.norm2(tgt)
@ -291,24 +253,18 @@ class TransformerDecoderLayer(nn.Module):
self, self,
tgt, tgt,
memory, memory,
tgt_mask: Optional[Tensor] = None,
memory_mask: Optional[Tensor] = None,
tgt_key_padding_mask: Optional[Tensor] = None,
memory_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
query_pos: Optional[Tensor] = None, query_pos: Optional[Tensor] = None,
): ):
tgt2 = self.norm1(tgt) tgt2 = self.norm1(tgt)
q = k = self.with_pos_embed(tgt2, query_pos) q = k = self.with_pos_embed(tgt2, query_pos)
tgt2 = self.self_attn(q, k, value=tgt2, attn_mask=tgt_mask, key_padding_mask=tgt_key_padding_mask)[0] tgt2 = self.self_attn(q, k, value=tgt2)[0]
tgt = tgt + self.dropout1(tgt2) tgt = tgt + self.dropout1(tgt2)
tgt2 = self.norm2(tgt) tgt2 = self.norm2(tgt)
tgt2 = self.multihead_attn( tgt2 = self.multihead_attn(
query=self.with_pos_embed(tgt2, query_pos), query=self.with_pos_embed(tgt2, query_pos),
key=self.with_pos_embed(memory, pos), key=self.with_pos_embed(memory, pos),
value=memory, value=memory,
attn_mask=memory_mask,
key_padding_mask=memory_key_padding_mask,
)[0] )[0]
tgt = tgt + self.dropout2(tgt2) tgt = tgt + self.dropout2(tgt2)
tgt2 = self.norm3(tgt) tgt2 = self.norm3(tgt)
@ -320,10 +276,6 @@ class TransformerDecoderLayer(nn.Module):
self, self,
tgt, tgt,
memory, memory,
tgt_mask: Optional[Tensor] = None,
memory_mask: Optional[Tensor] = None,
tgt_key_padding_mask: Optional[Tensor] = None,
memory_key_padding_mask: Optional[Tensor] = None,
pos: Optional[Tensor] = None, pos: Optional[Tensor] = None,
query_pos: Optional[Tensor] = None, query_pos: Optional[Tensor] = None,
): ):
@ -331,16 +283,10 @@ class TransformerDecoderLayer(nn.Module):
return self.forward_pre( return self.forward_pre(
tgt, tgt,
memory, memory,
tgt_mask,
memory_mask,
tgt_key_padding_mask,
memory_key_padding_mask,
pos, pos,
query_pos, query_pos,
) )
return self.forward_post( return self.forward_post(tgt, memory, pos, query_pos)
tgt, memory, tgt_mask, memory_mask, tgt_key_padding_mask, memory_key_padding_mask, pos, query_pos
)
def _get_clones(module, n): def _get_clones(module, n):
@ -356,7 +302,6 @@ def build_transformer(args):
num_encoder_layers=args.enc_layers, num_encoder_layers=args.enc_layers,
num_decoder_layers=args.dec_layers, num_decoder_layers=args.dec_layers,
normalize_before=args.pre_norm, normalize_before=args.pre_norm,
return_intermediate_dec=True,
) )

View File

@ -29,7 +29,7 @@ policy:
hidden_dim: 512 hidden_dim: 512
dim_feedforward: 3200 dim_feedforward: 3200
enc_layers: 4 enc_layers: 4
dec_layers: 7 dec_layers: 1
nheads: 8 nheads: 8
#camera_names: [top, front_close, left_pillar, right_pillar] #camera_names: [top, front_close, left_pillar, right_pillar]
camera_names: [top] camera_names: [top]

View File

@ -0,0 +1,64 @@
import torch
from lerobot.common.policies.factory import make_policy
from lerobot.common.utils import init_hydra_config
cfg = init_hydra_config(
"/home/alexander/Projects/lerobot/outputs/train/act_aloha_sim_transfer_cube_human/.hydra/config.yaml"
)
policy = make_policy(cfg)
state_dict = torch.load("/home/alexander/Projects/act/outputs/sim_transfer_cube_human_vae/policy_last.ckpt")
# Replace keys based on what they start with.
start_replacements = [
("model.query_embed.weight", "model.pos_embed.weight"),
("model.pos_table", "model.vae_encoder_pos_enc"),
("model.pos_embed.weight", "model.decoder_pos_embed.weight"),
("model.encoder.", "model.vae_encoder."),
("model.encoder_action_proj.", "model.vae_encoder_action_input_proj."),
("model.encoder_joint_proj.", "model.vae_encoder_robot_state_input_proj."),
("model.latent_proj.", "model.vae_encoder_latent_output_proj."),
("model.latent_proj.", "model.vae_encoder_latent_output_proj."),
("model.input_proj.", "model.encoder_img_feat_input_proj."),
("model.input_proj_robot_state", "model.encoder_robot_state_input_proj"),
("model.latent_out_proj.", "model.encoder_latent_input_proj."),
]
for to_replace, replace_with in start_replacements:
for k in list(state_dict.keys()):
if k.startswith(to_replace):
k_ = replace_with + k.removeprefix(to_replace)
state_dict[k_] = state_dict[k]
del state_dict[k]
# Remove keys based on what they start with.
start_removals = [
# There is a bug that means the pretrained model doesn't even use the final decoder layers.
*[f"model.transformer.decoder.layers.{i}" for i in range(1, 7)],
"model.is_pad_head.",
]
for to_remove in start_removals:
for k in list(state_dict.keys()):
if k.startswith(to_remove):
del state_dict[k]
missing_keys, unexpected_keys = policy.load_state_dict(state_dict, strict=False)
if len(missing_keys) != 0:
print("MISSING KEYS")
print(missing_keys)
if len(unexpected_keys) != 0:
print("UNEXPECTED KEYS")
print(unexpected_keys)
# if len(missing_keys) != 0 or len(unexpected_keys) != 0:
# print("Failed due to mismatch in state dicts.")
# exit()
policy.save("/tmp/weights.pth")