#include "../config-host.h"
/* SPDX-License-Identifier: MIT */
/*
 * Description: run various truncate tests
 *
 */
#include <errno.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/ioctl.h>

#include "liburing.h"
#include "helpers.h"

#define TWO_GIG_SIZE ((loff_t)2 * 1024 * 1024 * 1024)
#define ONE_GIG_SIZE ((loff_t)1024 * 1024 * 1024)
#define HALF_GIG_SIZE ((loff_t)512 * 1024 * 1024)

static int test_truncate(struct io_uring *ring, int fd)
{
	struct io_uring_cqe *cqe;
	struct io_uring_sqe *sqe;
	int ret = -1;

	sqe = io_uring_get_sqe(ring);
	if (!sqe) {
		fprintf(stderr, "get sqe failed\n");
		return T_EXIT_FAIL;
	}

	memset(sqe, 0, sizeof(*sqe));

	io_uring_prep_rw(IORING_OP_FTRUNCATE, sqe, fd, "fail", 0, 4);

	ret = io_uring_submit(ring);
	if (ret <= 0) {
		fprintf(stderr, "sqe submit failed: %d\n", ret);
		return T_EXIT_FAIL;
	}

	ret = io_uring_wait_cqe(ring, &cqe);
	if (ret < 0) {
		fprintf(stderr, "wait completion %d\n", ret);
		return T_EXIT_FAIL;
	}
	ret = cqe->res;
	io_uring_cqe_seen(ring, cqe);
	if (ret == -EINVAL)
		return T_EXIT_PASS;

	fprintf(stderr, "unexpected truncate res %d\n", ret);
	return T_EXIT_FAIL;
}

static int test_ftruncate(struct io_uring *ring, int fd, loff_t len)
{
	struct io_uring_cqe *cqe;
	struct io_uring_sqe *sqe;
	int ret;

	sqe = io_uring_get_sqe(ring);
	if (!sqe) {
		fprintf(stderr, "get sqe failed\n");
		goto err;
	}

	memset(sqe, 0, sizeof(*sqe));

	io_uring_prep_ftruncate(sqe, fd, len);

	ret = io_uring_submit(ring);
	if (ret <= 0) {
		fprintf(stderr, "sqe submit failed: %d\n", ret);
		goto err;
	}

	ret = io_uring_wait_cqe(ring, &cqe);
	if (ret < 0) {
		fprintf(stderr, "wait completion %d\n", ret);
		goto err;
	}
	ret = cqe->res;
	io_uring_cqe_seen(ring, cqe);
	return ret;
err:
	return 1;
}

static int get_file_size(int fd, loff_t *size)
{
	struct stat st;

	if (fstat(fd, &st) < 0) {
		perror("fstat");
		return -1;
	}
	if (S_ISREG(st.st_mode)) {
		*size = st.st_size;
		return 0;
	} else if (S_ISBLK(st.st_mode)) {
		unsigned long long bytes;

		if (ioctl(fd, BLKGETSIZE64, &bytes) != 0) {
			perror("ioctl");
			return -1;
		}

		*size = bytes;
		return 0;
	}

	return -1;
}

int main(int argc, char *argv[])
{
	struct io_uring ring;
	char path[32] = ".truncate.XXXXXX";
	int ret;
	int fd;
	int i;
	loff_t size;
	loff_t test_sizes[3];

	if (argc > 1)
		return T_EXIT_SKIP;

	ret = io_uring_queue_init(1, &ring, 0);
	if (ret) {
		fprintf(stderr, "ring setup failed: %d\n", ret);
		return T_EXIT_FAIL;
	}

	fd = mkostemp(path, O_WRONLY | O_CREAT | O_TRUNC);
	if (fd < 0) {
		perror("mkostemp");
		return T_EXIT_FAIL;
	}

	test_sizes[0] = TWO_GIG_SIZE;
	test_sizes[1] = ONE_GIG_SIZE;
	test_sizes[2] = HALF_GIG_SIZE;

	for (i = 0; i < 3; i++) {
		ret = test_ftruncate(&ring, fd, test_sizes[i]);
		if (ret < 0) {
			if (ret == -EBADF || ret == -EINVAL) {
				if (i == 0) {
					fprintf(stdout, "Ftruncate not supported, skipping\n");
					ret = T_EXIT_SKIP;
					goto out;
				}
				goto err;
			}
			fprintf(stderr, "ftruncate: %s\n", strerror(-ret));
			goto err;
		} else if (ret) {
			fprintf(stderr, "unexpected cqe->res %d\n", ret);
			goto err;
		}
		if (get_file_size(fd, &size))
			goto err;
		if (size != test_sizes[i]) {
			fprintf(stderr, "fail %d size=%llu, %llu\n", i,
				(unsigned long long) size,
				(unsigned long long) test_sizes[i]);
			goto err;
		}
	}

	ret = test_truncate(&ring, fd);
	if (ret != T_EXIT_PASS)
		goto err;

out:
	unlink(path);
	close(fd);
	return T_EXIT_PASS;
err:
	unlink(path);
	close(fd);
	return T_EXIT_FAIL;
}