spi.h 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  1. /*
  2. * Copyright (c) 2015 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Public API for SPI drivers and applications
  9. */
  10. #ifndef ZEPHYR_INCLUDE_DRIVERS_SPI_H_
  11. #define ZEPHYR_INCLUDE_DRIVERS_SPI_H_
  12. /**
  13. * @brief SPI Interface
  14. * @defgroup spi_interface SPI Interface
  15. * @ingroup io_interfaces
  16. * @{
  17. */
  18. #include <zephyr/types.h>
  19. #include <stddef.h>
  20. #include <device.h>
  21. #include <drivers/gpio.h>
  22. #ifdef __cplusplus
  23. extern "C" {
  24. #endif
  25. /**
  26. * @brief SPI operational mode
  27. */
  28. #define SPI_OP_MODE_MASTER 0
  29. #define SPI_OP_MODE_SLAVE BIT(0)
  30. #define SPI_OP_MODE_MASK 0x1
  31. #define SPI_OP_MODE_GET(_operation_) ((_operation_) & SPI_OP_MODE_MASK)
  32. /**
  33. * @brief SPI Polarity & Phase Modes
  34. */
  35. /**
  36. * Clock Polarity: if set, clock idle state will be 1
  37. * and active state will be 0. If untouched, the inverse will be true
  38. * which is the default.
  39. */
  40. #define SPI_MODE_CPOL BIT(1)
  41. /**
  42. * Clock Phase: this dictates when is the data captured, and depends
  43. * clock's polarity. When SPI_MODE_CPOL is set and this bit as well,
  44. * capture will occur on low to high transition and high to low if
  45. * this bit is not set (default). This is fully reversed if CPOL is
  46. * not set.
  47. */
  48. #define SPI_MODE_CPHA BIT(2)
  49. /**
  50. * Whatever data is transmitted is looped-back to the receiving buffer of
  51. * the controller. This is fully controller dependent as some may not
  52. * support this, and can be used for testing purposes only.
  53. */
  54. #define SPI_MODE_LOOP BIT(3)
  55. #define SPI_MODE_MASK (0xE)
  56. #define SPI_MODE_GET(_mode_) \
  57. ((_mode_) & SPI_MODE_MASK)
  58. /**
  59. * @brief SPI Transfer modes (host controller dependent)
  60. */
  61. #define SPI_TRANSFER_MSB (0)
  62. #define SPI_TRANSFER_LSB BIT(4)
  63. /**
  64. * @brief SPI word size
  65. */
  66. #define SPI_WORD_SIZE_SHIFT (5)
  67. #define SPI_WORD_SIZE_MASK (0x3F << SPI_WORD_SIZE_SHIFT)
  68. #define SPI_WORD_SIZE_GET(_operation_) \
  69. (((_operation_) & SPI_WORD_SIZE_MASK) >> SPI_WORD_SIZE_SHIFT)
  70. #define SPI_WORD_SET(_word_size_) \
  71. ((_word_size_) << SPI_WORD_SIZE_SHIFT)
  72. /**
  73. * @brief SPI MISO lines
  74. *
  75. * Some controllers support dual, quad or octal MISO lines connected to slaves.
  76. * Default is single, which is the case most of the time.
  77. */
  78. #define SPI_LINES_SINGLE (0 << 11)
  79. #define SPI_LINES_DUAL (1 << 11)
  80. #define SPI_LINES_QUAD (2 << 11)
  81. #define SPI_LINES_OCTAL (3 << 11)
  82. #define SPI_LINES_MASK (0x3 << 11)
  83. /**
  84. * @brief Specific SPI devices control bits
  85. */
  86. /* Requests - if possible - to keep CS asserted after the transaction */
  87. #define SPI_HOLD_ON_CS BIT(13)
  88. /* Keep the device locked after the transaction for the current config.
  89. * Use this with extreme caution (see spi_release() below) as it will
  90. * prevent other callers to access the SPI device until spi_release() is
  91. * properly called.
  92. */
  93. #define SPI_LOCK_ON BIT(14)
  94. /* Active high logic on CS - Usually, and by default, CS logic is active
  95. * low. However, some devices may require the reverse logic: active high.
  96. * This bit will request the controller to use that logic. Note that not
  97. * all controllers are able to handle that natively. In this case deferring
  98. * the CS control to a gpio line through struct spi_cs_control would be
  99. * the solution.
  100. */
  101. #define SPI_CS_ACTIVE_HIGH BIT(15)
  102. /**
  103. * @brief SPI Chip Select control structure
  104. *
  105. * This can be used to control a CS line via a GPIO line, instead of
  106. * using the controller inner CS logic.
  107. *
  108. * @param gpio_dev is a valid pointer to an actual GPIO device. A NULL pointer
  109. * can be provided to full inhibit CS control if necessary.
  110. * @param gpio_pin is a number representing the gpio PIN that will be used
  111. * to act as a CS line
  112. * @param delay is a delay in microseconds to wait before starting the
  113. * transmission and before releasing the CS line
  114. * @param gpio_dt_flags is the devicetree flags corresponding to how the CS
  115. * line should be driven. GPIO_ACTIVE_LOW/GPIO_ACTIVE_HIGH should be
  116. * equivalent to SPI_CS_ACTIVE_HIGH/SPI_CS_ACTIVE_LOW options in struct
  117. * spi_config.
  118. */
  119. struct spi_cs_control {
  120. const struct device *gpio_dev;
  121. uint32_t delay;
  122. gpio_pin_t gpio_pin;
  123. gpio_dt_flags_t gpio_dt_flags;
  124. };
  125. #ifndef __cplusplus
  126. /**
  127. * @brief Initialize and get a pointer to a @p spi_cs_control from a
  128. * devicetree node identifier
  129. *
  130. * This helper is useful for initializing a device on a SPI bus. It
  131. * initializes a struct spi_cs_control and returns a pointer to it.
  132. * Here, @p node_id is a node identifier for a SPI device, not a SPI
  133. * controller.
  134. *
  135. * Example devicetree fragment:
  136. *
  137. * spi@... {
  138. * cs-gpios = <&gpio0 1 GPIO_ACTIVE_LOW>;
  139. * spidev: spi-device@0 { ... };
  140. * };
  141. *
  142. * Assume that @p gpio0 follows the standard convention for specifying
  143. * GPIOs, i.e. it has the following in its binding:
  144. *
  145. * gpio-cells:
  146. * - pin
  147. * - flags
  148. *
  149. * Example usage:
  150. *
  151. * struct spi_cs_control *ctrl =
  152. * SPI_CS_CONTROL_PTR_DT(DT_NODELABEL(spidev), 2);
  153. *
  154. * This example is equivalent to:
  155. *
  156. * struct spi_cs_control *ctrl =
  157. * &(struct spi_cs_control) {
  158. * .gpio_dev = DEVICE_DT_GET(DT_NODELABEL(gpio0)),
  159. * .delay = 2,
  160. * .gpio_pin = 1,
  161. * .gpio_dt_flags = GPIO_ACTIVE_LOW
  162. * };
  163. *
  164. * This macro is not available in C++.
  165. *
  166. * @param node_id Devicetree node identifier for a device on a SPI bus
  167. * @param delay_ The @p delay field to set in the @p spi_cs_control
  168. * @return a pointer to the @p spi_cs_control structure
  169. */
  170. #define SPI_CS_CONTROL_PTR_DT(node_id, delay_) \
  171. (&(struct spi_cs_control) { \
  172. .gpio_dev = DEVICE_DT_GET( \
  173. DT_SPI_DEV_CS_GPIOS_CTLR(node_id)), \
  174. .delay = (delay_), \
  175. .gpio_pin = DT_SPI_DEV_CS_GPIOS_PIN(node_id), \
  176. .gpio_dt_flags = DT_SPI_DEV_CS_GPIOS_FLAGS(node_id), \
  177. })
  178. /**
  179. * @brief Get a pointer to a @p spi_cs_control from a devicetree node
  180. *
  181. * This is equivalent to
  182. * <tt>SPI_CS_CONTROL_PTR_DT(DT_DRV_INST(inst), delay)</tt>.
  183. *
  184. * Therefore, @p DT_DRV_COMPAT must already be defined before using
  185. * this macro.
  186. *
  187. * This macro is not available in C++.
  188. *
  189. * @param inst Devicetree node instance number
  190. * @param delay_ The @p delay field to set in the @p spi_cs_control
  191. * @return a pointer to the @p spi_cs_control structure
  192. */
  193. #define SPI_CS_CONTROL_PTR_DT_INST(inst, delay_) \
  194. SPI_CS_CONTROL_PTR_DT(DT_DRV_INST(inst), delay_)
  195. #endif
  196. /**
  197. * @brief SPI controller configuration structure
  198. *
  199. * @param frequency is the bus frequency in Hertz
  200. * @param operation is a bit field with the following parts:
  201. *
  202. * operational mode [ 0 ] - master or slave.
  203. * mode [ 1 : 3 ] - Polarity, phase and loop mode.
  204. * transfer [ 4 ] - LSB or MSB first.
  205. * word_size [ 5 : 10 ] - Size of a data frame in bits.
  206. * lines [ 11 : 12 ] - MISO lines: Single/Dual/Quad/Octal.
  207. * cs_hold [ 13 ] - Hold on the CS line if possible.
  208. * lock_on [ 14 ] - Keep resource locked for the caller.
  209. * cs_active_high [ 15 ] - Active high CS logic.
  210. * @param slave is the slave number from 0 to host controller slave limit.
  211. * @param cs is a valid pointer on a struct spi_cs_control is CS line is
  212. * emulated through a gpio line, or NULL otherwise.
  213. *
  214. * @note Only cs_hold and lock_on can be changed between consecutive
  215. * transceive call. Rest of the attributes are not meant to be tweaked.
  216. *
  217. * @warning Most drivers use pointer comparison to determine whether a
  218. * passed configuration is different from one used in a previous
  219. * transaction. Changes to fields in the structure may not be
  220. * detected.
  221. */
  222. struct spi_config {
  223. uint32_t frequency;
  224. uint16_t operation;
  225. uint16_t slave;
  226. const struct spi_cs_control *cs;
  227. };
  228. #ifndef __cplusplus
  229. /**
  230. * @brief Structure initializer for spi_config from devicetree
  231. *
  232. * This helper macro expands to a static initializer for a <tt>struct
  233. * spi_config</tt> by reading the relevant @p frequency, @p slave, and
  234. * @p cs data from the devicetree.
  235. *
  236. * Important: the @p cs field is initialized using
  237. * SPI_CS_CONTROL_PTR_DT(). The @p gpio_dev value pointed to by this
  238. * structure must be checked using device_is_ready() before use.
  239. *
  240. * This macro is not available in C++.
  241. *
  242. * @param node_id Devicetree node identifier for the SPI device whose
  243. * struct spi_config to create an initializer for
  244. * @param operation_ the desired @p operation field in the struct spi_config
  245. * @param delay_ the desired @p delay field in the struct spi_config's
  246. * spi_cs_control, if there is one
  247. */
  248. #define SPI_CONFIG_DT(node_id, operation_, delay_) \
  249. { \
  250. .frequency = DT_PROP(node_id, spi_max_frequency), \
  251. .operation = (operation_), \
  252. .slave = DT_REG_ADDR(node_id), \
  253. .cs = COND_CODE_1( \
  254. DT_SPI_DEV_HAS_CS_GPIOS(node_id), \
  255. (SPI_CS_CONTROL_PTR_DT(node_id, delay_)), \
  256. (NULL)), \
  257. }
  258. /**
  259. * @brief Structure initializer for spi_config from devicetree instance
  260. *
  261. * This is equivalent to
  262. * <tt>SPI_CONFIG_DT(DT_DRV_INST(inst), operation_, delay_)</tt>.
  263. *
  264. * This macro is not available in C++.
  265. *
  266. * @param inst Devicetree instance number
  267. * @param operation_ the desired @p operation field in the struct spi_config
  268. * @param delay_ the desired @p delay field in the struct spi_config's
  269. * spi_cs_control, if there is one
  270. */
  271. #define SPI_CONFIG_DT_INST(inst, operation_, delay_) \
  272. SPI_CONFIG_DT(DT_DRV_INST(inst), operation_, delay_)
  273. #endif
  274. /**
  275. * @brief Complete SPI DT information
  276. *
  277. * @param bus is the SPI bus
  278. * @param config is the slave specific configuration
  279. */
  280. struct spi_dt_spec {
  281. const struct device *bus;
  282. struct spi_config config;
  283. };
  284. #ifndef __cplusplus
  285. /**
  286. * @brief Structure initializer for spi_dt_spec from devicetree
  287. *
  288. * This helper macro expands to a static initializer for a <tt>struct
  289. * spi_dt_spec</tt> by reading the relevant bus, frequency, slave, and cs
  290. * data from the devicetree.
  291. *
  292. * Important: multiple fields are automatically constructed by this macro
  293. * which must be checked before use. @ref spi_is_ready performs the required
  294. * @ref device_is_ready checks.
  295. *
  296. * This macro is not available in C++.
  297. *
  298. * @param node_id Devicetree node identifier for the SPI device whose
  299. * struct spi_dt_spec to create an initializer for
  300. * @param operation_ the desired @p operation field in the struct spi_config
  301. * @param delay_ the desired @p delay field in the struct spi_config's
  302. * spi_cs_control, if there is one
  303. */
  304. #define SPI_DT_SPEC_GET(node_id, operation_, delay_) \
  305. { \
  306. .bus = DEVICE_DT_GET(DT_BUS(node_id)), \
  307. .config = SPI_CONFIG_DT(node_id, operation_, delay_) \
  308. }
  309. /**
  310. * @brief Structure initializer for spi_dt_spec from devicetree instance
  311. *
  312. * This is equivalent to
  313. * <tt>SPI_DT_SPEC_GET(DT_DRV_INST(inst), operation_, delay_)</tt>.
  314. *
  315. * This macro is not available in C++.
  316. *
  317. * @param inst Devicetree instance number
  318. * @param operation_ the desired @p operation field in the struct spi_config
  319. * @param delay_ the desired @p delay field in the struct spi_config's
  320. * spi_cs_control, if there is one
  321. */
  322. #define SPI_DT_SPEC_INST_GET(inst, operation_, delay_) \
  323. SPI_DT_SPEC_GET(DT_DRV_INST(inst), operation_, delay_)
  324. #endif
  325. /**
  326. * @brief SPI buffer structure
  327. *
  328. * @param buf is a valid pointer on a data buffer, or NULL otherwise.
  329. * @param len is the length of the buffer or, if buf is NULL, will be the
  330. * length which as to be sent as dummy bytes (as TX buffer) or
  331. * the length of bytes that should be skipped (as RX buffer).
  332. */
  333. struct spi_buf {
  334. void *buf;
  335. size_t len;
  336. };
  337. /**
  338. * @brief SPI buffer array structure
  339. *
  340. * @param buffers is a valid pointer on an array of spi_buf, or NULL.
  341. * @param count is the length of the array pointed by buffers.
  342. */
  343. struct spi_buf_set {
  344. const struct spi_buf *buffers;
  345. size_t count;
  346. };
  347. /**
  348. * @typedef spi_api_io
  349. * @brief Callback API for I/O
  350. * See spi_transceive() for argument descriptions
  351. */
  352. typedef int (*spi_api_io)(const struct device *dev,
  353. const struct spi_config *config,
  354. const struct spi_buf_set *tx_bufs,
  355. const struct spi_buf_set *rx_bufs);
  356. /**
  357. * @typedef spi_api_io
  358. * @brief Callback API for asynchronous I/O
  359. * See spi_transceive_async() for argument descriptions
  360. */
  361. typedef int (*spi_api_io_async)(const struct device *dev,
  362. const struct spi_config *config,
  363. const struct spi_buf_set *tx_bufs,
  364. const struct spi_buf_set *rx_bufs,
  365. struct k_poll_signal *async);
  366. /**
  367. * @typedef spi_api_release
  368. * @brief Callback API for unlocking SPI device.
  369. * See spi_release() for argument descriptions
  370. */
  371. typedef int (*spi_api_release)(const struct device *dev,
  372. const struct spi_config *config);
  373. /**
  374. * @brief SPI driver API
  375. * This is the mandatory API any SPI driver needs to expose.
  376. */
  377. __subsystem struct spi_driver_api {
  378. spi_api_io transceive;
  379. #ifdef CONFIG_SPI_ASYNC
  380. spi_api_io_async transceive_async;
  381. #endif /* CONFIG_SPI_ASYNC */
  382. spi_api_release release;
  383. };
  384. /**
  385. * @brief Validate that SPI bus is ready.
  386. *
  387. * @param spec SPI specification from devicetree
  388. *
  389. * @retval true if the SPI bus is ready for use.
  390. * @retval false if the SPI bus is not ready for use.
  391. */
  392. static inline bool spi_is_ready(const struct spi_dt_spec *spec)
  393. {
  394. /* Validate bus is ready */
  395. if (!device_is_ready(spec->bus)) {
  396. return false;
  397. }
  398. /* Validate CS gpio port is ready, if it is used */
  399. if (spec->config.cs &&
  400. !device_is_ready(spec->config.cs->gpio_dev)) {
  401. return false;
  402. }
  403. return true;
  404. }
  405. /**
  406. * @brief Read/write the specified amount of data from the SPI driver.
  407. *
  408. * @note This function is synchronous.
  409. *
  410. * @param dev Pointer to the device structure for the driver instance
  411. * @param config Pointer to a valid spi_config structure instance.
  412. * Pointer-comparison may be used to detect changes from
  413. * previous operations.
  414. * @param tx_bufs Buffer array where data to be sent originates from,
  415. * or NULL if none.
  416. * @param rx_bufs Buffer array where data to be read will be written to,
  417. * or NULL if none.
  418. *
  419. * @retval frames Positive number of frames received in slave mode.
  420. * @retval 0 If successful in master mode.
  421. * @retval -errno Negative errno code on failure.
  422. */
  423. __syscall int spi_transceive(const struct device *dev,
  424. const struct spi_config *config,
  425. const struct spi_buf_set *tx_bufs,
  426. const struct spi_buf_set *rx_bufs);
  427. static inline int z_impl_spi_transceive(const struct device *dev,
  428. const struct spi_config *config,
  429. const struct spi_buf_set *tx_bufs,
  430. const struct spi_buf_set *rx_bufs)
  431. {
  432. const struct spi_driver_api *api =
  433. (const struct spi_driver_api *)dev->api;
  434. return api->transceive(dev, config, tx_bufs, rx_bufs);
  435. }
  436. /**
  437. * @brief Read/write data from an SPI bus specified in @p spi_dt_spec.
  438. *
  439. * This is equivalent to:
  440. *
  441. * spi_transceive(spec->bus, &spec->config, tx_bufs, rx_bufs);
  442. *
  443. * @param spec SPI specification from devicetree
  444. * @param tx_bufs Buffer array where data to be sent originates from,
  445. * or NULL if none.
  446. * @param rx_bufs Buffer array where data to be read will be written to,
  447. * or NULL if none.
  448. *
  449. * @return a value from spi_transceive().
  450. */
  451. static inline int spi_transceive_dt(const struct spi_dt_spec *spec,
  452. const struct spi_buf_set *tx_bufs,
  453. const struct spi_buf_set *rx_bufs)
  454. {
  455. return spi_transceive(spec->bus, &spec->config, tx_bufs, rx_bufs);
  456. }
  457. /**
  458. * @brief Read the specified amount of data from the SPI driver.
  459. *
  460. * @note This function is synchronous.
  461. *
  462. * @note This function is an helper function calling spi_transceive.
  463. *
  464. * @param dev Pointer to the device structure for the driver instance
  465. * @param config Pointer to a valid spi_config structure instance.
  466. * Pointer-comparison may be used to detect changes from
  467. * previous operations.
  468. * @param rx_bufs Buffer array where data to be read will be written to.
  469. *
  470. * @retval 0 If successful.
  471. * @retval -errno Negative errno code on failure.
  472. */
  473. static inline int spi_read(const struct device *dev,
  474. const struct spi_config *config,
  475. const struct spi_buf_set *rx_bufs)
  476. {
  477. return spi_transceive(dev, config, NULL, rx_bufs);
  478. }
  479. /**
  480. * @brief Read data from a SPI bus specified in @p spi_dt_spec.
  481. *
  482. * This is equivalent to:
  483. *
  484. * spi_read(spec->bus, &spec->config, rx_bufs);
  485. *
  486. * @param spec SPI specification from devicetree
  487. * @param rx_bufs Buffer array where data to be read will be written to.
  488. *
  489. * @return a value from spi_read().
  490. */
  491. static inline int spi_read_dt(const struct spi_dt_spec *spec,
  492. const struct spi_buf_set *rx_bufs)
  493. {
  494. return spi_read(spec->bus, &spec->config, rx_bufs);
  495. }
  496. /**
  497. * @brief Write the specified amount of data from the SPI driver.
  498. *
  499. * @note This function is synchronous.
  500. *
  501. * @note This function is an helper function calling spi_transceive.
  502. *
  503. * @param dev Pointer to the device structure for the driver instance
  504. * @param config Pointer to a valid spi_config structure instance.
  505. * Pointer-comparison may be used to detect changes from
  506. * previous operations.
  507. * @param tx_bufs Buffer array where data to be sent originates from.
  508. *
  509. * @retval 0 If successful.
  510. * @retval -errno Negative errno code on failure.
  511. */
  512. static inline int spi_write(const struct device *dev,
  513. const struct spi_config *config,
  514. const struct spi_buf_set *tx_bufs)
  515. {
  516. return spi_transceive(dev, config, tx_bufs, NULL);
  517. }
  518. /**
  519. * @brief Write data to a SPI bus specified in @p spi_dt_spec.
  520. *
  521. * This is equivalent to:
  522. *
  523. * spi_write(spec->bus, &spec->config, tx_bufs);
  524. *
  525. * @param spec SPI specification from devicetree
  526. * @param tx_bufs Buffer array where data to be sent originates from.
  527. *
  528. * @return a value from spi_write().
  529. */
  530. static inline int spi_write_dt(const struct spi_dt_spec *spec,
  531. const struct spi_buf_set *tx_bufs)
  532. {
  533. return spi_write(spec->bus, &spec->config, tx_bufs);
  534. }
  535. /* Doxygen defines this so documentation is generated. */
  536. #ifdef CONFIG_SPI_ASYNC
  537. /**
  538. * @brief Read/write the specified amount of data from the SPI driver.
  539. *
  540. * @note This function is asynchronous.
  541. *
  542. * @note This function is available only if @kconfig{CONFIG_SPI_ASYNC}
  543. * is selected.
  544. *
  545. * @param dev Pointer to the device structure for the driver instance
  546. * @param config Pointer to a valid spi_config structure instance.
  547. * Pointer-comparison may be used to detect changes from
  548. * previous operations.
  549. * @param tx_bufs Buffer array where data to be sent originates from,
  550. * or NULL if none.
  551. * @param rx_bufs Buffer array where data to be read will be written to,
  552. * or NULL if none.
  553. * @param async A pointer to a valid and ready to be signaled
  554. * struct k_poll_signal. (Note: if NULL this function will not
  555. * notify the end of the transaction, and whether it went
  556. * successfully or not).
  557. *
  558. * @retval frames Positive number of frames received in slave mode.
  559. * @retval 0 If successful in master mode.
  560. * @retval -errno Negative errno code on failure.
  561. */
  562. static inline int spi_transceive_async(const struct device *dev,
  563. const struct spi_config *config,
  564. const struct spi_buf_set *tx_bufs,
  565. const struct spi_buf_set *rx_bufs,
  566. struct k_poll_signal *async)
  567. {
  568. const struct spi_driver_api *api =
  569. (const struct spi_driver_api *)dev->api;
  570. return api->transceive_async(dev, config, tx_bufs, rx_bufs, async);
  571. }
  572. /**
  573. * @brief Read the specified amount of data from the SPI driver.
  574. *
  575. * @note This function is asynchronous.
  576. *
  577. * @note This function is an helper function calling spi_transceive_async.
  578. *
  579. * @note This function is available only if @kconfig{CONFIG_SPI_ASYNC}
  580. * is selected.
  581. *
  582. * @param dev Pointer to the device structure for the driver instance
  583. * @param config Pointer to a valid spi_config structure instance.
  584. * Pointer-comparison may be used to detect changes from
  585. * previous operations.
  586. * @param rx_bufs Buffer array where data to be read will be written to.
  587. * @param async A pointer to a valid and ready to be signaled
  588. * struct k_poll_signal. (Note: if NULL this function will not
  589. * notify the end of the transaction, and whether it went
  590. * successfully or not).
  591. *
  592. * @retval 0 If successful
  593. * @retval -errno Negative errno code on failure.
  594. */
  595. static inline int spi_read_async(const struct device *dev,
  596. const struct spi_config *config,
  597. const struct spi_buf_set *rx_bufs,
  598. struct k_poll_signal *async)
  599. {
  600. return spi_transceive_async(dev, config, NULL, rx_bufs, async);
  601. }
  602. /**
  603. * @brief Write the specified amount of data from the SPI driver.
  604. *
  605. * @note This function is asynchronous.
  606. *
  607. * @note This function is an helper function calling spi_transceive_async.
  608. *
  609. * @note This function is available only if @kconfig{CONFIG_SPI_ASYNC}
  610. * is selected.
  611. *
  612. * @param dev Pointer to the device structure for the driver instance
  613. * @param config Pointer to a valid spi_config structure instance.
  614. * Pointer-comparison may be used to detect changes from
  615. * previous operations.
  616. * @param tx_bufs Buffer array where data to be sent originates from.
  617. * @param async A pointer to a valid and ready to be signaled
  618. * struct k_poll_signal. (Note: if NULL this function will not
  619. * notify the end of the transaction, and whether it went
  620. * successfully or not).
  621. *
  622. * @retval 0 If successful.
  623. * @retval -errno Negative errno code on failure.
  624. */
  625. static inline int spi_write_async(const struct device *dev,
  626. const struct spi_config *config,
  627. const struct spi_buf_set *tx_bufs,
  628. struct k_poll_signal *async)
  629. {
  630. return spi_transceive_async(dev, config, tx_bufs, NULL, async);
  631. }
  632. #endif /* CONFIG_SPI_ASYNC */
  633. /**
  634. * @brief Release the SPI device locked on by the current config
  635. *
  636. * Note: This synchronous function is used to release the lock on the SPI
  637. * device that was kept if, and if only, given config parameter was
  638. * the last one to be used (in any of the above functions) and if
  639. * it has the SPI_LOCK_ON bit set into its operation bits field.
  640. * This can be used if the caller needs to keep its hand on the SPI
  641. * device for consecutive transactions.
  642. *
  643. * @param dev Pointer to the device structure for the driver instance
  644. * @param config Pointer to a valid spi_config structure instance.
  645. * Pointer-comparison may be used to detect changes from
  646. * previous operations.
  647. *
  648. * @retval 0 If successful.
  649. * @retval -errno Negative errno code on failure.
  650. */
  651. __syscall int spi_release(const struct device *dev,
  652. const struct spi_config *config);
  653. static inline int z_impl_spi_release(const struct device *dev,
  654. const struct spi_config *config)
  655. {
  656. const struct spi_driver_api *api =
  657. (const struct spi_driver_api *)dev->api;
  658. return api->release(dev, config);
  659. }
  660. /**
  661. * @brief Release the SPI device specified in @p spi_dt_spec.
  662. *
  663. * This is equivalent to:
  664. *
  665. * spi_release(spec->bus, &spec->config);
  666. *
  667. * @param spec SPI specification from devicetree
  668. *
  669. * @return a value from spi_release().
  670. */
  671. static inline int spi_release_dt(const struct spi_dt_spec *spec)
  672. {
  673. return spi_release(spec->bus, &spec->config);
  674. }
  675. #ifdef __cplusplus
  676. }
  677. #endif
  678. /**
  679. * @}
  680. */
  681. #include <syscalls/spi.h>
  682. #endif /* ZEPHYR_INCLUDE_DRIVERS_SPI_H_ */