nuttx/drivers/sensors/fakesensor.c

418 lines
12 KiB
C
Raw Normal View History

/****************************************************************************
* drivers/sensors/fakesensor.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <fcntl.h>
#include <math.h>
#include <stdio.h>
#include <string.h>
#include <nuttx/fs/fs.h>
#include <nuttx/kmalloc.h>
#include <nuttx/kthread.h>
#include <nuttx/nuttx.h>
#include <nuttx/semaphore.h>
#include <nuttx/sensors/fakesensor.h>
#include <nuttx/sensors/sensor.h>
#include <nuttx/signal.h>
#include <debug.h>
/****************************************************************************
* Private Types
****************************************************************************/
struct fakesensor_s
{
struct sensor_lowerhalf_s lower;
struct file data;
unsigned int interval;
unsigned int batch;
int raw_start;
FAR const char *file_path;
sem_t wakeup;
volatile bool running;
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static int fakesensor_activate(FAR struct sensor_lowerhalf_s *lower,
bool sw);
static int fakesensor_set_interval(FAR struct sensor_lowerhalf_s *lower,
FAR unsigned int *period_us);
static int fakesensor_batch(FAR struct sensor_lowerhalf_s *lower,
FAR unsigned int *latency_us);
static void fakesensor_push_event(FAR struct sensor_lowerhalf_s *lower);
static int fakesensor_thread(int argc, char** argv);
/****************************************************************************
* Private Data
****************************************************************************/
static struct sensor_ops_s g_fakesensor_ops =
{
.activate = fakesensor_activate,
.set_interval = fakesensor_set_interval,
.batch = fakesensor_batch,
};
/****************************************************************************
* Private Functions
****************************************************************************/
static int fakesensor_read_csv_line(FAR struct file *file,
char *buffer, int len, int start)
{
int i;
len = file_read(file, buffer, len);
if (len == 0)
{
/* Loop reads */
file_seek(file, start, SEEK_SET);
len = file_read(file, buffer, len);
}
for (i = 0; i < len; i++)
{
if (buffer[i] == '\n')
{
file_seek(file, i - len + 1, SEEK_CUR);
buffer[i + 1] = '\0';
break;
}
}
return i + 1;
}
static int fakesensor_read_csv_header(FAR struct fakesensor_s *sensor)
{
char buffer[40];
/* Set interval */
sensor->raw_start =
fakesensor_read_csv_line(&sensor->data, buffer, sizeof(buffer), 0);
if (sensor->interval == 0)
{
sscanf(buffer, "interval:%d\n", &sensor->interval);
sensor->interval *= 1000;
}
/* Skip the CSV header */
sensor->raw_start +=
fakesensor_read_csv_line(&sensor->data, buffer, sizeof(buffer), 0);
return OK;
}
static inline void fakesensor_read_accel(FAR struct fakesensor_s *sensor)
{
struct sensor_event_accel accel;
char raw[50];
fakesensor_read_csv_line(
&sensor->data, raw, sizeof(raw), sensor->raw_start);
sscanf(raw, "%f,%f,%f\n", &accel.x, &accel.y, &accel.z);
accel.temperature = NAN;
accel.timestamp = sensor_get_timestamp();
sensor->lower.push_event(sensor->lower.priv, &accel,
sizeof(struct sensor_event_accel));
}
static inline void fakesensor_read_mag(FAR struct fakesensor_s *sensor)
{
struct sensor_event_mag mag;
char raw[50];
fakesensor_read_csv_line(
&sensor->data, raw, sizeof(raw), sensor->raw_start);
sscanf(raw, "%f,%f,%f\n", &mag.x, &mag.y, &mag.z);
mag.temperature = NAN;
mag.timestamp = sensor_get_timestamp();
sensor->lower.push_event(sensor->lower.priv, &mag,
sizeof(struct sensor_event_mag));
}
static inline void fakesensor_read_gyro(FAR struct fakesensor_s *sensor)
{
struct sensor_event_gyro gyro;
char raw[50];
fakesensor_read_csv_line(
&sensor->data, raw, sizeof(raw), sensor->raw_start);
sscanf(raw, "%f,%f,%f\n", &gyro.x, &gyro.y, &gyro.z);
gyro.temperature = NAN;
gyro.timestamp = sensor_get_timestamp();
sensor->lower.push_event(sensor->lower.priv, &gyro,
sizeof(struct sensor_event_gyro));
}
static inline void fakesensor_read_gps(FAR struct fakesensor_s *sensor)
{
struct sensor_event_gps gps;
float time;
char latitude;
char longitude;
int status;
int sate_num;
float hoop;
float altitude;
char raw[150];
memset(&gps, 0, sizeof(struct sensor_event_gps));
read:
fakesensor_read_csv_line(
&sensor->data, raw, sizeof(raw), sensor->raw_start);
FAR char *pos = strstr(raw, "GGA");
if (pos == NULL)
{
goto read;
}
pos += 4;
sscanf(pos, "%f,%f,%c,%f,%c,%d,%d,%f,%f,", &time, &gps.latitude, &latitude,
&gps.longitude, &longitude, &status, &sate_num, &hoop, &altitude);
if (latitude == 'S')
{
gps.latitude = -gps.latitude;
}
if (longitude == 'W')
{
gps.longitude = -gps.longitude;
}
gps.height = altitude;
sensor->lower.push_event(sensor->lower.priv, &gps,
sizeof(struct sensor_event_gps));
}
static int fakesensor_activate(FAR struct sensor_lowerhalf_s *lower, bool sw)
{
FAR struct fakesensor_s *sensor = container_of(lower,
struct fakesensor_s, lower);
if (sw)
{
sensor->running = true;
/* Wake up the thread */
nxsem_post(&sensor->wakeup);
}
else
{
sensor->running = false;
}
return OK;
}
static int fakesensor_set_interval(FAR struct sensor_lowerhalf_s *lower,
FAR unsigned int *period_us)
{
FAR struct fakesensor_s *sensor = container_of(lower,
struct fakesensor_s, lower);
sensor->interval = *period_us;
return OK;
}
static int fakesensor_batch(FAR struct sensor_lowerhalf_s *lower,
FAR unsigned int *latency_us)
{
FAR struct fakesensor_s *sensor = container_of(lower,
struct fakesensor_s, lower);
uint32_t max_latency = sensor->lower.buffer_number * sensor->interval;
if (*latency_us > max_latency)
{
*latency_us = max_latency;
}
else if (*latency_us < sensor->interval && *latency_us > 0)
{
*latency_us = sensor->interval;
}
sensor->batch = *latency_us;
return OK;
}
static void fakesensor_push_event(FAR struct sensor_lowerhalf_s *lower)
{
FAR struct fakesensor_s *sensor = container_of(lower,
struct fakesensor_s, lower);
switch (lower->type)
{
case SENSOR_TYPE_ACCELEROMETER:
fakesensor_read_accel(sensor);
break;
case SENSOR_TYPE_MAGNETIC_FIELD:
fakesensor_read_mag(sensor);
break;
case SENSOR_TYPE_GYROSCOPE:
fakesensor_read_gyro(sensor);
break;
case SENSOR_TYPE_GPS:
fakesensor_read_gps(sensor);
break;
default:
snerr("fakesensor: unsupported type sensor type\n");
break;
}
}
static int fakesensor_thread(int argc, char** argv)
{
FAR struct fakesensor_s *sensor = (FAR struct fakesensor_s *)
((uintptr_t)strtoul(argv[1], NULL, 0));
int ret;
while (true)
{
/* Waiting to be woken up */
nxsem_wait_uninterruptible(&sensor->wakeup);
/* Open csv file and init file handle */
ret = file_open(&sensor->data, sensor->file_path, O_RDONLY);
if (ret < 0)
{
snerr("Failed to open file:%s, err:%d", sensor->file_path, ret);
return ret;
}
fakesensor_read_csv_header(sensor);
while (sensor->running)
{
/* Sleeping thread for interval */
nxsig_usleep(sensor->batch ? sensor->batch : sensor->interval);
/* Notify upper */
if (sensor->batch)
{
uint32_t batch_num = sensor->batch / sensor->interval;
for (int i = 0; i < batch_num; i++)
{
fakesensor_push_event(&sensor->lower);
}
}
else
{
fakesensor_push_event(&sensor->lower);
}
}
/* Close csv file handle when running change true to false */
ret = file_close(&sensor->data);
if (ret < 0)
{
snerr("Failed to close file:%s, err:%d", sensor->file_path, ret);
return ret;
}
}
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: fakesensor_init
*
* Description:
* This function generates a sensor node under /dev/sensor/. And then
* report the data from csv file.
*
* Input Parameters:
* type - The type of sensor and defined in <nuttx/sensors/sensor.h>
* file_name - The name of csv name and the file structure is as follows:
* First row : set interval, unit millisecond
* Second row: csv file header
* third row : data
* (Each line should not exceed 50 characters)
* For example:
* interval:12
* x,y,z
* 2.1234,3.23443,2.23456
* ...
* devno - The user specifies which device of this type, from 0.
* batch_number- The maximum number of batch
****************************************************************************/
int fakesensor_init(int type, FAR const char *file_name,
int devno, uint32_t batch_number)
{
FAR struct fakesensor_s *sensor;
FAR char *argv[2];
char arg1[32];
int ret;
/* Alloc memory for sensor */
sensor = kmm_zalloc(sizeof(struct fakesensor_s));
if (!sensor)
{
snerr("Memory cannot be allocated for fakesensor\n");
return -ENOMEM;
}
sensor->lower.type = type;
sensor->lower.ops = &g_fakesensor_ops;
sensor->lower.buffer_number = batch_number;
sensor->file_path = file_name;
nxsem_init(&sensor->wakeup, 0, 0);
nxsem_set_protocol(&sensor->wakeup, SEM_PRIO_NONE);
/* Create thread for sensor */
snprintf(arg1, 32, "%p", sensor);
argv[0] = arg1;
argv[1] = NULL;
ret = kthread_create("fakesensor_thread", SCHED_PRIORITY_DEFAULT,
CONFIG_DEFAULT_TASK_STACKSIZE, fakesensor_thread, argv);
if (ret < 0)
{
kmm_free(sensor);
return ERROR;
}
/* Register sensor */
sensor_register(&sensor->lower, devno);
return OK;
}