1 // SPDX-License-Identifier: GPL-2.0
3 * w1_ds250x.c - w1 family 09/0b/89/91 (DS250x) driver
6 #include <linux/kernel.h>
7 #include <linux/module.h>
8 #include <linux/moduleparam.h>
9 #include <linux/device.h>
10 #include <linux/types.h>
11 #include <linux/delay.h>
12 #include <linux/slab.h>
13 #include <linux/crc16.h>
16 #include <linux/nvmem-provider.h>
18 #define W1_DS2501_UNW_FAMILY 0x91
19 #define W1_DS2501_SIZE 64
21 #define W1_DS2502_FAMILY 0x09
22 #define W1_DS2502_UNW_FAMILY 0x89
23 #define W1_DS2502_SIZE 128
25 #define W1_DS2505_FAMILY 0x0b
26 #define W1_DS2505_SIZE 2048
28 #define W1_PAGE_SIZE 32
30 #define W1_EXT_READ_MEMORY 0xA5
31 #define W1_READ_DATA_CRC 0xC3
33 #define OFF2PG(off) ((off) / W1_PAGE_SIZE)
36 #define CRC16_VALID 0xb001
38 struct w1_eprom_data
{
40 int (*read
)(struct w1_slave
*sl
, int pageno
);
41 u8 eprom
[W1_DS2505_SIZE
];
42 DECLARE_BITMAP(page_present
, W1_DS2505_SIZE
/ W1_PAGE_SIZE
);
46 static int w1_ds2502_read_page(struct w1_slave
*sl
, int pageno
)
48 struct w1_eprom_data
*data
= sl
->family_data
;
49 int pgoff
= pageno
* W1_PAGE_SIZE
;
54 if (test_bit(pageno
, data
->page_present
))
55 return 0; /* page already present */
57 mutex_lock(&sl
->master
->bus_mutex
);
59 if (w1_reset_select_slave(sl
))
62 buf
[0] = W1_READ_DATA_CRC
;
63 buf
[1] = pgoff
& 0xff;
65 w1_write_block(sl
->master
, buf
, 3);
67 crc8
= w1_read_8(sl
->master
);
68 if (w1_calc_crc8(buf
, 3) != crc8
)
71 w1_read_block(sl
->master
, &data
->eprom
[pgoff
], W1_PAGE_SIZE
);
73 crc8
= w1_read_8(sl
->master
);
74 if (w1_calc_crc8(&data
->eprom
[pgoff
], W1_PAGE_SIZE
) != crc8
)
77 set_bit(pageno
, data
->page_present
); /* mark page present */
80 mutex_unlock(&sl
->master
->bus_mutex
);
84 static int w1_ds2505_read_page(struct w1_slave
*sl
, int pageno
)
86 struct w1_eprom_data
*data
= sl
->family_data
;
87 int redir_retries
= 16;
94 if (test_bit(pageno
, data
->page_present
))
95 return 0; /* page already present */
97 epoff
= pgoff
= pageno
* W1_PAGE_SIZE
;
98 mutex_lock(&sl
->master
->bus_mutex
);
101 if (w1_reset_select_slave(sl
))
104 buf
[0] = W1_EXT_READ_MEMORY
;
105 buf
[1] = pgoff
& 0xff;
107 w1_write_block(sl
->master
, buf
, 3);
108 w1_read_block(sl
->master
, buf
+ 3, 3); /* redir, crc16 */
110 crc
= crc16(CRC16_INIT
, buf
, 6);
112 if (crc
!= CRC16_VALID
)
118 if (redir_retries
< 0)
121 pgoff
= (redir
^ 0xff) * W1_PAGE_SIZE
;
125 w1_read_block(sl
->master
, &data
->eprom
[epoff
], W1_PAGE_SIZE
);
126 w1_read_block(sl
->master
, buf
, 2); /* crc16 */
127 crc
= crc16(CRC16_INIT
, &data
->eprom
[epoff
], W1_PAGE_SIZE
);
128 crc
= crc16(crc
, buf
, 2);
130 if (crc
!= CRC16_VALID
)
133 set_bit(pageno
, data
->page_present
);
136 mutex_unlock(&sl
->master
->bus_mutex
);
140 static int w1_nvmem_read(void *priv
, unsigned int off
, void *buf
, size_t count
)
142 struct w1_slave
*sl
= priv
;
143 struct w1_eprom_data
*data
= sl
->family_data
;
144 size_t eprom_size
= data
->size
;
148 if (off
> eprom_size
)
151 if ((off
+ count
) > eprom_size
)
152 count
= eprom_size
- off
;
156 ret
= data
->read(sl
, i
++);
159 } while (i
< OFF2PG(off
+ count
));
161 memcpy(buf
, &data
->eprom
[off
], count
);
165 static int w1_eprom_add_slave(struct w1_slave
*sl
)
167 struct w1_eprom_data
*data
;
168 struct nvmem_device
*nvmem
;
169 struct nvmem_config nvmem_cfg
= {
171 .add_legacy_fixed_of_cells
= true,
172 .reg_read
= w1_nvmem_read
,
173 .type
= NVMEM_TYPE_OTP
,
180 data
= devm_kzalloc(&sl
->dev
, sizeof(struct w1_eprom_data
), GFP_KERNEL
);
184 sl
->family_data
= data
;
185 switch (sl
->family
->fid
) {
186 case W1_DS2501_UNW_FAMILY
:
187 data
->size
= W1_DS2501_SIZE
;
188 data
->read
= w1_ds2502_read_page
;
190 case W1_DS2502_FAMILY
:
191 case W1_DS2502_UNW_FAMILY
:
192 data
->size
= W1_DS2502_SIZE
;
193 data
->read
= w1_ds2502_read_page
;
195 case W1_DS2505_FAMILY
:
196 data
->size
= W1_DS2505_SIZE
;
197 data
->read
= w1_ds2505_read_page
;
201 if (sl
->master
->bus_master
->dev_id
)
202 snprintf(data
->nvmem_name
, sizeof(data
->nvmem_name
),
204 sl
->master
->bus_master
->dev_id
, sl
->reg_num
.family
,
205 (unsigned long long)sl
->reg_num
.id
);
207 snprintf(data
->nvmem_name
, sizeof(data
->nvmem_name
),
210 (unsigned long long)sl
->reg_num
.id
);
212 nvmem_cfg
.name
= data
->nvmem_name
;
213 nvmem_cfg
.size
= data
->size
;
215 nvmem
= devm_nvmem_register(&sl
->dev
, &nvmem_cfg
);
216 return PTR_ERR_OR_ZERO(nvmem
);
219 static const struct w1_family_ops w1_eprom_fops
= {
220 .add_slave
= w1_eprom_add_slave
,
223 static struct w1_family w1_family_09
= {
224 .fid
= W1_DS2502_FAMILY
,
225 .fops
= &w1_eprom_fops
,
228 static struct w1_family w1_family_0b
= {
229 .fid
= W1_DS2505_FAMILY
,
230 .fops
= &w1_eprom_fops
,
233 static struct w1_family w1_family_89
= {
234 .fid
= W1_DS2502_UNW_FAMILY
,
235 .fops
= &w1_eprom_fops
,
238 static struct w1_family w1_family_91
= {
239 .fid
= W1_DS2501_UNW_FAMILY
,
240 .fops
= &w1_eprom_fops
,
243 static int __init
w1_ds250x_init(void)
247 err
= w1_register_family(&w1_family_09
);
251 err
= w1_register_family(&w1_family_0b
);
255 err
= w1_register_family(&w1_family_89
);
259 err
= w1_register_family(&w1_family_91
);
266 w1_unregister_family(&w1_family_89
);
268 w1_unregister_family(&w1_family_0b
);
270 w1_unregister_family(&w1_family_09
);
274 static void __exit
w1_ds250x_exit(void)
276 w1_unregister_family(&w1_family_09
);
277 w1_unregister_family(&w1_family_0b
);
278 w1_unregister_family(&w1_family_89
);
279 w1_unregister_family(&w1_family_91
);
282 module_init(w1_ds250x_init
);
283 module_exit(w1_ds250x_exit
);
285 MODULE_AUTHOR("Thomas Bogendoerfer <tbogendoerfe@suse.de>");
286 MODULE_DESCRIPTION("w1 family driver for DS250x Add Only Memory");
287 MODULE_LICENSE("GPL");
288 MODULE_ALIAS("w1-family-" __stringify(W1_DS2502_FAMILY
));
289 MODULE_ALIAS("w1-family-" __stringify(W1_DS2505_FAMILY
));
290 MODULE_ALIAS("w1-family-" __stringify(W1_DS2501_UNW_FAMILY
));
291 MODULE_ALIAS("w1-family-" __stringify(W1_DS2502_UNW_FAMILY
));