123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119 |
- /*
- * RTC subsystem, nvmem interface
- *
- * Copyright (C) 2017 Alexandre Belloni
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
- #include <linux/err.h>
- #include <linux/types.h>
- #include <linux/nvmem-consumer.h>
- #include <linux/rtc.h>
- #include <linux/sysfs.h>
- /*
- * Deprecated ABI compatibility, this should be removed at some point
- */
- static const char nvram_warning[] = "Deprecated ABI, please use nvmem";
- static ssize_t
- rtc_nvram_read(struct file *filp, struct kobject *kobj,
- struct bin_attribute *attr,
- char *buf, loff_t off, size_t count)
- {
- struct rtc_device *rtc = attr->private;
- dev_warn_once(kobj_to_dev(kobj), nvram_warning);
- return nvmem_device_read(rtc->nvmem, off, count, buf);
- }
- static ssize_t
- rtc_nvram_write(struct file *filp, struct kobject *kobj,
- struct bin_attribute *attr,
- char *buf, loff_t off, size_t count)
- {
- struct rtc_device *rtc = attr->private;
- dev_warn_once(kobj_to_dev(kobj), nvram_warning);
- return nvmem_device_write(rtc->nvmem, off, count, buf);
- }
- static int rtc_nvram_register(struct rtc_device *rtc, size_t size)
- {
- int err;
- rtc->nvram = devm_kzalloc(rtc->dev.parent,
- sizeof(struct bin_attribute),
- GFP_KERNEL);
- if (!rtc->nvram)
- return -ENOMEM;
- rtc->nvram->attr.name = "nvram";
- rtc->nvram->attr.mode = 0644;
- rtc->nvram->private = rtc;
- sysfs_bin_attr_init(rtc->nvram);
- rtc->nvram->read = rtc_nvram_read;
- rtc->nvram->write = rtc_nvram_write;
- rtc->nvram->size = size;
- err = sysfs_create_bin_file(&rtc->dev.parent->kobj,
- rtc->nvram);
- if (err) {
- devm_kfree(rtc->dev.parent, rtc->nvram);
- rtc->nvram = NULL;
- }
- return err;
- }
- static void rtc_nvram_unregister(struct rtc_device *rtc)
- {
- sysfs_remove_bin_file(&rtc->dev.parent->kobj, rtc->nvram);
- }
- /*
- * New ABI, uses nvmem
- */
- int rtc_nvmem_register(struct rtc_device *rtc,
- struct nvmem_config *nvmem_config)
- {
- if (!IS_ERR_OR_NULL(rtc->nvmem))
- return -EBUSY;
- if (!nvmem_config)
- return -ENODEV;
- nvmem_config->dev = rtc->dev.parent;
- nvmem_config->owner = rtc->owner;
- rtc->nvmem = nvmem_register(nvmem_config);
- if (IS_ERR(rtc->nvmem))
- return PTR_ERR(rtc->nvmem);
- /* Register the old ABI */
- if (rtc->nvram_old_abi)
- rtc_nvram_register(rtc, nvmem_config->size);
- return 0;
- }
- EXPORT_SYMBOL_GPL(rtc_nvmem_register);
- void rtc_nvmem_unregister(struct rtc_device *rtc)
- {
- if (IS_ERR_OR_NULL(rtc->nvmem))
- return;
- /* unregister the old ABI */
- if (rtc->nvram)
- rtc_nvram_unregister(rtc);
- nvmem_unregister(rtc->nvmem);
- }
|