I2C: Driver changes for FDT support
Functions added to get the I2C bus number and reset I2C bus using FDT node. Signed-off-by: Rajeshwari Shinde <rajeshwari.s@samsung.com> Acked-by: Simon Glass <sjg@chromium.org> Acked-by: Heiko Schocher <hs@denx.de> Signed-off-by: Minkyu Kang <mk7.kang@samsung.com>
This commit is contained in:
parent
d055911887
commit
a9d2ae7014
|
@ -27,9 +27,11 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <common.h>
|
#include <common.h>
|
||||||
|
#include <fdtdec.h>
|
||||||
#if (defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
|
#if (defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
|
||||||
#include <asm/arch/clk.h>
|
#include <asm/arch/clk.h>
|
||||||
#include <asm/arch/cpu.h>
|
#include <asm/arch/cpu.h>
|
||||||
|
#include <asm/arch/pinmux.h>
|
||||||
#else
|
#else
|
||||||
#include <asm/arch/s3c24x0_cpu.h>
|
#include <asm/arch/s3c24x0_cpu.h>
|
||||||
#endif
|
#endif
|
||||||
|
@ -60,7 +62,14 @@
|
||||||
#define I2C_TIMEOUT 1 /* 1 second */
|
#define I2C_TIMEOUT 1 /* 1 second */
|
||||||
|
|
||||||
|
|
||||||
static unsigned int g_current_bus; /* Stores Current I2C Bus */
|
/*
|
||||||
|
* For SPL boot some boards need i2c before SDRAM is initialised so force
|
||||||
|
* variables to live in SRAM
|
||||||
|
*/
|
||||||
|
static unsigned int g_current_bus __attribute__((section(".data")));
|
||||||
|
static struct s3c24x0_i2c_bus i2c_bus[CONFIG_MAX_I2C_NUM]
|
||||||
|
__attribute__((section(".data")));
|
||||||
|
static int i2c_busses __attribute__((section(".data")));
|
||||||
|
|
||||||
#if !(defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
|
#if !(defined CONFIG_EXYNOS4 || defined CONFIG_EXYNOS5)
|
||||||
static int GetI2CSDA(void)
|
static int GetI2CSDA(void)
|
||||||
|
@ -512,4 +521,76 @@ int i2c_write(uchar chip, uint addr, int alen, uchar *buffer, int len)
|
||||||
(i2c, I2C_WRITE, chip << 1, &xaddr[4 - alen], alen, buffer,
|
(i2c, I2C_WRITE, chip << 1, &xaddr[4 - alen], alen, buffer,
|
||||||
len) != 0);
|
len) != 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef CONFIG_OF_CONTROL
|
||||||
|
void board_i2c_init(const void *blob)
|
||||||
|
{
|
||||||
|
int node_list[CONFIG_MAX_I2C_NUM];
|
||||||
|
int count, i;
|
||||||
|
|
||||||
|
count = fdtdec_find_aliases_for_id(blob, "i2c",
|
||||||
|
COMPAT_SAMSUNG_S3C2440_I2C, node_list,
|
||||||
|
CONFIG_MAX_I2C_NUM);
|
||||||
|
|
||||||
|
for (i = 0; i < count; i++) {
|
||||||
|
struct s3c24x0_i2c_bus *bus;
|
||||||
|
int node = node_list[i];
|
||||||
|
|
||||||
|
if (node <= 0)
|
||||||
|
continue;
|
||||||
|
bus = &i2c_bus[i];
|
||||||
|
bus->regs = (struct s3c24x0_i2c *)
|
||||||
|
fdtdec_get_addr(blob, node, "reg");
|
||||||
|
bus->id = pinmux_decode_periph_id(blob, node);
|
||||||
|
bus->node = node;
|
||||||
|
bus->bus_num = i2c_busses++;
|
||||||
|
exynos_pinmux_config(bus->id, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct s3c24x0_i2c_bus *get_bus(unsigned int bus_idx)
|
||||||
|
{
|
||||||
|
if (bus_idx < i2c_busses)
|
||||||
|
return &i2c_bus[bus_idx];
|
||||||
|
|
||||||
|
debug("Undefined bus: %d\n", bus_idx);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
int i2c_get_bus_num_fdt(int node)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < i2c_busses; i++) {
|
||||||
|
if (node == i2c_bus[i].node)
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
debug("%s: Can't find any matched I2C bus\n", __func__);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int i2c_reset_port_fdt(const void *blob, int node)
|
||||||
|
{
|
||||||
|
struct s3c24x0_i2c_bus *i2c;
|
||||||
|
int bus;
|
||||||
|
|
||||||
|
bus = i2c_get_bus_num_fdt(node);
|
||||||
|
if (bus < 0) {
|
||||||
|
debug("could not get bus for node %d\n", node);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
i2c = get_bus(bus);
|
||||||
|
if (!i2c) {
|
||||||
|
debug("get_bus() failed for node node %d\n", node);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
i2c_ch_init(i2c->regs, CONFIG_SYS_I2C_SPEED, CONFIG_SYS_I2C_SLAVE);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif /* CONFIG_HARD_I2C */
|
#endif /* CONFIG_HARD_I2C */
|
||||||
|
|
|
@ -30,4 +30,12 @@ struct s3c24x0_i2c {
|
||||||
u32 iicds;
|
u32 iicds;
|
||||||
u32 iiclc;
|
u32 iiclc;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct s3c24x0_i2c_bus {
|
||||||
|
int node; /* device tree node */
|
||||||
|
int bus_num; /* i2c bus number */
|
||||||
|
struct s3c24x0_i2c *regs;
|
||||||
|
enum periph_id id;
|
||||||
|
};
|
||||||
|
|
||||||
#endif /* _S3C24X0_I2C_H */
|
#endif /* _S3C24X0_I2C_H */
|
||||||
|
|
|
@ -262,4 +262,30 @@ extern int get_multi_scl_pin(void);
|
||||||
extern int get_multi_sda_pin(void);
|
extern int get_multi_sda_pin(void);
|
||||||
extern int multi_i2c_init(void);
|
extern int multi_i2c_init(void);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get FDT values for i2c bus.
|
||||||
|
*
|
||||||
|
* @param blob Device tree blbo
|
||||||
|
* @return the number of I2C bus
|
||||||
|
*/
|
||||||
|
void board_i2c_init(const void *blob);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Find the I2C bus number by given a FDT I2C node.
|
||||||
|
*
|
||||||
|
* @param blob Device tree blbo
|
||||||
|
* @param node FDT I2C node to find
|
||||||
|
* @return the number of I2C bus (zero based), or -1 on error
|
||||||
|
*/
|
||||||
|
int i2c_get_bus_num_fdt(int node);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset the I2C bus represented by the given a FDT I2C node.
|
||||||
|
*
|
||||||
|
* @param blob Device tree blbo
|
||||||
|
* @param node FDT I2C node to find
|
||||||
|
* @return 0 if port was reset, -1 if not found
|
||||||
|
*/
|
||||||
|
int i2c_reset_port_fdt(const void *blob, int node);
|
||||||
#endif /* _I2C_H_ */
|
#endif /* _I2C_H_ */
|
||||||
|
|
Loading…
Reference in New Issue