1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * drivers/dma-buf/sync_file.c 4 * 5 * Copyright (C) 2012 Google, Inc. 6 */ 7 8 #include <linux/export.h> 9 #include <linux/file.h> 10 #include <linux/fs.h> 11 #include <linux/kernel.h> 12 #include <linux/poll.h> 13 #include <linux/sched.h> 14 #include <linux/slab.h> 15 #include <linux/uaccess.h> 16 #include <linux/anon_inodes.h> 17 #include <linux/sync_file.h> 18 #include <uapi/linux/sync_file.h> 19 20 static const struct file_operations sync_file_fops; 21 22 static struct sync_file *sync_file_alloc(void) 23 { 24 struct sync_file *sync_file; 25 26 sync_file = kzalloc(sizeof(*sync_file), GFP_KERNEL); 27 if (!sync_file) 28 return NULL; 29 30 sync_file->file = anon_inode_getfile("sync_file", &sync_file_fops, 31 sync_file, 0); 32 if (IS_ERR(sync_file->file)) 33 goto err; 34 35 init_waitqueue_head(&sync_file->wq); 36 37 INIT_LIST_HEAD(&sync_file->cb.node); 38 39 return sync_file; 40 41 err: 42 kfree(sync_file); 43 return NULL; 44 } 45 46 static void fence_check_cb_func(struct dma_fence *f, struct dma_fence_cb *cb) 47 { 48 struct sync_file *sync_file; 49 50 sync_file = container_of(cb, struct sync_file, cb); 51 52 wake_up_all(&sync_file->wq); 53 } 54 55 /** 56 * sync_file_create() - creates a sync file 57 * @fence: fence to add to the sync_fence 58 * 59 * Creates a sync_file containg @fence. This function acquires and additional 60 * reference of @fence for the newly-created &sync_file, if it succeeds. The 61 * sync_file can be released with fput(sync_file->file). Returns the 62 * sync_file or NULL in case of error. 63 */ 64 struct sync_file *sync_file_create(struct dma_fence *fence) 65 { 66 struct sync_file *sync_file; 67 68 sync_file = sync_file_alloc(); 69 if (!sync_file) 70 return NULL; 71 72 sync_file->fence = dma_fence_get(fence); 73 74 return sync_file; 75 } 76 EXPORT_SYMBOL(sync_file_create); 77 78 static struct sync_file *sync_file_fdget(int fd) 79 { 80 struct file *file = fget(fd); 81 82 if (!file) 83 return NULL; 84 85 if (file->f_op != &sync_file_fops) 86 goto err; 87 88 return file->private_data; 89 90 err: 91 fput(file); 92 return NULL; 93 } 94 95 /** 96 * sync_file_get_fence - get the fence related to the sync_file fd 97 * @fd: sync_file fd to get the fence from 98 * 99 * Ensures @fd references a valid sync_file and returns a fence that 100 * represents all fence in the sync_file. On error NULL is returned. 101 */ 102 struct dma_fence *sync_file_get_fence(int fd) 103 { 104 struct sync_file *sync_file; 105 struct dma_fence *fence; 106 107 sync_file = sync_file_fdget(fd); 108 if (!sync_file) 109 return NULL; 110 111 fence = dma_fence_get(sync_file->fence); 112 fput(sync_file->file); 113 114 return fence; 115 } 116 EXPORT_SYMBOL(sync_file_get_fence); 117 118 /** 119 * sync_file_get_name - get the name of the sync_file 120 * @sync_file: sync_file to get the fence from 121 * @buf: destination buffer to copy sync_file name into 122 * @len: available size of destination buffer. 123 * 124 * Each sync_file may have a name assigned either by the user (when merging 125 * sync_files together) or created from the fence it contains. In the latter 126 * case construction of the name is deferred until use, and so requires 127 * sync_file_get_name(). 128 * 129 * Returns: a string representing the name. 130 */ 131 char *sync_file_get_name(struct sync_file *sync_file, char *buf, int len) 132 { 133 if (sync_file->user_name[0]) { 134 strlcpy(buf, sync_file->user_name, len); 135 } else { 136 struct dma_fence *fence = sync_file->fence; 137 138 snprintf(buf, len, "%s-%s%llu-%lld", 139 fence->ops->get_driver_name(fence), 140 fence->ops->get_timeline_name(fence), 141 fence->context, 142 fence->seqno); 143 } 144 145 return buf; 146 } 147 148 static int sync_file_set_fence(struct sync_file *sync_file, 149 struct dma_fence **fences, int num_fences) 150 { 151 struct dma_fence_array *array; 152 153 /* 154 * The reference for the fences in the new sync_file and held 155 * in add_fence() during the merge procedure, so for num_fences == 1 156 * we already own a new reference to the fence. For num_fence > 1 157 * we own the reference of the dma_fence_array creation. 158 */ 159 if (num_fences == 1) { 160 sync_file->fence = fences[0]; 161 kfree(fences); 162 } else { 163 array = dma_fence_array_create(num_fences, fences, 164 dma_fence_context_alloc(1), 165 1, false); 166 if (!array) 167 return -ENOMEM; 168 169 sync_file->fence = &array->base; 170 } 171 172 return 0; 173 } 174 175 static struct dma_fence **get_fences(struct sync_file *sync_file, 176 int *num_fences) 177 { 178 if (dma_fence_is_array(sync_file->fence)) { 179 struct dma_fence_array *array = to_dma_fence_array(sync_file->fence); 180 181 *num_fences = array->num_fences; 182 return array->fences; 183 } 184 185 *num_fences = 1; 186 return &sync_file->fence; 187 } 188 189 static void add_fence(struct dma_fence **fences, 190 int *i, struct dma_fence *fence) 191 { 192 fences[*i] = fence; 193 194 if (!dma_fence_is_signaled(fence)) { 195 dma_fence_get(fence); 196 (*i)++; 197 } 198 } 199 200 /** 201 * sync_file_merge() - merge two sync_files 202 * @name: name of new fence 203 * @a: sync_file a 204 * @b: sync_file b 205 * 206 * Creates a new sync_file which contains copies of all the fences in both 207 * @a and @b. @a and @b remain valid, independent sync_file. Returns the 208 * new merged sync_file or NULL in case of error. 209 */ 210 static struct sync_file *sync_file_merge(const char *name, struct sync_file *a, 211 struct sync_file *b) 212 { 213 struct sync_file *sync_file; 214 struct dma_fence **fences, **nfences, **a_fences, **b_fences; 215 int i, i_a, i_b, num_fences, a_num_fences, b_num_fences; 216 217 sync_file = sync_file_alloc(); 218 if (!sync_file) 219 return NULL; 220 221 a_fences = get_fences(a, &a_num_fences); 222 b_fences = get_fences(b, &b_num_fences); 223 if (a_num_fences > INT_MAX - b_num_fences) 224 goto err; 225 226 num_fences = a_num_fences + b_num_fences; 227 228 fences = kcalloc(num_fences, sizeof(*fences), GFP_KERNEL); 229 if (!fences) 230 goto err; 231 232 /* 233 * Assume sync_file a and b are both ordered and have no 234 * duplicates with the same context. 235 * 236 * If a sync_file can only be created with sync_file_merge 237 * and sync_file_create, this is a reasonable assumption. 238 */ 239 for (i = i_a = i_b = 0; i_a < a_num_fences && i_b < b_num_fences; ) { 240 struct dma_fence *pt_a = a_fences[i_a]; 241 struct dma_fence *pt_b = b_fences[i_b]; 242 243 if (pt_a->context < pt_b->context) { 244 add_fence(fences, &i, pt_a); 245 246 i_a++; 247 } else if (pt_a->context > pt_b->context) { 248 add_fence(fences, &i, pt_b); 249 250 i_b++; 251 } else { 252 if (__dma_fence_is_later(pt_a->seqno, pt_b->seqno, 253 pt_a->ops)) 254 add_fence(fences, &i, pt_a); 255 else 256 add_fence(fences, &i, pt_b); 257 258 i_a++; 259 i_b++; 260 } 261 } 262 263 for (; i_a < a_num_fences; i_a++) 264 add_fence(fences, &i, a_fences[i_a]); 265 266 for (; i_b < b_num_fences; i_b++) 267 add_fence(fences, &i, b_fences[i_b]); 268 269 if (i == 0) 270 fences[i++] = dma_fence_get(a_fences[0]); 271 272 if (num_fences > i) { 273 nfences = krealloc_array(fences, i, sizeof(*fences), GFP_KERNEL); 274 if (!nfences) 275 goto err; 276 277 fences = nfences; 278 } 279 280 if (sync_file_set_fence(sync_file, fences, i) < 0) { 281 kfree(fences); 282 goto err; 283 } 284 285 strlcpy(sync_file->user_name, name, sizeof(sync_file->user_name)); 286 return sync_file; 287 288 err: 289 fput(sync_file->file); 290 return NULL; 291 292 } 293 294 static int sync_file_release(struct inode *inode, struct file *file) 295 { 296 struct sync_file *sync_file = file->private_data; 297 298 if (test_bit(POLL_ENABLED, &sync_file->flags)) 299 dma_fence_remove_callback(sync_file->fence, &sync_file->cb); 300 dma_fence_put(sync_file->fence); 301 kfree(sync_file); 302 303 return 0; 304 } 305 306 static __poll_t sync_file_poll(struct file *file, poll_table *wait) 307 { 308 struct sync_file *sync_file = file->private_data; 309 310 poll_wait(file, &sync_file->wq, wait); 311 312 if (list_empty(&sync_file->cb.node) && 313 !test_and_set_bit(POLL_ENABLED, &sync_file->flags)) { 314 if (dma_fence_add_callback(sync_file->fence, &sync_file->cb, 315 fence_check_cb_func) < 0) 316 wake_up_all(&sync_file->wq); 317 } 318 319 return dma_fence_is_signaled(sync_file->fence) ? EPOLLIN : 0; 320 } 321 322 static long sync_file_ioctl_merge(struct sync_file *sync_file, 323 unsigned long arg) 324 { 325 int fd = get_unused_fd_flags(O_CLOEXEC); 326 int err; 327 struct sync_file *fence2, *fence3; 328 struct sync_merge_data data; 329 330 if (fd < 0) 331 return fd; 332 333 if (copy_from_user(&data, (void __user *)arg, sizeof(data))) { 334 err = -EFAULT; 335 goto err_put_fd; 336 } 337 338 if (data.flags || data.pad) { 339 err = -EINVAL; 340 goto err_put_fd; 341 } 342 343 fence2 = sync_file_fdget(data.fd2); 344 if (!fence2) { 345 err = -ENOENT; 346 goto err_put_fd; 347 } 348 349 data.name[sizeof(data.name) - 1] = '\0'; 350 fence3 = sync_file_merge(data.name, sync_file, fence2); 351 if (!fence3) { 352 err = -ENOMEM; 353 goto err_put_fence2; 354 } 355 356 data.fence = fd; 357 if (copy_to_user((void __user *)arg, &data, sizeof(data))) { 358 err = -EFAULT; 359 goto err_put_fence3; 360 } 361 362 fd_install(fd, fence3->file); 363 fput(fence2->file); 364 return 0; 365 366 err_put_fence3: 367 fput(fence3->file); 368 369 err_put_fence2: 370 fput(fence2->file); 371 372 err_put_fd: 373 put_unused_fd(fd); 374 return err; 375 } 376 377 static int sync_fill_fence_info(struct dma_fence *fence, 378 struct sync_fence_info *info) 379 { 380 strlcpy(info->obj_name, fence->ops->get_timeline_name(fence), 381 sizeof(info->obj_name)); 382 strlcpy(info->driver_name, fence->ops->get_driver_name(fence), 383 sizeof(info->driver_name)); 384 385 info->status = dma_fence_get_status(fence); 386 while (test_bit(DMA_FENCE_FLAG_SIGNALED_BIT, &fence->flags) && 387 !test_bit(DMA_FENCE_FLAG_TIMESTAMP_BIT, &fence->flags)) 388 cpu_relax(); 389 info->timestamp_ns = 390 test_bit(DMA_FENCE_FLAG_TIMESTAMP_BIT, &fence->flags) ? 391 ktime_to_ns(fence->timestamp) : 392 ktime_set(0, 0); 393 394 return info->status; 395 } 396 397 static long sync_file_ioctl_fence_info(struct sync_file *sync_file, 398 unsigned long arg) 399 { 400 struct sync_file_info info; 401 struct sync_fence_info *fence_info = NULL; 402 struct dma_fence **fences; 403 __u32 size; 404 int num_fences, ret, i; 405 406 if (copy_from_user(&info, (void __user *)arg, sizeof(info))) 407 return -EFAULT; 408 409 if (info.flags || info.pad) 410 return -EINVAL; 411 412 fences = get_fences(sync_file, &num_fences); 413 414 /* 415 * Passing num_fences = 0 means that userspace doesn't want to 416 * retrieve any sync_fence_info. If num_fences = 0 we skip filling 417 * sync_fence_info and return the actual number of fences on 418 * info->num_fences. 419 */ 420 if (!info.num_fences) { 421 info.status = dma_fence_get_status(sync_file->fence); 422 goto no_fences; 423 } else { 424 info.status = 1; 425 } 426 427 if (info.num_fences < num_fences) 428 return -EINVAL; 429 430 size = num_fences * sizeof(*fence_info); 431 fence_info = kzalloc(size, GFP_KERNEL); 432 if (!fence_info) 433 return -ENOMEM; 434 435 for (i = 0; i < num_fences; i++) { 436 int status = sync_fill_fence_info(fences[i], &fence_info[i]); 437 info.status = info.status <= 0 ? info.status : status; 438 } 439 440 if (copy_to_user(u64_to_user_ptr(info.sync_fence_info), fence_info, 441 size)) { 442 ret = -EFAULT; 443 goto out; 444 } 445 446 no_fences: 447 sync_file_get_name(sync_file, info.name, sizeof(info.name)); 448 info.num_fences = num_fences; 449 450 if (copy_to_user((void __user *)arg, &info, sizeof(info))) 451 ret = -EFAULT; 452 else 453 ret = 0; 454 455 out: 456 kfree(fence_info); 457 458 return ret; 459 } 460 461 static long sync_file_ioctl(struct file *file, unsigned int cmd, 462 unsigned long arg) 463 { 464 struct sync_file *sync_file = file->private_data; 465 466 switch (cmd) { 467 case SYNC_IOC_MERGE: 468 return sync_file_ioctl_merge(sync_file, arg); 469 470 case SYNC_IOC_FILE_INFO: 471 return sync_file_ioctl_fence_info(sync_file, arg); 472 473 default: 474 return -ENOTTY; 475 } 476 } 477 478 static const struct file_operations sync_file_fops = { 479 .release = sync_file_release, 480 .poll = sync_file_poll, 481 .unlocked_ioctl = sync_file_ioctl, 482 .compat_ioctl = compat_ptr_ioctl, 483 }; 484