mirror of
https://github.com/sle118/squeezelite-esp32.git
synced 2025-12-06 19:47:02 +03:00
161 lines
4.5 KiB
C
161 lines
4.5 KiB
C
/*
|
|
* Squeezelite for esp32
|
|
*
|
|
* (c) Sebastien 2019
|
|
* Philippe G. 2019, philippe_44@outlook.com
|
|
*
|
|
* This software is released under the MIT License.
|
|
* https://opensource.org/licenses/MIT
|
|
*
|
|
*/
|
|
|
|
#include <freertos/FreeRTOS.h>
|
|
#include <freertos/task.h>
|
|
#include <driver/i2s.h>
|
|
#include "driver/i2c.h"
|
|
#include "esp_log.h"
|
|
#include "cJSON.h"
|
|
#include "platform_config.h"
|
|
#include "adac.h"
|
|
|
|
static const char TAG[] = "DAC external";
|
|
|
|
static void speaker(bool active);
|
|
static void headset(bool active);
|
|
static bool volume(unsigned left, unsigned right) { return false; }
|
|
static void power(adac_power_e mode);
|
|
static bool init(char *config, int i2c_port_num, i2s_config_t *i2s_config);
|
|
|
|
static bool i2c_json_execute(char *set);
|
|
|
|
const struct adac_s dac_external = { "i2s", init, adac_deinit, power, speaker, headset, volume };
|
|
static cJSON *i2c_json;
|
|
static int i2c_addr;
|
|
|
|
static struct {
|
|
char *model;
|
|
bool mclk;
|
|
char *controlset;
|
|
} codecs[] = {
|
|
{ "es8388", true,
|
|
"{\"init\":[ \
|
|
{\"reg\":8,\"val\":0}, {\"reg\":2,\"val\":243}, {\"reg\":43,\"val\":128}, {\"reg\":0,\"val\":5}, \
|
|
{\"reg\":1,\"val\":64}, {\"reg\":4,\"val\":60}, {\"reg\":23,\"val\":24}, {\"reg\":24,\"val\":2}, \
|
|
{\"reg\":26,\"val\":0}, {\"reg\":27,\"val\":0}, {\"reg\":25,\"val\":50}, {\"reg\":38,\"val\":0}, \
|
|
{\"reg\":39,\"val\":184}, {\"reg\":42,\"val\":184}, {\"reg\":46,\"val\":30}, {\"reg\":47,\"val\":30}, \
|
|
{\"reg\":48,\"val\":30}, {\"reg\":49,\"val\":30}, {\"reg\":2,\"val\":170}]}" },
|
|
{ NULL, false, NULL }
|
|
};
|
|
|
|
/****************************************************************************************
|
|
* init
|
|
*/
|
|
static bool init(char *config, int i2c_port_num, i2s_config_t *i2s_config) {
|
|
char *p;
|
|
|
|
i2c_addr = adac_init(config, i2c_port_num);
|
|
if (!i2c_addr) return true;
|
|
|
|
ESP_LOGI(TAG, "DAC on I2C @%d", i2c_addr);
|
|
|
|
p = config_alloc_get_str("dac_controlset", CONFIG_DAC_CONTROLSET, NULL);
|
|
|
|
if ((!p || !*p) && (p = strcasestr(config, "model")) != NULL) {
|
|
char model[32] = "";
|
|
int i;
|
|
sscanf(p, "%*[^=]=%31[^,]", model);
|
|
for (i = 0; *model && ((p = codecs[i].controlset) != NULL) && strcasecmp(codecs[i].model, model); i++);
|
|
if (p && codecs[i].mclk) {
|
|
ESP_LOGI(TAG, "Configuring MCLK on GPIO0");
|
|
PIN_FUNC_SELECT(PERIPHS_IO_MUX_GPIO0_U, FUNC_GPIO0_CLK_OUT1);
|
|
REG_WRITE(PIN_CTRL, 0xFFFFFFF0);
|
|
}
|
|
}
|
|
|
|
i2c_json = cJSON_Parse(p);
|
|
|
|
if (!i2c_json) {
|
|
ESP_LOGW(TAG, "no i2c controlset found");
|
|
return true;
|
|
}
|
|
|
|
if (!i2c_json_execute("init")) {
|
|
ESP_LOGE(TAG, "could not intialize DAC");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/****************************************************************************************
|
|
* power
|
|
*/
|
|
static void power(adac_power_e mode) {
|
|
if (mode == ADAC_STANDBY || mode == ADAC_OFF) i2c_json_execute("poweroff");
|
|
else i2c_json_execute("poweron");
|
|
}
|
|
|
|
/****************************************************************************************
|
|
* speaker
|
|
*/
|
|
static void speaker(bool active) {
|
|
if (active) i2c_json_execute("speakeron");
|
|
else i2c_json_execute("speakeroff");
|
|
}
|
|
|
|
/****************************************************************************************
|
|
* headset
|
|
*/
|
|
static void headset(bool active) {
|
|
if (active) i2c_json_execute("headseton");
|
|
else i2c_json_execute("headsetoff");
|
|
}
|
|
|
|
/****************************************************************************************
|
|
*
|
|
*/
|
|
bool i2c_json_execute(char *set) {
|
|
cJSON *json_set = cJSON_GetObjectItemCaseSensitive(i2c_json, set);
|
|
cJSON *item;
|
|
|
|
if (!json_set) return true;
|
|
|
|
cJSON_ArrayForEach(item, json_set) {
|
|
cJSON *reg = cJSON_GetObjectItemCaseSensitive(item, "reg");
|
|
cJSON *val = cJSON_GetObjectItemCaseSensitive(item, "val");
|
|
|
|
if (cJSON_IsArray(val)) {
|
|
cJSON *value;
|
|
uint8_t *data = malloc(cJSON_GetArraySize(val));
|
|
int count = 0;
|
|
|
|
if (!data) continue;
|
|
|
|
cJSON_ArrayForEach(value, val) {
|
|
data[count++] = value->valueint;
|
|
}
|
|
|
|
adac_write(i2c_addr, reg->valueint, data, count);
|
|
free(data);
|
|
} else {
|
|
cJSON *mode = cJSON_GetObjectItemCaseSensitive(item, "mode");
|
|
|
|
if (!reg || !val) continue;
|
|
|
|
if (!mode) {
|
|
adac_write_byte(i2c_addr, reg->valueint, val->valueint);
|
|
} else if (!strcasecmp(mode->valuestring, "or")) {
|
|
uint8_t data = adac_read_byte(i2c_addr,reg->valueint);
|
|
data |= (uint8_t) val->valueint;
|
|
adac_write_byte(i2c_addr, reg->valueint, data);
|
|
} else if (!strcasecmp(mode->valuestring, "and")) {
|
|
uint8_t data = adac_read_byte(i2c_addr, reg->valueint);
|
|
data &= (uint8_t) val->valueint;
|
|
adac_write_byte(i2c_addr, reg->valueint, data);
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|