blob: 1539d7c90ddd320ecf06036afc79b2da7878eac7 [file] [log] [blame]
/*
* Copyright (c) 2018, Nordic Semiconductor ASA
*
* SPDX-License-Identifier: Apache-2.0
*/
#include <zephyr/drivers/i2c.h>
#include <zephyr/drivers/pinctrl.h>
#include <zephyr/dt-bindings/i2c/i2c.h>
#include <zephyr/logging/log.h>
#include <zephyr/pm/device_runtime.h>
#include "i2c_nrfx_twim_common.h"
LOG_MODULE_DECLARE(i2c_nrfx_twim);
int i2c_nrfx_twim_recover_bus(const struct device *dev)
{
const struct i2c_nrfx_twim_common_config *config = dev->config;
enum pm_device_state state;
uint32_t scl_pin;
uint32_t sda_pin;
nrfx_err_t err;
scl_pin = nrf_twim_scl_pin_get(config->twim.p_twim);
sda_pin = nrf_twim_sda_pin_get(config->twim.p_twim);
/* disable peripheral if active (required to release SCL/SDA lines) */
(void)pm_device_state_get(dev, &state);
if (state == PM_DEVICE_STATE_ACTIVE) {
nrfx_twim_disable(&config->twim);
}
err = nrfx_twim_bus_recover(scl_pin, sda_pin);
/* restore peripheral if it was active before */
if (state == PM_DEVICE_STATE_ACTIVE) {
(void)pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT);
nrfx_twim_enable(&config->twim);
}
return (err == NRFX_SUCCESS ? 0 : -EBUSY);
}
int i2c_nrfx_twim_configure(const struct device *dev, uint32_t i2c_config)
{
const struct i2c_nrfx_twim_common_config *config = dev->config;
if (I2C_ADDR_10_BITS & i2c_config) {
return -EINVAL;
}
switch (I2C_SPEED_GET(i2c_config)) {
case I2C_SPEED_STANDARD:
nrf_twim_frequency_set(config->twim.p_twim, NRF_TWIM_FREQ_100K);
break;
case I2C_SPEED_FAST:
nrf_twim_frequency_set(config->twim.p_twim, NRF_TWIM_FREQ_400K);
break;
#if NRF_TWIM_HAS_1000_KHZ_FREQ
case I2C_SPEED_FAST_PLUS:
nrf_twim_frequency_set(config->twim.p_twim, NRF_TWIM_FREQ_1000K);
break;
#endif
default:
LOG_ERR("unsupported speed");
return -EINVAL;
}
return 0;
}
int i2c_nrfx_twim_msg_transfer(const struct device *dev, uint8_t flags, uint8_t *buf,
size_t buf_len, uint16_t i2c_addr)
{
const struct i2c_nrfx_twim_common_config *config = dev->config;
nrfx_twim_xfer_desc_t cur_xfer = {
.address = i2c_addr,
.type = (flags & I2C_MSG_READ) ? NRFX_TWIM_XFER_RX : NRFX_TWIM_XFER_TX,
.p_primary_buf = buf,
.primary_length = buf_len,
};
nrfx_err_t res;
int ret = 0;
if (buf_len > config->max_transfer_size) {
LOG_ERR("Trying to transfer more than the maximum size "
"for this device: %d > %d",
buf_len, config->max_transfer_size);
return -ENOSPC;
}
res = nrfx_twim_xfer(&config->twim, &cur_xfer,
(flags & I2C_MSG_STOP) ? 0 : NRFX_TWIM_FLAG_TX_NO_STOP);
if (res != NRFX_SUCCESS) {
if (res == NRFX_ERROR_BUSY) {
ret = -EBUSY;
} else {
ret = -EIO;
}
}
return ret;
}
int twim_nrfx_pm_action(const struct device *dev, enum pm_device_action action)
{
const struct i2c_nrfx_twim_common_config *config = dev->config;
switch (action) {
case PM_DEVICE_ACTION_RESUME:
(void)pinctrl_apply_state(config->pcfg, PINCTRL_STATE_DEFAULT);
nrfx_twim_enable(&config->twim);
break;
case PM_DEVICE_ACTION_SUSPEND:
nrfx_twim_disable(&config->twim);
(void)pinctrl_apply_state(config->pcfg, PINCTRL_STATE_SLEEP);
break;
default:
return -ENOTSUP;
}
return 0;
}
int i2c_nrfx_twim_common_init(const struct device *dev)
{
const struct i2c_nrfx_twim_common_config *config = dev->config;
config->irq_connect();
(void)pinctrl_apply_state(config->pcfg, PINCTRL_STATE_SLEEP);
if (nrfx_twim_init(&config->twim, &config->twim_config, config->event_handler,
(void *)dev) != NRFX_SUCCESS) {
LOG_ERR("Failed to initialize device: %s", dev->name);
return -EIO;
}
return pm_device_driver_init(dev, twim_nrfx_pm_action);
}