2012-02-20 12:49:32

by Jaganath Kanakkassery

[permalink] [raw]
Subject: [PATCH obexd] client: Fix ABORT command not sending when user cancels the transfer

Issue: When user cancels the tranfer, after queuing the ABORT command
transfer->callback() with error will be called. In that session_unref
is called which disconnects the transport.

Fix: Just send ABORT command during abort and transfer->callback()
will be called after getting response of ABORT command.
---
client/transfer.c | 12 ------------
gobex/gobex-transfer.c | 6 +++++-
2 files changed, 5 insertions(+), 13 deletions(-)

diff --git a/client/transfer.c b/client/transfer.c
index 157811d..88e8ef7 100644
--- a/client/transfer.c
+++ b/client/transfer.c
@@ -141,23 +141,11 @@ static DBusMessage *obc_transfer_get_properties(DBusConnection *connection,

static void obc_transfer_abort(struct obc_transfer *transfer)
{
- struct transfer_callback *callback = transfer->callback;
-
if (transfer->xfer == 0)
return;

g_obex_cancel_transfer(transfer->xfer);
transfer->xfer = 0;
-
- if (callback) {
- GError *err;
-
- err = g_error_new(OBC_TRANSFER_ERROR, -ECANCELED, "%s",
- strerror(ECANCELED));
- callback->func(transfer, transfer->transferred, err,
- callback->data);
- g_error_free(err);
- }
}

static DBusMessage *obc_transfer_cancel(DBusConnection *connection,
diff --git a/gobex/gobex-transfer.c b/gobex/gobex-transfer.c
index c62a91e..706952f 100644
--- a/gobex/gobex-transfer.c
+++ b/gobex/gobex-transfer.c
@@ -635,6 +635,10 @@ gboolean g_obex_cancel_transfer(guint id)
if (transfer == NULL)
return FALSE;

- transfer_free(transfer);
+ if (transfer->req_id > 0)
+ g_obex_cancel_req(transfer->obex, transfer->req_id, FALSE);
+ else
+ transfer_free(transfer);
+
return TRUE;
}
--
1.7.1



2012-02-20 14:02:40

by Jaganath Kanakkassery

[permalink] [raw]
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user cancels the transfer

Hi Mikel,

--------------------------------------------------
From: "Mikel Astiz" <[email protected]>
Sent: Monday, February 20, 2012 6:50 PM
To: "Jaganath Kanakkassery" <[email protected]>
Cc: <[email protected]>
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user
cancels the transfer

> Hi Jaganath,
>
>> Issue: When user cancels the tranfer, after queuing the ABORT command
>> transfer->callback() with error will be called. In that session_unref
>> is called which disconnects the transport.
>>
>> Fix: Just send ABORT command during abort and transfer->callback()
>> will be called after getting response of ABORT command.
>> ---
>> client/transfer.c | 12 ------------
>> gobex/gobex-transfer.c | 6 +++++-
>> 2 files changed, 5 insertions(+), 13 deletions(-)
>>
>> diff --git a/client/transfer.c b/client/transfer.c
>> index 157811d..88e8ef7 100644
>> --- a/client/transfer.c
>> +++ b/client/transfer.c
>> @@ -141,23 +141,11 @@ static DBusMessage
>> *obc_transfer_get_properties(DBusConnection *connection,
>>
>> static void obc_transfer_abort(struct obc_transfer *transfer)
>> {
>> - struct transfer_callback *callback = transfer->callback;
>> -
>> if (transfer->xfer == 0)
>> return;
>>
>> g_obex_cancel_transfer(transfer->xfer);
>> transfer->xfer = 0;
>> -
>> - if (callback) {
>> - GError *err;
>> -
>> - err = g_error_new(OBC_TRANSFER_ERROR, -ECANCELED, "%s",
>> - strerror(ECANCELED));
>> - callback->func(transfer, transfer->transferred, err,
>> - callback->data);
>> - g_error_free(err);
>> - }
>> }
>>
>
> I think this callback is needed here, particularly for transfers that are
> queued but have not been started yet. This does not work right now

For queued transfer also reqid will be non zero. So in case if it is not
pending request response callback will be called in cancel_complete()
which is invoked in g_obex_cancel_req()

> but I was planning to propose a patch series to fix it soon. Also, IMO it
> seems fair to report the abortion as soon as possible to the upper layers,
> no matter what the response will be.
>
> Having said that, I agree that transfer canceling is broken right now.
>
> Cheers,
> Mikel
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-bluetooth"
> in
> the body of a message to [email protected]
> More majordomo info at http://vger.kernel.org/majordomo-info.html


2012-02-20 13:55:05

by Jaganath Kanakkassery

[permalink] [raw]
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user cancels the transfer

Hi Luiz.

--------------------------------------------------
From: "Luiz Augusto von Dentz" <[email protected]>
Sent: Monday, February 20, 2012 6:53 PM
To: "Jaganath Kanakkassery" <[email protected]>
Cc: <[email protected]>
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user
cancels the transfer

> Hi Jaganath,
>
> On Mon, Feb 20, 2012 at 2:49 PM, Jaganath Kanakkassery
> <[email protected]> wrote:
>> Issue: When user cancels the tranfer, after queuing the ABORT command
>> transfer->callback() with error will be called. In that session_unref
>> is called which disconnects the transport.
>
> I don't get the issue, you don't want to disconnect immediately if the
> transfer aborts and the last reference is released? IMO that is fine
> in most of the cases, btw by disconnecting it also serves as an ABORT.
User perspective this is fine but PTS requires ABORT command before
transport disconnection
> The only cases where we would not do it is if the session object is
> created, e.g. ftp, but if that happens than the module is responsible
> to hold a reference to the session preventing it to disconnect.
>
>> diff --git a/gobex/gobex-transfer.c b/gobex/gobex-transfer.c
>> index c62a91e..706952f 100644
>> --- a/gobex/gobex-transfer.c
>> +++ b/gobex/gobex-transfer.c
>> @@ -635,6 +635,10 @@ gboolean g_obex_cancel_transfer(guint id)
>> if (transfer == NULL)
>> return FALSE;
>>
>> - transfer_free(transfer);
>> + if (transfer->req_id > 0)
>> + g_obex_cancel_req(transfer->obex, transfer->req_id,
>> FALSE);
>> + else
>> + transfer_free(transfer);
>> +
>> return TRUE;
>> }
>
> Btw, when you changing gobex please send its changes separately
> starting with gobex:, so not that after g_obex_cancel_transfer the id
> and callback should no longer be accessible so the application should
> be able to free its resources, so if we call the callback latter it
> will very likely cause a crash.
I don't get you. In g_obex_cancel_transfer() it just sends abort command.
Transfer will be freed in transfer_complete().
>
> --
> Luiz Augusto von Dentz


2012-02-20 13:23:56

by Luiz Augusto von Dentz

[permalink] [raw]
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user cancels the transfer

Hi Jaganath,

On Mon, Feb 20, 2012 at 2:49 PM, Jaganath Kanakkassery
<[email protected]> wrote:
> Issue: When user cancels the tranfer, after queuing the ABORT command
> transfer->callback() with error will be called. In that session_unref
> is called which disconnects the transport.

I don't get the issue, you don't want to disconnect immediately if the
transfer aborts and the last reference is released? IMO that is fine
in most of the cases, btw by disconnecting it also serves as an ABORT.
The only cases where we would not do it is if the session object is
created, e.g. ftp, but if that happens than the module is responsible
to hold a reference to the session preventing it to disconnect.

> diff --git a/gobex/gobex-transfer.c b/gobex/gobex-transfer.c
> index c62a91e..706952f 100644
> --- a/gobex/gobex-transfer.c
> +++ b/gobex/gobex-transfer.c
> @@ -635,6 +635,10 @@ gboolean g_obex_cancel_transfer(guint id)
> ? ? ? ?if (transfer == NULL)
> ? ? ? ? ? ? ? ?return FALSE;
>
> - ? ? ? transfer_free(transfer);
> + ? ? ? if (transfer->req_id > 0)
> + ? ? ? ? ? ? ? g_obex_cancel_req(transfer->obex, transfer->req_id, FALSE);
> + ? ? ? else
> + ? ? ? ? ? ? ? transfer_free(transfer);
> +
> ? ? ? ?return TRUE;
> ?}

