From: Mat Martineau <mathewm@codeaurora.org>
To: Gustavo Padovan <gustavo@padovan.org>
Cc: linux-bluetooth@vger.kernel.org
Subject: Re: [PATCH 3/4] Bluetooth: Add MSG_MORE support to L2CAP sockets
Date: Fri, 11 May 2012 11:41:55 -0700 (PDT) [thread overview]
Message-ID: <alpine.DEB.2.02.1205111139080.17554@mathewm-linux> (raw)
In-Reply-To: <alpine.DEB.2.02.1205111126420.17554@mathewm-linux>
Gustavo -
On Fri, 11 May 2012, Mat Martineau wrote:
>
> Hi Gustavo -
>
> On Fri, 11 May 2012, Gustavo Padovan wrote:
>
>> MSG_MORE enables us to save buffer space in userspace, the packet are
>> built directly in the kernel and sent when their size reaches the Output
>> MTU value.
>>
>> Only MSG_MORE for Basic Mode is supported right now. chan->skb_more keeps
>> a pointer to the L2CAP packet that is being build through many calls to
>> send().
>
> Could you explain more about how you expect it to work?
>
> I would assume the application would do a series of sends:
>
> send(fd, buf, len, MSG_MORE);
> ...
> send(fd, buf, len, MSG_MORE);
> ...
> send(fd, buf, len, MSG_MORE);
> ...
> send(fd, buf, len, 0);
>
> and the SDU would be sent the first time there is no MSG_MORE flag. If the
> MTU is exceeded, the SDU is not sent and an error is returned.
>
> What should happen if a send() with MSG_MORE completely fills an SDU (length
> of data sent is equal to MTU)? Does it make sense to treat it like a normal
> send, or return an error so that application knows that later calls with
> MSG_MORE will not append? Or does the full SDU not get sent, and a
> zero-length send() with no MSG_MORE would trigger the transmission?
>
>>
>> Signed-off-by: Gustavo Padovan <gustavo@padovan.org>
>> ---
>> include/net/bluetooth/l2cap.h | 2 +
>> net/bluetooth/l2cap_core.c | 116
>> ++++++++++++++++++++++++++++++++++++++---
>> 2 files changed, 110 insertions(+), 8 deletions(-)
>>
>> diff --git a/include/net/bluetooth/l2cap.h b/include/net/bluetooth/l2cap.h
>> index 1c7d1cd..5f2845d 100644
>> --- a/include/net/bluetooth/l2cap.h
>> +++ b/include/net/bluetooth/l2cap.h
>> @@ -520,6 +520,8 @@ struct l2cap_chan {
>> struct list_head list;
>> struct list_head global_l;
>>
>> + struct sk_buff *skb_more;
>> +
>> void *data;
>> struct l2cap_ops *ops;
>> struct mutex lock;
>> diff --git a/net/bluetooth/l2cap_core.c b/net/bluetooth/l2cap_core.c
>> index e5a4fd9..73bf8a8 100644
>> --- a/net/bluetooth/l2cap_core.c
>> +++ b/net/bluetooth/l2cap_core.c
>> @@ -1827,6 +1827,8 @@ static inline int l2cap_skbuff_fromiovec(struct
>> l2cap_chan *chan,
>> struct sk_buff **frag;
>> int sent = 0;
>>
>> + count = min_t(unsigned int, count, len);
>> +
>> if (memcpy_fromiovec(skb_put(skb, count), msg->msg_iov, count))
>> return -EFAULT;
>>
>> @@ -1903,9 +1905,13 @@ static struct sk_buff *l2cap_create_basic_pdu(struct
>> l2cap_chan *chan,
>> int err, count;
>> struct l2cap_hdr *lh;
>>
>> - BT_DBG("chan %p len %d", chan, (int)len);
>> + BT_DBG("chan %p len %d aclmtu %d" , chan, (int)len, conn->mtu);
>>
>> - count = min_t(unsigned int, (conn->mtu - L2CAP_HDR_SIZE), len);
>> + if (msg->msg_flags & MSG_MORE)
>> + count = min_t(unsigned int, (conn->mtu - L2CAP_HDR_SIZE),
>> + chan->omtu);
>> + else
>> + count = min_t(unsigned int, (conn->mtu - L2CAP_HDR_SIZE),
>> len);
>>
>> skb = chan->ops->alloc_skb(chan, count + L2CAP_HDR_SIZE,
>> msg->msg_flags & MSG_DONTWAIT);
>> @@ -2048,6 +2054,75 @@ static int l2cap_segment_sdu(struct l2cap_chan
>> *chan,
>> return err;
>> }
>>
>> +static int l2cap_append_pdu(struct l2cap_chan *chan, struct msghdr *msg,
>> + size_t len)
>> +{
>> + struct l2cap_conn *conn = chan->conn;
>> + struct sk_buff **frag, *skb = chan->skb_more;
>> + int sent = 0;
>> + unsigned int count;
>> + struct l2cap_hdr *lh;
>> +
>> + BT_DBG("chan %p len %ld", chan, len);
>> +
>> + frag = &skb_shinfo(skb)->frag_list;
>> + if (*frag)
>> + goto frags;
>
> I think this would be more readable without the goto - just use a normal if
> statement with a code block. There's only one nested if statement.
>
>> +
>> + count = min_t(unsigned int, (conn->mtu - L2CAP_HDR_SIZE),
>> + chan->omtu);
>> + count = count - skb->len + L2CAP_HDR_SIZE;
>> + count = min_t(unsigned int, count, len);
>> +
>> + if (memcpy_fromiovec(skb_put(skb, count), msg->msg_iov, count))
>> + return -EFAULT;
>> +
>> + sent += count;
>> + len -= count;
>> +
>> +frags:
>> + while (*frag)
>> + frag = &(*frag)->next;
>> +
>> + while (len) {
>> + count = chan->omtu - skb->len + L2CAP_HDR_SIZE;
>> + count = min_t(unsigned int, count, len);
>> + count = min_t(unsigned int, conn->mtu, count);
>> +
>> + *frag = chan->ops->alloc_skb(chan, count,
>> + msg->msg_flags & MSG_DONTWAIT);
>> + if (IS_ERR(*frag))
>> + return PTR_ERR(*frag);
>> +
>> + if (memcpy_fromiovec(skb_put(*frag, count), msg->msg_iov,
>> + count))
>> + return -EFAULT;
>> +
>> + (*frag)->priority = skb->priority;
>> +
>> + sent += count;
>> + len -= count;
>> +
>> + skb->len += (*frag)->len;
>> + skb->data_len += (*frag)->len;
>> +
>> + frag = &(*frag)->next;
>> +
>> + if (skb->len == chan->omtu + L2CAP_HDR_SIZE)
>> + break;
>
> Don't you want to return -EMSGSIZE if the data doesn't fit in one SDU?
>
>> + }
>> +
>> + lh = (struct l2cap_hdr *) skb->data;
>> + lh->len = cpu_to_le16(skb->len - L2CAP_HDR_SIZE);
>> +
>> + if (skb->len == chan->omtu + L2CAP_HDR_SIZE) {
>> + l2cap_do_send(chan, skb);
>
> I don't think it's good to put a send in here. Let the calling function do
> the send, so it's in one place.
>
>> + chan->skb_more = NULL;
>> + }
>> +
>> + return sent;
>> +}
>> +
>> int l2cap_chan_send(struct l2cap_chan *chan, struct msghdr *msg, size_t
>> len,
>> u32 priority)
>> {
>> @@ -2068,16 +2143,41 @@ int l2cap_chan_send(struct l2cap_chan *chan, struct
>> msghdr *msg, size_t len,
>> switch (chan->mode) {
>> case L2CAP_MODE_BASIC:
>> /* Check outgoing MTU */
>> - if (len > chan->omtu)
>> + if (len > chan->omtu) {
>> + kfree_skb(chan->skb_more);
>
> Set skb_more to NULL after freeing.
>
>> return -EMSGSIZE;
>> + }
>>
>> - /* Create a basic PDU */
>> - skb = l2cap_create_basic_pdu(chan, msg, len, priority);
>> - if (IS_ERR(skb))
>> + err = len;
>> + if (chan->skb_more) {
>> + int sent = l2cap_append_pdu(chan, msg, len);
>> +
>> + if (sent < 0) {
>> + kfree_skb(chan->skb_more);
>> + return sent;
>> + }
>> +
>> + len -= sent;
>> + }
>> +
>> + if (len)
>> + skb = l2cap_create_basic_pdu(chan, msg, len,
>> priority);
>
> Shouldn't this be the 'else' clause for the above if statement? You should
> either call l2cap_append_pdu or l2cap_create_basic_pdu, but never both.
> Better to structure the logic so that they are obviously mutually exclusive.
>
>> + else
>> + skb = chan->skb_more;
>> +
>> + if (IS_ERR(skb)) {
>> + kfree_skb(chan->skb_more);
>
> Set skb_more to NULL after freeing.
>
>> return PTR_ERR(skb);
>> + }
>> +
>> + if (!skb)
>> + return err;
>> +
>> + if (msg->msg_flags & MSG_MORE && skb->len < chan->omtu)
>> + chan->skb_more = skb;
>> + else
>> + l2cap_do_send(chan, skb);
>
> I think l2cap_do_send() should be called if and only if MSG_MORE is not set,
> unless there is an MTU problem.
>
> Also, do you need to account for L2CAP_HDR_SIZE when checking skb->len?
>
>>
>> - l2cap_do_send(chan, skb);
>> - err = len;
>> break;
>>
>> case L2CAP_MODE_ERTM:
>> --
>> 1.7.10.1
>
> Overall, I would suggest that l2cap_chan_send should be kept simple and most
> of the MSG_MORE code should be in a function called by l2cap_chan_send.
One more thing: also make sure that chan->skb_more is freed when the
channel is deleted, since there may be data queued when an L2CAP
channel, ACL, or socket is disconnected.
--
Mat Martineau
Employee of Qualcomm Innovation Center, Inc.
Qualcomm Innovation Center, Inc. is a member of Code Aurora Forum
next prev parent reply other threads:[~2012-05-11 18:41 UTC|newest]
Thread overview: 12+ messages / expand[flat|nested] mbox.gz Atom feed top
2012-05-11 16:16 [PATCH 1/4] Bluetooth: Fix packet size provided to the controller Gustavo Padovan
2012-05-11 16:16 ` [PATCH 2/4] Bluetooth: Fix skb length calculation Gustavo Padovan
2012-05-11 16:16 ` [PATCH 3/4] Bluetooth: Add MSG_MORE support to L2CAP sockets Gustavo Padovan
2012-05-11 16:16 ` [PATCH 4/4] Bluetooth: add a timer to L2CAP MSG_MORE code Gustavo Padovan
2012-05-11 18:38 ` Mat Martineau
2012-05-11 19:23 ` Gustavo Padovan
2012-05-11 22:29 ` Mat Martineau
2012-05-11 18:31 ` [PATCH 3/4] Bluetooth: Add MSG_MORE support to L2CAP sockets Mat Martineau
2012-05-11 18:41 ` Mat Martineau [this message]
2012-05-11 19:15 ` Gustavo Padovan
2012-05-11 21:55 ` Mat Martineau
2012-05-16 7:49 ` [PATCH 1/4] Bluetooth: Fix packet size provided to the controller Johan Hedberg
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=alpine.DEB.2.02.1205111139080.17554@mathewm-linux \
--to=mathewm@codeaurora.org \
--cc=gustavo@padovan.org \
--cc=linux-bluetooth@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox