*idsn = be64_to_cpu(*((__be64 *)&mptcp_hashed_key[6]));
 }
 
-void mptcp_crypto_hmac_sha(u64 key1, u64 key2, u32 nonce1, u32 nonce2,
-                          void *hmac)
+void mptcp_crypto_hmac_sha(u64 key1, u64 key2, u8 *msg, int len, void *hmac)
 {
        u8 input[SHA256_BLOCK_SIZE + SHA256_DIGEST_SIZE];
        __be32 mptcp_hashed_key[SHA256_DIGEST_WORDS];
        u8 key2be[8];
        int i;
 
+       if (WARN_ON_ONCE(len > SHA256_DIGEST_SIZE))
+               len = SHA256_DIGEST_SIZE;
+
        put_unaligned_be64(key1, key1be);
        put_unaligned_be64(key2, key2be);
 
        for (i = 0; i < 8; i++)
                input[i + 8] ^= key2be[i];
 
-       put_unaligned_be32(nonce1, &input[SHA256_BLOCK_SIZE]);
-       put_unaligned_be32(nonce2, &input[SHA256_BLOCK_SIZE + 4]);
+       memcpy(&input[SHA256_BLOCK_SIZE], msg, len);
 
        sha256_init(&state);
-       sha256_update(&state, input, SHA256_BLOCK_SIZE + 8);
+       sha256_update(&state, input, SHA256_BLOCK_SIZE + len);
 
        /* emit sha256(K1 || msg) on the second input block, so we can
         * reuse 'input' for the last hashing
        char hmac[20], hmac_hex[41];
        u32 nonce1, nonce2;
        u64 key1, key2;
+       u8 msg[8];
        int i, j;
 
        for (i = 0; i < ARRAY_SIZE(tests); ++i) {
                nonce1 = be32_to_cpu(*((__be32 *)&tests[i].msg[0]));
                nonce2 = be32_to_cpu(*((__be32 *)&tests[i].msg[4]));
 
-               mptcp_crypto_hmac_sha(key1, key2, nonce1, nonce2, hmac);
+               put_unaligned_be32(nonce1, &msg[0]);
+               put_unaligned_be32(nonce2, &msg[4]);
+
+               mptcp_crypto_hmac_sha(key1, key2, msg, 8, hmac);
                for (j = 0; j < 20; ++j)
                        sprintf(&hmac_hex[j << 1], "%02x", hmac[j] & 0xff);
                hmac_hex[40] = 0;
 
 
                break;
 
+       case MPTCPOPT_ADD_ADDR:
+               mp_opt->echo = (*ptr++) & MPTCP_ADDR_ECHO;
+               if (!mp_opt->echo) {
+                       if (opsize == TCPOLEN_MPTCP_ADD_ADDR ||
+                           opsize == TCPOLEN_MPTCP_ADD_ADDR_PORT)
+                               mp_opt->family = MPTCP_ADDR_IPVERSION_4;
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+                       else if (opsize == TCPOLEN_MPTCP_ADD_ADDR6 ||
+                                opsize == TCPOLEN_MPTCP_ADD_ADDR6_PORT)
+                               mp_opt->family = MPTCP_ADDR_IPVERSION_6;
+#endif
+                       else
+                               break;
+               } else {
+                       if (opsize == TCPOLEN_MPTCP_ADD_ADDR_BASE ||
+                           opsize == TCPOLEN_MPTCP_ADD_ADDR_BASE_PORT)
+                               mp_opt->family = MPTCP_ADDR_IPVERSION_4;
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+                       else if (opsize == TCPOLEN_MPTCP_ADD_ADDR6_BASE ||
+                                opsize == TCPOLEN_MPTCP_ADD_ADDR6_BASE_PORT)
+                               mp_opt->family = MPTCP_ADDR_IPVERSION_6;
+#endif
+                       else
+                               break;
+               }
+
+               mp_opt->add_addr = 1;
+               mp_opt->port = 0;
+               mp_opt->addr_id = *ptr++;
+               pr_debug("ADD_ADDR: id=%d", mp_opt->addr_id);
+               if (mp_opt->family == MPTCP_ADDR_IPVERSION_4) {
+                       memcpy((u8 *)&mp_opt->addr.s_addr, (u8 *)ptr, 4);
+                       ptr += 4;
+                       if (opsize == TCPOLEN_MPTCP_ADD_ADDR_PORT ||
+                           opsize == TCPOLEN_MPTCP_ADD_ADDR_BASE_PORT) {
+                               mp_opt->port = get_unaligned_be16(ptr);
+                               ptr += 2;
+                       }
+               }
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+               else {
+                       memcpy(mp_opt->addr6.s6_addr, (u8 *)ptr, 16);
+                       ptr += 16;
+                       if (opsize == TCPOLEN_MPTCP_ADD_ADDR6_PORT ||
+                           opsize == TCPOLEN_MPTCP_ADD_ADDR6_BASE_PORT) {
+                               mp_opt->port = get_unaligned_be16(ptr);
+                               ptr += 2;
+                       }
+               }
+#endif
+               if (!mp_opt->echo) {
+                       mp_opt->ahmac = get_unaligned_be64(ptr);
+                       ptr += 8;
+               }
+               break;
+
+       case MPTCPOPT_RM_ADDR:
+               if (opsize != TCPOLEN_MPTCP_RM_ADDR_BASE)
+                       break;
+
+               mp_opt->rm_addr = 1;
+               mp_opt->rm_id = *ptr++;
+               pr_debug("RM_ADDR: id=%d", mp_opt->rm_id);
+               break;
+
        default:
                break;
        }
        return true;
 }
 
+static u64 add_addr_generate_hmac(u64 key1, u64 key2, u8 addr_id,
+                                 struct in_addr *addr)
+{
+       u8 hmac[MPTCP_ADDR_HMAC_LEN];
+       u8 msg[7];
+
+       msg[0] = addr_id;
+       memcpy(&msg[1], &addr->s_addr, 4);
+       msg[5] = 0;
+       msg[6] = 0;
+
+       mptcp_crypto_hmac_sha(key1, key2, msg, 7, hmac);
+
+       return get_unaligned_be64(hmac);
+}
+
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+static u64 add_addr6_generate_hmac(u64 key1, u64 key2, u8 addr_id,
+                                  struct in6_addr *addr)
+{
+       u8 hmac[MPTCP_ADDR_HMAC_LEN];
+       u8 msg[19];
+
+       msg[0] = addr_id;
+       memcpy(&msg[1], &addr->s6_addr, 16);
+       msg[17] = 0;
+       msg[18] = 0;
+
+       mptcp_crypto_hmac_sha(key1, key2, msg, 19, hmac);
+
+       return get_unaligned_be64(hmac);
+}
+#endif
+
+static bool mptcp_established_options_addr(struct sock *sk,
+                                          unsigned int *size,
+                                          unsigned int remaining,
+                                          struct mptcp_out_options *opts)
+{
+       struct mptcp_subflow_context *subflow = mptcp_subflow_ctx(sk);
+       struct mptcp_sock *msk = mptcp_sk(subflow->conn);
+       struct sockaddr_storage saddr;
+       u8 id;
+
+       id = 0;
+       memset(&saddr, 0, sizeof(saddr));
+
+       if (saddr.ss_family == AF_INET) {
+               if (remaining < TCPOLEN_MPTCP_ADD_ADDR)
+                       return false;
+               opts->suboptions |= OPTION_MPTCP_ADD_ADDR;
+               opts->addr_id = id;
+               opts->addr = ((struct sockaddr_in *)&saddr)->sin_addr;
+               opts->ahmac = add_addr_generate_hmac(msk->local_key,
+                                                    msk->remote_key,
+                                                    opts->addr_id,
+                                                    &opts->addr);
+               *size = TCPOLEN_MPTCP_ADD_ADDR;
+       }
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+       else if (saddr.ss_family == AF_INET6) {
+               if (remaining < TCPOLEN_MPTCP_ADD_ADDR6)
+                       return false;
+               opts->suboptions |= OPTION_MPTCP_ADD_ADDR6;
+               opts->addr_id = id;
+               opts->ahmac = add_addr6_generate_hmac(msk->local_key,
+                                                     msk->remote_key,
+                                                     opts->addr_id,
+                                                     &opts->addr6);
+               opts->addr6 = ((struct sockaddr_in6 *)&saddr)->sin6_addr;
+               *size = TCPOLEN_MPTCP_ADD_ADDR6;
+       }
+#endif
+       pr_debug("addr_id=%d, ahmac=%llu", opts->addr_id, opts->ahmac);
+
+       return true;
+}
+
 bool mptcp_established_options(struct sock *sk, struct sk_buff *skb,
                               unsigned int *size, unsigned int remaining,
                               struct mptcp_out_options *opts)
        unsigned int opt_size = 0;
        bool ret = false;
 
+       opts->suboptions = 0;
+
        if (mptcp_established_options_mp(sk, skb, &opt_size, remaining, opts))
                ret = true;
        else if (mptcp_established_options_dss(sk, skb, &opt_size, remaining,
 
        *size += opt_size;
        remaining -= opt_size;
+       if (mptcp_established_options_addr(sk, &opt_size, remaining, opts)) {
+               *size += opt_size;
+               remaining -= opt_size;
+               ret = true;
+       }
 
        return ret;
 }
                else
                        len = TCPOLEN_MPTCP_MPC_ACK;
 
-               *ptr++ = htonl((TCPOPT_MPTCP << 24) | (len << 16) |
-                              (MPTCPOPT_MP_CAPABLE << 12) |
-                              (MPTCP_SUPPORTED_VERSION << 8) |
-                              MPTCP_CAP_HMAC_SHA256);
+               *ptr++ = mptcp_option(MPTCPOPT_MP_CAPABLE, len,
+                                     MPTCP_SUPPORTED_VERSION,
+                                     MPTCP_CAP_HMAC_SHA256);
 
                if (!((OPTION_MPTCP_MPC_SYNACK | OPTION_MPTCP_MPC_ACK) &
                    opts->suboptions))
        }
 
 mp_capable_done:
+       if (OPTION_MPTCP_ADD_ADDR & opts->suboptions) {
+               if (opts->ahmac)
+                       *ptr++ = mptcp_option(MPTCPOPT_ADD_ADDR,
+                                             TCPOLEN_MPTCP_ADD_ADDR, 0,
+                                             opts->addr_id);
+               else
+                       *ptr++ = mptcp_option(MPTCPOPT_ADD_ADDR,
+                                             TCPOLEN_MPTCP_ADD_ADDR_BASE,
+                                             MPTCP_ADDR_ECHO,
+                                             opts->addr_id);
+               memcpy((u8 *)ptr, (u8 *)&opts->addr.s_addr, 4);
+               ptr += 1;
+               if (opts->ahmac) {
+                       put_unaligned_be64(opts->ahmac, ptr);
+                       ptr += 2;
+               }
+       }
+
+#if IS_ENABLED(CONFIG_MPTCP_IPV6)
+       if (OPTION_MPTCP_ADD_ADDR6 & opts->suboptions) {
+               if (opts->ahmac)
+                       *ptr++ = mptcp_option(MPTCPOPT_ADD_ADDR,
+                                             TCPOLEN_MPTCP_ADD_ADDR6, 0,
+                                             opts->addr_id);
+               else
+                       *ptr++ = mptcp_option(MPTCPOPT_ADD_ADDR,
+                                             TCPOLEN_MPTCP_ADD_ADDR6_BASE,
+                                             MPTCP_ADDR_ECHO,
+                                             opts->addr_id);
+               memcpy((u8 *)ptr, opts->addr6.s6_addr, 16);
+               ptr += 4;
+               if (opts->ahmac) {
+                       put_unaligned_be64(opts->ahmac, ptr);
+                       ptr += 2;
+               }
+       }
+#endif
+
+       if (OPTION_MPTCP_RM_ADDR & opts->suboptions) {
+               *ptr++ = mptcp_option(MPTCPOPT_RM_ADDR,
+                                     TCPOLEN_MPTCP_RM_ADDR_BASE,
+                                     0, opts->rm_id);
+       }
+
        if (opts->ext_copy.use_ack || opts->ext_copy.use_map) {
                struct mptcp_ext *mpext = &opts->ext_copy;
                u8 len = TCPOLEN_MPTCP_DSS_BASE;
                                flags |= MPTCP_DSS_DATA_FIN;
                }
 
-               *ptr++ = htonl((TCPOPT_MPTCP << 24) |
-                              (len  << 16) |
-                              (MPTCPOPT_DSS << 12) |
-                              (flags));
+               *ptr++ = mptcp_option(MPTCPOPT_DSS, len, 0, flags);
 
                if (mpext->use_ack) {
                        put_unaligned_be64(mpext->data_ack, ptr);
 
 #define OPTION_MPTCP_MPC_SYN   BIT(0)
 #define OPTION_MPTCP_MPC_SYNACK        BIT(1)
 #define OPTION_MPTCP_MPC_ACK   BIT(2)
+#define OPTION_MPTCP_ADD_ADDR  BIT(6)
+#define OPTION_MPTCP_ADD_ADDR6 BIT(7)
+#define OPTION_MPTCP_RM_ADDR   BIT(8)
 
 /* MPTCP option subtypes */
 #define MPTCPOPT_MP_CAPABLE    0
 #define TCPOLEN_MPTCP_DSS_MAP32                10
 #define TCPOLEN_MPTCP_DSS_MAP64                14
 #define TCPOLEN_MPTCP_DSS_CHECKSUM     2
+#define TCPOLEN_MPTCP_ADD_ADDR         16
+#define TCPOLEN_MPTCP_ADD_ADDR_PORT    18
+#define TCPOLEN_MPTCP_ADD_ADDR_BASE    8
+#define TCPOLEN_MPTCP_ADD_ADDR_BASE_PORT       10
+#define TCPOLEN_MPTCP_ADD_ADDR6                28
+#define TCPOLEN_MPTCP_ADD_ADDR6_PORT   30
+#define TCPOLEN_MPTCP_ADD_ADDR6_BASE   20
+#define TCPOLEN_MPTCP_ADD_ADDR6_BASE_PORT      22
+#define TCPOLEN_MPTCP_PORT_LEN         2
+#define TCPOLEN_MPTCP_RM_ADDR_BASE     4
 
 /* MPTCP MP_CAPABLE flags */
 #define MPTCP_VERSION_MASK     (0x0F)
 #define MPTCP_DSS_HAS_ACK      BIT(0)
 #define MPTCP_DSS_FLAG_MASK    (0x1F)
 
+/* MPTCP ADD_ADDR flags */
+#define MPTCP_ADDR_ECHO                BIT(0)
+#define MPTCP_ADDR_HMAC_LEN    20
+#define MPTCP_ADDR_IPVERSION_4 4
+#define MPTCP_ADDR_IPVERSION_6 6
+
 /* MPTCP socket flags */
 #define MPTCP_DATA_READY       0
 #define MPTCP_SEND_SPACE       1
 
+static inline __be32 mptcp_option(u8 subopt, u8 len, u8 nib, u8 field)
+{
+       return htonl((TCPOPT_MPTCP << 24) | (len << 16) | (subopt << 12) |
+                    ((nib & 0xF) << 8) | field);
+}
+
 /* MPTCP connection sock */
 struct mptcp_sock {
        /* inet_connection_sock must be the first member */
        mptcp_crypto_key_sha(*key, token, idsn);
 }
 
-void mptcp_crypto_hmac_sha(u64 key1, u64 key2, u32 nonce1, u32 nonce2,
-                          void *hash_out);
+void mptcp_crypto_hmac_sha(u64 key1, u64 key2, u8 *msg, int len, void *hmac);
 
 static inline struct mptcp_ext *mptcp_get_ext(struct sk_buff *skb)
 {