155766568SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only 2bbcb478eSHarald Freudenberger /* 3bbcb478eSHarald Freudenberger * s390 TRNG device driver 4bbcb478eSHarald Freudenberger * 5bbcb478eSHarald Freudenberger * Driver for the TRNG (true random number generation) command 6bbcb478eSHarald Freudenberger * available via CPACF extension MSA 7 on the s390 arch. 7bbcb478eSHarald Freudenberger 8bbcb478eSHarald Freudenberger * Copyright IBM Corp. 2017 9bbcb478eSHarald Freudenberger * Author(s): Harald Freudenberger <freude@de.ibm.com> 10bbcb478eSHarald Freudenberger */ 11bbcb478eSHarald Freudenberger 12bbcb478eSHarald Freudenberger #define KMSG_COMPONENT "trng" 13bbcb478eSHarald Freudenberger #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt 14bbcb478eSHarald Freudenberger 15bbcb478eSHarald Freudenberger #include <linux/hw_random.h> 16bbcb478eSHarald Freudenberger #include <linux/kernel.h> 17bbcb478eSHarald Freudenberger #include <linux/module.h> 18bbcb478eSHarald Freudenberger #include <linux/cpufeature.h> 19bbcb478eSHarald Freudenberger #include <linux/miscdevice.h> 20bbcb478eSHarald Freudenberger #include <linux/debugfs.h> 21bbcb478eSHarald Freudenberger #include <linux/atomic.h> 22bbcb478eSHarald Freudenberger #include <linux/random.h> 23bbcb478eSHarald Freudenberger #include <linux/sched/signal.h> 24bbcb478eSHarald Freudenberger #include <asm/debug.h> 25bbcb478eSHarald Freudenberger #include <asm/cpacf.h> 26*6bb20c15SJason A. Donenfeld #include <asm/archrandom.h> 27bbcb478eSHarald Freudenberger 28bbcb478eSHarald Freudenberger MODULE_LICENSE("GPL v2"); 29bbcb478eSHarald Freudenberger MODULE_AUTHOR("IBM Corporation"); 30bbcb478eSHarald Freudenberger MODULE_DESCRIPTION("s390 CPACF TRNG device driver"); 31bbcb478eSHarald Freudenberger 32bbcb478eSHarald Freudenberger 33bbcb478eSHarald Freudenberger /* trng related debug feature things */ 34bbcb478eSHarald Freudenberger 35bbcb478eSHarald Freudenberger static debug_info_t *debug_info; 36bbcb478eSHarald Freudenberger 37bbcb478eSHarald Freudenberger #define DEBUG_DBG(...) debug_sprintf_event(debug_info, 6, ##__VA_ARGS__) 38bbcb478eSHarald Freudenberger #define DEBUG_INFO(...) debug_sprintf_event(debug_info, 5, ##__VA_ARGS__) 39bbcb478eSHarald Freudenberger #define DEBUG_WARN(...) debug_sprintf_event(debug_info, 4, ##__VA_ARGS__) 40bbcb478eSHarald Freudenberger #define DEBUG_ERR(...) debug_sprintf_event(debug_info, 3, ##__VA_ARGS__) 41bbcb478eSHarald Freudenberger 42bbcb478eSHarald Freudenberger 43bbcb478eSHarald Freudenberger /* trng helpers */ 44bbcb478eSHarald Freudenberger 45bbcb478eSHarald Freudenberger static atomic64_t trng_dev_counter = ATOMIC64_INIT(0); 46bbcb478eSHarald Freudenberger static atomic64_t trng_hwrng_counter = ATOMIC64_INIT(0); 47bbcb478eSHarald Freudenberger 48bbcb478eSHarald Freudenberger 49bbcb478eSHarald Freudenberger /* file io functions */ 50bbcb478eSHarald Freudenberger 51bbcb478eSHarald Freudenberger static int trng_open(struct inode *inode, struct file *file) 52bbcb478eSHarald Freudenberger { 53bbcb478eSHarald Freudenberger return nonseekable_open(inode, file); 54bbcb478eSHarald Freudenberger } 55bbcb478eSHarald Freudenberger 56bbcb478eSHarald Freudenberger static ssize_t trng_read(struct file *file, char __user *ubuf, 57bbcb478eSHarald Freudenberger size_t nbytes, loff_t *ppos) 58bbcb478eSHarald Freudenberger { 59bbcb478eSHarald Freudenberger u8 buf[32]; 60bbcb478eSHarald Freudenberger u8 *p = buf; 61bbcb478eSHarald Freudenberger unsigned int n; 62bbcb478eSHarald Freudenberger ssize_t ret = 0; 63bbcb478eSHarald Freudenberger 64bbcb478eSHarald Freudenberger /* 65bbcb478eSHarald Freudenberger * use buf for requests <= sizeof(buf), 66bbcb478eSHarald Freudenberger * otherwise allocate one page and fetch 67bbcb478eSHarald Freudenberger * pagewise. 68bbcb478eSHarald Freudenberger */ 69bbcb478eSHarald Freudenberger 70bbcb478eSHarald Freudenberger if (nbytes > sizeof(buf)) { 71bbcb478eSHarald Freudenberger p = (u8 *) __get_free_page(GFP_KERNEL); 72bbcb478eSHarald Freudenberger if (!p) 73bbcb478eSHarald Freudenberger return -ENOMEM; 74bbcb478eSHarald Freudenberger } 75bbcb478eSHarald Freudenberger 76bbcb478eSHarald Freudenberger while (nbytes) { 77bbcb478eSHarald Freudenberger if (need_resched()) { 78bbcb478eSHarald Freudenberger if (signal_pending(current)) { 79bbcb478eSHarald Freudenberger if (ret == 0) 80bbcb478eSHarald Freudenberger ret = -ERESTARTSYS; 81bbcb478eSHarald Freudenberger break; 82bbcb478eSHarald Freudenberger } 83bbcb478eSHarald Freudenberger schedule(); 84bbcb478eSHarald Freudenberger } 85bbcb478eSHarald Freudenberger n = nbytes > PAGE_SIZE ? PAGE_SIZE : nbytes; 86bbcb478eSHarald Freudenberger cpacf_trng(NULL, 0, p, n); 87bbcb478eSHarald Freudenberger atomic64_add(n, &trng_dev_counter); 88bbcb478eSHarald Freudenberger if (copy_to_user(ubuf, p, n)) { 89bbcb478eSHarald Freudenberger ret = -EFAULT; 90bbcb478eSHarald Freudenberger break; 91bbcb478eSHarald Freudenberger } 92bbcb478eSHarald Freudenberger nbytes -= n; 93bbcb478eSHarald Freudenberger ubuf += n; 94bbcb478eSHarald Freudenberger ret += n; 95bbcb478eSHarald Freudenberger } 96bbcb478eSHarald Freudenberger 97bbcb478eSHarald Freudenberger if (p != buf) 98bbcb478eSHarald Freudenberger free_page((unsigned long) p); 99bbcb478eSHarald Freudenberger 100bbcb478eSHarald Freudenberger DEBUG_DBG("trng_read()=%zd\n", ret); 101bbcb478eSHarald Freudenberger return ret; 102bbcb478eSHarald Freudenberger } 103bbcb478eSHarald Freudenberger 104bbcb478eSHarald Freudenberger 105bbcb478eSHarald Freudenberger /* sysfs */ 106bbcb478eSHarald Freudenberger 107bbcb478eSHarald Freudenberger static ssize_t trng_counter_show(struct device *dev, 108bbcb478eSHarald Freudenberger struct device_attribute *attr, char *buf) 109bbcb478eSHarald Freudenberger { 110bbcb478eSHarald Freudenberger u64 dev_counter = atomic64_read(&trng_dev_counter); 111bbcb478eSHarald Freudenberger u64 hwrng_counter = atomic64_read(&trng_hwrng_counter); 112bbcb478eSHarald Freudenberger u64 arch_counter = atomic64_read(&s390_arch_random_counter); 113bbcb478eSHarald Freudenberger 1147e75c337SQing Wang return sysfs_emit(buf, 115bbcb478eSHarald Freudenberger "trng: %llu\n" 116bbcb478eSHarald Freudenberger "hwrng: %llu\n" 117bbcb478eSHarald Freudenberger "arch: %llu\n" 118bbcb478eSHarald Freudenberger "total: %llu\n", 119bbcb478eSHarald Freudenberger dev_counter, hwrng_counter, arch_counter, 120bbcb478eSHarald Freudenberger dev_counter + hwrng_counter + arch_counter); 121bbcb478eSHarald Freudenberger } 122bbcb478eSHarald Freudenberger static DEVICE_ATTR(byte_counter, 0444, trng_counter_show, NULL); 123bbcb478eSHarald Freudenberger 124bbcb478eSHarald Freudenberger static struct attribute *trng_dev_attrs[] = { 125bbcb478eSHarald Freudenberger &dev_attr_byte_counter.attr, 126bbcb478eSHarald Freudenberger NULL 127bbcb478eSHarald Freudenberger }; 128bbcb478eSHarald Freudenberger 129bbcb478eSHarald Freudenberger static const struct attribute_group trng_dev_attr_group = { 130bbcb478eSHarald Freudenberger .attrs = trng_dev_attrs 131bbcb478eSHarald Freudenberger }; 132bbcb478eSHarald Freudenberger 133bbcb478eSHarald Freudenberger static const struct attribute_group *trng_dev_attr_groups[] = { 134bbcb478eSHarald Freudenberger &trng_dev_attr_group, 135bbcb478eSHarald Freudenberger NULL 136bbcb478eSHarald Freudenberger }; 137bbcb478eSHarald Freudenberger 138bbcb478eSHarald Freudenberger static const struct file_operations trng_fops = { 139bbcb478eSHarald Freudenberger .owner = THIS_MODULE, 140bbcb478eSHarald Freudenberger .open = &trng_open, 141bbcb478eSHarald Freudenberger .release = NULL, 142bbcb478eSHarald Freudenberger .read = &trng_read, 143bbcb478eSHarald Freudenberger .llseek = noop_llseek, 144bbcb478eSHarald Freudenberger }; 145bbcb478eSHarald Freudenberger 146bbcb478eSHarald Freudenberger static struct miscdevice trng_dev = { 147bbcb478eSHarald Freudenberger .name = "trng", 148bbcb478eSHarald Freudenberger .minor = MISC_DYNAMIC_MINOR, 149bbcb478eSHarald Freudenberger .mode = 0444, 150bbcb478eSHarald Freudenberger .fops = &trng_fops, 151bbcb478eSHarald Freudenberger .groups = trng_dev_attr_groups, 152bbcb478eSHarald Freudenberger }; 153bbcb478eSHarald Freudenberger 154bbcb478eSHarald Freudenberger 155bbcb478eSHarald Freudenberger /* hwrng_register */ 156bbcb478eSHarald Freudenberger 157bbcb478eSHarald Freudenberger static inline void _trng_hwrng_read(u8 *buf, size_t len) 158bbcb478eSHarald Freudenberger { 159bbcb478eSHarald Freudenberger cpacf_trng(NULL, 0, buf, len); 160bbcb478eSHarald Freudenberger atomic64_add(len, &trng_hwrng_counter); 161bbcb478eSHarald Freudenberger } 162bbcb478eSHarald Freudenberger 163bbcb478eSHarald Freudenberger static int trng_hwrng_data_read(struct hwrng *rng, u32 *data) 164bbcb478eSHarald Freudenberger { 165bbcb478eSHarald Freudenberger size_t len = sizeof(*data); 166bbcb478eSHarald Freudenberger 167bbcb478eSHarald Freudenberger _trng_hwrng_read((u8 *) data, len); 168bbcb478eSHarald Freudenberger 169bbcb478eSHarald Freudenberger DEBUG_DBG("trng_hwrng_data_read()=%zu\n", len); 170bbcb478eSHarald Freudenberger 171bbcb478eSHarald Freudenberger return len; 172bbcb478eSHarald Freudenberger } 173bbcb478eSHarald Freudenberger 174bbcb478eSHarald Freudenberger static int trng_hwrng_read(struct hwrng *rng, void *data, size_t max, bool wait) 175bbcb478eSHarald Freudenberger { 176bbcb478eSHarald Freudenberger size_t len = max <= PAGE_SIZE ? max : PAGE_SIZE; 177bbcb478eSHarald Freudenberger 178bbcb478eSHarald Freudenberger _trng_hwrng_read((u8 *) data, len); 179bbcb478eSHarald Freudenberger 180bbcb478eSHarald Freudenberger DEBUG_DBG("trng_hwrng_read()=%zu\n", len); 181bbcb478eSHarald Freudenberger 182bbcb478eSHarald Freudenberger return len; 183bbcb478eSHarald Freudenberger } 184bbcb478eSHarald Freudenberger 185bbcb478eSHarald Freudenberger /* 186bbcb478eSHarald Freudenberger * hwrng register struct 187d041315eSChristian Borntraeger * The trng is supposed to have 100% entropy, and thus we register with a very 188d041315eSChristian Borntraeger * high quality value. If we ever have a better driver in the future, we should 189d041315eSChristian Borntraeger * change this value again when we merge this driver. 190bbcb478eSHarald Freudenberger */ 191bbcb478eSHarald Freudenberger static struct hwrng trng_hwrng_dev = { 192bbcb478eSHarald Freudenberger .name = "s390-trng", 193bbcb478eSHarald Freudenberger .data_read = trng_hwrng_data_read, 194bbcb478eSHarald Freudenberger .read = trng_hwrng_read, 195bbcb478eSHarald Freudenberger }; 196bbcb478eSHarald Freudenberger 197bbcb478eSHarald Freudenberger 198bbcb478eSHarald Freudenberger /* init and exit */ 199bbcb478eSHarald Freudenberger 200bbcb478eSHarald Freudenberger static void __init trng_debug_init(void) 201bbcb478eSHarald Freudenberger { 202bbcb478eSHarald Freudenberger debug_info = debug_register("trng", 1, 1, 4 * sizeof(long)); 203bbcb478eSHarald Freudenberger debug_register_view(debug_info, &debug_sprintf_view); 204bbcb478eSHarald Freudenberger debug_set_level(debug_info, 3); 205bbcb478eSHarald Freudenberger } 206bbcb478eSHarald Freudenberger 207bbcb478eSHarald Freudenberger static void trng_debug_exit(void) 208bbcb478eSHarald Freudenberger { 209bbcb478eSHarald Freudenberger debug_unregister(debug_info); 210bbcb478eSHarald Freudenberger } 211bbcb478eSHarald Freudenberger 212bbcb478eSHarald Freudenberger static int __init trng_init(void) 213bbcb478eSHarald Freudenberger { 214bbcb478eSHarald Freudenberger int ret; 215bbcb478eSHarald Freudenberger 216bbcb478eSHarald Freudenberger trng_debug_init(); 217bbcb478eSHarald Freudenberger 218bbcb478eSHarald Freudenberger /* check if subfunction CPACF_PRNO_TRNG is available */ 219bbcb478eSHarald Freudenberger if (!cpacf_query_func(CPACF_PRNO, CPACF_PRNO_TRNG)) { 220bbcb478eSHarald Freudenberger DEBUG_INFO("trng_init CPACF_PRNO_TRNG not available\n"); 221bbcb478eSHarald Freudenberger ret = -ENODEV; 222bbcb478eSHarald Freudenberger goto out_dbg; 223bbcb478eSHarald Freudenberger } 224bbcb478eSHarald Freudenberger 225bbcb478eSHarald Freudenberger ret = misc_register(&trng_dev); 226bbcb478eSHarald Freudenberger if (ret) { 227bbcb478eSHarald Freudenberger DEBUG_WARN("trng_init misc_register() failed rc=%d\n", ret); 228bbcb478eSHarald Freudenberger goto out_dbg; 229bbcb478eSHarald Freudenberger } 230bbcb478eSHarald Freudenberger 231bbcb478eSHarald Freudenberger ret = hwrng_register(&trng_hwrng_dev); 232bbcb478eSHarald Freudenberger if (ret) { 233bbcb478eSHarald Freudenberger DEBUG_WARN("trng_init hwrng_register() failed rc=%d\n", ret); 234bbcb478eSHarald Freudenberger goto out_misc; 235bbcb478eSHarald Freudenberger } 236bbcb478eSHarald Freudenberger 237bbcb478eSHarald Freudenberger DEBUG_DBG("trng_init successful\n"); 238bbcb478eSHarald Freudenberger 239bbcb478eSHarald Freudenberger return 0; 240bbcb478eSHarald Freudenberger 241bbcb478eSHarald Freudenberger out_misc: 242bbcb478eSHarald Freudenberger misc_deregister(&trng_dev); 243bbcb478eSHarald Freudenberger out_dbg: 244bbcb478eSHarald Freudenberger trng_debug_exit(); 245bbcb478eSHarald Freudenberger return ret; 246bbcb478eSHarald Freudenberger } 247bbcb478eSHarald Freudenberger 248bbcb478eSHarald Freudenberger static void __exit trng_exit(void) 249bbcb478eSHarald Freudenberger { 250bbcb478eSHarald Freudenberger hwrng_unregister(&trng_hwrng_dev); 251bbcb478eSHarald Freudenberger misc_deregister(&trng_dev); 252bbcb478eSHarald Freudenberger trng_debug_exit(); 253bbcb478eSHarald Freudenberger } 254bbcb478eSHarald Freudenberger 2550a5f9b38SHeiko Carstens module_cpu_feature_match(S390_CPU_FEATURE_MSA, trng_init); 256bbcb478eSHarald Freudenberger module_exit(trng_exit); 257