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 // Allways 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 {
94 public:
101  SPP(BTD *p, const char* name = "Arduino", const char* pin = "1234");
102 
108  virtual void ACLData(uint8_t* ACLData);
110  virtual void Run();
112  virtual void Reset();
114  virtual void disconnect();
118  bool connected;
119 
126  void print(const String &str);
131  void println(const String &str);
136  void print(const char* str);
141  void println(const char* str);
146  void print(uint8_t data);
151  void println(uint8_t data);
157  void print(uint8_t* array, uint8_t length);
163  void println(uint8_t* array, uint8_t length);
168  void print(const __FlashStringHelper *ifsh);
173  void println(const __FlashStringHelper *ifsh);
175  void println(void);
176 
181  void printNumber(int32_t n);
186  void printNumberln(int32_t n);
192  void printNumber(double n, uint8_t digits = 2);
198  void printNumberln(double n, uint8_t digits = 2);
199 
206  void doubleToString(double input, char* output, uint8_t digits = 2);
207 
212  uint8_t available() { return rfcommAvailable; };
217  uint8_t read();
219  void flush() { rfcommAvailable = 0; };
222 private:
223  /* Bluetooth dongle library pointer */
224  BTD *pBtd;
225 
226  /* Set true when a channel is created */
227  bool SDPConnected;
228  bool RFCOMMConnected;
229 
230  uint16_t hci_handle; // The HCI Handle for the connection
231 
232  /* Variables used by L2CAP state maschines */
233  uint8_t l2cap_sdp_state;
234  uint8_t l2cap_rfcomm_state;
235  uint16_t l2cap_event_flag; // l2cap flags of received bluetooth events
236 
237  uint8_t l2capoutbuf[BULK_MAXPKTSIZE]; // General purpose buffer for l2cap out data
238  uint8_t rfcommbuf[10]; // Buffer for RFCOMM Commands
239 
240  /* L2CAP Channels */
241  uint8_t sdp_scid[2]; // L2CAP source CID for SDP
242  uint8_t sdp_dcid[2]; // 0x0050
243  uint8_t rfcomm_scid[2]; // L2CAP source CID for RFCOMM
244  uint8_t rfcomm_dcid[2]; // 0x0051
245  uint8_t identifier; // Identifier for command
246 
247  /* RFCOMM Variables */
248  uint8_t rfcommChannel;
249  uint8_t rfcommChannelConnection; // This is the channel the SPP chanel will be running at
250  uint8_t rfcommDirection;
251  uint8_t rfcommCommandResponse;
252  uint8_t rfcommChannelType;
253  uint8_t rfcommPfBit;
254 
255  unsigned long timer;
256  bool waitForLastCommand;
257  bool creditSent;
258 
259  uint8_t rfcommDataBuffer[100]; // Create a 100 sized buffer for incoming data
260  uint8_t rfcommAvailable;
261 
262  bool firstMessage; // Used to see if it's the first SDP request received
263  uint8_t bytesRead; // Counter to see when it's time to send more credit
264 
265  unsigned long printTimer; // Used to set a delay, so it doesn't try to print too fast
266 
267  /* State machines */
268  void SDP_task(); // SDP state machine
269  void RFCOMM_task(); // RFCOMM state machine
270 
271  /* SDP Commands */
272  void SDP_Command(uint8_t* data, uint8_t nbytes);
273  void serviceNotSupported(uint8_t transactionIDHigh, uint8_t transactionIDLow);
274  void serialPortResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
275  void serialPortResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
276  void l2capResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
277  void l2capResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
278 
279  /* RFCOMM Commands */
280  void RFCOMM_Command(uint8_t* data, uint8_t nbytes);
281  void sendRfcomm(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t* data, uint8_t length);
282  void sendRfcommCredit(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t credit);
283  uint8_t calcFcs(uint8_t *data);
284  uint8_t __crc(uint8_t* data);
285 };
286 #endif