return true;
 }
 
+static bool mptcp_established_options_fastclose(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);
+
+       if (likely(!subflow->send_fastclose))
+               return false;
+
+       if (remaining < TCPOLEN_MPTCP_FASTCLOSE)
+               return false;
+
+       *size = TCPOLEN_MPTCP_FASTCLOSE;
+       opts->suboptions |= OPTION_MPTCP_FASTCLOSE;
+       opts->rcvr_key = msk->remote_key;
+
+       pr_debug("FASTCLOSE key=%llu", opts->rcvr_key);
+       return true;
+}
+
 static bool mptcp_established_options_mp_fail(struct sock *sk,
                                              unsigned int *size,
                                              unsigned int remaining,
                return false;
 
        if (unlikely(skb && TCP_SKB_CB(skb)->tcp_flags & TCPHDR_RST)) {
-               if (mptcp_established_options_mp_fail(sk, &opt_size, remaining, opts)) {
+               if (mptcp_established_options_fastclose(sk, &opt_size, remaining, opts) ||
+                   mptcp_established_options_mp_fail(sk, &opt_size, remaining, opts)) {
                        *size += opt_size;
                        remaining -= opt_size;
                }
+               /* MP_RST can be used with MP_FASTCLOSE and MP_FAIL if there is room */
                if (mptcp_established_options_rst(sk, skb, &opt_size, remaining, opts)) {
                        *size += opt_size;
                        remaining -= opt_size;
                ptr += 2;
        }
 
-       /* RST is mutually exclusive with everything else */
-       if (unlikely(OPTION_MPTCP_RST & opts->suboptions)) {
-               *ptr++ = mptcp_option(MPTCPOPT_RST,
-                                     TCPOLEN_MPTCP_RST,
-                                     opts->reset_transient,
-                                     opts->reset_reason);
-               return;
-       }
-
-       /* DSS, MPC, MPJ and ADD_ADDR are mutually exclusive, see
-        * mptcp_established_options*()
+       /* DSS, MPC, MPJ, ADD_ADDR, FASTCLOSE and RST are mutually exclusive,
+        * see mptcp_established_options*()
         */
        if (likely(OPTION_MPTCP_DSS & opts->suboptions)) {
                struct mptcp_ext *mpext = &opts->ext_copy;
                                ptr += 1;
                        }
                }
+       } else if (unlikely(OPTION_MPTCP_FASTCLOSE & opts->suboptions)) {
+               /* FASTCLOSE is mutually exclusive with others except RST */
+               *ptr++ = mptcp_option(MPTCPOPT_MP_FASTCLOSE,
+                                     TCPOLEN_MPTCP_FASTCLOSE,
+                                     0, 0);
+               put_unaligned_be64(opts->rcvr_key, ptr);
+               ptr += 2;
+
+               if (OPTION_MPTCP_RST & opts->suboptions)
+                       goto mp_rst;
+               return;
+       } else if (unlikely(OPTION_MPTCP_RST & opts->suboptions)) {
+mp_rst:
+               *ptr++ = mptcp_option(MPTCPOPT_RST,
+                                     TCPOLEN_MPTCP_RST,
+                                     opts->reset_transient,
+                                     opts->reset_reason);
+               return;
        }
 
        if (OPTION_MPTCP_PRIO & opts->suboptions) {