dac.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. // Copyright 2019-2020 Espressif Systems (Shanghai) PTE LTD
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #include <string.h>
  15. #include "esp_log.h"
  16. #include "esp_err.h"
  17. #include "esp_pm.h"
  18. #include "freertos/FreeRTOS.h"
  19. #include "freertos/semphr.h"
  20. #include "freertos/timers.h"
  21. #include "driver/rtc_io.h"
  22. #include "driver/dac.h"
  23. #include "soc/dac_periph.h"
  24. #include "hal/dac_hal.h"
  25. static const char *DAC_TAG = "DAC";
  26. #define DAC_CHECK(a, str, ret_val) ({ \
  27. if (!(a)) { \
  28. ESP_LOGE(DAC_TAG,"%s:%d (%s):%s", __FILE__, __LINE__, __FUNCTION__, str); \
  29. return (ret_val); \
  30. } \
  31. })
  32. extern portMUX_TYPE rtc_spinlock; //TODO: Will be placed in the appropriate position after the rtc module is finished.
  33. #define DAC_ENTER_CRITICAL() portENTER_CRITICAL(&rtc_spinlock)
  34. #define DAC_EXIT_CRITICAL() portEXIT_CRITICAL(&rtc_spinlock)
  35. #ifdef CONFIG_PM_ENABLE
  36. static esp_pm_lock_handle_t s_dac_digi_lock = NULL;
  37. #endif //CONFIG_PM_ENABLE
  38. /*---------------------------------------------------------------
  39. Digital controller setting
  40. ---------------------------------------------------------------*/
  41. esp_err_t dac_digi_init(void)
  42. {
  43. DAC_ENTER_CRITICAL();
  44. dac_hal_digi_init();
  45. DAC_EXIT_CRITICAL();
  46. return ESP_OK;
  47. }
  48. esp_err_t dac_digi_deinit(void)
  49. {
  50. #ifdef CONFIG_PM_ENABLE
  51. if (s_dac_digi_lock) {
  52. esp_pm_lock_delete(s_dac_digi_lock);
  53. s_dac_digi_lock = NULL;
  54. }
  55. #endif
  56. DAC_ENTER_CRITICAL();
  57. dac_hal_digi_deinit();
  58. DAC_EXIT_CRITICAL();
  59. return ESP_OK;
  60. }
  61. esp_err_t dac_digi_controller_config(const dac_digi_config_t *cfg)
  62. {
  63. DAC_CHECK(cfg->mode < DAC_CONV_MAX, "DAC mode error", ESP_ERR_INVALID_ARG);
  64. DAC_CHECK(cfg->interval > 0 && cfg->interval < 4096, "DAC interval error", ESP_ERR_INVALID_ARG);
  65. DAC_CHECK(cfg->dig_clk.div_num < 256, "DAC clk div_num error", ESP_ERR_INVALID_ARG);
  66. DAC_CHECK(cfg->dig_clk.div_b > 0 && cfg->dig_clk.div_b < 64, "DAC clk div_b error", ESP_ERR_INVALID_ARG);
  67. DAC_CHECK(cfg->dig_clk.div_a < 64, "DAC clk div_a error", ESP_ERR_INVALID_ARG);
  68. #ifdef CONFIG_PM_ENABLE
  69. esp_err_t err;
  70. if (s_dac_digi_lock == NULL) {
  71. if (cfg->dig_clk.use_apll) {
  72. err = esp_pm_lock_create(ESP_PM_NO_LIGHT_SLEEP, 0, "dac_dma", &s_dac_digi_lock);
  73. } else {
  74. err = esp_pm_lock_create(ESP_PM_APB_FREQ_MAX, 0, "dac_dma", &s_dac_digi_lock);
  75. }
  76. if (err != ESP_OK) {
  77. s_dac_digi_lock = NULL;
  78. ESP_LOGE(DAC_TAG, "DAC-DMA pm lock error");
  79. return err;
  80. }
  81. }
  82. #endif //CONFIG_PM_ENABLE
  83. DAC_ENTER_CRITICAL();
  84. dac_hal_digi_controller_config(cfg);
  85. DAC_EXIT_CRITICAL();
  86. return ESP_OK;
  87. }
  88. esp_err_t dac_digi_start(void)
  89. {
  90. #ifdef CONFIG_PM_ENABLE
  91. DAC_CHECK((s_dac_digi_lock), "Should start after call `dac_digi_controller_config`", ESP_FAIL);
  92. esp_pm_lock_acquire(s_dac_digi_lock);
  93. #endif
  94. DAC_ENTER_CRITICAL();
  95. dac_hal_digi_start();
  96. DAC_EXIT_CRITICAL();
  97. return ESP_OK;
  98. }
  99. esp_err_t dac_digi_stop(void)
  100. {
  101. #ifdef CONFIG_PM_ENABLE
  102. if (s_dac_digi_lock) {
  103. esp_pm_lock_release(s_dac_digi_lock);
  104. }
  105. #endif
  106. DAC_ENTER_CRITICAL();
  107. dac_hal_digi_stop();
  108. DAC_EXIT_CRITICAL();
  109. return ESP_OK;
  110. }
  111. esp_err_t dac_digi_fifo_reset(void)
  112. {
  113. DAC_ENTER_CRITICAL();
  114. dac_hal_digi_fifo_reset();
  115. DAC_EXIT_CRITICAL();
  116. return ESP_OK;
  117. }
  118. esp_err_t dac_digi_reset(void)
  119. {
  120. DAC_ENTER_CRITICAL();
  121. dac_hal_digi_reset();
  122. DAC_EXIT_CRITICAL();
  123. return ESP_OK;
  124. }