parent
							
								
									c67e3fe3b7
								
							
						
					
					
						commit
						3984c81faa
					
				| @ -0,0 +1,144 @@ | |||||||
|  | /*
 | ||||||
|  |  * This an unstable interface of wlroots. No guarantees are made regarding the | ||||||
|  |  * future consistency of this API. | ||||||
|  |  */ | ||||||
|  | #ifndef WLR_USE_UNSTABLE | ||||||
|  | #error "Add -DWLR_USE_UNSTABLE to enable unstable wlroots features" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #ifndef WLR_TYPES_WLR_DRM_LEASE_V1_H | ||||||
|  | #define WLR_TYPES_WLR_DRM_LEASE_V1_H | ||||||
|  | 
 | ||||||
|  | #include <wayland-server-core.h> | ||||||
|  | 
 | ||||||
|  | struct wlr_backend; | ||||||
|  | struct wlr_output; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_v1_manager { | ||||||
|  | 	struct wl_list devices; // wlr_drm_lease_device_v1::link;
 | ||||||
|  | 
 | ||||||
|  | 	struct wl_display *display; | ||||||
|  | 	struct wl_listener display_destroy; | ||||||
|  | 
 | ||||||
|  | 	struct { | ||||||
|  | 		/**
 | ||||||
|  | 		 * Upon receiving this signal, call | ||||||
|  | 		 * wlr_drm_lease_device_v1_grant_lease_request to grant a lease of the | ||||||
|  | 		 * requested DRM resources, or | ||||||
|  | 		 * wlr_drm_lease_device_v1_reject_lease_request to reject the request. | ||||||
|  | 		 */ | ||||||
|  | 		struct wl_signal request; | ||||||
|  | 	} events; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_device_v1 { | ||||||
|  | 	struct wl_list resources; | ||||||
|  | 	struct wl_global *global; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager; | ||||||
|  | 	struct wlr_backend *backend; | ||||||
|  | 
 | ||||||
|  | 	struct wl_list connectors; // wlr_drm_lease_connector_v1::link
 | ||||||
|  | 	struct wl_list leases; // wlr_drm_lease_v1::link
 | ||||||
|  | 	struct wl_list requests; // wlr_drm_lease_request_v1::link
 | ||||||
|  | 	struct wl_list link; // wlr_drm_lease_v1_manager::devices
 | ||||||
|  | 
 | ||||||
|  | 	struct wl_listener backend_destroy; | ||||||
|  | 
 | ||||||
|  | 	void *data; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_v1; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_connector_v1 { | ||||||
|  | 	struct wl_list resources; // wl_resource_get_link
 | ||||||
|  | 
 | ||||||
|  | 	struct wlr_output *output; | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device; | ||||||
|  | 	/** NULL if no client is currently leasing this connector */ | ||||||
|  | 	struct wlr_drm_lease_v1 *active_lease; | ||||||
|  | 
 | ||||||
|  | 	struct wl_listener destroy; | ||||||
|  | 
 | ||||||
|  | 	struct wl_list link; // wlr_drm_lease_device_v1::connectors
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_request_v1 { | ||||||
|  | 	struct wl_resource *resource; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 **connectors; | ||||||
|  | 	size_t n_connectors; | ||||||
|  | 
 | ||||||
|  | 	/** NULL until the lease is submitted */ | ||||||
|  | 	struct wlr_drm_lease_v1 *lease; | ||||||
|  | 
 | ||||||
|  | 	bool invalid; | ||||||
|  | 
 | ||||||
|  | 	struct wl_list link; // wlr_drm_lease_device_v1::requests
 | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_v1 { | ||||||
|  | 	struct wl_resource *resource; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 **connectors; | ||||||
|  | 	size_t n_connectors; | ||||||
|  | 
 | ||||||
|  | 	uint32_t lessee_id; | ||||||
|  | 
 | ||||||
|  | 	struct wl_list link; // wlr_drm_lease_device_v1::leases
 | ||||||
|  | 
 | ||||||
|  | 	void *data; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Creates a DRM lease manager. A DRM lease device will be created for each | ||||||
|  |  * DRM backend supplied in case of a wlr_multi_backend. | ||||||
|  |  * Returns NULL if no DRM backend is supplied. | ||||||
|  |  */ | ||||||
|  | struct wlr_drm_lease_v1_manager *wlr_drm_lease_v1_manager_create( | ||||||
|  | 	struct wl_display *display, struct wlr_backend *backend); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Offers a wlr_output for lease. | ||||||
|  |  * Returns false if the output can't be offered to lease. | ||||||
|  |  */ | ||||||
|  | bool wlr_drm_lease_v1_manager_offer_output( | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager, struct wlr_output *output); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Withdraws a previously offered output for lease. If the output is leased to | ||||||
|  |  * a client, a finished event will be send and the lease will be terminated. | ||||||
|  |  */ | ||||||
|  | void wlr_drm_lease_v1_manager_withdraw_output( | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager, struct wlr_output *output); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Grants a client's lease request. The lease device will then provision the | ||||||
|  |  * DRM lease and transfer the file descriptor to the client. After calling this, | ||||||
|  |  * each wlr_output leased is destroyed, and will be re-issued through | ||||||
|  |  * wlr_backend.events.new_outputs when the lease is revoked. | ||||||
|  |  * | ||||||
|  |  * This will return NULL without leasing any resources if the lease is invalid; | ||||||
|  |  * this can happen for example if two clients request the same resources and an | ||||||
|  |  * attempt to grant both leases is made. | ||||||
|  |  */ | ||||||
|  | struct wlr_drm_lease_v1 *wlr_drm_lease_request_v1_grant( | ||||||
|  | 	struct wlr_drm_lease_request_v1 *request); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Rejects a client's lease request. The output will still be available to | ||||||
|  |  * lease until withdrawn by the compositor. | ||||||
|  |  */ | ||||||
|  | void wlr_drm_lease_request_v1_reject(struct wlr_drm_lease_request_v1 *request); | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * Revokes a client's lease request. The output will still be available to | ||||||
|  |  * lease until withdrawn by the compositor. | ||||||
|  |  */ | ||||||
|  | void wlr_drm_lease_v1_revoke(struct wlr_drm_lease_v1 *lease); | ||||||
|  | 
 | ||||||
|  | #endif | ||||||
| @ -0,0 +1,716 @@ | |||||||
|  | #include <assert.h> | ||||||
|  | #include <errno.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <wlr/backend/drm.h> | ||||||
|  | #include <wlr/backend/multi.h> | ||||||
|  | #include <wlr/types/wlr_drm_lease_v1.h> | ||||||
|  | #include <wlr/util/log.h> | ||||||
|  | #include "backend/drm/drm.h" | ||||||
|  | #include "drm-lease-v1-protocol.h" | ||||||
|  | #include "util/signal.h" | ||||||
|  | #include "util/global.h" | ||||||
|  | 
 | ||||||
|  | #define DRM_LEASE_DEVICE_V1_VERSION 1 | ||||||
|  | 
 | ||||||
|  | static struct wp_drm_lease_device_v1_interface lease_device_impl; | ||||||
|  | static struct wp_drm_lease_connector_v1_interface lease_connector_impl; | ||||||
|  | static struct wp_drm_lease_request_v1_interface lease_request_impl; | ||||||
|  | static struct wp_drm_lease_v1_interface lease_impl; | ||||||
|  | 
 | ||||||
|  | static struct wlr_drm_lease_device_v1 *drm_lease_device_v1_from_resource( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	assert(wl_resource_instance_of(resource, | ||||||
|  | 				&wp_drm_lease_device_v1_interface, &lease_device_impl)); | ||||||
|  | 	return wl_resource_get_user_data(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wlr_drm_lease_connector_v1 * | ||||||
|  | drm_lease_connector_v1_from_resource(struct wl_resource *resource) { | ||||||
|  | 	assert(wl_resource_instance_of(resource, | ||||||
|  | 				&wp_drm_lease_connector_v1_interface, &lease_connector_impl)); | ||||||
|  | 	return wl_resource_get_user_data(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wlr_drm_lease_request_v1 *drm_lease_request_v1_from_resource( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	assert(wl_resource_instance_of(resource, | ||||||
|  | 				&wp_drm_lease_request_v1_interface, &lease_request_impl)); | ||||||
|  | 	return wl_resource_get_user_data(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wlr_drm_lease_v1 *drm_lease_v1_from_resource( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	assert(wl_resource_instance_of(resource, | ||||||
|  | 				&wp_drm_lease_v1_interface, &lease_impl)); | ||||||
|  | 	return wl_resource_get_user_data(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_v1_destroy(struct wlr_drm_lease_v1 *lease) { | ||||||
|  | 	if (!lease) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Destroying lease %"PRIu32, lease->lessee_id); | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_v1_send_finished(lease->resource); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = lease->device; | ||||||
|  | 	wlr_drm_backend_terminate_lease(device->backend, lease->lessee_id); | ||||||
|  | 	lease->lessee_id = 0; | ||||||
|  | 
 | ||||||
|  | 	for (size_t i = 0; i < lease->n_connectors; ++i) { | ||||||
|  | 		lease->connectors[i]->active_lease = NULL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_list_remove(&lease->link); | ||||||
|  | 	wl_resource_set_user_data(lease->resource, NULL); | ||||||
|  | 
 | ||||||
|  | 	free(lease->connectors); | ||||||
|  | 	free(lease); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_request_v1_destroy( | ||||||
|  | 		struct wlr_drm_lease_request_v1 *request) { | ||||||
|  | 	if (!request) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Destroying request %p", request); | ||||||
|  | 
 | ||||||
|  | 	wl_list_remove(&request->link); | ||||||
|  | 	wl_resource_set_user_data(request->resource, NULL); | ||||||
|  | 
 | ||||||
|  | 	free(request->connectors); | ||||||
|  | 	free(request); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_connector_v1_destroy( | ||||||
|  | 		struct wlr_drm_lease_connector_v1 *connector) { | ||||||
|  | 	if (!connector) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Destroying connector %s", connector->output->name); | ||||||
|  | 
 | ||||||
|  | 	if (connector->active_lease) { | ||||||
|  | 		drm_lease_v1_destroy(connector->active_lease); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wl_resource *resource, *tmp; | ||||||
|  | 	wl_resource_for_each_safe(resource, tmp, &connector->resources) { | ||||||
|  | 		wp_drm_lease_connector_v1_send_withdrawn(resource); | ||||||
|  | 
 | ||||||
|  | 		wl_resource_set_user_data(resource, NULL); | ||||||
|  | 		wl_list_remove(wl_resource_get_link(resource)); | ||||||
|  | 		wl_list_init(wl_resource_get_link(resource)); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wl_resource *device_resource; | ||||||
|  | 	wl_resource_for_each(device_resource, &connector->device->resources) { | ||||||
|  | 		wp_drm_lease_device_v1_send_done(device_resource); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_list_remove(&connector->link); | ||||||
|  | 	wl_list_remove(&connector->destroy.link); | ||||||
|  | 	free(connector); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_device_v1_destroy( | ||||||
|  | 		struct wlr_drm_lease_device_v1 *device) { | ||||||
|  | 	if (!device) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_backend *backend = | ||||||
|  | 			get_drm_backend_from_backend(device->backend); | ||||||
|  | 	wlr_log(WLR_DEBUG, "Destroying wlr_drm_lease_device_v1 for %s", | ||||||
|  | 			backend->name); | ||||||
|  | 
 | ||||||
|  | 	struct wl_resource *resource, *tmp_resource; | ||||||
|  | 	wl_resource_for_each_safe(resource, tmp_resource, &device->resources) { | ||||||
|  | 		wl_list_remove(wl_resource_get_link(resource)); | ||||||
|  | 		wl_list_init(wl_resource_get_link(resource)); | ||||||
|  | 		wl_resource_set_user_data(resource, NULL); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_request_v1 *request, *tmp_request; | ||||||
|  | 	wl_list_for_each_safe(request, tmp_request, &device->requests, link) { | ||||||
|  | 		drm_lease_request_v1_destroy(request); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_v1 *lease, *tmp_lease; | ||||||
|  | 	wl_list_for_each_safe(lease, tmp_lease, &device->leases, link) { | ||||||
|  | 		drm_lease_v1_destroy(lease); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *connector, *tmp_connector; | ||||||
|  | 	wl_list_for_each_safe(connector, tmp_connector, &device->connectors, link) { | ||||||
|  | 		drm_lease_connector_v1_destroy(connector); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_list_remove(&device->link); | ||||||
|  | 	wlr_global_destroy_safe(device->global, device->manager->display); | ||||||
|  | 
 | ||||||
|  | 	free(device); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_v1 *wlr_drm_lease_request_v1_grant( | ||||||
|  | 		struct wlr_drm_lease_request_v1 *request) { | ||||||
|  | 	assert(request->lease); | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Attempting to grant request %p", request); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_v1 *lease = request->lease; | ||||||
|  | 	assert(!request->invalid); | ||||||
|  | 
 | ||||||
|  | 	/* Transform connectors list into wlr_output for leasing */ | ||||||
|  | 	struct wlr_output *outputs[request->n_connectors + 1]; | ||||||
|  | 	for(size_t i = 0; i < request->n_connectors; ++i) { | ||||||
|  | 		outputs[i] = request->connectors[i]->output; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	int fd = wlr_drm_create_lease(outputs, request->n_connectors, | ||||||
|  | 			&lease->lessee_id); | ||||||
|  | 	if (fd < 0) { | ||||||
|  | 		wlr_log_errno(WLR_ERROR, "drm_create_lease failed"); | ||||||
|  | 		wp_drm_lease_v1_send_finished(lease->resource); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lease->connectors = calloc(request->n_connectors, | ||||||
|  | 			sizeof(struct wlr_drm_lease_connector_v1 *)); | ||||||
|  | 	if (!lease->connectors) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate lease connectors list"); | ||||||
|  | 		wp_drm_lease_v1_send_finished(lease->resource); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  | 	lease->n_connectors = request->n_connectors; | ||||||
|  | 	for (size_t i = 0; i < request->n_connectors; ++i) { | ||||||
|  | 		lease->connectors[i] = request->connectors[i]; | ||||||
|  | 		lease->connectors[i]->active_lease = lease; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Granting request %p", request); | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_v1_send_lease_fd(lease->resource, fd); | ||||||
|  | 	close(fd); | ||||||
|  | 
 | ||||||
|  | 	return lease; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlr_drm_lease_request_v1_reject( | ||||||
|  | 		struct wlr_drm_lease_request_v1 *request) { | ||||||
|  | 	assert(request && request->lease); | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Rejecting request %p", request); | ||||||
|  | 
 | ||||||
|  | 	request->invalid = true; | ||||||
|  | 	wp_drm_lease_v1_send_finished(request->lease->resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlr_drm_lease_v1_revoke(struct wlr_drm_lease_v1 *lease) { | ||||||
|  | 	assert(lease); | ||||||
|  | 	wlr_log(WLR_DEBUG, "Revoking lease %"PRIu32, lease->lessee_id); | ||||||
|  | 
 | ||||||
|  | 	drm_lease_v1_destroy(lease); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_v1_handle_resource_destroy(struct wl_resource *resource) { | ||||||
|  | 	struct wlr_drm_lease_v1 *lease = drm_lease_v1_from_resource(resource); | ||||||
|  | 	drm_lease_v1_destroy(lease); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_v1_handle_destroy( | ||||||
|  | 		struct wl_client *client, struct wl_resource *resource) { | ||||||
|  | 	wl_resource_destroy(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wp_drm_lease_v1_interface lease_impl = { | ||||||
|  | 	.destroy = drm_lease_v1_handle_destroy, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void drm_lease_request_v1_handle_resource_destroy( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	struct wlr_drm_lease_request_v1 *request = | ||||||
|  | 		drm_lease_request_v1_from_resource(resource); | ||||||
|  | 	drm_lease_request_v1_destroy(request); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_request_v1_handle_request_connector( | ||||||
|  | 		struct wl_client *client, struct wl_resource *request_resource, | ||||||
|  | 		struct wl_resource *connector_resource) { | ||||||
|  | 	struct wlr_drm_lease_request_v1 *request = | ||||||
|  | 		drm_lease_request_v1_from_resource(request_resource); | ||||||
|  | 	if (!request) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Request has been destroyed"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *connector = | ||||||
|  | 		drm_lease_connector_v1_from_resource(connector_resource); | ||||||
|  | 
 | ||||||
|  | 	if (!connector) { | ||||||
|  | 		/* This connector offer has been withdrawn or is leased */ | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to request connector"); | ||||||
|  | 		request->invalid = true; | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Requesting connector %s", connector->output->name); | ||||||
|  | 
 | ||||||
|  | 	if (request->device != connector->device) { | ||||||
|  | 		wlr_log(WLR_ERROR, "The connector belongs to another device"); | ||||||
|  | 		wl_resource_post_error(request_resource, | ||||||
|  | 				WP_DRM_LEASE_REQUEST_V1_ERROR_WRONG_DEVICE, | ||||||
|  | 				"The requested connector belongs to another device"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for (size_t i = 0; i < request->n_connectors; ++i) { | ||||||
|  | 		struct wlr_drm_lease_connector_v1 *tmp = request->connectors[i]; | ||||||
|  | 
 | ||||||
|  | 		if (connector == tmp) { | ||||||
|  | 			wlr_log(WLR_ERROR, "The connector has already been requested"); | ||||||
|  | 			wl_resource_post_error(request_resource, | ||||||
|  | 					WP_DRM_LEASE_REQUEST_V1_ERROR_DUPLICATE_CONNECTOR, | ||||||
|  | 					"The connector has already been requested"); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	size_t n_connectors = request->n_connectors + 1; | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 **tmp_connectors = | ||||||
|  | 			realloc(request->connectors, | ||||||
|  | 			n_connectors * sizeof(struct wlr_drm_lease_connector_v1 *)); | ||||||
|  | 	if (!tmp_connectors) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to grow connectors request array"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	request->connectors = tmp_connectors; | ||||||
|  | 	request->connectors[request->n_connectors] = connector; | ||||||
|  | 	request->n_connectors = n_connectors; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_request_v1_handle_submit( | ||||||
|  | 		struct wl_client *client, struct wl_resource *resource, uint32_t id) { | ||||||
|  | 	uint32_t version = wl_resource_get_version(resource); | ||||||
|  | 	struct wl_resource *lease_resource = wl_resource_create(client, | ||||||
|  | 			&wp_drm_lease_v1_interface, version, id); | ||||||
|  | 	if (!lease_resource) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wl_resource"); | ||||||
|  | 		wl_resource_post_no_memory(resource); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_implementation(lease_resource, &lease_impl, NULL, | ||||||
|  | 			drm_lease_v1_handle_resource_destroy); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_request_v1 *request = | ||||||
|  | 			drm_lease_request_v1_from_resource(resource); | ||||||
|  | 	if (!request) { | ||||||
|  | 		wlr_log(WLR_DEBUG, "Request has been destroyed"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Pre-emptively reject invalid lease requests */ | ||||||
|  | 	if (request->invalid) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Invalid request"); | ||||||
|  | 		return; | ||||||
|  | 	} else if (request->n_connectors == 0) { | ||||||
|  | 		wl_resource_post_error(lease_resource, | ||||||
|  | 				WP_DRM_LEASE_REQUEST_V1_ERROR_EMPTY_LEASE, | ||||||
|  | 				"Lease request has no connectors"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for (size_t i = 0; i < request->n_connectors; ++i) { | ||||||
|  | 		struct wlr_drm_lease_connector_v1 *conn = request->connectors[i]; | ||||||
|  | 		if (conn->active_lease) { | ||||||
|  | 			wlr_log(WLR_ERROR, "Failed to create lease, connector %s has " | ||||||
|  | 					"already been leased", conn->output->name); | ||||||
|  | 			return; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_v1 *lease = calloc(1, sizeof(struct wlr_drm_lease_v1)); | ||||||
|  | 	if (!lease) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wlr_drm_lease_v1"); | ||||||
|  | 		wl_resource_post_no_memory(resource); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lease->device = request->device; | ||||||
|  | 	wl_list_insert(&lease->device->leases, &lease->link); | ||||||
|  | 
 | ||||||
|  | 	lease->resource = lease_resource; | ||||||
|  | 	wl_resource_set_user_data(lease_resource, lease); | ||||||
|  | 
 | ||||||
|  | 	request->lease = lease; | ||||||
|  | 
 | ||||||
|  | 	/* TODO: reject the request if the user does not grant it */ | ||||||
|  | 	wlr_signal_emit_safe(&request->device->manager->events.request, | ||||||
|  | 			request); | ||||||
|  | 
 | ||||||
|  | 	/* Request is done */ | ||||||
|  | 	wl_resource_destroy(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wp_drm_lease_request_v1_interface lease_request_impl = { | ||||||
|  | 	.request_connector = drm_lease_request_v1_handle_request_connector, | ||||||
|  | 	.submit = drm_lease_request_v1_handle_submit, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void drm_lease_device_v1_handle_resource_destroy( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	wl_list_remove(wl_resource_get_link(resource)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_device_v1_handle_release( | ||||||
|  | 		struct wl_client *client, struct wl_resource *resource) { | ||||||
|  | 	wp_drm_lease_device_v1_send_released(resource); | ||||||
|  | 	wl_resource_destroy(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_device_v1_handle_create_lease_request( | ||||||
|  | 		struct wl_client *client, struct wl_resource *resource, uint32_t id) { | ||||||
|  | 	uint32_t version = wl_resource_get_version(resource); | ||||||
|  | 	struct wl_resource *request_resource = wl_resource_create(client, | ||||||
|  | 		&wp_drm_lease_request_v1_interface, version, id); | ||||||
|  | 	if (!request_resource) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wl_resource"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_implementation(request_resource, &lease_request_impl, | ||||||
|  | 		NULL, drm_lease_request_v1_handle_resource_destroy); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = | ||||||
|  | 		drm_lease_device_v1_from_resource(resource); | ||||||
|  | 	if (!device) { | ||||||
|  | 		wlr_log(WLR_DEBUG, "Failed to create lease request, " | ||||||
|  | 				"wlr_drm_lease_device_v1 has been destroyed"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_request_v1 *req = | ||||||
|  | 		calloc(1, sizeof(struct wlr_drm_lease_request_v1)); | ||||||
|  | 	if (!req) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wlr_drm_lease_request_v1"); | ||||||
|  | 		wl_resource_post_no_memory(resource); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Created request %p", req); | ||||||
|  | 
 | ||||||
|  | 	req->device = device; | ||||||
|  | 	req->resource = request_resource; | ||||||
|  | 	req->connectors = NULL; | ||||||
|  | 	req->n_connectors = 0; | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_user_data(request_resource, req); | ||||||
|  | 
 | ||||||
|  | 	wl_list_insert(&device->requests, &req->link); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wp_drm_lease_device_v1_interface lease_device_impl = { | ||||||
|  | 	.release = drm_lease_device_v1_handle_release, | ||||||
|  | 	.create_lease_request = drm_lease_device_v1_handle_create_lease_request, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void drm_connector_v1_handle_resource_destroy( | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	wl_list_remove(wl_resource_get_link(resource)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_connector_v1_handle_destroy( | ||||||
|  | 		struct wl_client *client, struct wl_resource *resource) { | ||||||
|  | 	wl_resource_destroy(resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct wp_drm_lease_connector_v1_interface lease_connector_impl = { | ||||||
|  | 	.destroy = drm_connector_v1_handle_destroy, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static void drm_lease_connector_v1_send_to_client( | ||||||
|  | 		struct wlr_drm_lease_connector_v1 *connector, | ||||||
|  | 		struct wl_resource *resource) { | ||||||
|  | 	if (connector->active_lease) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wl_client *client = wl_resource_get_client(resource); | ||||||
|  | 
 | ||||||
|  | 	uint32_t version = wl_resource_get_version(resource); | ||||||
|  | 	struct wl_resource *connector_resource = wl_resource_create(client, | ||||||
|  | 			&wp_drm_lease_connector_v1_interface, version, 0); | ||||||
|  | 	if (!connector_resource) { | ||||||
|  | 		wl_client_post_no_memory(client); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_implementation(connector_resource, &lease_connector_impl, | ||||||
|  | 			connector, drm_connector_v1_handle_resource_destroy); | ||||||
|  | 	wp_drm_lease_device_v1_send_connector(resource, connector_resource); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_output *output = connector->output; | ||||||
|  | 	wp_drm_lease_connector_v1_send_name(connector_resource, output->name); | ||||||
|  | 
 | ||||||
|  | 	// TODO: re-send the description when it's updated
 | ||||||
|  | 	wp_drm_lease_connector_v1_send_description(connector_resource, | ||||||
|  | 			output->description); | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_connector_v1_send_connector_id(connector_resource, | ||||||
|  | 			wlr_drm_connector_get_id(output)); | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_connector_v1_send_done(connector_resource); | ||||||
|  | 
 | ||||||
|  | 	wl_list_insert(&connector->resources, | ||||||
|  | 			wl_resource_get_link(connector_resource)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void lease_device_bind(struct wl_client *wl_client, void *data, | ||||||
|  | 		uint32_t version, uint32_t id) { | ||||||
|  | 	struct wl_resource *device_resource  = wl_resource_create(wl_client, | ||||||
|  | 		&wp_drm_lease_device_v1_interface, version, id); | ||||||
|  | 	if (!device_resource) { | ||||||
|  | 		wl_client_post_no_memory(wl_client); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_implementation(device_resource, &lease_device_impl, NULL, | ||||||
|  | 			drm_lease_device_v1_handle_resource_destroy); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = data; | ||||||
|  | 	if (!device) { | ||||||
|  | 		wlr_log(WLR_DEBUG, "Failed to bind lease device, " | ||||||
|  | 				"the wlr_drm_lease_device_v1 has been destroyed"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_resource_set_user_data(device_resource, device); | ||||||
|  | 
 | ||||||
|  | 	int fd = wlr_drm_backend_get_non_master_fd(device->backend); | ||||||
|  | 	if (fd < 0) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Unable to get read only DRM fd for leasing"); | ||||||
|  | 		wl_client_post_no_memory(wl_client); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_device_v1_send_drm_fd(device_resource, fd); | ||||||
|  | 	close(fd); | ||||||
|  | 
 | ||||||
|  | 	wl_list_insert(&device->resources, wl_resource_get_link(device_resource)); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *connector; | ||||||
|  | 	wl_list_for_each(connector, &device->connectors, link) { | ||||||
|  | 		drm_lease_connector_v1_send_to_client(connector, device_resource); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wp_drm_lease_device_v1_send_done(device_resource); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void handle_output_destroy(struct wl_listener *listener, void *data) { | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *conn = wl_container_of(listener, conn, | ||||||
|  | 			destroy); | ||||||
|  | 	wlr_log(WLR_DEBUG, "Handle destruction of output %s", conn->output->name); | ||||||
|  | 	wlr_drm_lease_v1_manager_withdraw_output(conn->device->manager, conn->output); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | bool wlr_drm_lease_v1_manager_offer_output( | ||||||
|  | 		struct wlr_drm_lease_v1_manager *manager, struct wlr_output *output) { | ||||||
|  | 	assert(manager && output); | ||||||
|  | 	assert(wlr_output_is_drm(output)); | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Offering output %s", output->name); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = NULL, *tmp_device; | ||||||
|  | 	wl_list_for_each(tmp_device, &manager->devices, link) { | ||||||
|  | 		if (tmp_device->backend == output->backend) { | ||||||
|  | 			device = tmp_device; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!device) { | ||||||
|  | 		wlr_log(WLR_ERROR, "No wlr_drm_lease_device_v1 associated with the " | ||||||
|  | 				"offered output"); | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *tmp_connector; | ||||||
|  | 	wl_list_for_each(tmp_connector, &device->connectors, link) { | ||||||
|  | 		if (tmp_connector->output == output) { | ||||||
|  | 			wlr_log(WLR_ERROR, "Output %s has already been offered", | ||||||
|  | 					output->name); | ||||||
|  | 			return false; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *connector = | ||||||
|  | 			calloc(1, sizeof(struct wlr_drm_lease_connector_v1)); | ||||||
|  | 	if (!connector) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wlr_drm_lease_connector_v1"); | ||||||
|  | 		return false; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	connector->output = output; | ||||||
|  | 	connector->device = device; | ||||||
|  | 
 | ||||||
|  | 	connector->destroy.notify = handle_output_destroy; | ||||||
|  | 	wl_signal_add(&output->events.destroy, &connector->destroy); | ||||||
|  | 
 | ||||||
|  | 	wl_list_init(&connector->resources); | ||||||
|  | 	wl_list_insert(&device->connectors, &connector->link); | ||||||
|  | 
 | ||||||
|  | 	struct wl_resource *resource; | ||||||
|  | 	wl_resource_for_each(resource, &device->resources) { | ||||||
|  | 		drm_lease_connector_v1_send_to_client(connector, resource); | ||||||
|  | 		wp_drm_lease_device_v1_send_done(resource); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return true; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void wlr_drm_lease_v1_manager_withdraw_output( | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager, struct wlr_output *output) { | ||||||
|  | 	assert(manager && output); | ||||||
|  | 
 | ||||||
|  | 	wlr_log(WLR_DEBUG, "Withdrawing output %s", output->name); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = NULL, *tmp_device; | ||||||
|  | 	wl_list_for_each(tmp_device, &manager->devices, link) { | ||||||
|  | 		if (tmp_device->backend == output->backend) { | ||||||
|  | 			device = tmp_device; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!device) { | ||||||
|  | 		wlr_log(WLR_ERROR, "No wlr_drm_lease_device_v1 associated with the " | ||||||
|  | 				"given output"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_connector_v1 *connector = NULL, *tmp_conn; | ||||||
|  | 	wl_list_for_each(tmp_conn, &device->connectors, link) { | ||||||
|  | 		if (tmp_conn->output == output) { | ||||||
|  | 			connector = tmp_conn; | ||||||
|  | 			break; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!connector) { | ||||||
|  | 		wlr_log(WLR_DEBUG, "No wlr_drm_connector_v1 associated with the given " | ||||||
|  | 				"output"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	drm_lease_connector_v1_destroy(connector); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void handle_backend_destroy(struct wl_listener *listener, void *data) { | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device = | ||||||
|  | 		wl_container_of(listener, device, backend_destroy); | ||||||
|  | 	drm_lease_device_v1_destroy(device); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void drm_lease_device_v1_create(struct wlr_drm_lease_v1_manager *manager, | ||||||
|  | 		struct wlr_backend *backend) { | ||||||
|  | 	assert(backend); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_backend *drm_backend = | ||||||
|  | 			get_drm_backend_from_backend(backend); | ||||||
|  | 	wlr_log(WLR_DEBUG, "Creating wlr_drm_lease_device_v1 for %s", | ||||||
|  | 			drm_backend->name); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *lease_device = | ||||||
|  | 		calloc(1, sizeof(struct wlr_drm_lease_device_v1)); | ||||||
|  | 
 | ||||||
|  | 	if (!lease_device) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wlr_drm_lease_device_v1"); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lease_device->manager = manager; | ||||||
|  | 	lease_device->backend = backend; | ||||||
|  | 	wl_list_init(&lease_device->resources); | ||||||
|  | 	wl_list_init(&lease_device->connectors); | ||||||
|  | 	wl_list_init(&lease_device->requests); | ||||||
|  | 	wl_list_init(&lease_device->leases); | ||||||
|  | 	wl_list_init(&lease_device->link); | ||||||
|  | 
 | ||||||
|  | 	lease_device->global = wl_global_create(manager->display, | ||||||
|  | 		&wp_drm_lease_device_v1_interface, DRM_LEASE_DEVICE_V1_VERSION, | ||||||
|  | 		lease_device, lease_device_bind); | ||||||
|  | 
 | ||||||
|  | 	if (!lease_device->global) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wp_drm_lease_device_v1 global"); | ||||||
|  | 		free(lease_device); | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	lease_device->backend_destroy.notify = handle_backend_destroy; | ||||||
|  | 	wl_signal_add(&backend->events.destroy, &lease_device->backend_destroy); | ||||||
|  | 
 | ||||||
|  | 	wl_list_insert(&manager->devices, &lease_device->link); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void multi_backend_cb(struct wlr_backend *backend, void *data) { | ||||||
|  | 	if (!wlr_backend_is_drm(backend)) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager = data; | ||||||
|  | 	drm_lease_device_v1_create(manager, backend); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void handle_display_destroy(struct wl_listener *listener, void *data) { | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager = wl_container_of(listener, manager, | ||||||
|  | 			display_destroy); | ||||||
|  | 	wlr_log(WLR_DEBUG, "Destroying wlr_drm_lease_v1_manager"); | ||||||
|  | 
 | ||||||
|  | 	struct wlr_drm_lease_device_v1 *device, *tmp; | ||||||
|  | 	wl_list_for_each_safe(device, tmp, &manager->devices, link) { | ||||||
|  | 		drm_lease_device_v1_destroy(device); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	free(manager); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct wlr_drm_lease_v1_manager *wlr_drm_lease_v1_manager_create( | ||||||
|  | 		struct wl_display *display, struct wlr_backend *backend) { | ||||||
|  | 	struct wlr_drm_lease_v1_manager *manager = calloc(1, | ||||||
|  | 			sizeof(struct wlr_drm_lease_v1_manager)); | ||||||
|  | 	if (!manager) { | ||||||
|  | 		wlr_log(WLR_ERROR, "Failed to allocate wlr_drm_lease_v1_manager"); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	wl_list_init(&manager->devices); | ||||||
|  | 	manager->display = display; | ||||||
|  | 
 | ||||||
|  | 	if (wlr_backend_is_multi(backend)) { | ||||||
|  | 		/* TODO: handle backends added after the manager is created */ | ||||||
|  | 		wlr_multi_for_each_backend(backend, multi_backend_cb, manager); | ||||||
|  | 	} else if (wlr_backend_is_drm(backend)) { | ||||||
|  | 		drm_lease_device_v1_create(manager, backend); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (wl_list_empty(&manager->devices)) { | ||||||
|  | 		wlr_log(WLR_ERROR, "No DRM backend supplied, failed to create " | ||||||
|  | 				"wlr_drm_lease_v1_manager"); | ||||||
|  | 		free(manager); | ||||||
|  | 		return NULL; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	manager->display_destroy.notify = handle_display_destroy; | ||||||
|  | 	wl_display_add_destroy_listener(display, &manager->display_destroy); | ||||||
|  | 
 | ||||||
|  | 	wl_signal_init(&manager->events.request); | ||||||
|  | 
 | ||||||
|  | 	return manager; | ||||||
|  | } | ||||||
					Loading…
					
					
				
		Reference in new issue