Fix for audio stream XCASE connect:connect failures

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



For several days now, I've been exploring what I felt was a race condition in audio/avdtp.c (many thanks to Johan for some helpful guidance). The behavior I was observing was that AVDTP connections were erroring out when the headset device (Motorola S9) was powering on. The following is a typical error log of was I was observing:

Sep 4 07:30:05 dev-wkstn bluetoothd[2713]: State changed /org/bluez/2713/hci0/dev_00_0D_FD_1E_99_30: HEADSET_STATE_CONNECT_IN_PROGRESS -> HEADSET_STATE_CONNECTED Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_ref(0x7fd0c11c9f50): ref=2 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_ref(0x7fd0c11c9f50): ref=3 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_unref(0x7fd0c11c9f50): ref=2
Sep  4 07:30:06 dev-wkstn bluetoothd[2713]: Received AT+CKPD=200
Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: telephony-dummy: got key press request for 200 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: AVDTP: incoming connect from 00:0D:FD:1E:99:30 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: AVDTP: connected signaling channel to 00:0D:FD:1E:99:30
Sep  4 07:30:06 dev-wkstn bluetoothd[2713]: AVDTP imtu=672, omtu=895
Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: send: Transport endpoint is not connected (107)
Sep  4 07:30:06 dev-wkstn bluetoothd[2713]: session_cb
Sep  4 07:30:06 dev-wkstn bluetoothd[2713]: Received DISCOVER_CMD
Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: send: Transport endpoint is not connected (107) Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: Unable to handle command. Disconnecting Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: Disconnected from 00:0D:FD:1E:99:30 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_unref(0x7fd0c11c9f50): ref=1
Sep  4 07:30:06 dev-wkstn bluetoothd[2713]: connect:connect XCASE detected
Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_unref(0x7fd0c11c9f50): ref=0 Sep 4 07:30:06 dev-wkstn bluetoothd[2713]: avdtp_unref(0x7fd0c11c9f50): freeing session and removing from list Sep 4 07:30:09 dev-wkstn bluetoothd[2713]: Stream setup failed, after XCASE connect:connect

Note the "Transport endpoint is not connected (107)" errors. Until I more fully understood the architecture, I did not understand that those errors were related to the "connect:connect XCASE detected" and "Stream setup failed, after XCASE connect:connect" messages. Now that I do understand, I thought I'd share what I learned and a suggested patch to resolve the problem. (I'm sure I'm rediscovering what others already know but I could find no reference to it in the mail archive).

The short version of the problem is that it is possible for a device to initiate an l2cap channel for the avdtp 'signalling' channel even though a host-initiated avdtp connection is already pending with that device. In the hcidump capture below, the first packet is the Connect initiated from the host side. The 4th packet shows the device initiating it's own connection (even though it has already acknowledged a pending 'signalling' channel connection from the host). (Per the AVDTP spec, the 'signalling' channel is simply the 1st avdtp channel created when none exist.)

2009-09-04 22:32:09.000236 < ACL data: handle 11 flags 0x02 dlen 12
   L2CAP(s): Connect req: psm 25 scid 0x0041
2009-09-04 22:32:09.018962 > HCI Event: Number of Completed Packets (0x13) plen 5
   handle 11 packets 2
2009-09-04 22:32:09.024959 > ACL data: handle 11 flags 0x02 dlen 16
   L2CAP(s): Connect rsp: dcid 0x005d scid 0x0041 result 1 status 2
     Connection pending - Authorization pending
2009-09-04 22:32:09.226945 > ACL data: handle 11 flags 0x02 dlen 12
   L2CAP(s): Connect req: psm 25 scid 0x005e
2009-09-04 22:32:09.227002 < ACL data: handle 11 flags 0x02 dlen 16
   L2CAP(s): Connect rsp: dcid 0x0042 scid 0x005e result 1 status 2
     Connection pending - Authorization pending
2009-09-04 22:32:09.282939 > ACL data: handle 11 flags 0x02 dlen 16
   L2CAP(s): Connect rsp: dcid 0x005d scid 0x0041 result 0 status 0
     Connection successful
2009-09-04 22:32:09.282980 < ACL data: handle 11 flags 0x02 dlen 12
   L2CAP(s): Config req: dcid 0x005d flags 0x00 clen 0
2009-09-04 22:32:09.292932 > ACL data: handle 11 flags 0x02 dlen 16
   L2CAP(s): Config req: dcid 0x0041 flags 0x00 clen 4
     MTU 895
2009-09-04 22:32:09.292976 < ACL data: handle 11 flags 0x02 dlen 18
   L2CAP(s): Config rsp: scid 0x005d flags 0x00 result 0 clen 4
     MTU 895
2009-09-04 22:32:09.293932 > HCI Event: Number of Completed Packets (0x13) plen 5
   handle 11 packets 2
