dac_external.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. /*
  2. * Squeezelite for esp32
  3. *
  4. * (c) Sebastien 2019
  5. * Philippe G. 2019, philippe_44@outlook.com
  6. *
  7. * This software is released under the MIT License.
  8. * https://opensource.org/licenses/MIT
  9. *
  10. */
  11. #include <freertos/FreeRTOS.h>
  12. #include <freertos/task.h>
  13. #include <driver/i2s.h>
  14. #include "driver/i2c.h"
  15. #include "esp_log.h"
  16. #include "cJSON.h"
  17. #include "platform_config.h"
  18. #include "adac.h"
  19. static const char TAG[] = "DAC external";
  20. static void speaker(bool active) { }
  21. static void headset(bool active) { }
  22. static bool volume(unsigned left, unsigned right) { return false; }
  23. static void power(adac_power_e mode);
  24. static bool init(char *config, int i2c_port_num, i2s_config_t *i2s_config);
  25. static bool i2c_json_execute(char *set);
  26. const struct adac_s dac_external = { "i2s", init, adac_deinit, power, speaker, headset, volume };
  27. static cJSON *i2c_json;
  28. static int i2c_addr;
  29. static struct {
  30. char *model;
  31. bool mclk;
  32. char *controlset;
  33. } codecs[] = {
  34. { "es8388", true,
  35. "{\"init\":[ \
  36. {\"reg\":8,\"val\":0}, {\"reg\":2,\"val\":243}, {\"reg\":43,\"val\":128}, {\"reg\":0,\"val\":5}, \
  37. {\"reg\":1,\"val\":64}, {\"reg\":4,\"val\":60}, {\"reg\":23,\"val\":24}, {\"reg\":24,\"val\":2}, \
  38. {\"reg\":26,\"val\":0}, {\"reg\":27,\"val\":0}, {\"reg\":25,\"val\":50}, {\"reg\":38,\"val\":0}, \
  39. {\"reg\":39,\"val\":184}, {\"reg\":42,\"val\":184}, {\"reg\":46,\"val\":30}, {\"reg\":47,\"val\":30}, \
  40. {\"reg\":48,\"val\":30}, {\"reg\":49,\"val\":30}, {\"reg\":2,\"val\":170}]}" },
  41. { NULL, false, NULL }
  42. };
  43. /****************************************************************************************
  44. * init
  45. */
  46. static bool init(char *config, int i2c_port_num, i2s_config_t *i2s_config) {
  47. char *p;
  48. i2c_addr = adac_init(config, i2c_port_num);
  49. if (!i2c_addr) return true;
  50. ESP_LOGI(TAG, "DAC on I2C @%d", i2c_addr);
  51. p = config_alloc_get_str("dac_controlset", CONFIG_DAC_CONTROLSET, NULL);
  52. if ((!p || !*p) && (p = strcasestr(config, "model")) != NULL) {
  53. char model[32] = "";
  54. int i;
  55. sscanf(p, "%*[^=]=%31[^,]", model);
  56. for (i = 0; *model && ((p = codecs[i].controlset) != NULL) && strcasecmp(codecs[i].model, model); i++);
  57. if (p && codecs[i].mclk) {
  58. ESP_LOGI(TAG, "Configuring MCLK on GPIO0");
  59. PIN_FUNC_SELECT(PERIPHS_IO_MUX_GPIO0_U, FUNC_GPIO0_CLK_OUT1);
  60. REG_WRITE(PIN_CTRL, 0xFFFFFFF0);
  61. }
  62. }
  63. i2c_json = cJSON_Parse(p);
  64. if (!i2c_json) {
  65. ESP_LOGW(TAG, "no i2c controlset found");
  66. return true;
  67. }
  68. if (!i2c_json_execute("init")) {
  69. ESP_LOGE(TAG, "could not intialize DAC");
  70. return false;
  71. }
  72. return true;
  73. }
  74. /****************************************************************************************
  75. * power
  76. */
  77. static void power(adac_power_e mode) {
  78. if (mode == ADAC_STANDBY || mode == ADAC_OFF) i2c_json_execute("poweroff");
  79. else i2c_json_execute("poweron");
  80. }
  81. /****************************************************************************************
  82. *
  83. */
  84. bool i2c_json_execute(char *set) {
  85. cJSON *json_set = cJSON_GetObjectItemCaseSensitive(i2c_json, set);
  86. cJSON *item;
  87. if (!json_set) return true;
  88. cJSON_ArrayForEach(item, json_set)
  89. {
  90. cJSON *reg = cJSON_GetObjectItemCaseSensitive(item, "reg");
  91. cJSON *val = cJSON_GetObjectItemCaseSensitive(item, "val");
  92. cJSON *mode = cJSON_GetObjectItemCaseSensitive(item, "mode");
  93. if (!reg || !val) continue;
  94. if (!mode) {
  95. adac_write_byte(i2c_addr, reg->valueint, val->valueint);
  96. } else if (!strcasecmp(mode->valuestring, "or")) {
  97. uint8_t data = adac_read_byte(i2c_addr,reg->valueint);
  98. data |= (uint8_t) val->valueint;
  99. adac_write_byte(i2c_addr, reg->valueint, data);
  100. } else if (!strcasecmp(mode->valuestring, "and")) {
  101. uint8_t data = adac_read_byte(i2c_addr, reg->valueint);
  102. data &= (uint8_t) val->valueint;
  103. adac_write_byte(i2c_addr, reg->valueint, data);
  104. }
  105. }
  106. return true;
  107. }