xref: /linux-6.15/tools/testing/selftests/sync/sync.c (revision ed2d8fa7)
182208160SEmilio López /*
282208160SEmilio López  *  sync / sw_sync abstraction
382208160SEmilio López  *  Copyright 2015-2016 Collabora Ltd.
482208160SEmilio López  *
582208160SEmilio López  *  Based on the implementation from the Android Open Source Project,
682208160SEmilio López  *
782208160SEmilio López  *  Copyright 2012 Google, Inc
882208160SEmilio López  *
982208160SEmilio López  *  Permission is hereby granted, free of charge, to any person obtaining a
1082208160SEmilio López  *  copy of this software and associated documentation files (the "Software"),
1182208160SEmilio López  *  to deal in the Software without restriction, including without limitation
1282208160SEmilio López  *  the rights to use, copy, modify, merge, publish, distribute, sublicense,
1382208160SEmilio López  *  and/or sell copies of the Software, and to permit persons to whom the
1482208160SEmilio López  *  Software is furnished to do so, subject to the following conditions:
1582208160SEmilio López  *
1682208160SEmilio López  *  The above copyright notice and this permission notice shall be included in
1782208160SEmilio López  *  all copies or substantial portions of the Software.
1882208160SEmilio López  *
1982208160SEmilio López  *  THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
2082208160SEmilio López  *  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
2182208160SEmilio López  *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
2282208160SEmilio López  *  THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR
2382208160SEmilio López  *  OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
2482208160SEmilio López  *  ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
2582208160SEmilio López  *  OTHER DEALINGS IN THE SOFTWARE.
2682208160SEmilio López  */
2782208160SEmilio López 
2882208160SEmilio López #include <fcntl.h>
2982208160SEmilio López #include <malloc.h>
3082208160SEmilio López #include <poll.h>
3182208160SEmilio López #include <stdint.h>
3282208160SEmilio López #include <string.h>
3382208160SEmilio López #include <unistd.h>
3482208160SEmilio López 
3582208160SEmilio López #include <sys/ioctl.h>
3682208160SEmilio López #include <sys/stat.h>
3782208160SEmilio López #include <sys/types.h>
3882208160SEmilio López 
3982208160SEmilio López #include "sync.h"
4082208160SEmilio López #include "sw_sync.h"
4182208160SEmilio López 
4282208160SEmilio López #include <linux/sync_file.h>
4382208160SEmilio López 
4482208160SEmilio López 
4582208160SEmilio López /* SW_SYNC ioctls */
4682208160SEmilio López struct sw_sync_create_fence_data {
4782208160SEmilio López 	__u32	value;
4882208160SEmilio López 	char	name[32];
4982208160SEmilio López 	__s32	fence;
5082208160SEmilio López };
5182208160SEmilio López 
5282208160SEmilio López #define SW_SYNC_IOC_MAGIC		'W'
5382208160SEmilio López #define SW_SYNC_IOC_CREATE_FENCE	_IOWR(SW_SYNC_IOC_MAGIC, 0,\
5482208160SEmilio López 					      struct sw_sync_create_fence_data)
5582208160SEmilio López #define SW_SYNC_IOC_INC			_IOW(SW_SYNC_IOC_MAGIC, 1, __u32)
5682208160SEmilio López 
5782208160SEmilio López 
sync_wait(int fd,int timeout)5882208160SEmilio López int sync_wait(int fd, int timeout)
5982208160SEmilio López {
6082208160SEmilio López 	struct pollfd fds;
6182208160SEmilio López 
6282208160SEmilio López 	fds.fd = fd;
6382208160SEmilio López 	fds.events = POLLIN | POLLERR;
6482208160SEmilio López 
6582208160SEmilio López 	return poll(&fds, 1, timeout);
6682208160SEmilio López }
6782208160SEmilio López 
sync_merge(const char * name,int fd1,int fd2)6882208160SEmilio López int sync_merge(const char *name, int fd1, int fd2)
6982208160SEmilio López {
7082208160SEmilio López 	struct sync_merge_data data = {};
7182208160SEmilio López 	int err;
7282208160SEmilio López 
7382208160SEmilio López 	data.fd2 = fd2;
7482208160SEmilio López 	strncpy(data.name, name, sizeof(data.name) - 1);
7582208160SEmilio López 	data.name[sizeof(data.name) - 1] = '\0';
7682208160SEmilio López 
7782208160SEmilio López 	err = ioctl(fd1, SYNC_IOC_MERGE, &data);
7882208160SEmilio López 	if (err < 0)
7982208160SEmilio López 		return err;
8082208160SEmilio López 
8182208160SEmilio López 	return data.fence;
8282208160SEmilio López }
8382208160SEmilio López 
sync_file_info(int fd)8482208160SEmilio López static struct sync_file_info *sync_file_info(int fd)
8582208160SEmilio López {
8682208160SEmilio López 	struct sync_file_info *info;
8782208160SEmilio López 	struct sync_fence_info *fence_info;
8882208160SEmilio López 	int err, num_fences;
8982208160SEmilio López 
9082208160SEmilio López 	info = calloc(1, sizeof(*info));
9182208160SEmilio López 	if (info == NULL)
9282208160SEmilio López 		return NULL;
9382208160SEmilio López 
9482208160SEmilio López 	err = ioctl(fd, SYNC_IOC_FILE_INFO, info);
9582208160SEmilio López 	if (err < 0) {
9682208160SEmilio López 		free(info);
9782208160SEmilio López 		return NULL;
9882208160SEmilio López 	}
9982208160SEmilio López 
10082208160SEmilio López 	num_fences = info->num_fences;
10182208160SEmilio López 
10282208160SEmilio López 	if (num_fences) {
10382208160SEmilio López 		info->flags = 0;
10482208160SEmilio López 		info->num_fences = num_fences;
10582208160SEmilio López 
10682208160SEmilio López 		fence_info = calloc(num_fences, sizeof(*fence_info));
10782208160SEmilio López 		if (!fence_info) {
10882208160SEmilio López 			free(info);
10982208160SEmilio López 			return NULL;
11082208160SEmilio López 		}
11182208160SEmilio López 
112*ed2d8fa7SMasami Hiramatsu 		info->sync_fence_info = (uint64_t)(unsigned long)fence_info;
11382208160SEmilio López 
11482208160SEmilio López 		err = ioctl(fd, SYNC_IOC_FILE_INFO, info);
11582208160SEmilio López 		if (err < 0) {
11682208160SEmilio López 			free(fence_info);
11782208160SEmilio López 			free(info);
11882208160SEmilio López 			return NULL;
11982208160SEmilio López 		}
12082208160SEmilio López 	}
12182208160SEmilio López 
12282208160SEmilio López 	return info;
12382208160SEmilio López }
12482208160SEmilio López 
sync_file_info_free(struct sync_file_info * info)12582208160SEmilio López static void sync_file_info_free(struct sync_file_info *info)
12682208160SEmilio López {
127*ed2d8fa7SMasami Hiramatsu 	free((void *)(unsigned long)info->sync_fence_info);
12882208160SEmilio López 	free(info);
12982208160SEmilio López }
13082208160SEmilio López 
sync_fence_size(int fd)13182208160SEmilio López int sync_fence_size(int fd)
13282208160SEmilio López {
13382208160SEmilio López 	int count;
13482208160SEmilio López 	struct sync_file_info *info = sync_file_info(fd);
13582208160SEmilio López 
13682208160SEmilio López 	if (!info)
13782208160SEmilio López 		return 0;
13882208160SEmilio López 
13982208160SEmilio López 	count = info->num_fences;
14082208160SEmilio López 
14182208160SEmilio López 	sync_file_info_free(info);
14282208160SEmilio López 
14382208160SEmilio López 	return count;
14482208160SEmilio López }
14582208160SEmilio López 
sync_fence_count_with_status(int fd,int status)14682208160SEmilio López int sync_fence_count_with_status(int fd, int status)
14782208160SEmilio López {
14882208160SEmilio López 	unsigned int i, count = 0;
14982208160SEmilio López 	struct sync_fence_info *fence_info = NULL;
15082208160SEmilio López 	struct sync_file_info *info = sync_file_info(fd);
15182208160SEmilio López 
15282208160SEmilio López 	if (!info)
15382208160SEmilio López 		return -1;
15482208160SEmilio López 
155*ed2d8fa7SMasami Hiramatsu 	fence_info = (struct sync_fence_info *)(unsigned long)info->sync_fence_info;
15682208160SEmilio López 	for (i = 0 ; i < info->num_fences ; i++) {
15782208160SEmilio López 		if (fence_info[i].status == status)
15882208160SEmilio López 			count++;
15982208160SEmilio López 	}
16082208160SEmilio López 
16182208160SEmilio López 	sync_file_info_free(info);
16282208160SEmilio López 
16382208160SEmilio López 	return count;
16482208160SEmilio López }
16582208160SEmilio López 
sw_sync_timeline_create(void)16682208160SEmilio López int sw_sync_timeline_create(void)
16782208160SEmilio López {
16882208160SEmilio López 	return open("/sys/kernel/debug/sync/sw_sync", O_RDWR);
16982208160SEmilio López }
17082208160SEmilio López 
sw_sync_timeline_inc(int fd,unsigned int count)17182208160SEmilio López int sw_sync_timeline_inc(int fd, unsigned int count)
17282208160SEmilio López {
17382208160SEmilio López 	__u32 arg = count;
17482208160SEmilio López 
17582208160SEmilio López 	return ioctl(fd, SW_SYNC_IOC_INC, &arg);
17682208160SEmilio López }
17782208160SEmilio López 
sw_sync_timeline_is_valid(int fd)17882208160SEmilio López int sw_sync_timeline_is_valid(int fd)
17982208160SEmilio López {
18082208160SEmilio López 	int status;
18182208160SEmilio López 
18282208160SEmilio López 	if (fd == -1)
18382208160SEmilio López 		return 0;
18482208160SEmilio López 
18582208160SEmilio López 	status = fcntl(fd, F_GETFD, 0);
18682208160SEmilio López 	return (status >= 0);
18782208160SEmilio López }
18882208160SEmilio López 
sw_sync_timeline_destroy(int fd)18982208160SEmilio López void sw_sync_timeline_destroy(int fd)
19082208160SEmilio López {
19182208160SEmilio López 	if (sw_sync_timeline_is_valid(fd))
19282208160SEmilio López 		close(fd);
19382208160SEmilio López }
19482208160SEmilio López 
sw_sync_fence_create(int fd,const char * name,unsigned int value)19582208160SEmilio López int sw_sync_fence_create(int fd, const char *name, unsigned int value)
19682208160SEmilio López {
19782208160SEmilio López 	struct sw_sync_create_fence_data data = {};
19882208160SEmilio López 	int err;
19982208160SEmilio López 
20082208160SEmilio López 	data.value = value;
20182208160SEmilio López 	strncpy(data.name, name, sizeof(data.name) - 1);
20282208160SEmilio López 	data.name[sizeof(data.name) - 1] = '\0';
20382208160SEmilio López 
20482208160SEmilio López 	err = ioctl(fd, SW_SYNC_IOC_CREATE_FENCE, &data);
20582208160SEmilio López 	if (err < 0)
20682208160SEmilio López 		return err;
20782208160SEmilio López 
20882208160SEmilio López 	return data.fence;
20982208160SEmilio López }
21082208160SEmilio López 
sw_sync_fence_is_valid(int fd)21182208160SEmilio López int sw_sync_fence_is_valid(int fd)
21282208160SEmilio López {
21382208160SEmilio López 	/* Same code! */
21482208160SEmilio López 	return sw_sync_timeline_is_valid(fd);
21582208160SEmilio López }
21682208160SEmilio López 
sw_sync_fence_destroy(int fd)21782208160SEmilio López void sw_sync_fence_destroy(int fd)
21882208160SEmilio López {
21982208160SEmilio López 	if (sw_sync_fence_is_valid(fd))
22082208160SEmilio López 		close(fd);
22182208160SEmilio López }
222