All of lore.kernel.org
 help / color / mirror / Atom feed
* [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct
@ 2016-03-18  7:54 Stefan Roese
  2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
                   ` (6 more replies)
  0 siblings, 7 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

Add the ic_enable_status register to the ic_regs struct. Additionally
the register offsets are added, to better check, if the offset matches
the register description in the datasheet.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.h | 68 +++++++++++++++++++++++---------------------
 1 file changed, 35 insertions(+), 33 deletions(-)

diff --git a/drivers/i2c/designware_i2c.h b/drivers/i2c/designware_i2c.h
index 19b09df..270c29c 100644
--- a/drivers/i2c/designware_i2c.h
+++ b/drivers/i2c/designware_i2c.h
@@ -9,39 +9,41 @@
 #define __DW_I2C_H_
 
 struct i2c_regs {
-	u32 ic_con;
-	u32 ic_tar;
-	u32 ic_sar;
-	u32 ic_hs_maddr;
-	u32 ic_cmd_data;
-	u32 ic_ss_scl_hcnt;
-	u32 ic_ss_scl_lcnt;
-	u32 ic_fs_scl_hcnt;
-	u32 ic_fs_scl_lcnt;
-	u32 ic_hs_scl_hcnt;
-	u32 ic_hs_scl_lcnt;
-	u32 ic_intr_stat;
-	u32 ic_intr_mask;
-	u32 ic_raw_intr_stat;
-	u32 ic_rx_tl;
-	u32 ic_tx_tl;
-	u32 ic_clr_intr;
-	u32 ic_clr_rx_under;
-	u32 ic_clr_rx_over;
-	u32 ic_clr_tx_over;
-	u32 ic_clr_rd_req;
-	u32 ic_clr_tx_abrt;
-	u32 ic_clr_rx_done;
-	u32 ic_clr_activity;
-	u32 ic_clr_stop_det;
-	u32 ic_clr_start_det;
-	u32 ic_clr_gen_call;
-	u32 ic_enable;
-	u32 ic_status;
-	u32 ic_txflr;
-	u32 ix_rxflr;
-	u32 reserved_1;
-	u32 ic_tx_abrt_source;
+	u32 ic_con;		/* 0x00 */
+	u32 ic_tar;		/* 0x04 */
+	u32 ic_sar;		/* 0x08 */
+	u32 ic_hs_maddr;	/* 0x0c */
+	u32 ic_cmd_data;	/* 0x10 */
+	u32 ic_ss_scl_hcnt;	/* 0x14 */
+	u32 ic_ss_scl_lcnt;	/* 0x18 */
+	u32 ic_fs_scl_hcnt;	/* 0x1c */
+	u32 ic_fs_scl_lcnt;	/* 0x20 */
+	u32 ic_hs_scl_hcnt;	/* 0x24 */
+	u32 ic_hs_scl_lcnt;	/* 0x28 */
+	u32 ic_intr_stat;	/* 0x2c */
+	u32 ic_intr_mask;	/* 0x30 */
+	u32 ic_raw_intr_stat;	/* 0x34 */
+	u32 ic_rx_tl;		/* 0x38 */
+	u32 ic_tx_tl;		/* 0x3c */
+	u32 ic_clr_intr;	/* 0x40 */
+	u32 ic_clr_rx_under;	/* 0x44 */
+	u32 ic_clr_rx_over;	/* 0x48 */
+	u32 ic_clr_tx_over;	/* 0x4c */
+	u32 ic_clr_rd_req;	/* 0x50 */
+	u32 ic_clr_tx_abrt;	/* 0x54 */
+	u32 ic_clr_rx_done;	/* 0x58 */
+	u32 ic_clr_activity;	/* 0x5c */
+	u32 ic_clr_stop_det;	/* 0x60 */
+	u32 ic_clr_start_det;	/* 0x64 */
+	u32 ic_clr_gen_call;	/* 0x68 */
+	u32 ic_enable;		/* 0x6c */
+	u32 ic_status;		/* 0x70 */
+	u32 ic_txflr;		/* 0x74 */
+	u32 ic_rxflr;		/* 0x78 */
+	u32 ic_sda_hold;	/* 0x7c */
+	u32 ic_tx_abrt_source;	/* 0x80 */
+	u8 res1[0x18];		/* 0x84 */
+	u32 ic_enable_status;	/* 0x9c */
 };
 
 #if !defined(IC_CLK)
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
@ 2016-03-18  7:54 ` Stefan Roese
  2016-03-18 11:12   ` Marek Vasut
  2016-03-21  8:54   ` Bin Meng
  2016-03-18  7:54 ` [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed() Stefan Roese
                   ` (5 subsequent siblings)
  6 siblings, 2 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

dw_i2c_enable() is used to dis-/en-able the I2C controller. It makes
sense to add such a function, as the controller is dis-/en-abled
multiple times in the code. Additionally, this function now checks,
if the controller is really dis-/en-abled. This code is copied
from the Linux I2C driver version.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.c | 46 +++++++++++++++++++++++++-------------------
 1 file changed, 26 insertions(+), 20 deletions(-)

diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
index e768cde..c8ea520 100644
--- a/drivers/i2c/designware_i2c.c
+++ b/drivers/i2c/designware_i2c.c
@@ -34,6 +34,26 @@ static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
 	return NULL;
 }
 
+static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
+{
+	int timeout = 100;
+
+	do {
+		writel(enable, &i2c_base->ic_enable);
+		if ((readl(&i2c_base->ic_enable_status) & 1) == enable)
+			return;
+
+		/*
+		 * Wait 10 times the signaling period of the highest I2C
+		 * transfer supported by the driver (for 400KHz this is
+		 * 25us) as described in the DesignWare I2C databook.
+		 */
+		udelay(25);
+	} while (timeout--);
+
+	printf("timeout in %sabling I2C adapter\n", enable ? "en" : "dis");
+}
+
 /*
  * set_speed - Set the i2c speed mode (standard, high, fast)
  * @i2c_spd:	required i2c speed mode
@@ -45,12 +65,9 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
 	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	unsigned int cntl;
 	unsigned int hcnt, lcnt;
-	unsigned int enbl;
 
 	/* to set speed cltr must be disabled */
-	enbl = readl(&i2c_base->ic_enable);
-	enbl &= ~IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 0);
 
 	cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
 
@@ -84,8 +101,7 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
 	writel(cntl, &i2c_base->ic_con);
 
 	/* Enable back i2c now speed set */
-	enbl |= IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 1);
 }
 
 /*
@@ -123,12 +139,9 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
 			int slaveaddr)
 {
 	struct i2c_regs *i2c_base = i2c_get_base(adap);
-	unsigned int enbl;
 
 	/* Disable i2c */
-	enbl = readl(&i2c_base->ic_enable);
-	enbl &= ~IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 0);
 
 	writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
 	writel(IC_RX_TL, &i2c_base->ic_rx_tl);
@@ -138,9 +151,7 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
 	writel(slaveaddr, &i2c_base->ic_sar);
 
 	/* Enable i2c */
-	enbl = readl(&i2c_base->ic_enable);
-	enbl |= IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 1);
 }
 
 /*
@@ -152,19 +163,14 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
 static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
 {
 	struct i2c_regs *i2c_base = i2c_get_base(adap);
-	unsigned int enbl;
 
 	/* Disable i2c */
-	enbl = readl(&i2c_base->ic_enable);
-	enbl &= ~IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 0);
 
 	writel(i2c_addr, &i2c_base->ic_tar);
 
 	/* Enable i2c */
-	enbl = readl(&i2c_base->ic_enable);
-	enbl |= IC_ENABLE_0B;
-	writel(enbl, &i2c_base->ic_enable);
+	dw_i2c_enable(i2c_base, 1);
 }
 
 /*
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed()
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
  2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
@ 2016-03-18  7:54 ` Stefan Roese
  2016-03-18 11:13   ` Marek Vasut
  2016-03-21  8:54   ` Bin Meng
  2016-03-18  7:54 ` [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion Stefan Roese
                   ` (4 subsequent siblings)
  6 siblings, 2 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

Integrating set_speed() into dw_i2c_set_bus_speed() will make the
conversion to DM easier for this driver.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.c | 37 +++++++++++++------------------------
 1 file changed, 13 insertions(+), 24 deletions(-)

diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
index c8ea520..508dac9 100644
--- a/drivers/i2c/designware_i2c.c
+++ b/drivers/i2c/designware_i2c.c
@@ -55,16 +55,25 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
 }
 
 /*
- * set_speed - Set the i2c speed mode (standard, high, fast)
- * @i2c_spd:	required i2c speed mode
+ * i2c_set_bus_speed - Set the i2c speed
+ * @speed:	required i2c speed
  *
- * Set the i2c speed mode (standard, high, fast)
+ * Set the i2c speed.
  */
