#include <linux/time64.h>
 #include <pthread.h>
 #include <fcntl.h>
+#include <linux/sockios.h>
 
 #include "vsock_test_zerocopy.h"
 #include "timeout.h"
        close(fd);
 }
 
+static void test_unread_bytes_server(const struct test_opts *opts, int type)
+{
+       unsigned char buf[MSG_BUF_IOCTL_LEN];
+       int client_fd;
+
+       client_fd = vsock_accept(VMADDR_CID_ANY, opts->peer_port, NULL, type);
+       if (client_fd < 0) {
+               perror("accept");
+               exit(EXIT_FAILURE);
+       }
+
+       for (int i = 0; i < sizeof(buf); i++)
+               buf[i] = rand() & 0xFF;
+
+       send_buf(client_fd, buf, sizeof(buf), 0, sizeof(buf));
+       control_writeln("SENT");
+
+       close(client_fd);
+}
+
+static void test_unread_bytes_client(const struct test_opts *opts, int type)
+{
+       unsigned char buf[MSG_BUF_IOCTL_LEN];
+       int fd;
+
+       fd = vsock_connect(opts->peer_cid, opts->peer_port, type);
+       if (fd < 0) {
+               perror("connect");
+               exit(EXIT_FAILURE);
+       }
+
+       control_expectln("SENT");
+       /* The data has arrived but has not been read. The expected is
+        * MSG_BUF_IOCTL_LEN.
+        */
+       if (!vsock_ioctl_int(fd, SIOCINQ, MSG_BUF_IOCTL_LEN)) {
+               fprintf(stderr, "Test skipped, SIOCINQ not supported.\n");
+               goto out;
+       }
+
+       recv_buf(fd, buf, sizeof(buf), 0, sizeof(buf));
+       /* All data has been consumed, so the expected is 0. */
+       vsock_ioctl_int(fd, SIOCINQ, 0);
+
+out:
+       close(fd);
+}
+
 static void test_stream_unsent_bytes_client(const struct test_opts *opts)
 {
        test_unsent_bytes_client(opts, SOCK_STREAM);
        test_unsent_bytes_server(opts, SOCK_SEQPACKET);
 }
 
+static void test_stream_unread_bytes_client(const struct test_opts *opts)
+{
+       test_unread_bytes_client(opts, SOCK_STREAM);
+}
+
+static void test_stream_unread_bytes_server(const struct test_opts *opts)
+{
+       test_unread_bytes_server(opts, SOCK_STREAM);
+}
+
+static void test_seqpacket_unread_bytes_client(const struct test_opts *opts)
+{
+       test_unread_bytes_client(opts, SOCK_SEQPACKET);
+}
+
+static void test_seqpacket_unread_bytes_server(const struct test_opts *opts)
+{
+       test_unread_bytes_server(opts, SOCK_SEQPACKET);
+}
+
 #define RCVLOWAT_CREDIT_UPD_BUF_SIZE   (1024 * 128)
 /* This define is the same as in 'include/linux/virtio_vsock.h':
  * it is used to decide when to send credit update message during
                .run_client = test_stream_transport_change_client,
                .run_server = test_stream_transport_change_server,
        },
+       {
+               .name = "SOCK_STREAM ioctl(SIOCINQ) functionality",
+               .run_client = test_stream_unread_bytes_client,
+               .run_server = test_stream_unread_bytes_server,
+       },
+       {
+               .name = "SOCK_SEQPACKET ioctl(SIOCINQ) functionality",
+               .run_client = test_seqpacket_unread_bytes_client,
+               .run_server = test_seqpacket_unread_bytes_server,
+       },
        {},
 };