USB Host Shield 2.0
 All Classes Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
SPP.h
Go to the documentation of this file.
1 /* Copyright (C) 2012 Kristian Lauszus, TKJ Electronics. All rights reserved.
2 
3  This software may be distributed and modified under the terms of the GNU
4  General Public License version 2 (GPL2) as published by the Free Software
5  Foundation and appearing in the file GPL2.TXT included in the packaging of
6  this file. Please note that GPL2 Section 2[b] requires that all works based
7  on this software must also be made publicly available under the terms of
8  the GPL2 ("Copyleft").
9 
10  Contact information
11  -------------------
12 
13  Kristian Lauszus, TKJ Electronics
14  Web : http://www.tkjelectronics.com
15  e-mail : kristianl@tkjelectronics.com
16  */
17 
18 #ifndef _spp_h_
19 #define _spp_h_
20 
21 #include "BTD.h"
22 
23 /* Bluetooth L2CAP states for SDP_task() */
24 #define L2CAP_SDP_WAIT 0
25 #define L2CAP_SDP_REQUEST 1
26 #define L2CAP_SDP_SUCCESS 2
27 #define L2CAP_SDP_DONE 3
28 #define L2CAP_DISCONNECT_RESPONSE 4
29 
30 /* Bluetooth L2CAP states for RFCOMM_task() */
31 #define L2CAP_RFCOMM_WAIT 0
32 #define L2CAP_RFCOMM_REQUEST 1
33 #define L2CAP_RFCOMM_SUCCESS 2
34 #define L2CAP_RFCOMM_DONE 3
35 
36 /* L2CAP event flags */
37 #define L2CAP_FLAG_CONNECTION_SDP_REQUEST 0x001
38 #define L2CAP_FLAG_CONNECTION_RFCOMM_REQUEST 0x002
39 #define L2CAP_FLAG_CONFIG_SDP_REQUEST 0x004
40 #define L2CAP_FLAG_CONFIG_RFCOMM_REQUEST 0x008
41 #define L2CAP_FLAG_CONFIG_SDP_SUCCESS 0x010
42 #define L2CAP_FLAG_CONFIG_RFCOMM_SUCCESS 0x020
43 #define L2CAP_FLAG_DISCONNECT_SDP_REQUEST 0x040
44 #define L2CAP_FLAG_DISCONNECT_RFCOMM_REQUEST 0x080
45 #define L2CAP_FLAG_DISCONNECT_RESPONSE 0x100
46 
47 /* Macros for L2CAP event flag tests */
48 #define l2cap_connection_request_sdp_flag (l2cap_event_flag & L2CAP_FLAG_CONNECTION_SDP_REQUEST)
49 #define l2cap_connection_request_rfcomm_flag (l2cap_event_flag & L2CAP_FLAG_CONNECTION_RFCOMM_REQUEST)
50 #define l2cap_config_request_sdp_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_SDP_REQUEST)
51 #define l2cap_config_request_rfcomm_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_RFCOMM_REQUEST)
52 #define l2cap_config_success_sdp_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_SDP_SUCCESS)
53 #define l2cap_config_success_rfcomm_flag (l2cap_event_flag & L2CAP_FLAG_CONFIG_RFCOMM_SUCCESS)
54 #define l2cap_disconnect_request_sdp_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_SDP_REQUEST)
55 #define l2cap_disconnect_request_rfcomm_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_RFCOMM_REQUEST)
56 #define l2cap_disconnect_response_flag (l2cap_event_flag & L2CAP_FLAG_DISCONNECT_RESPONSE)
57 
58 /* Used for SDP */
59 #define SDP_SERVICE_SEARCH_ATTRIBUTE_REQUEST_PDU 0x06 // See the RFCOMM specs
60 #define SDP_SERVICE_SEARCH_ATTRIBUTE_RESPONSE_PDU 0x07 // See the RFCOMM specs
61 #define SERIALPORT_UUID 0x1101 // See http://www.bluetooth.org/Technical/AssignedNumbers/service_discovery.htm
62 #define L2CAP_UUID 0x0100
63 
64 /* Used for RFCOMM */
65 #define RFCOMM_SABM 0x2F
66 #define RFCOMM_UA 0x63
67 #define RFCOMM_UIH 0xEF
68 //#define RFCOMM_DM 0x0F
69 #define RFCOMM_DISC 0x43
70 
71 #define extendAddress 0x01 // Always 1
72 
73 // Multiplexer message types
74 #define BT_RFCOMM_PN_CMD 0x83
75 #define BT_RFCOMM_PN_RSP 0x81
76 #define BT_RFCOMM_MSC_CMD 0xE3
77 #define BT_RFCOMM_MSC_RSP 0xE1
78 #define BT_RFCOMM_RPN_CMD 0x93
79 #define BT_RFCOMM_RPN_RSP 0x91
80 /*
81 #define BT_RFCOMM_TEST_CMD 0x23
82 #define BT_RFCOMM_TEST_RSP 0x21
83 #define BT_RFCOMM_FCON_CMD 0xA3
84 #define BT_RFCOMM_FCON_RSP 0xA1
85 #define BT_RFCOMM_FCOFF_CMD 0x63
86 #define BT_RFCOMM_FCOFF_RSP 0x61
87 #define BT_RFCOMM_RLS_CMD 0x53
88 #define BT_RFCOMM_RLS_RSP 0x51
89 #define BT_RFCOMM_NSC_RSP 0x11
90  */
91 
93 class SPP : public BluetoothService, public Stream {
94 public:
101  SPP(BTD *p, const char* name = "Arduino", const char* pin = "1234");
102 
107  operator bool() {
108  return connected;
109  }
111  bool connected;
112 
118  virtual void ACLData(uint8_t* ACLData);
120  virtual void Run();
122  virtual void Reset();
124  virtual void disconnect();
132  virtual int available(void);
134  virtual void flush(void);
139  virtual int peek(void);
144  virtual int read(void);
150  virtual size_t write(uint8_t data);
157  virtual size_t write(const uint8_t* data, size_t size);
159  using Print::write;
165  void send(void);
168 private:
169  /* Bluetooth dongle library pointer */
170  BTD *pBtd;
171 
172  /* Set true when a channel is created */
173  bool SDPConnected;
174  bool RFCOMMConnected;
175 
176  uint16_t hci_handle; // The HCI Handle for the connection
177 
178  /* Variables used by L2CAP state machines */
179  uint8_t l2cap_sdp_state;
180  uint8_t l2cap_rfcomm_state;
181  uint16_t l2cap_event_flag; // l2cap flags of received Bluetooth events
182 
183  uint8_t l2capoutbuf[BULK_MAXPKTSIZE]; // General purpose buffer for l2cap out data
184  uint8_t rfcommbuf[10]; // Buffer for RFCOMM Commands
185 
186  /* L2CAP Channels */
187  uint8_t sdp_scid[2]; // L2CAP source CID for SDP
188  uint8_t sdp_dcid[2]; // 0x0050
189  uint8_t rfcomm_scid[2]; // L2CAP source CID for RFCOMM
190  uint8_t rfcomm_dcid[2]; // 0x0051
191  uint8_t identifier; // Identifier for command
192 
193  /* RFCOMM Variables */
194  uint8_t rfcommChannel;
195  uint8_t rfcommChannelConnection; // This is the channel the SPP channel will be running at
196  uint8_t rfcommDirection;
197  uint8_t rfcommCommandResponse;
198  uint8_t rfcommChannelType;
199  uint8_t rfcommPfBit;
200 
201  unsigned long timer;
202  bool waitForLastCommand;
203  bool creditSent;
204 
205  uint8_t rfcommDataBuffer[100]; // Create a 100 sized buffer for incoming data
206  uint8_t sppOutputBuffer[100]; // Create a 100 sized buffer for outgoing SPP data
207  uint8_t sppIndex;
208  uint8_t rfcommAvailable;
209 
210  bool firstMessage; // Used to see if it's the first SDP request received
211  uint8_t bytesRead; // Counter to see when it's time to send more credit
212 
213  /* State machines */
214  void SDP_task(); // SDP state machine
215  void RFCOMM_task(); // RFCOMM state machine
216 
217  /* SDP Commands */
218  void SDP_Command(uint8_t *data, uint8_t nbytes);
219  void serviceNotSupported(uint8_t transactionIDHigh, uint8_t transactionIDLow);
220  void serialPortResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
221  void serialPortResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
222  void l2capResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
223  void l2capResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
224 
225  /* RFCOMM Commands */
226  void RFCOMM_Command(uint8_t *data, uint8_t nbytes);
227  void sendRfcomm(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t *data, uint8_t length);
228  void sendRfcommCredit(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t credit);
229  uint8_t calcFcs(uint8_t *data);
230  bool checkFcs(uint8_t *data, uint8_t fcs);
231  uint8_t crc(uint8_t *data);
232 };
233 #endif
virtual size_t write(uint8_t data)
Definition: SPP.cpp:759
Definition: BTD.h:158
virtual void flush(void)
Definition: SPP.cpp:804
bool connected
Definition: SPP.h:111
virtual int read(void)
Definition: SPP.cpp:814
virtual void Run()
Definition: SPP.cpp:429
virtual void Reset()
Definition: SPP.cpp:63
SPP(BTD *p, const char *name="Arduino", const char *pin="1234")
Definition: SPP.cpp:45
virtual int available(void)
Definition: SPP.cpp:800
#define BULK_MAXPKTSIZE
Definition: BTD.h:33
virtual void disconnect()
Definition: SPP.cpp:74
Definition: SPP.h:93
void send(void)
Definition: SPP.cpp:772
virtual void ACLData(uint8_t *ACLData)
Definition: SPP.cpp:86
virtual int peek(void)
Definition: SPP.cpp:808