-static void set_speed(struct i2c_adapter *adap, int i2c_spd)
+static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
+					 unsigned int speed)
 {
 	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	unsigned int cntl;
 	unsigned int hcnt, lcnt;
+	int i2c_spd;
+
+	if (speed >= I2C_MAX_SPEED)
+		i2c_spd = IC_SPEED_MODE_MAX;
+	else if (speed >= I2C_FAST_SPEED)
+		i2c_spd = IC_SPEED_MODE_FAST;
+	else
+		i2c_spd = IC_SPEED_MODE_STANDARD;
 
 	/* to set speed cltr must be disabled */
 	dw_i2c_enable(i2c_base, 0);
@@ -102,27 +111,7 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
 
 	/* Enable back i2c now speed set */
 	dw_i2c_enable(i2c_base, 1);
-}
-
-/*
- * i2c_set_bus_speed - Set the i2c speed
- * @speed:	required i2c speed
- *
- * Set the i2c speed.
- */
-static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
-					 unsigned int speed)
-{
-	int i2c_spd;
-
-	if (speed >= I2C_MAX_SPEED)
-		i2c_spd = IC_SPEED_MODE_MAX;
-	else if (speed >= I2C_FAST_SPEED)
-		i2c_spd = IC_SPEED_MODE_FAST;
-	else
-		i2c_spd = IC_SPEED_MODE_STANDARD;
 
-	set_speed(adap, i2c_spd);
 	adap->speed = speed;
 
 	return 0;
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
  2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
  2016-03-18  7:54 ` [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed() Stefan Roese
@ 2016-03-18  7:54 ` Stefan Roese
  2016-03-21  8:54   ` Bin Meng
  2016-03-18  7:54 ` [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support Stefan Roese
                   ` (3 subsequent siblings)
  6 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

This patch prepares the designware I2C driver for the DM conversion.
This is mainly done by removing struct i2c_adapter from the functions
that shall be used by the DM driver version as well.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.c | 173 ++++++++++++++++++++++---------------------
 1 file changed, 90 insertions(+), 83 deletions(-)

diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
index 508dac9..e51e6de 100644
--- a/drivers/i2c/designware_i2c.c
+++ b/drivers/i2c/designware_i2c.c
@@ -10,30 +10,6 @@
 #include <asm/io.h>
 #include "designware_i2c.h"
 
-static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
-{
-	switch (adap->hwadapnr) {
-#if CONFIG_SYS_I2C_BUS_MAX >= 4
-	case 3:
-		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE3;
-#endif
-#if CONFIG_SYS_I2C_BUS_MAX >= 3
-	case 2:
-		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE2;
-#endif
-#if CONFIG_SYS_I2C_BUS_MAX >= 2
-	case 1:
-		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE1;
-#endif
-	case 0:
-		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE;
-	default:
-		printf("Wrong I2C-adapter number %d\n", adap->hwadapnr);
-	}
-
-	return NULL;
-}
-
 static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
 {
 	int timeout = 100;
@@ -60,10 +36,9 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
  *
  * Set the i2c speed.
  */
-static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
-					 unsigned int speed)
+static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
+					   unsigned int speed)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	unsigned int cntl;
 	unsigned int hcnt, lcnt;
 	int i2c_spd;
@@ -112,47 +87,17 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
 	/* Enable back i2c now speed set */
 	dw_i2c_enable(i2c_base, 1);
 
-	adap->speed = speed;
-
 	return 0;
 }
 
 /*
- * i2c_init - Init function
- * @speed:	required i2c speed
- * @slaveaddr:	slave address for the device
- *
- * Initialization function.
- */
-static void dw_i2c_init(struct i2c_adapter *adap, int speed,
-			int slaveaddr)
-{
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
-
-	/* Disable i2c */
-	dw_i2c_enable(i2c_base, 0);
-
-	writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
-	writel(IC_RX_TL, &i2c_base->ic_rx_tl);
-	writel(IC_TX_TL, &i2c_base->ic_tx_tl);
-	dw_i2c_set_bus_speed(adap, speed);
-	writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
-	writel(slaveaddr, &i2c_base->ic_sar);
-
-	/* Enable i2c */
-	dw_i2c_enable(i2c_base, 1);
-}
-
-/*
  * i2c_setaddress - Sets the target slave address
  * @i2c_addr:	target i2c address
  *
  * Sets the target slave address.
  */
-static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
+static void i2c_setaddress(struct i2c_regs *i2c_base, unsigned int i2c_addr)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
-
 	/* Disable i2c */
 	dw_i2c_enable(i2c_base, 0);
 
@@ -167,10 +112,8 @@ static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
  *
  * Flushes the i2c RX FIFO
  */
-static void i2c_flush_rxfifo(struct i2c_adapter *adap)
+static void i2c_flush_rxfifo(struct i2c_regs *i2c_base)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
-
 	while (readl(&i2c_base->ic_status) & IC_STATUS_RFNE)
 		readl(&i2c_base->ic_cmd_data);
 }
@@ -180,9 +123,8 @@ static void i2c_flush_rxfifo(struct i2c_adapter *adap)
  *
  * Waits for bus busy
  */
-static int i2c_wait_for_bb(struct i2c_adapter *adap)
+static int i2c_wait_for_bb(struct i2c_regs *i2c_base)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	unsigned long start_time_bb = get_timer(0);
 
 	while ((readl(&i2c_base->ic_status) & IC_STATUS_MA) ||
@@ -196,15 +138,13 @@ static int i2c_wait_for_bb(struct i2c_adapter *adap)
 	return 0;
 }
 
-static int i2c_xfer_init(struct i2c_adapter *adap, uchar chip, uint addr,
+static int i2c_xfer_init(struct i2c_regs *i2c_base, uchar chip, uint addr,
 			 int alen)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
-
-	if (i2c_wait_for_bb(adap))
+	if (i2c_wait_for_bb(i2c_base))
 		return 1;
 
-	i2c_setaddress(adap, chip);
+	i2c_setaddress(i2c_base, chip);
 	while (alen) {
 		alen--;
 		/* high byte address going out first */
@@ -214,9 +154,8 @@ static int i2c_xfer_init(struct i2c_adapter *adap, uchar chip, uint addr,
 	return 0;
 }
 
-static int i2c_xfer_finish(struct i2c_adapter *adap)
+static int i2c_xfer_finish(struct i2c_regs *i2c_base)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	ulong start_stop_det = get_timer(0);
 
 	while (1) {
@@ -228,12 +167,12 @@ static int i2c_xfer_finish(struct i2c_adapter *adap)
 		}
 	}
 
-	if (i2c_wait_for_bb(adap)) {
+	if (i2c_wait_for_bb(i2c_base)) {
 		printf("Timed out waiting for bus\n");
 		return 1;
 	}
 
-	i2c_flush_rxfifo(adap);
+	i2c_flush_rxfifo(i2c_base);
 
 	return 0;
 }
@@ -248,10 +187,9 @@ static int i2c_xfer_finish(struct i2c_adapter *adap)
  *
  * Read from i2c memory.
  */
-static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
-		       int alen, u8 *buffer, int len)
+static int __dw_i2c_read(struct i2c_regs *i2c_base, u8 dev, uint addr,
+			 int alen, u8 *buffer, int len)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	unsigned long start_time_rx;
 
 #ifdef CONFIG_SYS_I2C_EEPROM_ADDR_OVERFLOW
@@ -273,7 +211,7 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
 	      addr);
 #endif
 
-	if (i2c_xfer_init(adap, dev, addr, alen))
+	if (i2c_xfer_init(i2c_base, dev, addr, alen))
 		return 1;
 
 	start_time_rx = get_timer(0);
@@ -293,7 +231,7 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
 		}
 	}
 
-	return i2c_xfer_finish(adap);
+	return i2c_xfer_finish(i2c_base);
 }
 
 /*
@@ -306,10 +244,9 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
  *
  * Write to i2c memory.
  */
-static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
-			int alen, u8 *buffer, int len)
+static int __dw_i2c_write(struct i2c_regs *i2c_base, u8 dev, uint addr,
+			  int alen, u8 *buffer, int len)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	int nb = len;
 	unsigned long start_time_tx;
 
@@ -332,7 +269,7 @@ static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
 	      addr);
 #endif
 
-	if (i2c_xfer_init(adap, dev, addr, alen))
+	if (i2c_xfer_init(i2c_base, dev, addr, alen))
 		return 1;
 
 	start_time_tx = get_timer(0);
@@ -353,7 +290,76 @@ static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
 		}
 	}
 
-	return i2c_xfer_finish(adap);
+	return i2c_xfer_finish(i2c_base);
+}
+
+static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
+{
+	switch (adap->hwadapnr) {
+#if CONFIG_SYS_I2C_BUS_MAX >= 4
+	case 3:
+		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE3;
+#endif
+#if CONFIG_SYS_I2C_BUS_MAX >= 3
+	case 2:
+		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE2;
+#endif
+#if CONFIG_SYS_I2C_BUS_MAX >= 2
+	case 1:
+		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE1;
+#endif
+	case 0:
+		return (struct i2c_regs *)CONFIG_SYS_I2C_BASE;
+	default:
+		printf("Wrong I2C-adapter number %d\n", adap->hwadapnr);
+	}
+
+	return NULL;
+}
+
+static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
+					 unsigned int speed)
+{
+	adap->speed = speed;
+	return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
+}
+
+/*
+ * i2c_init - Init function
+ * @speed:	required i2c speed
+ * @slaveaddr:	slave address for the device
+ *
+ * Initialization function.
+ */
+static void dw_i2c_init(struct i2c_adapter *adap, int speed,
+			int slaveaddr)
+{
+	struct i2c_regs *i2c_base = i2c_get_base(adap);
+
+	/* Disable i2c */
+	dw_i2c_enable(i2c_base, 0);
+
+	writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
+	writel(IC_RX_TL, &i2c_base->ic_rx_tl);
+	writel(IC_TX_TL, &i2c_base->ic_tx_tl);
+	dw_i2c_set_bus_speed(adap, speed);
+	writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
+	writel(slaveaddr, &i2c_base->ic_sar);
+
+	/* Enable i2c */
+	dw_i2c_enable(i2c_base, 1);
+}
+
+static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
+		       int alen, u8 *buffer, int len)
+{
+	return __dw_i2c_read(i2c_get_base(adap), dev, addr, alen, buffer, len);
+}
+
+static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
+			int alen, u8 *buffer, int len)
+{
+	return __dw_i2c_write(i2c_get_base(adap), dev, addr, alen, buffer, len);
 }
 
 /*
@@ -361,13 +367,14 @@ static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
  */
 static int dw_i2c_probe(struct i2c_adapter *adap, u8 dev)
 {
+	struct i2c_regs *i2c_base = i2c_get_base(adap);
 	u32 tmp;
 	int ret;
 
 	/*
 	 * Try to read the first location of the chip.
 	 */
-	ret = dw_i2c_read(adap, dev, 0, 1, (uchar *)&tmp, 1);
+	ret = __dw_i2c_read(i2c_base, dev, 0, 1, (uchar *)&tmp, 1);
 	if (ret)
 		dw_i2c_init(adap, adap->speed, adap->slaveaddr);
 
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
                   ` (2 preceding siblings ...)
  2016-03-18  7:54 ` [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion Stefan Roese
@ 2016-03-18  7:54 ` Stefan Roese
  2016-03-21  8:54   ` Bin Meng
  2016-04-09 18:35   ` Simon Glass
  2016-03-18  7:54 ` [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86) Stefan Roese
                   ` (2 subsequent siblings)
  6 siblings, 2 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

This patch adds DM support to the designware I2C driver. It currently
supports DM and the legacy I2C support. The legacy support should be
removed, once all platforms using it have DM enabled.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.c | 149 ++++++++++++++++++++++++++++++++++++-------
 1 file changed, 126 insertions(+), 23 deletions(-)

diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
index e51e6de..4e5340d 100644
--- a/drivers/i2c/designware_i2c.c
+++ b/drivers/i2c/designware_i2c.c
@@ -6,10 +6,15 @@
  */
 
 #include <common.h>
+#include <dm.h>
 #include <i2c.h>
 #include <asm/io.h>
 #include "designware_i2c.h"
 
+struct dw_i2c {
+	struct i2c_regs *regs;
+};
+
 static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
 {
 	int timeout = 100;
@@ -293,6 +298,36 @@ static int __dw_i2c_write(struct i2c_regs *i2c_base, u8 dev, uint addr,
 	return i2c_xfer_finish(i2c_base);
 }
 
+/*
+ * i2c_init - Init function
+ * @speed:	required i2c speed
+ * @slaveaddr:	slave address for the device
+ *
+ * Initialization function.
+ */
+static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
+{
+	/* Disable i2c */
+	dw_i2c_enable(i2c_base, 0);
+
+	writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
+	writel(IC_RX_TL, &i2c_base->ic_rx_tl);
+	writel(IC_TX_TL, &i2c_base->ic_tx_tl);
+	writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
+#ifndef CONFIG_DM_I2C
+	__dw_i2c_set_bus_speed(i2c_base, speed);
+	writel(slaveaddr, &i2c_base->ic_sar);
+#endif
+
+	/* Enable i2c */
+	dw_i2c_enable(i2c_base, 1);
+}
+
+#ifndef CONFIG_DM_I2C
+/*
+ * The legacy I2C functions. These need to get removed once
+ * all users of this driver are converted to DM.
+ */
 static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
 {
 	switch (adap->hwadapnr) {
@@ -324,30 +359,9 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
 	return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
 }
 
-/*
- * i2c_init - Init function
- * @speed:	required i2c speed
- * @slaveaddr:	slave address for the device
- *
- * Initialization function.
- */
-static void dw_i2c_init(struct i2c_adapter *adap, int speed,
-			int slaveaddr)
+static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
 {
-	struct i2c_regs *i2c_base = i2c_get_base(adap);
-
-	/* Disable i2c */
-	dw_i2c_enable(i2c_base, 0);
-
-	writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
-	writel(IC_RX_TL, &i2c_base->ic_rx_tl);
-	writel(IC_TX_TL, &i2c_base->ic_tx_tl);
-	dw_i2c_set_bus_speed(adap, speed);
-	writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
-	writel(slaveaddr, &i2c_base->ic_sar);
-
-	/* Enable i2c */
-	dw_i2c_enable(i2c_base, 1);
+	__dw_i2c_init(i2c_get_base(adap), speed, slaveaddr);
 }
 
 static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
@@ -402,3 +416,92 @@ U_BOOT_I2C_ADAP_COMPLETE(dw_3, dw_i2c_init, dw_i2c_probe, dw_i2c_read,
 			 dw_i2c_write, dw_i2c_set_bus_speed,
 			 CONFIG_SYS_I2C_SPEED3, CONFIG_SYS_I2C_SLAVE3, 3)
 #endif
+
+#else /* CONFIG_DM_I2C */
+/*
+ * The DM I2C functions
+ */
+
+static int designware_i2c_xfer(struct udevice *bus, struct i2c_msg *msg,
+			       int nmsgs)
+{
+	struct dw_i2c *i2c = dev_get_priv(bus);
+	int ret;
+
+	debug("i2c_xfer: %d messages\n", nmsgs);
+	for (; nmsgs > 0; nmsgs--, msg++) {
+		debug("i2c_xfer: chip=0x%x, len=0x%x\n", msg->addr, msg->len);
+		if (msg->flags & I2C_M_RD) {
+			ret = __dw_i2c_read(i2c->regs, msg->addr, 0, 0,
+					    msg->buf, msg->len);
+		} else {
+			ret = __dw_i2c_write(i2c->regs, msg->addr, 0, 0,
+					     msg->buf, msg->len);
+		}
+		if (ret) {
+			debug("i2c_write: error sending\n");
+			return -EREMOTEIO;
+		}
+	}
+
+	return 0;
+}
+
+static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
+{
+	struct dw_i2c *i2c = dev_get_priv(bus);
+
+	return __dw_i2c_set_bus_speed(i2c->regs, speed);
+}
+
+static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
+				     uint chip_flags)
+{
+	struct dw_i2c *i2c = dev_get_priv(bus);
+	struct i2c_regs *i2c_base = i2c->regs;
+	u32 tmp;
+	int ret;
+
+	/*
+	 * Try to read the first location of the chip.
+	 */
+	ret = __dw_i2c_read(i2c_base, chip_addr, 0, 1, (uchar *)&tmp, 1);
+	if (ret)
+		__dw_i2c_init(i2c_base, 0, 0);
+
+	return ret;
+}
+
+static int designware_i2c_probe(struct udevice *bus)
+{
+	struct dw_i2c *priv = dev_get_priv(bus);
+
+	/* Save base address from device-tree */
+	priv->regs = (struct i2c_regs *)dev_get_addr(bus);
+
+	__dw_i2c_init(priv->regs, 0, 0);
+
+	return 0;
+}
+
+static const struct dm_i2c_ops designware_i2c_ops = {
+	.xfer		= designware_i2c_xfer,
+	.probe_chip	= designware_i2c_probe_chip,
+	.set_bus_speed	= designware_i2c_set_bus_speed,
+};
+
+static const struct udevice_id designware_i2c_ids[] = {
+	{ .compatible = "snps,designware-i2c" },
+	{ }
+};
+
+U_BOOT_DRIVER(i2c_designware) = {
+	.name	= "i2c_designware",
+	.id	= UCLASS_I2C,
+	.of_match = designware_i2c_ids,
+	.probe	= designware_i2c_probe,
+	.priv_auto_alloc_size = sizeof(struct dw_i2c),
+	.ops	= &designware_i2c_ops,
+};
+
+#endif /* CONFIG_DM_I2C */
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
                   ` (3 preceding siblings ...)
  2016-03-18  7:54 ` [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support Stefan Roese
@ 2016-03-18  7:54 ` Stefan Roese
  2016-03-21  8:54   ` Bin Meng
  2016-03-18 11:09 ` [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Marek Vasut
  2016-03-21  8:54 ` Bin Meng
  6 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-03-18  7:54 UTC (permalink / raw)
  To: u-boot

This patch adds support for the PCI(e) based I2C cores. Which can be
found for example on the Intel Bay Trail SoC. It has 7 I2C controllers
implemented as PCI devices.

This patch also adds the fixed values for the timing registers for
BayTrail which are taken from the Linux designware I2C driver.

Signed-off-by: Stefan Roese <sr@denx.de>
Cc: Simon Glass <sjg@chromium.org>
Cc: Bin Meng <bmeng.cn@gmail.com>
Cc: Marek Vasut <marex@denx.de>
Cc: Heiko Schocher <hs@denx.de>
---
 drivers/i2c/designware_i2c.c | 111 +++++++++++++++++++++++++++++++++++++++----
 1 file changed, 101 insertions(+), 10 deletions(-)

diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
index 4e5340d..f7f2eba 100644
--- a/drivers/i2c/designware_i2c.c
+++ b/drivers/i2c/designware_i2c.c
@@ -8,11 +8,32 @@
 #include <common.h>
 #include <dm.h>
 #include <i2c.h>
+#include <pci.h>
 #include <asm/io.h>
 #include "designware_i2c.h"
 
+struct dw_scl_sda_cfg {
+	u32 ss_hcnt;
+	u32 fs_hcnt;
+	u32 ss_lcnt;
+	u32 fs_lcnt;
+	u32 sda_hold;
+};
+
+#ifdef CONFIG_X86
+/* BayTrail HCNT/LCNT/SDA hold time */
+static struct dw_scl_sda_cfg byt_config = {
+	.ss_hcnt = 0x200,
+	.fs_hcnt = 0x55,
+	.ss_lcnt = 0x200,
+	.fs_lcnt = 0x99,
+	.sda_hold = 0x6,
+};
+#endif
+
 struct dw_i2c {
 	struct i2c_regs *regs;
+	struct dw_scl_sda_cfg *scl_sda_cfg;
 };
 
 static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
@@ -42,6 +63,7 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
  * Set the i2c speed.
  */
 static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
+					   struct dw_scl_sda_cfg *scl_sda_cfg,
 					   unsigned int speed)
 {
 	unsigned int cntl;
@@ -61,34 +83,55 @@ static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
 	cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
 
 	switch (i2c_spd) {
+#ifndef CONFIG_X86 /* No High-speed for BayTrail yet */
 	case IC_SPEED_MODE_MAX:
-		cntl |= IC_CON_SPD_HS;
-		hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
+		cntl |= IC_CON_SPD_SS;
+		if (scl_sda_cfg) {
+			hcnt = scl_sda_cfg->fs_hcnt;
+			lcnt = scl_sda_cfg->fs_lcnt;
+		} else {
+			hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
+			lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
+		}
 		writel(hcnt, &i2c_base->ic_hs_scl_hcnt);
-		lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
 		writel(lcnt, &i2c_base->ic_hs_scl_lcnt);
 		break;
+#endif
 
 	case IC_SPEED_MODE_STANDARD:
 		cntl |= IC_CON_SPD_SS;
-		hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
+		if (scl_sda_cfg) {
+			hcnt = scl_sda_cfg->ss_hcnt;
+			lcnt = scl_sda_cfg->ss_lcnt;
+		} else {
+			hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
+			lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
+		}
 		writel(hcnt, &i2c_base->ic_ss_scl_hcnt);
-		lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
 		writel(lcnt, &i2c_base->ic_ss_scl_lcnt);
 		break;
 
 	case IC_SPEED_MODE_FAST:
 	default:
 		cntl |= IC_CON_SPD_FS;
-		hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
+		if (scl_sda_cfg) {
+			hcnt = scl_sda_cfg->fs_hcnt;
+			lcnt = scl_sda_cfg->fs_lcnt;
+		} else {
+			hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
+			lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
+		}
 		writel(hcnt, &i2c_base->ic_fs_scl_hcnt);
-		lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
 		writel(lcnt, &i2c_base->ic_fs_scl_lcnt);
 		break;
 	}
 
 	writel(cntl, &i2c_base->ic_con);
 
+	/* Configure SDA Hold Time if required */
+	if (scl_sda_cfg)
+		writel(scl_sda_cfg->sda_hold, &i2c_base->ic_sda_hold);
+
 	/* Enable back i2c now speed set */
 	dw_i2c_enable(i2c_base, 1);
 
@@ -315,7 +358,7 @@ static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
 	writel(IC_TX_TL, &i2c_base->ic_tx_tl);
 	writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
 #ifndef CONFIG_DM_I2C
-	__dw_i2c_set_bus_speed(i2c_base, speed);
+	__dw_i2c_set_bus_speed(i2c_base, NULL, speed);
 	writel(slaveaddr, &i2c_base->ic_sar);
 #endif
 
@@ -356,7 +399,7 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
 					 unsigned int speed)
 {
 	adap->speed = speed;
-	return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
+	return __dw_i2c_set_bus_speed(i2c_get_base(adap), NULL, speed);
 }
 
 static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
@@ -451,7 +494,7 @@ static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
 {
 	struct dw_i2c *i2c = dev_get_priv(bus);
 
-	return __dw_i2c_set_bus_speed(i2c->regs, speed);
+	return __dw_i2c_set_bus_speed(i2c->regs, i2c->scl_sda_cfg, speed);
 }
 
 static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
@@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
 {
 	struct dw_i2c *priv = dev_get_priv(bus);
 
+#ifdef CONFIG_X86
+	/* Save base address from PCI BAR */
+	priv->regs = (struct i2c_regs *)
+		dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
+	/* Use BayTrail specific timing values */
+	priv->scl_sda_cfg = &byt_config;
+#else
 	/* Save base address from device-tree */
 	priv->regs = (struct i2c_regs *)dev_get_addr(bus);
+#endif
 
 	__dw_i2c_init(priv->regs, 0, 0);
 
 	return 0;
 }
 
+static int designware_i2c_bind(struct udevice *dev)
+{
+	static int num_cards;
+	char name[20];
+
+	/* Create a unique device name for PCI type devices */
+	if (device_is_on_pci_bus(dev)) {
+		/*
+		 * ToDo:
+		 * Setting req_seq in the driver is probably not recommended.
+		 * But without a DT alias the number is not configured. And
+		 * using this driver is impossible for PCIe I2C devices.
+		 * This can be removed, once a better (correct) way for this
+		 * is found and implemented.
+		 */
+		dev->req_seq = num_cards;
+		sprintf(name, "i2c_designware#%u", num_cards++);
+		device_set_name(dev, name);
+	}
+
+	return 0;
+}
+
 static const struct dm_i2c_ops designware_i2c_ops = {
 	.xfer		= designware_i2c_xfer,
 	.probe_chip	= designware_i2c_probe_chip,
@@ -499,9 +573,26 @@ U_BOOT_DRIVER(i2c_designware) = {
 	.name	= "i2c_designware",
 	.id	= UCLASS_I2C,
 	.of_match = designware_i2c_ids,
+	.bind	= designware_i2c_bind,
 	.probe	= designware_i2c_probe,
 	.priv_auto_alloc_size = sizeof(struct dw_i2c),
 	.ops	= &designware_i2c_ops,
 };
 
+#ifdef CONFIG_X86
+static struct pci_device_id designware_pci_supported[] = {
+	/* Intel BayTrail has 7 I2C controller located on the PCI bus */
+	{ PCI_VDEVICE(INTEL, 0x0f41) },
+	{ PCI_VDEVICE(INTEL, 0x0f42) },
+	{ PCI_VDEVICE(INTEL, 0x0f43) },
+	{ PCI_VDEVICE(INTEL, 0x0f44) },
+	{ PCI_VDEVICE(INTEL, 0x0f45) },
+	{ PCI_VDEVICE(INTEL, 0x0f46) },
+	{ PCI_VDEVICE(INTEL, 0x0f47) },
+	{},
+};
+
+U_BOOT_PCI_DEVICE(i2c_designware, designware_pci_supported);
+#endif
+
 #endif /* CONFIG_DM_I2C */
-- 
2.7.3

^ permalink raw reply related	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
                   ` (4 preceding siblings ...)
  2016-03-18  7:54 ` [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86) Stefan Roese
@ 2016-03-18 11:09 ` Marek Vasut
  2016-03-21  8:54 ` Bin Meng
  6 siblings, 0 replies; 31+ messages in thread
From: Marek Vasut @ 2016-03-18 11:09 UTC (permalink / raw)
  To: u-boot

On 03/18/2016 08:54 AM, Stefan Roese wrote:
> Add the ic_enable_status register to the ic_regs struct. Additionally
> the register offsets are added, to better check, if the offset matches
> the register description in the datasheet.
> 
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>

Acked-by: Marek Vasut <marex@denx.de>

Best regards,
Marek Vasut

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function
  2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
@ 2016-03-18 11:12   ` Marek Vasut
  2016-03-18 12:04     ` Stefan Roese
  2016-03-21  8:54   ` Bin Meng
  1 sibling, 1 reply; 31+ messages in thread
From: Marek Vasut @ 2016-03-18 11:12 UTC (permalink / raw)
  To: u-boot

On 03/18/2016 08:54 AM, Stefan Roese wrote:
> dw_i2c_enable() is used to dis-/en-able the I2C controller. It makes
> sense to add such a function, as the controller is dis-/en-abled
> multiple times in the code. Additionally, this function now checks,
> if the controller is really dis-/en-abled. This code is copied
> from the Linux I2C driver version.
> 
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 46 +++++++++++++++++++++++++-------------------
>  1 file changed, 26 insertions(+), 20 deletions(-)
> 
> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
> index e768cde..c8ea520 100644
> --- a/drivers/i2c/designware_i2c.c
> +++ b/drivers/i2c/designware_i2c.c
> @@ -34,6 +34,26 @@ static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
>  	return NULL;
>  }
>  
> +static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
> +{
> +	int timeout = 100;
> +
> +	do {
> +		writel(enable, &i2c_base->ic_enable);

This should at least use IC_ENABLE_0B and not the boot enable.

> +		if ((readl(&i2c_base->ic_enable_status) & 1) == enable)
> +			return;
> +
> +		/*
> +		 * Wait 10 times the signaling period of the highest I2C
> +		 * transfer supported by the driver (for 400KHz this is
> +		 * 25us) as described in the DesignWare I2C databook.
> +		 */
> +		udelay(25);
> +	} while (timeout--);
> +
> +	printf("timeout in %sabling I2C adapter\n", enable ? "en" : "dis");
> +}
> +
>  /*
>   * set_speed - Set the i2c speed mode (standard, high, fast)
>   * @i2c_spd:	required i2c speed mode
> @@ -45,12 +65,9 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
>  	struct i2c_regs *i2c_base = i2c_get_base(adap);
>  	unsigned int cntl;
>  	unsigned int hcnt, lcnt;
> -	unsigned int enbl;
>  
>  	/* to set speed cltr must be disabled */
> -	enbl = readl(&i2c_base->ic_enable);
> -	enbl &= ~IC_ENABLE_0B;
> -	writel(enbl, &i2c_base->ic_enable);
> +	dw_i2c_enable(i2c_base, 0);

This and all the other places which you changed actually change the
logic of the code, right ? Is that a problem ?

Best regards,
Marek Vasut

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed()
  2016-03-18  7:54 ` [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed() Stefan Roese
@ 2016-03-18 11:13   ` Marek Vasut
  2016-03-21  8:54   ` Bin Meng
  1 sibling, 0 replies; 31+ messages in thread
From: Marek Vasut @ 2016-03-18 11:13 UTC (permalink / raw)
  To: u-boot

On 03/18/2016 08:54 AM, Stefan Roese wrote:
> Integrating set_speed() into dw_i2c_set_bus_speed() will make the
> conversion to DM easier for this driver.
> 
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>

Acked-by: Marek Vasut <marex@denx.de>

-- 
Best regards,
Marek Vasut

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function
  2016-03-18 11:12   ` Marek Vasut
@ 2016-03-18 12:04     ` Stefan Roese
  2016-03-18 12:14       ` Marek Vasut
  0 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-03-18 12:04 UTC (permalink / raw)
  To: u-boot

On 18.03.2016 12:12, Marek Vasut wrote:
> On 03/18/2016 08:54 AM, Stefan Roese wrote:
>> dw_i2c_enable() is used to dis-/en-able the I2C controller. It makes
>> sense to add such a function, as the controller is dis-/en-abled
>> multiple times in the code. Additionally, this function now checks,
>> if the controller is really dis-/en-abled. This code is copied
>> from the Linux I2C driver version.
>>
>> Signed-off-by: Stefan Roese <sr@denx.de>
>> Cc: Simon Glass <sjg@chromium.org>
>> Cc: Bin Meng <bmeng.cn@gmail.com>
>> Cc: Marek Vasut <marex@denx.de>
>> Cc: Heiko Schocher <hs@denx.de>
>> ---
>>   drivers/i2c/designware_i2c.c | 46 +++++++++++++++++++++++++-------------------
>>   1 file changed, 26 insertions(+), 20 deletions(-)
>>
>> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
>> index e768cde..c8ea520 100644
>> --- a/drivers/i2c/designware_i2c.c
>> +++ b/drivers/i2c/designware_i2c.c
>> @@ -34,6 +34,26 @@ static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
>>   	return NULL;
>>   }
>>
>> +static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>> +{
>> +	int timeout = 100;
>> +
>> +	do {
>> +		writel(enable, &i2c_base->ic_enable);
>
> This should at least use IC_ENABLE_0B and not the boot enable.
>
>> +		if ((readl(&i2c_base->ic_enable_status) & 1) == enable)
>> +			return;
>> +
>> +		/*
>> +		 * Wait 10 times the signaling period of the highest I2C
>> +		 * transfer supported by the driver (for 400KHz this is
>> +		 * 25us) as described in the DesignWare I2C databook.
>> +		 */
>> +		udelay(25);
>> +	} while (timeout--);
>> +
>> +	printf("timeout in %sabling I2C adapter\n", enable ? "en" : "dis");
>> +}
>> +
>>   /*
>>    * set_speed - Set the i2c speed mode (standard, high, fast)
>>    * @i2c_spd:	required i2c speed mode
>> @@ -45,12 +65,9 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
>>   	struct i2c_regs *i2c_base = i2c_get_base(adap);
>>   	unsigned int cntl;
>>   	unsigned int hcnt, lcnt;
>> -	unsigned int enbl;
>>
>>   	/* to set speed cltr must be disabled */
>> -	enbl = readl(&i2c_base->ic_enable);
>> -	enbl &= ~IC_ENABLE_0B;
>> -	writel(enbl, &i2c_base->ic_enable);
>> +	dw_i2c_enable(i2c_base, 0);
>
> This and all the other places which you changed actually change the
> logic of the code, right ? Is that a problem ?

It is a functional change, yes. With a now added check, if the
controller is actually getting enabled or disabled. The code is
taken from the Linux kernel, as noted in the commit text. And
I've tested this code on SoCFPGA without any issues so far.

Additional testing would be very welcome though. ;)

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function
  2016-03-18 12:04     ` Stefan Roese
@ 2016-03-18 12:14       ` Marek Vasut
  0 siblings, 0 replies; 31+ messages in thread
From: Marek Vasut @ 2016-03-18 12:14 UTC (permalink / raw)
  To: u-boot

On 03/18/2016 01:04 PM, Stefan Roese wrote:
> On 18.03.2016 12:12, Marek Vasut wrote:
>> On 03/18/2016 08:54 AM, Stefan Roese wrote:
>>> dw_i2c_enable() is used to dis-/en-able the I2C controller. It makes
>>> sense to add such a function, as the controller is dis-/en-abled
>>> multiple times in the code. Additionally, this function now checks,
>>> if the controller is really dis-/en-abled. This code is copied
>>> from the Linux I2C driver version.
>>>
>>> Signed-off-by: Stefan Roese <sr@denx.de>
>>> Cc: Simon Glass <sjg@chromium.org>
>>> Cc: Bin Meng <bmeng.cn@gmail.com>
>>> Cc: Marek Vasut <marex@denx.de>
>>> Cc: Heiko Schocher <hs@denx.de>
>>> ---
>>>   drivers/i2c/designware_i2c.c | 46
>>> +++++++++++++++++++++++++-------------------
>>>   1 file changed, 26 insertions(+), 20 deletions(-)
>>>
>>> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
>>> index e768cde..c8ea520 100644
>>> --- a/drivers/i2c/designware_i2c.c
>>> +++ b/drivers/i2c/designware_i2c.c
>>> @@ -34,6 +34,26 @@ static struct i2c_regs *i2c_get_base(struct
>>> i2c_adapter *adap)
>>>       return NULL;
>>>   }
>>>
>>> +static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>>> +{
>>> +    int timeout = 100;
>>> +
>>> +    do {
>>> +        writel(enable, &i2c_base->ic_enable);
>>
>> This should at least use IC_ENABLE_0B and not the boot enable.
>>
>>> +        if ((readl(&i2c_base->ic_enable_status) & 1) == enable)
>>> +            return;
>>> +
>>> +        /*
>>> +         * Wait 10 times the signaling period of the highest I2C
>>> +         * transfer supported by the driver (for 400KHz this is
>>> +         * 25us) as described in the DesignWare I2C databook.
>>> +         */
>>> +        udelay(25);
>>> +    } while (timeout--);
>>> +
>>> +    printf("timeout in %sabling I2C adapter\n", enable ? "en" : "dis");
>>> +}
>>> +
>>>   /*
>>>    * set_speed - Set the i2c speed mode (standard, high, fast)
>>>    * @i2c_spd:    required i2c speed mode
>>> @@ -45,12 +65,9 @@ static void set_speed(struct i2c_adapter *adap,
>>> int i2c_spd)
>>>       struct i2c_regs *i2c_base = i2c_get_base(adap);
>>>       unsigned int cntl;
>>>       unsigned int hcnt, lcnt;
>>> -    unsigned int enbl;
>>>
>>>       /* to set speed cltr must be disabled */
>>> -    enbl = readl(&i2c_base->ic_enable);
>>> -    enbl &= ~IC_ENABLE_0B;
>>> -    writel(enbl, &i2c_base->ic_enable);
>>> +    dw_i2c_enable(i2c_base, 0);
>>
>> This and all the other places which you changed actually change the
>> logic of the code, right ? Is that a problem ?
> 
> It is a functional change, yes. With a now added check, if the
> controller is actually getting enabled or disabled. The code is
> taken from the Linux kernel, as noted in the commit text. And
> I've tested this code on SoCFPGA without any issues so far.
> 
> Additional testing would be very welcome though. ;)

I will have a board ready for mainlining that uses i2c, so if something
breaks, I will cry ;-)

-- 
Best regards,
Marek Vasut

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct
  2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
                   ` (5 preceding siblings ...)
  2016-03-18 11:09 ` [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Marek Vasut
@ 2016-03-21  8:54 ` Bin Meng
  6 siblings, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> Add the ic_enable_status register to the ic_regs struct. Additionally

typo: i2c_regs

> the register offsets are added, to better check, if the offset matches
> the register description in the datasheet.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.h | 68 +++++++++++++++++++++++---------------------
>  1 file changed, 35 insertions(+), 33 deletions(-)
>
> diff --git a/drivers/i2c/designware_i2c.h b/drivers/i2c/designware_i2c.h
> index 19b09df..270c29c 100644
> --- a/drivers/i2c/designware_i2c.h
> +++ b/drivers/i2c/designware_i2c.h
> @@ -9,39 +9,41 @@
>  #define __DW_I2C_H_
>
>  struct i2c_regs {
> -       u32 ic_con;
> -       u32 ic_tar;
> -       u32 ic_sar;
> -       u32 ic_hs_maddr;
> -       u32 ic_cmd_data;
> -       u32 ic_ss_scl_hcnt;
> -       u32 ic_ss_scl_lcnt;
> -       u32 ic_fs_scl_hcnt;
> -       u32 ic_fs_scl_lcnt;
> -       u32 ic_hs_scl_hcnt;
> -       u32 ic_hs_scl_lcnt;
> -       u32 ic_intr_stat;
> -       u32 ic_intr_mask;
> -       u32 ic_raw_intr_stat;
> -       u32 ic_rx_tl;
> -       u32 ic_tx_tl;
> -       u32 ic_clr_intr;
> -       u32 ic_clr_rx_under;
> -       u32 ic_clr_rx_over;
> -       u32 ic_clr_tx_over;
> -       u32 ic_clr_rd_req;
> -       u32 ic_clr_tx_abrt;
> -       u32 ic_clr_rx_done;
> -       u32 ic_clr_activity;
> -       u32 ic_clr_stop_det;
> -       u32 ic_clr_start_det;
> -       u32 ic_clr_gen_call;
> -       u32 ic_enable;
> -       u32 ic_status;
> -       u32 ic_txflr;
> -       u32 ix_rxflr;
> -       u32 reserved_1;
> -       u32 ic_tx_abrt_source;
> +       u32 ic_con;             /* 0x00 */
> +       u32 ic_tar;             /* 0x04 */
> +       u32 ic_sar;             /* 0x08 */
> +       u32 ic_hs_maddr;        /* 0x0c */
> +       u32 ic_cmd_data;        /* 0x10 */
> +       u32 ic_ss_scl_hcnt;     /* 0x14 */
> +       u32 ic_ss_scl_lcnt;     /* 0x18 */
> +       u32 ic_fs_scl_hcnt;     /* 0x1c */
> +       u32 ic_fs_scl_lcnt;     /* 0x20 */
> +       u32 ic_hs_scl_hcnt;     /* 0x24 */
> +       u32 ic_hs_scl_lcnt;     /* 0x28 */
> +       u32 ic_intr_stat;       /* 0x2c */
> +       u32 ic_intr_mask;       /* 0x30 */
> +       u32 ic_raw_intr_stat;   /* 0x34 */
> +       u32 ic_rx_tl;           /* 0x38 */
> +       u32 ic_tx_tl;           /* 0x3c */
> +       u32 ic_clr_intr;        /* 0x40 */
> +       u32 ic_clr_rx_under;    /* 0x44 */
> +       u32 ic_clr_rx_over;     /* 0x48 */
> +       u32 ic_clr_tx_over;     /* 0x4c */
> +       u32 ic_clr_rd_req;      /* 0x50 */
> +       u32 ic_clr_tx_abrt;     /* 0x54 */
> +       u32 ic_clr_rx_done;     /* 0x58 */
> +       u32 ic_clr_activity;    /* 0x5c */
> +       u32 ic_clr_stop_det;    /* 0x60 */
> +       u32 ic_clr_start_det;   /* 0x64 */
> +       u32 ic_clr_gen_call;    /* 0x68 */
> +       u32 ic_enable;          /* 0x6c */
> +       u32 ic_status;          /* 0x70 */
> +       u32 ic_txflr;           /* 0x74 */
> +       u32 ic_rxflr;           /* 0x78 */
> +       u32 ic_sda_hold;        /* 0x7c */
> +       u32 ic_tx_abrt_source;  /* 0x80 */
> +       u8 res1[0x18];          /* 0x84 */
> +       u32 ic_enable_status;   /* 0x9c */
>  };
>

Other than that,
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function
  2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
  2016-03-18 11:12   ` Marek Vasut
@ 2016-03-21  8:54   ` Bin Meng
  1 sibling, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> dw_i2c_enable() is used to dis-/en-able the I2C controller. It makes
> sense to add such a function, as the controller is dis-/en-abled
> multiple times in the code. Additionally, this function now checks,
> if the controller is really dis-/en-abled. This code is copied
> from the Linux I2C driver version.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 46 +++++++++++++++++++++++++-------------------
>  1 file changed, 26 insertions(+), 20 deletions(-)
>
> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
> index e768cde..c8ea520 100644
> --- a/drivers/i2c/designware_i2c.c
> +++ b/drivers/i2c/designware_i2c.c
> @@ -34,6 +34,26 @@ static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
>         return NULL;
>  }
>
> +static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
> +{
> +       int timeout = 100;
> +
> +       do {
> +               writel(enable, &i2c_base->ic_enable);
> +               if ((readl(&i2c_base->ic_enable_status) & 1) == enable)
> +                       return;
> +
> +               /*
> +                * Wait 10 times the signaling period of the highest I2C
> +                * transfer supported by the driver (for 400KHz this is
> +                * 25us) as described in the DesignWare I2C databook.
> +                */
> +               udelay(25);
> +       } while (timeout--);
> +
> +       printf("timeout in %sabling I2C adapter\n", enable ? "en" : "dis");
> +}
> +
>  /*
>   * set_speed - Set the i2c speed mode (standard, high, fast)
>   * @i2c_spd:   required i2c speed mode
> @@ -45,12 +65,9 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
>         struct i2c_regs *i2c_base = i2c_get_base(adap);
>         unsigned int cntl;
>         unsigned int hcnt, lcnt;
> -       unsigned int enbl;
>
>         /* to set speed cltr must be disabled */
> -       enbl = readl(&i2c_base->ic_enable);
> -       enbl &= ~IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 0);

nits: change 0 to false since the parameter type is bool?

>
>         cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
>
> @@ -84,8 +101,7 @@ static void set_speed(struct i2c_adapter *adap, int i2c_spd)
>         writel(cntl, &i2c_base->ic_con);
>
>         /* Enable back i2c now speed set */
> -       enbl |= IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 1);

ditto, 1-> true?

>  }
>
>  /*
> @@ -123,12 +139,9 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
>                         int slaveaddr)
>  {
>         struct i2c_regs *i2c_base = i2c_get_base(adap);
> -       unsigned int enbl;
>
>         /* Disable i2c */
> -       enbl = readl(&i2c_base->ic_enable);
> -       enbl &= ~IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 0);

