2012-08-04 12:20:47 +02:00
/* Copyright (C) 2012 Kristian Lauszus, TKJ Electronics. All rights reserved.
2013-03-28 09:37:09 +01:00
2012-08-04 12:20:47 +02:00
This software may be distributed and modified under the terms of the GNU
General Public License version 2 ( GPL2 ) as published by the Free Software
Foundation and appearing in the file GPL2 . TXT included in the packaging of
this file . Please note that GPL2 Section 2 [ b ] requires that all works based
on this software must also be made publicly available under the terms of
the GPL2 ( " Copyleft " ) .
2013-03-28 09:37:09 +01:00
2012-08-04 12:20:47 +02:00
Contact information
- - - - - - - - - - - - - - - - - - -
2013-03-28 09:37:09 +01:00
2012-08-04 12:20:47 +02:00
Kristian Lauszus , TKJ Electronics
Web : http : //www.tkjelectronics.com
e - mail : kristianl @ tkjelectronics . com
*/
# include "SPP.h"
2013-11-11 11:48:09 +01:00
// To enable serial debugging see "settings.h"
2012-08-04 12:20:47 +02:00
//#define EXTRADEBUG // Uncomment to get even more debugging data
//#define PRINTREPORT // Uncomment to print the report sent to the Arduino
/*
* CRC ( reversed crc ) lookup table as calculated by the table generator in ETSI TS 101 369 V6 .3 .0 .
*/
2013-03-28 09:46:43 +01:00
const uint8_t rfcomm_crc_table [ 256 ] PROGMEM = { /* reversed, 8-bit, poly=0x07 */
0x00 , 0x91 , 0xE3 , 0x72 , 0x07 , 0x96 , 0xE4 , 0x75 , 0x0E , 0x9F , 0xED , 0x7C , 0x09 , 0x98 , 0xEA , 0x7B ,
0x1C , 0x8D , 0xFF , 0x6E , 0x1B , 0x8A , 0xF8 , 0x69 , 0x12 , 0x83 , 0xF1 , 0x60 , 0x15 , 0x84 , 0xF6 , 0x67 ,
0x38 , 0xA9 , 0xDB , 0x4A , 0x3F , 0xAE , 0xDC , 0x4D , 0x36 , 0xA7 , 0xD5 , 0x44 , 0x31 , 0xA0 , 0xD2 , 0x43 ,
0x24 , 0xB5 , 0xC7 , 0x56 , 0x23 , 0xB2 , 0xC0 , 0x51 , 0x2A , 0xBB , 0xC9 , 0x58 , 0x2D , 0xBC , 0xCE , 0x5F ,
0x70 , 0xE1 , 0x93 , 0x02 , 0x77 , 0xE6 , 0x94 , 0x05 , 0x7E , 0xEF , 0x9D , 0x0C , 0x79 , 0xE8 , 0x9A , 0x0B ,
0x6C , 0xFD , 0x8F , 0x1E , 0x6B , 0xFA , 0x88 , 0x19 , 0x62 , 0xF3 , 0x81 , 0x10 , 0x65 , 0xF4 , 0x86 , 0x17 ,
0x48 , 0xD9 , 0xAB , 0x3A , 0x4F , 0xDE , 0xAC , 0x3D , 0x46 , 0xD7 , 0xA5 , 0x34 , 0x41 , 0xD0 , 0xA2 , 0x33 ,
0x54 , 0xC5 , 0xB7 , 0x26 , 0x53 , 0xC2 , 0xB0 , 0x21 , 0x5A , 0xCB , 0xB9 , 0x28 , 0x5D , 0xCC , 0xBE , 0x2F ,
0xE0 , 0x71 , 0x03 , 0x92 , 0xE7 , 0x76 , 0x04 , 0x95 , 0xEE , 0x7F , 0x0D , 0x9C , 0xE9 , 0x78 , 0x0A , 0x9B ,
0xFC , 0x6D , 0x1F , 0x8E , 0xFB , 0x6A , 0x18 , 0x89 , 0xF2 , 0x63 , 0x11 , 0x80 , 0xF5 , 0x64 , 0x16 , 0x87 ,
0xD8 , 0x49 , 0x3B , 0xAA , 0xDF , 0x4E , 0x3C , 0xAD , 0xD6 , 0x47 , 0x35 , 0xA4 , 0xD1 , 0x40 , 0x32 , 0xA3 ,
0xC4 , 0x55 , 0x27 , 0xB6 , 0xC3 , 0x52 , 0x20 , 0xB1 , 0xCA , 0x5B , 0x29 , 0xB8 , 0xCD , 0x5C , 0x2E , 0xBF ,
0x90 , 0x01 , 0x73 , 0xE2 , 0x97 , 0x06 , 0x74 , 0xE5 , 0x9E , 0x0F , 0x7D , 0xEC , 0x99 , 0x08 , 0x7A , 0xEB ,
0x8C , 0x1D , 0x6F , 0xFE , 0x8B , 0x1A , 0x68 , 0xF9 , 0x82 , 0x13 , 0x61 , 0xF0 , 0x85 , 0x14 , 0x66 , 0xF7 ,
0xA8 , 0x39 , 0x4B , 0xDA , 0xAF , 0x3E , 0x4C , 0xDD , 0xA6 , 0x37 , 0x45 , 0xD4 , 0xA1 , 0x30 , 0x42 , 0xD3 ,
0xB4 , 0x25 , 0x57 , 0xC6 , 0xB3 , 0x22 , 0x50 , 0xC1 , 0xBA , 0x2B , 0x59 , 0xC8 , 0xBD , 0x2C , 0x5E , 0xCF
2012-08-04 12:20:47 +02:00
} ;
2013-03-28 09:46:43 +01:00
SPP : : SPP ( BTD * p , const char * name , const char * pin ) :
2012-08-04 12:20:47 +02:00
pBtd ( p ) // Pointer to BTD class instance - mandatory
{
2013-03-28 09:46:43 +01:00
if ( pBtd )
pBtd - > registerServiceClass ( this ) ; // Register it as a Bluetooth service
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
pBtd - > btdName = name ;
pBtd - > btdPin = pin ;
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
/* Set device cid for the SDP and RFCOMM channelse */
sdp_dcid [ 0 ] = 0x50 ; // 0x0050
sdp_dcid [ 1 ] = 0x00 ;
rfcomm_dcid [ 0 ] = 0x51 ; // 0x0051
rfcomm_dcid [ 1 ] = 0x00 ;
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
Reset ( ) ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-08 05:40:53 +02:00
void SPP : : Reset ( ) {
2013-03-28 09:46:43 +01:00
connected = false ;
RFCOMMConnected = false ;
SDPConnected = false ;
2013-09-27 11:07:05 +02:00
waitForLastCommand = false ;
2013-03-28 09:46:43 +01:00
l2cap_sdp_state = L2CAP_SDP_WAIT ;
l2cap_rfcomm_state = L2CAP_RFCOMM_WAIT ;
l2cap_event_flag = 0 ;
2013-05-07 00:06:49 +02:00
sppIndex = 0 ;
2013-03-28 09:46:43 +01:00
}
void SPP : : disconnect ( ) {
connected = false ;
// First the two L2CAP channels has to be disconencted and then the HCI connection
if ( RFCOMMConnected )
pBtd - > l2cap_disconnection_request ( hci_handle , 0x0A , rfcomm_scid , rfcomm_dcid ) ;
if ( RFCOMMConnected & & SDPConnected )
delay ( 1 ) ; // Add delay between commands
if ( SDPConnected )
pBtd - > l2cap_disconnection_request ( hci_handle , 0x0B , sdp_scid , sdp_dcid ) ;
l2cap_sdp_state = L2CAP_DISCONNECT_RESPONSE ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-08 20:01:10 +02:00
void SPP : : ACLData ( uint8_t * l2capinbuf ) {
2013-03-28 09:46:43 +01:00
if ( ! connected ) {
if ( l2capinbuf [ 8 ] = = L2CAP_CMD_CONNECTION_REQUEST ) {
if ( ( l2capinbuf [ 12 ] | ( l2capinbuf [ 13 ] < < 8 ) ) = = SDP_PSM & & ! pBtd - > sdpConnectionClaimed ) {
pBtd - > sdpConnectionClaimed = true ;
hci_handle = pBtd - > hci_handle ; // Store the HCI Handle for the connection
l2cap_sdp_state = L2CAP_SDP_WAIT ; // Reset state
} else if ( ( l2capinbuf [ 12 ] | ( l2capinbuf [ 13 ] < < 8 ) ) = = RFCOMM_PSM & & ! pBtd - > rfcommConnectionClaimed ) {
pBtd - > rfcommConnectionClaimed = true ;
hci_handle = pBtd - > hci_handle ; // Store the HCI Handle for the connection
l2cap_rfcomm_state = L2CAP_RFCOMM_WAIT ; // Reset state
}
}
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
if ( ( ( l2capinbuf [ 0 ] | ( l2capinbuf [ 1 ] < < 8 ) ) = = ( hci_handle | 0x2000 ) ) ) { // acl_handle_ok
if ( ( l2capinbuf [ 6 ] | ( l2capinbuf [ 7 ] < < 8 ) ) = = 0x0001 ) { //l2cap_control - Channel ID for ACL-U
if ( l2capinbuf [ 8 ] = = L2CAP_CMD_COMMAND_REJECT ) {
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n L2CAP Command Rejected - Reason: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 13 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 12 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " Data: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 17 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 16 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 15 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 14 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
# endif
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_CONNECTION_REQUEST ) {
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n L2CAP Connection Request - PSM: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 13 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 12 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " SCID: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 15 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 14 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " Identifier: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 9 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
# endif
if ( ( l2capinbuf [ 12 ] | ( l2capinbuf [ 13 ] < < 8 ) ) = = SDP_PSM ) { // It doesn't matter if it receives another reqeust, since it waits for the channel to disconnect in the L2CAP_SDP_DONE state, and the l2cap_event_flag will be cleared if so
identifier = l2capinbuf [ 9 ] ;
sdp_scid [ 0 ] = l2capinbuf [ 14 ] ;
sdp_scid [ 1 ] = l2capinbuf [ 15 ] ;
l2cap_event_flag | = L2CAP_FLAG_CONNECTION_SDP_REQUEST ;
} else if ( ( l2capinbuf [ 12 ] | ( l2capinbuf [ 13 ] < < 8 ) ) = = RFCOMM_PSM ) { // ----- || -----
identifier = l2capinbuf [ 9 ] ;
rfcomm_scid [ 0 ] = l2capinbuf [ 14 ] ;
rfcomm_scid [ 1 ] = l2capinbuf [ 15 ] ;
l2cap_event_flag | = L2CAP_FLAG_CONNECTION_RFCOMM_REQUEST ;
}
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_CONFIG_RESPONSE ) {
if ( ( l2capinbuf [ 16 ] | ( l2capinbuf [ 17 ] < < 8 ) ) = = 0x0000 ) { // Success
if ( l2capinbuf [ 12 ] = = sdp_dcid [ 0 ] & & l2capinbuf [ 13 ] = = sdp_dcid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nSDP Configuration Complete"), 0x80);
2013-03-28 09:46:43 +01:00
l2cap_event_flag | = L2CAP_FLAG_CONFIG_SDP_SUCCESS ;
} else if ( l2capinbuf [ 12 ] = = rfcomm_dcid [ 0 ] & & l2capinbuf [ 13 ] = = rfcomm_dcid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nRFCOMM Configuration Complete"), 0x80);
2013-03-28 09:46:43 +01:00
l2cap_event_flag | = L2CAP_FLAG_CONFIG_RFCOMM_SUCCESS ;
}
}
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_CONFIG_REQUEST ) {
if ( l2capinbuf [ 12 ] = = sdp_dcid [ 0 ] & & l2capinbuf [ 13 ] = = sdp_dcid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nSDP Configuration Request"), 0x80);
2013-03-28 09:46:43 +01:00
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_CONFIG_SDP_REQUEST ;
} else if ( l2capinbuf [ 12 ] = = rfcomm_dcid [ 0 ] & & l2capinbuf [ 13 ] = = rfcomm_dcid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nRFCOMM Configuration Request"), 0x80);
2013-03-28 09:46:43 +01:00
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_CONFIG_RFCOMM_REQUEST ;
}
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_DISCONNECT_REQUEST ) {
if ( l2capinbuf [ 12 ] = = sdp_dcid [ 0 ] & & l2capinbuf [ 13 ] = = sdp_dcid [ 1 ] ) {
//Notify(PSTR("\r\nDisconnect Request: SDP Channel"), 0x80);
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_DISCONNECT_SDP_REQUEST ;
} else if ( l2capinbuf [ 12 ] = = rfcomm_dcid [ 0 ] & & l2capinbuf [ 13 ] = = rfcomm_dcid [ 1 ] ) {
//Notify(PSTR("\r\nDisconnect Request: RFCOMM Channel"), 0x80);
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_DISCONNECT_RFCOMM_REQUEST ;
}
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_DISCONNECT_RESPONSE ) {
if ( l2capinbuf [ 12 ] = = sdp_scid [ 0 ] & & l2capinbuf [ 13 ] = = sdp_scid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nDisconnect Response: SDP Channel"), 0x80);
2013-03-28 09:46:43 +01:00
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_DISCONNECT_RESPONSE ;
} else if ( l2capinbuf [ 12 ] = = rfcomm_scid [ 0 ] & & l2capinbuf [ 13 ] = = rfcomm_scid [ 1 ] ) {
2013-04-26 23:50:39 +02:00
//Notify(PSTR("\r\nDisconnect Response: RFCOMM Channel"), 0x80);
2013-03-28 09:46:43 +01:00
identifier = l2capinbuf [ 9 ] ;
l2cap_event_flag | = L2CAP_FLAG_DISCONNECT_RESPONSE ;
}
} else if ( l2capinbuf [ 8 ] = = L2CAP_CMD_INFORMATION_REQUEST ) {
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Information request " ) , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
identifier = l2capinbuf [ 9 ] ;
pBtd - > l2cap_information_response ( hci_handle , identifier , l2capinbuf [ 12 ] , l2capinbuf [ 13 ] ) ;
}
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
else {
Notify ( PSTR ( " \r \n L2CAP Unknown Signaling Command: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 8 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
}
# endif
} else if ( l2capinbuf [ 6 ] = = sdp_dcid [ 0 ] & & l2capinbuf [ 7 ] = = sdp_dcid [ 1 ] ) { // SDP
if ( l2capinbuf [ 8 ] = = SDP_SERVICE_SEARCH_ATTRIBUTE_REQUEST_PDU ) {
2013-04-01 03:09:23 +02:00
if ( ( ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) = = SERIALPORT_UUID ) | | ( ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) = = 0x0000 & & ( l2capinbuf [ 18 ] < < 8 | l2capinbuf [ 19 ] ) = = SERIALPORT_UUID ) ) { // Check if it's sending the full UUID, see: https://www.bluetooth.org/Technical/AssignedNumbers/service_discovery.htm, we will just check the first four bytes
2013-03-28 09:46:43 +01:00
if ( firstMessage ) {
serialPortResponse1 ( l2capinbuf [ 9 ] , l2capinbuf [ 10 ] ) ;
firstMessage = false ;
} else {
serialPortResponse2 ( l2capinbuf [ 9 ] , l2capinbuf [ 10 ] ) ; // Serialport continuation state
firstMessage = true ;
}
2013-04-03 17:55:53 +02:00
} else if ( ( ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) = = L2CAP_UUID ) | | ( ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) = = 0x0000 & & ( l2capinbuf [ 18 ] < < 8 | l2capinbuf [ 19 ] ) = = L2CAP_UUID ) ) {
2013-03-28 09:46:43 +01:00
if ( firstMessage ) {
l2capResponse1 ( l2capinbuf [ 9 ] , l2capinbuf [ 10 ] ) ;
firstMessage = false ;
} else {
l2capResponse2 ( l2capinbuf [ 9 ] , l2capinbuf [ 10 ] ) ; // L2CAP continuation state
firstMessage = true ;
}
2013-04-08 00:22:15 +02:00
} else
serviceNotSupported ( l2capinbuf [ 9 ] , l2capinbuf [ 10 ] ) ; // The service is not supported
2013-04-01 03:09:23 +02:00
# ifdef EXTRADEBUG
2013-04-08 00:23:52 +02:00
Notify ( PSTR ( " \r \n UUID: " ) , 0x80 ) ;
uint16_t uuid ;
if ( ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) = = 0x0000 ) // Check if it's sending the UUID as a 128-bit UUID
uuid = ( l2capinbuf [ 18 ] < < 8 | l2capinbuf [ 19 ] ) ;
else // Short UUID
uuid = ( l2capinbuf [ 16 ] < < 8 | l2capinbuf [ 17 ] ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint16_t > ( uuid , 0x80 ) ;
2013-04-03 17:52:12 +02:00
2013-04-08 00:23:52 +02:00
Notify ( PSTR ( " \r \n Length: " ) , 0x80 ) ;
uint16_t length = l2capinbuf [ 11 ] < < 8 | l2capinbuf [ 12 ] ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint16_t > ( length , 0x80 ) ;
2013-04-08 00:23:52 +02:00
Notify ( PSTR ( " \r \n Data: " ) , 0x80 ) ;
for ( uint8_t i = 0 ; i < length ; i + + ) {
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 13 + i ] , 0x80 ) ;
2013-04-08 00:23:52 +02:00
Notify ( PSTR ( " " ) , 0x80 ) ;
}
2013-04-01 03:09:23 +02:00
# endif
2013-03-28 09:46:43 +01:00
}
2013-06-15 17:47:48 +02:00
# ifdef EXTRADEBUG
else {
2013-07-04 15:37:29 +02:00
Notify ( PSTR ( " \r \n Unknown PDU: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 8 ] , 0x80 ) ;
2013-06-15 17:47:48 +02:00
}
# endif
2013-03-28 09:46:43 +01:00
} else if ( l2capinbuf [ 6 ] = = rfcomm_dcid [ 0 ] & & l2capinbuf [ 7 ] = = rfcomm_dcid [ 1 ] ) { // RFCOMM
rfcommChannel = l2capinbuf [ 8 ] & 0xF8 ;
rfcommDirection = l2capinbuf [ 8 ] & 0x04 ;
rfcommCommandResponse = l2capinbuf [ 8 ] & 0x02 ;
rfcommChannelType = l2capinbuf [ 9 ] & 0xEF ;
rfcommPfBit = l2capinbuf [ 9 ] & 0x10 ;
if ( rfcommChannel > > 3 ! = 0x00 )
rfcommChannelConnection = rfcommChannel ;
2013-03-28 09:37:09 +01:00
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Channel: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommChannel > > 3 , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " Direction: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommDirection > > 2 , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " CommandResponse: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommCommandResponse > > 1 , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " ChannelType: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommChannelType , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " PF_BIT: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommPfBit , 0x80 ) ;
2013-03-28 09:46:43 +01:00
# endif
if ( rfcommChannelType = = RFCOMM_DISC ) {
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received Disconnect RFCOMM Command on channel: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommChannel > > 3 , 0x80 ) ;
2013-03-28 09:46:43 +01:00
# endif
connected = false ;
sendRfcomm ( rfcommChannel , rfcommDirection , rfcommCommandResponse , RFCOMM_UA , rfcommPfBit , rfcommbuf , 0x00 ) ; // UA Command
}
if ( connected ) {
/* Read the incoming message */
if ( rfcommChannelType = = RFCOMM_UIH & & rfcommChannel = = rfcommChannelConnection ) {
uint8_t length = l2capinbuf [ 10 ] > > 1 ; // Get length
uint8_t offset = l2capinbuf [ 4 ] - length - 4 ; // See if there is credit
if ( rfcommAvailable + length < = sizeof ( rfcommDataBuffer ) ) { // Don't add data to buffer if it would be full
for ( uint8_t i = 0 ; i < length ; i + + )
rfcommDataBuffer [ rfcommAvailable + i ] = l2capinbuf [ 11 + i + offset ] ;
rfcommAvailable + = length ;
}
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Data Available: " ) , 0x80 ) ;
2013-04-26 23:50:39 +02:00
Notify ( rfcommAvailable , 0x80 ) ;
2013-03-28 09:46:43 +01:00
if ( offset ) {
Notify ( PSTR ( " - Credit: 0x " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 11 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
}
2012-08-04 12:20:47 +02:00
# endif
# ifdef PRINTREPORT // Uncomment "#define PRINTREPORT" to print the report send to the Arduino via Bluetooth
2013-03-28 09:46:43 +01:00
for ( uint8_t i = 0 ; i < length ; i + + )
2013-04-26 23:50:39 +02:00
Notifyc ( l2capinbuf [ i + 11 + offset ] , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_RPN_CMD ) { // UIH Remote Port Negotiation Command
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received UIH Remote Port Negotiation Command " ) , 0x80 ) ;
# endif
rfcommbuf [ 0 ] = BT_RFCOMM_RPN_RSP ; // Command
rfcommbuf [ 1 ] = l2capinbuf [ 12 ] ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: channel << 1 | 1
rfcommbuf [ 3 ] = l2capinbuf [ 14 ] ; // Pre difined for Bluetooth, see 5.5.3 of TS 07.10 Adaption for RFCOMM
rfcommbuf [ 4 ] = l2capinbuf [ 15 ] ; // Priority
rfcommbuf [ 5 ] = l2capinbuf [ 16 ] ; // Timer
rfcommbuf [ 6 ] = l2capinbuf [ 17 ] ; // Max Fram Size LSB
rfcommbuf [ 7 ] = l2capinbuf [ 18 ] ; // Max Fram Size MSB
rfcommbuf [ 8 ] = l2capinbuf [ 19 ] ; // MaxRatransm.
rfcommbuf [ 9 ] = l2capinbuf [ 20 ] ; // Number of Frames
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x0A ) ; // UIH Remote Port Negotiation Response
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_MSC_CMD ) { // UIH Modem Status Command
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Send UIH Modem Status Response " ) , 0x80 ) ;
# endif
rfcommbuf [ 0 ] = BT_RFCOMM_MSC_RSP ; // UIH Modem Status Response
rfcommbuf [ 1 ] = 2 < < 1 | 1 ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: (1 << 0) | (1 << 1) | (0 << 2) | (channel << 3)
rfcommbuf [ 3 ] = l2capinbuf [ 14 ] ;
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x04 ) ;
}
} else {
if ( rfcommChannelType = = RFCOMM_SABM ) { // SABM Command - this is sent twice: once for channel 0 and then for the channel to establish
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received SABM Command " ) , 0x80 ) ;
2013-03-28 09:37:09 +01:00
# endif
2013-03-28 09:46:43 +01:00
sendRfcomm ( rfcommChannel , rfcommDirection , rfcommCommandResponse , RFCOMM_UA , rfcommPfBit , rfcommbuf , 0x00 ) ; // UA Command
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_PN_CMD ) { // UIH Parameter Negotiation Command
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received UIH Parameter Negotiation Command " ) , 0x80 ) ;
# endif
rfcommbuf [ 0 ] = BT_RFCOMM_PN_RSP ; // UIH Parameter Negotiation Response
rfcommbuf [ 1 ] = l2capinbuf [ 12 ] ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: channel << 1 | 1
rfcommbuf [ 3 ] = 0xE0 ; // Pre difined for Bluetooth, see 5.5.3 of TS 07.10 Adaption for RFCOMM
rfcommbuf [ 4 ] = 0x00 ; // Priority
rfcommbuf [ 5 ] = 0x00 ; // Timer
rfcommbuf [ 6 ] = BULK_MAXPKTSIZE - 14 ; // Max Fram Size LSB - set to the size of received data (50)
rfcommbuf [ 7 ] = 0x00 ; // Max Fram Size MSB
rfcommbuf [ 8 ] = 0x00 ; // MaxRatransm.
rfcommbuf [ 9 ] = 0x00 ; // Number of Frames
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x0A ) ;
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_MSC_CMD ) { // UIH Modem Status Command
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Send UIH Modem Status Response " ) , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
rfcommbuf [ 0 ] = BT_RFCOMM_MSC_RSP ; // UIH Modem Status Response
rfcommbuf [ 1 ] = 2 < < 1 | 1 ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: (1 << 0) | (1 << 1) | (0 << 2) | (channel << 3)
rfcommbuf [ 3 ] = l2capinbuf [ 14 ] ;
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x04 ) ;
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
delay ( 1 ) ;
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Send UIH Modem Status Command " ) , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
rfcommbuf [ 0 ] = BT_RFCOMM_MSC_CMD ; // UIH Modem Status Command
rfcommbuf [ 1 ] = 2 < < 1 | 1 ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: (1 << 0) | (1 << 1) | (0 << 2) | (channel << 3)
rfcommbuf [ 3 ] = 0x8D ; // Can receive frames (YES), Ready to Communicate (YES), Ready to Receive (YES), Incomig Call (NO), Data is Value (YES)
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x04 ) ;
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_MSC_RSP ) { // UIH Modem Status Response
if ( ! creditSent ) {
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Send UIH Command with credit " ) , 0x80 ) ;
# endif
sendRfcommCredit ( rfcommChannelConnection , rfcommDirection , 0 , RFCOMM_UIH , 0x10 , sizeof ( rfcommDataBuffer ) ) ; // Send credit
creditSent = true ;
timer = millis ( ) ;
waitForLastCommand = true ;
}
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 10 ] = = 0x01 ) { // UIH Command with credit
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received UIH Command with credit " ) , 0x80 ) ;
2013-03-28 09:37:09 +01:00
# endif
2013-03-28 09:46:43 +01:00
} else if ( rfcommChannelType = = RFCOMM_UIH & & l2capinbuf [ 11 ] = = BT_RFCOMM_RPN_CMD ) { // UIH Remote Port Negotiation Command
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Received UIH Remote Port Negotiation Command " ) , 0x80 ) ;
# endif
rfcommbuf [ 0 ] = BT_RFCOMM_RPN_RSP ; // Command
rfcommbuf [ 1 ] = l2capinbuf [ 12 ] ; // Length and shiftet like so: length << 1 | 1
rfcommbuf [ 2 ] = l2capinbuf [ 13 ] ; // Channel: channel << 1 | 1
rfcommbuf [ 3 ] = l2capinbuf [ 14 ] ; // Pre difined for Bluetooth, see 5.5.3 of TS 07.10 Adaption for RFCOMM
rfcommbuf [ 4 ] = l2capinbuf [ 15 ] ; // Priority
rfcommbuf [ 5 ] = l2capinbuf [ 16 ] ; // Timer
rfcommbuf [ 6 ] = l2capinbuf [ 17 ] ; // Max Fram Size LSB
rfcommbuf [ 7 ] = l2capinbuf [ 18 ] ; // Max Fram Size MSB
rfcommbuf [ 8 ] = l2capinbuf [ 19 ] ; // MaxRatransm.
rfcommbuf [ 9 ] = l2capinbuf [ 20 ] ; // Number of Frames
sendRfcomm ( rfcommChannel , rfcommDirection , 0 , RFCOMM_UIH , rfcommPfBit , rfcommbuf , 0x0A ) ; // UIH Remote Port Negotiation Response
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Connection is now established \r \n " ) , 0x80 ) ;
2013-03-28 09:37:09 +01:00
# endif
2013-03-28 09:46:43 +01:00
waitForLastCommand = false ;
creditSent = false ;
connected = true ; // The RFCOMM channel is now established
2013-05-07 00:06:49 +02:00
sppIndex = 0 ;
2013-03-28 09:46:43 +01:00
}
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
else if ( rfcommChannelType ! = RFCOMM_DISC ) {
Notify ( PSTR ( " \r \n Unsupported RFCOMM Data - ChannelType: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( rfcommChannelType , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " Command: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 11 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
}
# endif
}
2012-08-04 12:20:47 +02:00
}
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
else {
Notify ( PSTR ( " \r \n Unsupported L2CAP Data - Channel ID: " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 7 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capinbuf [ 6 ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
}
2012-08-09 21:22:55 +02:00
# endif
2013-03-28 09:46:43 +01:00
SDP_task ( ) ;
RFCOMM_task ( ) ;
}
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-08 05:40:53 +02:00
void SPP : : Run ( ) {
2013-03-28 09:46:43 +01:00
if ( waitForLastCommand & & ( millis ( ) - timer ) > 100 ) { // We will only wait 100ms and see if the UIH Remote Port Negotiation Command is send, as some deviced don't send it
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Connection is now established - Automatic \r \n " ) , 0x80 ) ;
2012-08-05 01:14:39 +02:00
# endif
2013-03-28 09:46:43 +01:00
creditSent = false ;
waitForLastCommand = false ;
connected = true ; // The RFCOMM channel is now established
2013-05-07 00:06:49 +02:00
sppIndex = 0 ;
2013-03-28 09:46:43 +01:00
}
2013-05-07 00:06:49 +02:00
send ( ) ; // Send all bytes currently in the buffer
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : SDP_task ( ) {
2013-03-28 09:46:43 +01:00
switch ( l2cap_sdp_state ) {
case L2CAP_SDP_WAIT :
if ( l2cap_connection_request_sdp_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONNECTION_SDP_REQUEST ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n SDP Incoming Connection Request " ) , 0x80 ) ;
# endif
pBtd - > l2cap_connection_response ( hci_handle , identifier , sdp_dcid , sdp_scid , PENDING ) ;
delay ( 1 ) ;
pBtd - > l2cap_connection_response ( hci_handle , identifier , sdp_dcid , sdp_scid , SUCCESSFUL ) ;
identifier + + ;
delay ( 1 ) ;
pBtd - > l2cap_config_request ( hci_handle , identifier , sdp_scid ) ;
l2cap_sdp_state = L2CAP_SDP_REQUEST ;
}
break ;
case L2CAP_SDP_REQUEST :
if ( l2cap_config_request_sdp_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONFIG_SDP_REQUEST ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n SDP Configuration Request " ) , 0x80 ) ;
# endif
pBtd - > l2cap_config_response ( hci_handle , identifier , sdp_scid ) ;
l2cap_sdp_state = L2CAP_SDP_SUCCESS ;
}
break ;
case L2CAP_SDP_SUCCESS :
if ( l2cap_config_success_sdp_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONFIG_SDP_SUCCESS ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n SDP Successfully Configured " ) , 0x80 ) ;
# endif
firstMessage = true ; // Reset bool
SDPConnected = true ;
l2cap_sdp_state = L2CAP_SDP_DONE ;
}
break ;
case L2CAP_SDP_DONE :
if ( l2cap_disconnect_request_sdp_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_DISCONNECT_SDP_REQUEST ; // Clear flag
SDPConnected = false ;
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Disconnected SDP Channel " ) , 0x80 ) ;
# endif
pBtd - > l2cap_disconnection_response ( hci_handle , identifier , sdp_dcid , sdp_scid ) ;
l2cap_sdp_state = L2CAP_SDP_WAIT ;
} else if ( l2cap_connection_request_sdp_flag )
l2cap_rfcomm_state = L2CAP_SDP_WAIT ;
break ;
case L2CAP_DISCONNECT_RESPONSE : // This is for both disconnection response from the RFCOMM and SDP channel if they were connected
if ( l2cap_disconnect_response_flag ) {
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Disconnected L2CAP Connection " ) , 0x80 ) ;
# endif
RFCOMMConnected = false ;
SDPConnected = false ;
pBtd - > hci_disconnect ( hci_handle ) ;
hci_handle = - 1 ; // Reset handle
l2cap_event_flag = 0 ; // Reset flags
l2cap_sdp_state = L2CAP_SDP_WAIT ;
l2cap_rfcomm_state = L2CAP_RFCOMM_WAIT ;
}
break ;
}
}
void SPP : : RFCOMM_task ( ) {
switch ( l2cap_rfcomm_state ) {
case L2CAP_RFCOMM_WAIT :
if ( l2cap_connection_request_rfcomm_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONNECTION_RFCOMM_REQUEST ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Incoming Connection Request " ) , 0x80 ) ;
# endif
pBtd - > l2cap_connection_response ( hci_handle , identifier , rfcomm_dcid , rfcomm_scid , PENDING ) ;
delay ( 1 ) ;
pBtd - > l2cap_connection_response ( hci_handle , identifier , rfcomm_dcid , rfcomm_scid , SUCCESSFUL ) ;
identifier + + ;
delay ( 1 ) ;
pBtd - > l2cap_config_request ( hci_handle , identifier , rfcomm_scid ) ;
l2cap_rfcomm_state = L2CAP_RFCOMM_REQUEST ;
}
break ;
case L2CAP_RFCOMM_REQUEST :
if ( l2cap_config_request_rfcomm_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONFIG_RFCOMM_REQUEST ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Configuration Request " ) , 0x80 ) ;
# endif
pBtd - > l2cap_config_response ( hci_handle , identifier , rfcomm_scid ) ;
l2cap_rfcomm_state = L2CAP_RFCOMM_SUCCESS ;
}
break ;
case L2CAP_RFCOMM_SUCCESS :
if ( l2cap_config_success_rfcomm_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_CONFIG_RFCOMM_SUCCESS ; // Clear flag
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n RFCOMM Successfully Configured " ) , 0x80 ) ;
# endif
rfcommAvailable = 0 ; // Reset number of bytes available
bytesRead = 0 ; // Reset number of bytes received
RFCOMMConnected = true ;
l2cap_rfcomm_state = L2CAP_RFCOMM_DONE ;
}
break ;
case L2CAP_RFCOMM_DONE :
if ( l2cap_disconnect_request_rfcomm_flag ) {
l2cap_event_flag & = ~ L2CAP_FLAG_DISCONNECT_RFCOMM_REQUEST ; // Clear flag
RFCOMMConnected = false ;
connected = false ;
2013-06-12 05:11:43 +02:00
# ifdef DEBUG_USB_HOST
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Disconnected RFCOMM Channel " ) , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
pBtd - > l2cap_disconnection_response ( hci_handle , identifier , rfcomm_dcid , rfcomm_scid ) ;
l2cap_rfcomm_state = L2CAP_RFCOMM_WAIT ;
} else if ( l2cap_connection_request_rfcomm_flag )
l2cap_rfcomm_state = L2CAP_RFCOMM_WAIT ;
break ;
}
2012-08-04 12:20:47 +02:00
}
/************************************************************/
/* SDP Commands */
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
/************************************************************/
void SPP : : SDP_Command ( uint8_t * data , uint8_t nbytes ) { // See page 223 in the Bluetooth specs
2013-03-28 09:46:43 +01:00
pBtd - > L2CAP_Command ( hci_handle , data , nbytes , sdp_scid [ 0 ] , sdp_scid [ 1 ] ) ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : serviceNotSupported ( uint8_t transactionIDHigh , uint8_t transactionIDLow ) { // See page 235 in the Bluetooth specs
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 0 ] = SDP_SERVICE_SEARCH_ATTRIBUTE_RESPONSE_PDU ;
l2capoutbuf [ 1 ] = transactionIDHigh ;
l2capoutbuf [ 2 ] = transactionIDLow ;
l2capoutbuf [ 3 ] = 0x00 ; // Parameter Length
l2capoutbuf [ 4 ] = 0x05 ; // Parameter Length
l2capoutbuf [ 5 ] = 0x00 ; // AttributeListsByteCount
l2capoutbuf [ 6 ] = 0x02 ; // AttributeListsByteCount
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
/* Attribute ID/Value Sequence: */
l2capoutbuf [ 7 ] = 0x35 ;
l2capoutbuf [ 8 ] = 0x00 ;
l2capoutbuf [ 9 ] = 0x00 ;
2013-03-28 09:37:09 +01:00
2013-03-28 09:46:43 +01:00
SDP_Command ( l2capoutbuf , 10 ) ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : serialPortResponse1 ( uint8_t transactionIDHigh , uint8_t transactionIDLow ) {
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 0 ] = SDP_SERVICE_SEARCH_ATTRIBUTE_RESPONSE_PDU ;
l2capoutbuf [ 1 ] = transactionIDHigh ;
l2capoutbuf [ 2 ] = transactionIDLow ;
l2capoutbuf [ 3 ] = 0x00 ; // Parameter Length
l2capoutbuf [ 4 ] = 0x2B ; // Parameter Length
l2capoutbuf [ 5 ] = 0x00 ; // AttributeListsByteCount
l2capoutbuf [ 6 ] = 0x26 ; // AttributeListsByteCount
/* Attribute ID/Value Sequence: */
l2capoutbuf [ 7 ] = 0x36 ;
l2capoutbuf [ 8 ] = 0x00 ;
l2capoutbuf [ 9 ] = 0x3C ;
l2capoutbuf [ 10 ] = 0x36 ;
l2capoutbuf [ 11 ] = 0x00 ;
l2capoutbuf [ 12 ] = 0x39 ;
l2capoutbuf [ 13 ] = 0x09 ;
l2capoutbuf [ 14 ] = 0x00 ;
l2capoutbuf [ 15 ] = 0x00 ;
l2capoutbuf [ 16 ] = 0x0A ;
l2capoutbuf [ 17 ] = 0x00 ;
l2capoutbuf [ 18 ] = 0x01 ;
l2capoutbuf [ 19 ] = 0x00 ;
l2capoutbuf [ 20 ] = 0x06 ;
l2capoutbuf [ 21 ] = 0x09 ;
l2capoutbuf [ 22 ] = 0x00 ;
l2capoutbuf [ 23 ] = 0x01 ;
l2capoutbuf [ 24 ] = 0x35 ;
l2capoutbuf [ 25 ] = 0x03 ;
l2capoutbuf [ 26 ] = 0x19 ;
l2capoutbuf [ 27 ] = 0x11 ;
l2capoutbuf [ 28 ] = 0x01 ;
l2capoutbuf [ 29 ] = 0x09 ;
l2capoutbuf [ 30 ] = 0x00 ;
l2capoutbuf [ 31 ] = 0x04 ;
l2capoutbuf [ 32 ] = 0x35 ;
l2capoutbuf [ 33 ] = 0x0C ;
l2capoutbuf [ 34 ] = 0x35 ;
l2capoutbuf [ 35 ] = 0x03 ;
l2capoutbuf [ 36 ] = 0x19 ;
l2capoutbuf [ 37 ] = 0x01 ;
l2capoutbuf [ 38 ] = 0x00 ;
l2capoutbuf [ 39 ] = 0x35 ;
l2capoutbuf [ 40 ] = 0x05 ;
l2capoutbuf [ 41 ] = 0x19 ;
l2capoutbuf [ 42 ] = 0x00 ;
l2capoutbuf [ 43 ] = 0x03 ;
l2capoutbuf [ 44 ] = 0x08 ;
l2capoutbuf [ 45 ] = 0x02 ; // Two extra bytes
l2capoutbuf [ 46 ] = 0x00 ; // 25 (0x19) more bytes to come
l2capoutbuf [ 47 ] = 0x19 ;
SDP_Command ( l2capoutbuf , 48 ) ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : serialPortResponse2 ( uint8_t transactionIDHigh , uint8_t transactionIDLow ) {
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 0 ] = SDP_SERVICE_SEARCH_ATTRIBUTE_RESPONSE_PDU ;
l2capoutbuf [ 1 ] = transactionIDHigh ;
l2capoutbuf [ 2 ] = transactionIDLow ;
l2capoutbuf [ 3 ] = 0x00 ; // Parameter Length
l2capoutbuf [ 4 ] = 0x1C ; // Parameter Length
l2capoutbuf [ 5 ] = 0x00 ; // AttributeListsByteCount
l2capoutbuf [ 6 ] = 0x19 ; // AttributeListsByteCount
/* Attribute ID/Value Sequence: */
l2capoutbuf [ 7 ] = 0x01 ;
l2capoutbuf [ 8 ] = 0x09 ;
l2capoutbuf [ 9 ] = 0x00 ;
l2capoutbuf [ 10 ] = 0x06 ;
l2capoutbuf [ 11 ] = 0x35 ;
l2capoutbuf [ 12 ] = 0x09 ;
l2capoutbuf [ 13 ] = 0x09 ;
l2capoutbuf [ 14 ] = 0x65 ;
l2capoutbuf [ 15 ] = 0x6E ;
l2capoutbuf [ 16 ] = 0x09 ;
l2capoutbuf [ 17 ] = 0x00 ;
l2capoutbuf [ 18 ] = 0x6A ;
l2capoutbuf [ 19 ] = 0x09 ;
l2capoutbuf [ 20 ] = 0x01 ;
l2capoutbuf [ 21 ] = 0x00 ;
l2capoutbuf [ 22 ] = 0x09 ;
l2capoutbuf [ 23 ] = 0x01 ;
l2capoutbuf [ 24 ] = 0x00 ;
l2capoutbuf [ 25 ] = 0x25 ;
l2capoutbuf [ 26 ] = 0x05 ; // Name length
l2capoutbuf [ 27 ] = ' T ' ;
l2capoutbuf [ 28 ] = ' K ' ;
l2capoutbuf [ 29 ] = ' J ' ;
l2capoutbuf [ 30 ] = ' S ' ;
l2capoutbuf [ 31 ] = ' P ' ;
l2capoutbuf [ 32 ] = 0x00 ; // No more data
SDP_Command ( l2capoutbuf , 33 ) ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : l2capResponse1 ( uint8_t transactionIDHigh , uint8_t transactionIDLow ) {
2013-03-28 09:46:43 +01:00
serialPortResponse1 ( transactionIDHigh , transactionIDLow ) ; // These has to send all the supported functions, since it only supports virtual serialport it just sends the message again
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
void SPP : : l2capResponse2 ( uint8_t transactionIDHigh , uint8_t transactionIDLow ) {
2013-03-28 09:46:43 +01:00
serialPortResponse2 ( transactionIDHigh , transactionIDLow ) ; // Same data as serialPortResponse2
2012-08-04 12:20:47 +02:00
}
/************************************************************/
/* RFCOMM Commands */
2013-03-28 09:46:43 +01:00
2012-08-04 12:20:47 +02:00
/************************************************************/
void SPP : : RFCOMM_Command ( uint8_t * data , uint8_t nbytes ) {
2013-03-28 09:46:43 +01:00
pBtd - > L2CAP_Command ( hci_handle , data , nbytes , rfcomm_scid [ 0 ] , rfcomm_scid [ 1 ] ) ;
2012-08-04 12:20:47 +02:00
}
void SPP : : sendRfcomm ( uint8_t channel , uint8_t direction , uint8_t CR , uint8_t channelType , uint8_t pfBit , uint8_t * data , uint8_t length ) {
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 0 ] = channel | direction | CR | extendAddress ; // RFCOMM Address
l2capoutbuf [ 1 ] = channelType | pfBit ; // RFCOMM Control
2013-04-07 17:21:28 +02:00
l2capoutbuf [ 2 ] = length < < 1 | 0x01 ; // Length and format (always 0x01 bytes format)
2013-03-28 09:46:43 +01:00
uint8_t i = 0 ;
for ( ; i < length ; i + + )
l2capoutbuf [ i + 3 ] = data [ i ] ;
l2capoutbuf [ i + 3 ] = calcFcs ( l2capoutbuf ) ;
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " - RFCOMM Data: " ) , 0x80 ) ;
for ( i = 0 ; i < length + 4 ; i + + ) {
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capoutbuf [ i ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
}
2013-03-28 09:37:09 +01:00
# endif
2013-03-28 09:46:43 +01:00
RFCOMM_Command ( l2capoutbuf , length + 4 ) ;
2013-03-28 09:37:09 +01:00
}
2012-08-04 12:20:47 +02:00
void SPP : : sendRfcommCredit ( uint8_t channel , uint8_t direction , uint8_t CR , uint8_t channelType , uint8_t pfBit , uint8_t credit ) {
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 0 ] = channel | direction | CR | extendAddress ; // RFCOMM Address
l2capoutbuf [ 1 ] = channelType | pfBit ; // RFCOMM Control
l2capoutbuf [ 2 ] = 0x01 ; // Length = 0
l2capoutbuf [ 3 ] = credit ; // Credit
l2capoutbuf [ 4 ] = calcFcs ( l2capoutbuf ) ;
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " - RFCOMM Credit Data: " ) , 0x80 ) ;
for ( uint8_t i = 0 ; i < 5 ; i + + ) {
2013-06-17 21:37:09 +02:00
D_PrintHex < uint8_t > ( l2capoutbuf [ i ] , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " " ) , 0x80 ) ;
}
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
RFCOMM_Command ( l2capoutbuf , 5 ) ;
2012-08-04 12:20:47 +02:00
}
/* CRC on 2 bytes */
uint8_t SPP : : __crc ( uint8_t * data ) {
2013-03-28 09:46:43 +01:00
return ( pgm_read_byte ( & rfcomm_crc_table [ pgm_read_byte ( & rfcomm_crc_table [ 0xff ^ data [ 0 ] ] ) ^ data [ 1 ] ] ) ) ;
2012-08-04 12:20:47 +02:00
}
/* Calculate FCS - we never actually check if the host sends correct FCS to the Arduino */
uint8_t SPP : : calcFcs ( uint8_t * data ) {
2013-03-28 09:46:43 +01:00
if ( ( data [ 1 ] & 0xEF ) = = RFCOMM_UIH )
return ( 0xff - __crc ( data ) ) ; // FCS on 2 bytes
else
return ( 0xff - pgm_read_byte ( & rfcomm_crc_table [ __crc ( data ) ^ data [ 2 ] ] ) ) ; // FCS on 3 bytes
2012-08-04 12:20:47 +02:00
}
/* Serial commands */
2013-05-07 00:06:49 +02:00
size_t SPP : : write ( uint8_t data ) {
return write ( & data , 1 ) ;
2013-04-08 00:25:01 +02:00
}
2013-04-03 20:36:55 +02:00
2013-05-07 00:06:49 +02:00
size_t SPP : : write ( const uint8_t * data , size_t size ) {
for ( uint8_t i = 0 ; i < size ; i + + ) {
if ( sppIndex > = sizeof ( sppOutputBuffer ) / sizeof ( sppOutputBuffer [ 0 ] ) )
2013-05-07 00:37:43 +02:00
send ( ) ; // Send the current data in the buffer
2013-06-17 21:37:09 +02:00
sppOutputBuffer [ sppIndex + + ] = data [ i ] ; // All the bytes are put into a buffer and then send using the send() function
2013-05-07 00:06:49 +02:00
}
return size ;
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2013-05-07 00:06:49 +02:00
void SPP : : send ( ) {
if ( ! connected | | ! sppIndex )
2013-03-28 09:46:43 +01:00
return ;
2013-04-03 20:36:55 +02:00
uint8_t length ; // This is the length of the string we are sending
uint8_t offset = 0 ; // This is used to keep track of where we are in the string
2013-04-03 19:42:24 +02:00
l2capoutbuf [ 0 ] = rfcommChannelConnection | 0 | 0 | extendAddress ; // RFCOMM Address
2013-03-28 09:46:43 +01:00
l2capoutbuf [ 1 ] = RFCOMM_UIH ; // RFCOMM Control
2013-05-07 00:06:49 +02:00
while ( sppIndex ) { // We will run this while loop until this variable is 0
if ( sppIndex > ( sizeof ( l2capoutbuf ) - 4 ) ) // Check if the string is larger than the outgoing buffer
2013-04-03 20:36:55 +02:00
length = sizeof ( l2capoutbuf ) - 4 ;
else
2013-05-07 00:06:49 +02:00
length = sppIndex ;
2013-04-03 20:36:55 +02:00
l2capoutbuf [ 2 ] = length < < 1 | 1 ; // Length
uint8_t i = 0 ;
for ( ; i < length ; i + + )
2013-05-07 00:06:49 +02:00
l2capoutbuf [ i + 3 ] = sppOutputBuffer [ i + offset ] ;
2013-04-03 20:36:55 +02:00
l2capoutbuf [ i + 3 ] = calcFcs ( l2capoutbuf ) ; // Calculate checksum
RFCOMM_Command ( l2capoutbuf , length + 4 ) ;
2013-05-07 00:06:49 +02:00
sppIndex - = length ;
2013-04-03 20:36:55 +02:00
offset + = length ; // Increment the offset
2013-04-07 17:21:28 +02:00
}
2012-08-04 12:20:47 +02:00
}
2013-03-28 09:46:43 +01:00
2013-05-07 00:06:49 +02:00
int SPP : : available ( void ) {
return rfcommAvailable ;
} ;
2013-03-28 09:46:43 +01:00
2013-05-07 00:06:49 +02:00
void SPP : : flush ( void ) {
rfcommAvailable = 0 ;
2012-12-16 18:42:02 +01:00
}
2013-03-28 09:46:43 +01:00
2013-05-07 00:06:49 +02:00
int SPP : : peek ( void ) {
if ( rfcommAvailable = = 0 ) // Don't read if there is nothing in the buffer
return - 1 ;
return rfcommDataBuffer [ 0 ] ;
2012-12-16 18:42:02 +01:00
}
2012-08-04 12:20:47 +02:00
2013-05-07 00:06:49 +02:00
int SPP : : read ( void ) {
2013-03-28 09:46:43 +01:00
if ( rfcommAvailable = = 0 ) // Don't read if there is nothing in the buffer
2013-05-07 00:06:49 +02:00
return - 1 ;
2013-03-28 09:46:43 +01:00
uint8_t output = rfcommDataBuffer [ 0 ] ;
for ( uint8_t i = 1 ; i < rfcommAvailable ; i + + )
rfcommDataBuffer [ i - 1 ] = rfcommDataBuffer [ i ] ; // Shift the buffer one left
rfcommAvailable - - ;
bytesRead + + ;
if ( bytesRead > ( sizeof ( rfcommDataBuffer ) - 5 ) ) { // We will send the command just before it runs out of credit
bytesRead = 0 ;
sendRfcommCredit ( rfcommChannelConnection , rfcommDirection , 0 , RFCOMM_UIH , 0x10 , sizeof ( rfcommDataBuffer ) ) ; // Send more credit
2012-08-04 12:20:47 +02:00
# ifdef EXTRADEBUG
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " \r \n Sent " ) , 0x80 ) ;
2013-04-26 23:50:39 +02:00
Notify ( ( uint8_t ) sizeof ( rfcommDataBuffer ) , 0x80 ) ;
2013-03-28 09:46:43 +01:00
Notify ( PSTR ( " more credit " ) , 0x80 ) ;
2012-08-04 12:20:47 +02:00
# endif
2013-03-28 09:46:43 +01:00
}
return output ;
2013-09-27 11:07:05 +02:00
}