2 * Copyright (c) 2011 Samsung Electronics Co., Ltd All Rights Reserved
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
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
21 #include <mime_type.h>
26 #include "bluetooth-api.h"
27 #include "bt-internal-types.h"
29 #include "bt-service-common.h"
30 #include "bt-service-event.h"
31 #include "bt-service-util.h"
32 #include "bt-service-opp-client.h"
33 #include "bt-service-obex-agent.h"
34 #include "bt-service-adapter.h"
36 #define BT_MIME_TYPE_MAX_LEN 20
38 static GSList *transfer_list = NULL;
39 bt_sending_info_t *sending_info;
40 static int file_offset = 0;
42 #define DBUS_TIEMOUT 20 * 1000 /* 20 Seconds */
43 static gboolean __bt_sending_release();
44 static int _bt_remove_session();
46 static int __bt_opp_client_start_sending(int request_id, char *address,
47 char **file_name_array, int file_count);
49 static GQuark __bt_opc_error_quark(void)
51 static GQuark quark = 0;
53 quark = g_quark_from_static_string("agent");
58 static void __bt_free_transfer_info(bt_transfer_info_t *info)
63 g_object_unref(info->proxy);
65 if (info->properties_proxy)
66 g_object_unref(info->properties_proxy);
69 g_free(info->transfer_name);
70 g_free(info->file_name);
74 static void __bt_free_sending_info(bt_sending_info_t *info)
78 /* Free the sending variable */
79 __bt_free_transfer_info(info->transfer_info);
81 g_free(info->file_name_array);
83 g_free(info->address);
87 static gboolean __bt_cancel_push_cb(gpointer data)
91 int result = BLUETOOTH_ERROR_CANCEL_BY_USER;
92 GVariant *param = NULL;
93 retv_if(sending_info == NULL, FALSE);
94 sending_info->result = result;
96 param = g_variant_new("(isi)", result,
97 sending_info->address,
98 sending_info->request_id);
99 /* Send the event in only error none case */
100 _bt_send_event(BT_OPP_CLIENT_EVENT,
101 BLUETOOTH_EVENT_OPC_CONNECTED,
103 __bt_free_sending_info(sending_info);
106 _bt_opp_client_event_deinit();
108 BT_DBG("Length of transfer list is %d", g_slist_length(transfer_list));
110 /*Operate remain works*/
111 if (g_slist_length(transfer_list) > 0) {
112 bt_sending_data_t *node = NULL;
114 node = transfer_list->data;
116 BT_ERR("data is NULL");
120 transfer_list = g_slist_remove(transfer_list, node);
122 if (__bt_opp_client_start_sending(node->request_id,
123 node->address, node->file_path,
124 node->file_count) != BLUETOOTH_ERROR_NONE) {
125 BT_ERR("Fail to start sending");
132 gboolean _bt_obex_client_progress(const char *transfer_path, guint64 transferred)
136 int percentage_progress;
137 int previous_progress;
139 int result = BLUETOOTH_ERROR_NONE;
140 GVariant *param = NULL;
141 retv_if(sending_info == NULL, TRUE);
142 retv_if(sending_info->transfer_info == NULL, TRUE);
144 if (g_strcmp0(sending_info->transfer_info->transfer_path,
145 transfer_path) != 0) {
146 BT_INFO("Path mismatch, previous transfer failed! Returning");
150 size = sending_info->transfer_info->size;
152 percentage_progress = (int)(((gdouble)transferred /(gdouble)size) * 100);
154 percentage_progress = 0;
156 sending_info->transfer_info->transfer_status = BT_TRANSFER_STATUS_PROGRESS;
157 sending_info->result = result;
159 previous_progress = (int)(((gdouble)sending_info->transfer_info->progress /(gdouble)size) * 100);
160 if (percentage_progress == previous_progress &&
161 sending_info->transfer_info->progress) {
162 sending_info->transfer_info->progress = transferred;
165 BT_DBG("Sending progress [prev %d] [curr %d]",
166 previous_progress, percentage_progress);
168 sending_info->transfer_info->progress = transferred;
170 /* Send the event in only error none case */
171 param = g_variant_new("(istii)", result,
172 sending_info->transfer_info->file_name,
173 sending_info->transfer_info->size,
175 sending_info->request_id);
178 _bt_send_event(BT_OPP_CLIENT_EVENT,
179 BLUETOOTH_EVENT_OPC_TRANSFER_PROGRESS,
186 gboolean _bt_obex_client_completed(const char *transfer_path, gboolean success)
190 int result = BLUETOOTH_ERROR_NONE;
191 GVariant *param = NULL;
192 retv_if(sending_info == NULL, TRUE);
193 retv_if(sending_info->transfer_info == NULL, TRUE);
195 if (g_strcmp0(sending_info->transfer_info->transfer_path,
196 transfer_path) != 0) {
197 BT_INFO("Path mismatch, previous transfer failed! Returning");
201 result = (success == TRUE) ? BLUETOOTH_ERROR_NONE : BLUETOOTH_ERROR_CANCEL;
203 sending_info->transfer_info->transfer_status = BT_TRANSFER_STATUS_COMPLETED;
204 sending_info->result = result;
206 if (!success) { /*In case of remote device reject, we need to send BLUETOOTH_EVENT_OPC_DISCONNECTED */
207 BT_DBG("completed with error");
208 if (!sending_info->is_canceled) {
209 param = g_variant_new("(issti)", result,
210 sending_info->address,
211 sending_info->transfer_info->file_name,
212 sending_info->transfer_info->size,
213 sending_info->request_id);
214 _bt_send_event(BT_OPP_CLIENT_EVENT,
215 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
217 __bt_free_transfer_info(sending_info->transfer_info);
218 sending_info->transfer_info = NULL;
219 /* Reset the file offset as we will cancelled remaining files also */
222 param = g_variant_new("(isi)", sending_info->result,
223 sending_info->address,
224 sending_info->request_id);
225 _bt_send_event(BT_OPP_CLIENT_EVENT,
226 BLUETOOTH_EVENT_OPC_DISCONNECTED,
228 __bt_sending_release();
229 /* Sending info should not freed after sending_release it's
230 * already freed in that API and if any pending request is
231 * present then it recreate sending_info again.
232 * And if we free it here then CreateSession method call will
233 * made but RemoveSession method call will not done.
236 BT_DBG("complete success");
237 /* Send the event in only error none case */
238 param = g_variant_new("(issti)", result,
239 sending_info->address,
240 sending_info->transfer_info->file_name,
241 sending_info->transfer_info->size,
242 sending_info->request_id);
243 _bt_send_event(BT_OPP_CLIENT_EVENT,
244 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
246 __bt_free_transfer_info(sending_info->transfer_info);
247 sending_info->transfer_info = NULL;
255 gboolean _bt_obex_client_started(const char *transfer_path)
259 int result = BLUETOOTH_ERROR_NONE;
260 GError *error = NULL;
261 GVariant *param = NULL;
262 GDBusConnection *g_conn;
263 GDBusProxy *properties_proxy;
264 GDBusProxy *transfer_proxy;
266 if (sending_info == NULL || sending_info->is_canceled == TRUE) {
267 result = BLUETOOTH_ERROR_CANCEL_BY_USER;
271 /* Get the session bus. */
272 g_conn = _bt_get_session_gconn();
273 retv_if(g_conn == NULL, FALSE);
274 properties_proxy = g_dbus_proxy_new_sync(g_conn, G_DBUS_PROXY_FLAGS_NONE,
275 NULL, BT_OBEXD_DBUS_NAME,
276 transfer_path, BT_PROPERTIES_INTERFACE,
279 retv_if(properties_proxy == NULL, FALSE);
281 sending_info->transfer_info->properties_proxy = properties_proxy;
283 transfer_proxy = g_dbus_proxy_new_sync(g_conn, G_DBUS_PROXY_FLAGS_NONE,
284 NULL, BT_OBEXD_DBUS_NAME,
285 transfer_path, BT_OBEX_TRANSFER_INTERFACE,
288 retv_if(transfer_proxy == NULL, FALSE);
290 sending_info->transfer_info->proxy = transfer_proxy;
292 sending_info->transfer_info->transfer_status = BT_TRANSFER_STATUS_STARTED;
293 sending_info->result = result;
295 param = g_variant_new("(issti)", result,
296 sending_info->address,
297 sending_info->transfer_info->file_name,
298 sending_info->transfer_info->size,
299 sending_info->request_id);
300 _bt_send_event(BT_OPP_CLIENT_EVENT,
301 BLUETOOTH_EVENT_OPC_TRANSFER_STARTED,
307 error = g_error_new(__bt_opc_error_quark(), BT_OBEX_AGENT_ERROR_CANCEL,
316 static void __bt_free_sending_data(gpointer data)
319 bt_sending_data_t *info = data;
321 ret_if(info == NULL);
323 for (i = 0; i < info->file_count; i++) {
324 g_free(info->file_path[i]);
327 _bt_delete_request_id(info->request_id);
329 g_free(info->file_path);
330 g_free(info->address);
334 static void __bt_sending_release_cb(GDBusProxy *proxy,
335 GAsyncResult *res, gpointer user_data)
338 ret_if(sending_info == NULL);
340 GError *error = NULL;
341 int result = BLUETOOTH_ERROR_NONE;
342 GVariant *param = NULL;
343 g_dbus_proxy_call_finish(proxy, res, &error);
345 g_object_unref(proxy);
348 BT_ERR("%s", error->message);
351 result = BLUETOOTH_ERROR_INTERNAL;
354 BT_DBG("Session Removed");
357 sending_info->result = result;
358 param = g_variant_new("(isi)", sending_info->result,
359 sending_info->address,
360 sending_info->request_id);
361 /* Send the event in only error none case */
362 _bt_send_event(BT_OPP_CLIENT_EVENT,
363 BLUETOOTH_EVENT_OPC_DISCONNECTED,
366 __bt_free_sending_info(sending_info);
369 _bt_opp_client_event_deinit();
371 /* Operate remain works */
372 if (g_slist_length(transfer_list) > 0) {
373 bt_sending_data_t *data = NULL;
375 data = transfer_list->data;
379 transfer_list = g_slist_remove(transfer_list, data);
381 BT_DBG("calling __bt_opp_client_start_sending");
383 if (__bt_opp_client_start_sending(data->request_id,
384 data->address, data->file_path,
385 data->file_count) != BLUETOOTH_ERROR_NONE) {
392 g_slist_free_full(transfer_list,
393 (GDestroyNotify)__bt_free_sending_data);
394 transfer_list = NULL;
401 static int _bt_remove_session()
403 GDBusConnection *g_conn;
404 GDBusProxy *session_proxy;
407 g_conn = _bt_get_session_gconn();
408 retv_if(g_conn == NULL, BLUETOOTH_ERROR_INTERNAL);
409 retv_if(sending_info->session_path == NULL, BLUETOOTH_ERROR_INVALID_PARAM);
411 session_proxy = g_dbus_proxy_new_sync(g_conn, G_DBUS_PROXY_FLAGS_NONE,
412 NULL, BT_OBEXD_DBUS_NAME,
414 BT_OBEX_CLIENT_INTERFACE,
417 retv_if(session_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
419 g_dbus_proxy_call(session_proxy, "RemoveSession",
420 g_variant_new("(o)", sending_info->session_path),
421 G_DBUS_CALL_FLAGS_NONE,
423 (GAsyncReadyCallback)__bt_sending_release_cb,
426 return BLUETOOTH_ERROR_NONE;
429 static gboolean __bt_sending_release()
433 retv_if(sending_info == NULL, FALSE);
435 retv_if(_bt_remove_session() != BLUETOOTH_ERROR_NONE, FALSE);
441 void _bt_opc_disconnected(const char *session_path)
444 GVariant *param = NULL;
445 ret_if(sending_info == NULL);
447 if (g_strcmp0(sending_info->session_path,
448 session_path) != 0) {
449 BT_INFO("Path mismatch, previous transfer failed! Returning");
453 if (sending_info->transfer_info) {
454 if (sending_info->transfer_info->transfer_status == BT_TRANSFER_STATUS_PROGRESS ||
455 sending_info->transfer_info->transfer_status == BT_TRANSFER_STATUS_STARTED) {
456 BT_INFO("Abnormal termination");
457 param = g_variant_new("(issti)", sending_info->result,
458 sending_info->address,
459 sending_info->transfer_info->file_name,
460 sending_info->transfer_info->size,
461 sending_info->request_id);
462 _bt_send_event(BT_OPP_CLIENT_EVENT,
463 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
465 __bt_free_transfer_info(sending_info->transfer_info);
468 param = g_variant_new("(isi)", sending_info->result,
469 sending_info->address,
470 sending_info->request_id);
471 _bt_send_event(BT_OPP_CLIENT_EVENT,
472 BLUETOOTH_EVENT_OPC_DISCONNECTED,
475 __bt_free_sending_info(sending_info);
481 static void __bt_send_file_cb(GDBusProxy *proxy,
482 GAsyncResult *res, gpointer user_data)
485 GVariant *value = NULL;
486 GError *error = NULL;
487 char *session_path = NULL;
488 const char *transfer_name = NULL;
489 const char *file_name = NULL;
491 GVariantIter *iter = NULL;
492 value = g_dbus_proxy_call_finish(proxy, res, &error);
494 g_dbus_error_strip_remote_error(error);
495 BT_ERR("%s", error->message);
496 /* If Obex is not able to open a file then continue with other if any */
497 if (g_strcmp0("Unable to open file", error->message) == 0) {
498 GVariant *param = NULL;
501 BT_ERR("Unable to open file [%s]", sending_info->file_name_array[file_offset]);
503 param = g_variant_new("(issti)", BLUETOOTH_ERROR_NOT_FOUND,
504 sending_info->address,
505 sending_info->file_name_array[file_offset],
507 sending_info->request_id);
508 _bt_send_event(BT_OPP_CLIENT_EVENT,
509 BLUETOOTH_EVENT_OPC_TRANSFER_STARTED,
512 param = g_variant_new("(issti)", BLUETOOTH_ERROR_NOT_FOUND,
513 sending_info->address,
514 sending_info->file_name_array[file_offset],
516 sending_info->request_id);
517 _bt_send_event(BT_OPP_CLIENT_EVENT,
518 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
522 g_object_unref(proxy);
529 g_object_unref(proxy);
532 g_variant_get(value, "(oa{sv})", &session_path, &iter);
533 g_variant_unref(value);
536 __bt_free_transfer_info(sending_info->transfer_info);
538 sending_info->transfer_info = g_malloc0(sizeof(bt_transfer_info_t));
544 while (g_variant_iter_loop(iter, "{sv}", &key, &val)) {
545 if (g_strcmp0(key, "Name") == 0) {
546 transfer_name = g_variant_dup_string(val, &len);
547 } else if (g_strcmp0(key, "Filename") == 0) {
548 file_name = g_variant_dup_string(val, &len);
549 } else if (g_strcmp0(key, "Size") == 0) {
550 size = g_variant_get_uint64(val);
553 g_variant_iter_free(iter);
556 sending_info->transfer_info->transfer_name = g_strdup(transfer_name);
557 sending_info->transfer_info->file_name = g_strdup(file_name);
558 sending_info->transfer_info->size = size;
559 sending_info->transfer_info->progress = 0;
560 sending_info->transfer_info->transfer_path = session_path;
561 sending_info->transfer_info->transfer_status = BT_TRANSFER_STATUS_QUEUED;
562 sending_info->result = BLUETOOTH_ERROR_NONE;
566 void _bt_sending_files(void)
571 GDBusConnection *g_conn;
572 GDBusProxy *client_proxy;
573 char mime_type[BT_MIME_TYPE_MAX_LEN + 1] = { 0 };
575 if (sending_info == NULL)
577 if (file_offset < sending_info->file_count) {
578 /* Get the session bus. */
579 g_conn = _bt_get_session_gconn();
580 ret_if(g_conn == NULL);
582 client_proxy = g_dbus_proxy_new_sync(g_conn, G_DBUS_PROXY_FLAGS_NONE,
583 NULL, BT_OBEXD_DBUS_NAME,
584 sending_info->session_path,
585 BT_OBEX_OBJECT_PUSH_INTERFACE,
587 ret_if(client_proxy == NULL);
588 if (aul_get_mime_from_file(sending_info->file_name_array[file_offset],
589 mime_type, BT_MIME_TYPE_MAX_LEN) == AUL_R_OK) {
590 BT_DBG("MLME type = %s", mime_type);
592 /* For IOPT compliance, change "text/x-iMelody" to "audio/imelody"
593 * because few devices(multimedia players) reject the OPP put for text objects
594 * since they support only multimedia files exchange */
595 if(!strcasecmp(mime_type, "text/x-iMelody")) {
596 strncpy(mime_type, "audio/imelody", BT_MIME_TYPE_MAX_LEN);
597 BT_DBG("over writing mime type to = %s", mime_type);
599 if(!strcasecmp(mime_type, "text/vcard")) {
600 strncpy(mime_type, "text/x-vcard", BT_MIME_TYPE_MAX_LEN);
601 BT_DBG("over writing mime type to = %s", mime_type);
605 BT_DBG("Calling SendFile");
606 g_dbus_proxy_call(client_proxy, "SendFile",
607 g_variant_new("(ss)", sending_info->file_name_array[file_offset],
609 G_DBUS_CALL_FLAGS_NONE,
611 (GAsyncReadyCallback)__bt_send_file_cb,
614 BT_ERR("Calling SendFile failed: [%s]\n", err->message);
621 __bt_sending_release();
627 static void __bt_create_session_cb(GDBusProxy *proxy,
628 GAsyncResult *res, gpointer user_data)
632 GError *error = NULL;
634 int result = BLUETOOTH_ERROR_NONE;
635 char *session_path = NULL;
636 GVariant *param = NULL;
638 value = g_dbus_proxy_call_finish(proxy, res, &error);
640 g_variant_get(value, "(o)", &session_path);
641 g_variant_unref(value);
645 BT_ERR("%s", error->message);
646 g_clear_error(&error);
648 result = BLUETOOTH_ERROR_INTERNAL;
650 BT_DBG("Session created");
651 if (sending_info != NULL)
652 sending_info->session_path = g_strdup(session_path);
654 g_free(session_path);
655 g_object_unref(proxy);
656 ret_if(sending_info == NULL);
658 sending_info->result = result;
659 param = g_variant_new("(isi)", result,
660 sending_info->address,
661 sending_info->request_id);
662 /* Send the event in only error none case */
663 _bt_send_event(BT_OPP_CLIENT_EVENT,
664 BLUETOOTH_EVENT_OPC_CONNECTED,
667 if (result != BLUETOOTH_ERROR_NONE) {
668 BT_ERR("Calling __bt_sending_release");
669 gboolean ret = __bt_sending_release();
671 __bt_free_sending_info(sending_info);
675 BT_DBG("ReleaseSession Not called");
676 /* Operate remain works */
677 if (g_slist_length(transfer_list) > 0) {
678 bt_sending_data_t *data = NULL;
680 data = transfer_list->data;
681 ret_if(data == NULL);
683 transfer_list = g_slist_remove(transfer_list, data);
685 BT_DBG("calling __bt_opp_client_start_sending");
687 if (__bt_opp_client_start_sending(data->request_id,
688 data->address, data->file_path,
689 data->file_count) != BLUETOOTH_ERROR_NONE) {
690 BT_ERR("Sending Enqueued Transfer Failed");
695 BT_DBG("Calling sending_files");
702 static int __bt_opp_client_start_sending(int request_id, char *address,
703 char **file_name_array, int file_count)
705 GVariantBuilder *builder;
707 GDBusConnection *g_conn;
708 GDBusProxy *client_proxy;
709 GError *error = NULL;
712 BT_CHECK_PARAMETER(address, return);
713 BT_CHECK_PARAMETER(file_name_array, return);
715 /* Get the session bus. */
716 g_conn = _bt_get_session_gconn();
717 retv_if(g_conn == NULL, BLUETOOTH_ERROR_INTERNAL);
719 client_proxy = g_dbus_proxy_new_sync(g_conn, G_DBUS_PROXY_FLAGS_NONE,
720 NULL, BT_OBEX_SERVICE_NAME,
722 BT_OBEX_CLIENT_INTERFACE,
726 BT_ERR("Unable to create client proxy: %s", error->message);
727 g_clear_error(&error);
730 retv_if(client_proxy == NULL, BLUETOOTH_ERROR_INTERNAL);
732 builder = g_variant_builder_new(
733 G_VARIANT_TYPE("a{sv}"));
735 g_variant_builder_add(builder, "{sv}", "Target",
736 g_variant_new_string("OPP"));
738 __bt_free_sending_info(sending_info);
740 sending_info = g_malloc0(sizeof(bt_sending_info_t));
741 sending_info->address = g_strdup(address);
742 sending_info->request_id = request_id;
744 sending_info->file_count = file_count;
745 sending_info->file_offset = 0;
746 sending_info->file_name_array = g_new0(char *, file_count + 1);
748 for (i = 0; i < file_count; i++) {
749 sending_info->file_name_array[i] = g_strdup(file_name_array[i]);
750 BT_DBG("file[%d]: %s", i, sending_info->file_name_array[i]);
753 _bt_opp_client_event_deinit();
754 _bt_opp_client_event_init();
755 //_bt_obex_client_started(agent_path);
757 BT_DBG("Adapter Status %d", _bt_adapter_get_status());
758 if (_bt_adapter_get_status() == BT_ACTIVATED) {
759 BT_DBG("Going to call CreateSession");
760 g_dbus_proxy_call(client_proxy, "CreateSession",
761 g_variant_new("(sa{sv})", address, builder),
762 G_DBUS_CALL_FLAGS_NONE,
764 (GAsyncReadyCallback)__bt_create_session_cb,
767 GVariant *param = g_variant_new("(isi)", BLUETOOTH_ERROR_INTERNAL,
768 sending_info->address, sending_info->request_id);
770 BT_DBG("Address[%s] RequestID[%d]", sending_info->address, sending_info->request_id);
771 _bt_send_event(BT_OPP_CLIENT_EVENT, BLUETOOTH_EVENT_OPC_CONNECTED,
773 __bt_free_sending_info(sending_info);
776 g_variant_builder_unref(builder);
780 return BLUETOOTH_ERROR_NONE;
783 int _bt_opp_client_push_files(int request_id, GDBusMethodInvocation *context,
784 bluetooth_device_address_t *remote_address,
785 char **file_path, int file_count)
788 char address[BT_ADDRESS_STRING_SIZE] = { 0 };
789 bt_sending_data_t *data;
791 GVariant *out_param1 = NULL;
793 int result = BLUETOOTH_ERROR_NONE;
796 BT_CHECK_PARAMETER(remote_address, return);
797 BT_CHECK_PARAMETER(file_path, return);
799 /* Implement the queue */
800 _bt_convert_addr_type_to_string(address, remote_address->addr);
802 if (sending_info == NULL) {
803 result = __bt_opp_client_start_sending(request_id,
804 address, file_path, file_count);
805 if (result != BLUETOOTH_ERROR_NONE)
808 /* Insert data in the queue */
809 data = g_malloc0(sizeof(bt_sending_data_t));
811 return BLUETOOTH_ERROR_MEMORY_ALLOCATION;
813 data->file_path = g_new0(char *, file_count + 1);
814 data->address = g_strdup(address);
815 data->file_count = file_count;
816 data->request_id = request_id;
818 for (i = 0; i < file_count; i++) {
819 data->file_path[i] = g_strdup(file_path[i]);
820 DBG_SECURE("file[%d]: %s", i, data->file_path[i]);
823 transfer_list = g_slist_append(transfer_list, data);
826 out_param1 = g_variant_new_from_data((const GVariantType *)"ay",
827 &request_id, sizeof(int),
831 g_dbus_method_invocation_return_value(context,
832 g_variant_new("(iv)", result, out_param1));
839 int _bt_opp_client_cancel_push(void)
844 int result = BLUETOOTH_ERROR_CANCEL_BY_USER;
845 GVariant *ret = NULL;
846 GVariant *param = NULL;
847 retv_if(sending_info == NULL, BLUETOOTH_ERROR_NOT_IN_OPERATION);
849 sending_info->is_canceled = TRUE;
850 sending_info->result = result;
852 if (sending_info->transfer_info) {
854 ret = g_dbus_proxy_call_sync(sending_info->transfer_info->proxy,
856 G_DBUS_CALL_FLAGS_NONE, -1,
860 BT_ERR("Cancel Error: %s\n", err->message);
864 g_variant_unref(ret);
867 param = g_variant_new("(issti)", result,
868 sending_info->address,
869 sending_info->transfer_info->file_name,
870 sending_info->transfer_info->size,
871 sending_info->request_id);
872 _bt_send_event(BT_OPP_CLIENT_EVENT,
873 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
876 if (result == BLUETOOTH_ERROR_CANCEL_BY_USER) {
877 BT_ERR("result is not BLUETOOTH_ERROR_NONE");
878 __bt_sending_release();
883 g_idle_add(__bt_cancel_push_cb, NULL);
888 return BLUETOOTH_ERROR_NONE;
891 int _bt_opp_client_cancel_all_transfers(void)
895 g_slist_free_full(transfer_list,
896 (GDestroyNotify)__bt_free_sending_data);
898 transfer_list = NULL;
901 _bt_opp_client_cancel_push();
903 return BLUETOOTH_ERROR_NONE;
906 int _bt_opp_client_is_sending(gboolean *sending)
908 BT_CHECK_PARAMETER(sending, return);
910 *sending = sending_info ? TRUE : FALSE;
912 return BLUETOOTH_ERROR_NONE;
915 void _bt_opp_client_check_pending_transfer(const char *address)
919 int result = BLUETOOTH_ERROR_CANCEL;
920 GVariant *param = NULL;
921 ret_if(sending_info == NULL);
922 ret_if(sending_info->transfer_info == NULL);
924 if (g_strcmp0(sending_info->address, address) == 0) {
925 BT_INFO("Address Match.Cancel current transfer");
926 sending_info->transfer_info->transfer_status = BT_TRANSFER_STATUS_COMPLETED;
927 sending_info->result = result;
929 if (!sending_info->is_canceled) {
930 param = g_variant_new("(issti)", result,
931 sending_info->address,
932 sending_info->transfer_info->file_name,
933 sending_info->transfer_info->size,
934 sending_info->request_id);
935 _bt_send_event(BT_OPP_CLIENT_EVENT,
936 BLUETOOTH_EVENT_OPC_TRANSFER_COMPLETE,
938 __bt_free_transfer_info(sending_info->transfer_info);
939 sending_info->transfer_info = NULL;
940 /* Reset the file offset as we will cancelled remaining files also */
943 param = g_variant_new("(isi)", sending_info->result,
944 sending_info->address,
945 sending_info->request_id);
946 _bt_send_event(BT_OPP_CLIENT_EVENT,
947 BLUETOOTH_EVENT_OPC_DISCONNECTED,
950 __bt_sending_release();
955 int _bt_opp_get_client_progress(guint8 *progress)
957 if (sending_info == NULL || sending_info->transfer_info == NULL) {
958 BT_ERR("No Active Outbound transfer");
959 return BLUETOOTH_ERROR_NOT_FOUND;
962 *progress = (int)(((double)sending_info->transfer_info->progress /
963 sending_info->transfer_info->size) * 100);
965 BT_DBG("Percentage: %d", *progress);
966 return BLUETOOTH_ERROR_NONE;
969 void _bt_cancel_queued_transfers(void)
971 bt_sending_data_t *data = NULL;
972 GVariant *param = NULL;
974 BT_INFO("Cancel queued Transfers:: Length of transfer list is %d",
975 g_slist_length(transfer_list));
977 while (transfer_list) {
978 data = transfer_list->data;
979 param = g_variant_new("(isi)", BLUETOOTH_ERROR_INTERNAL,
980 data->address, data->request_id);
982 BT_DBG("Address[%s] RequestID[%d]", data->address, data->request_id);
983 _bt_send_event(BT_OPP_CLIENT_EVENT, BLUETOOTH_EVENT_OPC_CONNECTED,
986 transfer_list = g_slist_remove(transfer_list, data);