162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci
362306a36Sopenharmony_ci#define _GNU_SOURCE
462306a36Sopenharmony_ci#include <fcntl.h>
562306a36Sopenharmony_ci#include <assert.h>
662306a36Sopenharmony_ci#include <stdio.h>
762306a36Sopenharmony_ci#include <unistd.h>
862306a36Sopenharmony_ci#include <string.h>
962306a36Sopenharmony_ci#include "../kselftest.h"
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_cistatic int lock_set(int fd, struct flock *fl)
1262306a36Sopenharmony_ci{
1362306a36Sopenharmony_ci	int ret;
1462306a36Sopenharmony_ci
1562306a36Sopenharmony_ci	fl->l_pid = 0;		// needed for OFD locks
1662306a36Sopenharmony_ci	fl->l_whence = SEEK_SET;
1762306a36Sopenharmony_ci	ret = fcntl(fd, F_OFD_SETLK, fl);
1862306a36Sopenharmony_ci	if (ret)
1962306a36Sopenharmony_ci		perror("fcntl()");
2062306a36Sopenharmony_ci	return ret;
2162306a36Sopenharmony_ci}
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_cistatic int lock_get(int fd, struct flock *fl)
2462306a36Sopenharmony_ci{
2562306a36Sopenharmony_ci	int ret;
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci	fl->l_pid = 0;		// needed for OFD locks
2862306a36Sopenharmony_ci	fl->l_whence = SEEK_SET;
2962306a36Sopenharmony_ci	ret = fcntl(fd, F_OFD_GETLK, fl);
3062306a36Sopenharmony_ci	if (ret)
3162306a36Sopenharmony_ci		perror("fcntl()");
3262306a36Sopenharmony_ci	return ret;
3362306a36Sopenharmony_ci}
3462306a36Sopenharmony_ci
3562306a36Sopenharmony_ciint main(void)
3662306a36Sopenharmony_ci{
3762306a36Sopenharmony_ci	int rc;
3862306a36Sopenharmony_ci	struct flock fl, fl2;
3962306a36Sopenharmony_ci	int fd = open("/tmp/aa", O_RDWR | O_CREAT | O_EXCL, 0600);
4062306a36Sopenharmony_ci	int fd2 = open("/tmp/aa", O_RDONLY);
4162306a36Sopenharmony_ci
4262306a36Sopenharmony_ci	unlink("/tmp/aa");
4362306a36Sopenharmony_ci	assert(fd != -1);
4462306a36Sopenharmony_ci	assert(fd2 != -1);
4562306a36Sopenharmony_ci	ksft_print_msg("[INFO] opened fds %i %i\n", fd, fd2);
4662306a36Sopenharmony_ci
4762306a36Sopenharmony_ci	/* Set some read lock */
4862306a36Sopenharmony_ci	fl.l_type = F_RDLCK;
4962306a36Sopenharmony_ci	fl.l_start = 5;
5062306a36Sopenharmony_ci	fl.l_len = 3;
5162306a36Sopenharmony_ci	rc = lock_set(fd, &fl);
5262306a36Sopenharmony_ci	if (rc == 0) {
5362306a36Sopenharmony_ci		ksft_print_msg
5462306a36Sopenharmony_ci		    ("[SUCCESS] set OFD read lock on first fd\n");
5562306a36Sopenharmony_ci	} else {
5662306a36Sopenharmony_ci		ksft_print_msg("[FAIL] to set OFD read lock on first fd\n");
5762306a36Sopenharmony_ci		return -1;
5862306a36Sopenharmony_ci	}
5962306a36Sopenharmony_ci	/* Make sure read locks do not conflict on different fds. */
6062306a36Sopenharmony_ci	fl.l_type = F_RDLCK;
6162306a36Sopenharmony_ci	fl.l_start = 5;
6262306a36Sopenharmony_ci	fl.l_len = 1;
6362306a36Sopenharmony_ci	rc = lock_get(fd2, &fl);
6462306a36Sopenharmony_ci	if (rc != 0)
6562306a36Sopenharmony_ci		return -1;
6662306a36Sopenharmony_ci	if (fl.l_type != F_UNLCK) {
6762306a36Sopenharmony_ci		ksft_print_msg("[FAIL] read locks conflicted\n");
6862306a36Sopenharmony_ci		return -1;
6962306a36Sopenharmony_ci	}
7062306a36Sopenharmony_ci	/* Make sure read/write locks do conflict on different fds. */
7162306a36Sopenharmony_ci	fl.l_type = F_WRLCK;
7262306a36Sopenharmony_ci	fl.l_start = 5;
7362306a36Sopenharmony_ci	fl.l_len = 1;
7462306a36Sopenharmony_ci	rc = lock_get(fd2, &fl);
7562306a36Sopenharmony_ci	if (rc != 0)
7662306a36Sopenharmony_ci		return -1;
7762306a36Sopenharmony_ci	if (fl.l_type != F_UNLCK) {
7862306a36Sopenharmony_ci		ksft_print_msg
7962306a36Sopenharmony_ci		    ("[SUCCESS] read and write locks conflicted\n");
8062306a36Sopenharmony_ci	} else {
8162306a36Sopenharmony_ci		ksft_print_msg
8262306a36Sopenharmony_ci		    ("[SUCCESS] read and write locks not conflicted\n");
8362306a36Sopenharmony_ci		return -1;
8462306a36Sopenharmony_ci	}
8562306a36Sopenharmony_ci	/* Get info about the lock on first fd. */
8662306a36Sopenharmony_ci	fl.l_type = F_UNLCK;
8762306a36Sopenharmony_ci	fl.l_start = 5;
8862306a36Sopenharmony_ci	fl.l_len = 1;
8962306a36Sopenharmony_ci	rc = lock_get(fd, &fl);
9062306a36Sopenharmony_ci	if (rc != 0) {
9162306a36Sopenharmony_ci		ksft_print_msg
9262306a36Sopenharmony_ci		    ("[FAIL] F_OFD_GETLK with F_UNLCK not supported\n");
9362306a36Sopenharmony_ci		return -1;
9462306a36Sopenharmony_ci	}
9562306a36Sopenharmony_ci	if (fl.l_type != F_UNLCK) {
9662306a36Sopenharmony_ci		ksft_print_msg
9762306a36Sopenharmony_ci		    ("[SUCCESS] F_UNLCK test returns: locked, type %i pid %i len %zi\n",
9862306a36Sopenharmony_ci		     fl.l_type, fl.l_pid, fl.l_len);
9962306a36Sopenharmony_ci	} else {
10062306a36Sopenharmony_ci		ksft_print_msg
10162306a36Sopenharmony_ci		    ("[FAIL] F_OFD_GETLK with F_UNLCK did not return lock info\n");
10262306a36Sopenharmony_ci		return -1;
10362306a36Sopenharmony_ci	}
10462306a36Sopenharmony_ci	/* Try the same but by locking everything by len==0. */
10562306a36Sopenharmony_ci	fl2.l_type = F_UNLCK;
10662306a36Sopenharmony_ci	fl2.l_start = 0;
10762306a36Sopenharmony_ci	fl2.l_len = 0;
10862306a36Sopenharmony_ci	rc = lock_get(fd, &fl2);
10962306a36Sopenharmony_ci	if (rc != 0) {
11062306a36Sopenharmony_ci		ksft_print_msg
11162306a36Sopenharmony_ci		    ("[FAIL] F_OFD_GETLK with F_UNLCK not supported\n");
11262306a36Sopenharmony_ci		return -1;
11362306a36Sopenharmony_ci	}
11462306a36Sopenharmony_ci	if (memcmp(&fl, &fl2, sizeof(fl))) {
11562306a36Sopenharmony_ci		ksft_print_msg
11662306a36Sopenharmony_ci		    ("[FAIL] F_UNLCK test returns: locked, type %i pid %i len %zi\n",
11762306a36Sopenharmony_ci		     fl.l_type, fl.l_pid, fl.l_len);
11862306a36Sopenharmony_ci		return -1;
11962306a36Sopenharmony_ci	}
12062306a36Sopenharmony_ci	ksft_print_msg("[SUCCESS] F_UNLCK with len==0 returned the same\n");
12162306a36Sopenharmony_ci	/* Get info about the lock on second fd - no locks on it. */
12262306a36Sopenharmony_ci	fl.l_type = F_UNLCK;
12362306a36Sopenharmony_ci	fl.l_start = 0;
12462306a36Sopenharmony_ci	fl.l_len = 0;
12562306a36Sopenharmony_ci	lock_get(fd2, &fl);
12662306a36Sopenharmony_ci	if (fl.l_type != F_UNLCK) {
12762306a36Sopenharmony_ci		ksft_print_msg
12862306a36Sopenharmony_ci		    ("[FAIL] F_OFD_GETLK with F_UNLCK return lock info from another fd\n");
12962306a36Sopenharmony_ci		return -1;
13062306a36Sopenharmony_ci	}
13162306a36Sopenharmony_ci	return 0;
13262306a36Sopenharmony_ci}
133