/**************************************************************************** * drivers/sensors/fakesensor_uorb.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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include /**************************************************************************** * Private Types ****************************************************************************/ struct fakesensor_s { union { struct sensor_lowerhalf_s lower; struct gps_lowerhalf_s gps; }; int type; struct file data; unsigned long interval; unsigned long 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, FAR struct file *filep, bool enable); static int fakesensor_set_interval(FAR struct sensor_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *period_us); static int fakesensor_batch(FAR struct sensor_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *latency_us); static int fakegps_activate(FAR struct gps_lowerhalf_s *lower, FAR struct file *filep, bool sw); static int fakegps_set_interval(FAR struct gps_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *period_us); static void fakesensor_push_event(FAR struct fakesensor_s *sensor, uint64_t event_timestamp); 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, }; static struct gps_ops_s g_fakegps_ops = { .activate = fakegps_activate, .set_interval = fakegps_set_interval, }; /**************************************************************************** * Private Functions ****************************************************************************/ static int fakesensor_read_csv_line(FAR struct file *file, FAR 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:%lu\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, uint64_t event_timestamp) { struct sensor_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 = event_timestamp; sensor->lower.push_event(sensor->lower.priv, &accel, sizeof(struct sensor_accel)); } static inline void fakesensor_read_mag(FAR struct fakesensor_s *sensor, uint64_t event_timestamp) { struct sensor_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 = event_timestamp; sensor->lower.push_event(sensor->lower.priv, &mag, sizeof(struct sensor_mag)); } static inline void fakesensor_read_gyro(FAR struct fakesensor_s *sensor, uint64_t event_timestamp) { struct sensor_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 = event_timestamp; sensor->lower.push_event(sensor->lower.priv, &gyro, sizeof(struct sensor_gyro)); } static inline void fakesensor_read_gps(FAR struct fakesensor_s *sensor) { char raw[150]; while (1) { fakesensor_read_csv_line(&sensor->data, raw, sizeof(raw), sensor->raw_start); sensor->gps.push_data(sensor->gps.priv, raw, strlen(raw), true); if (strstr(raw, "GGA") != NULL) { break; } } } static int fakesensor_activate(FAR struct sensor_lowerhalf_s *lower, FAR struct file *filep, bool enable) { FAR struct fakesensor_s *sensor = container_of(lower, struct fakesensor_s, lower); if (enable) { sensor->running = true; /* Wake up the thread */ nxsem_post(&sensor->wakeup); } else { sensor->running = false; } return OK; } static int fakegps_activate(FAR struct gps_lowerhalf_s *lower, FAR struct file *filep, bool enable) { return fakesensor_activate((FAR void *)lower, filep, enable); } static int fakesensor_set_interval(FAR struct sensor_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *period_us) { FAR struct fakesensor_s *sensor = container_of(lower, struct fakesensor_s, lower); sensor->interval = *period_us; return OK; } static int fakegps_set_interval(FAR struct gps_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *period_us) { return fakesensor_set_interval((FAR void *)lower, filep, period_us); } static int fakesensor_batch(FAR struct sensor_lowerhalf_s *lower, FAR struct file *filep, FAR unsigned long *latency_us) { FAR struct fakesensor_s *sensor = container_of(lower, struct fakesensor_s, lower); unsigned long max_latency = sensor->lower.nbuffer * 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; } void fakesensor_push_event(FAR struct fakesensor_s *sensor, uint64_t event_timestamp) { switch (sensor->type) { case SENSOR_TYPE_ACCELEROMETER: fakesensor_read_accel(sensor, event_timestamp); break; case SENSOR_TYPE_MAGNETIC_FIELD: fakesensor_read_mag(sensor, event_timestamp); break; case SENSOR_TYPE_GYROSCOPE: fakesensor_read_gyro(sensor, event_timestamp); break; case SENSOR_TYPE_GPS: case SENSOR_TYPE_GPS_SATELLITE: 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, 16)); 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 | O_CLOEXEC); 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; uint64_t event_timestamp = sensor_get_timestamp() - sensor->interval * batch_num; int i; for (i = 0; i < batch_num; i++) { fakesensor_push_event(sensor, event_timestamp); event_timestamp += sensor->interval; } } else { fakesensor_push_event(sensor, sensor_get_timestamp()); } } /* 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/uorb/. And then * report the data from csv file. * * Input Parameters: * type - The type of sensor and defined in * 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->file_path = file_name; sensor->type = type; nxsem_init(&sensor->wakeup, 0, 0); /* 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 */ if (type == SENSOR_TYPE_GPS || type == SENSOR_TYPE_GPS_SATELLITE) { sensor->gps.ops = &g_fakegps_ops; gps_register(&sensor->gps, devno, batch_number); } else { sensor->lower.type = type; sensor->lower.ops = &g_fakesensor_ops; sensor->lower.nbuffer = batch_number; sensor_register(&sensor->lower, devno); } return OK; }