[U-Boot] [PATCH 5/6] I2C: Driver changes for FDT support
Simon Glass
sjg at chromium.org
Mon Nov 12 22:49:23 CET 2012
Hi Rajeshwari,
On Mon, Nov 12, 2012 at 5:03 AM, Rajeshwari Birje
<rajeshwari.birje at gmail.com> wrote:
> Hi Simon Glass,
>
> Thank you for comments.
>
> On Fri, Nov 9, 2012 at 6:34 AM, Simon Glass <sjg at chromium.org> wrote:
>> Hi Rajeshwari,
>>
>> On Wed, Nov 7, 2012 at 9:30 PM, Rajeshwari Shinde
>> <rajeshwari.s at samsung.com> wrote:
>>> Functions added to get the I2C bus number and reset I2C bus using
>>> FDT node.
>>>
>>> Signed-off-by: Simon Glass <sjg at chromium.org>
>>> Signed-off-by: Rajeshwari Shinde <rajeshwari.s at samsung.com>
>>> ---
>>> drivers/i2c/s3c24x0_i2c.c | 89 ++++++++++++++++++++++++++++++++++++++++++++-
>>> drivers/i2c/s3c24x0_i2c.h | 7 ++++
>>> include/i2c.h | 28 ++++++++++++++
>>> 3 files changed, 123 insertions(+), 1 deletions(-)
>>>
>>> diff --git a/drivers/i2c/s3c24x0_i2c.c b/drivers/i2c/s3c24x0_i2c.c
>>> index 9bc4c7f..952f76c 100644
>>> --- a/drivers/i2c/s3c24x0_i2c.c
>>> +++ b/drivers/i2c/s3c24x0_i2c.c
>>> @@ -27,6 +27,7 @@
>>> */
>>>
>>> #include <common.h>
>>> +#include <fdtdec.h>
>>> #ifdef CONFIG_EXYNOS5
>>> #include <asm/arch/clk.h>
>>> #include <asm/arch/cpu.h>
>>> @@ -60,7 +61,14 @@
>>> #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")));
>>>
>>> #ifndef CONFIG_EXYNOS5
>>> static int GetI2CSDA(void)
>>> @@ -507,4 +515,83 @@ int i2c_write(uchar chip, uint addr, int alen, uchar *buffer, int len)
>>> (i2c, I2C_WRITE, chip << 1, &xaddr[4 - alen], alen, buffer,
>>> len) != 0);
>>> }
>>> +
>>> +#ifdef CONFIG_OF_CONTROL
>>> +unsigned int i2c_fdt_bus_values(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->node = node;
>>> + bus->bus_num = i2c_busses++;
>>> + }
>>> +
>>> + return count;
>>> +}
>>> +
>>
>> This seems to be the init function for i2c, so how about calling it
>> board_i2c_init() or similar. I don't think you need to return the
>> count of i2c ports - you can just use CONFIG_MAX_I2C_NUM anywhere that
>> it is needed.
>>
>>> +static struct s3c24x0_i2c_bus *get_bus(int bus_idx)
>>
>> should probably be unsigned so that you don't need to check < 0
>>
>>> +{
>>> + 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(const void *blob, int node)
>>> +{
>>> + enum fdt_compat_id compat;
>>> + fdt_addr_t reg;
>>> + int i;
>>> +
>>> + compat = fdtdec_lookup(blob, node);
>>> + if (compat != COMPAT_SAMSUNG_S3C2440_I2C) {
>>> + debug("%s: Not a supported I2C node\n", __func__);
>>> + return -1;
>>> + }
>>> + reg = fdtdec_get_addr(blob, node, "reg");
>>> + for (i = 0; i < i2c_busses; i++) {
>>> + if (reg == (fdt_addr_t)(uintptr_t)i2c_bus[i].regs)
>>> + return i;
>>> + }
>>
>> This seems painful. You already have a table of node versus bus
>> number, created in the function above, so why not just look in that?
>>
> You want me to use get_bus function? but that function takes the bus
> number and gives the handle to the structure and not the i2c channel
> register address. Hence we cannot use that function here, instead I am
> using the structures populated in board_i2c_init.
> Do clarify if I have misunderstood the comment.
What I mean is that you have set up a table of node versus bus number
in i2c_fdt_bus_values(). So there should be no need to go and look
again at the fdt in i2c_get_bus_num_fdt() - you can just write a
function to search your table.
[snip]
Regards,
Simon
More information about the U-Boot
mailing list