xref: /openbmc/linux/drivers/gpu/drm/drm_lease.c (revision dd2939ef)
1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3  * Copyright © 2017 Keith Packard <keithp@keithp.com>
4  */
5 #include <linux/file.h>
6 #include <linux/uaccess.h>
7 
8 #include <drm/drm_auth.h>
9 #include <drm/drm_crtc_helper.h>
10 #include <drm/drm_drv.h>
11 #include <drm/drm_file.h>
12 #include <drm/drm_lease.h>
13 #include <drm/drm_print.h>
14 
15 #include "drm_crtc_internal.h"
16 #include "drm_internal.h"
17 
18 /**
19  * DOC: drm leasing
20  *
21  * DRM leases provide information about whether a DRM master may control a DRM
22  * mode setting object. This enables the creation of multiple DRM masters that
23  * manage subsets of display resources.
24  *
25  * The original DRM master of a device 'owns' the available drm resources. It
26  * may create additional DRM masters and 'lease' resources which it controls
27  * to the new DRM master. This gives the new DRM master control over the
28  * leased resources until the owner revokes the lease, or the new DRM master
29  * is closed. Some helpful terminology:
30  *
31  * - An 'owner' is a &struct drm_master that is not leasing objects from
32  *   another &struct drm_master, and hence 'owns' the objects. The owner can be
33  *   identified as the &struct drm_master for which &drm_master.lessor is NULL.
34  *
35  * - A 'lessor' is a &struct drm_master which is leasing objects to one or more
36  *   other &struct drm_master. Currently, lessees are not allowed to
37  *   create sub-leases, hence the lessor is the same as the owner.
38  *
39  * - A 'lessee' is a &struct drm_master which is leasing objects from some
40  *   other &struct drm_master. Each lessee only leases resources from a single
41  *   lessor recorded in &drm_master.lessor, and holds the set of objects that
42  *   it is leasing in &drm_master.leases.
43  *
44  * - A 'lease' is a contract between the lessor and lessee that identifies
45  *   which resources may be controlled by the lessee. All of the resources
46  *   that are leased must be owned by or leased to the lessor, and lessors are
47  *   not permitted to lease the same object to multiple lessees.
48  *
49  * The set of objects any &struct drm_master 'controls' is limited to the set
50  * of objects it leases (for lessees) or all objects (for owners).
51  *
52  * Objects not controlled by a &struct drm_master cannot be modified through
53  * the various state manipulating ioctls, and any state reported back to user
54  * space will be edited to make them appear idle and/or unusable. For
55  * instance, connectors always report 'disconnected', while encoders
56  * report no possible crtcs or clones.
57  *
58  * Since each lessee may lease objects from a single lessor, display resource
59  * leases form a tree of &struct drm_master. As lessees are currently not
60  * allowed to create sub-leases, the tree depth is limited to 1. All of
61  * these get activated simultaneously when the top level device owner changes
62  * through the SETMASTER or DROPMASTER IOCTL, so &drm_device.master points to
63  * the owner at the top of the lease tree (i.e. the &struct drm_master for which
64  * &drm_master.lessor is NULL). The full list of lessees that are leasing
65  * objects from the owner can be searched via the owner's
66  * &drm_master.lessee_idr.
67  */
68 
69 #define drm_for_each_lessee(lessee, lessor) \
70 	list_for_each_entry((lessee), &(lessor)->lessees, lessee_list)
71 
72 static uint64_t drm_lease_idr_object;
73 
74 /**
75  * drm_lease_owner - return ancestor owner drm_master
76  * @master: drm_master somewhere within tree of lessees and lessors
77  *
78  * RETURN:
79  *
80  * drm_master at the top of the tree (i.e, with lessor NULL
81  */
82 struct drm_master *drm_lease_owner(struct drm_master *master)
83 {
84 	while (master->lessor != NULL)
85 		master = master->lessor;
86 	return master;
87 }
88 
89 /**
90  * _drm_find_lessee - find lessee by id (idr_mutex held)
91  * @master: drm_master of lessor
92  * @lessee_id: id
93  *
94  * RETURN:
95  *
96  * drm_master of the lessee if valid, NULL otherwise
97  */
98 
99 static struct drm_master*
100 _drm_find_lessee(struct drm_master *master, int lessee_id)
101 {
102 	lockdep_assert_held(&master->dev->mode_config.idr_mutex);
103 	return idr_find(&drm_lease_owner(master)->lessee_idr, lessee_id);
104 }
105 
106 /**
107  * _drm_lease_held_master - check to see if an object is leased (or owned) by master (idr_mutex held)
108  * @master: the master to check the lease status of
109  * @id: the id to check
110  *
111  * Checks if the specified master holds a lease on the object. Return
112  * value:
113  *
114  *	true		'master' holds a lease on (or owns) the object
115  *	false		'master' does not hold a lease.
116  */
117 static int _drm_lease_held_master(struct drm_master *master, int id)
118 {
119 	lockdep_assert_held(&master->dev->mode_config.idr_mutex);
120 	if (master->lessor)
121 		return idr_find(&master->leases, id) != NULL;
122 	return true;
123 }
124 
125 /**
126  * _drm_has_leased - check to see if an object has been leased (idr_mutex held)
127  * @master: the master to check the lease status of
128  * @id: the id to check
129  *
130  * Checks if any lessee of 'master' holds a lease on 'id'. Return
131  * value:
132  *
133  *	true		Some lessee holds a lease on the object.
134  *	false		No lessee has a lease on the object.
135  */
136 static bool _drm_has_leased(struct drm_master *master, int id)
137 {
138 	struct drm_master *lessee;
139 
140 	lockdep_assert_held(&master->dev->mode_config.idr_mutex);
141 	drm_for_each_lessee(lessee, master)
142 		if (_drm_lease_held_master(lessee, id))
143 			return true;
144 	return false;
145 }
146 
147 /**
148  * _drm_lease_held - check drm_mode_object lease status (idr_mutex held)
149  * @file_priv: the master drm_file
150  * @id: the object id
151  *
152  * Checks if the specified master holds a lease on the object. Return
153  * value:
154  *
155  *	true		'master' holds a lease on (or owns) the object
156  *	false		'master' does not hold a lease.
157  */
158 bool _drm_lease_held(struct drm_file *file_priv, int id)
159 {
160 	bool ret;
161 	struct drm_master *master;
162 
163 	if (!file_priv)
164 		return true;
165 
166 	master = drm_file_get_master(file_priv);
167 	if (!master)
168 		return true;
169 	ret = _drm_lease_held_master(master, id);
170 	drm_master_put(&master);
171 
172 	return ret;
173 }
174 
175 /**
176  * drm_lease_held - check drm_mode_object lease status (idr_mutex not held)
177  * @file_priv: the master drm_file
178  * @id: the object id
179  *
180  * Checks if the specified master holds a lease on the object. Return
181  * value:
182  *
183  *	true		'master' holds a lease on (or owns) the object
184  *	false		'master' does not hold a lease.
185  */
186 bool drm_lease_held(struct drm_file *file_priv, int id)
187 {
188 	struct drm_master *master;
189 	bool ret;
190 
191 	if (!file_priv)
192 		return true;
193 
194 	master = drm_file_get_master(file_priv);
195 	if (!master)
196 		return true;
197 	if (!master->lessor) {
198 		ret = true;
199 		goto out;
200 	}
201 	mutex_lock(&master->dev->mode_config.idr_mutex);
202 	ret = _drm_lease_held_master(master, id);
203 	mutex_unlock(&master->dev->mode_config.idr_mutex);
204 
205 out:
206 	drm_master_put(&master);
207 	return ret;
208 }
209 
210 /**
211  * drm_lease_filter_crtcs - restricted crtc set to leased values (idr_mutex not held)
212  * @file_priv: requestor file
213  * @crtcs_in: bitmask of crtcs to check
214  *
215  * Reconstructs a crtc mask based on the crtcs which are visible
216  * through the specified file.
217  */
218 uint32_t drm_lease_filter_crtcs(struct drm_file *file_priv, uint32_t crtcs_in)
219 {
220 	struct drm_master *master;
221 	struct drm_device *dev;
222 	struct drm_crtc *crtc;
223 	int count_in, count_out;
224 	uint32_t crtcs_out = 0;
225 
226 	if (!file_priv)
227 		return crtcs_in;
228 
229 	master = drm_file_get_master(file_priv);
230 	if (!master)
231 		return crtcs_in;
232 	if (!master->lessor) {
233 		crtcs_out = crtcs_in;
234 		goto out;
235 	}
236 	dev = master->dev;
237 
238 	count_in = count_out = 0;
239 	mutex_lock(&master->dev->mode_config.idr_mutex);
240 	list_for_each_entry(crtc, &dev->mode_config.crtc_list, head) {
241 		if (_drm_lease_held_master(master, crtc->base.id)) {
242 			uint32_t mask_in = 1ul << count_in;
243 
244 			if ((crtcs_in & mask_in) != 0) {
245 				uint32_t mask_out = 1ul << count_out;
246 
247 				crtcs_out |= mask_out;
248 			}
249 			count_out++;
250 		}
251 		count_in++;
252 	}
253 	mutex_unlock(&master->dev->mode_config.idr_mutex);
254 
255 out:
256 	drm_master_put(&master);
257 	return crtcs_out;
258 }
259 
260 /*
261  * drm_lease_create - create a new drm_master with leased objects (idr_mutex not held)
262  * @lessor: lease holder (or owner) of objects
263  * @leases: objects to lease to the new drm_master
264  *
265  * Uses drm_master_create to allocate a new drm_master, then checks to
266  * make sure all of the desired objects can be leased, atomically
267  * leasing them to the new drmmaster.
268  *
269  * 	ERR_PTR(-EACCES)	some other master holds the title to any object
270  * 	ERR_PTR(-ENOENT)	some object is not a valid DRM object for this device
271  * 	ERR_PTR(-EBUSY)		some other lessee holds title to this object
272  *	ERR_PTR(-EEXIST)	same object specified more than once in the provided list
273  *	ERR_PTR(-ENOMEM)	allocation failed
274  */
275 static struct drm_master *drm_lease_create(struct drm_master *lessor, struct idr *leases)
276 {
277 	struct drm_device *dev = lessor->dev;
278 	int error;
279 	struct drm_master *lessee;
280 	int object;
281 	int id;
282 	void *entry;
283 
284 	DRM_DEBUG_LEASE("lessor %d\n", lessor->lessee_id);
285 
286 	lessee = drm_master_create(lessor->dev);
287 	if (!lessee) {
288 		DRM_DEBUG_LEASE("drm_master_create failed\n");
289 		return ERR_PTR(-ENOMEM);
290 	}
291 
292 	mutex_lock(&dev->mode_config.idr_mutex);
293 
294 	idr_for_each_entry(leases, entry, object) {
295 		error = 0;
296 		if (!idr_find(&dev->mode_config.object_idr, object))
297 			error = -ENOENT;
298 		else if (_drm_has_leased(lessor, object))
299 			error = -EBUSY;
300 
301 		if (error != 0) {
302 			DRM_DEBUG_LEASE("object %d failed %d\n", object, error);
303 			goto out_lessee;
304 		}
305 	}
306 
307 	/* Insert the new lessee into the tree */
308 	id = idr_alloc(&(drm_lease_owner(lessor)->lessee_idr), lessee, 1, 0, GFP_KERNEL);
309 	if (id < 0) {
310 		error = id;
311 		goto out_lessee;
312 	}
313 
314 	lessee->lessee_id = id;
315 	lessee->lessor = drm_master_get(lessor);
316 	list_add_tail(&lessee->lessee_list, &lessor->lessees);
317 
318 	/* Move the leases over */
319 	lessee->leases = *leases;
320 	DRM_DEBUG_LEASE("new lessee %d %p, lessor %d %p\n", lessee->lessee_id, lessee, lessor->lessee_id, lessor);
321 
322 	mutex_unlock(&dev->mode_config.idr_mutex);
323 	return lessee;
324 
325 out_lessee:
326 	mutex_unlock(&dev->mode_config.idr_mutex);
327 
328 	drm_master_put(&lessee);
329 
330 	return ERR_PTR(error);
331 }
332 
333 /**
334  * drm_lease_destroy - a master is going away (idr_mutex not held)
335  * @master: the drm_master being destroyed
336  *
337  * All lessees will have been destroyed as they
338  * hold a reference on their lessor. Notify any
339  * lessor for this master so that it can check
340  * the list of lessees.
341  */
342 void drm_lease_destroy(struct drm_master *master)
343 {
344 	struct drm_device *dev = master->dev;
345 
346 	mutex_lock(&dev->mode_config.idr_mutex);
347 
348 	DRM_DEBUG_LEASE("drm_lease_destroy %d\n", master->lessee_id);
349 
350 	/* This master is referenced by all lessees, hence it cannot be destroyed
351 	 * until all of them have been
352 	 */
353 	WARN_ON(!list_empty(&master->lessees));
354 
355 	/* Remove this master from the lessee idr in the owner */
356 	if (master->lessee_id != 0) {
357 		DRM_DEBUG_LEASE("remove master %d from device list of lessees\n", master->lessee_id);
358 		idr_remove(&(drm_lease_owner(master)->lessee_idr), master->lessee_id);
359 	}
360 
361 	/* Remove this master from any lessee list it may be on */
362 	list_del(&master->lessee_list);
363 
364 	mutex_unlock(&dev->mode_config.idr_mutex);
365 
366 	if (master->lessor) {
367 		/* Tell the master to check the lessee list */
368 		drm_sysfs_lease_event(dev);
369 		drm_master_put(&master->lessor);
370 	}
371 
372 	DRM_DEBUG_LEASE("drm_lease_destroy done %d\n", master->lessee_id);
373 }
374 
375 /**
376  * _drm_lease_revoke - revoke access to all leased objects (idr_mutex held)
377  * @top: the master losing its lease
378  */
379 static void _drm_lease_revoke(struct drm_master *top)
380 {
381 	int object;
382 	void *entry;
383 	struct drm_master *master = top;
384 
385 	lockdep_assert_held(&top->dev->mode_config.idr_mutex);
386 
387 	/*
388 	 * Walk the tree starting at 'top' emptying all leases. Because
389 	 * the tree is fully connected, we can do this without recursing
390 	 */
391 	for (;;) {
392 		DRM_DEBUG_LEASE("revoke leases for %p %d\n", master, master->lessee_id);
393 
394 		/* Evacuate the lease */
395 		idr_for_each_entry(&master->leases, entry, object)
396 			idr_remove(&master->leases, object);
397 
398 		/* Depth-first list walk */
399 
400 		/* Down */
401 		if (!list_empty(&master->lessees)) {
402 			master = list_first_entry(&master->lessees, struct drm_master, lessee_list);
403 		} else {
404 			/* Up */
405 			while (master != top && master == list_last_entry(&master->lessor->lessees, struct drm_master, lessee_list))
406 				master = master->lessor;
407 
408 			if (master == top)
409 				break;
410 
411 			/* Over */
412 			master = list_next_entry(master, lessee_list);
413 		}
414 	}
415 }
416 
417 /**
418  * drm_lease_revoke - revoke access to all leased objects (idr_mutex not held)
419  * @top: the master losing its lease
420  */
421 void drm_lease_revoke(struct drm_master *top)
422 {
423 	mutex_lock(&top->dev->mode_config.idr_mutex);
424 	_drm_lease_revoke(top);
425 	mutex_unlock(&top->dev->mode_config.idr_mutex);
426 }
427 
428 static int validate_lease(struct drm_device *dev,
429 			  int object_count,
430 			  struct drm_mode_object **objects,
431 			  bool universal_planes)
432 {
433 	int o;
434 	int has_crtc = -1;
435 	int has_connector = -1;
436 	int has_plane = -1;
437 
438 	/* we want to confirm that there is at least one crtc, plane
439 	   connector object. */
440 
441 	for (o = 0; o < object_count; o++) {
442 		if (objects[o]->type == DRM_MODE_OBJECT_CRTC && has_crtc == -1) {
443 			has_crtc = o;
444 		}
445 		if (objects[o]->type == DRM_MODE_OBJECT_CONNECTOR && has_connector == -1)
446 			has_connector = o;
447 
448 		if (universal_planes) {
449 			if (objects[o]->type == DRM_MODE_OBJECT_PLANE && has_plane == -1)
450 				has_plane = o;
451 		}
452 	}
453 	if (has_crtc == -1 || has_connector == -1)
454 		return -EINVAL;
455 	if (universal_planes && has_plane == -1)
456 		return -EINVAL;
457 	return 0;
458 }
459 
460 static int fill_object_idr(struct drm_device *dev,
461 			   struct drm_file *lessor_priv,
462 			   struct idr *leases,
463 			   int object_count,
464 			   u32 *object_ids)
465 {
466 	struct drm_mode_object **objects;
467 	u32 o;
468 	int ret;
469 	bool universal_planes = READ_ONCE(lessor_priv->universal_planes);
470 
471 	objects = kcalloc(object_count, sizeof(struct drm_mode_object *),
472 			  GFP_KERNEL);
473 	if (!objects)
474 		return -ENOMEM;
475 
476 	/* step one - get references to all the mode objects
477 	   and check for validity. */
478 	for (o = 0; o < object_count; o++) {
479 		objects[o] = drm_mode_object_find(dev, lessor_priv,
480 						  object_ids[o],
481 						  DRM_MODE_OBJECT_ANY);
482 		if (!objects[o]) {
483 			ret = -ENOENT;
484 			goto out_free_objects;
485 		}
486 
487 		if (!drm_mode_object_lease_required(objects[o]->type)) {
488 			DRM_DEBUG_KMS("invalid object for lease\n");
489 			ret = -EINVAL;
490 			goto out_free_objects;
491 		}
492 	}
493 
494 	ret = validate_lease(dev, object_count, objects, universal_planes);
495 	if (ret) {
496 		DRM_DEBUG_LEASE("lease validation failed\n");
497 		goto out_free_objects;
498 	}
499 
500 	/* add their IDs to the lease request - taking into account
501 	   universal planes */
502 	for (o = 0; o < object_count; o++) {
503 		struct drm_mode_object *obj = objects[o];
504 		u32 object_id = objects[o]->id;
505 
506 		DRM_DEBUG_LEASE("Adding object %d to lease\n", object_id);
507 
508 		/*
509 		 * We're using an IDR to hold the set of leased
510 		 * objects, but we don't need to point at the object's
511 		 * data structure from the lease as the main object_idr
512 		 * will be used to actually find that. Instead, all we
513 		 * really want is a 'leased/not-leased' result, for
514 		 * which any non-NULL pointer will work fine.
515 		 */
516 		ret = idr_alloc(leases, &drm_lease_idr_object , object_id, object_id + 1, GFP_KERNEL);
517 		if (ret < 0) {
518 			DRM_DEBUG_LEASE("Object %d cannot be inserted into leases (%d)\n",
519 					object_id, ret);
520 			goto out_free_objects;
521 		}
522 		if (obj->type == DRM_MODE_OBJECT_CRTC && !universal_planes) {
523 			struct drm_crtc *crtc = obj_to_crtc(obj);
524 
525 			ret = idr_alloc(leases, &drm_lease_idr_object, crtc->primary->base.id, crtc->primary->base.id + 1, GFP_KERNEL);
526 			if (ret < 0) {
527 				DRM_DEBUG_LEASE("Object primary plane %d cannot be inserted into leases (%d)\n",
528 						object_id, ret);
529 				goto out_free_objects;
530 			}
531 			if (crtc->cursor) {
532 				ret = idr_alloc(leases, &drm_lease_idr_object, crtc->cursor->base.id, crtc->cursor->base.id + 1, GFP_KERNEL);
533 				if (ret < 0) {
534 					DRM_DEBUG_LEASE("Object cursor plane %d cannot be inserted into leases (%d)\n",
535 							object_id, ret);
536 					goto out_free_objects;
537 				}
538 			}
539 		}
540 	}
541 
542 	ret = 0;
543 out_free_objects:
544 	for (o = 0; o < object_count; o++) {
545 		if (objects[o])
546 			drm_mode_object_put(objects[o]);
547 	}
548 	kfree(objects);
549 	return ret;
550 }
551 
552 /**
553  * drm_mode_create_lease_ioctl - create a new lease
554  * @dev: the drm device
555  * @data: pointer to struct drm_mode_create_lease
556  * @lessor_priv: the file being manipulated
557  *
558  * The master associated with the specified file will have a lease
559  * created containing the objects specified in the ioctl structure.
560  * A file descriptor will be allocated for that and returned to the
561  * application.
562  */
563 int drm_mode_create_lease_ioctl(struct drm_device *dev,
564 				void *data, struct drm_file *lessor_priv)
565 {
566 	struct drm_mode_create_lease *cl = data;
567 	size_t object_count;
568 	int ret = 0;
569 	struct idr leases;
570 	struct drm_master *lessor;
571 	struct drm_master *lessee = NULL;
572 	struct file *lessee_file = NULL;
573 	struct file *lessor_file = lessor_priv->filp;
574 	struct drm_file *lessee_priv;
575 	int fd = -1;
576 	uint32_t *object_ids;
577 
578 	/* Can't lease without MODESET */
579 	if (!drm_core_check_feature(dev, DRIVER_MODESET))
580 		return -EOPNOTSUPP;
581 
582 	/* need some objects */
583 	if (cl->object_count == 0) {
584 		DRM_DEBUG_LEASE("no objects in lease\n");
585 		return -EINVAL;
586 	}
587 
588 	if (cl->flags && (cl->flags & ~(O_CLOEXEC | O_NONBLOCK))) {
589 		DRM_DEBUG_LEASE("invalid flags\n");
590 		return -EINVAL;
591 	}
592 
593 	lessor = drm_file_get_master(lessor_priv);
594 	/* Do not allow sub-leases */
595 	if (lessor->lessor) {
596 		DRM_DEBUG_LEASE("recursive leasing not allowed\n");
597 		ret = -EINVAL;
598 		goto out_lessor;
599 	}
600 
601 	object_count = cl->object_count;
602 
603 	object_ids = memdup_user(u64_to_user_ptr(cl->object_ids),
604 			array_size(object_count, sizeof(__u32)));
605 	if (IS_ERR(object_ids)) {
606 		ret = PTR_ERR(object_ids);
607 		goto out_lessor;
608 	}
609 
610 	idr_init(&leases);
611 
612 	/* fill and validate the object idr */
613 	ret = fill_object_idr(dev, lessor_priv, &leases,
614 			      object_count, object_ids);
615 	kfree(object_ids);
616 	if (ret) {
617 		DRM_DEBUG_LEASE("lease object lookup failed: %i\n", ret);
618 		idr_destroy(&leases);
619 		goto out_lessor;
620 	}
621 
622 	/* Allocate a file descriptor for the lease */
623 	fd = get_unused_fd_flags(cl->flags & (O_CLOEXEC | O_NONBLOCK));
624 	if (fd < 0) {
625 		idr_destroy(&leases);
626 		ret = fd;
627 		goto out_lessor;
628 	}
629 
630 	DRM_DEBUG_LEASE("Creating lease\n");
631 	/* lessee will take the ownership of leases */
632 	lessee = drm_lease_create(lessor, &leases);
633 
634 	if (IS_ERR(lessee)) {
635 		ret = PTR_ERR(lessee);
636 		idr_destroy(&leases);
637 		goto out_leases;
638 	}
639 
640 	/* Clone the lessor file to create a new file for us */
641 	DRM_DEBUG_LEASE("Allocating lease file\n");
642 	lessee_file = file_clone_open(lessor_file);
643 	if (IS_ERR(lessee_file)) {
644 		ret = PTR_ERR(lessee_file);
645 		goto out_lessee;
646 	}
647 
648 	lessee_priv = lessee_file->private_data;
649 	/* Change the file to a master one */
650 	drm_master_put(&lessee_priv->master);
651 	lessee_priv->master = lessee;
652 	lessee_priv->is_master = 1;
653 	lessee_priv->authenticated = 1;
654 
655 	/* Pass fd back to userspace */
656 	DRM_DEBUG_LEASE("Returning fd %d id %d\n", fd, lessee->lessee_id);
657 	cl->fd = fd;
658 	cl->lessee_id = lessee->lessee_id;
659 
660 	/* Hook up the fd */
661 	fd_install(fd, lessee_file);
662 
663 	drm_master_put(&lessor);
664 	DRM_DEBUG_LEASE("drm_mode_create_lease_ioctl succeeded\n");
665 	return 0;
666 
667 out_lessee:
668 	drm_master_put(&lessee);
669 
670 out_leases:
671 	put_unused_fd(fd);
672 
673 out_lessor:
674 	drm_master_put(&lessor);
675 	DRM_DEBUG_LEASE("drm_mode_create_lease_ioctl failed: %d\n", ret);
676 	return ret;
677 }
678 
679 /**
680  * drm_mode_list_lessees_ioctl - list lessee ids
681  * @dev: the drm device
682  * @data: pointer to struct drm_mode_list_lessees
683  * @lessor_priv: the file being manipulated
684  *
685  * Starting from the master associated with the specified file,
686  * the master with the provided lessee_id is found, and then
687  * an array of lessee ids associated with leases from that master
688  * are returned.
689  */
690 
691 int drm_mode_list_lessees_ioctl(struct drm_device *dev,
692 			       void *data, struct drm_file *lessor_priv)
693 {
694 	struct drm_mode_list_lessees *arg = data;
695 	__u32 __user *lessee_ids = (__u32 __user *) (uintptr_t) (arg->lessees_ptr);
696 	__u32 count_lessees = arg->count_lessees;
697 	struct drm_master *lessor, *lessee;
698 	int count;
699 	int ret = 0;
700 
701 	if (arg->pad)
702 		return -EINVAL;
703 
704 	/* Can't lease without MODESET */
705 	if (!drm_core_check_feature(dev, DRIVER_MODESET))
706 		return -EOPNOTSUPP;
707 
708 	lessor = drm_file_get_master(lessor_priv);
709 	DRM_DEBUG_LEASE("List lessees for %d\n", lessor->lessee_id);
710 
711 	mutex_lock(&dev->mode_config.idr_mutex);
712 
713 	count = 0;
714 	drm_for_each_lessee(lessee, lessor) {
715 		/* Only list un-revoked leases */
716 		if (!idr_is_empty(&lessee->leases)) {
717 			if (count_lessees > count) {
718 				DRM_DEBUG_LEASE("Add lessee %d\n", lessee->lessee_id);
719 				ret = put_user(lessee->lessee_id, lessee_ids + count);
720 				if (ret)
721 					break;
722 			}
723 			count++;
724 		}
725 	}
726 
727 	DRM_DEBUG_LEASE("Lessor leases to %d\n", count);
728 	if (ret == 0)
729 		arg->count_lessees = count;
730 
731 	mutex_unlock(&dev->mode_config.idr_mutex);
732 	drm_master_put(&lessor);
733 
734 	return ret;
735 }
736 
737 /**
738  * drm_mode_get_lease_ioctl - list leased objects
739  * @dev: the drm device
740  * @data: pointer to struct drm_mode_get_lease
741  * @lessee_priv: the file being manipulated
742  *
743  * Return the list of leased objects for the specified lessee
744  */
745 
746 int drm_mode_get_lease_ioctl(struct drm_device *dev,
747 			     void *data, struct drm_file *lessee_priv)
748 {
749 	struct drm_mode_get_lease *arg = data;
750 	__u32 __user *object_ids = (__u32 __user *) (uintptr_t) (arg->objects_ptr);
751 	__u32 count_objects = arg->count_objects;
752 	struct drm_master *lessee;
753 	struct idr *object_idr;
754 	int count;
755 	void *entry;
756 	int object;
757 	int ret = 0;
758 
759 	if (arg->pad)
760 		return -EINVAL;
761 
762 	/* Can't lease without MODESET */
763 	if (!drm_core_check_feature(dev, DRIVER_MODESET))
764 		return -EOPNOTSUPP;
765 
766 	lessee = drm_file_get_master(lessee_priv);
767 	DRM_DEBUG_LEASE("get lease for %d\n", lessee->lessee_id);
768 
769 	mutex_lock(&dev->mode_config.idr_mutex);
770 
771 	if (lessee->lessor == NULL)
772 		/* owner can use all objects */
773 		object_idr = &lessee->dev->mode_config.object_idr;
774 	else
775 		/* lessee can only use allowed object */
776 		object_idr = &lessee->leases;
777 
778 	count = 0;
779 	idr_for_each_entry(object_idr, entry, object) {
780 		if (count_objects > count) {
781 			DRM_DEBUG_LEASE("adding object %d\n", object);
782 			ret = put_user(object, object_ids + count);
783 			if (ret)
784 				break;
785 		}
786 		count++;
787 	}
788 
789 	DRM_DEBUG("lease holds %d objects\n", count);
790 	if (ret == 0)
791 		arg->count_objects = count;
792 
793 	mutex_unlock(&dev->mode_config.idr_mutex);
794 	drm_master_put(&lessee);
795 
796 	return ret;
797 }
798 
799 /**
800  * drm_mode_revoke_lease_ioctl - revoke lease
801  * @dev: the drm device
802  * @data: pointer to struct drm_mode_revoke_lease
803  * @lessor_priv: the file being manipulated
804  *
805  * This removes all of the objects from the lease without
806  * actually getting rid of the lease itself; that way all
807  * references to it still work correctly
808  */
809 int drm_mode_revoke_lease_ioctl(struct drm_device *dev,
810 				void *data, struct drm_file *lessor_priv)
811 {
812 	struct drm_mode_revoke_lease *arg = data;
813 	struct drm_master *lessor;
814 	struct drm_master *lessee;
815 	int ret = 0;
816 
817 	DRM_DEBUG_LEASE("revoke lease for %d\n", arg->lessee_id);
818 
819 	/* Can't lease without MODESET */
820 	if (!drm_core_check_feature(dev, DRIVER_MODESET))
821 		return -EOPNOTSUPP;
822 
823 	lessor = drm_file_get_master(lessor_priv);
824 	mutex_lock(&dev->mode_config.idr_mutex);
825 
826 	lessee = _drm_find_lessee(lessor, arg->lessee_id);
827 
828 	/* No such lessee */
829 	if (!lessee) {
830 		ret = -ENOENT;
831 		goto fail;
832 	}
833 
834 	/* Lease is not held by lessor */
835 	if (lessee->lessor != lessor) {
836 		ret = -EACCES;
837 		goto fail;
838 	}
839 
840 	_drm_lease_revoke(lessee);
841 
842 fail:
843 	mutex_unlock(&dev->mode_config.idr_mutex);
844 	drm_master_put(&lessor);
845 
846 	return ret;
847 }
848