Geo beacon for VF.
Dependencies: MMA8452 aconno_bsp adc52832_common
main.cpp
- Committer:
- jurica238814
- Date:
- 2017-07-19
- Revision:
- 6:d14e3df498f4
- Parent:
- 5:0f42dcae4cdf
- Child:
- 8:570eb66d50b5
File content as of revision 6:d14e3df498f4:
/*
*
* Made by Jurica Resetar @ aconno
* aconno.de
* All rights reserved
*
*/
#include "mbed.h"
#include "ble/BLE.h"
#include "GapAdvertisingData.h"
#include "acd52832_bsp.h"
#include "mma8452.h"
#define SLEEP_TIME (0.85) // Sleep time in seconds WAS 0.85
#define AWAKE_TIME (0.15) // Was 0.15
#define BUZZER (p31)
/* Static constants for the BLE example */
#define MAX_BLE_PACKET_SIZE (31)
#define MSD_SIZE (18)
#define MSD_ID (0xFF)
#define BUZZ_TIME (1.0) // Buzz time in s
/* Static constants for the accelerometer */
#define WHO_AM_I 0x0D // Type 'read' : This should return the device id of 0x2A
#define OUT_Z_MSB 0x05 // Type 'read' : z axis - 8 most significatn bit of a 12 bit sample
#define I2C_DATA (p29)
#define I2C_CLK (p2)
#define INT2_PIN (p4)
//uint8_t SLEEP = true;
int8_t txPower = 4;
uint8_t MSD[MSD_SIZE] = {0x59, 0x00, 0xE1, 0x61, 0x35, 0xBA, 0xC0, 0xEC, 0x47, 0x2A, 0x98, 0x00, 0xAF, 0x18, 0x43, 0xFF, 0x05, 0x00};
uint8_t my_mac_address[6] = {};
uint8_t buzzer_flag = 0;
void turnBuzzOff(void);
void goToSleep();
Ticker WakeSleepT;
Ticker turnBuzzOffT;
PwmOut buzzer(BUZZER);
PwmOut gyro_power(p7);
PwmOut i2c_power(p5); // I2C Pull-ups power pin
InterruptIn gyro_pulse(INT2_PIN);
Acc_MMA8452 acc(I2C_DATA, I2C_CLK, MMA8452_ADDRESS);
BLE &ble = BLE::Instance();
/* Restart Advertising on disconnection*/
void disconnectionCallback(const Gap::DisconnectionCallbackParams_t *params){
BLE::Instance().gap().startAdvertising();
}
/**
* The function is called when ticker generates interrupt
*/
void turnBuzzOff(void){
buzzer.write(0.0F);
turnBuzzOffT.detach();
WakeSleepT.attach(goToSleep, AWAKE_TIME);
}
/**
* This function is called when the ble initialization process has failed
*/
void onBleInitError(BLE &ble, ble_error_t error){
/* Avoid compiler warnings */
(void) ble;
(void) error;
/* Initialization error handling should go here */
}
/**
* Callback triggered when the ble initialization process has finished
*/
void bleInitComplete(BLE::InitializationCompleteCallbackContext *params){
BLE& ble = params->ble;
ble_error_t error = params->error;
if (error != BLE_ERROR_NONE) {
/* In case of error, forward the error handling to onBleInitError */
onBleInitError(ble, error);
return;
}
/* Ensure that it is the default instance of BLE */
if(ble.getInstanceID() != BLE::DEFAULT_INSTANCE) {
return;
}
ble.gap().onDisconnection(disconnectionCallback);
/* Get my MAC address */
BLEProtocol::AddressType_t temp_address_type;
ble.gap().getAddress(&temp_address_type, my_mac_address);
/* setup advertising */
ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::BREDR_NOT_SUPPORTED);
ble.gap().accumulateAdvertisingPayload(GapAdvertisingData::MANUFACTURER_SPECIFIC_DATA, (uint8_t *)MSD, MSD_SIZE);
ble.gap().setAdvertisingType(GapAdvertisingParams::ADV_NON_CONNECTABLE_UNDIRECTED);
ble.gap().setAdvertisingInterval(100); // --> Has to be at least 100ms!
ble.gap().startAdvertising();
}
uint8_t findMSDIndex(const Gap::AdvertisementCallbackParams_t *params){
uint8_t i=0;
uint8_t len;
do{
len = params->advertisingData[i];
i++;
if(params->advertisingData[i] == MSD_ID) return i;
else i += (len-1);
}while(i<MAX_BLE_PACKET_SIZE);
return 0;
}
/**
* Function is called when BLE radio discovers any kind of advertisment
*/
void advertisementCallback(const Gap::AdvertisementCallbackParams_t *params){
uint8_t i=0;
uint8_t msdOffset;
msdOffset = findMSDIndex(params);
if(msdOffset == 0){
// There's no MSD in BLE advertisement data
return;
}
if ((params->advertisingData[msdOffset]) == MSD_ID){
// Follows Manufacturer Specific Data
if ((params->advertisingData[msdOffset+1]) == 0x59){
if ((params->advertisingData[msdOffset+2]) == 0x00){
for(i=0; i<6; i++){
if((params->advertisingData[msdOffset+i+3]) == my_mac_address[5-i]){
continue;
}
else{
return;
}
}
turnBuzzOffT.detach();
WakeSleepT.detach();
buzzer.write(0.5F);
turnBuzzOffT.attach(turnBuzzOff, BUZZ_TIME);
}
}
}
}
void WakeMeUp(){
WakeSleepT.detach();
WakeSleepT.attach(goToSleep, AWAKE_TIME);
ble.gap().startScan(advertisementCallback);
ble.gap().startAdvertising();
SLEEP = false;
}
void goToSleep(){
WakeSleepT.detach();
WakeSleepT.attach(WakeMeUp, SLEEP_TIME);
ble.gap().stopAdvertising();
ble.gap().stopScan();
SLEEP = true;
}
void buzz(void){
buzzer.write(0.5f);
wait_ms(100);
buzzer.write(0.0f);
buzzer_flag = 0;
}
void pulse_handler(void){
#if DEBUG
int_led = !int_led;
#endif
i2c_power.write(1.0F);
buzzer_flag = 1;
}
int main(void){
WakeSleepT.attach(goToSleep, AWAKE_TIME);
ble.init(bleInitComplete);
ble.gap().setTxPower(txPower);
GapAdvertisingData postavke = GapAdvertisingData();
ble.gap().setScanParams(100, 100);
ble.gap().startScan(advertisementCallback);
buzzer.period(0.001F);
buzzer.write(0.0F);
gyro_power.period(0.01F);
gyro_power.write(1.0F);
i2c_power.period(0.01F);
i2c_power.write(1.0F);
wait_ms(1000);
/* Pulse interrupt detection */
acc.set_register((char)CTRL_REG_4, (char) 0x04); // INT_EN_FF_MT Freefall/motion interrupt enabled
wait_ms(1);
acc.set_register((char)FF_MT_CFG, (char) 0b01011000); //ELE, Motion Flag ON, YEFE, X Event Flag Enable
wait_ms(1);
acc.set_register((char)CTRL_REG_5, (char) 0x00); // INT_EN_FF_MT interrupt is router t0 INT2
wait_ms(1);
acc.set_register((char)FF_COUNT, (char) 0x08); // Set Counter degister value (10ms)
wait_ms(1);
acc.set_register((char)FF_MT_THS, (char) 0x90); // Set TH value for motion detection on 1 G (1/0.063) and DBCNTM = 1 (Increments or clears counter)
wait_ms(1);
/* Setup for the interrupt handler */
gyro_pulse.rise(&pulse_handler);
acc.set_register((char)CTRL_REG_1, (char) 0x01); // Flow data rate and Active mode
wait(1);
__enable_irq();
/* SpinWait for initialization to complete. This is necessary because the
* BLE object is used in the main loop below. */
while (ble.hasInitialized() == false){
/* spin loop */
}
while(true){
if(SLEEP) __WFI();
else if(buzzer_flag) buzz();
else ble.waitForEvent();
}
}