2009-09-04 22:32:09.338933 > ACL data: handle 11 flags 0x02 dlen 14
   L2CAP(s): Config rsp: scid 0x0041 flags 0x00 result 0 clen 0
     Success
2009-09-04 22:32:09.387923 > ACL data: handle 11 flags 0x02 dlen 6
   L2CAP(d): cid 0x0041 len 2 [psm 25]
     AVDTP(s): Discover cmd: transaction 2

Since the kernel l2cap driver has limited knowledge of avdtp protocol, it continues to handle the l2cap channel configuration ultimately resulting in 2 'signalling' channels being created (behavior which is outside the AVDTP spec). This situation also confuses the software state machine in audio/avdtp.c because it does not account for 2 signalling channels (nor should it).

Possible solutions to this situation (perhaps not exhaustive) include:
1) Reject incoming avdtp connection attempt from device when outgoing connection request is pending. 2) Give priority to device-initiated connection, killing outgoing connection request. 3) Move knowledge of avdtp into l2cap kernel driver (or make separate avdtp kernel driver). 4) Lengthen timer setup in device.c to give devices more time to initiate an avdtp connection.

I discarded option 4 as not solving the problem, but simply avoiding it. Option 3 seems like taking an architectural step backwards (not to mention how significant the effort would be). Initially, I tried option 2 by state-checking and parameter-testing avdtp_connect_cb(). I had this working to a certain extent but ultimately option 1 proved to be a much cleaner implementation.

Patch for option 1 is attached. Here's an excerpt from the daemon.log when this patch works (note the 'connect already in progress' message).

Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: AVDTP: incoming connect from 00:0D:FD:1E:99:30 Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: avdtp_confirm_cb: connect already in progress (XCASE connect:connect) Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: AVDTP: connected signaling channel to 00:0D:FD:1E:99:30
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: AVDTP imtu=672, omtu=895
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: AVCTP Connecting
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: session_cb
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: DISCOVER request succeeded
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: seid 1 type 1 media 0 in use 0
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: seid 2 type 1 media 0 in use 0
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: session_cb
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: GET_CAPABILITIES request succeeded
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: seid 1 type 1 media 0
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: session_cb
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: GET_CAPABILITIES request succeeded
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: seid 2 type 1 media 0
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: Discovery complete
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: a2dp_config: selected SEP 0x7f89d7a8d190
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: avdtp_ref(0x7f89d7a86c30): ref=3
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: setup_ref(0x7f89d7a94a40): ref=1
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: avdtp_set_configuration(0x7f89d7a86c30): int_seid=1, acp_seid=1
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: session_cb
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: SET_CONFIGURATION request succeeded Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: Source 0x7f89d7a8d190: Set_Configuration_Cfm Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: stream state changed: IDLE -> CONFIGURED
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: session_cb
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: OPEN request succeeded
Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: AVCTP: connected to 00:0D:FD:1E:99:30 Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: Can't open input device: No such file or directory (2) Sep 8 15:25:03 dev-wkstn bluetoothd[2676]: AVRCP: failed to init uinput for 00:0D:FD:1E:99:30
Sep  8 15:25:03 dev-wkstn bluetoothd[2676]: AVCTP Connected
Sep 8 15:25:04 dev-wkstn bluetoothd[2676]: AVDTP: connected transport channel to 00:0D:FD:1E:99:30
Sep  8 15:25:04 dev-wkstn bluetoothd[2676]: Source 0x7f89d7a8d190: Open_Cfm
Sep  8 15:25:04 dev-wkstn bluetoothd[2676]: setup_ref(0x7f89d7a94a40): ref=2
Sep  8 15:25:04 dev-wkstn bluetoothd[2676]: Stream successfully created

Comments?

Regards,
Peter Hurley

Also, I have prepared a more extensive source code analysis which is more complicated if anyone would prefer that.
diff --git a/audio/avdtp.c b/audio/avdtp.c
index 131ff4f..d97aff2 100644
--- a/audio/avdtp.c
+++ b/audio/avdtp.c
@@ -2119,6 +2119,16 @@ static void avdtp_confirm_cb(GIOChannel *chan, gpointer data)
 	if (!session)
 		goto drop;
 
+	// This state (ie, session is already *connecting*) happens when the device
+	//  initiates a connect (really a config'd L2CAP channel) even though
+	//  there is a connect we initiated in progress.  In sink.c & source.c,
+	//  this state is referred to as XCASE connect:connect.
+	// Abort the device's channel in favor of our own.
+	if (session->state == AVDTP_SESSION_STATE_CONNECTING) {
+		debug("avdtp_confirm_cb: connect already in progress (XCASE connect:connect)");
+		goto drop;
+	}
+
 	if (session->pending_open && session->pending_open->open_acp) {
 		if (!bt_io_accept(chan, avdtp_connect_cb, session, NULL, NULL))
 			goto drop;

[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux