2 * Copyright © 2020 Collabora, Ltd.
4 * Permission is hereby granted, free of charge, to any person obtaining
5 * a copy of this software and associated documentation files (the
6 * "Software"), to deal in the Software without restriction, including
7 * without limitation the rights to use, copy, modify, merge, publish,
8 * distribute, sublicense, and/or sell copies of the Software, and to
9 * permit persons to whom the Software is furnished to do so, subject to
10 * the following conditions:
12 * The above copyright notice and this permission notice (including the
13 * next paragraph) shall be included in all copies or substantial
14 * portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
20 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
21 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
22 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
26 #include "ivi-compositor.h"
30 #include <sys/smack.h>
34 #include "shared/helpers.h"
37 static const char *const bind_agl_shell[] = {
38 "User::App::homescreen",
39 "User::App::cluster-gauges" /* cluster-dashboard */
42 static const char *const bind_agl_shell_desktop[] = {
43 "User::App::launcher",
44 "User::App::alexa-viewer",
47 "User::App::xdg-cluster-receiver", /* cluster-receiver, native XDG app*/
48 "User::App::cluster-receiver" /* cluster-receiver, Qt app */
52 static const char *const applications_permitted[] = {
53 "homescreen", "alexa-viewer", "launcher", "hvac",
54 "navigation", "mediaplayer"
57 /* helper start searches the applications_permitted for the
61 ivi_policy_verify_permitted_app(const char *app_id)
63 for (size_t i = 0; i < ARRAY_LENGTH(applications_permitted); i++)
64 if (strcmp(app_id, applications_permitted[i]) == 0)
71 /* helper to determine which applications are allowed to bind to the
75 ivi_policy_check_bind_agl_shell(const char *app_id)
77 for (size_t i = 0; i < ARRAY_LENGTH(bind_agl_shell); i++)
78 if (strcmp(app_id, bind_agl_shell[i]) == 0)
85 ivi_policy_check_bind_agl_shell_desktop(const char *app_id)
87 for (size_t i = 0; i < ARRAY_LENGTH(bind_agl_shell_desktop); i++)
88 if (strcmp(app_id, bind_agl_shell_desktop[i]) == 0)
96 ivi_policy_verify_ivi_surface(struct ivi_surface *surf)
98 const char *app_id = weston_desktop_surface_get_app_id(surf->dsurface);
99 return ivi_policy_verify_permitted_app(app_id);
103 * deny-all policy implementation allows every action to be denied
105 * This is an example, that implements the API
107 * For injecting rules back in the compositor one should use ivi_policy_add()
108 * - policy_rule_allow_to_add is required in order to add further policy rules
109 * - policy_rule_try_event will be callback executed when handling the state
113 ivi_policy_default_surface_create(struct ivi_surface *surf, void *user_data)
115 return ivi_policy_verify_ivi_surface(surf);
119 ivi_policy_default_surface_commmited(struct ivi_surface *surf, void *user_data)
121 return ivi_policy_verify_ivi_surface(surf);
125 ivi_policy_default_surface_activate(struct ivi_surface *surf, void *user_data)
127 return ivi_policy_verify_ivi_surface(surf);
131 ivi_policy_default_surface_deactivate(struct ivi_surface *surf, void *user_data)
133 return ivi_policy_verify_ivi_surface(surf);
137 ivi_policy_default_surface_activate_default(struct ivi_surface *surf, void *user_data)
139 return ivi_policy_verify_ivi_surface(surf);
143 ivi_policy_default_surface_advertise_state_change(struct ivi_surface *surf, void *user_data)
145 return ivi_policy_verify_ivi_surface(surf);
150 ivi_policy_default_shell_bind_interface(void *client, void *interface)
152 struct wl_interface *shell_interface = interface;
153 struct wl_client *conn_client = client;
160 wl_client_get_credentials(conn_client, &pid, &uid, &gid);
162 client_fd = wl_client_get_fd(conn_client);
163 if (smack_new_label_from_socket(client_fd, &label) < 0) {
167 if (strcmp(shell_interface->name, "agl_shell") == 0)
168 ret = ivi_policy_check_bind_agl_shell(label);
170 if (strcmp(shell_interface->name, "agl_shell_desktop") == 0)
171 ret = ivi_policy_check_bind_agl_shell_desktop(label);
174 weston_log("Client with pid %d, uid %d, gid %d, allowed "
175 "to bind to %s for label %s\n", pid, uid, gid,
176 shell_interface->name, label);
178 /* client responsible for free'ing */
184 ivi_policy_default_shell_bind_interface(void *client, void *interface)
191 ivi_policy_default_allow_to_add(void *user_data)
193 /* verify that policy rules can be added with ivi_policy_add() */
198 * Policy rules added by ivi_policy_add() will be handled by this callback, and
199 * should be treated depending on the event. Note this is just an example.
202 ivi_policy_default_try_event(struct ivi_a_policy *a_policy)
204 uint32_t event = a_policy->event;
207 case AGL_SHELL_POLICY_EVENT_SHOW:
208 ivi_layout_activate(a_policy->output, a_policy->app_id);
210 case AGL_SHELL_POLICY_EVENT_HIDE:
211 ivi_layout_deactivate(a_policy->policy->ivi, a_policy->app_id);
217 static const struct ivi_policy_api policy_api = {
218 .struct_size = sizeof(policy_api),
219 .surface_create = ivi_policy_default_surface_create,
220 .surface_commited = ivi_policy_default_surface_commmited,
221 .surface_activate = ivi_policy_default_surface_activate,
222 .surface_deactivate = ivi_policy_default_surface_deactivate,
223 .surface_activate_by_default = ivi_policy_default_surface_activate_default,
224 .surface_advertise_state_change = ivi_policy_default_surface_advertise_state_change,
225 .shell_bind_interface = ivi_policy_default_shell_bind_interface,
226 .policy_rule_allow_to_add = ivi_policy_default_allow_to_add,
227 .policy_rule_try_event = ivi_policy_default_try_event,
231 ivi_policy_init(struct ivi_compositor *ivi)
233 ivi->policy = ivi_policy_create(ivi, &policy_api, ivi);
237 weston_log("Installing 'deny-all' policy engine\n");