digital-cdc-rsc-mgr.c 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2019-2020, The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
  5. */
  6. #include <linux/clk.h>
  7. #include <linux/clk-provider.h>
  8. #include <linux/ratelimit.h>
  9. #include <dsp/digital-cdc-rsc-mgr.h>
  10. #include <linux/dev_printk.h>
  11. struct mutex hw_vote_lock;
  12. static bool is_init_done;
  13. /**
  14. * digital_cdc_rsc_mgr_hw_vote_enable - Enables hw vote in DSP
  15. *
  16. * @vote_handle: vote handle for which voting needs to be done
  17. * @dev: indicate which device votes
  18. *
  19. * Returns 0 on success or -EINVAL/error code on failure
  20. */
  21. int digital_cdc_rsc_mgr_hw_vote_enable(struct clk *vote_handle, struct device *dev)
  22. {
  23. int ret = 0;
  24. if (!is_init_done || vote_handle == NULL) {
  25. pr_err_ratelimited("%s: init failed or vote handle NULL\n",
  26. __func__);
  27. return -EINVAL;
  28. }
  29. mutex_lock(&hw_vote_lock);
  30. ret = clk_prepare_enable(vote_handle);
  31. mutex_unlock(&hw_vote_lock);
  32. dev_dbg(dev, "%s: return %d\n", __func__, ret);
  33. trace_printk("%s: return %d\n", __func__, ret);
  34. return ret;
  35. }
  36. EXPORT_SYMBOL(digital_cdc_rsc_mgr_hw_vote_enable);
  37. /**
  38. * digital_cdc_rsc_mgr_hw_vote_disable - Disables hw vote in DSP
  39. *
  40. * @vote_handle: vote handle for which voting needs to be disabled
  41. * @dev: indicate which device unvotes
  42. *
  43. */
  44. void digital_cdc_rsc_mgr_hw_vote_disable(struct clk *vote_handle, struct device *dev)
  45. {
  46. if (!is_init_done || vote_handle == NULL) {
  47. pr_err_ratelimited("%s: init failed or vote handle NULL\n",
  48. __func__);
  49. return;
  50. }
  51. mutex_lock(&hw_vote_lock);
  52. clk_disable_unprepare(vote_handle);
  53. mutex_unlock(&hw_vote_lock);
  54. dev_dbg(dev, "%s: leave\n", __func__);
  55. trace_printk("%s\n", __func__);
  56. }
  57. EXPORT_SYMBOL(digital_cdc_rsc_mgr_hw_vote_disable);
  58. /**
  59. * digital_cdc_rsc_mgr_hw_vote_reset - Resets hw vote count
  60. *
  61. */
  62. void digital_cdc_rsc_mgr_hw_vote_reset(struct clk* vote_handle)
  63. {
  64. int count = 0;
  65. if (!is_init_done || vote_handle == NULL) {
  66. pr_err_ratelimited("%s: init failed or vote handle NULL\n",
  67. __func__);
  68. return;
  69. }
  70. mutex_lock(&hw_vote_lock);
  71. while (__clk_is_enabled(vote_handle)) {
  72. clk_disable_unprepare(vote_handle);
  73. count++;
  74. }
  75. pr_debug("%s: Vote count after SSR: %d\n", __func__, count);
  76. trace_printk("%s: Vote count after SSR: %d\n", __func__, count);
  77. while (count--)
  78. clk_prepare_enable(vote_handle);
  79. mutex_unlock(&hw_vote_lock);
  80. }
  81. EXPORT_SYMBOL(digital_cdc_rsc_mgr_hw_vote_reset);
  82. void digital_cdc_rsc_mgr_init(void)
  83. {
  84. mutex_init(&hw_vote_lock);
  85. is_init_done = true;
  86. }
  87. void digital_cdc_rsc_mgr_exit(void)
  88. {
  89. mutex_destroy(&hw_vote_lock);
  90. is_init_done = false;
  91. }