Btw, when you changing gobex please send its changes separately
starting with gobex:, so not that after g_obex_cancel_transfer the id
and callback should no longer be accessible so the application should
be able to free its resources, so if we call the callback latter it
will very likely cause a crash.

--
Luiz Augusto von Dentz

2012-02-20 13:20:57

by Mikel Astiz

[permalink] [raw]
Subject: Re: [PATCH obexd] client: Fix ABORT command not sending when user cancels the transfer

Hi Jaganath,

> Issue: When user cancels the tranfer, after queuing the ABORT command
> transfer->callback() with error will be called. In that session_unref
> is called which disconnects the transport.
>
> Fix: Just send ABORT command during abort and transfer->callback()
> will be called after getting response of ABORT command.
> ---
> client/transfer.c | 12 ------------
> gobex/gobex-transfer.c | 6 +++++-
> 2 files changed, 5 insertions(+), 13 deletions(-)
>
> diff --git a/client/transfer.c b/client/transfer.c
> index 157811d..88e8ef7 100644
> --- a/client/transfer.c
> +++ b/client/transfer.c
> @@ -141,23 +141,11 @@ static DBusMessage *obc_transfer_get_properties(DBusConnection *connection,
>
> static void obc_transfer_abort(struct obc_transfer *transfer)
> {
> - struct transfer_callback *callback = transfer->callback;
> -
> if (transfer->xfer == 0)
> return;
>
> g_obex_cancel_transfer(transfer->xfer);
> transfer->xfer = 0;
> -
> - if (callback) {
> - GError *err;
> -
> - err = g_error_new(OBC_TRANSFER_ERROR, -ECANCELED, "%s",
> - strerror(ECANCELED));
> - callback->func(transfer, transfer->transferred, err,
> - callback->data);
> - g_error_free(err);
> - }
> }
>

I think this callback is needed here, particularly for transfers that
are queued but have not been started yet. This does not work right now
but I was planning to propose a patch series to fix it soon. Also, IMO
it seems fair to report the abortion as soon as possible to the upper
layers, no matter what the response will be.

Having said that, I agree that transfer canceling is broken right now.

Cheers,
Mikel