mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-06-30 01:00:35 +08:00
3198610f85
and remove the px4_ prefix, except for px4_config.h.
command to update includes:
for k in app.h atomic.h cli.h console_buffer.h defines.h getopt.h i2c.h init.h log.h micro_hal.h module.h module_params.h param.h param_macros.h posix.h sem.h sem.hpp shmem.h shutdown.h tasks.h time.h workqueue.h; do for i in $(grep -rl 'include <px4_'$k src platforms boards); do sed -i 's/#include <px4_'$k'/#include <px4_platform_common\/'$k/ $i; done; done
for in $(grep -rl 'include <px4_config.h' src platforms boards); do sed -i 's/#include <px4_config.h/#include <px4_platform_common\/px4_config.h'/ $i; done
Transitional headers for submodules are added (px4_{defines,log,time}.h)
197 lines
5.2 KiB
C++
197 lines
5.2 KiB
C++
/****************************************************************************
|
||
*
|
||
* Copyright (C) 2019 PX4 Development Team. All rights reserved.
|
||
*
|
||
* Redistribution and use in source and binary forms, with or without
|
||
* modification, are permitted provided that the following conditions
|
||
* are met:
|
||
*
|
||
* 1. Redistributions of source code must retain the above copyright
|
||
* notice, this list of conditions and the following disclaimer.
|
||
* 2. Redistributions in binary form must reproduce the above copyright
|
||
* notice, this list of conditions and the following disclaimer in
|
||
* the documentation and/or other materials provided with the
|
||
* distribution.
|
||
* 3. Neither the name PX4 nor the names of its contributors may be
|
||
* used to endorse or promote products derived from this software
|
||
* without specific prior written permission.
|
||
*
|
||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
||
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
||
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
||
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
||
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
||
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
||
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
* POSSIBILITY OF SUCH DAMAGE.
|
||
*
|
||
****************************************************************************/
|
||
|
||
/**
|
||
* @file SPI.cpp
|
||
*
|
||
* Base class for devices connected via SPI.
|
||
*
|
||
*/
|
||
|
||
#include "SPI.hpp"
|
||
|
||
#ifdef __PX4_LINUX
|
||
|
||
#include <fcntl.h>
|
||
#include <unistd.h>
|
||
#include <sys/ioctl.h>
|
||
#include <linux/types.h>
|
||
#include <linux/spi/spidev.h>
|
||
|
||
#include <px4_platform_common/px4_config.h>
|
||
|
||
namespace device
|
||
{
|
||
|
||
SPI::SPI(const char *name, const char *devname, int bus, uint32_t device, enum spi_mode_e mode, uint32_t frequency) :
|
||
CDev(name, devname),
|
||
_device(device),
|
||
_mode(mode),
|
||
_frequency(frequency)
|
||
{
|
||
DEVICE_DEBUG("SPI::SPI name = %s devname = %s", name, devname);
|
||
|
||
// fill in _device_id fields for a SPI device
|
||
_device_id.devid_s.bus_type = DeviceBusType_SPI;
|
||
_device_id.devid_s.bus = bus;
|
||
_device_id.devid_s.address = (uint8_t)device;
|
||
// devtype needs to be filled in by the driver
|
||
_device_id.devid_s.devtype = 0;
|
||
}
|
||
|
||
SPI::~SPI()
|
||
{
|
||
if (_fd >= 0) {
|
||
::close(_fd);
|
||
_fd = -1;
|
||
}
|
||
}
|
||
|
||
int
|
||
SPI::init()
|
||
{
|
||
// Open the actual SPI device
|
||
char dev_path[16];
|
||
snprintf(dev_path, sizeof(dev_path), "/dev/spidev%i.%i", get_device_bus(), PX4_SPI_DEV_ID(_device));
|
||
DEVICE_DEBUG("%s", dev_path);
|
||
_fd = ::open(dev_path, O_RDWR);
|
||
|
||
if (_fd < 0) {
|
||
PX4_ERR("could not open %s", dev_path);
|
||
return PX4_ERROR;
|
||
}
|
||
|
||
/* call the probe function to check whether the device is present */
|
||
int ret = probe();
|
||
|
||
if (ret != OK) {
|
||
DEVICE_DEBUG("probe failed");
|
||
return ret;
|
||
}
|
||
|
||
/* do base class init, which will create the device node, etc. */
|
||
ret = CDev::init();
|
||
|
||
if (ret != OK) {
|
||
DEVICE_DEBUG("cdev init failed");
|
||
return ret;
|
||
}
|
||
|
||
/* tell the workd where we are */
|
||
DEVICE_LOG("on SPI bus %d at %d (%u KHz)", get_device_bus(), PX4_SPI_DEV_ID(_device), _frequency / 1000);
|
||
|
||
return PX4_OK;
|
||
}
|
||
|
||
int
|
||
SPI::transfer(uint8_t *send, uint8_t *recv, unsigned len)
|
||
{
|
||
if ((send == nullptr) && (recv == nullptr)) {
|
||
return -EINVAL;
|
||
}
|
||
|
||
// set write mode of SPI
|
||
int result = ::ioctl(_fd, SPI_IOC_WR_MODE, &_mode);
|
||
|
||
if (result == -1) {
|
||
PX4_ERR("can’t set spi mode");
|
||
return PX4_ERROR;
|
||
}
|
||
|
||
spi_ioc_transfer spi_transfer[1] {}; // datastructures for linux spi interface
|
||
|
||
spi_transfer[0].tx_buf = (uint64_t)send;
|
||
spi_transfer[0].rx_buf = (uint64_t)recv;
|
||
spi_transfer[0].len = len;
|
||
spi_transfer[0].speed_hz = _frequency;
|
||
spi_transfer[0].bits_per_word = 8;
|
||
//spi_transfer[0].delay_usecs = 10;
|
||
spi_transfer[0].cs_change = true;
|
||
|
||
result = ::ioctl(_fd, SPI_IOC_MESSAGE(1), &spi_transfer);
|
||
|
||
if (result != (int)len) {
|
||
PX4_ERR("write failed. Reported %d bytes written (%s)", result, strerror(errno));
|
||
return -1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
int
|
||
SPI::transferhword(uint16_t *send, uint16_t *recv, unsigned len)
|
||
{
|
||
if ((send == nullptr) && (recv == nullptr)) {
|
||
return -EINVAL;
|
||
}
|
||
|
||
// set write mode of SPI
|
||
int result = ::ioctl(_fd, SPI_IOC_WR_MODE, &_mode);
|
||
|
||
if (result == -1) {
|
||
PX4_ERR("can’t set spi mode");
|
||
return PX4_ERROR;
|
||
}
|
||
|
||
int bits = 16;
|
||
result = ::ioctl(_fd, SPI_IOC_WR_BITS_PER_WORD, &bits);
|
||
|
||
if (result == -1) {
|
||
PX4_ERR("can’t set 16 bit spi mode");
|
||
return PX4_ERROR;
|
||
}
|
||
|
||
spi_ioc_transfer spi_transfer[1] {}; // datastructures for linux spi interface
|
||
|
||
spi_transfer[0].tx_buf = (uint64_t)send;
|
||
spi_transfer[0].rx_buf = (uint64_t)recv;
|
||
spi_transfer[0].len = len * 2;
|
||
spi_transfer[0].speed_hz = _frequency;
|
||
//spi_transfer[0].bits_per_word = 8;
|
||
//spi_transfer[0].delay_usecs = 10;
|
||
spi_transfer[0].cs_change = true;
|
||
|
||
result = ::ioctl(_fd, SPI_IOC_MESSAGE(1), &spi_transfer);
|
||
|
||
if (result != (int)(len * 2)) {
|
||
PX4_ERR("write failed. Reported %d bytes written (%s)", result, strerror(errno));
|
||
return -1;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
} // namespace device
|
||
|
||
#endif // __PX4_LINUX
|