cros_ec_vbc.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // SPDX-License-Identifier: GPL-2.0+
  2. // Expose the vboot context nvram to userspace
  3. //
  4. // Copyright (C) 2012 Google, Inc.
  5. // Copyright (C) 2015 Collabora Ltd.
  6. #include <linux/of.h>
  7. #include <linux/platform_device.h>
  8. #include <linux/mod_devicetable.h>
  9. #include <linux/module.h>
  10. #include <linux/platform_data/cros_ec_commands.h>
  11. #include <linux/platform_data/cros_ec_proto.h>
  12. #include <linux/slab.h>
  13. #define DRV_NAME "cros-ec-vbc"
  14. static ssize_t vboot_context_read(struct file *filp, struct kobject *kobj,
  15. struct bin_attribute *att, char *buf,
  16. loff_t pos, size_t count)
  17. {
  18. struct device *dev = kobj_to_dev(kobj);
  19. struct cros_ec_dev *ec = to_cros_ec_dev(dev);
  20. struct cros_ec_device *ecdev = ec->ec_dev;
  21. struct cros_ec_command *msg;
  22. /*
  23. * This should be a pointer to the same type as op field in
  24. * struct ec_params_vbnvcontext.
  25. */
  26. uint32_t *params_op;
  27. int err;
  28. const size_t para_sz = sizeof(*params_op);
  29. const size_t resp_sz = sizeof(struct ec_response_vbnvcontext);
  30. const size_t payload = max(para_sz, resp_sz);
  31. msg = kmalloc(sizeof(*msg) + payload, GFP_KERNEL);
  32. if (!msg)
  33. return -ENOMEM;
  34. /* NB: we only kmalloc()ated enough space for the op field */
  35. params_op = (uint32_t *)msg->data;
  36. *params_op = EC_VBNV_CONTEXT_OP_READ;
  37. msg->version = EC_VER_VBNV_CONTEXT;
  38. msg->command = EC_CMD_VBNV_CONTEXT;
  39. msg->outsize = para_sz;
  40. msg->insize = resp_sz;
  41. err = cros_ec_cmd_xfer_status(ecdev, msg);
  42. if (err < 0) {
  43. dev_err(dev, "Error sending read request: %d\n", err);
  44. kfree(msg);
  45. return err;
  46. }
  47. memcpy(buf, msg->data, resp_sz);
  48. kfree(msg);
  49. return resp_sz;
  50. }
  51. static ssize_t vboot_context_write(struct file *filp, struct kobject *kobj,
  52. struct bin_attribute *attr, char *buf,
  53. loff_t pos, size_t count)
  54. {
  55. struct device *dev = kobj_to_dev(kobj);
  56. struct cros_ec_dev *ec = to_cros_ec_dev(dev);
  57. struct cros_ec_device *ecdev = ec->ec_dev;
  58. struct ec_params_vbnvcontext *params;
  59. struct cros_ec_command *msg;
  60. int err;
  61. const size_t para_sz = sizeof(*params);
  62. const size_t data_sz = sizeof(params->block);
  63. /* Only write full values */
  64. if (count != data_sz)
  65. return -EINVAL;
  66. msg = kmalloc(sizeof(*msg) + para_sz, GFP_KERNEL);
  67. if (!msg)
  68. return -ENOMEM;
  69. params = (struct ec_params_vbnvcontext *)msg->data;
  70. params->op = EC_VBNV_CONTEXT_OP_WRITE;
  71. memcpy(params->block, buf, data_sz);
  72. msg->version = EC_VER_VBNV_CONTEXT;
  73. msg->command = EC_CMD_VBNV_CONTEXT;
  74. msg->outsize = para_sz;
  75. msg->insize = 0;
  76. err = cros_ec_cmd_xfer_status(ecdev, msg);
  77. if (err < 0) {
  78. dev_err(dev, "Error sending write request: %d\n", err);
  79. kfree(msg);
  80. return err;
  81. }
  82. kfree(msg);
  83. return data_sz;
  84. }
  85. static BIN_ATTR_RW(vboot_context, 16);
  86. static struct bin_attribute *cros_ec_vbc_bin_attrs[] = {
  87. &bin_attr_vboot_context,
  88. NULL
  89. };
  90. static const struct attribute_group cros_ec_vbc_attr_group = {
  91. .name = "vbc",
  92. .bin_attrs = cros_ec_vbc_bin_attrs,
  93. };
  94. static int cros_ec_vbc_probe(struct platform_device *pd)
  95. {
  96. struct cros_ec_dev *ec_dev = dev_get_drvdata(pd->dev.parent);
  97. struct device *dev = &pd->dev;
  98. int ret;
  99. ret = sysfs_create_group(&ec_dev->class_dev.kobj,
  100. &cros_ec_vbc_attr_group);
  101. if (ret < 0)
  102. dev_err(dev, "failed to create %s attributes. err=%d\n",
  103. cros_ec_vbc_attr_group.name, ret);
  104. return ret;
  105. }
  106. static void cros_ec_vbc_remove(struct platform_device *pd)
  107. {
  108. struct cros_ec_dev *ec_dev = dev_get_drvdata(pd->dev.parent);
  109. sysfs_remove_group(&ec_dev->class_dev.kobj,
  110. &cros_ec_vbc_attr_group);
  111. }
  112. static const struct platform_device_id cros_ec_vbc_id[] = {
  113. { DRV_NAME, 0 },
  114. {}
  115. };
  116. MODULE_DEVICE_TABLE(platform, cros_ec_vbc_id);
  117. static struct platform_driver cros_ec_vbc_driver = {
  118. .driver = {
  119. .name = DRV_NAME,
  120. },
  121. .probe = cros_ec_vbc_probe,
  122. .remove_new = cros_ec_vbc_remove,
  123. .id_table = cros_ec_vbc_id,
  124. };
  125. module_platform_driver(cros_ec_vbc_driver);
  126. MODULE_LICENSE("GPL");
  127. MODULE_DESCRIPTION("Expose the vboot context nvram to userspace");