/****************************************************************************
 * apps/examples/sensor_fusion/sensor_fusion_main.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 <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <nuttx/sensors/sensor.h>
#include "Fusion/Fusion.h"

/****************************************************************************
 * Pre-processor Definitions
 ****************************************************************************/

#define REG_LOW_MASK 0xFF00
#define REG_HIGH_MASK 0x00FF
#define MPU6050_FS_SEL 32.8f
#define MPU6050_AFS_SEL 4096.0f

/****************************************************************************
 * Private Types
 ****************************************************************************/

struct mpu6050_imu_msg
{
  int16_t acc_x;
  int16_t acc_y;
  int16_t acc_z;
  int16_t temp;
  int16_t gyro_x;
  int16_t gyro_y;
  int16_t gyro_z;
};

/****************************************************************************
 * Public Functions
 ****************************************************************************/

void read_mpu6050(int fd, struct sensor_accel *acc_data,
                  struct sensor_gyro *gyro_data);

/****************************************************************************
 * sensor_fusion_main
 ****************************************************************************/

int main(int argc, char *argv[])
{
  int fd;
  int iterations = CONFIG_EXAMPLES_SENSOR_FUSION_SAMPLES;
  float acq_period = CONFIG_EXAMPLES_SENSOR_FUSION_SAMPLE_RATE / 1000.0f;

  /* Minimal required data for Fusion library. Use of magnetometer data
   * is optional and can improve performance.
   */

  struct sensor_accel imu_acc_data;
  struct sensor_gyro imu_gyro_data;
  FusionVector accelerometer;
  FusionVector gyroscope;
  FusionEuler euler;
  FusionAhrs ahrs;

  FusionAhrsInitialise(&ahrs);

  printf("Sensor Fusion example\n");
  printf("Sample Rate: %.2f Hz\n", 1.0 / acq_period);

  fd = open("/dev/imu0", O_RDONLY);
  if (fd < 0)
    {
      printf("Failed to open imu0\n");
      return EXIT_FAILURE;
    }

  for (int i = 0; i < iterations; i++)
    {
      read_mpu6050(fd, &imu_acc_data, &imu_gyro_data);

      accelerometer.axis.x = imu_acc_data.x;
      accelerometer.axis.y = imu_acc_data.y;
      accelerometer.axis.z = imu_acc_data.z;

      gyroscope.axis.x = imu_gyro_data.x;
      gyroscope.axis.y = imu_gyro_data.y;
      gyroscope.axis.z = imu_gyro_data.z;

      FusionAhrsUpdateNoMagnetometer(&ahrs,
                                     gyroscope,
                                     accelerometer,
                                     acq_period);
      euler = FusionQuaternionToEuler(FusionAhrsGetQuaternion(&ahrs));

      printf("Yaw: %.3f | Pitch: %.3f | Roll: %.3f\n",
             euler.angle.yaw, euler.angle.pitch, euler.angle.roll);
      usleep(CONFIG_EXAMPLES_SENSOR_FUSION_SAMPLE_RATE * 1000);
    }

  close(fd);

  return EXIT_SUCCESS;
}

void read_mpu6050(int fd,
                  struct sensor_accel *acc_data,
                  struct sensor_gyro *gyro_data)
{
  struct mpu6050_imu_msg raw_imu;
  int16_t raw_data[7];
  memset(&raw_imu, 0, sizeof(raw_imu));

  int ret = read(fd, &raw_data, sizeof(raw_data));
  if (ret != sizeof(raw_data))
    {
      printf("Failed to read accelerometer data\n");
    }
    else
    {
      raw_imu.acc_x = ((raw_data[0] & REG_HIGH_MASK) << 8) +
                        ((raw_data[0] & REG_LOW_MASK) >> 8);
      raw_imu.acc_y = ((raw_data[1] & REG_HIGH_MASK) << 8) +
                        ((raw_data[1] & REG_LOW_MASK) >> 8);
      raw_imu.acc_z = ((raw_data[2] & REG_HIGH_MASK) << 8) +
                        ((raw_data[2] & REG_LOW_MASK) >> 8);
      raw_imu.gyro_x = ((raw_data[4] & REG_HIGH_MASK) << 8) +
                        ((raw_data[4] & REG_LOW_MASK) >> 8);
      raw_imu.gyro_y = ((raw_data[5] & REG_HIGH_MASK) << 8) +
                        ((raw_data[5] & REG_LOW_MASK) >> 8);
      raw_imu.gyro_z = ((raw_data[6] & REG_HIGH_MASK) << 8) +
                        ((raw_data[6] & REG_LOW_MASK) >> 8);
    }

  acc_data->x = raw_imu.acc_x / MPU6050_AFS_SEL;
  acc_data->y = raw_imu.acc_y / MPU6050_AFS_SEL;
  acc_data->z = raw_imu.acc_z / MPU6050_AFS_SEL;

  /* Gyro data in ยบ/s (degrees per second) */

  gyro_data->x = raw_imu.gyro_x / MPU6050_FS_SEL;
  gyro_data->y = raw_imu.gyro_y / MPU6050_FS_SEL;
  gyro_data->z = raw_imu.gyro_z / MPU6050_FS_SEL;
}