Introduction
Car Tracker utilizes the LinkIt ONE development board’s GNSS features to acquire the position and the speed of a vehicle. The data is then uploaded to the MediaTek Cloud Sandbox (MCS) over a GPRS connection. MCS then provides a visualization of the data on a map. For more advanced metrics an accelerometer is also attached to the board, to collect data on the g-forces generated by the vehicle.
By the end of this tutorial you’ll have a fully functioning IoT prototype, which gathers GPS and accelerometer data, then transmits the data through a GPRS connection to enable control and visualization using a cloud service through a web interface or Android app.
This tutorial guides you through:
- Building the prototype, with details of the hardware requirements and how to put them together to create the Car Tracker.
- Adding your Car Tracker to the MCS, creating the data channels to collect the sensor data and obtaining the information to use in the software to send data to MCS.
- Creating the software to provide the interfaces with the sensor to collect the data and implement data communications to send that data to the MCS.
- Using the MCS to view that data collected on the Car Tracker, including using the MCS app on an Android smartphone or tablet (running Android v. 4.0 or above) to view data while away from your PC.
At the end of the tutorial there are details on where to go for additional information on the LinkIt ONE development board and how to create software for it.
Before you start
If you haven’t built a LinkIt ONE project before, this section describes the steps you need to follow before commencing this project.
Install the development software
Full details on downloading and installing the Arduino IDE and LinkIt ONE SDK then configuring the IDE and upgrading the board firmware are provided in the LinkIt ONE quick start guide. Complete this before you continue if you haven’t already set up your development environment.
Activate your MediaTek Cloud Sandbox account
To make use of the MediaTek Cloud Sandbox register for a Labs account, it’s free, andactivate your MCS account to prototype your own devices and applications. By registering on Labs you also gain access to the hardware reference designs and the ability to submit and respond to items in the forums.
An Android Device
The application supports smartphones and tablets running Android v. 4.0 or above, with an active data connection (mobile network or Wi-Fi). In this particular scenario Android smartphone (Samsung Galaxy) is used.
Building the Car Tracker Hardware
This section describes the hardware and electronics needed to build Car Tracker and provides details on how to put them together.
What you need
To build the Car Tracker hardware, in addition to a LinkIt ONE development board, you need the following components:
Component | Description | Source |
---|---|---|
Sensor | ADXL345 digital accelerometer | Grove - 3-Axis Digital Accelerometer (±16g) |
Antennas | GSM/GPRS antenna covering 850/900/1800/1900 MHz frequency range and classified as Class 12. GPS + GLONASS antenna. |
LinkIt ONE Development Kit |
Cable | Micro-USB cable | |
Power | Polymer Li-ion 1000mAh battery | LinkIt ONE Development Kit |
Schematics
The components of the Car Tracker are assembled according to the following schematic diagram:
Electronic circuit diagram of the Car Tracker
This image was created with Fritzing.
Putting the components together
This section provides step-by-step instructions on putting the Car Tracker hardware together. At the end you’ll have a Car Tracker similar to the one shown below:
Hardware setup of the Car Tracker
Step 1 – Antennas
Attach the GPS+GLONASS, dual purpose Wi-Fi/Bluetooth and GSM antennas to their corresponding pins on the LinkIt ONE development board as follows:
Connecting the antennas
Step 2 – Accelerometer
Attach a digital accelerometer to the LinkIt ONE development board using the Grove - Universal 4 Pin buckled cable. The accelerometer is based on ADXL-345 and is connected to the Grove interface on the LinkIt ONE development board.
Step 3 – Power source
The power source in this tutorial is a polymer lithium-ion battery and is included in the LinkIt ONE development board kit. Attach the battery source to the Li-Battery (3.7~4.2V) socket, as shown in the figure below.
Breadboard view of the Car Tracker
This image was created with Fritzing.
Setup the MCS
In this project you’ll be using MCS to collect and visualize the data recorded by the Car Tracker hardware. The setup is described in two parts: first creating a new Car Tracker prototype and then create a new device — the cloud space that will interact with your physical device — based on the Car Tracker prototype.
MCS Development Cycle
Activate a MCS account to prototype your own devices and applications. The Car Tracker tutorial follows the general steps of application development on the MCS.
Create a new prototype for the Car Tracker
A prototype serves as a blueprint for the actual hardware setup. The prototype consists of one or more data channels of type display, controller and hybrid. The data channels are defined with a Data channel name
, Data channel id and data type as required parameters. The variety of data types and overall structure of the prototype in general is shown in the figure below.
General prototype content in MCS
The Car Tracker prototype in this tutorial has five data channels to display the vehicle location on a map based on GPS coordinates, speed of the car and the g-forces experienced by the car. This section describes how to create and configure a Car Tracker prototype with necessary data channels.
- Sign in to MCS with a pre-registered MediaTek Labs account to create and prototype your own devices and applications.
Signing into the MCS
- Click Development from the navigation toolbar, and then under Prototype list clickCreate to create a new prototype, as shown below.
Create a new prototype
- Provide information, to define a basic profile of this prototype.
Define a Car Tracker prototype
Save the information and proceed to the next step.
Note: Fields marked by a red asterisk (?) are required fields. |
- Click Detail to view the prototype information, as shown below.
View or modify the Car Tracker prototype
- Click Exit to bypass initiation instructions, as shown below:
Bypassing the data channel initiation instructions
Add Data Channels
The required data channels are created at this stage.
- Click Add under Data channel toolbar, as shown below to provide data channels for the Car Tracker prototype.
Add a data channel to the Car Tracker prototype
Controller and Display are data channel types available to use. Create a Display data channel for GPS map location, speed and g-force data visualization on MCS by clickingAdd. First, you’ll create the map location data channel.
Data channel types
- Enter the information for Data channel name, Data channel id, Description and Data type for position data channel and click Save, as shown in the figure below.
GPS location data channel details
The Data channel name and Data Channel id define the data channel and will be used in an Arduino sketch to communicate the data from the board to the MCS.
Repeat steps 1 and 2 to create data channels for speed and g-forces on the X, Y and Z axes. The Data type is float and the unitTypes is miles/hour for the speed of the car. The g-force on the Z axis has a Data channel name Acceleration/Breaking and Data channel idzForce
. The g-force on the X axis has a Data channel name Up/Down Force and Data channel id xForce
. The g-force on the Y axis has a Data channel name Centrifugal Force and Data channel id yForce
. The unitTypes of g-forces is Others. More details on the information you need to provide when setting up a data channel, please refer to Key concepts found in MCS documentation.
You should now have a complete definition of the Car Tracker prototype, as shown below.
Car Tracker prototype definition
Create a New Device Based on the Car Tracker Prototype
Each prototype on MCS is defined on a specific hardware platform. The hardware platform assigned in this tutorial is the LinkIt ONE development board. The Car Tracker prototype is mapped to the actual device by taking the following steps.
- Click Development from the MCS toolbar, and then click Car Tracker Prototype detail.
- Click Create test device to configure a new device.
Creating a test device for the Car Tracker prototype
- Provide Device name, as an example device name shown in the figure below, click OKto continue.
Car Tracker test device configuration
- Click Go to detail, as shown in the figure below to view the device information.
Confirmation on creating a device for the Car Tracker prototype
- It’s essential to store the DeviceID and DeviceKey values in a file or hardcode them in an Arduino sketch (see Overview of the Arduino Sketch) to enable API calls and device connectivity.
Device details for the Car Tracker prototype
You have now created a new device in MCS matching the Car Tracker prototype with five display data channels for the position on the map, speed and g-forces. Next section describes the Car Tracker’s software implementation.
Create your Car Tracker’s software
Your Car Tracker needs an Arduino Sketch to share data from an onboard GPS receiver and an accelerometer sensor with MCS. This section describes the software required as follows:
- Overview of the Arduino sketch.
- Configuring device and GPRS settings.
- Transmitting sensor data to the MCS.
- Integrating with MediaTek Cloud Sandbox on a PC and a mobile device.
Overview of the Arduino Sketch
An Arduino sketch is implemented, verified and uploaded to the LinkIt ONE development board. The source code is provided in a GitHub repository. For the accelerometer operation download the ADXL-345 driver. The HTTP communication is supported by an HTTPClientdriver. Add the downloaded libraries to the Arduino libraries folder.
Start your Arduino project
Create a new Arduino Sketch - software that will run on the LinkIt One development board.
- Open Arduino IDE and by default a new sketch is created and showed.
- Create a folder named
CarTracker
in your documents folder and on the File menu clickSave As navigate to the selected folder, name the sketchCarTracker.ino
and save it.
Empty project in Arduino IDE
Add the supporting libraries
Add the libraries to the CarTracker.ino
Arduino sketch. The libraries that are part of LinkIt ONE API are listed below:
#include <LGPS.h> |
These provide support for GPS, GPRS, and data storage communication and their detailed description can be found in the LinkIt ONE API guide.
HTTP communication
The HTTP communication header file is from the HTTP library.
#include "HttpClient.h" |
Wire communication
This library allows you to communicate with an I2C device.
#include <Wire.h> |
Accelerator communication
This library allows you to communicate with a digital accelerometer.
#include "ADXL345.h" |
Custom libraries
These libraries are custom defined and the detailed content will be described in sections The GPSWaypoint class and The GPSFunctions class.
#include "GPSWaypoint.h" |
Add source code to the sketch CarTracker.ini
A sketch is a source code file representing the core controlling logic for the LinkIt ONE development board. It consists of the following:
- A
setup()
function that initializes resources, such as the GPRS module. - A
loop()
function that continuously listens to and processes events from hardware sensors and software modules such as those for GPRS. Theloop()
function runs forever — until the device is shutdown.
Configure device settings
Once the libraries are included, provide the device ID and device key information. The device ID and device key were generated when the test device for Car Tracker prototype was created; see Create a Test Device Based on the Car Tracker Prototype. The device ID and device key are used to identify the Car Tracker hardware and they are defined as constant values in a CarTracker.ini
where you can replace the quoted text with your device’s information.
#define DeviceID "DD1XLRmX" |
Initializing the communication
The setup()
function is called only once at the beginning of the program and provides for initialization of the GPS, GPRS and digital accelerometer communication.
- Initiate serial communication by calling the
begin()
function of theSerial class
. Provide the communication data rate in bits per second (baud) for serial data transmission. For the Car Tracker tutorial the baud rate is set to 115200. Then call thepowerOn()
function of the LGPS class to start the GPS communication and wait for three seconds (delay(3000)
) for the connection to be established, as in the code below:
void setup() |
- Set a timeout on the HTTP communication by calling the
setHttpResponseTimeout
function of the HTTP class, as shown in the code below:
http.setHttpResponseTimeout(15); |
- Configure the Access Point Name (APN) settings of your GPRS connectivity to connect to the remote network services over TCP by calling the
attachGPRS()
function of theLGPRS class in a while loop. TheattachGPRS()
function requires APN, username and password as input parameters provided by your network telecom operator. If the settings are not verified allow 500 ms wait time to continuously establish the GPRS connection using your telecom operator’s information. After successful initialization, a messageSuccess
is sent to the output of the serial port. Then the accelerometer is turned on by callingpowerOn()
function of theadxl
class.
Serial.print("Connecting to GPRS:"); |
The complete setup()
function is shown below:
void setup() |
Add a loop function
- The loop function is executed continuously providing formatted GPS coordinates such as latitude and longitude, speed and extra g-force data. The function call hierarchy is provided in the figure below:
loop()
function’s call hierarchy
- The function
getGPSData()
is where you get the position data from. It is defined in theGPSFunctions
class and the details are provided in The GPSFunctions class.
void loop() |
- Sending data to the MCS
Upload the position and speed data to the device specified by its deviceID
and deviceKey
. Functions uploadData()
and uploadAccelerations()
use the formatted data to upload them to the Car Tracker. The first uploadData()
function sends the formatted position data to the device. Then the second uploadData()
function sends the car speed value to the device converting the floating point number into a string.
char* buffer_latitude = new char[30]; |
Note: Make sure the DeviceIDs in the source code match to the ones defined in the MCS prototype, see Add Data Channels. |
uploadData()
— while the HTTP connection is still established upload the string data invalueForKey
to the device specified by thedeviceID
anddeviceKey
.
Syntax | void uploadData(const char* deviceID, const char* deviceKey,const char* dataKey, String valueForKey) |
Parameters | const char* deviceID, const char* deviceKey,const char* dataKey, String valueForKey |
Return value | void |
uploadAccelerations()
— uploads accelerations and g-forces on the X, Y and Z axes while the HTTP connection is still established.
Syntax | void uploadAccelerations() |
Parameters | void |
Return value | void |
Add classes in Arduino
The project is based on several classes that provide the position estimation functionality of the LinkIt ONE development board.
- Add source and header files.
- Click on the top right drop down menu then select New Tab, as shown below.
Adding new source and header files
- Name the new file as
GPS_functions.cpp
. - Repeat the same step and create
GPS_functions.h
,GPSWaypoint.cpp
andGPSWaypoint.h
source and header files.
The GPSWaypoint
class
The GPSWaypoint
class is used to store information and pass it between the different functions.
- Define a class
GPSWaypoint
in aGPSWaypoint.h
header file. The class contains public attributes for longitude, latitude and speed as floating numbers, north or south (char* n_or_s
) and east or west (char* e_or_w
) directions as character arrays and an indicator for the position fix (int has_fix
) as an integer. Constructor and destructor methods areGPSWaypoint()
and~GPSWaypoint()
respectively.
class GPSWaypoint |
- Provide a default constructor and destructor source code for the
GPSWaypoint
class in aGPSWaypoint.cpp
file as follows.
#include "GPSWaypoint.h" |
The GPSFunctions
class
The ultimate goal of a GPS receiver is to provide position, velocity, and time information. The National Marine Electronics Association (NMEA) has developed a specification that defines the interface between various pieces of marine electronic equipment. The standard permits marine electronics to send information to computers and to other marine equipment. NMEA format is common for real time position estimation including the complete PVT (position, velocity, time) solution computed by the GPS receiver. The idea of NMEA is to send a line of data called a sentence that is totally self-contained and independent from other sentences. All NMEA sentences are sequences of ASCII symbols beginning with a '$' and ending with a carriage return/line feed sequence and can be no longer than 80 characters of visible text (plus the line terminators). The GPSFunctions
class contains position and speed estimation functionality based on the NMEA specification. The focus of this class is to decode NMEA sentences including the GGA (Global Positioning System Fix Data), the RMC (Recommended Minimum sentence C) and the GSA (Satellite status) data.
parseGPGGA()
— This function decodes the received sentence according to the NMEAspecification. It starts with anif
statement that looks for a ‘$’ sign in a NMEA sentence. Once it’s found an integer variable tmp is declared and assigned to the location of the first comma in a string. This will define the time information. The position information - GPS latitude and longitude coordinates are found after the second comma in a NMEA sentence. Directions indicating north, south, east or west are found similarly. The data is then converted into an appropriate format by callingconvertCoords()
function.
void parseGPGGA(const char* GPGGAstr, GPSWaypoint* wayPoint){ |
Syntax | void parseGPGGA(const char* GPGGAstr, GPSWaypoint* wayPoint) |
Parameters | const char* GPGGAstr, GPSWaypoint* wayPoint |
Return value | void |
printGPGGA()
— format the latitude and longitude data and send to the serial output when there is a position fix.
Syntax | boolean printGPGGA(char* str, char* GPS_formatted, GPSWaypoint* wayPoint) |
Parameters | char* str, char* GPS_formatted, GPSWaypoint* wayPoint |
Return value | boolean |
nextToken()
— copy content of thebuf
character array intosrc
character array from a specified location.
Syntax | const char *nextToken(char* src, char* buf) |
Parameters | char* src, char* buf |
Return value | const char* |
getGPSData()
— acquire the current position of the vehicle. Because acquiring the position can take some time, the function will check if there is a fix on the position and if not, wait for half a second and try again. Once the position is determined, the data is read out and stored in aGPSWaypoint
object.
Syntax | void getGPSData(gpsSentenceInfoStruct &g_info, char* GPS_formatted, GPSWaypoint* positionData) |
Parameters | gpsSentenceInfoStruct &g_info, char* GPS_formatted, GPSWaypoint* positionData |
Return value | void |
void getGPSData(gpsSentenceInfoStruct &g_info, char* GPS_formatted, GPSWaypoint* positionData) |
LGPS.getData()
is the API call to acquire the current position. The position comes back in a gpsSentenceInfoStruct
structure. Within this structure you find different types ofNMEA sentences. A sentence is a comma delimited string with different types of information about the acquired position. For this tutorial you’ll use the GPGGA (Global Positioning System Fix Data) for the location and the GPVTG (Track Made Good and Ground Speed) for reading out the speed.
Example of GPGGA NMEA sentence is shown below:
$GPGGA,123519,4807.038,N,01131.000,E,1,08,0.9,545.4,M,46.9,M,,*47 |
convertCoords()
— convert latitude and longitude from degrees, minutes and seconds to floating point numbers. The positive or negative sign indicates the direction.
Syntax | void convertCoords(float latitude, float longitude, const char* n_or_s, const char* e_or_w, float &lat_return, float &lon_return) |
Parameters | float latitude, float longitude, const char* n_or_s, const char* e_or_w, float &lat_return, float &lon_return |
Return value | void |
void convertCoords(float latitude, float longitude, const char* n_or_s, const char* e_or_w, float &lat_return, float &lon_return){ |
arrayToInt()
— converts character array into an integer.
Syntax | int arrayToInt(const char* char_array) |
Parameters | const char* char_array |
Return value | int |
arrayToFloat()
— converts character array into an floating point number.
Syntax | float arrayToFloat (const char* char_array) |
Parameters | const char* char_array |
Return value | float |
getComma()
— identifies the location of a comma in a given string
Syntax | static unsigned char getComma(unsigned char num, const char *str); |
Parameters | unsigned char num, const char *str |
Return value | static unsigned char |
getIntNumber()
— given a character array identifies the comma and converts each comma separated string into an integer.
Syntax | static int getIntNumber(const char *s); |
Parameters | const char *s |
Return value | static int |
getFloatNumber()
— given a character array identifies the comma and converts each comma separated string into a floating point number.
Syntax | static float getFloatNumber(const char *s) |
Parameters | const char *s |
Return value | static float |
getCharString()
— given a character array identifies the comma and returns each comma separated character.
Syntax | static float getCharString (const char *s) |
Parameters | const char *s |
Return value | static float |
readSpeed()
— read the GPVTG speed information from the NMEA sentence.
Syntax | void readSpeed(char* str, GPSWaypoint* wayPoint) |
Parameters | char* str, GPSWaypoint* wayPoint |
Return value | void |
void readSpeed(char* str, GPSWaypoint* wayPoint){ |
The full list of function definitions in GPSFunction.h
header file is shown below.
static unsigned char getComma(unsigned char num, const char *str); |
Now you can compile and upload the sketch to the LinkIt ONE development board as described in the LinkIt ONE quick start guide, then follow the instructions in the next section to visualize the output on the MCS.
Remote Access to the MCS
In previous sections you’ve successfully created a prototype associated with the Car Tracker using MCS and programmed the software to define the connectivity and data channels in Arduino IDE. To visualize the output of the source code on the MCS console click My Devices on the MCS toolbar to view existing devices or add one.
View available devices on MCS
Accessing Car Tracker device details
Click Detail on Car Tracker device from the Device list menu. The location on the map and values of speed and accelerometer will be on the display as shown in Add Data Channels. Each of those controls has a shortcut menu. Open the shortcut menu for data channel to view the available options.
Card menu History and API Hint options
The History option provides all the past and present values of the data channel, in this case Centrifugal force.
Values of centrifugal force over time
Congratulations, you have now completed the Car Tracker prototype setup.
MCS provides mobile app to remotely control the existing prototypes connected to the LinkIt ONE development board. Download and install the app on an Android smartphone or tablet device to get started with the Car Tracker tutorial on a mobile device.
Conclusion
In this tutorial you have implemented a remote controlled Car Tracker application using LinkIt ONE development board, accelerometer, MediaTek Cloud Sandbox and Arduino programming environment.
For more information on MediaTek LinkIt ONE development and prototyping board and cloud services refer to LinkIt ONE developer’s guide and MediaTek Cloud Sandbox.