The PSLab Android App allows users to log data from sensors connected to the PSLab hardware device. The Connected sensors should support I2C, SPI communication protocols to communicate with the PSLab device successfully. The only prerequisite is the additional support for the particular sensor plugin in Android App. The user can log data from various sensors and measure parameters like temperature, humidity, acceleration, magnetic field, etc. These parameters are useful in predicting and monitoring the environment and in performing many experiments.
The support for the sensor plugins was added during the porting python communication library code to Java. In this post, we will discuss how we logged real time sensor data from the PSLab Hardware Device. We used Realm database to store the sensor data locally. We have taken the MPU6050 sensor as an example to understand the complete process of logging sensor data.
Creating Realm Object for MPU6050 Sensor Data
The MPU6050 sensor gives the acceleration and gyroscope readings along the three axes X, Y and Z. So the data object storing the readings of the mpu sensor have variables to store the acceleration and gyroscope readings along all three axes.
public class DataMPU6050 extends RealmObject { private double ax, ay, az; private double gx, gy, gz; private double temperature; public DataMPU6050() { } public DataMPU6050(double ax, double ay, double az, double gx, double gy, double gz, double temperature) { this.ax = ax; this.ay = ay; this.az = az; this.gx = gx; this.gy = gy; this.gz = gz; this.temperature = temperature; } // getter and setter for all variables }
Creating Runnable to Start/Stop Data Logging
To sample the sensor data at 500ms interval, we created a runnable object and passed it to another thread which would prevent lagging of the UI thread. We can start/stop logging by changing the value of the boolean loggingThreadRunning on button click. TaskMPU6050 is an AsyncTask which reads each sample of sensor data from the PSLab device, it gets executed inside a while loop which is controlled by boolean loggingThreadRunning. Thread.sleep(500) pauses the thread for 500ms, this is also one of the reason to transfer the logging to another thread instead of logging the sensor data in UI thread. If such 500ms delays are incorporated in UI thread, app experience won’t be smooth for the users.
Runnable loggingRunnable = new Runnable() { @Override public void run() { try { MPU6050 sensorMPU6050 = new MPU6050(i2c); while (loggingThreadRunning) { TaskMPU6050 taskMPU6050 = new TaskMPU6050(sensorMPU6050); taskMPU6050.execute(); // use lock object to synchronize threads Thread.sleep(500); } } catch (IOException InterruptedException e) { e.printStackTrace(); } } };
Sampling of Sensor Data
We created an AsyncTask to read each sample of the sensor data from the PSLab device in the background thread. The getRaw() method read raw values from the sensor and returned an ArrayList containing the acceleration and gyro values. After the values were read successfully, they were updated in the data card in the foreground which was visible to the user. This data card acts as a real-time screen for the user. All the samples read are appended to ArrayList mpu6050DataList, when the user clicks on button Save Data, the collected samples are saved to the local realm database.
private ArrayList<DataMPU6050> mpu6050DataList = new ArrayList<>(); private class TaskMPU6050 extends AsyncTask<Void, Void, Void> { private MPU6050 sensorMPU6050; private ArrayList<Double> dataMPU6050 = new ArrayList<>(); TaskMPU6050(MPU6050 mpu6050) { this.sensorMPU6050 = mpu6050; } @Override protected Void doInBackground(Void... params) { try { dataMPU6050 = sensorMPU6050.getRaw(); } catch (IOException e) { e.printStackTrace(); } return null; } @Override protected void onPostExecute(Void aVoid) { super.onPostExecute(aVoid); // update data card TextViews with data read. DataMPU6050 tempObject = new DataMPU6050(dataMPU6050.get(0), dataMPU6050.get(1), dataMPU6050.get(2), dataMPU6050.get(4), dataMPU6050.get(5), dataMPU6050.get(6), dataMPU6050.get(3)); mpu6050DataList.add(tempObject); synchronized (lock) { lock.notify(); } } }
There is an option for Start/Stop Logging, clicking on which will change the value of boolean loggingThreadRunning which stops starts/stops the logging thread.
When the Save Data button is clicked, all the samples of sensor data collected from the PSLab device till that point are saved to the local realm database.
realm.beginTransaction(); for (DataMPU6050 tempObject : mpu6050DataList) { realm.copyToRealm(tempObject); } realm.commitTransaction();
Data can also be written asynchronously to the local realm database. For other methods to write to a real database refer write section of Realm docs.
Resources
- Medium Article on How to Use Realm for Android
- Article on Android Background Processing on vogella.com
- Realm Documentation