[U-Boot] [PATCH] RFC: samsung: i2c: Enable new CONFIG_SYS_I2C framework

Heiko Schocher hs at denx.de
Mon Oct 14 08:06:28 CEST 2013


Hello Naveen,

Am 30.09.2013 08:58, schrieb Naveen Krishna Chatradhi:
> This enables CONFIG_SYS_I2C on Samsung, updating existing s3c24x0
> i2c driver to support this.
>
> Note: Not for merge, Just for review and suggestions.
>
> Signed-off-by: Naveen Krishna Chatradhi<ch.naveen at samsung.com>
> ---
>   drivers/i2c/Makefile      |    2 +-
>   drivers/i2c/s3c24x0_i2c.c |  156 +++++++++++++++++++++++++++++++--------------
>   2 files changed, 108 insertions(+), 50 deletions(-)
>
> diff --git a/drivers/i2c/Makefile b/drivers/i2c/Makefile
> index 37ccbd1..96448a6 100644
> --- a/drivers/i2c/Makefile
> +++ b/drivers/i2c/Makefile
> @@ -20,7 +20,7 @@ COBJS-$(CONFIG_DRIVER_OMAP1510_I2C) += omap1510_i2c.o
>   COBJS-$(CONFIG_DRIVER_OMAP24XX_I2C) += omap24xx_i2c.o
>   COBJS-$(CONFIG_DRIVER_OMAP34XX_I2C) += omap24xx_i2c.o
>   COBJS-$(CONFIG_PCA9564_I2C) += pca9564_i2c.o
> -COBJS-$(CONFIG_DRIVER_S3C24X0_I2C) += s3c24x0_i2c.o
> +COBJS-$(CONFIG_SYS_I2C_S3C24X0_I2C) += s3c24x0_i2c.o

Please keep lists sorted.

