if (ret)
                return ret;
 
+       netdev_queue_set_dql_min_limit(netdev_get_tx_queue(cdev->net, 0),
+                                      cdev->tx_max_coalesced_frames);
+
        cdev->can.state = CAN_STATE_ERROR_ACTIVE;
 
        m_can_enable_all_interrupts(cdev);
                 */
                can_put_echo_skb(skb, dev, putidx, frame_len);
 
-               /* Enable TX FIFO element to start transfer  */
-               m_can_write(cdev, M_CAN_TXBAR, (1 << putidx));
+               if (cdev->is_peripheral) {
+                       /* Delay enabling TX FIFO element */
+                       cdev->tx_peripheral_submit |= BIT(putidx);
+               } else {
+                       /* Enable TX FIFO element to start transfer  */
+                       m_can_write(cdev, M_CAN_TXBAR, BIT(putidx));
+               }
                cdev->tx_fifo_putidx = (++cdev->tx_fifo_putidx >= cdev->can.echo_skb_max ?
                                        0 : cdev->tx_fifo_putidx);
        }
        return NETDEV_TX_BUSY;
 }
 
+static void m_can_tx_submit(struct m_can_classdev *cdev)
+{
+       if (cdev->version == 30)
+               return;
+       if (!cdev->is_peripheral)
+               return;
+
+       m_can_write(cdev, M_CAN_TXBAR, cdev->tx_peripheral_submit);
+       cdev->tx_peripheral_submit = 0;
+}
+
 static void m_can_tx_work_queue(struct work_struct *ws)
 {
        struct m_can_tx_op *op = container_of(ws, struct m_can_tx_op, work);
 
        op->skb = NULL;
        m_can_tx_handler(cdev, skb);
+       if (op->submit)
+               m_can_tx_submit(cdev);
 }
 
-static void m_can_tx_queue_skb(struct m_can_classdev *cdev, struct sk_buff *skb)
+static void m_can_tx_queue_skb(struct m_can_classdev *cdev, struct sk_buff *skb,
+                              bool submit)
 {
        cdev->tx_ops[cdev->next_tx_op].skb = skb;
+       cdev->tx_ops[cdev->next_tx_op].submit = submit;
        queue_work(cdev->tx_wq, &cdev->tx_ops[cdev->next_tx_op].work);
 
        ++cdev->next_tx_op;
 static netdev_tx_t m_can_start_peripheral_xmit(struct m_can_classdev *cdev,
                                               struct sk_buff *skb)
 {
-       m_can_tx_queue_skb(cdev, skb);
+       bool submit;
+
+       ++cdev->nr_txs_without_submit;
+       if (cdev->nr_txs_without_submit >= cdev->tx_max_coalesced_frames ||
+           !netdev_xmit_more()) {
+               cdev->nr_txs_without_submit = 0;
+               submit = true;
+       } else {
+               submit = false;
+       }
+       m_can_tx_queue_skb(cdev, skb, submit);
 
        return NETDEV_TX_OK;
 }
 
        ec->rx_max_coalesced_frames_irq = cdev->rx_max_coalesced_frames_irq;
        ec->rx_coalesce_usecs_irq = cdev->rx_coalesce_usecs_irq;
+       ec->tx_max_coalesced_frames = cdev->tx_max_coalesced_frames;
        ec->tx_max_coalesced_frames_irq = cdev->tx_max_coalesced_frames_irq;
        ec->tx_coalesce_usecs_irq = cdev->tx_coalesce_usecs_irq;
 
                netdev_err(dev, "tx-frames-irq and tx-usecs-irq can only be set together\n");
                return -EINVAL;
        }
+       if (ec->tx_max_coalesced_frames > cdev->mcfg[MRAM_TXE].num) {
+               netdev_err(dev, "tx-frames %u greater than the TX event FIFO %u\n",
+                          ec->tx_max_coalesced_frames,
+                          cdev->mcfg[MRAM_TXE].num);
+               return -EINVAL;
+       }
+       if (ec->tx_max_coalesced_frames > cdev->mcfg[MRAM_TXB].num) {
+               netdev_err(dev, "tx-frames %u greater than the TX FIFO %u\n",
+                          ec->tx_max_coalesced_frames,
+                          cdev->mcfg[MRAM_TXB].num);
+               return -EINVAL;
+       }
        if (ec->rx_coalesce_usecs_irq != 0 && ec->tx_coalesce_usecs_irq != 0 &&
            ec->rx_coalesce_usecs_irq != ec->tx_coalesce_usecs_irq) {
                netdev_err(dev, "rx-usecs-irq %u needs to be equal to tx-usecs-irq %u if both are enabled\n",
 
        cdev->rx_max_coalesced_frames_irq = ec->rx_max_coalesced_frames_irq;
        cdev->rx_coalesce_usecs_irq = ec->rx_coalesce_usecs_irq;
+       cdev->tx_max_coalesced_frames = ec->tx_max_coalesced_frames;
        cdev->tx_max_coalesced_frames_irq = ec->tx_max_coalesced_frames_irq;
        cdev->tx_coalesce_usecs_irq = ec->tx_coalesce_usecs_irq;
 
        .supported_coalesce_params = ETHTOOL_COALESCE_RX_USECS_IRQ |
                ETHTOOL_COALESCE_RX_MAX_FRAMES_IRQ |
                ETHTOOL_COALESCE_TX_USECS_IRQ |
+               ETHTOOL_COALESCE_TX_MAX_FRAMES |
                ETHTOOL_COALESCE_TX_MAX_FRAMES_IRQ,
        .get_ts_info = ethtool_op_get_ts_info,
        .get_coalesce = m_can_get_coalesce,