mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-13 11:28:30 -04:00
e8cceff543
Semaphore can be null, handled correctly if it is. Rather than check for which SPI bus the sensor is using, just check whether semaphore is not null before using it. More general and flexible.
146 lines
6.7 KiB
C++
146 lines
6.7 KiB
C++
#ifndef __AP_OPTICALFLOW_ADNS3080_H__
|
|
#define __AP_OPTICALFLOW_ADNS3080_H__
|
|
|
|
#include <AP_Semaphore.h>
|
|
#include "AP_OpticalFlow.h"
|
|
|
|
// default pin settings
|
|
#define ADNS3080_CHIP_SELECT 34 // PC3
|
|
#define ADNS3080_RESET 0 // reset pin is unattached by default
|
|
|
|
// orientations for ADNS3080 sensor
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_FORWARD ROTATION_YAW_180
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_FORWARD_RIGHT ROTATION_YAW_135
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_RIGHT ROTATION_YAW_90
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_BACK_RIGHT ROTATION_YAW_45
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_BACK ROTATION_NONE
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_BACK_LEFT ROTATION_YAW_315
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_LEFT ROTATION_YAW_270
|
|
#define AP_OPTICALFLOW_ADNS3080_PINS_FORWARD_LEFT ROTATION_YAW_225
|
|
|
|
// field of view of ADNS3080 sensor lenses
|
|
#define AP_OPTICALFLOW_ADNS3080_08_FOV 0.202458 // 11.6 degrees
|
|
|
|
// scaler - value returned when sensor is moved equivalent of 1 pixel
|
|
#define AP_OPTICALFLOW_ADNS3080_SCALER 1.1
|
|
|
|
// ADNS3080 hardware config
|
|
#define ADNS3080_PIXELS_X 30
|
|
#define ADNS3080_PIXELS_Y 30
|
|
#define ADNS3080_CLOCK_SPEED 24000000
|
|
|
|
// Register Map for the ADNS3080 Optical OpticalFlow Sensor
|
|
#define ADNS3080_PRODUCT_ID 0x00
|
|
#define ADNS3080_REVISION_ID 0x01
|
|
#define ADNS3080_MOTION 0x02
|
|
#define ADNS3080_DELTA_X 0x03
|
|
#define ADNS3080_DELTA_Y 0x04
|
|
#define ADNS3080_SQUAL 0x05
|
|
#define ADNS3080_PIXEL_SUM 0x06
|
|
#define ADNS3080_MAXIMUM_PIXEL 0x07
|
|
#define ADNS3080_CONFIGURATION_BITS 0x0a
|
|
#define ADNS3080_EXTENDED_CONFIG 0x0b
|
|
#define ADNS3080_DATA_OUT_LOWER 0x0c
|
|
#define ADNS3080_DATA_OUT_UPPER 0x0d
|
|
#define ADNS3080_SHUTTER_LOWER 0x0e
|
|
#define ADNS3080_SHUTTER_UPPER 0x0f
|
|
#define ADNS3080_FRAME_PERIOD_LOWER 0x10
|
|
#define ADNS3080_FRAME_PERIOD_UPPER 0x11
|
|
#define ADNS3080_MOTION_CLEAR 0x12
|
|
#define ADNS3080_FRAME_CAPTURE 0x13
|
|
#define ADNS3080_SROM_ENABLE 0x14
|
|
#define ADNS3080_FRAME_PERIOD_MAX_BOUND_LOWER 0x19
|
|
#define ADNS3080_FRAME_PERIOD_MAX_BOUND_UPPER 0x1a
|
|
#define ADNS3080_FRAME_PERIOD_MIN_BOUND_LOWER 0x1b
|
|
#define ADNS3080_FRAME_PERIOD_MIN_BOUND_UPPER 0x1c
|
|
#define ADNS3080_SHUTTER_MAX_BOUND_LOWER 0x1e
|
|
#define ADNS3080_SHUTTER_MAX_BOUND_UPPER 0x1e
|
|
#define ADNS3080_SROM_ID 0x1f
|
|
#define ADNS3080_OBSERVATION 0x3d
|
|
#define ADNS3080_INVERSE_PRODUCT_ID 0x3f
|
|
#define ADNS3080_PIXEL_BURST 0x40
|
|
#define ADNS3080_MOTION_BURST 0x50
|
|
#define ADNS3080_SROM_LOAD 0x60
|
|
|
|
// Configuration Bits
|
|
#define ADNS3080_LED_MODE_ALWAYS_ON 0x00
|
|
#define ADNS3080_LED_MODE_WHEN_REQUIRED 0x01
|
|
|
|
#define ADNS3080_RESOLUTION_400 400
|
|
#define ADNS3080_RESOLUTION_1600 1600
|
|
|
|
// Extended Configuration bits
|
|
#define ADNS3080_SERIALNPU_OFF 0x02
|
|
|
|
#define ADNS3080_FRAME_RATE_MAX 6469
|
|
#define ADNS3080_FRAME_RATE_MIN 2000
|
|
|
|
// SPI bus definitions
|
|
#define ADNS3080_SPI_UNKNOWN 0
|
|
#define ADNS3080_SPIBUS_1 1 // standard SPI bus
|
|
#define ADNS3080_SPIBUS_3 3 // SPI3
|
|
|
|
class AP_OpticalFlow_ADNS3080 : public AP_OpticalFlow
|
|
{
|
|
public:
|
|
AP_OpticalFlow_ADNS3080(AP_Semaphore* semaphore, int16_t cs_pin = ADNS3080_CHIP_SELECT, int16_t reset_pin = ADNS3080_RESET);
|
|
bool init(bool initCommAPI, AP_PeriodicProcess *scheduler); // parameter controls whether I2C/SPI interface is initialised (set to false if other devices are on the I2C/SPI bus and have already initialised the interface)
|
|
uint8_t read_register(uint8_t address);
|
|
void write_register(uint8_t address, uint8_t value);
|
|
void reset(); // reset sensor by holding a pin high (or is it low?) for 10us.
|
|
void update(uint32_t now); // read latest values from sensor and fill in x,y and totals, return true on successful read
|
|
|
|
// ADNS3080 specific features
|
|
|
|
// return true if there has been motion since the last time this was called
|
|
bool motion() { if( _motion ) { _motion = false; return true; }else{ return false; } }
|
|
|
|
bool overflow() { return _overflow; } // true if there has been an overflow
|
|
|
|
void disable_serial_pullup();
|
|
|
|
bool get_led_always_on(); // returns true if LED is always on, false if only on when required
|
|
void set_led_always_on( bool alwaysOn ); // set parameter to true if you want LED always on, otherwise false for only when required
|
|
|
|
int16_t get_resolution(); // returns resolution (either 400 or 1600 counts per inch)
|
|
void set_resolution(uint16_t resolution); // set parameter to 400 or 1600 counts per inch
|
|
|
|
bool get_frame_rate_auto(); // get_frame_rate_auto - return true if frame rate is set to "auto", false if manual
|
|
void set_frame_rate_auto(bool auto_frame_rate); // set_frame_rate_auto(bool) - set frame rate to auto (true), or manual (false)
|
|
|
|
uint16_t get_frame_period(); // get_frame_period
|
|
void set_frame_period(uint16_t period);
|
|
|
|
uint16_t get_frame_rate();
|
|
void set_frame_rate(uint16_t rate);
|
|
|
|
bool get_shutter_speed_auto(); // get_shutter_speed_auto - returns true if shutter speed is adjusted automatically, false if manual
|
|
void set_shutter_speed_auto(bool auto_shutter_speed); // set_shutter_speed_auto - set shutter speed to auto (true), or manual (false)
|
|
|
|
uint16_t get_shutter_speed();
|
|
void set_shutter_speed(uint16_t shutter_speed);
|
|
|
|
void clear_motion(); // will cause the x,y, dx, dy, and the sensor's motion registers to be cleared
|
|
|
|
void print_pixel_data(Stream *serPort); // dumps a 30x30 image to the Serial port
|
|
|
|
private:
|
|
// bytes to store SPI settings
|
|
uint8_t orig_spi_settings_spcr; // spi1's mode
|
|
uint8_t orig_spi3_settings_ucsr3c; // spi3's mode
|
|
uint8_t orig_spi3_settings_ubrr3; // spi3's speed
|
|
|
|
// save and restore SPI settings
|
|
void backup_spi_settings();
|
|
void restore_spi_settings();
|
|
|
|
int16_t _cs_pin; // pin used for chip select
|
|
int16_t _reset_pin; // pin used for chip reset
|
|
bool _motion; // true if there has been motion
|
|
bool _overflow; // true if the x or y data buffers overflowed
|
|
uint8_t _spi_bus; // 0 = unknown, 1 = using SPI, 3 = using SPI3
|
|
AP_Semaphore* _semaphore;
|
|
};
|
|
|
|
#endif
|