ditto

>
>         writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
>         writel(IC_RX_TL, &i2c_base->ic_rx_tl);
> @@ -138,9 +151,7 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
>         writel(slaveaddr, &i2c_base->ic_sar);
>
>         /* Enable i2c */
> -       enbl = readl(&i2c_base->ic_enable);
> -       enbl |= IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 1);

ditto

>  }
>
>  /*
> @@ -152,19 +163,14 @@ static void dw_i2c_init(struct i2c_adapter *adap, int speed,
>  static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
>  {
>         struct i2c_regs *i2c_base = i2c_get_base(adap);
> -       unsigned int enbl;
>
>         /* Disable i2c */
> -       enbl = readl(&i2c_base->ic_enable);
> -       enbl &= ~IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 0);
>
>         writel(i2c_addr, &i2c_base->ic_tar);
>
>         /* Enable i2c */
> -       enbl = readl(&i2c_base->ic_enable);
> -       enbl |= IC_ENABLE_0B;
> -       writel(enbl, &i2c_base->ic_enable);
> +       dw_i2c_enable(i2c_base, 1);
>  }
>

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed()
  2016-03-18  7:54 ` [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed() Stefan Roese
  2016-03-18 11:13   ` Marek Vasut
@ 2016-03-21  8:54   ` Bin Meng
  1 sibling, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> Integrating set_speed() into dw_i2c_set_bus_speed() will make the
> conversion to DM easier for this driver.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 37 +++++++++++++------------------------
>  1 file changed, 13 insertions(+), 24 deletions(-)
>

Reviewed-by: Bin Meng <bmeng.cn@gmail.com>

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion
  2016-03-18  7:54 ` [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion Stefan Roese
@ 2016-03-21  8:54   ` Bin Meng
  0 siblings, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> This patch prepares the designware I2C driver for the DM conversion.
> This is mainly done by removing struct i2c_adapter from the functions
> that shall be used by the DM driver version as well.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 173 ++++++++++++++++++++++---------------------
>  1 file changed, 90 insertions(+), 83 deletions(-)
>
> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
> index 508dac9..e51e6de 100644
> --- a/drivers/i2c/designware_i2c.c
> +++ b/drivers/i2c/designware_i2c.c
> @@ -10,30 +10,6 @@
>  #include <asm/io.h>
>  #include "designware_i2c.h"
>
> -static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
> -{
> -       switch (adap->hwadapnr) {
> -#if CONFIG_SYS_I2C_BUS_MAX >= 4
> -       case 3:
> -               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE3;
> -#endif
> -#if CONFIG_SYS_I2C_BUS_MAX >= 3
> -       case 2:
> -               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE2;
> -#endif
> -#if CONFIG_SYS_I2C_BUS_MAX >= 2
> -       case 1:
> -               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE1;
> -#endif
> -       case 0:
> -               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE;
> -       default:
> -               printf("Wrong I2C-adapter number %d\n", adap->hwadapnr);
> -       }
> -
> -       return NULL;
> -}
> -
>  static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>  {
>         int timeout = 100;
> @@ -60,10 +36,9 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>   *
>   * Set the i2c speed.
>   */
> -static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
> -                                        unsigned int speed)
> +static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
> +                                          unsigned int speed)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
>         unsigned int cntl;
>         unsigned int hcnt, lcnt;
>         int i2c_spd;
> @@ -112,47 +87,17 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
>         /* Enable back i2c now speed set */
>         dw_i2c_enable(i2c_base, 1);
>
> -       adap->speed = speed;
> -
>         return 0;
>  }
>
>  /*
> - * i2c_init - Init function
> - * @speed:     required i2c speed
> - * @slaveaddr: slave address for the device
> - *
> - * Initialization function.
> - */
> -static void dw_i2c_init(struct i2c_adapter *adap, int speed,
> -                       int slaveaddr)
> -{
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
> -
> -       /* Disable i2c */
> -       dw_i2c_enable(i2c_base, 0);
> -
> -       writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
> -       writel(IC_RX_TL, &i2c_base->ic_rx_tl);
> -       writel(IC_TX_TL, &i2c_base->ic_tx_tl);
> -       dw_i2c_set_bus_speed(adap, speed);
> -       writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
> -       writel(slaveaddr, &i2c_base->ic_sar);
> -
> -       /* Enable i2c */
> -       dw_i2c_enable(i2c_base, 1);
> -}
> -
> -/*
>   * i2c_setaddress - Sets the target slave address
>   * @i2c_addr:  target i2c address
>   *
>   * Sets the target slave address.
>   */
> -static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
> +static void i2c_setaddress(struct i2c_regs *i2c_base, unsigned int i2c_addr)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
> -
>         /* Disable i2c */
>         dw_i2c_enable(i2c_base, 0);
>
> @@ -167,10 +112,8 @@ static void i2c_setaddress(struct i2c_adapter *adap, unsigned int i2c_addr)
>   *
>   * Flushes the i2c RX FIFO
>   */
> -static void i2c_flush_rxfifo(struct i2c_adapter *adap)
> +static void i2c_flush_rxfifo(struct i2c_regs *i2c_base)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
> -
>         while (readl(&i2c_base->ic_status) & IC_STATUS_RFNE)
>                 readl(&i2c_base->ic_cmd_data);
>  }
> @@ -180,9 +123,8 @@ static void i2c_flush_rxfifo(struct i2c_adapter *adap)
>   *
>   * Waits for bus busy
>   */
> -static int i2c_wait_for_bb(struct i2c_adapter *adap)
> +static int i2c_wait_for_bb(struct i2c_regs *i2c_base)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
>         unsigned long start_time_bb = get_timer(0);
>
>         while ((readl(&i2c_base->ic_status) & IC_STATUS_MA) ||
> @@ -196,15 +138,13 @@ static int i2c_wait_for_bb(struct i2c_adapter *adap)
>         return 0;
>  }
>
> -static int i2c_xfer_init(struct i2c_adapter *adap, uchar chip, uint addr,
> +static int i2c_xfer_init(struct i2c_regs *i2c_base, uchar chip, uint addr,
>                          int alen)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
> -
> -       if (i2c_wait_for_bb(adap))
> +       if (i2c_wait_for_bb(i2c_base))
>                 return 1;
>
> -       i2c_setaddress(adap, chip);
> +       i2c_setaddress(i2c_base, chip);
>         while (alen) {
>                 alen--;
>                 /* high byte address going out first */
> @@ -214,9 +154,8 @@ static int i2c_xfer_init(struct i2c_adapter *adap, uchar chip, uint addr,
>         return 0;
>  }
>
> -static int i2c_xfer_finish(struct i2c_adapter *adap)
> +static int i2c_xfer_finish(struct i2c_regs *i2c_base)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
>         ulong start_stop_det = get_timer(0);
>
>         while (1) {
> @@ -228,12 +167,12 @@ static int i2c_xfer_finish(struct i2c_adapter *adap)
>                 }
>         }
>
> -       if (i2c_wait_for_bb(adap)) {
> +       if (i2c_wait_for_bb(i2c_base)) {
>                 printf("Timed out waiting for bus\n");
>                 return 1;
>         }
>
> -       i2c_flush_rxfifo(adap);
> +       i2c_flush_rxfifo(i2c_base);
>
>         return 0;
>  }
> @@ -248,10 +187,9 @@ static int i2c_xfer_finish(struct i2c_adapter *adap)
>   *
>   * Read from i2c memory.
>   */
> -static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
> -                      int alen, u8 *buffer, int len)
> +static int __dw_i2c_read(struct i2c_regs *i2c_base, u8 dev, uint addr,
> +                        int alen, u8 *buffer, int len)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
>         unsigned long start_time_rx;
>
>  #ifdef CONFIG_SYS_I2C_EEPROM_ADDR_OVERFLOW
> @@ -273,7 +211,7 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
>               addr);
>  #endif
>
> -       if (i2c_xfer_init(adap, dev, addr, alen))
> +       if (i2c_xfer_init(i2c_base, dev, addr, alen))
>                 return 1;
>
>         start_time_rx = get_timer(0);
> @@ -293,7 +231,7 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
>                 }
>         }
>
> -       return i2c_xfer_finish(adap);
> +       return i2c_xfer_finish(i2c_base);
>  }
>
>  /*
> @@ -306,10 +244,9 @@ static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
>   *
>   * Write to i2c memory.
>   */
> -static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
> -                       int alen, u8 *buffer, int len)
> +static int __dw_i2c_write(struct i2c_regs *i2c_base, u8 dev, uint addr,
> +                         int alen, u8 *buffer, int len)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
>         int nb = len;
>         unsigned long start_time_tx;
>
> @@ -332,7 +269,7 @@ static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
>               addr);
>  #endif
>
> -       if (i2c_xfer_init(adap, dev, addr, alen))
> +       if (i2c_xfer_init(i2c_base, dev, addr, alen))
>                 return 1;
>
>         start_time_tx = get_timer(0);
> @@ -353,7 +290,76 @@ static int dw_i2c_write(struct i2c_adapter *adap, u8 dev, uint addr,
>                 }
>         }
>
> -       return i2c_xfer_finish(adap);
> +       return i2c_xfer_finish(i2c_base);
> +}
> +
> +static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
> +{
> +       switch (adap->hwadapnr) {
> +#if CONFIG_SYS_I2C_BUS_MAX >= 4
> +       case 3:
> +               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE3;
> +#endif
> +#if CONFIG_SYS_I2C_BUS_MAX >= 3
> +       case 2:
> +               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE2;
> +#endif
> +#if CONFIG_SYS_I2C_BUS_MAX >= 2
> +       case 1:
> +               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE1;
> +#endif
> +       case 0:
> +               return (struct i2c_regs *)CONFIG_SYS_I2C_BASE;
> +       default:
> +               printf("Wrong I2C-adapter number %d\n", adap->hwadapnr);
> +       }
> +
> +       return NULL;
> +}
> +
> +static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
> +                                        unsigned int speed)
> +{
> +       adap->speed = speed;
> +       return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
> +}
> +
> +/*
> + * i2c_init - Init function

dw_i2c_init

> + * @speed:     required i2c speed
> + * @slaveaddr: slave address for the device
> + *
> + * Initialization function.
> + */
> +static void dw_i2c_init(struct i2c_adapter *adap, int speed,
> +                       int slaveaddr)
> +{
> +       struct i2c_regs *i2c_base = i2c_get_base(adap);
> +
> +       /* Disable i2c */
> +       dw_i2c_enable(i2c_base, 0);
> +
> +       writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
> +       writel(IC_RX_TL, &i2c_base->ic_rx_tl);
> +       writel(IC_TX_TL, &i2c_base->ic_tx_tl);
> +       dw_i2c_set_bus_speed(adap, speed);
> +       writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
> +       writel(slaveaddr, &i2c_base->ic_sar);
> +
> +       /* Enable i2c */
> +       dw_i2c_enable(i2c_base, 1);
> +}
> +

Reviewed-by: Bin Meng <bmeng.cn@gmail.com>

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support
  2016-03-18  7:54 ` [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support Stefan Roese
@ 2016-03-21  8:54   ` Bin Meng
  2016-04-09 18:35   ` Simon Glass
  1 sibling, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> This patch adds DM support to the designware I2C driver. It currently
> supports DM and the legacy I2C support. The legacy support should be
> removed, once all platforms using it have DM enabled.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 149 ++++++++++++++++++++++++++++++++++++-------
>  1 file changed, 126 insertions(+), 23 deletions(-)
>
> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
> index e51e6de..4e5340d 100644
> --- a/drivers/i2c/designware_i2c.c
> +++ b/drivers/i2c/designware_i2c.c
> @@ -6,10 +6,15 @@
>   */
>
>  #include <common.h>
> +#include <dm.h>
>  #include <i2c.h>
>  #include <asm/io.h>
>  #include "designware_i2c.h"
>
> +struct dw_i2c {
> +       struct i2c_regs *regs;
> +};
> +
>  static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>  {
>         int timeout = 100;
> @@ -293,6 +298,36 @@ static int __dw_i2c_write(struct i2c_regs *i2c_base, u8 dev, uint addr,
>         return i2c_xfer_finish(i2c_base);
>  }
>
> +/*
> + * i2c_init - Init function

__dw_i2c_init

> + * @speed:     required i2c speed
> + * @slaveaddr: slave address for the device
> + *
> + * Initialization function.
> + */
> +static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
> +{
> +       /* Disable i2c */
> +       dw_i2c_enable(i2c_base, 0);
> +
> +       writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
> +       writel(IC_RX_TL, &i2c_base->ic_rx_tl);
> +       writel(IC_TX_TL, &i2c_base->ic_tx_tl);
> +       writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
> +#ifndef CONFIG_DM_I2C
> +       __dw_i2c_set_bus_speed(i2c_base, speed);
> +       writel(slaveaddr, &i2c_base->ic_sar);
> +#endif
> +
> +       /* Enable i2c */
> +       dw_i2c_enable(i2c_base, 1);
> +}
> +
> +#ifndef CONFIG_DM_I2C
> +/*
> + * The legacy I2C functions. These need to get removed once
> + * all users of this driver are converted to DM.
> + */
>  static struct i2c_regs *i2c_get_base(struct i2c_adapter *adap)
>  {
>         switch (adap->hwadapnr) {
> @@ -324,30 +359,9 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
>         return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
>  }
>
> -/*
> - * i2c_init - Init function
> - * @speed:     required i2c speed
> - * @slaveaddr: slave address for the device
> - *
> - * Initialization function.
> - */
> -static void dw_i2c_init(struct i2c_adapter *adap, int speed,
> -                       int slaveaddr)
> +static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
>  {
> -       struct i2c_regs *i2c_base = i2c_get_base(adap);
> -
> -       /* Disable i2c */
> -       dw_i2c_enable(i2c_base, 0);
> -
> -       writel((IC_CON_SD | IC_CON_SPD_FS | IC_CON_MM), &i2c_base->ic_con);
> -       writel(IC_RX_TL, &i2c_base->ic_rx_tl);
> -       writel(IC_TX_TL, &i2c_base->ic_tx_tl);
> -       dw_i2c_set_bus_speed(adap, speed);
> -       writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
> -       writel(slaveaddr, &i2c_base->ic_sar);
> -
> -       /* Enable i2c */
> -       dw_i2c_enable(i2c_base, 1);
> +       __dw_i2c_init(i2c_get_base(adap), speed, slaveaddr);
>  }
>
>  static int dw_i2c_read(struct i2c_adapter *adap, u8 dev, uint addr,
> @@ -402,3 +416,92 @@ U_BOOT_I2C_ADAP_COMPLETE(dw_3, dw_i2c_init, dw_i2c_probe, dw_i2c_read,
>                          dw_i2c_write, dw_i2c_set_bus_speed,
>                          CONFIG_SYS_I2C_SPEED3, CONFIG_SYS_I2C_SLAVE3, 3)
>  #endif
> +
> +#else /* CONFIG_DM_I2C */
> +/*
> + * The DM I2C functions
> + */

nits: single line comment format

> +
> +static int designware_i2c_xfer(struct udevice *bus, struct i2c_msg *msg,
> +                              int nmsgs)
> +{
> +       struct dw_i2c *i2c = dev_get_priv(bus);
> +       int ret;
> +
> +       debug("i2c_xfer: %d messages\n", nmsgs);
> +       for (; nmsgs > 0; nmsgs--, msg++) {
> +               debug("i2c_xfer: chip=0x%x, len=0x%x\n", msg->addr, msg->len);
> +               if (msg->flags & I2C_M_RD) {
> +                       ret = __dw_i2c_read(i2c->regs, msg->addr, 0, 0,
> +                                           msg->buf, msg->len);
> +               } else {
> +                       ret = __dw_i2c_write(i2c->regs, msg->addr, 0, 0,
> +                                            msg->buf, msg->len);
> +               }
> +               if (ret) {
> +                       debug("i2c_write: error sending\n");
> +                       return -EREMOTEIO;
> +               }
> +       }
> +
> +       return 0;
> +}
> +
> +static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
> +{
> +       struct dw_i2c *i2c = dev_get_priv(bus);
> +
> +       return __dw_i2c_set_bus_speed(i2c->regs, speed);
> +}
> +
> +static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
> +                                    uint chip_flags)
> +{
> +       struct dw_i2c *i2c = dev_get_priv(bus);
> +       struct i2c_regs *i2c_base = i2c->regs;
> +       u32 tmp;
> +       int ret;
> +
> +       /*
> +        * Try to read the first location of the chip.
> +        */

nits: single line comment format

> +       ret = __dw_i2c_read(i2c_base, chip_addr, 0, 1, (uchar *)&tmp, 1);
> +       if (ret)
> +               __dw_i2c_init(i2c_base, 0, 0);
> +
> +       return ret;
> +}
> +
> +static int designware_i2c_probe(struct udevice *bus)
> +{
> +       struct dw_i2c *priv = dev_get_priv(bus);
> +
> +       /* Save base address from device-tree */
> +       priv->regs = (struct i2c_regs *)dev_get_addr(bus);
> +
> +       __dw_i2c_init(priv->regs, 0, 0);
> +
> +       return 0;
> +}
> +
> +static const struct dm_i2c_ops designware_i2c_ops = {
> +       .xfer           = designware_i2c_xfer,
> +       .probe_chip     = designware_i2c_probe_chip,
> +       .set_bus_speed  = designware_i2c_set_bus_speed,
> +};
> +
> +static const struct udevice_id designware_i2c_ids[] = {
> +       { .compatible = "snps,designware-i2c" },
> +       { }
> +};
> +
> +U_BOOT_DRIVER(i2c_designware) = {
> +       .name   = "i2c_designware",
> +       .id     = UCLASS_I2C,
> +       .of_match = designware_i2c_ids,
> +       .probe  = designware_i2c_probe,
> +       .priv_auto_alloc_size = sizeof(struct dw_i2c),
> +       .ops    = &designware_i2c_ops,
> +};
> +
> +#endif /* CONFIG_DM_I2C */
> --

Other than that,
Reviewed-by: Bin Meng <bmeng.cn@gmail.com>

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-18  7:54 ` [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86) Stefan Roese
@ 2016-03-21  8:54   ` Bin Meng
  2016-03-21  9:03     ` Stefan Roese
  0 siblings, 1 reply; 31+ messages in thread
From: Bin Meng @ 2016-03-21  8:54 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
> This patch adds support for the PCI(e) based I2C cores. Which can be
> found for example on the Intel Bay Trail SoC. It has 7 I2C controllers
> implemented as PCI devices.
>
> This patch also adds the fixed values for the timing registers for
> BayTrail which are taken from the Linux designware I2C driver.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 111 +++++++++++++++++++++++++++++++++++++++----
>  1 file changed, 101 insertions(+), 10 deletions(-)
>
> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
> index 4e5340d..f7f2eba 100644
> --- a/drivers/i2c/designware_i2c.c
> +++ b/drivers/i2c/designware_i2c.c
> @@ -8,11 +8,32 @@
>  #include <common.h>
>  #include <dm.h>
>  #include <i2c.h>
> +#include <pci.h>
>  #include <asm/io.h>
>  #include "designware_i2c.h"
>
> +struct dw_scl_sda_cfg {
> +       u32 ss_hcnt;
> +       u32 fs_hcnt;
> +       u32 ss_lcnt;
> +       u32 fs_lcnt;
> +       u32 sda_hold;
> +};
> +
> +#ifdef CONFIG_X86
> +/* BayTrail HCNT/LCNT/SDA hold time */
> +static struct dw_scl_sda_cfg byt_config = {
> +       .ss_hcnt = 0x200,
> +       .fs_hcnt = 0x55,
> +       .ss_lcnt = 0x200,
> +       .fs_lcnt = 0x99,
> +       .sda_hold = 0x6,
> +};
> +#endif
> +
>  struct dw_i2c {
>         struct i2c_regs *regs;
> +       struct dw_scl_sda_cfg *scl_sda_cfg;
>  };
>
>  static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
> @@ -42,6 +63,7 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>   * Set the i2c speed.
>   */
>  static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
> +                                          struct dw_scl_sda_cfg *scl_sda_cfg,
>                                            unsigned int speed)
>  {
>         unsigned int cntl;
> @@ -61,34 +83,55 @@ static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
>         cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
>
>         switch (i2c_spd) {
> +#ifndef CONFIG_X86 /* No High-speed for BayTrail yet */
>         case IC_SPEED_MODE_MAX:
> -               cntl |= IC_CON_SPD_HS;
> -               hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +               cntl |= IC_CON_SPD_SS;
> +               if (scl_sda_cfg) {
> +                       hcnt = scl_sda_cfg->fs_hcnt;
> +                       lcnt = scl_sda_cfg->fs_lcnt;
> +               } else {
> +                       hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +                       lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
> +               }
>                 writel(hcnt, &i2c_base->ic_hs_scl_hcnt);
> -               lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
>                 writel(lcnt, &i2c_base->ic_hs_scl_lcnt);
>                 break;
> +#endif
>
>         case IC_SPEED_MODE_STANDARD:
>                 cntl |= IC_CON_SPD_SS;
> -               hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +               if (scl_sda_cfg) {
> +                       hcnt = scl_sda_cfg->ss_hcnt;
> +                       lcnt = scl_sda_cfg->ss_lcnt;
> +               } else {
> +                       hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +                       lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
> +               }
>                 writel(hcnt, &i2c_base->ic_ss_scl_hcnt);
> -               lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
>                 writel(lcnt, &i2c_base->ic_ss_scl_lcnt);
>                 break;
>
>         case IC_SPEED_MODE_FAST:
>         default:
>                 cntl |= IC_CON_SPD_FS;
> -               hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +               if (scl_sda_cfg) {
> +                       hcnt = scl_sda_cfg->fs_hcnt;
> +                       lcnt = scl_sda_cfg->fs_lcnt;
> +               } else {
> +                       hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
> +                       lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
> +               }
>                 writel(hcnt, &i2c_base->ic_fs_scl_hcnt);
> -               lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
>                 writel(lcnt, &i2c_base->ic_fs_scl_lcnt);
>                 break;
>         }
>
>         writel(cntl, &i2c_base->ic_con);
>
> +       /* Configure SDA Hold Time if required */
> +       if (scl_sda_cfg)
> +               writel(scl_sda_cfg->sda_hold, &i2c_base->ic_sda_hold);
> +
>         /* Enable back i2c now speed set */
>         dw_i2c_enable(i2c_base, 1);
>
> @@ -315,7 +358,7 @@ static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
>         writel(IC_TX_TL, &i2c_base->ic_tx_tl);
>         writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
>  #ifndef CONFIG_DM_I2C
> -       __dw_i2c_set_bus_speed(i2c_base, speed);
> +       __dw_i2c_set_bus_speed(i2c_base, NULL, speed);
>         writel(slaveaddr, &i2c_base->ic_sar);
>  #endif
>
> @@ -356,7 +399,7 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
>                                          unsigned int speed)
>  {
>         adap->speed = speed;
> -       return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
> +       return __dw_i2c_set_bus_speed(i2c_get_base(adap), NULL, speed);
>  }
>
>  static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
> @@ -451,7 +494,7 @@ static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
>  {
>         struct dw_i2c *i2c = dev_get_priv(bus);
>
> -       return __dw_i2c_set_bus_speed(i2c->regs, speed);
> +       return __dw_i2c_set_bus_speed(i2c->regs, i2c->scl_sda_cfg, speed);
>  }
>
>  static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>  {
>         struct dw_i2c *priv = dev_get_priv(bus);
>
> +#ifdef CONFIG_X86
> +       /* Save base address from PCI BAR */
> +       priv->regs = (struct i2c_regs *)
> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
> +       /* Use BayTrail specific timing values */
> +       priv->scl_sda_cfg = &byt_config;
> +#else

How about:

    if (device_is_on_pci_bus(dev)) {
    do the PCI I2C stuff here;
    }

See driver/net/designware.c for example.

>         /* Save base address from device-tree */
>         priv->regs = (struct i2c_regs *)dev_get_addr(bus);
> +#endif
>
>         __dw_i2c_init(priv->regs, 0, 0);
>
>         return 0;
>  }
>
> +static int designware_i2c_bind(struct udevice *dev)
> +{
> +       static int num_cards;
> +       char name[20];
> +
> +       /* Create a unique device name for PCI type devices */
> +       if (device_is_on_pci_bus(dev)) {
> +               /*
> +                * ToDo:
> +                * Setting req_seq in the driver is probably not recommended.
> +                * But without a DT alias the number is not configured. And
> +                * using this driver is impossible for PCIe I2C devices.
> +                * This can be removed, once a better (correct) way for this
> +                * is found and implemented.
> +                */
> +               dev->req_seq = num_cards;
> +               sprintf(name, "i2c_designware#%u", num_cards++);
> +               device_set_name(dev, name);
> +       }

I believe the above should be wrapped by #ifdef CONFIG_DM_PCI #endif,
otherwise it won't build when DM_PCI is not enabled.

> +
> +       return 0;
> +}
> +
>  static const struct dm_i2c_ops designware_i2c_ops = {
>         .xfer           = designware_i2c_xfer,
>         .probe_chip     = designware_i2c_probe_chip,
> @@ -499,9 +573,26 @@ U_BOOT_DRIVER(i2c_designware) = {
>         .name   = "i2c_designware",
>         .id     = UCLASS_I2C,
>         .of_match = designware_i2c_ids,
> +       .bind   = designware_i2c_bind,
>         .probe  = designware_i2c_probe,
>         .priv_auto_alloc_size = sizeof(struct dw_i2c),
>         .ops    = &designware_i2c_ops,
>  };
>
> +#ifdef CONFIG_X86
> +static struct pci_device_id designware_pci_supported[] = {
> +       /* Intel BayTrail has 7 I2C controller located on the PCI bus */
> +       { PCI_VDEVICE(INTEL, 0x0f41) },
> +       { PCI_VDEVICE(INTEL, 0x0f42) },
> +       { PCI_VDEVICE(INTEL, 0x0f43) },
> +       { PCI_VDEVICE(INTEL, 0x0f44) },
> +       { PCI_VDEVICE(INTEL, 0x0f45) },
> +       { PCI_VDEVICE(INTEL, 0x0f46) },
> +       { PCI_VDEVICE(INTEL, 0x0f47) },
> +       {},
> +};
> +
> +U_BOOT_PCI_DEVICE(i2c_designware, designware_pci_supported);
> +#endif
> +
>  #endif /* CONFIG_DM_I2C */
> --

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21  8:54   ` Bin Meng
@ 2016-03-21  9:03     ` Stefan Roese
  2016-03-21  9:05       ` Bin Meng
  2016-03-21 12:04       ` Stefan Roese
  0 siblings, 2 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-21  9:03 UTC (permalink / raw)
  To: u-boot

Hi Bin,

On 21.03.2016 09:54, Bin Meng wrote:
> On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
>> This patch adds support for the PCI(e) based I2C cores. Which can be
>> found for example on the Intel Bay Trail SoC. It has 7 I2C controllers
>> implemented as PCI devices.
>>
>> This patch also adds the fixed values for the timing registers for
>> BayTrail which are taken from the Linux designware I2C driver.
>>
>> Signed-off-by: Stefan Roese <sr@denx.de>
>> Cc: Simon Glass <sjg@chromium.org>
>> Cc: Bin Meng <bmeng.cn@gmail.com>
>> Cc: Marek Vasut <marex@denx.de>
>> Cc: Heiko Schocher <hs@denx.de>
>> ---
>>   drivers/i2c/designware_i2c.c | 111 +++++++++++++++++++++++++++++++++++++++----
>>   1 file changed, 101 insertions(+), 10 deletions(-)
>>
>> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
>> index 4e5340d..f7f2eba 100644
>> --- a/drivers/i2c/designware_i2c.c
>> +++ b/drivers/i2c/designware_i2c.c
>> @@ -8,11 +8,32 @@
>>   #include <common.h>
>>   #include <dm.h>
>>   #include <i2c.h>
>> +#include <pci.h>
>>   #include <asm/io.h>
>>   #include "designware_i2c.h"
>>
>> +struct dw_scl_sda_cfg {
>> +       u32 ss_hcnt;
>> +       u32 fs_hcnt;
>> +       u32 ss_lcnt;
>> +       u32 fs_lcnt;
>> +       u32 sda_hold;
>> +};
>> +
>> +#ifdef CONFIG_X86
>> +/* BayTrail HCNT/LCNT/SDA hold time */
>> +static struct dw_scl_sda_cfg byt_config = {
>> +       .ss_hcnt = 0x200,
>> +       .fs_hcnt = 0x55,
>> +       .ss_lcnt = 0x200,
>> +       .fs_lcnt = 0x99,
>> +       .sda_hold = 0x6,
>> +};
>> +#endif
>> +
>>   struct dw_i2c {
>>          struct i2c_regs *regs;
>> +       struct dw_scl_sda_cfg *scl_sda_cfg;
>>   };
>>
>>   static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>> @@ -42,6 +63,7 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>>    * Set the i2c speed.
>>    */
>>   static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
>> +                                          struct dw_scl_sda_cfg *scl_sda_cfg,
>>                                             unsigned int speed)
>>   {
>>          unsigned int cntl;
>> @@ -61,34 +83,55 @@ static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
>>          cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
>>
>>          switch (i2c_spd) {
>> +#ifndef CONFIG_X86 /* No High-speed for BayTrail yet */
>>          case IC_SPEED_MODE_MAX:
>> -               cntl |= IC_CON_SPD_HS;
>> -               hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +               cntl |= IC_CON_SPD_SS;
>> +               if (scl_sda_cfg) {
>> +                       hcnt = scl_sda_cfg->fs_hcnt;
>> +                       lcnt = scl_sda_cfg->fs_lcnt;
>> +               } else {
>> +                       hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +                       lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
>> +               }
>>                  writel(hcnt, &i2c_base->ic_hs_scl_hcnt);
>> -               lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
>>                  writel(lcnt, &i2c_base->ic_hs_scl_lcnt);
>>                  break;
>> +#endif
>>
>>          case IC_SPEED_MODE_STANDARD:
>>                  cntl |= IC_CON_SPD_SS;
>> -               hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +               if (scl_sda_cfg) {
>> +                       hcnt = scl_sda_cfg->ss_hcnt;
>> +                       lcnt = scl_sda_cfg->ss_lcnt;
>> +               } else {
>> +                       hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +                       lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
>> +               }
>>                  writel(hcnt, &i2c_base->ic_ss_scl_hcnt);
>> -               lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
>>                  writel(lcnt, &i2c_base->ic_ss_scl_lcnt);
>>                  break;
>>
>>          case IC_SPEED_MODE_FAST:
>>          default:
>>                  cntl |= IC_CON_SPD_FS;
>> -               hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +               if (scl_sda_cfg) {
>> +                       hcnt = scl_sda_cfg->fs_hcnt;
>> +                       lcnt = scl_sda_cfg->fs_lcnt;
>> +               } else {
>> +                       hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
>> +                       lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
>> +               }
>>                  writel(hcnt, &i2c_base->ic_fs_scl_hcnt);
>> -               lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
>>                  writel(lcnt, &i2c_base->ic_fs_scl_lcnt);
>>                  break;
>>          }
>>
>>          writel(cntl, &i2c_base->ic_con);
>>
>> +       /* Configure SDA Hold Time if required */
>> +       if (scl_sda_cfg)
>> +               writel(scl_sda_cfg->sda_hold, &i2c_base->ic_sda_hold);
>> +
>>          /* Enable back i2c now speed set */
>>          dw_i2c_enable(i2c_base, 1);
>>
>> @@ -315,7 +358,7 @@ static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
>>          writel(IC_TX_TL, &i2c_base->ic_tx_tl);
>>          writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
>>   #ifndef CONFIG_DM_I2C
>> -       __dw_i2c_set_bus_speed(i2c_base, speed);
>> +       __dw_i2c_set_bus_speed(i2c_base, NULL, speed);
>>          writel(slaveaddr, &i2c_base->ic_sar);
>>   #endif
>>
>> @@ -356,7 +399,7 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
>>                                           unsigned int speed)
>>   {
>>          adap->speed = speed;
>> -       return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
>> +       return __dw_i2c_set_bus_speed(i2c_get_base(adap), NULL, speed);
>>   }
>>
>>   static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
>> @@ -451,7 +494,7 @@ static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
>>   {
>>          struct dw_i2c *i2c = dev_get_priv(bus);
>>
>> -       return __dw_i2c_set_bus_speed(i2c->regs, speed);
>> +       return __dw_i2c_set_bus_speed(i2c->regs, i2c->scl_sda_cfg, speed);
>>   }
>>
>>   static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>   {
>>          struct dw_i2c *priv = dev_get_priv(bus);
>>
>> +#ifdef CONFIG_X86
>> +       /* Save base address from PCI BAR */
>> +       priv->regs = (struct i2c_regs *)
>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>> +       /* Use BayTrail specific timing values */
>> +       priv->scl_sda_cfg = &byt_config;
>> +#else
> 
> How about:
> 
>      if (device_is_on_pci_bus(dev)) {
>      do the PCI I2C stuff here;
>      }

I've tried this but it generated compilation errors on socfpga, as the
dm_pci_xxx functions are not available there. So it definitely needs
some #ifdef here. I could go with your suggestion and use
#if CONFIG_DM_PCI as well.
 
> See driver/net/designware.c for example.
> 
>>          /* Save base address from device-tree */
>>          priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>> +#endif
>>
>>          __dw_i2c_init(priv->regs, 0, 0);
>>
>>          return 0;
>>   }
>>
>> +static int designware_i2c_bind(struct udevice *dev)
>> +{
>> +       static int num_cards;
>> +       char name[20];
>> +
>> +       /* Create a unique device name for PCI type devices */
>> +       if (device_is_on_pci_bus(dev)) {
>> +               /*
>> +                * ToDo:
>> +                * Setting req_seq in the driver is probably not recommended.
>> +                * But without a DT alias the number is not configured. And
>> +                * using this driver is impossible for PCIe I2C devices.
>> +                * This can be removed, once a better (correct) way for this
>> +                * is found and implemented.
>> +                */
>> +               dev->req_seq = num_cards;
>> +               sprintf(name, "i2c_designware#%u", num_cards++);
>> +               device_set_name(dev, name);
>> +       }
> 
> I believe the above should be wrapped by #ifdef CONFIG_DM_PCI #endif,
> otherwise it won't build when DM_PCI is not enabled.

It does build and work on socfpga without CONFIG_DM_PCI enabled. I've
double-checked this. The only problem is the dm_pci_xxx() in
designware_i2c_probe() as I mentioned above.

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21  9:03     ` Stefan Roese
@ 2016-03-21  9:05       ` Bin Meng
  2016-03-21 12:04       ` Stefan Roese
  1 sibling, 0 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21  9:05 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Mon, Mar 21, 2016 at 5:03 PM, Stefan Roese <sr@denx.de> wrote:
> Hi Bin,
>
> On 21.03.2016 09:54, Bin Meng wrote:
>> On Fri, Mar 18, 2016 at 3:54 PM, Stefan Roese <sr@denx.de> wrote:
>>> This patch adds support for the PCI(e) based I2C cores. Which can be
>>> found for example on the Intel Bay Trail SoC. It has 7 I2C controllers
>>> implemented as PCI devices.
>>>
>>> This patch also adds the fixed values for the timing registers for
>>> BayTrail which are taken from the Linux designware I2C driver.
>>>
>>> Signed-off-by: Stefan Roese <sr@denx.de>
>>> Cc: Simon Glass <sjg@chromium.org>
>>> Cc: Bin Meng <bmeng.cn@gmail.com>
>>> Cc: Marek Vasut <marex@denx.de>
>>> Cc: Heiko Schocher <hs@denx.de>
>>> ---
>>>   drivers/i2c/designware_i2c.c | 111 +++++++++++++++++++++++++++++++++++++++----
>>>   1 file changed, 101 insertions(+), 10 deletions(-)
>>>
>>> diff --git a/drivers/i2c/designware_i2c.c b/drivers/i2c/designware_i2c.c
>>> index 4e5340d..f7f2eba 100644
>>> --- a/drivers/i2c/designware_i2c.c
>>> +++ b/drivers/i2c/designware_i2c.c
>>> @@ -8,11 +8,32 @@
>>>   #include <common.h>
>>>   #include <dm.h>
>>>   #include <i2c.h>
>>> +#include <pci.h>
>>>   #include <asm/io.h>
>>>   #include "designware_i2c.h"
>>>
>>> +struct dw_scl_sda_cfg {
>>> +       u32 ss_hcnt;
>>> +       u32 fs_hcnt;
>>> +       u32 ss_lcnt;
>>> +       u32 fs_lcnt;
>>> +       u32 sda_hold;
>>> +};
>>> +
>>> +#ifdef CONFIG_X86
>>> +/* BayTrail HCNT/LCNT/SDA hold time */
>>> +static struct dw_scl_sda_cfg byt_config = {
>>> +       .ss_hcnt = 0x200,
>>> +       .fs_hcnt = 0x55,
>>> +       .ss_lcnt = 0x200,
>>> +       .fs_lcnt = 0x99,
>>> +       .sda_hold = 0x6,
>>> +};
>>> +#endif
>>> +
>>>   struct dw_i2c {
>>>          struct i2c_regs *regs;
>>> +       struct dw_scl_sda_cfg *scl_sda_cfg;
>>>   };
>>>
>>>   static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>>> @@ -42,6 +63,7 @@ static void dw_i2c_enable(struct i2c_regs *i2c_base, bool enable)
>>>    * Set the i2c speed.
>>>    */
>>>   static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
>>> +                                          struct dw_scl_sda_cfg *scl_sda_cfg,
>>>                                             unsigned int speed)
>>>   {
>>>          unsigned int cntl;
>>> @@ -61,34 +83,55 @@ static unsigned int __dw_i2c_set_bus_speed(struct i2c_regs *i2c_base,
>>>          cntl = (readl(&i2c_base->ic_con) & (~IC_CON_SPD_MSK));
>>>
>>>          switch (i2c_spd) {
>>> +#ifndef CONFIG_X86 /* No High-speed for BayTrail yet */
>>>          case IC_SPEED_MODE_MAX:
>>> -               cntl |= IC_CON_SPD_HS;
>>> -               hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +               cntl |= IC_CON_SPD_SS;
>>> +               if (scl_sda_cfg) {
>>> +                       hcnt = scl_sda_cfg->fs_hcnt;
>>> +                       lcnt = scl_sda_cfg->fs_lcnt;
>>> +               } else {
>>> +                       hcnt = (IC_CLK * MIN_HS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +                       lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
>>> +               }
>>>                  writel(hcnt, &i2c_base->ic_hs_scl_hcnt);
>>> -               lcnt = (IC_CLK * MIN_HS_SCL_LOWTIME) / NANO_TO_MICRO;
>>>                  writel(lcnt, &i2c_base->ic_hs_scl_lcnt);
>>>                  break;
>>> +#endif
>>>
>>>          case IC_SPEED_MODE_STANDARD:
>>>                  cntl |= IC_CON_SPD_SS;
>>> -               hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +               if (scl_sda_cfg) {
>>> +                       hcnt = scl_sda_cfg->ss_hcnt;
>>> +                       lcnt = scl_sda_cfg->ss_lcnt;
>>> +               } else {
>>> +                       hcnt = (IC_CLK * MIN_SS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +                       lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
>>> +               }
>>>                  writel(hcnt, &i2c_base->ic_ss_scl_hcnt);
>>> -               lcnt = (IC_CLK * MIN_SS_SCL_LOWTIME) / NANO_TO_MICRO;
>>>                  writel(lcnt, &i2c_base->ic_ss_scl_lcnt);
>>>                  break;
>>>
>>>          case IC_SPEED_MODE_FAST:
>>>          default:
>>>                  cntl |= IC_CON_SPD_FS;
>>> -               hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +               if (scl_sda_cfg) {
>>> +                       hcnt = scl_sda_cfg->fs_hcnt;
>>> +                       lcnt = scl_sda_cfg->fs_lcnt;
>>> +               } else {
>>> +                       hcnt = (IC_CLK * MIN_FS_SCL_HIGHTIME) / NANO_TO_MICRO;
>>> +                       lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
>>> +               }
>>>                  writel(hcnt, &i2c_base->ic_fs_scl_hcnt);
>>> -               lcnt = (IC_CLK * MIN_FS_SCL_LOWTIME) / NANO_TO_MICRO;
>>>                  writel(lcnt, &i2c_base->ic_fs_scl_lcnt);
>>>                  break;
>>>          }
>>>
>>>          writel(cntl, &i2c_base->ic_con);
>>>
>>> +       /* Configure SDA Hold Time if required */
>>> +       if (scl_sda_cfg)
>>> +               writel(scl_sda_cfg->sda_hold, &i2c_base->ic_sda_hold);
>>> +
>>>          /* Enable back i2c now speed set */
>>>          dw_i2c_enable(i2c_base, 1);
>>>
>>> @@ -315,7 +358,7 @@ static void __dw_i2c_init(struct i2c_regs *i2c_base, int speed, int slaveaddr)
>>>          writel(IC_TX_TL, &i2c_base->ic_tx_tl);
>>>          writel(IC_STOP_DET, &i2c_base->ic_intr_mask);
>>>   #ifndef CONFIG_DM_I2C
>>> -       __dw_i2c_set_bus_speed(i2c_base, speed);
>>> +       __dw_i2c_set_bus_speed(i2c_base, NULL, speed);
>>>          writel(slaveaddr, &i2c_base->ic_sar);
>>>   #endif
>>>
>>> @@ -356,7 +399,7 @@ static unsigned int dw_i2c_set_bus_speed(struct i2c_adapter *adap,
>>>                                           unsigned int speed)
>>>   {
>>>          adap->speed = speed;
>>> -       return __dw_i2c_set_bus_speed(i2c_get_base(adap), speed);
>>> +       return __dw_i2c_set_bus_speed(i2c_get_base(adap), NULL, speed);
>>>   }
>>>
>>>   static void dw_i2c_init(struct i2c_adapter *adap, int speed, int slaveaddr)
>>> @@ -451,7 +494,7 @@ static int designware_i2c_set_bus_speed(struct udevice *bus, unsigned int speed)
>>>   {
>>>          struct dw_i2c *i2c = dev_get_priv(bus);
>>>
>>> -       return __dw_i2c_set_bus_speed(i2c->regs, speed);
>>> +       return __dw_i2c_set_bus_speed(i2c->regs, i2c->scl_sda_cfg, speed);
>>>   }
>>>
>>>   static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>   {
>>>          struct dw_i2c *priv = dev_get_priv(bus);
>>>
>>> +#ifdef CONFIG_X86
>>> +       /* Save base address from PCI BAR */
>>> +       priv->regs = (struct i2c_regs *)
>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>> +       /* Use BayTrail specific timing values */
>>> +       priv->scl_sda_cfg = &byt_config;
>>> +#else
>>
>> How about:
>>
>>      if (device_is_on_pci_bus(dev)) {
>>      do the PCI I2C stuff here;
>>      }
>
> I've tried this but it generated compilation errors on socfpga, as the
> dm_pci_xxx functions are not available there. So it definitely needs
> some #ifdef here. I could go with your suggestion and use
> #if CONFIG_DM_PCI as well.
>
>> See driver/net/designware.c for example.
>>
>>>          /* Save base address from device-tree */
>>>          priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>> +#endif
>>>
>>>          __dw_i2c_init(priv->regs, 0, 0);
>>>
>>>          return 0;
>>>   }
>>>
>>> +static int designware_i2c_bind(struct udevice *dev)
>>> +{
>>> +       static int num_cards;
>>> +       char name[20];
>>> +
>>> +       /* Create a unique device name for PCI type devices */
>>> +       if (device_is_on_pci_bus(dev)) {
>>> +               /*
>>> +                * ToDo:
>>> +                * Setting req_seq in the driver is probably not recommended.
>>> +                * But without a DT alias the number is not configured. And
>>> +                * using this driver is impossible for PCIe I2C devices.
>>> +                * This can be removed, once a better (correct) way for this
>>> +                * is found and implemented.
>>> +                */
>>> +               dev->req_seq = num_cards;
>>> +               sprintf(name, "i2c_designware#%u", num_cards++);
>>> +               device_set_name(dev, name);
>>> +       }
>>
>> I believe the above should be wrapped by #ifdef CONFIG_DM_PCI #endif,
>> otherwise it won't build when DM_PCI is not enabled.
>
> It does build and work on socfpga without CONFIG_DM_PCI enabled. I've
> double-checked this. The only problem is the dm_pci_xxx() in
> designware_i2c_probe() as I mentioned above.
>

Great. So looks we may consolidate one usage for such both PCI and SoC
devices, like the one used in drivers/net/designware.c?

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21  9:03     ` Stefan Roese
  2016-03-21  9:05       ` Bin Meng
@ 2016-03-21 12:04       ` Stefan Roese
  2016-03-21 12:43         ` Bin Meng
  1 sibling, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-03-21 12:04 UTC (permalink / raw)
  To: u-boot

Hi Bin,

On 21.03.2016 10:03, Stefan Roese wrote:

<snip>

>>>    static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>    {
>>>           struct dw_i2c *priv = dev_get_priv(bus);
>>>
>>> +#ifdef CONFIG_X86
>>> +       /* Save base address from PCI BAR */
>>> +       priv->regs = (struct i2c_regs *)
>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>> +       /* Use BayTrail specific timing values */
>>> +       priv->scl_sda_cfg = &byt_config;
>>> +#else
>>
>> How about:
>>
>>       if (device_is_on_pci_bus(dev)) {
>>       do the PCI I2C stuff here;
>>       }
> 
> I've tried this but it generated compilation errors on socfpga, as the
> dm_pci_xxx functions are not available there. So it definitely needs
> some #ifdef here. I could go with your suggestion and use
> #if CONFIG_DM_PCI as well.
>   
>> See driver/net/designware.c for example.
>>
>>>           /* Save base address from device-tree */
>>>           priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>> +#endif

Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
in this ugly compilation warning:

drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
   priv->regs = (struct i2c_regs *)dev_get_addr(bus);
                ^

This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
I'm wondering, how dev_get_addr() should get used on x86. Has it
been used anywhere here at all? Should we perhaps go back to
a 32bit phy_addr representation again? So that dev_get_addr()
matches the (void *) size again?

The other option would to just leave the code as in v1 so that
dev_get_addr() is not referenced on x86 here.

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21 12:04       ` Stefan Roese
@ 2016-03-21 12:43         ` Bin Meng
  2016-03-21 12:52           ` Stefan Roese
  2016-03-21 14:04           ` Stefan Roese
  0 siblings, 2 replies; 31+ messages in thread
From: Bin Meng @ 2016-03-21 12:43 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
> Hi Bin,
>
> On 21.03.2016 10:03, Stefan Roese wrote:
>
> <snip>
>
>>>>    static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>    {
>>>>           struct dw_i2c *priv = dev_get_priv(bus);
>>>>
>>>> +#ifdef CONFIG_X86
>>>> +       /* Save base address from PCI BAR */
>>>> +       priv->regs = (struct i2c_regs *)
>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>> +       /* Use BayTrail specific timing values */
>>>> +       priv->scl_sda_cfg = &byt_config;
>>>> +#else
>>>
>>> How about:
>>>
>>>       if (device_is_on_pci_bus(dev)) {
>>>       do the PCI I2C stuff here;
>>>       }
>>
>> I've tried this but it generated compilation errors on socfpga, as the
>> dm_pci_xxx functions are not available there. So it definitely needs
>> some #ifdef here. I could go with your suggestion and use
>> #if CONFIG_DM_PCI as well.
>>
>>> See driver/net/designware.c for example.
>>>
>>>>           /* Save base address from device-tree */
>>>>           priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>> +#endif
>
> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
> in this ugly compilation warning:
>
> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>    priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>                 ^
>
> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
> I'm wondering, how dev_get_addr() should get used on x86. Has it
> been used anywhere here at all? Should we perhaps go back to
> a 32bit phy_addr representation again? So that dev_get_addr()
> matches the (void *) size again?
>

dev_get_addr() is being used on x86 drivers. See
ns16550_serial_ofdata_to_platdata() for example. There is no build
warning for the ns16550 driver.

> The other option would to just leave the code as in v1 so that
> dev_get_addr() is not referenced on x86 here.
>

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21 12:43         ` Bin Meng
@ 2016-03-21 12:52           ` Stefan Roese
  2016-03-21 14:04           ` Stefan Roese
  1 sibling, 0 replies; 31+ messages in thread
From: Stefan Roese @ 2016-03-21 12:52 UTC (permalink / raw)
  To: u-boot

Hi Bin,

On 21.03.2016 13:43, Bin Meng wrote:
> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>> Hi Bin,
>>
>> On 21.03.2016 10:03, Stefan Roese wrote:
>>
>> <snip>
>>
>>>>>     static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>>     {
>>>>>            struct dw_i2c *priv = dev_get_priv(bus);
>>>>>
>>>>> +#ifdef CONFIG_X86
>>>>> +       /* Save base address from PCI BAR */
>>>>> +       priv->regs = (struct i2c_regs *)
>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>>> +       /* Use BayTrail specific timing values */
>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>> +#else
>>>>
>>>> How about:
>>>>
>>>>        if (device_is_on_pci_bus(dev)) {
>>>>        do the PCI I2C stuff here;
>>>>        }
>>>
>>> I've tried this but it generated compilation errors on socfpga, as the
>>> dm_pci_xxx functions are not available there. So it definitely needs
>>> some #ifdef here. I could go with your suggestion and use
>>> #if CONFIG_DM_PCI as well.
>>>
>>>> See driver/net/designware.c for example.
>>>>
>>>>>            /* Save base address from device-tree */
>>>>>            priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>> +#endif
>>
>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>> in this ugly compilation warning:
>>
>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>>     priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>                  ^
>>
>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>> been used anywhere here at all? Should we perhaps go back to
>> a 32bit phy_addr representation again? So that dev_get_addr()
>> matches the (void *) size again?
>>
>
> dev_get_addr() is being used on x86 drivers. See
> ns16550_serial_ofdata_to_platdata() for example. There is no build
> warning for the ns16550 driver.

I see, thanks. Its used differently there though. Let me see, if I
cook something up...

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21 12:43         ` Bin Meng
  2016-03-21 12:52           ` Stefan Roese
@ 2016-03-21 14:04           ` Stefan Roese
  2016-03-28  6:01             ` Bin Meng
  1 sibling, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-03-21 14:04 UTC (permalink / raw)
  To: u-boot

Hi Bin,

On 21.03.2016 13:43, Bin Meng wrote:
> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>> Hi Bin,
>>
>> On 21.03.2016 10:03, Stefan Roese wrote:
>>
>> <snip>
>>
>>>>>     static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>>     {
>>>>>            struct dw_i2c *priv = dev_get_priv(bus);
>>>>>
>>>>> +#ifdef CONFIG_X86
>>>>> +       /* Save base address from PCI BAR */
>>>>> +       priv->regs = (struct i2c_regs *)
>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>>> +       /* Use BayTrail specific timing values */
>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>> +#else
>>>>
>>>> How about:
>>>>
>>>>        if (device_is_on_pci_bus(dev)) {
>>>>        do the PCI I2C stuff here;
>>>>        }
>>>
>>> I've tried this but it generated compilation errors on socfpga, as the
>>> dm_pci_xxx functions are not available there. So it definitely needs
>>> some #ifdef here. I could go with your suggestion and use
>>> #if CONFIG_DM_PCI as well.
>>>
>>>> See driver/net/designware.c for example.
>>>>
>>>>>            /* Save base address from device-tree */
>>>>>            priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>> +#endif
>>
>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>> in this ugly compilation warning:
>>
>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>>     priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>                  ^
>>
>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>> been used anywhere here at all? Should we perhaps go back to
>> a 32bit phy_addr representation again? So that dev_get_addr()
>> matches the (void *) size again?
>>
> 
> dev_get_addr() is being used on x86 drivers. See
> ns16550_serial_ofdata_to_platdata() for example. There is no build
> warning for the ns16550 driver.

Looking closer, the warning does not occur here, since the registers
are stored in a u32 variable "base". And assigning a 64bit value to a
32bit variable as in "plat->base = addr" in ns16550.c does not cause any
warnings.

Here in the I2C driver though, the base address is stored as a pointer
(pointer size is 32 bit for x86). And this triggers this warning, even
though its effectively the same assignment. I could cast to u32 but this
would cause problems on 64 bit architectures using this driver (in the
future). So I came up with this approach:

/*
 * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning the
 * register base directly in dev_get_addr() results in this compilation warning:
 *     warning: cast to pointer from integer of different size
 *
 * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
 * dev_get_addr() into a 32bit value before casting it to the pointer
 * (struct i2c_regs *).
 */
#ifdef CONFIG_X86
#define POINTER_SIZE_CAST	u32
#endif

...

static int designware_i2c_probe(struct udevice *bus)
{
	struct dw_i2c *priv = dev_get_priv(bus);

	if (device_is_on_pci_bus(bus)) {
#ifdef CONFIG_DM_PCI
		/* Save base address from PCI BAR */
		priv->regs = (struct i2c_regs *)
			dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
#ifdef CONFIG_X86
		/* Use BayTrail specific timing values */
		priv->scl_sda_cfg = &byt_config;
#endif
#endif
	} else {
		/* Save base address from device-tree */
		priv->regs = (struct i2c_regs *)(POINTER_SIZE_CAST)dev_get_addr(bus);
	}

But I'm not 100% happy with this approach.

So what are the alternatives:

a) Don't compile the  dev_get_addr() part for x86 similar to what I've
   done in v1

b) This approach with POINTER_SIZE_CAST

Any preferences of other ideas?

Side note: My general feeling is, that dev_get_addr() should be able to
get cast into a pointer on all platforms. This is how it is used in many
drivers, btw. Since this is not possible on x86, we might have a problem
here. Simon might have some ideas on this as well...

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-21 14:04           ` Stefan Roese
@ 2016-03-28  6:01             ` Bin Meng
  2016-04-04 14:53               ` Stefan Roese
  0 siblings, 1 reply; 31+ messages in thread
From: Bin Meng @ 2016-03-28  6:01 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
> Hi Bin,
>
> On 21.03.2016 13:43, Bin Meng wrote:
>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>> Hi Bin,
>>>
>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>
>>> <snip>
>>>
>>>>>>     static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>>>     {
>>>>>>            struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>
>>>>>> +#ifdef CONFIG_X86
>>>>>> +       /* Save base address from PCI BAR */
>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>>>> +       /* Use BayTrail specific timing values */
>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>> +#else
>>>>>
>>>>> How about:
>>>>>
>>>>>        if (device_is_on_pci_bus(dev)) {
>>>>>        do the PCI I2C stuff here;
>>>>>        }
>>>>
>>>> I've tried this but it generated compilation errors on socfpga, as the
>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>> some #ifdef here. I could go with your suggestion and use
>>>> #if CONFIG_DM_PCI as well.
>>>>
>>>>> See driver/net/designware.c for example.
>>>>>
>>>>>>            /* Save base address from device-tree */
>>>>>>            priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>> +#endif
>>>
>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>> in this ugly compilation warning:
>>>
>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>>>     priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>                  ^
>>>
>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>> been used anywhere here at all? Should we perhaps go back to
>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>> matches the (void *) size again?
>>>
>>
>> dev_get_addr() is being used on x86 drivers. See
>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>> warning for the ns16550 driver.
>
> Looking closer, the warning does not occur here, since the registers
> are stored in a u32 variable "base". And assigning a 64bit value to a
> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
> warnings.
>
> Here in the I2C driver though, the base address is stored as a pointer
> (pointer size is 32 bit for x86). And this triggers this warning, even
> though its effectively the same assignment. I could cast to u32 but this
> would cause problems on 64 bit architectures using this driver (in the
> future). So I came up with this approach:

Thanks for digging out these.

>
> /*
>  * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning the
>  * register base directly in dev_get_addr() results in this compilation warning:
>  *     warning: cast to pointer from integer of different size
>  *
>  * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>  * dev_get_addr() into a 32bit value before casting it to the pointer
>  * (struct i2c_regs *).
>  */
> #ifdef CONFIG_X86
> #define POINTER_SIZE_CAST       u32
> #endif
>
> ...
>
> static int designware_i2c_probe(struct udevice *bus)
> {
>         struct dw_i2c *priv = dev_get_priv(bus);
>
>         if (device_is_on_pci_bus(bus)) {
> #ifdef CONFIG_DM_PCI
>                 /* Save base address from PCI BAR */
>                 priv->regs = (struct i2c_regs *)
>                         dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
> #ifdef CONFIG_X86
>                 /* Use BayTrail specific timing values */
>                 priv->scl_sda_cfg = &byt_config;
> #endif
> #endif
>         } else {
>                 /* Save base address from device-tree */
>                 priv->regs = (struct i2c_regs *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>         }
>
> But I'm not 100% happy with this approach.
>

Yes, it's annoying.

> So what are the alternatives:
>
> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>    done in v1
>
> b) This approach with POINTER_SIZE_CAST
>
> Any preferences of other ideas?
>
> Side note: My general feeling is, that dev_get_addr() should be able to
> get cast into a pointer on all platforms. This is how it is used in many
> drivers, btw. Since this is not possible on x86, we might have a problem
> here. Simon might have some ideas on this as well...
>

I would like to hear Simon's input. Simon?

Regards,
Bin

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-03-28  6:01             ` Bin Meng
@ 2016-04-04 14:53               ` Stefan Roese
  2016-04-11 15:03                 ` Stefan Roese
  0 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-04-04 14:53 UTC (permalink / raw)
  To: u-boot

Hi Simon,

as you seem to be back from vacation (?), we (Bin and myself) would
like to hear your expert comment on a x86 issue I've discovered
while porting the Designware I2C driver to x86. Please see below:

On 28.03.2016 08:01, Bin Meng wrote:
> Hi Stefan,
> 
> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
>> Hi Bin,
>>
>> On 21.03.2016 13:43, Bin Meng wrote:
>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>> Hi Bin,
>>>>
>>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>>
>>>> <snip>
>>>>
>>>>>>>      static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>>>>      {
>>>>>>>             struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>
>>>>>>> +#ifdef CONFIG_X86
>>>>>>> +       /* Save base address from PCI BAR */
>>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>>>>> +       /* Use BayTrail specific timing values */
>>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>>> +#else
>>>>>>
>>>>>> How about:
>>>>>>
>>>>>>         if (device_is_on_pci_bus(dev)) {
>>>>>>         do the PCI I2C stuff here;
>>>>>>         }
>>>>>
>>>>> I've tried this but it generated compilation errors on socfpga, as the
>>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>>> some #ifdef here. I could go with your suggestion and use
>>>>> #if CONFIG_DM_PCI as well.
>>>>>
>>>>>> See driver/net/designware.c for example.
>>>>>>
>>>>>>>             /* Save base address from device-tree */
>>>>>>>             priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>> +#endif
>>>>
>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>>> in this ugly compilation warning:
>>>>
>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>>>>      priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>                   ^
>>>>
>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>>> been used anywhere here at all? Should we perhaps go back to
>>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>>> matches the (void *) size again?
>>>>
>>>
>>> dev_get_addr() is being used on x86 drivers. See
>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>>> warning for the ns16550 driver.
>>
>> Looking closer, the warning does not occur here, since the registers
>> are stored in a u32 variable "base". And assigning a 64bit value to a
>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
>> warnings.
>>
>> Here in the I2C driver though, the base address is stored as a pointer
>> (pointer size is 32 bit for x86). And this triggers this warning, even
>> though its effectively the same assignment. I could cast to u32 but this
>> would cause problems on 64 bit architectures using this driver (in the
>> future). So I came up with this approach:
> 
> Thanks for digging out these.
> 
>>
>> /*
>>   * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning the
>>   * register base directly in dev_get_addr() results in this compilation warning:
>>   *     warning: cast to pointer from integer of different size
>>   *
>>   * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>>   * dev_get_addr() into a 32bit value before casting it to the pointer
>>   * (struct i2c_regs *).
>>   */
>> #ifdef CONFIG_X86
>> #define POINTER_SIZE_CAST       u32
>> #endif
>>
>> ...
>>
>> static int designware_i2c_probe(struct udevice *bus)
>> {
>>          struct dw_i2c *priv = dev_get_priv(bus);
>>
>>          if (device_is_on_pci_bus(bus)) {
>> #ifdef CONFIG_DM_PCI
>>                  /* Save base address from PCI BAR */
>>                  priv->regs = (struct i2c_regs *)
>>                          dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>> #ifdef CONFIG_X86
>>                  /* Use BayTrail specific timing values */
>>                  priv->scl_sda_cfg = &byt_config;
>> #endif
>> #endif
>>          } else {
>>                  /* Save base address from device-tree */
>>                  priv->regs = (struct i2c_regs *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>>          }
>>
>> But I'm not 100% happy with this approach.
>>
> 
> Yes, it's annoying.
> 
>> So what are the alternatives:
>>
>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>>     done in v1
>>
>> b) This approach with POINTER_SIZE_CAST
>>
>> Any preferences of other ideas?
>>
>> Side note: My general feeling is, that dev_get_addr() should be able to
>> get cast into a pointer on all platforms. This is how it is used in many
>> drivers, btw. Since this is not possible on x86, we might have a problem
>> here. Simon might have some ideas on this as well...
>>
> 
> I would like to hear Simon's input. Simon?

Yes, Simon, what do you think?

Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
for the cast:

https://patchwork.ozlabs.org/patch/601113/

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support
  2016-03-18  7:54 ` [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support Stefan Roese
  2016-03-21  8:54   ` Bin Meng
@ 2016-04-09 18:35   ` Simon Glass
  1 sibling, 0 replies; 31+ messages in thread
From: Simon Glass @ 2016-04-09 18:35 UTC (permalink / raw)
  To: u-boot

On 18 March 2016 at 01:54, Stefan Roese <sr@denx.de> wrote:
> This patch adds DM support to the designware I2C driver. It currently
> supports DM and the legacy I2C support. The legacy support should be
> removed, once all platforms using it have DM enabled.
>
> Signed-off-by: Stefan Roese <sr@denx.de>
> Cc: Simon Glass <sjg@chromium.org>
> Cc: Bin Meng <bmeng.cn@gmail.com>
> Cc: Marek Vasut <marex@denx.de>
> Cc: Heiko Schocher <hs@denx.de>
> ---
>  drivers/i2c/designware_i2c.c | 149 ++++++++++++++++++++++++++++++++++++-------
>  1 file changed, 126 insertions(+), 23 deletions(-)

Reviewed-by: Simon Glass <sjg@chromium.org>

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-04-04 14:53               ` Stefan Roese
@ 2016-04-11 15:03                 ` Stefan Roese
  2016-04-20 14:40                   ` Simon Glass
  0 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-04-11 15:03 UTC (permalink / raw)
  To: u-boot

Hi Simon,

On 04.04.2016 16:53, Stefan Roese wrote:
> Hi Simon,
>
> as you seem to be back from vacation (?), we (Bin and myself) would
> like to hear your expert comment on a x86 issue I've discovered
> while porting the Designware I2C driver to x86. Please see below:
>
> On 28.03.2016 08:01, Bin Meng wrote:
>> Hi Stefan,
>>
>> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
>>> Hi Bin,
>>>
>>> On 21.03.2016 13:43, Bin Meng wrote:
>>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>> Hi Bin,
>>>>>
>>>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>>>
>>>>> <snip>
>>>>>
>>>>>>>>       static int designware_i2c_probe_chip(struct udevice *bus, uint chip_addr,
>>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct udevice *bus)
>>>>>>>>       {
>>>>>>>>              struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>>
>>>>>>>> +#ifdef CONFIG_X86
>>>>>>>> +       /* Save base address from PCI BAR */
>>>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>>>>>>> +       /* Use BayTrail specific timing values */
>>>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>>>> +#else
>>>>>>>
>>>>>>> How about:
>>>>>>>
>>>>>>>          if (device_is_on_pci_bus(dev)) {
>>>>>>>          do the PCI I2C stuff here;
>>>>>>>          }
>>>>>>
>>>>>> I've tried this but it generated compilation errors on socfpga, as the
>>>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>>>> some #ifdef here. I could go with your suggestion and use
>>>>>> #if CONFIG_DM_PCI as well.
>>>>>>
>>>>>>> See driver/net/designware.c for example.
>>>>>>>
>>>>>>>>              /* Save base address from device-tree */
>>>>>>>>              priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>> +#endif
>>>>>
>>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>>>> in this ugly compilation warning:
>>>>>
>>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from integer of different size [-Wint-to-pointer-cast]
>>>>>       priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>                    ^
>>>>>
>>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>>>> been used anywhere here at all? Should we perhaps go back to
>>>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>>>> matches the (void *) size again?
>>>>>
>>>>
>>>> dev_get_addr() is being used on x86 drivers. See
>>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>>>> warning for the ns16550 driver.
>>>
>>> Looking closer, the warning does not occur here, since the registers
>>> are stored in a u32 variable "base". And assigning a 64bit value to a
>>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
>>> warnings.
>>>
>>> Here in the I2C driver though, the base address is stored as a pointer
>>> (pointer size is 32 bit for x86). And this triggers this warning, even
>>> though its effectively the same assignment. I could cast to u32 but this
>>> would cause problems on 64 bit architectures using this driver (in the
>>> future). So I came up with this approach:
>>
>> Thanks for digging out these.
>>
>>>
>>> /*
>>>    * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning the
>>>    * register base directly in dev_get_addr() results in this compilation warning:
>>>    *     warning: cast to pointer from integer of different size
>>>    *
>>>    * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>>>    * dev_get_addr() into a 32bit value before casting it to the pointer
>>>    * (struct i2c_regs *).
>>>    */
>>> #ifdef CONFIG_X86
>>> #define POINTER_SIZE_CAST       u32
>>> #endif
>>>
>>> ...
>>>
>>> static int designware_i2c_probe(struct udevice *bus)
>>> {
>>>           struct dw_i2c *priv = dev_get_priv(bus);
>>>
>>>           if (device_is_on_pci_bus(bus)) {
>>> #ifdef CONFIG_DM_PCI
>>>                   /* Save base address from PCI BAR */
>>>                   priv->regs = (struct i2c_regs *)
>>>                           dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0, PCI_REGION_MEM);
>>> #ifdef CONFIG_X86
>>>                   /* Use BayTrail specific timing values */
>>>                   priv->scl_sda_cfg = &byt_config;
>>> #endif
>>> #endif
>>>           } else {
>>>                   /* Save base address from device-tree */
>>>                   priv->regs = (struct i2c_regs *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>>>           }
>>>
>>> But I'm not 100% happy with this approach.
>>>
>>
>> Yes, it's annoying.
>>
>>> So what are the alternatives:
>>>
>>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>>>      done in v1
>>>
>>> b) This approach with POINTER_SIZE_CAST
>>>
>>> Any preferences of other ideas?
>>>
>>> Side note: My general feeling is, that dev_get_addr() should be able to
>>> get cast into a pointer on all platforms. This is how it is used in many
>>> drivers, btw. Since this is not possible on x86, we might have a problem
>>> here. Simon might have some ideas on this as well...
>>>
>>
>> I would like to hear Simon's input. Simon?
>
> Yes, Simon, what do you think?
>
> Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
> for the cast:
>
> https://patchwork.ozlabs.org/patch/601113/

Simon, could you please take a quick look at this patch? With the
general problem of dev_get_addr() on x86 (as described above). Do you
have some other suggestions to solve this? Or is the solution in
v2 which uses (__UINTPTR_TYPE__) acceptable?

https://patchwork.ozlabs.org/patch/601113/

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-04-11 15:03                 ` Stefan Roese
@ 2016-04-20 14:40                   ` Simon Glass
  2016-04-20 14:58                     ` Stefan Roese
  0 siblings, 1 reply; 31+ messages in thread
From: Simon Glass @ 2016-04-20 14:40 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On 11 April 2016 at 09:03, Stefan Roese <sr@denx.de> wrote:
> Hi Simon,
>
>
> On 04.04.2016 16:53, Stefan Roese wrote:
>>
>> Hi Simon,
>>
>> as you seem to be back from vacation (?), we (Bin and myself) would
>> like to hear your expert comment on a x86 issue I've discovered
>> while porting the Designware I2C driver to x86. Please see below:
>>
>> On 28.03.2016 08:01, Bin Meng wrote:
>>>
>>> Hi Stefan,
>>>
>>> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>
>>>> Hi Bin,
>>>>
>>>> On 21.03.2016 13:43, Bin Meng wrote:
>>>>>
>>>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>>>
>>>>>> Hi Bin,
>>>>>>
>>>>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>>>>
>>>>>> <snip>
>>>>>>
>>>>>>>>>       static int designware_i2c_probe_chip(struct udevice *bus,
>>>>>>>>> uint chip_addr,
>>>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct
>>>>>>>>> udevice *bus)
>>>>>>>>>       {
>>>>>>>>>              struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>>>
>>>>>>>>> +#ifdef CONFIG_X86
>>>>>>>>> +       /* Save base address from PCI BAR */
>>>>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>>>>>>> PCI_REGION_MEM);
>>>>>>>>> +       /* Use BayTrail specific timing values */
>>>>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>>>>> +#else
>>>>>>>>
>>>>>>>>
>>>>>>>> How about:
>>>>>>>>
>>>>>>>>          if (device_is_on_pci_bus(dev)) {
>>>>>>>>          do the PCI I2C stuff here;
>>>>>>>>          }
>>>>>>>
>>>>>>>
>>>>>>> I've tried this but it generated compilation errors on socfpga, as
>>>>>>> the
>>>>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>>>>> some #ifdef here. I could go with your suggestion and use
>>>>>>> #if CONFIG_DM_PCI as well.
>>>>>>>
>>>>>>>> See driver/net/designware.c for example.
>>>>>>>>
>>>>>>>>>              /* Save base address from device-tree */
>>>>>>>>>              priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>>> +#endif
>>>>>>
>>>>>>
>>>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>>>>> in this ugly compilation warning:
>>>>>>
>>>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from
>>>>>> integer of different size [-Wint-to-pointer-cast]
>>>>>>       priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>                    ^
>>>>>>
>>>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>>>>> been used anywhere here at all? Should we perhaps go back to
>>>>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>>>>> matches the (void *) size again?
>>>>>>
>>>>>
>>>>> dev_get_addr() is being used on x86 drivers. See
>>>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>>>>> warning for the ns16550 driver.
>>>>
>>>>
>>>> Looking closer, the warning does not occur here, since the registers
>>>> are stored in a u32 variable "base". And assigning a 64bit value to a
>>>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
>>>> warnings.
>>>>
>>>> Here in the I2C driver though, the base address is stored as a pointer
>>>> (pointer size is 32 bit for x86). And this triggers this warning, even
>>>> though its effectively the same assignment. I could cast to u32 but this
>>>> would cause problems on 64 bit architectures using this driver (in the
>>>> future). So I came up with this approach:
>>>
>>>
>>> Thanks for digging out these.
>>>
>>>>
>>>> /*
>>>>    * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning
>>>> the
>>>>    * register base directly in dev_get_addr() results in this
>>>> compilation warning:
>>>>    *     warning: cast to pointer from integer of different size
>>>>    *
>>>>    * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>>>>    * dev_get_addr() into a 32bit value before casting it to the pointer
>>>>    * (struct i2c_regs *).
>>>>    */
>>>> #ifdef CONFIG_X86
>>>> #define POINTER_SIZE_CAST       u32
>>>> #endif
>>>>
>>>> ...
>>>>
>>>> static int designware_i2c_probe(struct udevice *bus)
>>>> {
>>>>           struct dw_i2c *priv = dev_get_priv(bus);
>>>>
>>>>           if (device_is_on_pci_bus(bus)) {
>>>> #ifdef CONFIG_DM_PCI
>>>>                   /* Save base address from PCI BAR */
>>>>                   priv->regs = (struct i2c_regs *)
>>>>                           dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>> PCI_REGION_MEM);
>>>> #ifdef CONFIG_X86
>>>>                   /* Use BayTrail specific timing values */
>>>>                   priv->scl_sda_cfg = &byt_config;
>>>> #endif
>>>> #endif
>>>>           } else {
>>>>                   /* Save base address from device-tree */
>>>>                   priv->regs = (struct i2c_regs
>>>> *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>>>>           }
>>>>
>>>> But I'm not 100% happy with this approach.
>>>>
>>>
>>> Yes, it's annoying.
>>>
>>>> So what are the alternatives:
>>>>
>>>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>>>>      done in v1
>>>>
>>>> b) This approach with POINTER_SIZE_CAST
>>>>
>>>> Any preferences of other ideas?
>>>>
>>>> Side note: My general feeling is, that dev_get_addr() should be able to
>>>> get cast into a pointer on all platforms. This is how it is used in many
>>>> drivers, btw. Since this is not possible on x86, we might have a problem
>>>> here. Simon might have some ideas on this as well...
>>>>
>>>
>>> I would like to hear Simon's input. Simon?
>>
>>
>> Yes, Simon, what do you think?
>>
>> Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
>> for the cast:
>>
>> https://patchwork.ozlabs.org/patch/601113/
>
>
> Simon, could you please take a quick look at this patch? With the
> general problem of dev_get_addr() on x86 (as described above). Do you
> have some other suggestions to solve this? Or is the solution in
> v2 which uses (__UINTPTR_TYPE__) acceptable?
>
> https://patchwork.ozlabs.org/patch/601113/

I feel that you should store the return value from dev_get_addr() in
an fdt_addr_t or a ulong. Then it can be cast to a pointer as you
wish. Platform data should hold the ulong, and private data
(dev_get_priv()) should hold the pointer.

I'm not keen on the POINTER_SIZE_CAST idea.

Does that fix the problem?

Regards,
Simon

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-04-20 14:40                   ` Simon Glass
@ 2016-04-20 14:58                     ` Stefan Roese
  2016-04-20 15:09                       ` Simon Glass
  0 siblings, 1 reply; 31+ messages in thread
From: Stefan Roese @ 2016-04-20 14:58 UTC (permalink / raw)
  To: u-boot

Hi Simon.

On 20.04.2016 16:40, Simon Glass wrote:

> On 11 April 2016 at 09:03, Stefan Roese <sr@denx.de> wrote:
>> Hi Simon,
>>
>>
>> On 04.04.2016 16:53, Stefan Roese wrote:
>>>
>>> Hi Simon,
>>>
>>> as you seem to be back from vacation (?), we (Bin and myself) would
>>> like to hear your expert comment on a x86 issue I've discovered
>>> while porting the Designware I2C driver to x86. Please see below:
>>>
>>> On 28.03.2016 08:01, Bin Meng wrote:
>>>>
>>>> Hi Stefan,
>>>>
>>>> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>>
>>>>> Hi Bin,
>>>>>
>>>>> On 21.03.2016 13:43, Bin Meng wrote:
>>>>>>
>>>>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>>>>
>>>>>>> Hi Bin,
>>>>>>>
>>>>>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>>>>>
>>>>>>> <snip>
>>>>>>>
>>>>>>>>>>        static int designware_i2c_probe_chip(struct udevice *bus,
>>>>>>>>>> uint chip_addr,
>>>>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct
>>>>>>>>>> udevice *bus)
>>>>>>>>>>        {
>>>>>>>>>>               struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>>>>
>>>>>>>>>> +#ifdef CONFIG_X86
>>>>>>>>>> +       /* Save base address from PCI BAR */
>>>>>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>>>>>>>> PCI_REGION_MEM);
>>>>>>>>>> +       /* Use BayTrail specific timing values */
>>>>>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>>>>>> +#else
>>>>>>>>>
>>>>>>>>>
>>>>>>>>> How about:
>>>>>>>>>
>>>>>>>>>           if (device_is_on_pci_bus(dev)) {
>>>>>>>>>           do the PCI I2C stuff here;
>>>>>>>>>           }
>>>>>>>>
>>>>>>>>
>>>>>>>> I've tried this but it generated compilation errors on socfpga, as
>>>>>>>> the
>>>>>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>>>>>> some #ifdef here. I could go with your suggestion and use
>>>>>>>> #if CONFIG_DM_PCI as well.
>>>>>>>>
>>>>>>>>> See driver/net/designware.c for example.
>>>>>>>>>
>>>>>>>>>>               /* Save base address from device-tree */
>>>>>>>>>>               priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>>>> +#endif
>>>>>>>
>>>>>>>
>>>>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>>>>>> in this ugly compilation warning:
>>>>>>>
>>>>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>>>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from
>>>>>>> integer of different size [-Wint-to-pointer-cast]
>>>>>>>        priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>                     ^
>>>>>>>
>>>>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>>>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>>>>>> been used anywhere here at all? Should we perhaps go back to
>>>>>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>>>>>> matches the (void *) size again?
>>>>>>>
>>>>>>
>>>>>> dev_get_addr() is being used on x86 drivers. See
>>>>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>>>>>> warning for the ns16550 driver.
>>>>>
>>>>>
>>>>> Looking closer, the warning does not occur here, since the registers
>>>>> are stored in a u32 variable "base". And assigning a 64bit value to a
>>>>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
>>>>> warnings.
>>>>>
>>>>> Here in the I2C driver though, the base address is stored as a pointer
>>>>> (pointer size is 32 bit for x86). And this triggers this warning, even
>>>>> though its effectively the same assignment. I could cast to u32 but this
>>>>> would cause problems on 64 bit architectures using this driver (in the
>>>>> future). So I came up with this approach:
>>>>
>>>>
>>>> Thanks for digging out these.
>>>>
>>>>>
>>>>> /*
>>>>>     * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning
>>>>> the
>>>>>     * register base directly in dev_get_addr() results in this
>>>>> compilation warning:
>>>>>     *     warning: cast to pointer from integer of different size
>>>>>     *
>>>>>     * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>>>>>     * dev_get_addr() into a 32bit value before casting it to the pointer
>>>>>     * (struct i2c_regs *).
>>>>>     */
>>>>> #ifdef CONFIG_X86
>>>>> #define POINTER_SIZE_CAST       u32
>>>>> #endif
>>>>>
>>>>> ...
>>>>>
>>>>> static int designware_i2c_probe(struct udevice *bus)
>>>>> {
>>>>>            struct dw_i2c *priv = dev_get_priv(bus);
>>>>>
>>>>>            if (device_is_on_pci_bus(bus)) {
>>>>> #ifdef CONFIG_DM_PCI
>>>>>                    /* Save base address from PCI BAR */
>>>>>                    priv->regs = (struct i2c_regs *)
>>>>>                            dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>>> PCI_REGION_MEM);
>>>>> #ifdef CONFIG_X86
>>>>>                    /* Use BayTrail specific timing values */
>>>>>                    priv->scl_sda_cfg = &byt_config;
>>>>> #endif
>>>>> #endif
>>>>>            } else {
>>>>>                    /* Save base address from device-tree */
>>>>>                    priv->regs = (struct i2c_regs
>>>>> *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>>>>>            }
>>>>>
>>>>> But I'm not 100% happy with this approach.
>>>>>
>>>>
>>>> Yes, it's annoying.
>>>>
>>>>> So what are the alternatives:
>>>>>
>>>>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>>>>>       done in v1
>>>>>
>>>>> b) This approach with POINTER_SIZE_CAST
>>>>>
>>>>> Any preferences of other ideas?
>>>>>
>>>>> Side note: My general feeling is, that dev_get_addr() should be able to
>>>>> get cast into a pointer on all platforms. This is how it is used in many
>>>>> drivers, btw. Since this is not possible on x86, we might have a problem
>>>>> here. Simon might have some ideas on this as well...
>>>>>
>>>>
>>>> I would like to hear Simon's input. Simon?
>>>
>>>
>>> Yes, Simon, what do you think?
>>>
>>> Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
>>> for the cast:
>>>
>>> https://patchwork.ozlabs.org/patch/601113/
>>
>>
>> Simon, could you please take a quick look at this patch? With the
>> general problem of dev_get_addr() on x86 (as described above). Do you
>> have some other suggestions to solve this? Or is the solution in
>> v2 which uses (__UINTPTR_TYPE__) acceptable?
>>
>> https://patchwork.ozlabs.org/patch/601113/
> 
> I feel that you should store the return value from dev_get_addr() in
> an fdt_addr_t or a ulong. Then it can be cast to a pointer as you
> wish. Platform data should hold the ulong, and private data
> (dev_get_priv()) should hold the pointer.
> 
> I'm not keen on the POINTER_SIZE_CAST idea.
> 
> Does that fix the problem?

Yes, it does. In a somewhat less ugly way. This is my current result:

	} else {
		ulong base;

		/* Save base address from device-tree */

		/*
		 * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit.
		 * So assigning the register base directly in dev_get_addr()
		 * results in this compilation warning:
		 *   warning: cast to pointer from integer of different size
		 *
		 * Using an intermediate "ulong" variable before assigning
		 * this pointer to the "regs" variable solves this issue.
		 */
		base = dev_get_addr(bus);
		priv->regs = (struct i2c_regs *)base;
	}

If you think this is acceptable, I'll send a new patch version to
the list.

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-04-20 14:58                     ` Stefan Roese
@ 2016-04-20 15:09                       ` Simon Glass
  2016-04-20 15:17                         ` Stefan Roese
  0 siblings, 1 reply; 31+ messages in thread
From: Simon Glass @ 2016-04-20 15:09 UTC (permalink / raw)
  To: u-boot

Hi Stefan,

On 20 April 2016 at 08:58, Stefan Roese <sr@denx.de> wrote:
>
> Hi Simon.
>
> On 20.04.2016 16:40, Simon Glass wrote:
>
> > On 11 April 2016 at 09:03, Stefan Roese <sr@denx.de> wrote:
> >> Hi Simon,
> >>
> >>
> >> On 04.04.2016 16:53, Stefan Roese wrote:
> >>>
> >>> Hi Simon,
> >>>
> >>> as you seem to be back from vacation (?), we (Bin and myself) would
> >>> like to hear your expert comment on a x86 issue I've discovered
> >>> while porting the Designware I2C driver to x86. Please see below:
> >>>
> >>> On 28.03.2016 08:01, Bin Meng wrote:
> >>>>
> >>>> Hi Stefan,
> >>>>
> >>>> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
> >>>>>
> >>>>> Hi Bin,
> >>>>>
> >>>>> On 21.03.2016 13:43, Bin Meng wrote:
> >>>>>>
> >>>>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
> >>>>>>>
> >>>>>>> Hi Bin,
> >>>>>>>
> >>>>>>> On 21.03.2016 10:03, Stefan Roese wrote:
> >>>>>>>
> >>>>>>> <snip>
> >>>>>>>
> >>>>>>>>>>        static int designware_i2c_probe_chip(struct udevice *bus,
> >>>>>>>>>> uint chip_addr,
> >>>>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct
> >>>>>>>>>> udevice *bus)
> >>>>>>>>>>        {
> >>>>>>>>>>               struct dw_i2c *priv = dev_get_priv(bus);
> >>>>>>>>>>
> >>>>>>>>>> +#ifdef CONFIG_X86
> >>>>>>>>>> +       /* Save base address from PCI BAR */
> >>>>>>>>>> +       priv->regs = (struct i2c_regs *)
> >>>>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
> >>>>>>>>>> PCI_REGION_MEM);
> >>>>>>>>>> +       /* Use BayTrail specific timing values */
> >>>>>>>>>> +       priv->scl_sda_cfg = &byt_config;
> >>>>>>>>>> +#else
> >>>>>>>>>
> >>>>>>>>>
> >>>>>>>>> How about:
> >>>>>>>>>
> >>>>>>>>>           if (device_is_on_pci_bus(dev)) {
> >>>>>>>>>           do the PCI I2C stuff here;
> >>>>>>>>>           }
> >>>>>>>>
> >>>>>>>>
> >>>>>>>> I've tried this but it generated compilation errors on socfpga, as
> >>>>>>>> the
> >>>>>>>> dm_pci_xxx functions are not available there. So it definitely needs
> >>>>>>>> some #ifdef here. I could go with your suggestion and use
> >>>>>>>> #if CONFIG_DM_PCI as well.
> >>>>>>>>
> >>>>>>>>> See driver/net/designware.c for example.
> >>>>>>>>>
> >>>>>>>>>>               /* Save base address from device-tree */
> >>>>>>>>>>               priv->regs = (struct i2c_regs *)dev_get_addr(bus);
> >>>>>>>>>> +#endif
> >>>>>>>
> >>>>>>>
> >>>>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
> >>>>>>> in this ugly compilation warning:
> >>>>>>>
> >>>>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
> >>>>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from
> >>>>>>> integer of different size [-Wint-to-pointer-cast]
> >>>>>>>        priv->regs = (struct i2c_regs *)dev_get_addr(bus);
> >>>>>>>                     ^
> >>>>>>>
> >>>>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
> >>>>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
> >>>>>>> been used anywhere here at all? Should we perhaps go back to
> >>>>>>> a 32bit phy_addr representation again? So that dev_get_addr()
> >>>>>>> matches the (void *) size again?
> >>>>>>>
> >>>>>>
> >>>>>> dev_get_addr() is being used on x86 drivers. See
> >>>>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
> >>>>>> warning for the ns16550 driver.
> >>>>>
> >>>>>
> >>>>> Looking closer, the warning does not occur here, since the registers
> >>>>> are stored in a u32 variable "base". And assigning a 64bit value to a
> >>>>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
> >>>>> warnings.
> >>>>>
> >>>>> Here in the I2C driver though, the base address is stored as a pointer
> >>>>> (pointer size is 32 bit for x86). And this triggers this warning, even
> >>>>> though its effectively the same assignment. I could cast to u32 but this
> >>>>> would cause problems on 64 bit architectures using this driver (in the
> >>>>> future). So I came up with this approach:
> >>>>
> >>>>
> >>>> Thanks for digging out these.
> >>>>
> >>>>>
> >>>>> /*
> >>>>>     * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning
> >>>>> the
> >>>>>     * register base directly in dev_get_addr() results in this
> >>>>> compilation warning:
> >>>>>     *     warning: cast to pointer from integer of different size
> >>>>>     *
> >>>>>     * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
> >>>>>     * dev_get_addr() into a 32bit value before casting it to the pointer
> >>>>>     * (struct i2c_regs *).
> >>>>>     */
> >>>>> #ifdef CONFIG_X86
> >>>>> #define POINTER_SIZE_CAST       u32
> >>>>> #endif
> >>>>>
> >>>>> ...
> >>>>>
> >>>>> static int designware_i2c_probe(struct udevice *bus)
> >>>>> {
> >>>>>            struct dw_i2c *priv = dev_get_priv(bus);
> >>>>>
> >>>>>            if (device_is_on_pci_bus(bus)) {
> >>>>> #ifdef CONFIG_DM_PCI
> >>>>>                    /* Save base address from PCI BAR */
> >>>>>                    priv->regs = (struct i2c_regs *)
> >>>>>                            dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
> >>>>> PCI_REGION_MEM);
> >>>>> #ifdef CONFIG_X86
> >>>>>                    /* Use BayTrail specific timing values */
> >>>>>                    priv->scl_sda_cfg = &byt_config;
> >>>>> #endif
> >>>>> #endif
> >>>>>            } else {
> >>>>>                    /* Save base address from device-tree */
> >>>>>                    priv->regs = (struct i2c_regs
> >>>>> *)(POINTER_SIZE_CAST)dev_get_addr(bus);
> >>>>>            }
> >>>>>
> >>>>> But I'm not 100% happy with this approach.
> >>>>>
> >>>>
> >>>> Yes, it's annoying.
> >>>>
> >>>>> So what are the alternatives:
> >>>>>
> >>>>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
> >>>>>       done in v1
> >>>>>
> >>>>> b) This approach with POINTER_SIZE_CAST
> >>>>>
> >>>>> Any preferences of other ideas?
> >>>>>
> >>>>> Side note: My general feeling is, that dev_get_addr() should be able to
> >>>>> get cast into a pointer on all platforms. This is how it is used in many
> >>>>> drivers, btw. Since this is not possible on x86, we might have a problem
> >>>>> here. Simon might have some ideas on this as well...
> >>>>>
> >>>>
> >>>> I would like to hear Simon's input. Simon?
> >>>
> >>>
> >>> Yes, Simon, what do you think?
> >>>
> >>> Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
> >>> for the cast:
> >>>
> >>> https://patchwork.ozlabs.org/patch/601113/
> >>
> >>
> >> Simon, could you please take a quick look at this patch? With the
> >> general problem of dev_get_addr() on x86 (as described above). Do you
> >> have some other suggestions to solve this? Or is the solution in
> >> v2 which uses (__UINTPTR_TYPE__) acceptable?
> >>
> >> https://patchwork.ozlabs.org/patch/601113/
> >
> > I feel that you should store the return value from dev_get_addr() in
> > an fdt_addr_t or a ulong. Then it can be cast to a pointer as you
> > wish. Platform data should hold the ulong, and private data
> > (dev_get_priv()) should hold the pointer.
> >
> > I'm not keen on the POINTER_SIZE_CAST idea.
> >
> > Does that fix the problem?
>
> Yes, it does. In a somewhat less ugly way. This is my current result:
>
>         } else {
>                 ulong base;
>
>                 /* Save base address from device-tree */
>
>                 /*
>                  * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit.
>                  * So assigning the register base directly in dev_get_addr()
>                  * results in this compilation warning:
>                  *   warning: cast to pointer from integer of different size
>                  *
>                  * Using an intermediate "ulong" variable before assigning
>                  * this pointer to the "regs" variable solves this issue.
>                  */
>                 base = dev_get_addr(bus);
>                 priv->regs = (struct i2c_regs *)base;
>         }
>
> If you think this is acceptable, I'll send a new patch version to
> the list.

Seems fine to me. Perhaps we should have dev_get_addr_ptr() to do this for us?

Regards,
Simon

^ permalink raw reply	[flat|nested] 31+ messages in thread

* [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86)
  2016-04-20 15:09                       ` Simon Glass
@ 2016-04-20 15:17                         ` Stefan Roese
  0 siblings, 0 replies; 31+ messages in thread
From: Stefan Roese @ 2016-04-20 15:17 UTC (permalink / raw)
  To: u-boot

Hi Simon,

On 20.04.2016 17:09, Simon Glass wrote:
> Hi Stefan,
>
> On 20 April 2016 at 08:58, Stefan Roese <sr@denx.de> wrote:
>>
>> Hi Simon.
>>
>> On 20.04.2016 16:40, Simon Glass wrote:
>>
>>> On 11 April 2016 at 09:03, Stefan Roese <sr@denx.de> wrote:
>>>> Hi Simon,
>>>>
>>>>
>>>> On 04.04.2016 16:53, Stefan Roese wrote:
>>>>>
>>>>> Hi Simon,
>>>>>
>>>>> as you seem to be back from vacation (?), we (Bin and myself) would
>>>>> like to hear your expert comment on a x86 issue I've discovered
>>>>> while porting the Designware I2C driver to x86. Please see below:
>>>>>
>>>>> On 28.03.2016 08:01, Bin Meng wrote:
>>>>>>
>>>>>> Hi Stefan,
>>>>>>
>>>>>> On Mon, Mar 21, 2016 at 10:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>>>>
>>>>>>> Hi Bin,
>>>>>>>
>>>>>>> On 21.03.2016 13:43, Bin Meng wrote:
>>>>>>>>
>>>>>>>> On Mon, Mar 21, 2016 at 8:04 PM, Stefan Roese <sr@denx.de> wrote:
>>>>>>>>>
>>>>>>>>> Hi Bin,
>>>>>>>>>
>>>>>>>>> On 21.03.2016 10:03, Stefan Roese wrote:
>>>>>>>>>
>>>>>>>>> <snip>
>>>>>>>>>
>>>>>>>>>>>>         static int designware_i2c_probe_chip(struct udevice *bus,
>>>>>>>>>>>> uint chip_addr,
>>>>>>>>>>>> @@ -476,14 +519,45 @@ static int designware_i2c_probe(struct
>>>>>>>>>>>> udevice *bus)
>>>>>>>>>>>>         {
>>>>>>>>>>>>                struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>>>>>>
>>>>>>>>>>>> +#ifdef CONFIG_X86
>>>>>>>>>>>> +       /* Save base address from PCI BAR */
>>>>>>>>>>>> +       priv->regs = (struct i2c_regs *)
>>>>>>>>>>>> +               dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>>>>>>>>>> PCI_REGION_MEM);
>>>>>>>>>>>> +       /* Use BayTrail specific timing values */
>>>>>>>>>>>> +       priv->scl_sda_cfg = &byt_config;
>>>>>>>>>>>> +#else
>>>>>>>>>>>
>>>>>>>>>>>
>>>>>>>>>>> How about:
>>>>>>>>>>>
>>>>>>>>>>>            if (device_is_on_pci_bus(dev)) {
>>>>>>>>>>>            do the PCI I2C stuff here;
>>>>>>>>>>>            }
>>>>>>>>>>
>>>>>>>>>>
>>>>>>>>>> I've tried this but it generated compilation errors on socfpga, as
>>>>>>>>>> the
>>>>>>>>>> dm_pci_xxx functions are not available there. So it definitely needs
>>>>>>>>>> some #ifdef here. I could go with your suggestion and use
>>>>>>>>>> #if CONFIG_DM_PCI as well.
>>>>>>>>>>
>>>>>>>>>>> See driver/net/designware.c for example.
>>>>>>>>>>>
>>>>>>>>>>>>                /* Save base address from device-tree */
>>>>>>>>>>>>                priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>>>>>> +#endif
>>>>>>>>>
>>>>>>>>>
>>>>>>>>> Enabling this code for x86 via if (device_is_on_pci_bus(dev)) results
>>>>>>>>> in this ugly compilation warning:
>>>>>>>>>
>>>>>>>>> drivers/i2c/designware_i2c.c: In function ?designware_i2c_probe?:
>>>>>>>>> drivers/i2c/designware_i2c.c:530:16: warning: cast to pointer from
>>>>>>>>> integer of different size [-Wint-to-pointer-cast]
>>>>>>>>>         priv->regs = (struct i2c_regs *)dev_get_addr(bus);
>>>>>>>>>                      ^
>>>>>>>>>
>>>>>>>>> This is because x86 defines fdt_addr_t / phys_addr_t as 64bit. So
>>>>>>>>> I'm wondering, how dev_get_addr() should get used on x86. Has it
>>>>>>>>> been used anywhere here at all? Should we perhaps go back to
>>>>>>>>> a 32bit phy_addr representation again? So that dev_get_addr()
>>>>>>>>> matches the (void *) size again?
>>>>>>>>>
>>>>>>>>
>>>>>>>> dev_get_addr() is being used on x86 drivers. See
>>>>>>>> ns16550_serial_ofdata_to_platdata() for example. There is no build
>>>>>>>> warning for the ns16550 driver.
>>>>>>>
>>>>>>>
>>>>>>> Looking closer, the warning does not occur here, since the registers
>>>>>>> are stored in a u32 variable "base". And assigning a 64bit value to a
>>>>>>> 32bit variable as in "plat->base = addr" in ns16550.c does not cause any
>>>>>>> warnings.
>>>>>>>
>>>>>>> Here in the I2C driver though, the base address is stored as a pointer
>>>>>>> (pointer size is 32 bit for x86). And this triggers this warning, even
>>>>>>> though its effectively the same assignment. I could cast to u32 but this
>>>>>>> would cause problems on 64 bit architectures using this driver (in the
>>>>>>> future). So I came up with this approach:
>>>>>>
>>>>>>
>>>>>> Thanks for digging out these.
>>>>>>
>>>>>>>
>>>>>>> /*
>>>>>>>      * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit. So assigning
>>>>>>> the
>>>>>>>      * register base directly in dev_get_addr() results in this
>>>>>>> compilation warning:
>>>>>>>      *     warning: cast to pointer from integer of different size
>>>>>>>      *
>>>>>>>      * Using this macro POINTER_SIZE_CAST, allows us to cast the result of
>>>>>>>      * dev_get_addr() into a 32bit value before casting it to the pointer
>>>>>>>      * (struct i2c_regs *).
>>>>>>>      */
>>>>>>> #ifdef CONFIG_X86
>>>>>>> #define POINTER_SIZE_CAST       u32
>>>>>>> #endif
>>>>>>>
>>>>>>> ...
>>>>>>>
>>>>>>> static int designware_i2c_probe(struct udevice *bus)
>>>>>>> {
>>>>>>>             struct dw_i2c *priv = dev_get_priv(bus);
>>>>>>>
>>>>>>>             if (device_is_on_pci_bus(bus)) {
>>>>>>> #ifdef CONFIG_DM_PCI
>>>>>>>                     /* Save base address from PCI BAR */
>>>>>>>                     priv->regs = (struct i2c_regs *)
>>>>>>>                             dm_pci_map_bar(bus, PCI_BASE_ADDRESS_0,
>>>>>>> PCI_REGION_MEM);
>>>>>>> #ifdef CONFIG_X86
>>>>>>>                     /* Use BayTrail specific timing values */
>>>>>>>                     priv->scl_sda_cfg = &byt_config;
>>>>>>> #endif
>>>>>>> #endif
>>>>>>>             } else {
>>>>>>>                     /* Save base address from device-tree */
>>>>>>>                     priv->regs = (struct i2c_regs
>>>>>>> *)(POINTER_SIZE_CAST)dev_get_addr(bus);
>>>>>>>             }
>>>>>>>
>>>>>>> But I'm not 100% happy with this approach.
>>>>>>>
>>>>>>
>>>>>> Yes, it's annoying.
>>>>>>
>>>>>>> So what are the alternatives:
>>>>>>>
>>>>>>> a) Don't compile the  dev_get_addr() part for x86 similar to what I've
>>>>>>>        done in v1
>>>>>>>
>>>>>>> b) This approach with POINTER_SIZE_CAST
>>>>>>>
>>>>>>> Any preferences of other ideas?
>>>>>>>
>>>>>>> Side note: My general feeling is, that dev_get_addr() should be able to
>>>>>>> get cast into a pointer on all platforms. This is how it is used in many
>>>>>>> drivers, btw. Since this is not possible on x86, we might have a problem
>>>>>>> here. Simon might have some ideas on this as well...
>>>>>>>
>>>>>>
>>>>>> I would like to hear Simon's input. Simon?
>>>>>
>>>>>
>>>>> Yes, Simon, what do you think?
>>>>>
>>>>> Please also see my v2 of this patch which uses (__UINTPTR_TYPE__)
>>>>> for the cast:
>>>>>
>>>>> https://patchwork.ozlabs.org/patch/601113/
>>>>
>>>>
>>>> Simon, could you please take a quick look at this patch? With the
>>>> general problem of dev_get_addr() on x86 (as described above). Do you
>>>> have some other suggestions to solve this? Or is the solution in
>>>> v2 which uses (__UINTPTR_TYPE__) acceptable?
>>>>
>>>> https://patchwork.ozlabs.org/patch/601113/
>>>
>>> I feel that you should store the return value from dev_get_addr() in
>>> an fdt_addr_t or a ulong. Then it can be cast to a pointer as you
>>> wish. Platform data should hold the ulong, and private data
>>> (dev_get_priv()) should hold the pointer.
>>>
>>> I'm not keen on the POINTER_SIZE_CAST idea.
>>>
>>> Does that fix the problem?
>>
>> Yes, it does. In a somewhat less ugly way. This is my current result:
>>
>>          } else {
>>                  ulong base;
>>
>>                  /* Save base address from device-tree */
>>
>>                  /*
>>                   * On x86, "fdt_addr_t" is 64bit but "void *" only 32bit.
>>                   * So assigning the register base directly in dev_get_addr()
>>                   * results in this compilation warning:
>>                   *   warning: cast to pointer from integer of different size
>>                   *
>>                   * Using an intermediate "ulong" variable before assigning
>>                   * this pointer to the "regs" variable solves this issue.
>>                   */
>>                  base = dev_get_addr(bus);
>>                  priv->regs = (struct i2c_regs *)base;
>>          }
>>
>> If you think this is acceptable, I'll send a new patch version to
>> the list.
>
> Seems fine to me. Perhaps we should have dev_get_addr_ptr() to do
> this for us?

Might make sense. I can generate a small patch for this.

Perhaps we should better use "uintptr_t" as type for the intermediate
variable instead. But then we can effectively drop the intermediate
variable and do the casting directly.

What do you think?

Thanks,
Stefan

^ permalink raw reply	[flat|nested] 31+ messages in thread

end of thread, other threads:[~2016-04-20 15:17 UTC | newest]

Thread overview: 31+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2016-03-18  7:54 [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Stefan Roese
2016-03-18  7:54 ` [U-Boot] [PATCH 2/6] i2c: designware_i2c: Add dw_i2c_enable() helper function Stefan Roese
2016-03-18 11:12   ` Marek Vasut
2016-03-18 12:04     ` Stefan Roese
2016-03-18 12:14       ` Marek Vasut
2016-03-21  8:54   ` Bin Meng
2016-03-18  7:54 ` [U-Boot] [PATCH 3/6] i2c: designware_i2c: Integrate set_speed() into dw_i2c_set_bus_speed() Stefan Roese
2016-03-18 11:13   ` Marek Vasut
2016-03-21  8:54   ` Bin Meng
2016-03-18  7:54 ` [U-Boot] [PATCH 4/6] i2c: designware_i2c: Prepare for DM driver conversion Stefan Roese
2016-03-21  8:54   ` Bin Meng
2016-03-18  7:54 ` [U-Boot] [PATCH 5/6] i2c: designware_i2c: Add DM support Stefan Roese
2016-03-21  8:54   ` Bin Meng
2016-04-09 18:35   ` Simon Glass
2016-03-18  7:54 ` [U-Boot] [PATCH 6/6] i2c: designware_i2c: Add support for PCI(e) based I2C cores (x86) Stefan Roese
2016-03-21  8:54   ` Bin Meng
2016-03-21  9:03     ` Stefan Roese
2016-03-21  9:05       ` Bin Meng
2016-03-21 12:04       ` Stefan Roese
2016-03-21 12:43         ` Bin Meng
2016-03-21 12:52           ` Stefan Roese
2016-03-21 14:04           ` Stefan Roese
2016-03-28  6:01             ` Bin Meng
2016-04-04 14:53               ` Stefan Roese
2016-04-11 15:03                 ` Stefan Roese
2016-04-20 14:40                   ` Simon Glass
2016-04-20 14:58                     ` Stefan Roese
2016-04-20 15:09                       ` Simon Glass
2016-04-20 15:17                         ` Stefan Roese
2016-03-18 11:09 ` [U-Boot] [PATCH 1/6] i2c: designware_i2c: Add ic_enable_status to ic_regs struct Marek Vasut
2016-03-21  8:54 ` Bin Meng

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.