>   COBJS-$(CONFIG_S3C44B0_I2C) += s3c44b0_i2c.o
>   COBJS-$(CONFIG_TSI108_I2C) += tsi108_i2c.o
>   COBJS-$(CONFIG_U8500_I2C) += u8500_i2c.o
> diff --git a/drivers/i2c/s3c24x0_i2c.c b/drivers/i2c/s3c24x0_i2c.c
> index 604d43d..89cb2d2 100644
> --- a/drivers/i2c/s3c24x0_i2c.c
> +++ b/drivers/i2c/s3c24x0_i2c.c
> @@ -117,6 +117,8 @@
>
>   #define	HSI2C_TIMEOUT_US 100000 /* 100 ms, finer granularity */
>
> +DECLARE_GLOBAL_DATA_PTR;
> +
>   /*
>    * For SPL boot some boards need i2c before SDRAM is initialised so force
>    * variables to live in SRAM
> @@ -126,6 +128,9 @@ static unsigned int g_current_bus __attribute__((section(".data")));
>   static struct s3c24x0_i2c_bus i2c_bus[CONFIG_MAX_I2C_NUM]
>   			__attribute__((section(".data")));
>   #endif
> +static void i2c_ch_init(struct s3c24x0_i2c *i2c, int speed, int slaveadd);
> +static int hsi2c_get_clk_details(struct s3c24x0_i2c_bus *);
> +static void hsi2c_ch_init(struct s3c24x0_i2c_bus *);
>
>   /**
>    * Get a pointer to the given bus index
> @@ -133,20 +138,40 @@ static struct s3c24x0_i2c_bus i2c_bus[CONFIG_MAX_I2C_NUM]
>    * @bus_idx: Bus index to look up
>    * @return pointer to bus, or NULL if invalid or not available
>    */
> -static struct s3c24x0_i2c_bus *get_bus(unsigned int bus_idx)
> +static struct s3c24x0_i2c_bus *s3c_i2c_get_bus(struct i2c_adapter *adap)
>   {
> -	if (bus_idx<  ARRAY_SIZE(i2c_bus)) {
> -		struct s3c24x0_i2c_bus *bus;
> +	struct s3c24x0_i2c_bus *bus;
>
> -		bus =&i2c_bus[bus_idx];
> -		if (bus->active)
> -			return bus;
> -	}
> +	bus =&i2c_bus[adap->hwadapnr];
> +	if (bus->active)
> +		return bus;
>
> -	debug("Undefined bus: %d\n", bus_idx);
> +	debug("Undefined bus: %d\n", adap->hwadapnr);
>   	return NULL;
>   }
>
> +static unsigned int s3c_i2c_set_bus_speed(struct i2c_adapter *adap,
> +					  unsigned int speed)
> +{
> +	struct s3c24x0_i2c_bus *i2c_bus;
> +
> +	i2c_bus = s3c_i2c_get_bus(adap);
> +	if (!i2c_bus)
> +		return -1;
> +	i2c_bus->clock_frequency = speed;
> +
> +	if (i2c_bus->is_highspeed) {
> +		if (hsi2c_get_clk_details(i2c_bus))
> +			return -1;
> +		hsi2c_ch_init(i2c_bus);
> +	} else {
> +		i2c_ch_init(i2c_bus->regs, i2c_bus->clock_frequency,
> +						CONFIG_SYS_I2C_SLAVE);
> +	}
> +
> +	return 0;
> +}
> +
>   #if !(defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
>   static int GetI2CSDA(void)
>   {
> @@ -392,39 +417,7 @@ static void exynos5_i2c_reset(struct s3c24x0_i2c_bus *i2c_bus)
>   	hsi2c_ch_init(i2c_bus);
>   }
>
> -/*
> - * MULTI BUS I2C support
> - */
> -
> -#ifdef CONFIG_I2C_MULTI_BUS
> -int i2c_set_bus_num(unsigned int bus)
> -{
> -	struct s3c24x0_i2c_bus *i2c_bus;
> -
> -	i2c_bus = get_bus(bus);
> -	if (!i2c_bus)
> -		return -1;
> -	g_current_bus = bus;
> -
> -	if (i2c_bus->is_highspeed) {
> -		if (hsi2c_get_clk_details(i2c_bus))
> -			return -1;
> -		hsi2c_ch_init(i2c_bus);
> -	} else {
> -		i2c_ch_init(i2c_bus->regs, i2c_bus->clock_frequency,
> -						CONFIG_SYS_I2C_SLAVE);
> -	}
> -
> -	return 0;
> -}
> -
> -unsigned int i2c_get_bus_num(void)
> -{
> -	return g_current_bus;
> -}
> -#endif
> -
> -void i2c_init(int speed, int slaveadd)
> +static void s3c_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
>   {
>   	struct s3c24x0_i2c *i2c;
>   #if !(defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
> @@ -493,9 +486,15 @@ void i2c_init(int speed, int slaveadd)
>   #endif
>   	}
>   #endif /* #if !(defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5) */
> -	i2c_ch_init(i2c, speed, slaveadd);
> +	/* This will override the speed selected in the fdt for that port */
> +	debug("i2c_init(speed=%u, slaveaddr=0x%x)\n", speed, slaveaddr);
> +	i2c_ch_init(i2c_bus->regs, speed, slaveaddr);
>   }
>
> +static void exynos_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
> +{
> +	i2c_set_bus_speed(speed);
> +}
>   /*
>    * Poll the appropriate bit of the fifo status register until the interface is
>    * ready to process the next byte or timeout expires.
> @@ -829,13 +828,13 @@ bailout:
>   	return result;
>   }
>
> -int i2c_probe(uchar chip)
> +int s3c_i2c_probe(struct i2c_adapter *adap, uchar chip)
>   {
>   	struct s3c24x0_i2c_bus *i2c_bus;
>   	uchar buf[1];
>   	int ret;
>
> -	i2c_bus = get_bus(g_current_bus);
> +	i2c_bus = s3c_i2c_get_bus(adap);
>   	if (!i2c_bus)
>   		return -1;
>   	buf[0] = 0;
> @@ -857,7 +856,8 @@ int i2c_probe(uchar chip)
>   	return ret != I2C_OK;
>   }
>
> -int i2c_read(uchar chip, uint addr, int alen, uchar *buffer, int len)
> +int s3c_i2c_read(struct i2c_adapter *adap, uchar chip, uint addr, int alen,
> +	     uchar *buffer, int len)
>   {
>   	struct s3c24x0_i2c_bus *i2c_bus;
>   	uchar xaddr[4];
> @@ -891,7 +891,7 @@ int i2c_read(uchar chip, uint addr, int alen, uchar *buffer, int len)
>   		chip |= ((addr>>  (alen * 8))&
>   			CONFIG_SYS_I2C_EEPROM_ADDR_OVERFLOW);
>   #endif
> -	i2c_bus = get_bus(g_current_bus);
> +	i2c_bus = s3c_i2c_get_bus(adap);
>   	if (!i2c_bus)
>   		return -1;
>
> @@ -911,7 +911,8 @@ int i2c_read(uchar chip, uint addr, int alen, uchar *buffer, int len)
>   	return 0;
>   }
>
> -int i2c_write(uchar chip, uint addr, int alen, uchar *buffer, int len)
> +int s3c_i2c_write(struct i2c_adapter *adap, uchar chip, uint addr, int alen,
> +	      uchar *buffer, int len)
>   {
>   	struct s3c24x0_i2c_bus *i2c_bus;
>   	uchar xaddr[4];
> @@ -944,7 +945,7 @@ int i2c_write(uchar chip, uint addr, int alen, uchar *buffer, int len)
>   		chip |= ((addr>>  (alen * 8))&
>   			CONFIG_SYS_I2C_EEPROM_ADDR_OVERFLOW);
>   #endif
> -	i2c_bus = get_bus(g_current_bus);
> +	i2c_bus = s3c_i2c_get_bus(adap);
>   	if (!i2c_bus)
>   		return -1;
>
> @@ -1000,6 +1001,25 @@ static void process_nodes(const void *blob, int node_list[], int count,
>   		node_list[i] = -1;
>   	}
>   }
> +void i2c_init_board()
> +{
> +	int node_list[CONFIG_MAX_I2C_NUM];
> +	const void *blob = gd->fdt_blob;
> +	int count;
> +
> +	/* First get the normal i2c ports */
> +	count = fdtdec_find_aliases_for_id(blob, "i2c",
> +		COMPAT_SAMSUNG_S3C2440_I2C, node_list,
> +		CONFIG_MAX_I2C_NUM);
> +	process_nodes(blob, node_list, count, 0);
> +
> +	/* Now look for high speed i2c ports */
> +	count = fdtdec_find_aliases_for_id(blob, "i2c",
> +		COMPAT_SAMSUNG_EXYNOS5_I2C, node_list,
> +		CONFIG_MAX_I2C_NUM);
> +	process_nodes(blob, node_list, count, 1);
> +
> +}
>
>   void board_i2c_init(const void *blob)
>   {
> @@ -1017,9 +1037,10 @@ void board_i2c_init(const void *blob)
>   		COMPAT_SAMSUNG_EXYNOS5_I2C, node_list,
>   		CONFIG_MAX_I2C_NUM);
>   	process_nodes(blob, node_list, count, 1);
> -
>   }
>
> +#ifndef CONFIG_SYS_I2C
> +/* TODO: fix this as per comments from upstream */

