2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
25 #include "drivers/bus_i2c.h"
26 #include "drivers/io_types.h"
27 #include "drivers/dma.h"
33 BUS_TYPE_MPU_SLAVE
, // Slave I2C on SPI master
34 BUS_TYPE_GYRO_AUTO
, // Only used by acc/gyro bus auto detection code
46 // Bus interface, independent of connected device
47 typedef struct busDevice_s
{
51 SPI_TypeDef
*instance
;
58 struct busMpuSlave_s
{
59 struct extDevice_s
*master
;
64 dmaChannelDescriptor_t
*dmaTx
;
65 dmaChannelDescriptor_t
*dmaRx
;
67 // Use a reference here as this saves RAM for unused descriptors
68 #if defined(USE_FULL_LL_DRIVER)
69 LL_DMA_InitTypeDef
*initTx
;
70 LL_DMA_InitTypeDef
*initRx
;
72 DMA_InitTypeDef
*initTx
;
73 DMA_InitTypeDef
*initRx
;
76 volatile struct busSegment_s
* volatile curSegment
;
80 // External device has an associated bus and bus dependent address
81 typedef struct extDevice_s
{
92 struct extMpuSlave_s
{
97 // Cache the init structure for the next DMA transfer to reduce inter-segment delay
98 #if defined(USE_FULL_LL_DRIVER)
99 LL_DMA_InitTypeDef initTx
;
100 LL_DMA_InitTypeDef initRx
;
102 DMA_InitTypeDef initTx
;
103 DMA_InitTypeDef initRx
;
106 // Support disabling DMA on a per device basis
108 // Per device buffer reference if needed
109 uint8_t *txBuf
, *rxBuf
;
110 // Connected devices on the same bus may support different speeds
111 uint32_t callbackArg
;
114 /* Each SPI access may comprise multiple parts, for example, wait/write enable/write/data each of which
115 * is defined by a segment, with optional callback after each is completed.
117 * If there are more than one segments, or a single segment with negateCS negated then DMA will be used irrespective of length
120 typedef struct busSegment_s
{
125 // Receive buffer, or in the case of the final segment to
129 // Link to the device associated with the next transfer
130 const extDevice_t
*dev
;
131 // Segments to process in the next transfer.
132 volatile struct busSegment_s
*segments
;
136 bool negateCS
; // Should CS be negated at the end of this segment
137 busStatus_e (*callback
)(uint32_t arg
);
140 #ifdef TARGET_BUS_INIT
141 void targetBusInit(void);
144 // Access routines where the register is accessed directly
145 bool busRawWriteRegister(const extDevice_t
*dev
, uint8_t reg
, uint8_t data
);
146 bool busRawWriteRegisterStart(const extDevice_t
*dev
, uint8_t reg
, uint8_t data
);
147 bool busRawReadRegisterBuffer(const extDevice_t
*dev
, uint8_t reg
, uint8_t *data
, uint8_t length
);
148 bool busRawReadRegisterBufferStart(const extDevice_t
*dev
, uint8_t reg
, uint8_t *data
, uint8_t length
);
149 // Write routines where the register is masked with 0x7f
150 bool busWriteRegister(const extDevice_t
*dev
, uint8_t reg
, uint8_t data
);
151 bool busWriteRegisterStart(const extDevice_t
*dev
, uint8_t reg
, uint8_t data
);
152 // Read routines where the register is ORed with 0x80
153 bool busReadRegisterBuffer(const extDevice_t
*dev
, uint8_t reg
, uint8_t *data
, uint8_t length
);
154 bool busReadRegisterBufferStart(const extDevice_t
*dev
, uint8_t reg
, uint8_t *data
, uint8_t length
);
155 uint8_t busReadRegister(const extDevice_t
*dev
, uint8_t reg
);
157 bool busBusy(const extDevice_t
*dev
, bool *error
);
158 void busDeviceRegister(const extDevice_t
*dev
);