diff --git a/drivers/Makefile.dep b/drivers/Makefile.dep
index 053d54a3c7ec1e066bc565e760d7e038d1cf1f98..115a0c1007eeb31b8185acbabcbc1a44549cdaa0 100644
--- a/drivers/Makefile.dep
+++ b/drivers/Makefile.dep
@@ -246,6 +246,7 @@ ifneq (,$(filter si70%,$(USEMODULE)))
 endif
 
 ifneq (,$(filter slipdev,$(USEMODULE)))
+  USEMODULE += tsrb
   FEATURES_REQUIRED += periph_uart
 endif
 
diff --git a/drivers/include/slipdev.h b/drivers/include/slipdev.h
index 3d10bf7f9b2f20a57bdeb2d853adc541bb9642a7..9ad916e258cbd670565c2e0f387cd5ff59e17209 100644
--- a/drivers/include/slipdev.h
+++ b/drivers/include/slipdev.h
@@ -26,7 +26,7 @@
 #include "cib.h"
 #include "net/netdev.h"
 #include "periph/uart.h"
-#include "ringbuffer.h"
+#include "tsrb.h"
 
 #ifdef __cplusplus
 extern "C" {
@@ -36,20 +36,12 @@ extern "C" {
  * @brief   UART buffer size used for TX and RX buffers
  *
  * Reduce this value if your expected traffic does not include full IPv6 MTU
- * sized packets
- */
-#ifndef SLIPDEV_BUFSIZE
-#define SLIPDEV_BUFSIZE (1500U)
-#endif
-
-/**
- * @brief   Packet FIFO size
+ * sized packets.
  *
- * @note    For GNRC it is recommended to have it the same size as the link-layer
- *          thread's message queue, but it MUST be of power of 2
+ * @pre Needs to be power of two and `<= INT_MAX`
  */
-#ifndef SLIPDEV_PKTFIFO_SIZE
-#define SLIPDEV_PKTFIFO_SIZE    (8U)
+#ifndef SLIPDEV_BUFSIZE
+#define SLIPDEV_BUFSIZE (2048U)
 #endif
 
 /**
@@ -68,13 +60,8 @@ typedef struct {
 typedef struct {
     netdev_t netdev;                        /**< parent class */
     slipdev_params_t config;                /**< configuration parameters */
-    ringbuffer_t inbuf;                     /**< RX buffer */
+    tsrb_t inbuf;                           /**< RX buffer */
     char rxmem[SLIPDEV_BUFSIZE];            /**< memory used by RX buffer */
-    uint16_t pktfifo[SLIPDEV_PKTFIFO_SIZE]; /**< FIFO of sizes of fully received
-                                             *   packets */
-    cib_t pktfifo_idx;                      /**< CIB for slipdev_t::pktfifo */
-    uint16_t inbytes;                       /**< the number of bytes received of
-                                             *   a currently incoming packet */
     uint16_t inesc;                         /**< device previously received an escape
                                              *   byte */
 } slipdev_t;
diff --git a/drivers/slipdev/slipdev.c b/drivers/slipdev/slipdev.c
index 80ee16e7eabbafa5c157bc72d80b63c6c3ded0ea..3670776d8ee3687b9327b62dbdde5a517d6fa684 100644
--- a/drivers/slipdev/slipdev.c
+++ b/drivers/slipdev/slipdev.c
@@ -48,53 +48,17 @@ void slipdev_setup(slipdev_t *dev, const slipdev_params_t *params)
 {
     /* set device descriptor fields */
     memcpy(&dev->config, params, sizeof(dev->config));
-    dev->inbytes = 0U;
     dev->inesc = 0U;
     dev->netdev.driver = &slip_driver;
 }
 
-static inline void _add_byte_to_inbuf(slipdev_t *dev, uint8_t byte)
-{
-    if (ringbuffer_add_one(&dev->inbuf, byte) < 0) {
-        dev->inbytes++;
-    }
-}
-
-static void _slip_rx_cb(void *arg, uint8_t data)
+static void _slip_rx_cb(void *arg, uint8_t byte)
 {
     slipdev_t *dev = arg;
 
-    if ((data == SLIP_END) && (dev->netdev.event_callback != NULL)) {
-        int idx = cib_put(&dev->pktfifo_idx);
-        if (idx >= 0) {
-            dev->netdev.event_callback((netdev_t *)dev, NETDEV_EVENT_ISR);
-            dev->pktfifo[idx] = dev->inbytes;
-        }
-        else {
-            /* can't handover packet => dropping it */
-            ringbuffer_remove(&dev->inbuf, dev->inbytes);
-        }
-        dev->inbytes = 0;
-    }
-    else if (dev->inesc) {
-        dev->inesc = 0U;
-        uint8_t actual = (data == SLIP_END_ESC) ? SLIP_END :
-                         ((data == SLIP_ESC_ESC) ? SLIP_ESC : 0);
-
-        switch (data) {
-            case SLIP_END_ESC:
-            case SLIP_ESC_ESC:
-                _add_byte_to_inbuf(dev, actual);
-                break;
-            default:
-                break;
-        }
-    }
-    else if (data == SLIP_ESC) {
-        dev->inesc = 1U;
-    }
-    else {
-        _add_byte_to_inbuf(dev, data);
+    tsrb_add_one(&dev->inbuf, byte);
+    if ((byte == SLIP_END) && (dev->netdev.event_callback != NULL)) {
+        dev->netdev.event_callback((netdev_t *)dev, NETDEV_EVENT_ISR);
     }
 }
 
@@ -105,8 +69,7 @@ static int _init(netdev_t *netdev)
     DEBUG("slipdev: initializing device %p on UART %i with baudrate %" PRIu32 "\n",
           (void *)dev, dev->config.uart, dev->config.baudrate);
     /* initialize buffers */
-    ringbuffer_init(&dev->inbuf, dev->rxmem, sizeof(dev->rxmem));
-    cib_init(&dev->pktfifo_idx, SLIPDEV_PKTFIFO_SIZE);
+    tsrb_init(&dev->inbuf, dev->rxmem, sizeof(dev->rxmem));
     if (uart_init(dev->config.uart, dev->config.baudrate, _slip_rx_cb,
                   dev) != UART_OK) {
         LOG_ERROR("slipdev: error initializing UART %i with baudrate %" PRIu32 "\n",
@@ -155,30 +118,69 @@ static int _send(netdev_t *netdev, const struct iovec *vector, unsigned count)
 static int _recv(netdev_t *netdev, void *buf, size_t len, void *info)
 {
     slipdev_t *dev = (slipdev_t *)netdev;
-    int res, idx = cib_peek(&dev->pktfifo_idx);
+    int res = 0;
 
     (void)info;
-    if (idx < 0) {
-        return -EFAULT;
-    }
     if (buf == NULL) {
         if (len > 0) {
-            /* drop packet */
-            cib_get(&dev->pktfifo_idx);
-            /* and remove data */
-            res = ringbuffer_remove(&dev->inbuf, len);
-        }
-        else {
-            res = dev->pktfifo[idx];
+            /* remove data */
+            for (; len > 0; len--) {
+                int byte = tsrb_get_one(&dev->inbuf);
+                if ((byte == (int)SLIP_END) || (byte < 0)) {
+                    /* end early if end of packet or ringbuffer is reached;
+                     * len might be larger than the actual packet */
+                    break;
+                }
+            }
+        } else {
+            /* the user was warned not to use a buffer size > `INT_MAX` ;-) */
+            res = (int)tsrb_avail(&dev->inbuf);
         }
     }
-    else if (len < dev->pktfifo[idx]) {
-        res = -ENOBUFS;
-    }
     else {
-        size_t bytes = dev->pktfifo[cib_get(&dev->pktfifo_idx)];
-        bytes = ringbuffer_get(&dev->inbuf, buf, bytes);
-        res = bytes;
+        int byte;
+        uint8_t *ptr = buf;
+
+        do {
+            if ((byte = tsrb_get_one(&dev->inbuf)) < 0) {
+                /* something went wrong, return error */
+                return -EIO;
+            }
+            switch (byte) {
+                case SLIP_END:
+                    break;
+                case SLIP_ESC:
+                    dev->inesc = 1;
+                    break;
+                case SLIP_END_ESC:
+                    if (dev->inesc) {
+                        *(ptr++) = SLIP_END;
+                        res++;
+                        dev->inesc = 0;
+                        break;
+                    }
+                    /* falls through intentionally to default when !dev->inesc */
+                case SLIP_ESC_ESC:
+                    if (dev->inesc) {
+                        *(ptr++) = SLIP_ESC;
+                        res++;
+                        dev->inesc = 0;
+                        break;
+                    }
+                    /* falls through intentionally to default when !dev->inesc */
+                default:
+                    *(ptr++) = (uint8_t)byte;
+                    res++;
+                    break;
+            }
+            if ((unsigned)res > len) {
+                while (byte != SLIP_END) {
+                    /* clear out unreceived packet */
+                    byte = tsrb_get_one(&dev->inbuf);
+                }
+                return -ENOBUFS;
+            }
+        } while (byte != SLIP_END);
     }
     return res;
 }