/**************************************************************************** * drivers/1wire/ds28e17.c * * Copyright (C) 2018 Haltian Ltd. All rights reserved. * Author: Juha Niskanen * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name NuttX nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include "1wire_internal.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ #ifndef CONFIG_DS28E17_I2C_FREQUENCY # define CONFIG_DS28E17_I2C_FREQUENCY 400000 #endif /* DS28E17 device 1-Wire family ID */ #define DS_FAMILY 0x19 /* DS28E17 device I2C commands */ #define DS_WRITE_DATA_WITH_STOP 0x4b #define DS_WRITE_DATA_NO_STOP 0x5a #define DS_WRITE_DATA_ONLY 0x69 #define DS_WRITE_DATA_ONLY_WITH_STOP 0x78 #define DS_READ_DATA_WITH_STOP 0x87 #define DS_WRITE_READ_DATA_WITH_STOP 0x2d #define DS_WRITE_CONFIGURATION 0xd2 #define DS_READ_CONFIGURATION 0xe1 #define DS_ENABLE_SLEEP_MODE 0x1e #define DS_READ_DEVICE_REVISION 0xc4 /* DS28E17 status bits */ #define DS_STATUS_CRC 0x01 #define DS_STATUS_ADDRESS 0x02 #define DS_STATUS_START 0x08 /* Maximum number of I2C bytes to transfer within one CRC16 protected onewire * command (same for either read and write). */ #define DS_DATA_LIMIT 255 /* Default I2C frequency to use when DS28E17 is detected. */ #define DS_DEFAULT_FREQUENCY CONFIG_DS28E17_I2C_FREQUENCY /* Default I2C stretch value. */ #define DS_DEFAULT_STRETCH 1 /* How many times to retry check for chip busy condition vanishing. */ #define DS_BUSYWAIT_RETRIES 500 /**************************************************************************** * Private Types ****************************************************************************/ struct ds28e17_dev_s /* Must be cast-compatible with onewire_master_s */ { FAR struct onewire_dev_s *dev; /* 1-wire interface */ }; /* I2C Device, Instance */ struct ds_i2c_inst_s /* Must be cast-compatible with i2c_master_s */ { FAR const struct i2c_ops_s *ops; /* Standard I2C operations */ /* 1-wire data */ FAR struct onewire_master_s *master; /* 1-wire bus master */ FAR struct onewire_slave_s slave; /* Our slave data on 1-wire bus */ /* I2C data */ uint32_t frequency; /* Current I2C frequency */ uint8_t stretch; /* Current I2C stretch value */ }; /**************************************************************************** * Private Function Prototypes ****************************************************************************/ static int ds_i2c_reset(FAR struct i2c_master_s *i2cdev); static int ds_i2c_transfer(FAR struct i2c_master_s *i2cdev, FAR struct i2c_msg_s *msgs, int count); /**************************************************************************** * Private Data ****************************************************************************/ /* Device Structures, Instantiation */ static const struct i2c_ops_s ds_i2c_ops = { .transfer = ds_i2c_transfer #ifdef CONFIG_I2C_RESET , .reset = ds_i2c_reset #endif }; /**************************************************************************** * Private Functions ****************************************************************************/ /**************************************************************************** * Name: ds_i2c_sem_wait * * Description: * Take the exclusive access, waiting as necessary * ****************************************************************************/ static inline void ds_i2c_sem_wait(FAR struct i2c_master_s *i2cdev) { FAR struct ds_i2c_inst_s *inst = (FAR struct ds_i2c_inst_s *)i2cdev; FAR struct onewire_master_s *master = inst->master; return onewire_sem_wait(master); } /**************************************************************************** * Name: ds_i2c_sem_post * * Description: * Release the mutual exclusion semaphore * ****************************************************************************/ static inline void ds_i2c_sem_post(FAR struct i2c_master_s *i2cdev) { FAR struct ds_i2c_inst_s *inst = (FAR struct ds_i2c_inst_s *)i2cdev; FAR struct onewire_master_s *master = inst->master; onewire_sem_post(master); } static int ds_error(uint8_t buf[]) { /* Warnings */ if (buf[0] & DS_STATUS_CRC) { i2cwarn("crc16 match failed\n"); } if ((buf[0] & (DS_STATUS_CRC | DS_STATUS_ADDRESS)) == 0 && buf[1] != 0) { i2cwarn("I2C short write, no ack for %d bytes\n", buf[1]); } /* Error conditions */ if (buf[0] & DS_STATUS_ADDRESS) { i2cerr("I2C device not responding\n"); return -ENXIO; } if (buf[0] & DS_STATUS_START) { i2cerr("I2C status start\n"); return -EAGAIN; } if (buf[0] != 0 || buf[1] != 0) { i2cerr("I2C IO error\n"); return -EIO; } return OK; } static inline int ds_busywait_time(FAR struct ds_i2c_inst_s *inst) { int d; /* Return busywait time (us) for I2C speeds 100 kHz, 400 kHz * and 900 kHz, adjusted with current stretch value. */ switch (inst->frequency) { case 100000: d = 90; break; case 400000: default: d = 23; break; case 900000: d = 10; break; } return d * inst->stretch; } static int ds_busywait(FAR struct ds_i2c_inst_s *inst, size_t length) { FAR struct onewire_master_s *master = inst->master; int retries = DS_BUSYWAIT_RETRIES; int delay; int ret; uint8_t bit = 1; /* Calculate delay time from current I2C settings. */ delay = ds_busywait_time(inst); do { ret = ONEWIRE_READBIT(master->dev, &bit); if (ret < 0) { i2cerr("ERROR: ONEWIRE_READBIT failed\n"); return ret; } if (bit == 0) { return OK; } if (retries == DS_BUSYWAIT_RETRIES) { /* First time, after checking bit once, do a big delay * for I2C to process all bytes in message. */ up_udelay(delay * length); } else { /* Otherwise, wait for one byte duration. */ up_udelay(delay); } } while (retries-- > 0); i2cwarn("busywait timeout\n"); return -ETIMEDOUT; } /**************************************************************************** * Name: ds_i2c_read * * Description: * Read data from I2C slave. * ****************************************************************************/ static int ds_i2c_read(FAR struct ds_i2c_inst_s *inst, uint16_t i2c_addr, FAR uint8_t *buffer, ssize_t length) { FAR struct onewire_master_s *master = inst->master; uint16_t crc; int ret; uint8_t buf[5]; if (length <= 0) { return -EINVAL; } if (length > DS_DATA_LIMIT) { i2cerr("ERROR: reading too many bytes!\n"); return -EINVAL; } /* Send command to DS28E17. */ buf[0] = DS_READ_DATA_WITH_STOP; buf[1] = i2c_addr << 1 | 0x01; buf[2] = length; crc = onewire_crc16(buf, 3, 0); buf[3] = ~(crc & 0xff); buf[4] = ~((crc >> 8) & 0xff); ret = ONEWIRE_WRITE(master->dev, buf, 5); if (ret < 0) { return ret; } /* Wait busy indication to vanish. */ ret = ds_busywait(inst, length + 1); if (ret < 0) { return ret; } /* Read status from DS28E17. */ ret = ONEWIRE_READ(master->dev, buf, 1); buf[1] = 0; /* Check error conditions. */ ret = ds_error(buf); if (ret < 0) { return ret; } /* Read received I2C data from DS28E17. */ return ONEWIRE_READ(master->dev, buffer, length); } /**************************************************************************** * Name: ds_i2c_write * * Description: * Write data to I2C slave. * ****************************************************************************/ static int ds_i2c_write(FAR struct ds_i2c_inst_s *inst, uint16_t i2c_addr, FAR const uint8_t *buffer, ssize_t length, bool stop) { FAR struct onewire_master_s *master = inst->master; uint16_t crc; int ret; uint8_t buf[5]; if (length <= 0) { return -EINVAL; } if (length > DS_DATA_LIMIT) { /* TODO: split big writes into multiple chunks. */ i2cerr("ERROR: writing too many bytes!\n"); return -EINVAL; } /* Write command header. */ buf[0] = stop ? DS_WRITE_DATA_WITH_STOP : DS_WRITE_DATA_NO_STOP; buf[1] = i2c_addr << 1; buf[2] = length; crc = onewire_crc16(buf, 3, 0); ret = ONEWIRE_WRITE(master->dev, buf, 3); if (ret < 0) { return ret; } /* Write payload I2C data to DS28E17. */ crc = onewire_crc16(buffer, length, crc); ret = ONEWIRE_WRITE(master->dev, buffer, length); if (ret < 0) { return ret; } /* Write checksum. */ buf[0] = ~(crc & 0xff); buf[1] = ~((crc >> 8) & 0xff); ret = ONEWIRE_WRITE(master->dev, buf, 2); if (ret < 0) { return ret; } /* Wait busy indication to vanish. */ ret = ds_busywait(inst, length + 1); if (ret < 0) { return ret; } /* Read status from DS28E17. */ ret = ONEWIRE_READ(master->dev, buf, 2); if (ret < 0) { return ret; } /* Check error conditions. */ ret = ds_error(buf); if (ret < 0) { return ret; } /* Return count of bytes written. */ return length; } static int ds_i2c_setfrequency(FAR struct ds_i2c_inst_s *inst, uint32_t frequency) { FAR struct onewire_master_s *master = inst->master; uint8_t buf[2]; int speed; int ret; switch (frequency) { case 100000: speed = 0; break; case 400000: speed = 1; break; case 900000: speed = 2; break; default: i2cerr("ERROR: bad I2C freq %u\n", frequency); return -EINVAL; } i2cinfo("Changing I2C freq %u -> %u\n", inst->frequency, frequency); /* Select DS28E17 */ ret = onewire_reset_select(&inst->slave); if (ret < 0) { i2cerr("ERROR: cannot change I2C freq\n"); return ret; } /* Write new speed to device */ buf[0] = DS_WRITE_CONFIGURATION; buf[1] = speed; ret = ONEWIRE_WRITE(master->dev, buf, 2); if (ret < 0) { i2cerr("ERROR: cannot change I2C freq\n"); return ret; } inst->frequency = frequency; return ret; } /**************************************************************************** * Name: ds_i2c_process * * Description: * Common I2C transfer logic * * Initiates a master mode transaction on the I2C bus to transfer the * provided messages to and from the slave devices. * ****************************************************************************/ static int ds_i2c_process(FAR struct i2c_master_s *i2cdev, FAR struct i2c_msg_s *msgs, int count) { FAR struct ds_i2c_inst_s *inst = (FAR struct ds_i2c_inst_s *)i2cdev; FAR struct onewire_master_s *master = inst->master; int ret; int i; /* Check from first message only, if we want to change I2C frequency. */ if (inst->frequency != msgs[0].frequency) { ds_i2c_setfrequency(inst, msgs[0].frequency); } /* Select DS28E17 */ i = onewire_reset_select(&inst->slave); if (i < 0) { goto errout; } while (i < count) { /* TODO: we could use DS_WRITE_READ_DATA_WITH_STOP to optimize the * common case of write followed by read to a same address. */ if (msgs[i].flags & I2C_M_READ) { /* Read transfer. */ ret = ds_i2c_read(inst, msgs[i].addr, msgs[i].buffer, msgs[i].length); if (ret < 0) { i = ret; goto errout; } } else { /* Write transfer. Stop condition only for last transfer. */ ret = ds_i2c_write(inst, msgs[i].addr, msgs[i].buffer, msgs[i].length, i == (count-1)); if (ret < 0) { i = ret; goto errout; } } /* Any more messages to process? */ i++; if (i < count) { /* Yes. Resume to same DS28E17. */ /* Oddness: Skip-ROM does not set RS-bit needed by resume. */ if (master->nslaves > 1) { ret = onewire_reset_resume(master); } else { ret = onewire_reset_select(&inst->slave); } if (ret < 0) { i = ret; goto errout; } } } errout: return i; } /**************************************************************************** * Name: ds_i2c_reset * * Description: * Reset an I2C bus * ****************************************************************************/ #ifdef CONFIG_I2C_RESET static int ds_i2c_reset(FAR struct i2c_master_s *i2cdev) { FAR struct ds_i2c_inst_s *inst = (FAR struct ds_i2c_inst_s *)i2cdev; FAR struct onewire_master_s *master = inst->master; int ret; ds_i2c_sem_wait(i2cdev); ret = ONEWIRE_RESET(master->dev); ds_i2c_sem_post(i2cdev); return ret; } #endif /**************************************************************************** * Name: ds_i2c_transfer * * Description: * Generic I2C transfer function * ****************************************************************************/ static int ds_i2c_transfer(FAR struct i2c_master_s *i2cdev, FAR struct i2c_msg_s *msgs, int count) { int ret; ds_i2c_sem_wait(i2cdev); ret = ds_i2c_process(i2cdev, msgs, count); ds_i2c_sem_post(i2cdev); return ret; } /**************************************************************************** * Name: ds28e17_selftest * * Description: * ****************************************************************************/ static int ds28e17_selftest(FAR struct ds_i2c_inst_s *inst) { FAR struct onewire_master_s *master = inst->master; uint8_t txbuf[] = { ONEWIRE_CMD_READ_ROM }; uint8_t rxbuf[8] = { 0 }; uint64_t rom; uint8_t crc; int ret; /* Read ROM-code of single connected slave and * check its checksum. */ ret = ONEWIRE_RESET(master->dev); if (ret < 0) { i2cerr("ERROR: ONEWIRE_RESET failed: %d\n", ret); return ret; } ret = ONEWIRE_WRITE(master->dev, txbuf, sizeof(txbuf)); if (ret < 0) { i2cerr("ERROR: ONEWIRE_WRITE failed: %d\n", ret); return ret; } ret = ONEWIRE_READ(master->dev, rxbuf, sizeof(rxbuf)); if (ret < 0) { i2cerr("ERROR: ONEWIRE_READ failed: %d\n", ret); return ret; } #ifdef CONFIG_DEBUG_I2C_INFO lib_dumpbuffer("ds28e17_selftest: rxbuf", rxbuf, sizeof(rxbuf)); #endif memcpy(&rom, rxbuf, 8); i2cinfo("recv rom: 0x%llx\n", rom); crc = onewire_crc8(rxbuf, sizeof(rxbuf)-1); i2cinfo("crc8=%d, recv crc8=%d\n", crc, (int)rxbuf[7]); if (crc != rxbuf[7]) { i2cerr("ERROR: crc8 does not match!\n"); ret = -EIO; } return ret; } /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: ds28e17_search * * Description: * Search all DS28E17 devices from a 1-wire network. * * Input Parameters: * priv - Pointer to the associated DS28E17 * cb_search - Callback to call on each device found * arg - Argument passed to cb_search * * Return Value: * Number of DS28E17 devices present. * ****************************************************************************/ int ds28e17_search(FAR struct ds28e17_dev_s *priv, void (*cb_search)(int family, uint64_t romcode, void *arg), void *arg) { FAR struct onewire_master_s *master = (FAR struct onewire_master_s *)priv; int ret; DEBUGASSERT(master != NULL && cb_search != NULL); onewire_sem_wait(master); ret = onewire_search(master, DS_FAMILY, false, cb_search, arg); onewire_sem_post(master); return ret; } /**************************************************************************** * Name: ds28e17_lower_half * * Description: * Initialize the lower half of the DS28E17 by creating a i2c_master_s * for the virtual i2c master and link it to the associated DS28E17 and * its port. * * Input Parameters: * dev - Pointer to the associated DS28E17 * romcode - The unique 64-bit address in 1-wire network. * Use zero for skip-ROM mode. * * Returned Value: * i2c device instance; NULL on failure. * ****************************************************************************/ FAR struct i2c_master_s * ds28e17_lower_half(FAR struct ds28e17_dev_s *priv, uint64_t romcode) { FAR struct ds_i2c_inst_s *inst; /* device, single instance */ FAR struct onewire_master_s *master = (FAR struct onewire_master_s *)priv; DEBUGASSERT(master != NULL); /* Allocate instance */ inst = kmm_zalloc(sizeof(struct ds_i2c_inst_s)); if (inst == NULL) { i2cerr("ERROR: Failed to allocate instance\n"); return NULL; } /* Initialize instance */ inst->ops = &ds_i2c_ops; inst->master = master; inst->frequency = 400000; /* power-on frequency */ inst->stretch = DS_DEFAULT_STRETCH; inst->slave.romcode = romcode; /* We need a recursive lock as this may be called from a search callback. */ onewire_sem_wait(master); if (onewire_addslave(master, &inst->slave) < 0) { kmm_free(inst); i2cerr("ERROR: Failed to add slave\n"); onewire_sem_post(master); return NULL; } /* Should default speed be different from DS28E17 power-on frequency? */ if (inst->frequency != DS_DEFAULT_FREQUENCY) { ds_i2c_setfrequency(inst, DS_DEFAULT_FREQUENCY); } /* TODO: better selftest */ if (master->maxslaves == 1) { ds28e17_selftest(inst); } onewire_sem_post(master); return (struct i2c_master_s *)inst; } /**************************************************************************** * Name: ds28e17_lower_half_unregister * * Description: * Put back the lower half of the DS28E17. * * Input Parameters: * priv - Pointer to the associated DS28E17 * i2cdev - i2c device instance from ds28e17_lower_half() * * Returned Value: * Zero on success; a negated errno value on failure. * ****************************************************************************/ int ds28e17_lower_half_unregister(FAR struct ds28e17_dev_s *priv, FAR struct i2c_master_s *i2cdev) { FAR struct ds_i2c_inst_s *inst = (FAR struct ds_i2c_inst_s *)i2cdev; FAR struct onewire_master_s *master = inst->master; int ret; onewire_sem_wait(master); ret = onewire_removeslave(master, &inst->slave); if (ret < 0) { kmm_free(inst); i2cerr("ERROR: Failed to remove slave\n"); onewire_sem_post(master); return ret; } kmm_free(inst); onewire_sem_post(master); return OK; } /**************************************************************************** * Name: ds28e17_initialize * * Description: * Returns a common DS28E17 device from 1-wire lower half device * * Input Parameters: * dev - The allocated 1-wire lower half * ****************************************************************************/ FAR struct ds28e17_dev_s *ds28e17_initialize(FAR struct onewire_dev_s *dev) { FAR struct onewire_master_s *priv; priv = onewire_initialize(dev, DS_DEFAULT_MAXSLAVES); /* We do not have our own data fields so just cast it. */ return (FAR struct ds28e17_dev_s *)priv; }