Why this? If we switch to the new i2c framework, please convert all boards,
which using this driver ...

>   int i2c_get_bus_num_fdt(int node)
>   {
>   	int i;
> @@ -1062,5 +1083,42 @@ int i2c_reset_port_fdt(const void *blob, int node)
>   	return 0;
>   }
>   #endif
> +#endif
>
> +/*
> + * Register soft i2c adapters
> + */
> +U_BOOT_I2C_ADAP_COMPLETE(s3c0, s3c_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 0)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c1, s3c_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 1)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c2, s3c_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 2)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c3, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 3)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c4, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 4)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c5, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 5)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c6, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 6)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c7, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 7)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c8, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 8)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c9, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 9)
> +U_BOOT_I2C_ADAP_COMPLETE(s3c10, exynos_i2c_init, s3c_i2c_probe,
> +			 s3c_i2c_read, s3c_i2c_write,
> +			 s3c_i2c_set_bus_speed, 100000, 0, 10)
>   #endif /* CONFIG_HARD_I2C */

Beside of that, it looks good to me.

Thanks for your work!

bye,
Heiko
-- 
DENX Software Engineering GmbH,     MD: Wolfgang Denk & Detlev Zundel
HRB 165235 Munich, Office: Kirchenstr.5, D-82194 Groebenzell, Germany


More information about the U-Boot mailing list