Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit acd7d370 authored by Ville Tervo's avatar Ville Tervo Committed by Gustavo Padovan
Browse files

Bluetooth: Add LE connection support to L2CAP



Add basic LE connection support to L2CAP. LE
connection can be created by specifying cid
in struct sockaddr_l2

Signed-off-by: default avatarVille Tervo <ville.tervo@nokia.com>
Acked-by: default avatarMarcel Holtmann <marcel@holtmann.org>
Signed-off-by: default avatarGustavo F. Padovan <padovan@profusion.mobi>
parent 6ed58ec5
Loading
Loading
Loading
Loading
+3 −0
Original line number Original line Diff line number Diff line
@@ -160,6 +160,9 @@ struct l2cap_conn_rsp {
/* channel indentifier */
/* channel indentifier */
#define L2CAP_CID_SIGNALING	0x0001
#define L2CAP_CID_SIGNALING	0x0001
#define L2CAP_CID_CONN_LESS	0x0002
#define L2CAP_CID_CONN_LESS	0x0002
#define L2CAP_CID_LE_DATA	0x0004
#define L2CAP_CID_LE_SIGNALING	0x0005
#define L2CAP_CID_SMP		0x0006
#define L2CAP_CID_DYN_START	0x0040
#define L2CAP_CID_DYN_START	0x0040
#define L2CAP_CID_DYN_END	0xffff
#define L2CAP_CID_DYN_END	0xffff


+19 −4
Original line number Original line Diff line number Diff line
@@ -593,6 +593,12 @@ static void l2cap_conn_ready(struct l2cap_conn *conn)
	for (sk = l->head; sk; sk = l2cap_pi(sk)->next_c) {
	for (sk = l->head; sk; sk = l2cap_pi(sk)->next_c) {
		bh_lock_sock(sk);
		bh_lock_sock(sk);


		if (conn->hcon->type == LE_LINK) {
			l2cap_sock_clear_timer(sk);
			sk->sk_state = BT_CONNECTED;
			sk->sk_state_change(sk);
		}

		if (sk->sk_type != SOCK_SEQPACKET &&
		if (sk->sk_type != SOCK_SEQPACKET &&
				sk->sk_type != SOCK_STREAM) {
				sk->sk_type != SOCK_STREAM) {
			l2cap_sock_clear_timer(sk);
			l2cap_sock_clear_timer(sk);
@@ -651,7 +657,11 @@ static struct l2cap_conn *l2cap_conn_add(struct hci_conn *hcon, u8 status)


	BT_DBG("hcon %p conn %p", hcon, conn);
	BT_DBG("hcon %p conn %p", hcon, conn);


	if (hcon->hdev->le_mtu && hcon->type == LE_LINK)
		conn->mtu = hcon->hdev->le_mtu;
	else
		conn->mtu = hcon->hdev->acl_mtu;
		conn->mtu = hcon->hdev->acl_mtu;

	conn->src = &hcon->hdev->bdaddr;
	conn->src = &hcon->hdev->bdaddr;
	conn->dst = &hcon->dst;
	conn->dst = &hcon->dst;


@@ -758,8 +768,13 @@ int l2cap_do_connect(struct sock *sk)


	auth_type = l2cap_get_auth_type(sk);
	auth_type = l2cap_get_auth_type(sk);


	if (l2cap_pi(sk)->dcid == L2CAP_CID_LE_DATA)
		hcon = hci_connect(hdev, LE_LINK, dst,
					l2cap_pi(sk)->sec_level, auth_type);
	else
		hcon = hci_connect(hdev, ACL_LINK, dst,
		hcon = hci_connect(hdev, ACL_LINK, dst,
					l2cap_pi(sk)->sec_level, auth_type);
					l2cap_pi(sk)->sec_level, auth_type);

	if (!hcon)
	if (!hcon)
		goto done;
		goto done;


@@ -3520,7 +3535,7 @@ static int l2cap_connect_cfm(struct hci_conn *hcon, u8 status)


	BT_DBG("hcon %p bdaddr %s status %d", hcon, batostr(&hcon->dst), status);
	BT_DBG("hcon %p bdaddr %s status %d", hcon, batostr(&hcon->dst), status);


	if (hcon->type != ACL_LINK)
	if (!(hcon->type == ACL_LINK || hcon->type == LE_LINK))
		return -EINVAL;
		return -EINVAL;


	if (!status) {
	if (!status) {
@@ -3549,7 +3564,7 @@ static int l2cap_disconn_cfm(struct hci_conn *hcon, u8 reason)
{
{
	BT_DBG("hcon %p reason %d", hcon, reason);
	BT_DBG("hcon %p reason %d", hcon, reason);


	if (hcon->type != ACL_LINK)
	if (!(hcon->type == ACL_LINK || hcon->type == LE_LINK))
		return -EINVAL;
		return -EINVAL;


	l2cap_conn_del(hcon, bt_err(reason));
	l2cap_conn_del(hcon, bt_err(reason));
+4 −3
Original line number Original line Diff line number Diff line
@@ -168,13 +168,13 @@ static int l2cap_sock_connect(struct socket *sock, struct sockaddr *addr, int al
	len = min_t(unsigned int, sizeof(la), alen);
	len = min_t(unsigned int, sizeof(la), alen);
	memcpy(&la, addr, len);
	memcpy(&la, addr, len);


	if (la.l2_cid)
	if (la.l2_cid && la.l2_psm)
		return -EINVAL;
		return -EINVAL;


	lock_sock(sk);
	lock_sock(sk);


	if ((sk->sk_type == SOCK_SEQPACKET || sk->sk_type == SOCK_STREAM)
	if ((sk->sk_type == SOCK_SEQPACKET || sk->sk_type == SOCK_STREAM)
			&& !la.l2_psm) {
			&& !(la.l2_psm || la.l2_cid)) {
		err = -EINVAL;
		err = -EINVAL;
		goto done;
		goto done;
	}
	}
@@ -216,7 +216,7 @@ static int l2cap_sock_connect(struct socket *sock, struct sockaddr *addr, int al


	/* PSM must be odd and lsb of upper byte must be 0 */
	/* PSM must be odd and lsb of upper byte must be 0 */
	if ((__le16_to_cpu(la.l2_psm) & 0x0101) != 0x0001 &&
	if ((__le16_to_cpu(la.l2_psm) & 0x0101) != 0x0001 &&
		sk->sk_type != SOCK_RAW) {
				sk->sk_type != SOCK_RAW && !la.l2_cid) {
		err = -EINVAL;
		err = -EINVAL;
		goto done;
		goto done;
	}
	}
@@ -224,6 +224,7 @@ static int l2cap_sock_connect(struct socket *sock, struct sockaddr *addr, int al
	/* Set destination address and psm */
	/* Set destination address and psm */
	bacpy(&bt_sk(sk)->dst, &la.l2_bdaddr);
	bacpy(&bt_sk(sk)->dst, &la.l2_bdaddr);
	l2cap_pi(sk)->psm = la.l2_psm;
	l2cap_pi(sk)->psm = la.l2_psm;
	l2cap_pi(sk)->dcid = la.l2_cid;


	err = l2cap_do_connect(sk);
	err = l2cap_do_connect(sk);
	if (err)
	if (err)