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