qti-smmu-proxy.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2023 Qualcomm Innovation Center, Inc. All rights reserved.
  4. */
  5. #include <linux/module.h>
  6. #include <linux/of_platform.h>
  7. #include <linux/platform_device.h>
  8. #include <linux/cdev.h>
  9. #include "linux/qti-smmu-proxy.h"
  10. #define SMMU_PROXY_MAX_DEVS 1
  11. static dev_t smmu_proxy_dev_no;
  12. static struct class *smmu_proxy_class;
  13. static struct cdev smmu_proxy_char_dev;
  14. union smmu_proxy_ioctl_arg {
  15. struct csf_version csf_version;
  16. };
  17. int smmu_proxy_get_csf_version(struct csf_version *csf_version)
  18. {
  19. csf_version->arch_ver = 2;
  20. csf_version->max_ver = 0;
  21. csf_version->min_ver = 0;
  22. return 0;
  23. }
  24. EXPORT_SYMBOL(smmu_proxy_get_csf_version);
  25. static long smmu_proxy_dev_ioctl(struct file *filp, unsigned int cmd,
  26. unsigned long arg)
  27. {
  28. unsigned int dir = _IOC_DIR(cmd);
  29. union smmu_proxy_ioctl_arg ioctl_arg;
  30. if (_IOC_SIZE(cmd) > sizeof(ioctl_arg))
  31. return -EINVAL;
  32. if (copy_from_user(&ioctl_arg, (void __user *)arg, _IOC_SIZE(cmd)))
  33. return -EFAULT;
  34. if (!(dir & _IOC_WRITE))
  35. memset(&ioctl_arg, 0, sizeof(ioctl_arg));
  36. switch (cmd) {
  37. case QTI_SMMU_PROXY_GET_VERSION_IOCTL:
  38. {
  39. struct csf_version *csf_version =
  40. &ioctl_arg.csf_version;
  41. int ret;
  42. ret = smmu_proxy_get_csf_version(csf_version);
  43. if(ret)
  44. return ret;
  45. break;
  46. }
  47. default:
  48. return -ENOTTY;
  49. }
  50. if (dir & _IOC_READ) {
  51. if (copy_to_user((void __user *)arg, &ioctl_arg,
  52. _IOC_SIZE(cmd)))
  53. return -EFAULT;
  54. }
  55. return 0;
  56. }
  57. static const struct file_operations smmu_proxy_dev_fops = {
  58. .unlocked_ioctl = smmu_proxy_dev_ioctl,
  59. .compat_ioctl = compat_ptr_ioctl,
  60. };
  61. static int smmu_proxy_create_dev(struct platform_device *pdev)
  62. {
  63. int ret;
  64. struct device *class_dev;
  65. cdev_init(&smmu_proxy_char_dev, &smmu_proxy_dev_fops);
  66. ret = cdev_add(&smmu_proxy_char_dev, smmu_proxy_dev_no,
  67. SMMU_PROXY_MAX_DEVS);
  68. if (ret < 0)
  69. return ret;
  70. class_dev = device_create(smmu_proxy_class, NULL, smmu_proxy_dev_no, NULL,
  71. "qti-smmu-proxy");
  72. if (IS_ERR(class_dev)) {
  73. ret = PTR_ERR(class_dev);
  74. goto err_dev_create;
  75. }
  76. return 0;
  77. err_dev_create:
  78. cdev_del(&smmu_proxy_char_dev);
  79. return ret;
  80. }
  81. static int smmu_proxy_probe(struct platform_device *pdev)
  82. {
  83. return smmu_proxy_create_dev(pdev);
  84. }
  85. static const struct of_device_id smmu_proxy_match_table[] = {
  86. {.compatible = "smmu-proxy-sender"},
  87. {},
  88. };
  89. static struct platform_driver smmu_proxy_driver = {
  90. .probe = smmu_proxy_probe,
  91. .driver = {
  92. .name = "qti-smmu-proxy",
  93. .of_match_table = smmu_proxy_match_table,
  94. },
  95. };
  96. static int __init init_smmu_proxy_driver(void)
  97. {
  98. int ret;
  99. ret = alloc_chrdev_region(&smmu_proxy_dev_no, 0, SMMU_PROXY_MAX_DEVS,
  100. "qti-smmu-proxy");
  101. if (ret < 0)
  102. goto err_chrdev_region;
  103. smmu_proxy_class = class_create(THIS_MODULE, "qti-smmu-proxy");
  104. if (IS_ERR(smmu_proxy_class)) {
  105. ret = PTR_ERR(smmu_proxy_class);
  106. goto err_class_create;
  107. }
  108. ret = platform_driver_register(&smmu_proxy_driver);
  109. if (ret < 0)
  110. goto err_platform_drvr_register;
  111. return 0;
  112. err_platform_drvr_register:
  113. class_destroy(smmu_proxy_class);
  114. err_class_create:
  115. unregister_chrdev_region(smmu_proxy_dev_no, SMMU_PROXY_MAX_DEVS);
  116. err_chrdev_region:
  117. return ret;
  118. }
  119. module_init(init_smmu_proxy_driver);
  120. MODULE_IMPORT_NS(DMA_BUF);
  121. MODULE_LICENSE("GPL v2");