123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772 |
- /*
- * w1_ds28e17.c - w1 family 19 (DS28E17) driver
- *
- * Copyright (c) 2016 Jan Kandziora <jjj@gmx.de>
- *
- * This source code is licensed under the GNU General Public License,
- * Version 2. See the file COPYING for more details.
- */
- #include <linux/crc16.h>
- #include <linux/delay.h>
- #include <linux/device.h>
- #include <linux/i2c.h>
- #include <linux/kernel.h>
- #include <linux/module.h>
- #include <linux/moduleparam.h>
- #include <linux/slab.h>
- #include <linux/types.h>
- #include <linux/uaccess.h>
- #define CRC16_INIT 0
- #include <linux/w1.h>
- #define W1_FAMILY_DS28E17 0x19
- /* Module setup. */
- MODULE_LICENSE("GPL v2");
- MODULE_AUTHOR("Jan Kandziora <jjj@gmx.de>");
- MODULE_DESCRIPTION("w1 family 19 driver for DS28E17, 1-wire to I2C master bridge");
- MODULE_ALIAS("w1-family-" __stringify(W1_FAMILY_DS28E17));
- /* Default I2C speed to be set when a DS28E17 is detected. */
- static int i2c_speed = 100;
- module_param_named(speed, i2c_speed, int, (S_IRUSR | S_IWUSR));
- MODULE_PARM_DESC(speed, "Default I2C speed to be set when a DS28E17 is detected");
- /* Default I2C stretch value to be set when a DS28E17 is detected. */
- static char i2c_stretch = 1;
- module_param_named(stretch, i2c_stretch, byte, (S_IRUSR | S_IWUSR));
- MODULE_PARM_DESC(stretch, "Default I2C stretch value to be set when a DS28E17 is detected");
- /* DS28E17 device command codes. */
- #define W1_F19_WRITE_DATA_WITH_STOP 0x4B
- #define W1_F19_WRITE_DATA_NO_STOP 0x5A
- #define W1_F19_WRITE_DATA_ONLY 0x69
- #define W1_F19_WRITE_DATA_ONLY_WITH_STOP 0x78
- #define W1_F19_READ_DATA_WITH_STOP 0x87
- #define W1_F19_WRITE_READ_DATA_WITH_STOP 0x2D
- #define W1_F19_WRITE_CONFIGURATION 0xD2
- #define W1_F19_READ_CONFIGURATION 0xE1
- #define W1_F19_ENABLE_SLEEP_MODE 0x1E
- #define W1_F19_READ_DEVICE_REVISION 0xC4
- /* DS28E17 status bits */
- #define W1_F19_STATUS_CRC 0x01
- #define W1_F19_STATUS_ADDRESS 0x02
- #define W1_F19_STATUS_START 0x08
- /*
- * Maximum number of I2C bytes to transfer within one CRC16 protected onewire
- * command.
- * */
- #define W1_F19_WRITE_DATA_LIMIT 255
- /* Maximum number of I2C bytes to read with one onewire command. */
- #define W1_F19_READ_DATA_LIMIT 255
- /* Constants for calculating the busy sleep. */
- #define W1_F19_BUSY_TIMEBASES { 90, 23, 10 }
- #define W1_F19_BUSY_GRATUITY 1000
- /* Number of checks for the busy flag before timeout. */
- #define W1_F19_BUSY_CHECKS 1000
- /* Slave specific data. */
- struct w1_f19_data {
- u8 speed;
- u8 stretch;
- struct i2c_adapter adapter;
- };
- /* Wait a while until the busy flag clears. */
- static int w1_f19_i2c_busy_wait(struct w1_slave *sl, size_t count)
- {
- const unsigned long timebases[3] = W1_F19_BUSY_TIMEBASES;
- struct w1_f19_data *data = sl->family_data;
- unsigned int checks;
- /* Check the busy flag first in any case.*/
- if (w1_touch_bit(sl->master, 1) == 0)
- return 0;
- /*
- * Do a generously long sleep in the beginning,
- * as we have to wait at least this time for all
- * the I2C bytes at the given speed to be transferred.
- */
- usleep_range(timebases[data->speed] * (data->stretch) * count,
- timebases[data->speed] * (data->stretch) * count
- + W1_F19_BUSY_GRATUITY);
- /* Now continusly check the busy flag sent by the DS28E17. */
- checks = W1_F19_BUSY_CHECKS;
- while ((checks--) > 0) {
- /* Return success if the busy flag is cleared. */
- if (w1_touch_bit(sl->master, 1) == 0)
- return 0;
- /* Wait one non-streched byte timeslot. */
- udelay(timebases[data->speed]);
- }
- /* Timeout. */
- dev_warn(&sl->dev, "busy timeout\n");
- return -ETIMEDOUT;
- }
- /* Utility function: result. */
- static size_t w1_f19_error(struct w1_slave *sl, u8 w1_buf[])
- {
- /* Warnings. */
- if (w1_buf[0] & W1_F19_STATUS_CRC)
- dev_warn(&sl->dev, "crc16 mismatch\n");
- if (w1_buf[0] & W1_F19_STATUS_ADDRESS)
- dev_warn(&sl->dev, "i2c device not responding\n");
- if ((w1_buf[0] & (W1_F19_STATUS_CRC | W1_F19_STATUS_ADDRESS)) == 0
- && w1_buf[1] != 0) {
- dev_warn(&sl->dev, "i2c short write, %d bytes not acknowledged\n",
- w1_buf[1]);
- }
- /* Check error conditions. */
- if (w1_buf[0] & W1_F19_STATUS_ADDRESS)
- return -ENXIO;
- if (w1_buf[0] & W1_F19_STATUS_START)
- return -EAGAIN;
- if (w1_buf[0] != 0 || w1_buf[1] != 0)
- return -EIO;
- /* All ok. */
- return 0;
- }
- /* Utility function: write data to I2C slave, single chunk. */
- static int __w1_f19_i2c_write(struct w1_slave *sl,
- const u8 *command, size_t command_count,
- const u8 *buffer, size_t count)
- {
- u16 crc;
- int error;
- u8 w1_buf[2];
- /* Send command and I2C data to DS28E17. */
- crc = crc16(CRC16_INIT, command, command_count);
- w1_write_block(sl->master, command, command_count);
- w1_buf[0] = count;
- crc = crc16(crc, w1_buf, 1);
- w1_write_8(sl->master, w1_buf[0]);
- crc = crc16(crc, buffer, count);
- w1_write_block(sl->master, buffer, count);
- w1_buf[0] = ~(crc & 0xFF);
- w1_buf[1] = ~((crc >> 8) & 0xFF);
- w1_write_block(sl->master, w1_buf, 2);
- /* Wait until busy flag clears (or timeout). */
- if (w1_f19_i2c_busy_wait(sl, count + 1) < 0)
- return -ETIMEDOUT;
- /* Read status from DS28E17. */
- w1_read_block(sl->master, w1_buf, 2);
- /* Check error conditions. */
- error = w1_f19_error(sl, w1_buf);
- if (error < 0)
- return error;
- /* Return number of bytes written. */
- return count;
- }
- /* Write data to I2C slave. */
- static int w1_f19_i2c_write(struct w1_slave *sl, u16 i2c_address,
- const u8 *buffer, size_t count, bool stop)
- {
- int result;
- int remaining = count;
- const u8 *p;
- u8 command[2];
- /* Check input. */
- if (count == 0)
- return -EOPNOTSUPP;
- /* Check whether we need multiple commands. */
- if (count <= W1_F19_WRITE_DATA_LIMIT) {
- /*
- * Small data amount. Data can be sent with
- * a single onewire command.
- */
- /* Send all data to DS28E17. */
- command[0] = (stop ? W1_F19_WRITE_DATA_WITH_STOP
- : W1_F19_WRITE_DATA_NO_STOP);
- command[1] = i2c_address << 1;
- result = __w1_f19_i2c_write(sl, command, 2, buffer, count);
- } else {
- /* Large data amount. Data has to be sent in multiple chunks. */
- /* Send first chunk to DS28E17. */
- p = buffer;
- command[0] = W1_F19_WRITE_DATA_NO_STOP;
- command[1] = i2c_address << 1;
- result = __w1_f19_i2c_write(sl, command, 2, p,
- W1_F19_WRITE_DATA_LIMIT);
- if (result < 0)
- return result;
- /* Resume to same DS28E17. */
- if (w1_reset_resume_command(sl->master))
- return -EIO;
- /* Next data chunk. */
- p += W1_F19_WRITE_DATA_LIMIT;
- remaining -= W1_F19_WRITE_DATA_LIMIT;
- while (remaining > W1_F19_WRITE_DATA_LIMIT) {
- /* Send intermediate chunk to DS28E17. */
- command[0] = W1_F19_WRITE_DATA_ONLY;
- result = __w1_f19_i2c_write(sl, command, 1, p,
- W1_F19_WRITE_DATA_LIMIT);
- if (result < 0)
- return result;
- /* Resume to same DS28E17. */
- if (w1_reset_resume_command(sl->master))
- return -EIO;
- /* Next data chunk. */
- p += W1_F19_WRITE_DATA_LIMIT;
- remaining -= W1_F19_WRITE_DATA_LIMIT;
- }
- /* Send final chunk to DS28E17. */
- command[0] = (stop ? W1_F19_WRITE_DATA_ONLY_WITH_STOP
- : W1_F19_WRITE_DATA_ONLY);
- result = __w1_f19_i2c_write(sl, command, 1, p, remaining);
- }
- return result;
- }
- /* Read data from I2C slave. */
- static int w1_f19_i2c_read(struct w1_slave *sl, u16 i2c_address,
- u8 *buffer, size_t count)
- {
- u16 crc;
- int error;
- u8 w1_buf[5];
- /* Check input. */
- if (count == 0)
- return -EOPNOTSUPP;
- /* Send command to DS28E17. */
- w1_buf[0] = W1_F19_READ_DATA_WITH_STOP;
- w1_buf[1] = i2c_address << 1 | 0x01;
- w1_buf[2] = count;
- crc = crc16(CRC16_INIT, w1_buf, 3);
- w1_buf[3] = ~(crc & 0xFF);
- w1_buf[4] = ~((crc >> 8) & 0xFF);
- w1_write_block(sl->master, w1_buf, 5);
- /* Wait until busy flag clears (or timeout). */
- if (w1_f19_i2c_busy_wait(sl, count + 1) < 0)
- return -ETIMEDOUT;
- /* Read status from DS28E17. */
- w1_buf[0] = w1_read_8(sl->master);
- w1_buf[1] = 0;
- /* Check error conditions. */
- error = w1_f19_error(sl, w1_buf);
- if (error < 0)
- return error;
- /* Read received I2C data from DS28E17. */
- return w1_read_block(sl->master, buffer, count);
- }
- /* Write to, then read data from I2C slave. */
- static int w1_f19_i2c_write_read(struct w1_slave *sl, u16 i2c_address,
- const u8 *wbuffer, size_t wcount, u8 *rbuffer, size_t rcount)
- {
- u16 crc;
- int error;
- u8 w1_buf[3];
- /* Check input. */
- if (wcount == 0 || rcount == 0)
- return -EOPNOTSUPP;
- /* Send command and I2C data to DS28E17. */
- w1_buf[0] = W1_F19_WRITE_READ_DATA_WITH_STOP;
- w1_buf[1] = i2c_address << 1;
- w1_buf[2] = wcount;
- crc = crc16(CRC16_INIT, w1_buf, 3);
- w1_write_block(sl->master, w1_buf, 3);
- crc = crc16(crc, wbuffer, wcount);
- w1_write_block(sl->master, wbuffer, wcount);
- w1_buf[0] = rcount;
- crc = crc16(crc, w1_buf, 1);
- w1_buf[1] = ~(crc & 0xFF);
- w1_buf[2] = ~((crc >> 8) & 0xFF);
- w1_write_block(sl->master, w1_buf, 3);
- /* Wait until busy flag clears (or timeout). */
- if (w1_f19_i2c_busy_wait(sl, wcount + rcount + 2) < 0)
- return -ETIMEDOUT;
- /* Read status from DS28E17. */
- w1_read_block(sl->master, w1_buf, 2);
- /* Check error conditions. */
- error = w1_f19_error(sl, w1_buf);
- if (error < 0)
- return error;
- /* Read received I2C data from DS28E17. */
- return w1_read_block(sl->master, rbuffer, rcount);
- }
- /* Do an I2C master transfer. */
- static int w1_f19_i2c_master_transfer(struct i2c_adapter *adapter,
- struct i2c_msg *msgs, int num)
- {
- struct w1_slave *sl = (struct w1_slave *) adapter->algo_data;
- int i = 0;
- int result = 0;
- /* Start onewire transaction. */
- mutex_lock(&sl->master->bus_mutex);
- /* Select DS28E17. */
- if (w1_reset_select_slave(sl)) {
- i = -EIO;
- goto error;
- }
- /* Loop while there are still messages to transfer. */
- while (i < num) {
- /*
- * Check for special case: Small write followed
- * by read to same I2C device.
- */
- if (i < (num-1)
- && msgs[i].addr == msgs[i+1].addr
- && !(msgs[i].flags & I2C_M_RD)
- && (msgs[i+1].flags & I2C_M_RD)
- && (msgs[i].len <= W1_F19_WRITE_DATA_LIMIT)) {
- /*
- * The DS28E17 has a combined transfer
- * for small write+read.
- */
- result = w1_f19_i2c_write_read(sl, msgs[i].addr,
- msgs[i].buf, msgs[i].len,
- msgs[i+1].buf, msgs[i+1].len);
- if (result < 0) {
- i = result;
- goto error;
- }
- /*
- * Check if we should interpret the read data
- * as a length byte. The DS28E17 unfortunately
- * has no read without stop, so we can just do
- * another simple read in that case.
- */
- if (msgs[i+1].flags & I2C_M_RECV_LEN) {
- result = w1_f19_i2c_read(sl, msgs[i+1].addr,
- &(msgs[i+1].buf[1]), msgs[i+1].buf[0]);
- if (result < 0) {
- i = result;
- goto error;
- }
- }
- /* Eat up read message, too. */
- i++;
- } else if (msgs[i].flags & I2C_M_RD) {
- /* Read transfer. */
- result = w1_f19_i2c_read(sl, msgs[i].addr,
- msgs[i].buf, msgs[i].len);
- if (result < 0) {
- i = result;
- goto error;
- }
- /*
- * Check if we should interpret the read data
- * as a length byte. The DS28E17 unfortunately
- * has no read without stop, so we can just do
- * another simple read in that case.
- */
- if (msgs[i].flags & I2C_M_RECV_LEN) {
- result = w1_f19_i2c_read(sl,
- msgs[i].addr,
- &(msgs[i].buf[1]),
- msgs[i].buf[0]);
- if (result < 0) {
- i = result;
- goto error;
- }
- }
- } else {
- /*
- * Write transfer.
- * Stop condition only for last
- * transfer.
- */
- result = w1_f19_i2c_write(sl,
- msgs[i].addr,
- msgs[i].buf,
- msgs[i].len,
- i == (num-1));
- if (result < 0) {
- i = result;
- goto error;
- }
- }
- /* Next message. */
- i++;
- /* Are there still messages to send/receive? */
- if (i < num) {
- /* Yes. Resume to same DS28E17. */
- if (w1_reset_resume_command(sl->master)) {
- i = -EIO;
- goto error;
- }
- }
- }
- error:
- /* End onewire transaction. */
- mutex_unlock(&sl->master->bus_mutex);
- /* Return number of messages processed or error. */
- return i;
- }
- /* Get I2C adapter functionality. */
- static u32 w1_f19_i2c_functionality(struct i2c_adapter *adapter)
- {
- /*
- * Plain I2C functions only.
- * SMBus is emulated by the kernel's I2C layer.
- * No "I2C_FUNC_SMBUS_QUICK"
- * No "I2C_FUNC_SMBUS_READ_BLOCK_DATA"
- * No "I2C_FUNC_SMBUS_BLOCK_PROC_CALL"
- */
- return I2C_FUNC_I2C |
- I2C_FUNC_SMBUS_BYTE |
- I2C_FUNC_SMBUS_BYTE_DATA |
- I2C_FUNC_SMBUS_WORD_DATA |
- I2C_FUNC_SMBUS_PROC_CALL |
- I2C_FUNC_SMBUS_WRITE_BLOCK_DATA |
- I2C_FUNC_SMBUS_I2C_BLOCK |
- I2C_FUNC_SMBUS_PEC;
- }
- /* I2C adapter quirks. */
- static const struct i2c_adapter_quirks w1_f19_i2c_adapter_quirks = {
- .max_read_len = W1_F19_READ_DATA_LIMIT,
- };
- /* I2C algorithm. */
- static const struct i2c_algorithm w1_f19_i2c_algorithm = {
- .master_xfer = w1_f19_i2c_master_transfer,
- .functionality = w1_f19_i2c_functionality,
- };
- /* Read I2C speed from DS28E17. */
- static int w1_f19_get_i2c_speed(struct w1_slave *sl)
- {
- struct w1_f19_data *data = sl->family_data;
- int result = -EIO;
- /* Start onewire transaction. */
- mutex_lock(&sl->master->bus_mutex);
- /* Select slave. */
- if (w1_reset_select_slave(sl))
- goto error;
- /* Read slave configuration byte. */
- w1_write_8(sl->master, W1_F19_READ_CONFIGURATION);
- result = w1_read_8(sl->master);
- if (result < 0 || result > 2) {
- result = -EIO;
- goto error;
- }
- /* Update speed in slave specific data. */
- data->speed = result;
- error:
- /* End onewire transaction. */
- mutex_unlock(&sl->master->bus_mutex);
- return result;
- }
- /* Set I2C speed on DS28E17. */
- static int __w1_f19_set_i2c_speed(struct w1_slave *sl, u8 speed)
- {
- struct w1_f19_data *data = sl->family_data;
- const int i2c_speeds[3] = { 100, 400, 900 };
- u8 w1_buf[2];
- /* Select slave. */
- if (w1_reset_select_slave(sl))
- return -EIO;
- w1_buf[0] = W1_F19_WRITE_CONFIGURATION;
- w1_buf[1] = speed;
- w1_write_block(sl->master, w1_buf, 2);
- /* Update speed in slave specific data. */
- data->speed = speed;
- dev_info(&sl->dev, "i2c speed set to %d kBaud\n", i2c_speeds[speed]);
- return 0;
- }
- static int w1_f19_set_i2c_speed(struct w1_slave *sl, u8 speed)
- {
- int result;
- /* Start onewire transaction. */
- mutex_lock(&sl->master->bus_mutex);
- /* Set I2C speed on DS28E17. */
- result = __w1_f19_set_i2c_speed(sl, speed);
- /* End onewire transaction. */
- mutex_unlock(&sl->master->bus_mutex);
- return result;
- }
- /* Sysfs attributes. */
- /* I2C speed attribute for a single chip. */
- static ssize_t speed_show(struct device *dev, struct device_attribute *attr,
- char *buf)
- {
- struct w1_slave *sl = dev_to_w1_slave(dev);
- int result;
- /* Read current speed from slave. Updates data->speed. */
- result = w1_f19_get_i2c_speed(sl);
- if (result < 0)
- return result;
- /* Return current speed value. */
- return sprintf(buf, "%d\n", result);
- }
- static ssize_t speed_store(struct device *dev, struct device_attribute *attr,
- const char *buf, size_t count)
- {
- struct w1_slave *sl = dev_to_w1_slave(dev);
- int error;
- /* Valid values are: "100", "400", "900" */
- if (count < 3 || count > 4 || !buf)
- return -EINVAL;
- if (count == 4 && buf[3] != '\n')
- return -EINVAL;
- if (buf[1] != '0' || buf[2] != '0')
- return -EINVAL;
- /* Set speed on slave. */
- switch (buf[0]) {
- case '1':
- error = w1_f19_set_i2c_speed(sl, 0);
- break;
- case '4':
- error = w1_f19_set_i2c_speed(sl, 1);
- break;
- case '9':
- error = w1_f19_set_i2c_speed(sl, 2);
- break;
- default:
- return -EINVAL;
- }
- if (error < 0)
- return error;
- /* Return bytes written. */
- return count;
- }
- static DEVICE_ATTR_RW(speed);
- /* Busy stretch attribute for a single chip. */
- static ssize_t stretch_show(struct device *dev, struct device_attribute *attr,
- char *buf)
- {
- struct w1_slave *sl = dev_to_w1_slave(dev);
- struct w1_f19_data *data = sl->family_data;
- /* Return current stretch value. */
- return sprintf(buf, "%d\n", data->stretch);
- }
- static ssize_t stretch_store(struct device *dev, struct device_attribute *attr,
- const char *buf, size_t count)
- {
- struct w1_slave *sl = dev_to_w1_slave(dev);
- struct w1_f19_data *data = sl->family_data;
- /* Valid values are '1' to '9' */
- if (count < 1 || count > 2 || !buf)
- return -EINVAL;
- if (count == 2 && buf[1] != '\n')
- return -EINVAL;
- if (buf[0] < '1' || buf[0] > '9')
- return -EINVAL;
- /* Set busy stretch value. */
- data->stretch = buf[0] & 0x0F;
- /* Return bytes written. */
- return count;
- }
- static DEVICE_ATTR_RW(stretch);
- /* All attributes. */
- static struct attribute *w1_f19_attrs[] = {
- &dev_attr_speed.attr,
- &dev_attr_stretch.attr,
- NULL,
- };
- static const struct attribute_group w1_f19_group = {
- .attrs = w1_f19_attrs,
- };
- static const struct attribute_group *w1_f19_groups[] = {
- &w1_f19_group,
- NULL,
- };
- /* Slave add and remove functions. */
- static int w1_f19_add_slave(struct w1_slave *sl)
- {
- struct w1_f19_data *data = NULL;
- /* Allocate memory for slave specific data. */
- data = devm_kzalloc(&sl->dev, sizeof(*data), GFP_KERNEL);
- if (!data)
- return -ENOMEM;
- sl->family_data = data;
- /* Setup default I2C speed on slave. */
- switch (i2c_speed) {
- case 100:
- __w1_f19_set_i2c_speed(sl, 0);
- break;
- case 400:
- __w1_f19_set_i2c_speed(sl, 1);
- break;
- case 900:
- __w1_f19_set_i2c_speed(sl, 2);
- break;
- default:
- /*
- * A i2c_speed module parameter of anything else
- * than 100, 400, 900 means not to touch the
- * speed of the DS28E17.
- * We assume 400kBaud, the power-on value.
- */
- data->speed = 1;
- }
- /*
- * Setup default busy stretch
- * configuration for the DS28E17.
- */
- data->stretch = i2c_stretch;
- /* Setup I2C adapter. */
- data->adapter.owner = THIS_MODULE;
- data->adapter.algo = &w1_f19_i2c_algorithm;
- data->adapter.algo_data = sl;
- strcpy(data->adapter.name, "w1-");
- strcat(data->adapter.name, sl->name);
- data->adapter.dev.parent = &sl->dev;
- data->adapter.quirks = &w1_f19_i2c_adapter_quirks;
- return i2c_add_adapter(&data->adapter);
- }
- static void w1_f19_remove_slave(struct w1_slave *sl)
- {
- struct w1_f19_data *family_data = sl->family_data;
- /* Delete I2C adapter. */
- i2c_del_adapter(&family_data->adapter);
- /* Free slave specific data. */
- devm_kfree(&sl->dev, family_data);
- sl->family_data = NULL;
- }
- /* Declarations within the w1 subsystem. */
- static struct w1_family_ops w1_f19_fops = {
- .add_slave = w1_f19_add_slave,
- .remove_slave = w1_f19_remove_slave,
- .groups = w1_f19_groups,
- };
- static struct w1_family w1_family_19 = {
- .fid = W1_FAMILY_DS28E17,
- .fops = &w1_f19_fops,
- };
- /* Module init and remove functions. */
- static int __init w1_f19_init(void)
- {
- return w1_register_family(&w1_family_19);
- }
- static void __exit w1_f19_fini(void)
- {
- w1_unregister_family(&w1_family_19);
- }
- module_init(w1_f19_init);
- module_exit(w1_f19_fini);
|