1 // SPDX-License-Identifier: GPL-2.0
3 * Microchip / Atmel SHA204A (I2C) driver.
5 * Copyright (c) 2019 Linaro, Ltd. <ard.biesheuvel@linaro.org>
8 #include <linux/delay.h>
9 #include <linux/device.h>
10 #include <linux/err.h>
11 #include <linux/errno.h>
12 #include <linux/i2c.h>
13 #include <linux/init.h>
14 #include <linux/kernel.h>
15 #include <linux/module.h>
16 #include <linux/scatterlist.h>
17 #include <linux/slab.h>
18 #include <linux/workqueue.h>
19 #include "atmel-i2c.h"
21 static void atmel_sha204a_rng_done(struct atmel_i2c_work_data
*work_data
,
22 void *areq
, int status
)
24 struct atmel_i2c_client_priv
*i2c_priv
= work_data
->ctx
;
25 struct hwrng
*rng
= areq
;
28 dev_warn_ratelimited(&i2c_priv
->client
->dev
,
29 "i2c transaction failed (%d)\n",
32 rng
->priv
= (unsigned long)work_data
;
33 atomic_dec(&i2c_priv
->tfm_count
);
36 static int atmel_sha204a_rng_read_nonblocking(struct hwrng
*rng
, void *data
,
39 struct atmel_i2c_client_priv
*i2c_priv
;
40 struct atmel_i2c_work_data
*work_data
;
42 i2c_priv
= container_of(rng
, struct atmel_i2c_client_priv
, hwrng
);
44 /* keep maximum 1 asynchronous read in flight at any time */
45 if (!atomic_add_unless(&i2c_priv
->tfm_count
, 1, 1))
49 work_data
= (struct atmel_i2c_work_data
*)rng
->priv
;
50 max
= min(sizeof(work_data
->cmd
.data
), max
);
51 memcpy(data
, &work_data
->cmd
.data
, max
);
54 work_data
= kmalloc(sizeof(*work_data
), GFP_ATOMIC
);
58 work_data
->ctx
= i2c_priv
;
59 work_data
->client
= i2c_priv
->client
;
64 atmel_i2c_init_random_cmd(&work_data
->cmd
);
65 atmel_i2c_enqueue(work_data
, atmel_sha204a_rng_done
, rng
);
70 static int atmel_sha204a_rng_read(struct hwrng
*rng
, void *data
, size_t max
,
73 struct atmel_i2c_client_priv
*i2c_priv
;
74 struct atmel_i2c_cmd cmd
;
78 return atmel_sha204a_rng_read_nonblocking(rng
, data
, max
);
80 i2c_priv
= container_of(rng
, struct atmel_i2c_client_priv
, hwrng
);
82 atmel_i2c_init_random_cmd(&cmd
);
84 ret
= atmel_i2c_send_receive(i2c_priv
->client
, &cmd
);
88 max
= min(sizeof(cmd
.data
), max
);
89 memcpy(data
, cmd
.data
, max
);
94 static int atmel_sha204a_probe(struct i2c_client
*client
,
95 const struct i2c_device_id
*id
)
97 struct atmel_i2c_client_priv
*i2c_priv
;
100 ret
= atmel_i2c_probe(client
, id
);
104 i2c_priv
= i2c_get_clientdata(client
);
106 memset(&i2c_priv
->hwrng
, 0, sizeof(i2c_priv
->hwrng
));
108 i2c_priv
->hwrng
.name
= dev_name(&client
->dev
);
109 i2c_priv
->hwrng
.read
= atmel_sha204a_rng_read
;
110 i2c_priv
->hwrng
.quality
= 1024;
112 ret
= devm_hwrng_register(&client
->dev
, &i2c_priv
->hwrng
);
114 dev_warn(&client
->dev
, "failed to register RNG (%d)\n", ret
);
119 static int atmel_sha204a_remove(struct i2c_client
*client
)
121 struct atmel_i2c_client_priv
*i2c_priv
= i2c_get_clientdata(client
);
123 if (atomic_read(&i2c_priv
->tfm_count
)) {
124 dev_err(&client
->dev
, "Device is busy\n");
128 if (i2c_priv
->hwrng
.priv
)
129 kfree((void *)i2c_priv
->hwrng
.priv
);
134 static const struct of_device_id atmel_sha204a_dt_ids
[] = {
135 { .compatible
= "atmel,atsha204a", },
138 MODULE_DEVICE_TABLE(of
, atmel_sha204a_dt_ids
);
140 static const struct i2c_device_id atmel_sha204a_id
[] = {
144 MODULE_DEVICE_TABLE(i2c
, atmel_sha204a_id
);
146 static struct i2c_driver atmel_sha204a_driver
= {
147 .probe
= atmel_sha204a_probe
,
148 .remove
= atmel_sha204a_remove
,
149 .id_table
= atmel_sha204a_id
,
151 .driver
.name
= "atmel-sha204a",
152 .driver
.of_match_table
= of_match_ptr(atmel_sha204a_dt_ids
),
155 static int __init
atmel_sha204a_init(void)
157 return i2c_add_driver(&atmel_sha204a_driver
);
160 static void __exit
atmel_sha204a_exit(void)
162 flush_scheduled_work();
163 i2c_del_driver(&atmel_sha204a_driver
);
166 module_init(atmel_sha204a_init
);
167 module_exit(atmel_sha204a_exit
);
169 MODULE_AUTHOR("Ard Biesheuvel <ard.biesheuvel@linaro.org>");
170 MODULE_LICENSE("GPL v2");