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 
92 class SPP : public BluetoothService {
93 public:
94  SPP(BTD *p, const char* name = "Arduino", const char* pin = "1234");
95 
96  // BluetoothService implementation
97  virtual void ACLData(uint8_t* ACLData); // Used to pass acldata to the services
98  virtual void Run(); // Used to establish the connection automatically
99  virtual void Reset(); // Use this to reset the service
100  virtual void disconnect(); // Used this void to disconnect the virtual serial port
101 
102  bool connected;// Variable used to indicate if the connection is established
103 
104  /* Serial port profile (SPP) commands */
105  void print(const String &); // Used to send strings
106  void print(const char* data); // Used to send strings
107  void print(uint8_t data); // Used to send single bytes
108  void print(uint8_t* array, uint8_t length); // Used to send arrays
109  void print(const __FlashStringHelper *); // Used to print strings stored in flash
110 
111  void println(const String &); // Include newline and carriage return
112  void println(const char* data); // Include newline and carriage return
113  void println(uint8_t data); // Include newline and carriage return
114  void println(uint8_t* array, uint8_t length); // Include newline and carriage return
115  void println(const __FlashStringHelper *); // Include newline and carriage return
116  void println(void); // Use this to print newline and carriage return
117 
118  void printNumber(int16_t n); // These must be used to print numbers
119  void printNumberln(int16_t n); // This will include newline and carriage return
120  void printNumber(double n, uint8_t digits = 2); // These must be used to print floating-point numbers
121  void printNumberln(double n, uint8_t digits = 2); // This will include newline and carriage return
122 
123  const char* doubleToString(double input, uint8_t digits); // Helper function to convert from double to string
124 
125  uint8_t available() { return rfcommAvailable; }; // Get the bytes waiting to be read
126  uint8_t read(); // Used to read the buffer
127  void flush() { rfcommAvailable = 0; }; // Discard all the bytes in the buffer
128 
129 private:
130  /* Bluetooth dongle library pointer */
131  BTD *pBtd;
132 
133  /* Set true when a channel is created */
134  bool SDPConnected;
135  bool RFCOMMConnected;
136 
137  uint16_t hci_handle; // The HCI Handle for the connection
138 
139  /* Variables used by L2CAP state maschines */
140  uint8_t l2cap_sdp_state;
141  uint8_t l2cap_rfcomm_state;
142  uint16_t l2cap_event_flag; // l2cap flags of received bluetooth events
143 
144  uint8_t l2capoutbuf[BULK_MAXPKTSIZE]; // General purpose buffer for l2cap out data
145  uint8_t rfcommbuf[10]; // Buffer for RFCOMM Commands
146 
147  /* L2CAP Channels */
148  uint8_t sdp_scid[2]; // L2CAP source CID for SDP
149  uint8_t sdp_dcid[2]; // 0x0050
150  uint8_t rfcomm_scid[2]; // L2CAP source CID for RFCOMM
151  uint8_t rfcomm_dcid[2]; // 0x0051
152  uint8_t identifier; // Identifier for command
153 
154  /* RFCOMM Variables */
155  uint8_t rfcommChannel;
156  uint8_t rfcommChannelConnection; // This is the channel the SPP chanel will be running at
157  uint8_t rfcommDirection;
158  uint8_t rfcommCommandResponse;
159  uint8_t rfcommChannelType;
160  uint8_t rfcommPfBit;
161 
162  unsigned long timer;
163  bool waitForLastCommand;
164  bool creditSent;
165 
166  uint8_t rfcommDataBuffer[100]; // Create a 100 sized buffer for incoming data
167  uint8_t rfcommAvailable;
168 
169  bool firstMessage; // Used to see if it's the first SDP request received
170  uint8_t bytesRead; // Counter to see when it's time to send more credit
171 
172  unsigned long printTimer; // Used to set a delay, so it doesn't try to print too fast
173 
174  /* State machines */
175  void SDP_task(); // SDP state machine
176  void RFCOMM_task(); // RFCOMM state machine
177 
178  /* SDP Commands */
179  void SDP_Command(uint8_t* data, uint8_t nbytes);
180  void serviceNotSupported(uint8_t transactionIDHigh, uint8_t transactionIDLow);
181  void serialPortResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
182  void serialPortResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
183  void l2capResponse1(uint8_t transactionIDHigh, uint8_t transactionIDLow);
184  void l2capResponse2(uint8_t transactionIDHigh, uint8_t transactionIDLow);
185 
186  /* RFCOMM Commands */
187  void RFCOMM_Command(uint8_t* data, uint8_t nbytes);
188  void sendRfcomm(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t* data, uint8_t length);
189  void sendRfcommCredit(uint8_t channel, uint8_t direction, uint8_t CR, uint8_t channelType, uint8_t pfBit, uint8_t credit);
190  uint8_t calcFcs(uint8_t *data);
191  uint8_t __crc(uint8_t* data);
192 };
193 #endif