const struct sctp_association *asoc,
                                        const struct sctp_chunk *chunk);
 struct sctp_chunk *sctp_make_heartbeat(const struct sctp_association *asoc,
-                                      const struct sctp_transport *transport);
+                                      const struct sctp_transport *transport,
+                                      __u32 probe_size);
 struct sctp_chunk *sctp_make_heartbeat_ack(const struct sctp_association *asoc,
                                           const struct sctp_chunk *chunk,
                                           const void *payload,
 
        union sctp_addr daddr;
        unsigned long sent_at;
        __u64 hb_nonce;
+       __u32 probe_size;
 };
 
 int sctp_stream_init(struct sctp_stream *stream, __u16 outcnt, __u16 incnt,
                data_accepted:1,        /* At least 1 chunk accepted */
                auth:1,                 /* IN: was auth'ed | OUT: needs auth */
                has_asconf:1,           /* IN: have seen an asconf before */
+               pmtu_probe:1,           /* Used by PLPMTUD, can be set in s HB chunk */
                tsn_missing_report:2,   /* Data chunk missing counter. */
                fast_retransmit:2;      /* Is this chunk fast retransmitted? */
 };
 
        return retval;
 }
 
+/* Try to bundle a pad chunk into a packet with a heartbeat chunk for PLPMTUTD probe */
+static enum sctp_xmit sctp_packet_bundle_pad(struct sctp_packet *pkt, struct sctp_chunk *chunk)
+{
+       struct sctp_transport *t = pkt->transport;
+       struct sctp_chunk *pad;
+       int overhead = 0;
+
+       if (!chunk->pmtu_probe)
+               return SCTP_XMIT_OK;
+
+       /* calculate the Padding Data size for the pad chunk */
+       overhead += sizeof(struct sctphdr) + sizeof(struct sctp_chunkhdr);
+       overhead += sizeof(struct sctp_sender_hb_info) + sizeof(struct sctp_pad_chunk);
+       pad = sctp_make_pad(t->asoc, t->pl.probe_size - overhead);
+       if (!pad)
+               return SCTP_XMIT_DELAY;
+
+       list_add_tail(&pad->list, &pkt->chunk_list);
+       pkt->size += SCTP_PAD4(ntohs(pad->chunk_hdr->length));
+       chunk->transport = t;
+
+       return SCTP_XMIT_OK;
+}
+
 /* Try to bundle an auth chunk into the packet. */
 static enum sctp_xmit sctp_packet_bundle_auth(struct sctp_packet *pkt,
                                              struct sctp_chunk *chunk)
                goto finish;
 
        retval = __sctp_packet_append_chunk(packet, chunk);
+       if (retval != SCTP_XMIT_OK)
+               goto finish;
+
+       retval = sctp_packet_bundle_pad(packet, chunk);
 
 finish:
        return retval;
        sk = chunk->skb->sk;
 
        /* check gso */
-       if (packet->size > tp->pathmtu && !packet->ipfragok) {
+       if (packet->size > tp->pathmtu && !packet->ipfragok && !chunk->pmtu_probe) {
                if (!sk_can_gso(sk)) {
                        pr_err_once("Trying to GSO but underlying device doesn't support it.");
                        goto out;
 
 
        sctp_packet_init(&singleton, transport, sport, dport);
        sctp_packet_config(&singleton, vtag, 0);
-       sctp_packet_append_chunk(&singleton, chunk);
+       if (sctp_packet_append_chunk(&singleton, chunk) != SCTP_XMIT_OK) {
+               list_del_init(&chunk->list);
+               sctp_chunk_free(chunk);
+               return -ENOMEM;
+       }
        return sctp_packet_transmit(&singleton, gfp);
 }
 
                        one_packet = 1;
                        fallthrough;
 
-               case SCTP_CID_SACK:
                case SCTP_CID_HEARTBEAT:
+                       if (chunk->pmtu_probe) {
+                               sctp_packet_singleton(ctx->transport, chunk, ctx->gfp);
+                               break;
+                       }
+                       fallthrough;
+               case SCTP_CID_SACK:
                case SCTP_CID_SHUTDOWN:
                case SCTP_CID_ECN_ECNE:
                case SCTP_CID_ASCONF:
 
 
 /* Make a HEARTBEAT chunk.  */
 struct sctp_chunk *sctp_make_heartbeat(const struct sctp_association *asoc,
-                                      const struct sctp_transport *transport)
+                                      const struct sctp_transport *transport,
+                                      __u32 probe_size)
 {
        struct sctp_sender_hb_info hbinfo;
        struct sctp_chunk *retval;
        hbinfo.daddr = transport->ipaddr;
        hbinfo.sent_at = jiffies;
        hbinfo.hb_nonce = transport->hb_nonce;
+       hbinfo.probe_size = probe_size;
 
        /* Cast away the 'const', as this is just telling the chunk
         * what transport it belongs to.
        retval->transport = (struct sctp_transport *) transport;
        retval->subh.hbs_hdr = sctp_addto_chunk(retval, sizeof(hbinfo),
                                                &hbinfo);
+       retval->pmtu_probe = !!probe_size;
 
 nodata:
        return retval;
 
        struct sctp_chunk *reply;
 
        /* Send a heartbeat to our peer.  */
-       reply = sctp_make_heartbeat(asoc, transport);
+       reply = sctp_make_heartbeat(asoc, transport, 0);
        if (!reply)
                return SCTP_DISPOSITION_NOMEM;
 
                                         struct sctp_cmd_seq *commands)
 {
        struct sctp_transport *transport = (struct sctp_transport *)arg;
+       struct sctp_chunk *reply;
+
+       if (!sctp_transport_pl_enabled(transport))
+               return SCTP_DISPOSITION_CONSUME;
 
-       /* The actual handling will be performed here in a later patch. */
+       reply = sctp_make_heartbeat(asoc, transport, transport->pl.probe_size);
+       if (!reply)
+               return SCTP_DISPOSITION_NOMEM;
+       sctp_add_cmd_sf(commands, SCTP_CMD_REPLY, SCTP_CHUNK(reply));
        sctp_add_cmd_sf(commands, SCTP_CMD_PROBE_TIMER_UPDATE,
                        SCTP_TRANSPORT(transport));
 
        if (hbinfo->hb_nonce != link->hb_nonce)
                return SCTP_DISPOSITION_DISCARD;
 
+       if (hbinfo->probe_size) {
+               if (hbinfo->probe_size != link->pl.probe_size ||
+                   !sctp_transport_pl_enabled(link))
+                       return SCTP_DISPOSITION_DISCARD;
+
+               /* The actual handling will be performed here in a later patch. */
+               return SCTP_DISPOSITION_CONSUME;
+       }
+
        max_interval = link->hbinterval + link->rto;
 
        /* Check if the timestamp looks valid.  */