123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245 |
- /**
- * @file
- *
- * @brief Public APIs for the I2C drivers.
- */
- /*
- * Copyright (c) 2015 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #ifndef ZEPHYR_INCLUDE_DRIVERS_I2C_H_
- #define ZEPHYR_INCLUDE_DRIVERS_I2C_H_
- /**
- * @brief I2C Interface
- * @defgroup i2c_interface I2C Interface
- * @ingroup io_interfaces
- * @{
- */
- #include <zephyr/types.h>
- #include <device.h>
- #ifdef __cplusplus
- extern "C" {
- #endif
- /*
- * The following #defines are used to configure the I2C controller.
- */
- /** I2C Standard Speed: 100 kHz */
- #define I2C_SPEED_STANDARD (0x1U)
- /** I2C Fast Speed: 400 kHz */
- #define I2C_SPEED_FAST (0x2U)
- /** I2C Fast Plus Speed: 1 MHz */
- #define I2C_SPEED_FAST_PLUS (0x3U)
- /** I2C High Speed: 3.4 MHz */
- #define I2C_SPEED_HIGH (0x4U)
- /** I2C Ultra Fast Speed: 5 MHz */
- #define I2C_SPEED_ULTRA (0x5U)
- #define I2C_SPEED_SHIFT (1U)
- #define I2C_SPEED_SET(speed) (((speed) << I2C_SPEED_SHIFT) \
- & I2C_SPEED_MASK)
- #define I2C_SPEED_MASK (0x7U << I2C_SPEED_SHIFT) /* 3 bits */
- #define I2C_SPEED_GET(cfg) (((cfg) & I2C_SPEED_MASK) \
- >> I2C_SPEED_SHIFT)
- /** Use 10-bit addressing. DEPRECATED - Use I2C_MSG_ADDR_10_BITS instead. */
- #define I2C_ADDR_10_BITS BIT(0)
- /** Controller to act as Master. */
- #define I2C_MODE_MASTER BIT(4)
- /**
- * @brief Complete I2C DT information
- *
- * @param bus is the I2C bus
- * @param addr is the slave address
- */
- struct i2c_dt_spec {
- const struct device *bus;
- uint16_t addr;
- };
- /**
- * @brief Structure initializer for i2c_dt_spec from devicetree
- *
- * This helper macro expands to a static initializer for a <tt>struct
- * i2c_dt_spec</tt> by reading the relevant bus and address data from
- * the devicetree.
- *
- * @param node_id Devicetree node identifier for the I2C device whose
- * struct i2c_dt_spec to create an initializer for
- */
- #define I2C_DT_SPEC_GET(node_id) \
- { \
- .bus = DEVICE_DT_GET(DT_BUS(node_id)), \
- .addr = DT_REG_ADDR(node_id) \
- }
- /**
- * @brief Structure initializer for i2c_dt_spec from devicetree instance
- *
- * This is equivalent to
- * <tt>I2C_DT_SPEC_GET(DT_DRV_INST(inst))</tt>.
- *
- * @param inst Devicetree instance number
- */
- #define I2C_DT_SPEC_INST_GET(inst) \
- I2C_DT_SPEC_GET(DT_DRV_INST(inst))
- /*
- * I2C_MSG_* are I2C Message flags.
- */
- /** Write message to I2C bus. */
- #define I2C_MSG_WRITE (0U << 0U)
- /** Read message from I2C bus. */
- #define I2C_MSG_READ BIT(0)
- /** @cond INTERNAL_HIDDEN */
- #define I2C_MSG_RW_MASK BIT(0)
- /** @endcond */
- /** Send STOP after this message. */
- #define I2C_MSG_STOP BIT(1)
- /** RESTART I2C transaction for this message.
- *
- * @note Not all I2C drivers have or require explicit support for this
- * feature. Some drivers require this be present on a read message
- * that follows a write, or vice-versa. Some drivers will merge
- * adjacent fragments into a single transaction using this flag; some
- * will not. */
- #define I2C_MSG_RESTART BIT(2)
- /** Use 10-bit addressing for this message.
- *
- * @note Not all SoC I2C implementations support this feature. */
- #define I2C_MSG_ADDR_10_BITS BIT(3)
- /**
- * @brief One I2C Message.
- *
- * This defines one I2C message to transact on the I2C bus.
- *
- * @note Some of the configurations supported by this API may not be
- * supported by specific SoC I2C hardware implementations, in
- * particular features related to bus transactions intended to read or
- * write data from different buffers within a single transaction.
- * Invocations of i2c_transfer() may not indicate an error when an
- * unsupported configuration is encountered. In some cases drivers
- * will generate separate transactions for each message fragment, with
- * or without presence of @ref I2C_MSG_RESTART in #flags.
- */
- struct i2c_msg {
- /** Data buffer in bytes */
- uint8_t *buf;
- /** Length of buffer in bytes */
- uint32_t len;
- /** Flags for this message */
- uint8_t flags;
- };
- /**
- * @cond INTERNAL_HIDDEN
- *
- * These are for internal use only, so skip these in
- * public documentation.
- */
- struct i2c_slave_config;
- typedef int (*i2c_api_configure_t)(const struct device *dev,
- uint32_t dev_config);
- typedef int (*i2c_api_full_io_t)(const struct device *dev,
- struct i2c_msg *msgs,
- uint8_t num_msgs,
- uint16_t addr);
- typedef int (*i2c_api_slave_register_t)(const struct device *dev,
- struct i2c_slave_config *cfg);
- typedef int (*i2c_api_slave_unregister_t)(const struct device *dev,
- struct i2c_slave_config *cfg);
- typedef int (*i2c_api_recover_bus_t)(const struct device *dev);
- #ifdef CONFIG_I2C_ASYNC
- typedef int (*i2c_async_func_t)(void *cb_data, struct i2c_msg *msgs,
- uint8_t num_msgs, bool is_err);
- #ifdef CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER
- #define I2C_MSG_ASYNC_RX_MAX_BUF (12)
- #define I2C_MSG_ASYNC_TX_MAX_BUF (4)
- #endif
- /* i2c message queue item used in async mode */
- struct i2c_msg_async {
- struct i2c_msg msg[2];
- uint8_t num_msgs;
- uint16_t addr;
- i2c_async_func_t async_func;
- void *cb_data;
- #ifdef CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER
- uint8_t rx_buf[I2C_MSG_ASYNC_RX_MAX_BUF];
- uint8_t tx_buf[I2C_MSG_ASYNC_TX_MAX_BUF];
- #endif
- };
- typedef int (*i2c_api_full_io_async_t)(const struct device *dev,
- struct i2c_msg_async *msg_async);
- #endif
- __subsystem struct i2c_driver_api {
- i2c_api_configure_t configure;
- i2c_api_full_io_t transfer;
- #ifdef CONFIG_I2C_ASYNC
- i2c_api_full_io_async_t transfer_async;
- #endif
- i2c_api_slave_register_t slave_register;
- i2c_api_slave_unregister_t slave_unregister;
- i2c_api_recover_bus_t recover_bus;
- };
- typedef int (*i2c_slave_api_register_t)(const struct device *dev);
- typedef int (*i2c_slave_api_unregister_t)(const struct device *dev);
- struct i2c_slave_driver_api {
- i2c_slave_api_register_t driver_register;
- i2c_slave_api_unregister_t driver_unregister;
- };
- /**
- * @endcond
- */
- /** Slave device responds to 10-bit addressing. */
- #define I2C_SLAVE_FLAGS_ADDR_10_BITS BIT(0)
- /** @brief Function called when a write to the device is initiated.
- *
- * This function is invoked by the controller when the bus completes a
- * start condition for a write operation to the address associated
- * with a particular device.
- *
- * A success return shall cause the controller to ACK the next byte
- * received. An error return shall cause the controller to NACK the
- * next byte received.
- *
- * @param config the configuration structure associated with the
- * device to which the operation is addressed.
- *
- * @return 0 if the write is accepted, or a negative error code.
- */
- typedef int (*i2c_slave_write_requested_cb_t)(
- struct i2c_slave_config *config);
- /** @brief Function called when a write to the device is continued.
- *
- * This function is invoked by the controller when it completes
- * reception of a byte of data in an ongoing write operation to the
- * device.
- *
- * A success return shall cause the controller to ACK the next byte
- * received. An error return shall cause the controller to NACK the
- * next byte received.
- *
- * @param config the configuration structure associated with the
- * device to which the operation is addressed.
- *
- * @param val the byte received by the controller.
- *
- * @return 0 if more data can be accepted, or a negative error
- * code.
- */
- typedef int (*i2c_slave_write_received_cb_t)(
- struct i2c_slave_config *config, uint8_t val);
- /** @brief Function called when a read from the device is initiated.
- *
- * This function is invoked by the controller when the bus completes a
- * start condition for a read operation from the address associated
- * with a particular device.
- *
- * The value returned in @p *val will be transmitted. A success
- * return shall cause the controller to react to additional read
- * operations. An error return shall cause the controller to ignore
- * bus operations until a new start condition is received.
- *
- * @param config the configuration structure associated with the
- * device to which the operation is addressed.
- *
- * @param val pointer to storage for the first byte of data to return
- * for the read request.
- *
- * @return 0 if more data can be requested, or a negative error code.
- */
- typedef int (*i2c_slave_read_requested_cb_t)(
- struct i2c_slave_config *config, uint8_t *val);
- /** @brief Function called when a read from the device is continued.
- *
- * This function is invoked by the controller when the bus is ready to
- * provide additional data for a read operation from the address
- * associated with the device device.
- *
- * The value returned in @p *val will be transmitted. A success
- * return shall cause the controller to react to additional read
- * operations. An error return shall cause the controller to ignore
- * bus operations until a new start condition is received.
- *
- * @param config the configuration structure associated with the
- * device to which the operation is addressed.
- *
- * @param val pointer to storage for the next byte of data to return
- * for the read request.
- *
- * @return 0 if data has been provided, or a negative error code.
- */
- typedef int (*i2c_slave_read_processed_cb_t)(
- struct i2c_slave_config *config, uint8_t *val);
- /** @brief Function called when a stop condition is observed after a
- * start condition addressed to a particular device.
- *
- * This function is invoked by the controller when the bus is ready to
- * provide additional data for a read operation from the address
- * associated with the device device. After the function returns the
- * controller shall enter a state where it is ready to react to new
- * start conditions.
- *
- * @param config the configuration structure associated with the
- * device to which the operation is addressed.
- *
- * @return Ignored.
- */
- typedef int (*i2c_slave_stop_cb_t)(struct i2c_slave_config *config);
- /** @brief Structure providing callbacks to be implemented for devices
- * that supports the I2C slave API.
- *
- * This structure may be shared by multiple devices that implement the
- * same API at different addresses on the bus.
- */
- struct i2c_slave_callbacks {
- i2c_slave_write_requested_cb_t write_requested;
- i2c_slave_read_requested_cb_t read_requested;
- i2c_slave_write_received_cb_t write_received;
- i2c_slave_read_processed_cb_t read_processed;
- i2c_slave_stop_cb_t stop;
- };
- /** @brief Structure describing a device that supports the I2C
- * slave API.
- *
- * Instances of this are passed to the i2c_slave_register() and
- * i2c_slave_unregister() functions to indicate addition and removal
- * of a slave device, respective.
- *
- * Fields other than @c node must be initialized by the module that
- * implements the device behavior prior to passing the object
- * reference to i2c_slave_register().
- */
- struct i2c_slave_config {
- /** Private, do not modify */
- sys_snode_t node;
- /** Flags for the slave device defined by I2C_SLAVE_FLAGS_* constants */
- uint8_t flags;
- /** Address for this slave device */
- uint16_t address;
- /** Callback functions */
- const struct i2c_slave_callbacks *callbacks;
- };
- /**
- * @brief Configure operation of a host controller.
- *
- * @param dev Pointer to the device structure for the driver instance.
- * @param dev_config Bit-packed 32-bit value to the device runtime configuration
- * for the I2C controller.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error, failed to configure device.
- */
- __syscall int i2c_configure(const struct device *dev, uint32_t dev_config);
- static inline int z_impl_i2c_configure(const struct device *dev,
- uint32_t dev_config)
- {
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- return api->configure(dev, dev_config);
- }
- /**
- * @brief Perform data transfer to another I2C device in master mode.
- *
- * This routine provides a generic interface to perform data transfer
- * to another I2C device synchronously. Use i2c_read()/i2c_write()
- * for simple read or write.
- *
- * The array of message @a msgs must not be NULL. The number of
- * message @a num_msgs may be zero,in which case no transfer occurs.
- *
- * @note Not all scatter/gather transactions can be supported by all
- * drivers. As an example, a gather write (multiple consecutive
- * `i2c_msg` buffers all configured for `I2C_MSG_WRITE`) may be packed
- * into a single transaction by some drivers, but others may emit each
- * fragment as a distinct write transaction, which will not produce
- * the same behavior. See the documentation of `struct i2c_msg` for
- * limitations on support for multi-message bus transactions.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param msgs Array of messages to transfer.
- * @param num_msgs Number of messages to transfer.
- * @param addr Address of the I2C target device.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- __syscall int i2c_transfer(const struct device *dev,
- struct i2c_msg *msgs, uint8_t num_msgs,
- uint16_t addr);
- static inline int z_impl_i2c_transfer(const struct device *dev,
- struct i2c_msg *msgs, uint8_t num_msgs,
- uint16_t addr)
- {
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- return api->transfer(dev, msgs, num_msgs, addr);
- }
- /**
- * @brief Perform data transfer to another I2C device in master mode.
- *
- * This is equivalent to:
- *
- * i2c_transfer(spec->bus, msgs, num_msgs, spec->addr);
- *
- * @param spec I2C specification from devicetree.
- * @param msgs Array of messages to transfer.
- * @param num_msgs Number of messages to transfer.
- *
- * @return a value from i2c_transfer()
- */
- static inline int i2c_transfer_dt(const struct i2c_dt_spec *spec,
- struct i2c_msg *msgs, uint8_t num_msgs)
- {
- return i2c_transfer(spec->bus, msgs, num_msgs, spec->addr);
- }
- /**
- * @brief Recover the I2C bus
- *
- * Attempt to recover the I2C bus.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @retval 0 If successful
- * @retval -EBUSY If bus is not clear after recovery attempt.
- * @retval -EIO General input / output error.
- * @retval -ENOSYS If bus recovery is not implemented
- */
- __syscall int i2c_recover_bus(const struct device *dev);
- static inline int z_impl_i2c_recover_bus(const struct device *dev)
- {
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- if (api->recover_bus == NULL) {
- return -ENOSYS;
- }
- return api->recover_bus(dev);
- }
- /**
- * @brief Registers the provided config as Slave device of a controller.
- *
- * Enable I2C slave mode for the 'dev' I2C bus driver using the provided
- * 'config' struct containing the functions and parameters to send bus
- * events. The I2C slave will be registered at the address provided as 'address'
- * struct member. Addressing mode - 7 or 10 bit - depends on the 'flags'
- * struct member. Any I2C bus events related to the slave mode will be passed
- * onto I2C slave device driver via a set of callback functions provided in
- * the 'callbacks' struct member.
- *
- * Most of the existing hardware allows simultaneous support for master
- * and slave mode. This is however not guaranteed.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in slave mode.
- * @param cfg Config struct with functions and parameters used by the I2C driver
- * to send bus events
- *
- * @retval 0 Is successful
- * @retval -EINVAL If parameters are invalid
- * @retval -EIO General input / output error.
- * @retval -ENOSYS If slave mode is not implemented
- */
- static inline int i2c_slave_register(const struct device *dev,
- struct i2c_slave_config *cfg)
- {
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- if (api->slave_register == NULL) {
- return -ENOSYS;
- }
- return api->slave_register(dev, cfg);
- }
- /**
- * @brief Unregisters the provided config as Slave device
- *
- * This routine disables I2C slave mode for the 'dev' I2C bus driver using
- * the provided 'config' struct containing the functions and parameters
- * to send bus events.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in slave mode.
- * @param cfg Config struct with functions and parameters used by the I2C driver
- * to send bus events
- *
- * @retval 0 Is successful
- * @retval -EINVAL If parameters are invalid
- * @retval -ENOSYS If slave mode is not implemented
- */
- static inline int i2c_slave_unregister(const struct device *dev,
- struct i2c_slave_config *cfg)
- {
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- if (api->slave_unregister == NULL) {
- return -ENOSYS;
- }
- return api->slave_unregister(dev, cfg);
- }
- /**
- * @brief Instructs the I2C Slave device to register itself to the I2C Controller
- *
- * This routine instructs the I2C Slave device to register itself to the I2C
- * Controller via its parent controller's i2c_slave_register() API.
- *
- * @param dev Pointer to the device structure for the I2C slave
- * device (not itself an I2C controller).
- *
- * @retval 0 Is successful
- * @retval -EINVAL If parameters are invalid
- * @retval -EIO General input / output error.
- */
- __syscall int i2c_slave_driver_register(const struct device *dev);
- static inline int z_impl_i2c_slave_driver_register(const struct device *dev)
- {
- const struct i2c_slave_driver_api *api =
- (const struct i2c_slave_driver_api *)dev->api;
- return api->driver_register(dev);
- }
- /**
- * @brief Instructs the I2C Slave device to unregister itself from the I2C
- * Controller
- *
- * This routine instructs the I2C Slave device to unregister itself from the I2C
- * Controller via its parent controller's i2c_slave_register() API.
- *
- * @param dev Pointer to the device structure for the I2C slave
- * device (not itself an I2C controller).
- *
- * @retval 0 Is successful
- * @retval -EINVAL If parameters are invalid
- */
- __syscall int i2c_slave_driver_unregister(const struct device *dev);
- static inline int z_impl_i2c_slave_driver_unregister(const struct device *dev)
- {
- const struct i2c_slave_driver_api *api =
- (const struct i2c_slave_driver_api *)dev->api;
- return api->driver_unregister(dev);
- }
- /*
- * Derived i2c APIs -- all implemented in terms of i2c_transfer()
- */
- /**
- * @brief Write a set amount of data to an I2C device.
- *
- * This routine writes a set amount of data synchronously.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param buf Memory pool from which the data is transferred.
- * @param num_bytes Number of bytes to write.
- * @param addr Address to the target I2C device for writing.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_write(const struct device *dev, const uint8_t *buf,
- uint32_t num_bytes, uint16_t addr)
- {
- struct i2c_msg msg;
- #if defined(CONFIG_I2C_ASYNC) && defined(CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER)
- if ((num_bytes > I2C_MSG_ASYNC_TX_MAX_BUF) || (!num_bytes)) {
- printk("invalid num_bytes:%d max:%d\n",
- num_bytes, I2C_MSG_ASYNC_TX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg.buf = (uint8_t *)buf;
- msg.len = num_bytes;
- msg.flags = I2C_MSG_WRITE | I2C_MSG_STOP;
- return i2c_transfer(dev, &msg, 1, addr);
- }
- /**
- * @brief Write a set amount of data to an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_write(spec->bus, buf, num_bytes, spec->addr);
- *
- * @param spec I2C specification from devicetree.
- * @param buf Memory pool from which the data is transferred.
- * @param num_bytes Number of bytes to write.
- *
- * @return a value from i2c_write()
- */
- static inline int i2c_write_dt(const struct i2c_dt_spec *spec,
- const uint8_t *buf, uint32_t num_bytes)
- {
- return i2c_write(spec->bus, buf, num_bytes, spec->addr);
- }
- /**
- * @brief Read a set amount of data from an I2C device.
- *
- * This routine reads a set amount of data synchronously.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param buf Memory pool that stores the retrieved data.
- * @param num_bytes Number of bytes to read.
- * @param addr Address of the I2C device being read.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_read(const struct device *dev, uint8_t *buf,
- uint32_t num_bytes, uint16_t addr)
- {
- struct i2c_msg msg;
- #if defined(CONFIG_I2C_ASYNC) && defined(CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER)
- if ((num_bytes > I2C_MSG_ASYNC_RX_MAX_BUF) || (!num_bytes)) {
- printk("invalid num_bytes:%d max:%d\n",
- num_bytes, I2C_MSG_ASYNC_RX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg.buf = buf;
- msg.len = num_bytes;
- msg.flags = I2C_MSG_READ | I2C_MSG_STOP;
- return i2c_transfer(dev, &msg, 1, addr);
- }
- /**
- * @brief Read a set amount of data from an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_read(spec->bus, buf, num_bytes, spec->addr);
- *
- * @param spec I2C specification from devicetree.
- * @param buf Memory pool that stores the retrieved data.
- * @param num_bytes Number of bytes to read.
- *
- * @return a value from i2c_read()
- */
- static inline int i2c_read_dt(const struct i2c_dt_spec *spec,
- uint8_t *buf, uint32_t num_bytes)
- {
- return i2c_read(spec->bus, buf, num_bytes, spec->addr);
- }
- /**
- * @brief Write then read data from an I2C device.
- *
- * This supports the common operation "this is what I want", "now give
- * it to me" transaction pair through a combined write-then-read bus
- * transaction.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param addr Address of the I2C device
- * @param write_buf Pointer to the data to be written
- * @param num_write Number of bytes to write
- * @param read_buf Pointer to storage for read data
- * @param num_read Number of bytes to read
- *
- * @retval 0 if successful
- * @retval negative on error.
- */
- static inline int i2c_write_read(const struct device *dev, uint16_t addr,
- const void *write_buf, size_t num_write,
- void *read_buf, size_t num_read)
- {
- struct i2c_msg msg[2];
- #if defined(CONFIG_I2C_ASYNC) && defined(CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER)
- if ((num_write > I2C_MSG_ASYNC_TX_MAX_BUF) || (!num_write)) {
- printk("invalid num_write:%d max:%d\n",
- num_write, I2C_MSG_ASYNC_TX_MAX_BUF);
- return -EINVAL;
- }
- if ((num_read > I2C_MSG_ASYNC_RX_MAX_BUF) || (!num_read)) {
- printk("invalid num_read:%d max:%d\n",
- num_read, I2C_MSG_ASYNC_RX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg[0].buf = (uint8_t *)write_buf;
- msg[0].len = num_write;
- msg[0].flags = I2C_MSG_WRITE;
- msg[1].buf = (uint8_t *)read_buf;
- msg[1].len = num_read;
- msg[1].flags = I2C_MSG_RESTART | I2C_MSG_READ | I2C_MSG_STOP;
- return i2c_transfer(dev, msg, 2, addr);
- }
- #ifdef CONFIG_I2C_ASYNC
- /*
- * Derived i2c APIs -- all implemented in terms of i2c_transfer()
- */
- /**
- * @brief Write a set amount of data to an I2C device in async mode.
- *
- * This routine writes a set amount of data synchronously.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param buf Memory pool from which the data is transferred.
- * @param num_bytes Number of bytes to write.
- * @param addr Address to the target I2C device for writing.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- *
- * @note In case of not define CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER, the parameter @a buf
- * shall be a global address.
- */
- static inline int i2c_write_async(const struct device *dev, const uint8_t *buf,
- uint32_t num_bytes, uint16_t addr,
- i2c_async_func_t async_func, void *cb_data)
- {
- struct i2c_msg_async msg_async = {0};
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- #ifdef CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER
- if ((num_bytes > I2C_MSG_ASYNC_TX_MAX_BUF) || (!num_bytes)) {
- printk("invalid num_bytes:%d max:%d\n",
- num_bytes, I2C_MSG_ASYNC_TX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg_async.msg[0].buf = (uint8_t *)buf;
- msg_async.msg[0].len = num_bytes;
- msg_async.msg[0].flags = I2C_MSG_WRITE | I2C_MSG_STOP;
- msg_async.addr = addr;
- msg_async.num_msgs = 1;
- msg_async.async_func = async_func;
- msg_async.cb_data = cb_data;
- return api->transfer_async(dev, &msg_async);
- }
- /**
- * @brief Read a set amount of data from an I2C device in async mode.
- *
- * This routine reads a set amount of data synchronously.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param buf Memory pool that stores the retrieved data.
- * @param num_bytes Number of bytes to read.
- * @param addr Address of the I2C device being read.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- *
- * @note In case of not define CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER, the parameter @a buf
- * shall be a global address.
- */
- static inline int i2c_read_async(const struct device *dev, uint8_t *buf,
- uint32_t num_bytes, uint16_t addr,
- i2c_async_func_t async_func, void *cb_data)
- {
- struct i2c_msg_async msg_async = {0};
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- #ifdef CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER
- if ((num_bytes > I2C_MSG_ASYNC_RX_MAX_BUF) || (!num_bytes)) {
- printk("invalid num_bytes:%d max:%d\n",
- num_bytes, I2C_MSG_ASYNC_RX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg_async.msg[0].buf = buf;
- msg_async.msg[0].len = num_bytes;
- msg_async.msg[0].flags = I2C_MSG_READ | I2C_MSG_STOP;
- msg_async.addr = addr;
- msg_async.num_msgs = 1;
- msg_async.async_func = async_func;
- msg_async.cb_data = cb_data;
- return api->transfer_async(dev, &msg_async);
- }
- /**
- * @brief Write then read data from an I2C device in async mode.
- *
- * This supports the common operation "this is what I want", "now give
- * it to me" transaction pair through a combined write-then-read bus
- * transaction.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param addr Address of the I2C device
- * @param write_buf Pointer to the data to be written
- * @param num_write Number of bytes to write
- * @param read_buf Pointer to storage for read data
- * @param num_read Number of bytes to read
- *
- * @retval 0 if successful
- * @retval negative on error.
- *
- * @note In case of not define CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER, the parameter
- * @a write_buf and read_buf shall be a global address.
- */
- static inline int i2c_write_read_async(const struct device *dev, uint16_t addr,
- const void *write_buf, size_t num_write,
- void *read_buf, size_t num_read,
- i2c_async_func_t async_func, void *cb_data)
- {
- struct i2c_msg_async msg_async = {0};
- const struct i2c_driver_api *api =
- (const struct i2c_driver_api *)dev->api;
- #ifdef CONFIG_I2C_ASYNC_MSG_INTERNAL_BUFFER
- if ((num_write > I2C_MSG_ASYNC_TX_MAX_BUF) || (!num_write)) {
- printk("invalid num_write:%d max:%d\n",
- num_write, I2C_MSG_ASYNC_TX_MAX_BUF);
- return -EINVAL;
- }
- if ((num_read > I2C_MSG_ASYNC_RX_MAX_BUF) || (!num_read)) {
- printk("invalid num_read:%d max:%d\n",
- num_read, I2C_MSG_ASYNC_RX_MAX_BUF);
- return -EINVAL;
- }
- #endif
- msg_async.msg[0].buf = (uint8_t *)write_buf;
- msg_async.msg[0].len = num_write;
- msg_async.msg[0].flags = I2C_MSG_WRITE;
- msg_async.msg[1].buf = (uint8_t *)read_buf;
- msg_async.msg[1].len = num_read;
- msg_async.msg[1].flags = I2C_MSG_RESTART | I2C_MSG_READ | I2C_MSG_STOP;
- msg_async.addr = addr;
- msg_async.num_msgs = 2;
- msg_async.async_func = async_func;
- msg_async.cb_data = cb_data;
- return api->transfer_async(dev, &msg_async);
- }
- #endif
- /**
- * @brief Write then read data from an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_write_read(spec->bus, spec->addr,
- * write_buf, num_write,
- * read_buf, num_read);
- *
- * @param spec I2C specification from devicetree.
- * @param write_buf Pointer to the data to be written
- * @param num_write Number of bytes to write
- * @param read_buf Pointer to storage for read data
- * @param num_read Number of bytes to read
- *
- * @return a value from i2c_write_read()
- */
- static inline int i2c_write_read_dt(const struct i2c_dt_spec *spec,
- const void *write_buf, size_t num_write,
- void *read_buf, size_t num_read)
- {
- return i2c_write_read(spec->bus, spec->addr,
- write_buf, num_write,
- read_buf, num_read);
- }
- /**
- * @brief Read multiple bytes from an internal address of an I2C device.
- *
- * This routine reads multiple bytes from an internal address of an
- * I2C device synchronously.
- *
- * Instances of this may be replaced by i2c_write_read().
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param dev_addr Address of the I2C device for reading.
- * @param start_addr Internal address from which the data is being read.
- * @param buf Memory pool that stores the retrieved data.
- * @param num_bytes Number of bytes being read.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_burst_read(const struct device *dev,
- uint16_t dev_addr,
- uint8_t start_addr,
- uint8_t *buf,
- uint32_t num_bytes)
- {
- return i2c_write_read(dev, dev_addr,
- &start_addr, sizeof(start_addr),
- buf, num_bytes);
- }
- /**
- * @brief Read multiple bytes from an internal address of an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_burst_read(spec->bus, spec->addr, start_addr, buf, num_bytes);
- *
- * @param spec I2C specification from devicetree.
- * @param start_addr Internal address from which the data is being read.
- * @param buf Memory pool that stores the retrieved data.
- * @param num_bytes Number of bytes to read.
- *
- * @return a value from i2c_burst_read()
- */
- static inline int i2c_burst_read_dt(const struct i2c_dt_spec *spec,
- uint8_t start_addr,
- uint8_t *buf,
- uint32_t num_bytes)
- {
- return i2c_burst_read(spec->bus, spec->addr,
- start_addr, buf, num_bytes);
- }
- /**
- * @brief Write multiple bytes to an internal address of an I2C device.
- *
- * This routine writes multiple bytes to an internal address of an
- * I2C device synchronously.
- *
- * @warning The combined write synthesized by this API may not be
- * supported on all I2C devices. Uses of this API may be made more
- * portable by replacing them with calls to i2c_write() passing a
- * buffer containing the combined address and data.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param dev_addr Address of the I2C device for writing.
- * @param start_addr Internal address to which the data is being written.
- * @param buf Memory pool from which the data is transferred.
- * @param num_bytes Number of bytes being written.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_burst_write(const struct device *dev,
- uint16_t dev_addr,
- uint8_t start_addr,
- const uint8_t *buf,
- uint32_t num_bytes)
- {
- struct i2c_msg msg[2];
- msg[0].buf = &start_addr;
- msg[0].len = 1U;
- msg[0].flags = I2C_MSG_WRITE;
- msg[1].buf = (uint8_t *)buf;
- msg[1].len = num_bytes;
- msg[1].flags = I2C_MSG_WRITE | I2C_MSG_STOP;
- return i2c_transfer(dev, msg, 2, dev_addr);
- }
- /**
- * @brief Write multiple bytes to an internal address of an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_burst_write(spec->bus, spec->addr, start_addr, buf, num_bytes);
- *
- * @param spec I2C specification from devicetree.
- * @param start_addr Internal address to which the data is being written.
- * @param buf Memory pool from which the data is transferred.
- * @param num_bytes Number of bytes being written.
- *
- * @return a value from i2c_burst_write()
- */
- static inline int i2c_burst_write_dt(const struct i2c_dt_spec *spec,
- uint8_t start_addr,
- const uint8_t *buf,
- uint32_t num_bytes)
- {
- return i2c_burst_write(spec->bus, spec->addr,
- start_addr, buf, num_bytes);
- }
- /**
- * @brief Read internal register of an I2C device.
- *
- * This routine reads the value of an 8-bit internal register of an I2C
- * device synchronously.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param dev_addr Address of the I2C device for reading.
- * @param reg_addr Address of the internal register being read.
- * @param value Memory pool that stores the retrieved register value.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_reg_read_byte(const struct device *dev,
- uint16_t dev_addr,
- uint8_t reg_addr, uint8_t *value)
- {
- return i2c_write_read(dev, dev_addr,
- ®_addr, sizeof(reg_addr),
- value, sizeof(*value));
- }
- /**
- * @brief Read internal register of an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_reg_read_byte(spec->bus, spec->addr, reg_addr, value);
- *
- * @param spec I2C specification from devicetree.
- * @param reg_addr Address of the internal register being read.
- * @param value Memory pool that stores the retrieved register value.
- *
- * @return a value from i2c_reg_read_byte()
- */
- static inline int i2c_reg_read_byte_dt(const struct i2c_dt_spec *spec,
- uint8_t reg_addr, uint8_t *value)
- {
- return i2c_reg_read_byte(spec->bus, spec->addr, reg_addr, value);
- }
- /**
- * @brief Write internal register of an I2C device.
- *
- * This routine writes a value to an 8-bit internal register of an I2C
- * device synchronously.
- *
- * @note This function internally combines the register and value into
- * a single bus transaction.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param dev_addr Address of the I2C device for writing.
- * @param reg_addr Address of the internal register being written.
- * @param value Value to be written to internal register.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_reg_write_byte(const struct device *dev,
- uint16_t dev_addr,
- uint8_t reg_addr, uint8_t value)
- {
- uint8_t tx_buf[2] = {reg_addr, value};
- return i2c_write(dev, tx_buf, 2, dev_addr);
- }
- /**
- * @brief Write internal register of an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_reg_write_byte(spec->bus, spec->addr, reg_addr, value);
- *
- * @param spec I2C specification from devicetree.
- * @param reg_addr Address of the internal register being written.
- * @param value Value to be written to internal register.
- *
- * @return a value from i2c_reg_write_byte()
- */
- static inline int i2c_reg_write_byte_dt(const struct i2c_dt_spec *spec,
- uint8_t reg_addr, uint8_t value)
- {
- return i2c_reg_write_byte(spec->bus, spec->addr, reg_addr, value);
- }
- /**
- * @brief Update internal register of an I2C device.
- *
- * This routine updates the value of a set of bits from an 8-bit internal
- * register of an I2C device synchronously.
- *
- * @note If the calculated new register value matches the value that
- * was read this function will not generate a write operation.
- *
- * @param dev Pointer to the device structure for an I2C controller
- * driver configured in master mode.
- * @param dev_addr Address of the I2C device for updating.
- * @param reg_addr Address of the internal register being updated.
- * @param mask Bitmask for updating internal register.
- * @param value Value for updating internal register.
- *
- * @retval 0 If successful.
- * @retval -EIO General input / output error.
- */
- static inline int i2c_reg_update_byte(const struct device *dev,
- uint8_t dev_addr,
- uint8_t reg_addr, uint8_t mask,
- uint8_t value)
- {
- uint8_t old_value, new_value;
- int rc;
- rc = i2c_reg_read_byte(dev, dev_addr, reg_addr, &old_value);
- if (rc != 0) {
- return rc;
- }
- new_value = (old_value & ~mask) | (value & mask);
- if (new_value == old_value) {
- return 0;
- }
- return i2c_reg_write_byte(dev, dev_addr, reg_addr, new_value);
- }
- /**
- * @brief Update internal register of an I2C device.
- *
- * This is equivalent to:
- *
- * i2c_reg_update_byte(spec->bus, spec->addr, reg_addr, mask, value);
- *
- * @param spec I2C specification from devicetree.
- * @param reg_addr Address of the internal register being updated.
- * @param mask Bitmask for updating internal register.
- * @param value Value for updating internal register.
- *
- * @return a value from i2c_reg_update_byte()
- */
- static inline int i2c_reg_update_byte_dt(const struct i2c_dt_spec *spec,
- uint8_t reg_addr, uint8_t mask,
- uint8_t value)
- {
- return i2c_reg_update_byte(spec->bus, spec->addr,
- reg_addr, mask, value);
- }
- /**
- * @brief Dump out an I2C message
- *
- * Dumps out a list of I2C messages. For any that are writes (W), the data is
- * displayed in hex.
- *
- * It looks something like this (with name "testing"):
- *
- * D: I2C msg: testing, addr=56
- * D: W len=01:
- * D: contents:
- * D: 06 |.
- * D: W len=0e:
- * D: contents:
- * D: 00 01 02 03 04 05 06 07 |........
- * D: 08 09 0a 0b 0c 0d |......
- *
- * @param name Name of this dump, displayed at the top.
- * @param msgs Array of messages to dump.
- * @param num_msgs Number of messages to dump.
- * @param addr Address of the I2C target device.
- */
- void i2c_dump_msgs(const char *name, const struct i2c_msg *msgs,
- uint8_t num_msgs, uint16_t addr);
- struct i2c_client_config {
- char *i2c_master;
- uint16_t i2c_addr;
- };
- #define I2C_DECLARE_CLIENT_CONFIG struct i2c_client_config i2c_client
- #define I2C_CLIENT(_master, _addr) \
- .i2c_client = { \
- .i2c_master = (_master), \
- .i2c_addr = (_addr), \
- }
- #define I2C_GET_MASTER(_conf) ((_conf)->i2c_client.i2c_master)
- #define I2C_GET_ADDR(_conf) ((_conf)->i2c_client.i2c_addr)
- #ifdef __cplusplus
- }
- #endif
- /**
- * @}
- */
- #include <syscalls/i2c.h>
- #endif /* ZEPHYR_INCLUDE_DRIVERS_I2C_H_ */
|