Gregory Kielian / Mbed 2 deprecated PollutionSensor

Dependencies:   libmDot mbed-rtos mbed

Fork of Adafruit_I2C_9dof_mDot_TTN by Gregory Kielian

Committer:
gkielian
Date:
Mon Sep 12 15:45:13 2016 +0000
Revision:
11:a2a6b1214ae4
Parent:
10:798cac7d3dcb
appears to work, has analog read, digital output for triggering the led read, next step is to procure more dust sensors from a reliable source (digikey mouser) which we can look for consistency of reading and I can start creating custom pcb's

Who changed what in which revision?

UserRevisionLine numberNew contents of line
elwinong 8:ebec2e50d421 1 /* Adafruit I2C Temp Sensor MCP9808 -> Multitech mDot -> The Things Network (TTN)
elwinong 8:ebec2e50d421 2 *
elwinong 8:ebec2e50d421 3 * This program connects the Adafruit temp sensor board MCP9808 through I2C
elwinong 8:ebec2e50d421 4 * and sends the temperature output to the The Things Network.
elwinong 8:ebec2e50d421 5 *
elwinong 8:ebec2e50d421 6 * Based on:
elwinong 8:ebec2e50d421 7 * - mDot TTN Connect Example: https://developer.mbed.org/users/ropu/code/mDot_LoRa_example_TTN_connect/rev/609e7bb06486
elwinong 8:ebec2e50d421 8 * - Adafruit MCP9808 Library: https://github.com/adafruit/Adafruit_MCP9808_Library
elwinong 8:ebec2e50d421 9 * - mbed I2C Example: https://developer.mbed.org/handbook/I2C
elwinong 8:ebec2e50d421 10 * - TTN Backend: https://www.thethingsnetwork.org/wiki/Backend/Overview
elwinong 8:ebec2e50d421 11 *
elwinong 8:ebec2e50d421 12 * Requiremens:
elwinong 8:ebec2e50d421 13 * - Multitech UDK board and mDot: https://developer.mbed.org/platforms/MTS-mDot-F411/
elwinong 8:ebec2e50d421 14 * - Adafruit MCP9809: https://learn.adafruit.com/adafruit-mcp9808-precision-i2c-temperature-sensor-guide/overview
elwinong 8:ebec2e50d421 15 * - Serial to USB like this or similar: https://www.amazon.com/Sabrent-Serial-RS-232-Converter-CB-DB9P/dp/B00IDSM6BW/ref=sr_1_5?ie=UTF8&qid=1470167137&sr=8-5&keywords=serial+usb
elwinong 8:ebec2e50d421 16 * - Default serial port baudrate is 9600
elwinong 8:ebec2e50d421 17 *
elwinong 8:ebec2e50d421 18 * Wiring:
elwinong 8:ebec2e50d421 19 * MCP9808 : UDK
elwinong 8:ebec2e50d421 20 * - Vdd : - D8 3.3V
elwinong 8:ebec2e50d421 21 * - Gnd : - D3 GND
elwinong 8:ebec2e50d421 22 * - SCL : - D15
elwinong 8:ebec2e50d421 23 * - SDA : - D14
elwinong 8:ebec2e50d421 24 */
elwinong 8:ebec2e50d421 25
mfiore 0:09250cd371d2 26 #include "mbed.h"
mfiore 0:09250cd371d2 27 #include "mDot.h"
mfiore 4:36e214ebfa56 28 #include "MTSLog.h"
mfiore 0:09250cd371d2 29 #include <string>
mfiore 0:09250cd371d2 30 #include <vector>
mfiore 4:36e214ebfa56 31 #include <algorithm>
elwinong 8:ebec2e50d421 32
gkielian 11:a2a6b1214ae4 33 AnalogIn ain(PB_1);
gkielian 11:a2a6b1214ae4 34 DigitalOut dout(PA_0);
gkielian 11:a2a6b1214ae4 35
gkielian 11:a2a6b1214ae4 36
elwinong 8:ebec2e50d421 37 // TTN Keys, setup your application and get your own keys at https://www.thethingsnetwork.org/wiki/Backend/Overview
gkielian 9:e3756ad55310 38 static const uint8_t netowork_session_key_array[] = { 0x22, 0xDF, 0x20, 0x0C, 0x71, 0x5A, 0xCC, 0x35, 0xB8, 0x20, 0x5F, 0xAE, 0xC7, 0x00, 0x5A, 0x63 };//
gkielian 9:e3756ad55310 39
gkielian 9:e3756ad55310 40 //app session key
gkielian 9:e3756ad55310 41 static const uint8_t data_session_key_array[] = { 0x66, 0xF2, 0x8B, 0x88, 0x76, 0x0D, 0xB0, 0xE2, 0xAE, 0x16, 0x29, 0x81, 0x29, 0xC9, 0x25, 0x11 };
gkielian 9:e3756ad55310 42
gkielian 9:e3756ad55310 43 //dev address (keep in mind these are all under abp)
gkielian 9:e3756ad55310 44 static const uint8_t network_address_array[] = { 0x30, 0xE2, 0x6C, 0xAF };
gkielian 9:e3756ad55310 45
gkielian 9:e3756ad55310 46 //static const uint8_t netowork_session_key_array[] = { 0x35, 0x41, 0x32, 0x67, 0x79, 0x76, 0x49, 0x8B, 0xBE, 0x98, 0x10, 0x80, 0x76, 0xB7, 0x61, 0x8B };
gkielian 9:e3756ad55310 47 //static const uint8_t data_session_key_array[] = { 0xD3, 0x6F, 0x5E, 0x66, 0x4A, 0x1B, 0xEC, 0x0C, 0x4A, 0x63, 0x8E, 0x1C, 0x2D, 0xB3, 0x18, 0xA4 };
gkielian 9:e3756ad55310 48 //static const uint8_t network_address_array[] = { 0x10, 0x81, 0xDA, 0x6F };
elwinong 8:ebec2e50d421 49
ropu 6:8f7276e7d206 50 static std::vector<uint8_t> netowork_session_key (netowork_session_key_array, netowork_session_key_array + sizeof(netowork_session_key_array) / sizeof(uint8_t));
ropu 6:8f7276e7d206 51 static std::vector<uint8_t> data_session_key (data_session_key_array, data_session_key_array + sizeof(data_session_key_array) / sizeof(uint8_t));
ropu 6:8f7276e7d206 52 static std::vector<uint8_t> network_address (network_address_array, network_address_array + sizeof(network_address_array) / sizeof(uint8_t));
elwinong 8:ebec2e50d421 53
elwinong 8:ebec2e50d421 54 static uint8_t config_frequency_sub_band = 2;
elwinong 8:ebec2e50d421 55
elwinong 8:ebec2e50d421 56 // Initialize I2C
elwinong 8:ebec2e50d421 57 I2C i2c(I2C_SDA , I2C_SCL );
gkielian 9:e3756ad55310 58 const int addr7bit = 0x1E; // 7 bit I2C address
elwinong 8:ebec2e50d421 59 const int addr8bit = addr7bit << 1; // 8bit I2C address, 0x90
elwinong 8:ebec2e50d421 60
mfiore 0:09250cd371d2 61
mfiore 0:09250cd371d2 62 int main() {
gkielian 11:a2a6b1214ae4 63 dout = 0;
gkielian 11:a2a6b1214ae4 64
gkielian 11:a2a6b1214ae4 65 // Declare and initialize variables
gkielian 11:a2a6b1214ae4 66 int16_t system_voltage = 3300;
gkielian 11:a2a6b1214ae4 67 float conversion_ratio = 0.2;
gkielian 11:a2a6b1214ae4 68 float zero_dust_voltage = 0.4;
gkielian 11:a2a6b1214ae4 69 float voltage = 0;
gkielian 11:a2a6b1214ae4 70 float concentration = 0;
elwinong 8:ebec2e50d421 71
mfiore 0:09250cd371d2 72 int32_t ret;
mfiore 0:09250cd371d2 73 mDot* dot;
elwinong 8:ebec2e50d421 74 std::vector<uint8_t> data; // mDot data->send variable
elwinong 8:ebec2e50d421 75 char data_str[64]; // Intermediate conversion variable
gkielian 11:a2a6b1214ae4 76 uint32_t update_interval = 4000; // TTN transmission interval (loop interval)
gkielian 11:a2a6b1214ae4 77 float analog_input;
gkielian 10:798cac7d3dcb 78 //initialize pins for analog input
mfiore 2:6e2c378339d9 79
mfiore 0:09250cd371d2 80 // get a mDot handle
mfiore 0:09250cd371d2 81 dot = mDot::getInstance();
ropu 6:8f7276e7d206 82
ropu 6:8f7276e7d206 83 dot->resetConfig();
mfiore 0:09250cd371d2 84
ropu 7:609e7bb06486 85 dot->setLogLevel(mts::MTSLog::INFO_LEVEL);
ropu 7:609e7bb06486 86
ropu 7:609e7bb06486 87 // too lazzy to check all errors
ropu 6:8f7276e7d206 88 dot->setJoinMode(mDot::MANUAL);
ropu 6:8f7276e7d206 89 dot->setPublicNetwork(true);
ropu 6:8f7276e7d206 90 dot->setFrequencySubBand(config_frequency_sub_band);
ropu 6:8f7276e7d206 91 dot->setNetworkSessionKey(netowork_session_key);
ropu 6:8f7276e7d206 92 dot->setDataSessionKey(data_session_key);
ropu 6:8f7276e7d206 93 dot->setNetworkAddress(network_address);
mfiore 4:36e214ebfa56 94
mfiore 4:36e214ebfa56 95
mfiore 4:36e214ebfa56 96 // a higher spreading factor allows for longer range but lower throughput
mfiore 4:36e214ebfa56 97 // in the 915 (US) frequency band, spreading factors 7 - 10 are available
mfiore 4:36e214ebfa56 98 // in the 868 (EU) frequency band, spreading factors 7 - 12 are available
mfiore 4:36e214ebfa56 99 logInfo("setting TX spreading factor");
mfiore 4:36e214ebfa56 100 if ((ret = dot->setTxDataRate(mDot::SF_10)) != mDot::MDOT_OK) {
mfiore 4:36e214ebfa56 101 logError("failed to set TX datarate %d:%s", ret, mDot::getReturnCodeString(ret).c_str());
mfiore 4:36e214ebfa56 102 }
mfiore 4:36e214ebfa56 103
ropu 6:8f7276e7d206 104
mfiore 4:36e214ebfa56 105 // request receive confirmation of packets from the gateway
mfiore 4:36e214ebfa56 106 logInfo("enabling ACKs");
ropu 6:8f7276e7d206 107 if ((ret = dot->setAck(0)) != mDot::MDOT_OK) {
mfiore 4:36e214ebfa56 108 logError("failed to enable ACKs %d:%s", ret, mDot::getReturnCodeString(ret).c_str());
mfiore 4:36e214ebfa56 109 }
mfiore 4:36e214ebfa56 110
mfiore 4:36e214ebfa56 111 // save this configuration to the mDot's NVM
mfiore 2:6e2c378339d9 112 logInfo("saving config");
mfiore 2:6e2c378339d9 113 if (! dot->saveConfig()) {
mfiore 2:6e2c378339d9 114 logError("failed to save configuration");
mfiore 0:09250cd371d2 115 }
mfiore 2:6e2c378339d9 116 //*******************************************
mfiore 2:6e2c378339d9 117 // end of configuration
mfiore 2:6e2c378339d9 118 //*******************************************
mfiore 0:09250cd371d2 119
mfiore 0:09250cd371d2 120 // attempt to join the network
mfiore 2:6e2c378339d9 121 logInfo("joining network");
mfiore 0:09250cd371d2 122 while ((ret = dot->joinNetwork()) != mDot::MDOT_OK) {
mfiore 2:6e2c378339d9 123 logError("failed to join network %d:%s", ret, mDot::getReturnCodeString(ret).c_str());
mfiore 4:36e214ebfa56 124 // in the 868 (EU) frequency band, we need to wait until another channel is available before transmitting again
mfiore 4:36e214ebfa56 125 osDelay(std::max((uint32_t)1000, (uint32_t)dot->getNextTxMs()));
mfiore 0:09250cd371d2 126 }
mfiore 0:09250cd371d2 127
elwinong 8:ebec2e50d421 128 while (true) {
gkielian 11:a2a6b1214ae4 129 dout = 1;
gkielian 11:a2a6b1214ae4 130 wait_us(280);
gkielian 11:a2a6b1214ae4 131 analog_input = ain;
gkielian 11:a2a6b1214ae4 132 dout = 0;
gkielian 11:a2a6b1214ae4 133 wait_us(720);
gkielian 11:a2a6b1214ae4 134
gkielian 11:a2a6b1214ae4 135 //convert to voltage in mv
gkielian 11:a2a6b1214ae4 136 voltage = (system_voltage / 1024.0 ) * analog_input;
gkielian 11:a2a6b1214ae4 137 logInfo("voltage = %f", voltage);
gkielian 11:a2a6b1214ae4 138
gkielian 11:a2a6b1214ae4 139
gkielian 11:a2a6b1214ae4 140 //voltage to concentration
gkielian 11:a2a6b1214ae4 141 concentration = voltage * conversion_ratio;
gkielian 11:a2a6b1214ae4 142
gkielian 11:a2a6b1214ae4 143
gkielian 11:a2a6b1214ae4 144
gkielian 11:a2a6b1214ae4 145
gkielian 11:a2a6b1214ae4 146
gkielian 11:a2a6b1214ae4 147 logInfo("concentration [ug/m^3] = %f", concentration);
elwinong 8:ebec2e50d421 148
elwinong 8:ebec2e50d421 149
elwinong 8:ebec2e50d421 150 // Empty data vector
elwinong 8:ebec2e50d421 151 data.clear();
elwinong 8:ebec2e50d421 152
elwinong 8:ebec2e50d421 153 // Push temperature value into data array
gkielian 11:a2a6b1214ae4 154 sprintf(data_str, "%f", analog_input);
elwinong 8:ebec2e50d421 155 for (int i = 0; i<strlen(data_str); i++)
elwinong 8:ebec2e50d421 156 {
elwinong 8:ebec2e50d421 157 data.push_back(((char*)data_str)[i]);
elwinong 8:ebec2e50d421 158 }
elwinong 8:ebec2e50d421 159
mfiore 4:36e214ebfa56 160 // send the data to the gateway
mfiore 0:09250cd371d2 161 if ((ret = dot->send(data)) != mDot::MDOT_OK) {
mfiore 2:6e2c378339d9 162 logError("failed to send", ret, mDot::getReturnCodeString(ret).c_str());
mfiore 0:09250cd371d2 163 } else {
elwinong 8:ebec2e50d421 164 logInfo("successfully sent data to gateway: %s", data_str);
mfiore 0:09250cd371d2 165 }
mfiore 0:09250cd371d2 166
mfiore 4:36e214ebfa56 167 // in the 868 (EU) frequency band, we need to wait until another channel is available before transmitting again
elwinong 8:ebec2e50d421 168 osDelay(std::max(update_interval, (uint32_t)dot->getNextTxMs()));
elwinong 8:ebec2e50d421 169
mfiore 0:09250cd371d2 170 }
mfiore 0:09250cd371d2 171
mfiore 0:09250cd371d2 172 return 0;
mfiore 0:09250cd371d2 173 }