-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathble_temperature_broadcast.ino
345 lines (307 loc) · 11.7 KB
/
ble_temperature_broadcast.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
/* Copyright (c) 2014, Nordic Semiconductor ASA
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
/**
*
* IMPORTANT: This example still is not compatible with CHIPKIT
*
* Click on the "Serial Monitor" button on the Arduino IDE to get reset the Arduino and start the application.
* The setup() function is called first and is called only one for each reset of the Arduino.
* The loop() function as the name implies is called in a loop.
* The setup() and loop() function are called in this way.
* main()
* {
* setup();
* while(1)
* {
* loop();
* }
* }
*
* Use the PERSONAL HEALTH DEVICES TRANSCODING WHITE PAPER in bluetooth.org to understand the
* format of the temperature measurement characteristic.
* The format used is IEEE 11073-20601 FLOAT
*/
#include <SPI.h>
#include <EEPROM.h>
#include <avr/sleep.h>
#include <avr/power.h>
#include "services.h"
#include <lib_aci.h>
#include <aci_setup.h>
#include "timer1.h"
#ifdef SERVICES_PIPE_TYPE_MAPPING_CONTENT
static services_pipe_type_mapping_t
services_pipe_type_mapping[NUMBER_OF_PIPES] = SERVICES_PIPE_TYPE_MAPPING_CONTENT;
#else
#define NUMBER_OF_PIPES 0
static services_pipe_type_mapping_t * services_pipe_type_mapping = NULL;
#endif
#define TEMPERATURE_NUM_SAMPLES 5
static const hal_aci_data_t setup_msgs[NB_SETUP_MESSAGES] PROGMEM = SETUP_MESSAGES_CONTENT;
// aci_struct that will contain
// total initial credits
// current credit
// current state of the aci (setup/standby/active/sleep)
// open remote pipe pending
// close remote pipe pending
// Current pipe available bitmap
// Current pipe closed bitmap
// Current connection interval, slave latency and link supervision timeout
// Current State of the the GATT client (Service Discovery)
static struct aci_state_t aci_state;
static hal_aci_evt_t aci_data;
/**
Counter in seconds.
When this counter counts down to zero -> wakeup the nRF8001
*/
static int16_t sleep_to_wakeup_timeout;
/*
Variables used for the temperature measurement and transmission
*/
//static h_thermo_temp_measure_t h_temperature;
//static h_temp_type_t current_type;
static int32_t temperature = 0; // Needs to be an int32 to measure negative values
static float temperature_f;
static uint8_t temperature_count = TEMPERATURE_NUM_SAMPLES;
static int32_t temperature_total = 0;
static bool first_temp_measure_pending = true;
/*
Variables used for the timer on the AVR
*/
volatile uint8_t timer1_f = 1;
/* Define how assert should function in the BLE library */
void __ble_assert(const char *file, uint16_t line)
{
Serial.print("ERROR ");
Serial.print(file);
Serial.print(": ");
Serial.print(line);
Serial.print("\n");
while(1);
}
void aci_loop()
{
static bool setup_required = false;
// We enter the if statement only when there is a ACI event available to be processed
if (lib_aci_event_get(&aci_state, &aci_data))
{
aci_evt_t * aci_evt;
aci_evt = &aci_data.evt;
switch(aci_evt->evt_opcode)
{
case ACI_EVT_DEVICE_STARTED:
{
aci_state.data_credit_available = aci_evt->params.device_started.credit_available;
switch(aci_evt->params.device_started.device_mode)
{
case ACI_DEVICE_SETUP:
/**
When the device is in the setup mode
*/
aci_state.device_state = ACI_DEVICE_SETUP;
Serial.println(F("Evt Device Started: Setup"));
setup_required = true;
break;
case ACI_DEVICE_STANDBY:
aci_state.device_state = ACI_DEVICE_STANDBY;
sleep_to_wakeup_timeout = 30;
Serial.println(F("Evt Device Started: Standby"));
if (aci_evt->params.device_started.hw_error)
{
delay(20); //Magic number used to make sure the HW error event is handled correctly.
}
else
{
// prepare first temperature measurement
Timer1start();
}
break;
}
}
break; //ACI Device Started Event
case ACI_EVT_CMD_RSP:
//If an ACI command response event comes with an error -> stop
if (ACI_STATUS_SUCCESS != aci_evt->params.cmd_rsp.cmd_status )
{
//ACI ReadDynamicData and ACI WriteDynamicData will have status codes of
//TRANSACTION_CONTINUE and TRANSACTION_COMPLETE
//all other ACI commands will have status code of ACI_STATUS_SCUCCESS for a successful command
Serial.print(F("ACI Status of ACI Evt Cmd Rsp 0x"));
Serial.println(aci_evt->params.cmd_rsp.cmd_status, HEX);
Serial.print(F("ACI Command 0x"));
Serial.println(aci_evt->params.cmd_rsp.cmd_opcode, HEX);
Serial.println(F("Evt Cmd respone: Error. Arduino is in an while(1); loop"));
while (1);
}
else
{
switch (aci_evt->params.cmd_rsp.cmd_opcode)
{
case ACI_CMD_GET_TEMPERATURE:
if (0 != temperature_count)
{
temperature_total = temperature_total +
(aci_evt->params.cmd_rsp.params.get_temperature.temperature_value);
Serial.print(F("Sampling Temperature "));
Serial.print(temperature_count);
Serial.print(F(" "));
Serial.println(aci_evt->params.cmd_rsp.params.get_temperature.temperature_value);
temperature_count--;
}
if (0 == temperature_count)
{
Serial.println(temperature_total);
temperature_count = TEMPERATURE_NUM_SAMPLES;
/**
Multiply by 100 for exp = -2 : divide by 4 -> See ACI GetTemperature in datasheet
*/
temperature = temperature_total * (float)(25/(float)TEMPERATURE_NUM_SAMPLES);
temperature_total = 0;
temperature_f = (float)temperature/100.00;
Serial.print(F("Temperature :"));
Serial.print(temperature_f, 2);
Serial.println(" C");
if (true)//lib_aci_is_pipe_available(&aci_state, PIPE_HEALTH_THERMOMETER_TEMPERATURE_MEASUREMENT_SET))
{
Serial.println(F("Setting the temperature"));
temperature &= 0x00FFFFFF; //Mask the exponent part
temperature |= 0xFE000000; //Exponent is -2 since we multipled by 100
lib_aci_set_local_data(&aci_state, PIPE_HEALTH_THERMOMETER_TEMPERATURE_MEASUREMENT_SET, (uint8_t*) &temperature, 4);
//Start broadcasting our temperature value, now that we have one
if (first_temp_measure_pending)
{
lib_aci_open_adv_pipe(PIPE_HEALTH_THERMOMETER_TEMPERATURE_MEASUREMENT_BROADCAST);
lib_aci_broadcast(0/* indefinitely */, 0x0100 /* advertising interval 100ms*/);
Serial.println(F("Broadcasting started"));
first_temp_measure_pending = false;
}
}
}
break;
}
}
break;
case ACI_EVT_HW_ERROR:
Serial.println(F("HW error: "));
Serial.println(aci_evt->params.hw_error.line_num, DEC);
for(uint8_t counter = 0; counter <= (aci_evt->len - 3); counter++)
{
Serial.write(aci_evt->params.hw_error.file_name[counter]); //uint8_t file_name[20];
}
break;
default:
Serial.print(F("Evt Opcode 0x"));
Serial.print(aci_evt->evt_opcode, HEX);
Serial.println(F(" unhandled"));
break;
}
}
else
{
/**
No event in the ACI Event queue and if there are no commands in the command queue
Arduino can go to sleep
*/
}
/* setup_required is set to true when the device starts up and enters setup mode.
* It indicates that do_aci_setup() should be called. The flag should be cleared if
* do_aci_setup() returns ACI_STATUS_TRANSACTION_COMPLETE.
*/
if(setup_required)
{
if (SETUP_SUCCESS == do_aci_setup(&aci_state))
{
setup_required = false;
}
}
}
void setup(void)
{
Serial.begin(115200);
//Wait until the serial port is available (useful only for the Leonardo)
//As the Leonardo board is not reseted every time you open the Serial Monitor
#if defined (__AVR_ATmega32U4__)
while(!Serial)
{}
delay(5000); //5 seconds delay for enabling to see the start up comments on the serial board
#elif defined(__PIC32MX__)
delay(1000);
#endif
Serial.println(F("Arduino setup"));
/**
Point ACI data structures to the the setup data that the nRFgo studio generated for the nRF8001
*/
if (NULL != services_pipe_type_mapping)
{
aci_state.aci_setup_info.services_pipe_type_mapping = &services_pipe_type_mapping[0];
}
else
{
aci_state.aci_setup_info.services_pipe_type_mapping = NULL;
}
aci_state.aci_setup_info.number_of_pipes = NUMBER_OF_PIPES;
aci_state.aci_setup_info.setup_msgs = (hal_aci_data_t*) setup_msgs;
aci_state.aci_setup_info.num_setup_msgs = NB_SETUP_MESSAGES;
/*
Tell the ACI library, the MCU to nRF8001 pin connections.
The Active pin is optional and can be marked UNUSED
*/
aci_state.aci_pins.board_name = BOARD_DEFAULT; //See board.h for details
aci_state.aci_pins.reqn_pin = 9;
aci_state.aci_pins.rdyn_pin = 8;
aci_state.aci_pins.mosi_pin = MOSI;
aci_state.aci_pins.miso_pin = MISO;
aci_state.aci_pins.sck_pin = SCK;
aci_state.aci_pins.spi_clock_divider = SPI_CLOCK_DIV8;//SPI_CLOCK_DIV8 = 2MHz SPI speed
//SPI_CLOCK_DIV16 = 1MHz SPI speed
aci_state.aci_pins.reset_pin = 4; //4 for Nordic board, UNUSED for REDBEARLAB_SHIELD_V1_1
aci_state.aci_pins.active_pin = UNUSED;
aci_state.aci_pins.optional_chip_sel_pin = UNUSED;
aci_state.aci_pins.interface_is_interrupt = false; //Interrupts still not available in Chipkit
aci_state.aci_pins.interrupt_number = 1;
/** We reset the nRF8001 here by toggling the RESET line connected to the nRF8001
* and initialize the data structures required to setup the nRF8001
*/
//The second parameter is for turning debug printing on for the ACI Commands and Events so they be printed on the Serial
lib_aci_init(&aci_state, false);
/**
* Initalize the data structures required for the GATT Health Thermometer Service
*/
}
void loop()
{
aci_loop();
/**
Temperature sampling application that samples the temperature every 4 seconds
OR
Wakes up the sleeping nRF8001 every 4 seconds
*/
if (1 == timer1_f)
{
uint8_t i = 0;
timer1_f = 0;
for(i=0; i<TEMPERATURE_NUM_SAMPLES; i++)
{
Serial.println(F("Get temperature from nRF8001 sensor"));
lib_aci_get_temperature();
}
}
}