mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-20 15:48:29 -04:00
a46d8dbd51
it should be one connection to the kernel per bus, not one per device
198 lines
5.4 KiB
C++
198 lines
5.4 KiB
C++
#include <AP_HAL.h>
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_LINUX
|
|
#include "SPIDriver.h"
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <stdio.h>
|
|
#include <stdint.h>
|
|
#include <sys/ioctl.h>
|
|
#include <linux/spi/spidev.h>
|
|
#include "GPIO.h"
|
|
|
|
#define SPI_DEBUGGING 1
|
|
|
|
using namespace Linux;
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
#define MHZ (1000U*1000U)
|
|
|
|
LinuxSPIDeviceDriver LinuxSPIDeviceManager::_device[LINUX_SPI_DEVICE_NUM_DEVICES] = {
|
|
LinuxSPIDeviceDriver(1, LINUX_SPI_DEVICE_MS5611, SPI_MODE_3, 8, BBB_P9_42, 6*MHZ, 6*MHZ),
|
|
LinuxSPIDeviceDriver(1, LINUX_SPI_DEVICE_MPU6000, SPI_MODE_3, 8, BBB_P9_28, 500*1000, 20*MHZ),
|
|
/* MPU9250 is restricted to 1MHz for non-data and interrupt registers */
|
|
LinuxSPIDeviceDriver(1, LINUX_SPI_DEVICE_MPU9250, SPI_MODE_3, 8, BBB_P9_23, 1*MHZ, 20*MHZ),
|
|
LinuxSPIDeviceDriver(0, LINUX_SPI_DEVICE_LSM9DS0, SPI_MODE_3, 8, BBB_P9_17, 10*MHZ,10*MHZ),
|
|
LinuxSPIDeviceDriver(1, LINUX_SPI_DEVICE_FRAM, SPI_MODE_0, 8, BBB_P8_12, 6*MHZ, 6*MHZ)
|
|
};
|
|
|
|
// have a separate semaphore per bus
|
|
LinuxSemaphore LinuxSPIDeviceManager::_semaphore[LINUX_SPI_NUM_BUSES];
|
|
int LinuxSPIDeviceManager::_fd[LINUX_SPI_NUM_BUSES];
|
|
|
|
LinuxSPIDeviceDriver::LinuxSPIDeviceDriver(uint8_t bus, LinuxSPIDeviceType type, uint8_t mode, uint8_t bitsPerWord, uint8_t cs_pin, uint32_t lowspeed, uint32_t highspeed):
|
|
_bus(bus),
|
|
_type(type),
|
|
_mode(mode),
|
|
_bitsPerWord(bitsPerWord),
|
|
_lowspeed(lowspeed),
|
|
_highspeed(highspeed),
|
|
_speed(highspeed),
|
|
_cs_pin(cs_pin)
|
|
{
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::init()
|
|
{
|
|
// Init the CS
|
|
_cs = hal.gpio->channel(_cs_pin);
|
|
if (_cs == NULL) {
|
|
hal.scheduler->panic("Unable to instantiate cs pin");
|
|
}
|
|
_cs->mode(HAL_GPIO_OUTPUT);
|
|
_cs->write(HIGH); // do not hold the SPI bus initially
|
|
}
|
|
|
|
AP_HAL::Semaphore* LinuxSPIDeviceDriver::get_semaphore()
|
|
{
|
|
return LinuxSPIDeviceManager::get_semaphore(_bus);
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::transaction(const uint8_t *tx, uint8_t *rx, uint16_t len)
|
|
{
|
|
LinuxSPIDeviceManager::transaction(*this, tx, rx, len);
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::set_bus_speed(enum bus_speed speed)
|
|
{
|
|
if (speed == SPI_SPEED_LOW) {
|
|
_speed = _lowspeed;
|
|
} else {
|
|
_speed = _highspeed;
|
|
}
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::cs_assert()
|
|
{
|
|
LinuxSPIDeviceManager::cs_assert(_type);
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::cs_release()
|
|
{
|
|
LinuxSPIDeviceManager::cs_release(_type);
|
|
}
|
|
|
|
uint8_t LinuxSPIDeviceDriver::transfer(uint8_t data)
|
|
{
|
|
uint8_t v = 0;
|
|
transaction(&data, &v, 1);
|
|
return v;
|
|
}
|
|
|
|
void LinuxSPIDeviceDriver::transfer(const uint8_t *data, uint16_t len)
|
|
{
|
|
transaction(data, NULL, len);
|
|
}
|
|
|
|
void LinuxSPIDeviceManager::init(void *)
|
|
{
|
|
char path[] = "/dev/spidevN.0";
|
|
for (uint8_t i=0; i<LINUX_SPI_NUM_BUSES; i++) {
|
|
path[11] = '1' + i;
|
|
_fd[i] = open(path, O_RDWR);
|
|
if (_fd[i] == -1) {
|
|
hal.scheduler->panic("SPIDriver: unable to open SPI bus");
|
|
}
|
|
}
|
|
for (uint8_t i=0; i<LINUX_SPI_DEVICE_NUM_DEVICES; i++) {
|
|
_device[i].init();
|
|
}
|
|
}
|
|
|
|
void LinuxSPIDeviceManager::cs_assert(LinuxSPIDeviceType type)
|
|
{
|
|
for (uint8_t i=0; i<LINUX_SPI_DEVICE_NUM_DEVICES; i++) {
|
|
if (_device[i]._bus != _device[type]._bus) {
|
|
// not the same bus
|
|
continue;
|
|
}
|
|
if (i != type) {
|
|
if (_device[i]._cs->read() != 1) {
|
|
hal.scheduler->panic("two CS enabled at once");
|
|
}
|
|
}
|
|
}
|
|
_device[type]._cs->write(0);
|
|
}
|
|
|
|
void LinuxSPIDeviceManager::cs_release(LinuxSPIDeviceType type)
|
|
{
|
|
for (uint8_t i=0; i<LINUX_SPI_DEVICE_NUM_DEVICES; i++) {
|
|
if (_device[i]._bus != _device[type]._bus) {
|
|
// not the same bus
|
|
continue;
|
|
}
|
|
_device[i]._cs->write(1);
|
|
}
|
|
}
|
|
|
|
void LinuxSPIDeviceManager::transaction(LinuxSPIDeviceDriver &driver, const uint8_t *tx, uint8_t *rx, uint16_t len)
|
|
{
|
|
// we set the mode before we assert the CS line so that the bus is
|
|
// in the correct idle state before the chip is selected
|
|
ioctl(_fd[driver._bus], SPI_IOC_WR_MODE, &driver._mode);
|
|
|
|
cs_assert(driver._type);
|
|
struct spi_ioc_transfer spi[1];
|
|
memset(spi, 0, sizeof(spi));
|
|
spi[0].tx_buf = (uint64_t)tx;
|
|
spi[0].rx_buf = (uint64_t)rx;
|
|
spi[0].len = len;
|
|
spi[0].delay_usecs = 0;
|
|
spi[0].speed_hz = driver._speed;
|
|
spi[0].bits_per_word = driver._bitsPerWord;
|
|
spi[0].cs_change = 0;
|
|
|
|
if (rx != NULL) {
|
|
// keep valgrind happy
|
|
memset(rx, 0, len);
|
|
}
|
|
|
|
ioctl(_fd[driver._bus], SPI_IOC_MESSAGE(1), &spi);
|
|
cs_release(driver._type);
|
|
}
|
|
|
|
/*
|
|
return a SPIDeviceDriver for a particular device
|
|
*/
|
|
AP_HAL::SPIDeviceDriver* LinuxSPIDeviceManager::device(enum AP_HAL::SPIDevice dev)
|
|
{
|
|
switch (dev) {
|
|
case AP_HAL::SPIDevice_MPU6000:
|
|
return &_device[LINUX_SPI_DEVICE_MPU6000];
|
|
case AP_HAL::SPIDevice_MPU9250:
|
|
return &_device[LINUX_SPI_DEVICE_MPU9250];
|
|
case AP_HAL::SPIDevice_MS5611:
|
|
return &_device[LINUX_SPI_DEVICE_MS5611];
|
|
case AP_HAL::SPIDevice_LSM9DS0:
|
|
return &_device[LINUX_SPI_DEVICE_LSM9DS0];
|
|
case AP_HAL::SPIDevice_Dataflash:
|
|
return &_device[LINUX_SPI_DEVICE_FRAM];
|
|
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
return the bus specific semaphore
|
|
*/
|
|
AP_HAL::Semaphore *LinuxSPIDeviceManager::get_semaphore(uint8_t bus)
|
|
{
|
|
return &_semaphore[bus];
|
|
}
|
|
|
|
#endif // CONFIG_HAL_BOARD
|