qfprom-sys.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2022-2023 Qualcomm Innovation Center, Inc. All rights reserved.
  4. */
  5. #include <linux/device.h>
  6. #include <linux/module.h>
  7. #include <linux/mod_devicetable.h>
  8. #include <linux/io.h>
  9. #include <linux/nvmem-consumer.h>
  10. #include <linux/of.h>
  11. #include <linux/platform_device.h>
  12. #include <linux/slab.h>
  13. #include <linux/sysfs.h>
  14. struct qfprom_sys {
  15. int cell_count;
  16. struct nvmem_cell **cells;
  17. struct bin_attribute **attrs;
  18. };
  19. static ssize_t qfprom_sys_cell_read(struct file *filp, struct kobject *kobj,
  20. struct bin_attribute *attr,
  21. char *buf, loff_t pos, size_t count)
  22. {
  23. struct nvmem_cell *cell;
  24. size_t len;
  25. u8 *data;
  26. cell = attr->private;
  27. if (!cell)
  28. return -EINVAL;
  29. data = nvmem_cell_read(cell, &len);
  30. if (IS_ERR(data))
  31. return -EINVAL;
  32. len = min(len, count);
  33. memcpy(buf, data, len);
  34. kfree(data);
  35. return len;
  36. }
  37. static int qfprom_sys_probe(struct platform_device *pdev)
  38. {
  39. struct device *dev = &pdev->dev;
  40. struct qfprom_sys *priv;
  41. struct nvmem_cell *cell;
  42. int cell_count;
  43. int i, ret;
  44. const char **cells;
  45. cell_count = of_property_count_strings(dev->of_node, "nvmem-cell-names");
  46. if (cell_count <= 0)
  47. return -EINVAL;
  48. priv = devm_kzalloc(dev, sizeof(struct qfprom_sys), GFP_KERNEL);
  49. if (!priv)
  50. return -ENOMEM;
  51. priv->cells = devm_kzalloc(dev,
  52. cell_count * sizeof(struct nvmem_cell *),
  53. GFP_KERNEL);
  54. if (!priv->cells)
  55. return -ENOMEM;
  56. cells = devm_kzalloc(dev, cell_count * sizeof(char *), GFP_KERNEL);
  57. if (!cells)
  58. return -ENOMEM;
  59. priv->attrs = devm_kzalloc(dev,
  60. cell_count * sizeof(struct bin_attribute *),
  61. GFP_KERNEL);
  62. if (!priv->attrs)
  63. return -ENOMEM;
  64. ret = of_property_read_string_array(dev->of_node, "nvmem-cell-names",
  65. cells, cell_count);
  66. for (i = 0; i < cell_count; i++) {
  67. cell = nvmem_cell_get(dev, cells[i]);
  68. if (IS_ERR_OR_NULL(cell)) {
  69. ret = PTR_ERR(cell);
  70. dev_err(dev, "cell corresponding to %s not found\n",
  71. cells[i]);
  72. goto remove_cells;
  73. }
  74. priv->attrs[i] = devm_kzalloc(dev, sizeof(struct bin_attribute),
  75. GFP_KERNEL);
  76. if (!priv->attrs[i]) {
  77. ret = -ENOMEM;
  78. nvmem_cell_put(cell);
  79. goto remove_cells;
  80. }
  81. priv->cells[i] = cell;
  82. sysfs_bin_attr_init(priv->attrs[i]);
  83. priv->attrs[i]->attr.name = cells[i];
  84. priv->attrs[i]->attr.mode = 0444;
  85. priv->attrs[i]->private = cell;
  86. priv->attrs[i]->size = 4;
  87. priv->attrs[i]->read = qfprom_sys_cell_read;
  88. ret = sysfs_create_bin_file(&dev->kobj, priv->attrs[i]);
  89. if (ret) {
  90. dev_err(dev, "cell corresponding to %s\n", cells[i]);
  91. nvmem_cell_put(cell);
  92. goto remove_cells;
  93. }
  94. }
  95. priv->cell_count = cell_count;
  96. dev->platform_data = priv;
  97. return 0;
  98. remove_cells:
  99. for (; i > 0; i--) {
  100. sysfs_remove_bin_file(&dev->kobj, priv->attrs[i-1]);
  101. nvmem_cell_put(priv->cells[i-1]);
  102. }
  103. priv->cell_count = 0;
  104. return ret;
  105. }
  106. static int qfprom_sys_remove(struct platform_device *pdev)
  107. {
  108. struct qfprom_sys *priv;
  109. int i;
  110. priv = dev_get_platdata(&pdev->dev);
  111. for (i = 0; i < priv->cell_count; i++) {
  112. nvmem_cell_put(priv->cells[i]);
  113. sysfs_remove_bin_file(&pdev->dev.kobj, priv->attrs[i]);
  114. }
  115. return 0;
  116. }
  117. static const struct of_device_id qfprom_sys_of_match[] = {
  118. { .compatible = "qcom,qfprom-sys",},
  119. {},
  120. };
  121. MODULE_DEVICE_TABLE(of, qfprom_sys_of_match);
  122. static struct platform_driver qfprom_sys_driver = {
  123. .probe = qfprom_sys_probe,
  124. .remove = qfprom_sys_remove,
  125. .driver = {
  126. .name = "qcom,qfprom-sys",
  127. .of_match_table = qfprom_sys_of_match,
  128. },
  129. };
  130. module_platform_driver(qfprom_sys_driver);
  131. MODULE_DESCRIPTION("Qualcomm Technologies, Inc. QFPROM_SYS driver");
  132. MODULE_LICENSE("GPL");