83d031c024
For systems with a true dynamic 1-Wire nature, the 1-Wire devices cannot be defined in the DeviceTree (the number of slaves in the system is not known during the build time). In other words, there are no pre-defined 1-Wire devices on the bus in DeviceTree, and thus the slave_count variable is always zero. That means the skip_rom functionality will always be called even if there are multiple devices connected to the bus. This commit allows the user to override this original behavior and avoid the skip_rom call by ignoring the slave count variable. Also, this work preserves full backward compatibility. Signed-off-by: Pavel Hübner <pavel.hubner@hardwario.com>
399 lines
9.5 KiB
C
399 lines
9.5 KiB
C
/*
|
|
* Copyright (c) 2022 Thomas Stranger
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
|
|
/**
|
|
* @brief 1-Wire network related functions.
|
|
*
|
|
* The following procedures wrap basic w1 syscalls, they should be callable
|
|
* from user mode as well as supervisor mode, therefore _ZEPHYR_SUPERVISOR__
|
|
* is not defined for this file such that inline macros do not skip
|
|
* the arch_is_user_context() check.
|
|
*/
|
|
|
|
#include <zephyr/logging/log.h>
|
|
#include <zephyr/drivers/w1.h>
|
|
|
|
LOG_MODULE_REGISTER(w1, CONFIG_W1_LOG_LEVEL);
|
|
|
|
#define W1_SEARCH_DISCREPANCY_INIT 0
|
|
#define W1_SEARCH_LAST_SLAVE 65
|
|
#define W1_SEARCH_NO_SLAVE 66
|
|
|
|
/* @brief Search bus for next slave.
|
|
*
|
|
* This function searches the next 1-Wire slave on the bus.
|
|
* It sets the found ROM and the last discrepancy in case more than one
|
|
* slave took part in the search.
|
|
* In case only one slave took part in the search, the discrepancy is set to
|
|
* W1_SEARCH_LAST_SLAVE, and in case no slave participated in the search,
|
|
* the discrepancy is set to W1_SEARCH_NO_SLAVE.
|
|
*
|
|
* The implementation is similar to suggested in the maxim application note 187.
|
|
* The master reads the first ROM bit and its complementary value of all slaves.
|
|
* Due to physical characteristics, the value received is a
|
|
* logical AND of all slaves' 1st bit. Slaves only continue to
|
|
* participate in the search procedure if the next bit the master sends matches
|
|
* their own addresses' bit. This allows the master to branch through 64-bit
|
|
* addresses in order to detect all slaves.
|
|
|
|
* The 1st bit received is stored in bit 1 of rom_inv_64, the 2nd in bit 2 and so
|
|
* on, until bit 64.
|
|
* As a result, each byte of the ROM has the correct bit order, but the received
|
|
* bytes (big-endian) stored in rom_inv_64 are in inverse byte order.
|
|
*
|
|
* Note: Filtering by families is currently not supported.
|
|
*
|
|
* @param dev Pointer to the device structure for the w1 instance.
|
|
* @param command Command to chose between normal and alarm search.
|
|
* @param family This parameter is currently not supported.
|
|
* @param last_discrepancy This must be set to W1_SEARCH_DISCREPANCY_INIT before
|
|
* the first call, it carries the search progress for
|
|
* further calls.
|
|
* @param rom_inv_64 The found ROM: It must be set to zero before first
|
|
* call and carries the last found ROM for furter calls.
|
|
* The ROM is stored in inverse byte order.
|
|
*
|
|
* @retval 0 If successful.
|
|
* @retval -errno Negative error code in case of 1-wire read/write error.
|
|
*/
|
|
static int search_slave(const struct device *dev, uint8_t command,
|
|
uint8_t family, size_t *last_discrepancy,
|
|
uint64_t *rom_inv_64)
|
|
{
|
|
int ret;
|
|
size_t next_discrepancy;
|
|
bool last_id_bit;
|
|
bool last_complement_id_bit;
|
|
|
|
ARG_UNUSED(family);
|
|
__ASSERT_NO_MSG(command == W1_CMD_SEARCH_ROM ||
|
|
command == W1_CMD_SEARCH_ALARM);
|
|
|
|
ret = w1_reset_bus(dev);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
if (ret == 0) {
|
|
*last_discrepancy = W1_SEARCH_NO_SLAVE;
|
|
return 0;
|
|
}
|
|
|
|
ret = w1_write_byte(dev, command);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
next_discrepancy = W1_SEARCH_LAST_SLAVE;
|
|
|
|
for (size_t id_bit_nr = 1; id_bit_nr < W1_SEARCH_LAST_SLAVE; id_bit_nr++) {
|
|
ret = w1_read_bit(dev);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
last_id_bit = (bool)ret;
|
|
ret = w1_read_bit(dev);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
last_complement_id_bit = (bool)ret;
|
|
|
|
if (last_id_bit && last_complement_id_bit) {
|
|
/*
|
|
* No slave participating:
|
|
* We can stop following the branch.
|
|
*/
|
|
LOG_DBG("No slave paricipating");
|
|
*last_discrepancy = W1_SEARCH_NO_SLAVE;
|
|
return 0;
|
|
} else if (last_id_bit != last_complement_id_bit) {
|
|
/*
|
|
* All slaves connected have same ROM bit value:
|
|
* We can directly follow last_id_bit branch.
|
|
*/
|
|
} else {
|
|
/*
|
|
* Discrepancy detected: bit value at id_bit_nr does
|
|
* not match for all slaves on the bus.
|
|
*/
|
|
if ((id_bit_nr > *last_discrepancy) ||
|
|
((id_bit_nr < *last_discrepancy) &&
|
|
(*rom_inv_64 & BIT64(id_bit_nr - 1)))) {
|
|
/*
|
|
* - id_bit_nr > last_discrepancy:
|
|
* Start always w/ branch of 1s
|
|
* - id_bit_nr < last_discrepancy:
|
|
* Follow same branch as before
|
|
*/
|
|
last_id_bit = true;
|
|
next_discrepancy = id_bit_nr;
|
|
} else {
|
|
/*
|
|
* - id_bit_nr == last_discrepancy:
|
|
* 1-path already done, therefore go 0 path
|
|
* - id_bit_nr < last_discrepancy:
|
|
* Follow same branch as before
|
|
*/
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Send and store the chosen bit: all not matching slaves will
|
|
* no longer participate in this search until they are reset.
|
|
*/
|
|
ret = w1_write_bit(dev, last_id_bit);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
*rom_inv_64 &= ~BIT64(id_bit_nr - 1);
|
|
*rom_inv_64 |= last_id_bit ? BIT64(id_bit_nr - 1) : 0;
|
|
}
|
|
|
|
*last_discrepancy = next_discrepancy;
|
|
return 0;
|
|
}
|
|
|
|
int z_impl_w1_search_bus(const struct device *dev, uint8_t command,
|
|
uint8_t family, w1_search_callback_t callback,
|
|
void *user_data)
|
|
{
|
|
size_t last_discrepancy = W1_SEARCH_DISCREPANCY_INIT;
|
|
uint64_t found_rom_inv_64 = 0;
|
|
struct w1_rom found_rom = { 0 };
|
|
int found_cnt = 0;
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
|
|
do {
|
|
ret = search_slave(dev, command, family, &last_discrepancy,
|
|
&found_rom_inv_64);
|
|
if (ret < 0) {
|
|
found_cnt = ret;
|
|
break;
|
|
}
|
|
if (last_discrepancy == W1_SEARCH_NO_SLAVE) {
|
|
break;
|
|
}
|
|
|
|
found_cnt++;
|
|
/*
|
|
* ROM is stored in found_rom_inv_64 in "inverse byte order" =>
|
|
* Only big-endian targets need to swap, such that struct's
|
|
* bytes are stored in big-endian byte order.
|
|
*/
|
|
if (IS_ENABLED(CONFIG_BIG_ENDIAN)) {
|
|
sys_memcpy_swap(&found_rom, &found_rom_inv_64, 8);
|
|
} else {
|
|
*(uint64_t *)&found_rom = found_rom_inv_64;
|
|
}
|
|
LOG_DBG("ROM found: nr %u, %016llx", found_cnt,
|
|
w1_rom_to_uint64(&found_rom));
|
|
|
|
if (callback != NULL) {
|
|
callback(found_rom, user_data);
|
|
}
|
|
|
|
} while (last_discrepancy != W1_SEARCH_LAST_SLAVE);
|
|
|
|
(void)w1_unlock_bus(dev);
|
|
return found_cnt;
|
|
}
|
|
|
|
int w1_read_rom(const struct device *dev, struct w1_rom *rom)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = w1_reset_bus(dev);
|
|
if (ret == 0) {
|
|
ret = -ENODEV;
|
|
goto out;
|
|
}
|
|
if (ret < 0) {
|
|
goto out;
|
|
}
|
|
|
|
ret = w1_write_byte(dev, W1_CMD_READ_ROM);
|
|
if (ret < 0) {
|
|
goto out;
|
|
}
|
|
ret = w1_read_block(dev, (uint8_t *)rom, sizeof(struct w1_rom));
|
|
if (ret < 0) {
|
|
goto out;
|
|
}
|
|
if (w1_crc8((uint8_t *)rom, sizeof(struct w1_rom)) != 0) {
|
|
ret = -EIO;
|
|
}
|
|
|
|
out:
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
};
|
|
|
|
static int match_rom(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
int ret;
|
|
uint8_t cmd;
|
|
|
|
if (!config->overdrive) {
|
|
if (w1_configure(dev, W1_SETTING_SPEED, 0) < 0) {
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
ret = w1_reset_bus(dev);
|
|
if (ret == 0) {
|
|
return -ENODEV;
|
|
}
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
cmd = config->overdrive ? W1_CMD_OVERDRIVE_MATCH_ROM : W1_CMD_MATCH_ROM;
|
|
ret = w1_write_byte(dev, cmd);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
ret = w1_write_block(dev, (uint8_t *)&config->rom, 8);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (config->overdrive) {
|
|
if (w1_configure(dev, W1_SETTING_SPEED, 1) < 0) {
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
};
|
|
|
|
int w1_match_rom(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = match_rom(dev, config);
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
}
|
|
|
|
int w1_resume_command(const struct device *dev)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = w1_reset_bus(dev);
|
|
if (ret == 0) {
|
|
ret = -ENODEV;
|
|
goto out;
|
|
}
|
|
if (ret < 0) {
|
|
goto out;
|
|
}
|
|
|
|
ret = w1_write_byte(dev, W1_CMD_RESUME);
|
|
out:
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
}
|
|
|
|
static int skip_rom(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
int ret;
|
|
uint8_t cmd;
|
|
|
|
if (!config->overdrive) {
|
|
if (w1_configure(dev, W1_SETTING_SPEED, 0) < 0) {
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
ret = w1_reset_bus(dev);
|
|
if (ret == 0) {
|
|
return -ENODEV;
|
|
}
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
cmd = config->overdrive ? W1_CMD_OVERDRIVE_SKIP_ROM : W1_CMD_SKIP_ROM;
|
|
ret = w1_write_byte(dev, cmd);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (config->overdrive) {
|
|
if (w1_configure(dev, W1_SETTING_SPEED, 1) < 0) {
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int w1_skip_rom(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = skip_rom(dev, config);
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
}
|
|
|
|
static int reset_select(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
if (IS_ENABLED(CONFIG_W1_NET_FORCE_MULTIDROP_ADDRESSING) || w1_get_slave_count(dev) > 1) {
|
|
return match_rom(dev, config);
|
|
}
|
|
|
|
return skip_rom(dev, config);
|
|
}
|
|
|
|
int w1_reset_select(const struct device *dev, const struct w1_slave_config *config)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = reset_select(dev, config);
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
}
|
|
|
|
static int write_read(const struct device *dev, const struct w1_slave_config *config,
|
|
const uint8_t *write_buf, size_t write_len,
|
|
uint8_t *read_buf, size_t read_len)
|
|
{
|
|
int ret;
|
|
|
|
ret = reset_select(dev, config);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
ret = w1_write_block(dev, write_buf, write_len);
|
|
if (ret < 0) {
|
|
return ret;
|
|
}
|
|
|
|
if (read_buf == NULL && read_len > 0) {
|
|
return -EIO;
|
|
}
|
|
return w1_read_block(dev, read_buf, read_len);
|
|
};
|
|
|
|
int w1_write_read(const struct device *dev, const struct w1_slave_config *config,
|
|
const uint8_t *write_buf, size_t write_len,
|
|
uint8_t *read_buf, size_t read_len)
|
|
{
|
|
int ret;
|
|
|
|
(void)w1_lock_bus(dev);
|
|
ret = write_read(dev, config, write_buf, write_len, read_buf, read_len);
|
|
(void)w1_unlock_bus(dev);
|
|
return ret;
|
|
};
|