mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-31 18:35:42 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			336 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			336 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0+
 | |
| /*
 | |
|  * Copyright 2019 Google LLC
 | |
|  */
 | |
| 
 | |
| #define LOG_CATEGORY UCLASS_SOUND
 | |
| 
 | |
| #include <common.h>
 | |
| #include <audio_codec.h>
 | |
| #include <dm.h>
 | |
| #include <i2c.h>
 | |
| #include "rt5677.h"
 | |
| #include <log.h>
 | |
| 
 | |
| struct rt5677_priv {
 | |
| 	struct udevice *dev;
 | |
| };
 | |
| 
 | |
| /* RT5677 has 256 8-bit register addresses, and 16-bit register data */
 | |
| struct rt5677_init_reg {
 | |
| 	u8 reg;
 | |
| 	u16 val;
 | |
| };
 | |
| 
 | |
| static struct rt5677_init_reg init_list[] = {
 | |
| 	{RT5677_LOUT1,		  0x0800},
 | |
| 	{RT5677_SIDETONE_CTRL,	  0x0000},
 | |
| 	{RT5677_STO1_ADC_DIG_VOL, 0x3F3F},
 | |
| 	{RT5677_DAC1_DIG_VOL,	  0x9090},
 | |
| 	{RT5677_STO2_ADC_MIXER,	  0xA441},
 | |
| 	{RT5677_STO1_ADC_MIXER,	  0x5480},
 | |
| 	{RT5677_STO1_DAC_MIXER,	  0x8A8A},
 | |
| 	{RT5677_PWR_DIG1,	  0x9800}, /* Power up I2S1 */
 | |
| 	{RT5677_PWR_ANLG1,	  0xE9D5},
 | |
| 	{RT5677_PWR_ANLG2,	  0x2CC0},
 | |
| 	{RT5677_PWR_DSP2,	  0x0C00},
 | |
| 	{RT5677_I2S2_SDP,	  0x0000},
 | |
| 	{RT5677_CLK_TREE_CTRL1,	  0x1111},
 | |
| 	{RT5677_PLL1_CTRL1,	  0x0000},
 | |
| 	{RT5677_PLL1_CTRL2,	  0x0000},
 | |
| 	{RT5677_DIG_MISC,	  0x0029},
 | |
| 	{RT5677_GEN_CTRL1,	  0x00FF},
 | |
| 	{RT5677_GPIO_CTRL2,	  0x0020},
 | |
| 	{RT5677_PWR_DIG2,	  0x9024}, /* Power on ADC Stereo Filters */
 | |
| 	{RT5677_PDM_OUT_CTRL,	  0x0088}, /* Unmute PDM, set stereo1 DAC */
 | |
| 	{RT5677_PDM_DATA_CTRL1,   0x0001}, /* Sysclk to PDM filter divider 2 */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * rt5677_i2c_read() - Read a 16-bit register
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @reg: Register number to read
 | |
|  * @returns data read or -ve on error
 | |
|  */
 | |
| static int rt5677_i2c_read(struct rt5677_priv *priv, uint reg)
 | |
| {
 | |
| 	u8 buf[2];
 | |
| 	int ret;
 | |
| 
 | |
| 	ret = dm_i2c_read(priv->dev, reg, buf, sizeof(u16));
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 	return buf[0] << 8 | buf[1];
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * rt5677_i2c_write() - Write a 16-bit register
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @reg: Register number to read
 | |
|  * @data: Data to write
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  */
 | |
| static int rt5677_i2c_write(struct rt5677_priv *priv, uint reg, uint data)
 | |
| {
 | |
| 	u8 buf[2];
 | |
| 
 | |
| 	buf[0] = (data >> 8) & 0xff;
 | |
| 	buf[1] = data & 0xff;
 | |
| 
 | |
| 	return dm_i2c_write(priv->dev, reg, buf, sizeof(u16));
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * rt5677_bic_or() - Set and clear bits of a codec register
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @reg: Register number to update
 | |
|  * @bic: Mask of bits to clear
 | |
|  * @set: Mask of bits to set
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  *
 | |
|  */
 | |
| static int rt5677_bic_or(struct rt5677_priv *priv, uint reg, uint bic,
 | |
| 			 uint set)
 | |
| {
 | |
| 	uint old, new_value;
 | |
| 	int ret;
 | |
| 
 | |
| 	old = rt5677_i2c_read(priv, reg);
 | |
| 	if (old < 0)
 | |
| 		return old;
 | |
| 
 | |
| 	new_value = (old & ~bic) | (set & bic);
 | |
| 
 | |
| 	if (old != new_value) {
 | |
| 		ret = rt5677_i2c_write(priv, reg, new_value);
 | |
| 		if (ret)
 | |
| 			return ret;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * rt5677_reg_init() - Initialise codec regs w/static/base values
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  */
 | |
| static int rt5677_reg_init(struct rt5677_priv *priv)
 | |
| {
 | |
| 	int ret;
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < ARRAY_SIZE(init_list); i++) {
 | |
| 		ret = rt5677_i2c_write(priv, init_list[i].reg, init_list[i].val);
 | |
| 		if (ret)
 | |
| 			return ret;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| #ifdef DEBUG
 | |
| static void debug_dump_5677_regs(struct rt5677_priv *priv, int swap)
 | |
| {
 | |
| 	uint i, reg_word;
 | |
| 
 | |
| 	/* Show all 16-bit codec regs */
 | |
| 	for (i = 0; i < RT5677_REG_CNT; i++) {
 | |
| 		if (i % 8 == 0)
 | |
| 			log_debug("\nMX%02x: ", i);
 | |
| 
 | |
| 		rt5677_i2c_read(priv, (u8)i, ®_word);
 | |
| 		if (swap)
 | |
| 			log_debug("%04x ", swap_bytes16(reg_word));
 | |
| 		else
 | |
| 			log_debug("%04x ", reg_word);
 | |
| 	}
 | |
| 	log_debug("\n");
 | |
| 
 | |
| 	/* Show all 16-bit 'private' codec regs */
 | |
| 	for (i = 0; i < RT5677_PR_REG_CNT; i++) {
 | |
| 		if (i % 8 == 0)
 | |
| 			log_debug("\nPR%02x: ", i);
 | |
| 
 | |
| 		rt5677_i2c_write(priv, RT5677_PRIV_INDEX, i);
 | |
| 		rt5677_i2c_read(priv, RT5677_PRIV_DATA, ®_word);
 | |
| 		if (swap)
 | |
| 			log_debug("%04x ", swap_bytes16(reg_word));
 | |
| 		else
 | |
| 			log_debug("%04x ", reg_word);
 | |
| 	}
 | |
| 	log_debug("\n");
 | |
| }
 | |
| #endif	/* DEBUG */
 | |
| 
 | |
| static int rt5677_hw_params(struct rt5677_priv *priv, uint bits_per_sample)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	switch (bits_per_sample) {
 | |
| 	case 16:
 | |
| 		ret = rt5677_bic_or(priv, RT5677_I2S1_SDP, RT5677_I2S_DL_MASK,
 | |
| 				    0);
 | |
| 		if (ret) {
 | |
| 			log_debug("Error updating I2S1 Interface Ctrl reg\n");
 | |
| 			return 1;
 | |
| 		}
 | |
| 		break;
 | |
| 	default:
 | |
| 		log_err("Illegal bits per sample %d\n", bits_per_sample);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * rt5677_set_fmt() - set rt5677 I2S format
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  */
 | |
| static int rt5677_set_fmt(struct rt5677_priv *priv)
 | |
| {
 | |
| 	int ret = 0;
 | |
| 
 | |
| 	/*
 | |
| 	 * Set format here: Assumes I2S, NB_NF, CBS_CFS
 | |
| 	 *
 | |
| 	 * CBS_CFS (Codec Bit Slave/Codec Frame Slave)
 | |
| 	 */
 | |
| 	ret = rt5677_bic_or(priv, RT5677_I2S1_SDP, RT5677_I2S_MS_MASK,
 | |
| 			    RT5677_I2S_MS_S);
 | |
| 
 | |
| 	/* NB_NF (Normal Bit/Normal Frame) */
 | |
| 	ret |= rt5677_bic_or(priv, RT5677_I2S1_SDP, RT5677_I2S_BP_MASK,
 | |
| 			     RT5677_I2S_BP_NOR);
 | |
| 
 | |
| 	/* I2S mode */
 | |
| 	ret |= rt5677_bic_or(priv, RT5677_I2S1_SDP, RT5677_I2S_DF_MASK,
 | |
| 			     RT5677_I2S_DF_I2S);
 | |
| 
 | |
| 	/* A44: I2S2 (going to speaker amp) is master */
 | |
| 	ret |= rt5677_bic_or(priv, RT5677_I2S2_SDP, RT5677_I2S_MS_MASK,
 | |
| 			     RT5677_I2S_MS_M);
 | |
| 
 | |
| 	if (ret) {
 | |
| 		log_err("Error updating I2S1 Interface Ctrl reg\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * rt5677_reset() - reset the audio codec
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  */
 | |
| static int rt5677_reset(struct rt5677_priv *priv)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	/* Reset the codec registers to their defaults */
 | |
| 	ret = rt5677_i2c_write(priv, RT5677_RESET, RT5677_SW_RESET);
 | |
| 	if (ret) {
 | |
| 		log_err("Error resetting codec\n");
 | |
| 		return ret;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Initialise rt5677 codec device
 | |
|  *
 | |
|  * @priv: Private driver data
 | |
|  * @returns 0 if OK, -ve on error
 | |
|  */
 | |
| int rt5677_device_init(struct rt5677_priv *priv)
 | |
| {
 | |
| 	int ret;
 | |
| 
 | |
| 	/* Read status reg */
 | |
| 	ret = rt5677_i2c_read(priv, RT5677_RESET);
 | |
| 	if (ret < 0)
 | |
| 		return ret;
 | |
| 	log_debug("reg 00h, Software Reset & Status = 0x%04x\n", ret);
 | |
| 
 | |
| 	/* Reset the codec/regs */
 | |
| 	ret = rt5677_reset(priv);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	ret = rt5677_i2c_read(priv, RT5677_VENDOR_ID1);
 | |
| 	if (ret < 0) {
 | |
| 		log_err("Error reading vendor ID\n");
 | |
| 		return 1;
 | |
| 	}
 | |
| 	log_debug("Hardware ID: %0xX\n", ret);
 | |
| 
 | |
| 	ret = rt5677_i2c_read(priv, RT5677_VENDOR_ID2);
 | |
| 	if (ret < 0) {
 | |
| 		log_err("Error reading vendor rev\n");
 | |
| 		return 1;
 | |
| 	}
 | |
| 	log_debug("Hardware revision: %04x\n", ret);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rt5677_set_params(struct udevice *dev, int interface, int rate,
 | |
| 			     int mclk_freq, int bits_per_sample,
 | |
| 			     uint channels)
 | |
| {
 | |
| 	struct rt5677_priv *priv = dev_get_priv(dev);
 | |
| 	int ret;
 | |
| 
 | |
| 	/* Initialise codec regs w/static/base values, same as Linux driver */
 | |
| 	ret = rt5677_reg_init(priv);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	ret = rt5677_hw_params(priv, bits_per_sample);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	ret = rt5677_set_fmt(priv);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int rt5677_probe(struct udevice *dev)
 | |
| {
 | |
| 	struct rt5677_priv *priv = dev_get_priv(dev);
 | |
| 
 | |
| 	priv->dev = dev;
 | |
| 
 | |
| 	return rt5677_device_init(priv);
 | |
| }
 | |
| 
 | |
| static const struct audio_codec_ops rt5677_ops = {
 | |
| 	.set_params	= rt5677_set_params,
 | |
| };
 | |
| 
 | |
| static const struct udevice_id rt5677_ids[] = {
 | |
| 	{ .compatible = "realtek,rt5677" },
 | |
| 	{ }
 | |
| };
 | |
| 
 | |
| U_BOOT_DRIVER(rt5677_drv) = {
 | |
| 	.name		= "rt5677",
 | |
| 	.id		= UCLASS_AUDIO_CODEC,
 | |
| 	.of_match	= rt5677_ids,
 | |
| 	.ops		= &rt5677_ops,
 | |
| 	.probe		= rt5677_probe,
 | |
| 	.priv_auto_alloc_size	= sizeof(struct rt5677_priv),
 | |
| };
 |