ecb23280931f922056de3c8cb4cd10e34b5a0048
[platform/core/appfw/shortcut.git] / lib / src / shortcut_internal.c
1 /*
2  * Copyright (c) 2011 - 2017 Samsung Electronics Co., Ltd. All rights reserved.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *
16  */
17 #include <gio/gio.h>
18 #include <stdlib.h>
19 #include <errno.h>
20 #include <unistd.h>
21 #include <fcntl.h>
22 #include <string.h>
23
24 #include <aul.h>
25 #include <dlog.h>
26 #include <glib.h>
27 #include <stdio.h>
28
29 #include "shortcut.h"
30 #include "shortcut_private.h"
31 #include "shortcut_internal.h"
32
33 #define SHORTCUT_PKGNAME_LEN 512
34 #define REQUEST_ID_LEN 40
35 #define TIMEOUT 4000
36
37 #define PROVIDER_BUS_NAME "org.tizen.data_provider_service"
38 #define PROVIDER_OBJECT_PATH "/org/tizen/data_provider_service"
39 #define PROVIDER_SHORTCUT_INTERFACE_NAME "org.tizen.data_provider_shortcut_service"
40
41 #define DBUS_SERVICE_DBUS "org.freedesktop.DBus"
42 #define DBUS_PATH_DBUS "/org/freedesktop/DBus"
43 #define DBUS_INTERFACE_DBUS "org.freedesktop.DBus"
44
45 static GDBusConnection *_gdbus_conn = NULL;
46 static int monitor_id = 0;
47 static int provider_monitor_id = 0;
48
49 typedef struct _shortcut_request_cb_info {
50         int (*request_cb)(const char *appid, const char *name, int type, const char *content, const char *icon, pid_t pid, double period, int allow_duplicate, void *data);
51         void *data;
52 } shortcut_request_cb_info;
53
54 typedef struct _shortcut_remove_cb_info {
55         int (*remove_cb)(const char *package_name, const char *name, int sender_pid, void *user_data);
56         void *data;
57 } shortcut_remove_cb_info;
58
59 static shortcut_request_cb_info _request_callback_info;
60 static shortcut_remove_cb_info _remove_callback_info;
61
62 static void _shortcut_send_return(int ret_val, const char *request_id)
63 {
64         int ret;
65         GDBusMessage *reply = NULL;
66
67         ret = _dbus_init();
68         if (ret != SHORTCUT_ERROR_NONE) {
69                 SHORTCUT_ERR("Can't init dbus %d", ret);
70                 return;
71         }
72
73         _send_sync_shortcut(g_variant_new("(is)", ret_val, request_id),
74                             &reply, "send_return_value");
75         if (reply)
76                 g_object_unref(reply);
77 }
78
79 /* LCOV_EXCL_START */
80 static void _add_shortcut_notify(GVariant *parameters)
81 {
82         int ret = SHORTCUT_ERROR_NONE;
83         const char *appid;
84         const char *name;
85         int type;
86         const char *content;
87         const char *icon;
88         int allow_duplicate;
89         int sender_pid;
90         const char *request_id;
91
92         g_variant_get(parameters, "(&si&s&si&s&si)", &request_id, &sender_pid, &appid, &name, &type, &content, &icon, &allow_duplicate);
93         SHORTCUT_DBG("_add_shortcut_notify sender pid : [%d] appid : [%s]", sender_pid, appid);
94
95         if (_request_callback_info.request_cb != NULL)
96                 ret = _request_callback_info.request_cb(appid, name, type, content, icon, sender_pid, -1.0f, allow_duplicate, _request_callback_info.data);
97         else
98                 SHORTCUT_DBG("request_cb is null.");
99
100         _shortcut_send_return(ret, request_id);
101 }
102 /* LCOV_EXCL_STOP */
103
104 /* LCOV_EXCL_START */
105 static void _add_shortcut_widget_notify(GVariant *parameters)
106 {
107         int ret = SHORTCUT_ERROR_NONE;
108         const char *appid;
109         const char *name;
110         int type;
111         const char *content;
112         const char *icon;
113         int allow_duplicate;
114         int sender_pid;
115         double period;
116         const char *request_id;
117
118         g_variant_get(parameters, "(&si&s&si&s&sdi)", &request_id, &sender_pid, &appid, &name, &type, &content, &icon, &period, &allow_duplicate);
119         SHORTCUT_DBG("_add_shortcut_widget_notify sender pid : [%d] appid : [%s]", sender_pid, appid);
120
121         if (_request_callback_info.request_cb != NULL)
122                 ret = _request_callback_info.request_cb(appid, name, type, content, icon, sender_pid, period, allow_duplicate, _request_callback_info.data);
123         else
124                 SHORTCUT_DBG("request_cb is null.");
125
126         _shortcut_send_return(ret, request_id);
127 }
128 /* LCOV_EXCL_STOP */
129
130 static void _remove_shortcut_notify(GVariant *parameters)
131 {
132         int ret = SHORTCUT_ERROR_NONE;
133         const char *appid;
134         const char *name;
135         int sender_pid;
136         const char *request_id;
137
138         g_variant_get(parameters, "(&si&s&s)", &request_id, &sender_pid, &appid, &name);
139         SHORTCUT_DBG("_remove_shortcut_notify sender pid : [%d] appid : [%s]", sender_pid, appid);
140
141         if (_remove_callback_info.remove_cb != NULL)
142                 ret = _remove_callback_info.remove_cb(appid, name, sender_pid, _remove_callback_info.data);
143         else
144                 SHORTCUT_DBG("remove_cb is null.");
145
146         _shortcut_send_return(ret, request_id);
147 }
148
149 /* LCOV_EXCL_START */
150 static void _handle_shortcut_notify(GDBusConnection *connection,
151                 const gchar     *sender_name,
152                 const gchar     *object_path,
153                 const gchar     *interface_name,
154                 const gchar     *signal_name,
155                 GVariant        *parameters,
156                 gpointer         user_data)
157 {
158         SHORTCUT_DBG("signal_name : [%s]", signal_name);
159         if (g_strcmp0(signal_name, "add_shortcut_notify") == 0)
160                 _add_shortcut_notify(parameters);
161         else if (g_strcmp0(signal_name, "add_shortcut_widget_notify") == 0)
162                 _add_shortcut_widget_notify(parameters);
163         else if (g_strcmp0(signal_name, "remove_shortcut_notify") == 0)
164                 _remove_shortcut_notify(parameters);
165 }
166 /* LCOV_EXCL_STOP */
167
168 int _dbus_init(void)
169 {
170         GError *error = NULL;
171
172         if (_gdbus_conn == NULL) {
173                 _gdbus_conn = g_bus_get_sync(G_BUS_TYPE_SYSTEM, NULL, &error);
174
175                 if (_gdbus_conn == NULL) {
176                         /* LCOV_EXCL_START */
177                         if (error != NULL) {
178
179                                 SHORTCUT_ERR("Failed to get dbus [%s]", error->message);
180                                 g_error_free(error);
181
182                         }
183                         return SHORTCUT_ERROR_IO_ERROR;
184                         /* LCOV_EXCL_STOP */
185                 }
186                 shortcut_error_quark();
187         }
188
189         return SHORTCUT_ERROR_NONE;
190 }
191
192 int _dbus_signal_init(void)
193 {
194         int ret = SHORTCUT_ERROR_NONE;
195         int id;
196
197         if (monitor_id == 0) {
198                 id = g_dbus_connection_signal_subscribe(
199                                 _gdbus_conn,
200                                 PROVIDER_BUS_NAME,
201                                 PROVIDER_SHORTCUT_INTERFACE_NAME,       /* interface */
202                                 NULL,                                   /* member */
203                                 PROVIDER_OBJECT_PATH,                   /* path */
204                                 NULL,                                   /* arg0 */
205                                 G_DBUS_SIGNAL_FLAGS_NONE,
206                                 _handle_shortcut_notify,
207                                 NULL,
208                                 NULL);
209                 SHORTCUT_DBG("subscribe id : %d", id);
210                 if (id == 0) {
211                         /* LCOV_EXCL_START */
212                         ret = SHORTCUT_ERROR_IO_ERROR;
213                         SHORTCUT_ERR("Failed to _register_noti_dbus_interface");
214                         /* LCOV_EXCL_STOP */
215                 } else {
216                         SHORTCUT_INFO("get dbus connection success");
217                         monitor_id = id;
218                 }
219         }
220
221         return ret;
222 }
223
224 char *_shortcut_get_pkgname_by_pid(void)
225 {
226         char pkgname[SHORTCUT_PKGNAME_LEN + 1] = { 0, };
227         char buf[SHORTCUT_PKGNAME_LEN + 1] = { 0, };
228         int pid, ret;
229         int fd;
230         char *dup_pkgname;
231
232         pid = getpid();
233
234         ret = aul_app_get_pkgname_bypid(pid, pkgname, sizeof(pkgname));
235         if (ret != 0) {
236                 /* LCOV_EXCL_START */
237                 snprintf(buf, sizeof(buf), "/proc/%d/cmdline", pid);
238
239                 fd = open(buf, O_RDONLY);
240                 if (fd < 0)
241                         return NULL;
242
243                 ret = read(fd, pkgname, sizeof(pkgname) - 1);
244                 close(fd);
245
246                 if (ret <= 0)
247                         return NULL;
248
249                 pkgname[ret] = '\0';
250                 /* LCOV_EXCL_STOP */
251                 /*!
252                  * \NOTE
253                  * "ret" is not able to be larger than "sizeof(pkgname) - 1",
254                  * if the system is not going wrong.
255                  */
256         } else {
257                 if (strlen(pkgname) <= 0)
258                         return NULL;
259         }
260
261         dup_pkgname = strdup(pkgname);
262         if (!dup_pkgname)
263                 SHORTCUT_ERR("Heap: %d\n", errno); /* LCOV_EXCL_LINE */
264
265         return dup_pkgname;
266 }
267
268 /*
269  * implement user request
270  */
271 int _send_sync_shortcut(GVariant *body, GDBusMessage **reply, char *cmd)
272 {
273         GError *err = NULL;
274         GDBusMessage *msg;
275         int ret = SHORTCUT_ERROR_NONE;
276
277         msg = g_dbus_message_new_method_call(
278                         PROVIDER_BUS_NAME,
279                         PROVIDER_OBJECT_PATH,
280                         PROVIDER_SHORTCUT_INTERFACE_NAME,
281                         cmd);
282         if (!msg) {
283                 /* LCOV_EXCL_START */
284                 SHORTCUT_ERR("Can't allocate new method call");
285                 if (body)
286                         g_variant_unref(body);
287                 return SHORTCUT_ERROR_OUT_OF_MEMORY;
288                 /* LCOV_EXCL_STOP */
289         }
290
291         if (body != NULL)
292                 g_dbus_message_set_body(msg, body);
293
294         *reply = g_dbus_connection_send_message_with_reply_sync(
295                         _gdbus_conn,
296                         msg,
297                         G_DBUS_SEND_MESSAGE_FLAGS_NONE,
298                         TIMEOUT,
299                         NULL,
300                         NULL,
301                         &err);
302
303         g_object_unref(msg);
304
305         if (!*reply) {
306                 /* LCOV_EXCL_START */
307                 ret = SHORTCUT_ERROR_COMM;
308                 if (err != NULL) {
309                         SHORTCUT_ERR("No reply. cmd = %s,  error = %s", cmd, err->message);
310                         if (err->code == G_DBUS_ERROR_ACCESS_DENIED)
311                                 ret = SHORTCUT_ERROR_PERMISSION_DENIED;
312                         g_error_free(err);
313                 }
314                 return ret;
315                 /* LCOV_EXCL_STOP */
316         }
317
318         if (g_dbus_message_to_gerror(*reply, &err)) {
319                 /* LCOV_EXCL_START */
320                 if (err->code == G_DBUS_ERROR_ACCESS_DENIED)
321                         ret = SHORTCUT_ERROR_PERMISSION_DENIED;
322                 else
323                         ret = err->code;
324
325                 SHORTCUT_ERR("_send_sync_shortcut error %s err code: %d", err->message, ret);
326                 g_error_free(err);
327                 return ret;
328                 /* LCOV_EXCL_STOP */
329         }
330
331         SHORTCUT_DBG("_send_sync_shortcut done !!");
332         return SHORTCUT_ERROR_NONE;
333 }
334
335 int _send_service_register(void)
336 {
337         GDBusMessage *reply = NULL;
338         int result;
339
340         result = _send_sync_shortcut(g_variant_new("(i)", getuid()), &reply, "shortcut_service_register");
341         SHORTCUT_DBG("_send_service_register done");
342         return result;
343 }
344
345 static void _send_message_with_reply_sync_cb(GDBusConnection *connection,
346                 GAsyncResult *res,
347                 gpointer user_data)
348 {
349         int result;
350         GError *err = NULL;
351         GDBusMessage *reply = NULL;
352         GVariant *body;
353         struct result_cb_item *cb_item = (struct result_cb_item *)user_data;
354
355         if (cb_item == NULL) {
356                 /* LCOV_EXCL_START */
357                 SHORTCUT_ERR("Failed to get a callback item");
358                 return;
359                 /* LCOV_EXCL_STOP */
360         }
361
362         reply = g_dbus_connection_send_message_with_reply_finish(
363                         connection,
364                         res,
365                         &err);
366
367         if (!reply) {
368                 /* LCOV_EXCL_START */
369                 if (err != NULL) {
370                         SHORTCUT_ERR("No reply. error = %s", err->message);
371                         g_error_free(err);
372                 }
373                 result = SHORTCUT_ERROR_COMM;
374                 /* LCOV_EXCL_STOP */
375         }
376
377         if (g_dbus_message_to_gerror(reply, &err)) {
378                 /* LCOV_EXCL_START */
379                 if (err->code == G_DBUS_ERROR_ACCESS_DENIED)
380                         result = SHORTCUT_ERROR_PERMISSION_DENIED;
381                 else
382                         result = err->code;
383
384                 SHORTCUT_ERR("_send_message_with_reply_sync_cb error %s err code: %d", err->message, result);
385                 g_error_free(err);
386                 /* LCOV_EXCL_STOP */
387         } else {
388                 body = g_dbus_message_get_body(reply);
389                 g_variant_get(body, "(i)", &result);
390         }
391
392         if (cb_item->result_internal_cb)
393                 cb_item->result_internal_cb(result, getpid(), cb_item->data); /* LCOV_EXCL_LINE */
394         else if (cb_item->result_cb)
395                 cb_item->result_cb(result, cb_item->data);
396
397         if (reply)
398                 g_object_unref(reply);
399
400         free(cb_item);
401 }
402
403 int _send_async_shortcut(GVariant *body, struct result_cb_item *cb_item, char *cmd)
404 {
405         GDBusMessage *msg;
406
407         msg = g_dbus_message_new_method_call(
408                         PROVIDER_BUS_NAME,
409                         PROVIDER_OBJECT_PATH,
410                         PROVIDER_SHORTCUT_INTERFACE_NAME,
411                         cmd);
412         if (!msg) {
413                 /* LCOV_EXCL_START */
414                 SHORTCUT_ERR("Can't allocate new method call");
415                 return SHORTCUT_ERROR_OUT_OF_MEMORY;
416                 /* LCOV_EXCL_STOP */
417         }
418
419         if (g_variant_is_floating(body))
420                 g_variant_ref(body);
421
422         if (body != NULL)
423                 g_dbus_message_set_body(msg, body);
424
425         g_dbus_connection_send_message_with_reply(
426                         _gdbus_conn,
427                         msg,
428                         G_DBUS_SEND_MESSAGE_FLAGS_NONE,
429                         -1,
430                         NULL,
431                         NULL,
432                         (GAsyncReadyCallback)_send_message_with_reply_sync_cb,
433                         cb_item);
434
435         if (msg)
436                 g_object_unref(msg);
437
438         SHORTCUT_DBG("_send_async_shortcut done !!");
439         return SHORTCUT_ERROR_NONE;
440 }
441
442 int _check_privilege(void)
443 {
444         GDBusMessage *reply = NULL;
445         int ret = SHORTCUT_ERROR_NONE;
446
447         ret = _send_sync_shortcut(NULL, &reply, "check_privilege");
448
449         if (reply)
450                 g_object_unref(reply);
451
452         return ret;
453 }
454
455
456 /* LCOV_EXCL_START */
457 static void _on_name_appeared(GDBusConnection *connection,
458                 const gchar     *name,
459                 const gchar     *name_owner,
460                 gpointer         user_data)
461 {
462         SHORTCUT_DBG("name appeared : %s", name);
463         _send_service_register();
464 }
465 /* LCOV_EXCL_STOP */
466
467 /* LCOV_EXCL_START */
468 static void _on_name_vanished(GDBusConnection *connection,
469                 const gchar     *name,
470                 gpointer         user_data)
471 {
472         SHORTCUT_DBG("name vanished : %s", name);
473 }
474 /* LCOV_EXCL_STOP */
475
476 void _ipc_monitor_fini(void)
477 {
478         if (provider_monitor_id) {
479                 g_bus_unwatch_name(provider_monitor_id);
480                 provider_monitor_id = 0;
481         }
482
483         if (monitor_id) {
484                 g_dbus_connection_signal_unsubscribe(_gdbus_conn, monitor_id);
485                 monitor_id = 0;
486         }
487
488         if (_gdbus_conn) {
489                 g_object_unref(_gdbus_conn);
490                 _gdbus_conn = NULL;
491         }
492
493 }
494
495 void _set_request_cb(shortcut_request_cb request_cb, void *data)
496 {
497         _request_callback_info.request_cb = request_cb;
498         _request_callback_info.data = data;
499 }
500
501 void _unset_request_cb(void)
502 {
503         if (_remove_callback_info.remove_cb == NULL &&
504                 _remove_callback_info.data == NULL)
505                 _ipc_monitor_fini();
506
507         _request_callback_info.request_cb = NULL;
508         _request_callback_info.data = NULL;
509 }
510
511 void _set_remove_cb(shortcut_remove_cb remove_cb, void *data)
512 {
513         _remove_callback_info.remove_cb = remove_cb;
514         _remove_callback_info.data = data;
515 }
516
517 void _unset_remove_cb(void)
518 {
519         if (_request_callback_info.request_cb == NULL &&
520                 _request_callback_info.data == NULL)
521                 _ipc_monitor_fini();
522
523         _remove_callback_info.remove_cb = NULL;
524         _remove_callback_info.data = NULL;
525 }
526
527 int _dbus_set_watch_name(void)
528 {
529         if (provider_monitor_id == 0) {
530                 provider_monitor_id = g_bus_watch_name_on_connection(
531                                 _gdbus_conn,
532                                 PROVIDER_BUS_NAME,
533                                 G_BUS_NAME_WATCHER_FLAGS_NONE,
534                                 _on_name_appeared,
535                                 _on_name_vanished,
536                                 NULL,
537                                 NULL);
538
539                 if (provider_monitor_id == 0) {
540                         /* LCOV_EXCL_START */
541                         SHORTCUT_ERR("watch on name fail");
542                         return SHORTCUT_ERROR_IO_ERROR;
543                         /* LCOV_EXCL_STOP */
544                 }
545         }
546
547         return SHORTCUT_ERROR_NONE;
548 }
549
550 char *_make_request_id(void)
551 {
552         static int id = 0;
553         char request_id[REQUEST_ID_LEN];
554
555         g_atomic_int_inc(&id);
556         snprintf(request_id, sizeof(request_id), "%d@%d", getpid(), id);
557
558         SHORTCUT_DBG("The request_id of shortcut is [%s]", request_id);
559
560         return strdup(request_id);
561 }
562
563 int _ready_to_send(char **appid, char **request_id)
564 {
565         int ret;
566
567         ret = _dbus_init();
568         if (ret != SHORTCUT_ERROR_NONE) {
569                 /* LCOV_EXCL_START */
570                 SHORTCUT_ERR("Can't init dbus %d", ret);
571                 return ret;
572                 /* LCOV_EXCL_STOP */
573         }
574
575         ret = _check_privilege();
576         if (ret != SHORTCUT_ERROR_NONE)
577                 return ret;
578
579         *appid = _shortcut_get_pkgname_by_pid();
580         if (*appid == NULL) {
581                 /* LCOV_EXCL_START */
582                 SHORTCUT_ERR("Can't get appid");
583                 return SHORTCUT_ERROR_IO_ERROR;
584                 /* LCOV_EXCL_STOP */
585         }
586
587         *request_id = _make_request_id();
588         if (*request_id == NULL) {
589                 SHORTCUT_ERR("Can't get request_id");
590                 free(*appid);
591                 *appid = NULL;
592                 return SHORTCUT_ERROR_OUT_OF_MEMORY;
593         }
594
595         return SHORTCUT_ERROR_NONE;
596 }
597
598 EAPI int shortcut_add_to_home_sync(const char *name, shortcut_type type,
599                 const char *uri, const char *icon, int allow_duplicate)
600 {
601         int ret;
602         char *appid;
603         char *request_id = NULL;
604         GVariant *body;
605         GVariant *reply_body;
606         GDBusMessage *reply = NULL;
607
608         CHECK_SHORTCUT_FEATURE();
609
610         if (ADD_TO_HOME_IS_DYNAMICBOX(type)) {
611                 /* LCOV_EXCL_START */
612                 SHORTCUT_ERR("Invalid type used for adding a shortcut\n");
613                 return SHORTCUT_ERROR_INVALID_PARAMETER;
614                 /* LCOV_EXCL_STOP */
615         }
616
617         ret = _ready_to_send(&appid, &request_id);
618         if (ret != SHORTCUT_ERROR_NONE) {
619                 /* LCOV_EXCL_START */
620                 SHORTCUT_ERR("ready to send error [%d]", ret);
621                 return ret;
622                 /* LCOV_EXCL_STOP */
623         }
624
625         if (!name)
626                 name = "";
627
628         if (!uri)
629                 uri = "";
630
631         if (!icon)
632                 icon = "";
633
634         body = g_variant_new("(sississi)", request_id, getpid(), appid, name,
635                         type, uri, icon, allow_duplicate);
636
637         ret = _send_sync_shortcut(body, &reply, "add_shortcut");
638         if (ret == SHORTCUT_ERROR_NONE) {
639                 reply_body = g_dbus_message_get_body(reply);
640                 g_variant_get(reply_body, "(i)", &ret);
641         }
642
643         if (appid)
644                 free(appid);
645         if (body)
646                 g_variant_unref(body);
647         if (request_id)
648                 free(request_id);
649         if (reply)
650                 g_object_unref(reply);
651
652         SHORTCUT_DBG("result[%d]", ret);
653
654         return ret;
655 }
656
657 EAPI int shortcut_add_to_home_widget_sync(const char *name,
658                 shortcut_widget_size_e size, const char *widget_id,
659                 const char *icon, double period, int allow_duplicate)
660 {
661         int ret;
662         char *appid;
663         char *request_id = NULL;
664         GVariant *body;
665         GVariant *reply_body;
666         GDBusMessage *reply = NULL;
667
668         CHECK_SHORTCUT_FEATURE();
669
670         if (name == NULL) {
671                 SHORTCUT_ERR("AppID is null\n");
672                 return SHORTCUT_ERROR_INVALID_PARAMETER;
673         }
674
675         if (!SHORTCUT_IS_WIDGET_SIZE(size)) {
676                 /* LCOV_EXCL_START */
677                 SHORTCUT_ERR("Invalid type used for adding a widget\n");
678                 return SHORTCUT_ERROR_INVALID_PARAMETER;
679                 /* LCOV_EXCL_STOP */
680         }
681
682         ret = _ready_to_send(&appid, &request_id);
683         if (ret != SHORTCUT_ERROR_NONE) {
684                 /* LCOV_EXCL_START */
685                 SHORTCUT_ERR("ready to send error [%d]", ret);
686                 return ret;
687                 /* LCOV_EXCL_STOP */
688         }
689
690         body = g_variant_new("(sississdi)", request_id, getpid(), widget_id,
691                         name, size, NULL, icon, period, allow_duplicate);
692
693         ret = _send_sync_shortcut(body, &reply, "add_shortcut_widget");
694         if (ret == SHORTCUT_ERROR_NONE) {
695                 reply_body = g_dbus_message_get_body(reply);
696                 g_variant_get(reply_body, "(i)", &ret);
697         }
698
699         if (appid)
700                 free(appid);
701         if (body)
702                 g_variant_unref(body);
703         if (request_id)
704                 free(request_id);
705         if (reply)
706                 g_object_unref(reply);
707
708         SHORTCUT_DBG("result[%d]", ret);
709
710         return ret;
711 }
712
713 EAPI int shortcut_remove_from_home_sync(const char *name)
714 {
715         int ret;
716         char *appid;
717         char *request_id = NULL;
718         GVariant *body;
719         GVariant *reply_body;
720         GDBusMessage *reply = NULL;
721
722         CHECK_SHORTCUT_FEATURE();
723
724         if (name == NULL) {
725                 SHORTCUT_ERR("name is NULL.");
726                 return SHORTCUT_ERROR_INVALID_PARAMETER;
727         }
728
729         ret = _ready_to_send(&appid, &request_id);
730         if (ret != SHORTCUT_ERROR_NONE) {
731                 /* LCOV_EXCL_START */
732                 SHORTCUT_ERR("ready to send error [%d]", ret);
733                 return ret;
734                 /* LCOV_EXCL_STOP */
735         }
736
737         body = g_variant_new("(siss)", request_id, getpid(), appid, name);
738
739         ret = _send_sync_shortcut(body, &reply, "remove_shortcut");
740         if (ret == SHORTCUT_ERROR_NONE) {
741                 reply_body = g_dbus_message_get_body(reply);
742                 g_variant_get(reply_body, "(i)", &ret);
743         }
744
745         if (appid)
746                 free(appid);
747         if (body)
748                 g_variant_unref(body);
749         if (request_id)
750                 free(request_id);
751         if (reply)
752                 g_object_unref(reply);
753
754         SHORTCUT_DBG("result[%d]", ret);
755
756         return ret;
757 }