root/tools/testing/selftests/net/bind_timewait.c
// SPDX-License-Identifier: GPL-2.0
/* Copyright Amazon.com Inc. or its affiliates. */

#include <sys/socket.h>
#include <netinet/in.h>

#include "kselftest_harness.h"

FIXTURE(bind_timewait)
{
        struct sockaddr_in addr;
        socklen_t addrlen;
};

FIXTURE_VARIANT(bind_timewait)
{
        __u32 addr_const;
};

FIXTURE_VARIANT_ADD(bind_timewait, localhost)
{
        .addr_const = INADDR_LOOPBACK
};

FIXTURE_VARIANT_ADD(bind_timewait, addrany)
{
        .addr_const = INADDR_ANY
};

FIXTURE_SETUP(bind_timewait)
{
        self->addr.sin_family = AF_INET;
        self->addr.sin_port = 0;
        self->addr.sin_addr.s_addr = htonl(variant->addr_const);
        self->addrlen = sizeof(self->addr);
}

FIXTURE_TEARDOWN(bind_timewait)
{
}

void create_timewait_socket(struct __test_metadata *_metadata,
                            FIXTURE_DATA(bind_timewait) *self)
{
        int server_fd, client_fd, child_fd, ret;
        struct sockaddr_in addr;
        socklen_t addrlen;

        server_fd = socket(AF_INET, SOCK_STREAM, 0);
        ASSERT_GT(server_fd, 0);

        ret = bind(server_fd, (struct sockaddr *)&self->addr, self->addrlen);
        ASSERT_EQ(ret, 0);

        ret = listen(server_fd, 1);
        ASSERT_EQ(ret, 0);

        ret = getsockname(server_fd, (struct sockaddr *)&self->addr, &self->addrlen);
        ASSERT_EQ(ret, 0);

        client_fd = socket(AF_INET, SOCK_STREAM, 0);
        ASSERT_GT(client_fd, 0);

        ret = connect(client_fd, (struct sockaddr *)&self->addr, self->addrlen);
        ASSERT_EQ(ret, 0);

        addrlen = sizeof(addr);
        child_fd = accept(server_fd, (struct sockaddr *)&addr, &addrlen);
        ASSERT_GT(child_fd, 0);

        close(child_fd);
        close(client_fd);
        close(server_fd);
}

TEST_F(bind_timewait, 1)
{
        int fd, ret;

        create_timewait_socket(_metadata, self);

        fd = socket(AF_INET, SOCK_STREAM, 0);
        ASSERT_GT(fd, 0);

        ret = bind(fd, (struct sockaddr *)&self->addr, self->addrlen);
        ASSERT_EQ(ret, -1);
        ASSERT_EQ(errno, EADDRINUSE);

        close(fd);
}

TEST_HARNESS_MAIN