/**************************************************************************** * 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 #include #include #include #include #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; }