On Sun, May 10, 2020 at 01:23:29PM +0300, Tali Perry wrote:
> Add Nuvoton NPCM BMC I2C controller driver.

Some cosmetic changes needs to be done.

...

> +/*
> + * Nuvoton NPCM7xx I2C Controller driver
> + *
> + * Copyright (C) 2020 Nuvoton Technologies tali.pe...@nuvoton.com
> + */

So, entire file has C99 comment style, but this and few other places.
Any reason of inconsistency?

...

> +#if IS_ENABLED(CONFIG_DEBUG_FS)

Why?

> +#include <linux/debugfs.h>
> +#endif

...

> +//#define _I2C_DEBUG_

Leftover, remove.

...

> +// Common regs
> +#define NPCM_I2CSDA                       0x00
> +#define NPCM_I2CST                        0x02
> +#define NPCM_I2CCST                       0x04

> +#define NPCM_I2CCTL1                   0x06

Indentation issue. And it's better to use TABs over spaces here and
in the similar places above and below.

> +#define NPCM_I2CADDR1                     0x08
> +#define NPCM_I2CCTL2                      0x0A
> +#define NPCM_I2CADDR2                     0x0C
> +#define NPCM_I2CCTL3                      0x0E
> +#define NPCM_I2CCST2                      0x18
> +#define NPCM_I2CCST3                      0x19

...

> +// supported clk settings. values in KHz.
> +#define I2C_FREQ_MIN                      10
> +#define I2C_FREQ_MAX                      1000

_KHZ to both.

...

> +#define I2C_NUM_OF_ADDR 10

Is it 10-bit address support or what?

...

> +#if IS_ENABLED(CONFIG_DEBUG_FS)
> +static struct dentry *npcm_i2c_debugfs_dir;   /* i2c debugfs directory */
> +static const char *ber_cnt_name      = "ber_count";
> +static const char *rec_succ_cnt_name = "rec_succ_count";
> +static const char *rec_fail_cnt_name = "rec_fail_count";
> +static const char *nack_cnt_name     = "nack_count";
> +static const char *timeout_cnt_name  = "timeout_count";
> +#endif

Why are these global?

...

> +static void npcm_i2c_write_to_fifo_master(struct npcm_i2c *bus,
> +                                       u16 max_bytes_to_send)
> +{
> +     // Fill the FIFO, while the FIFO is not full and there are more bytes to
> +     // write

> +     while ((max_bytes_to_send--) &&

Inner parentheses are redundant.

> +            (I2C_HW_FIFO_SIZE - npcm_i2c_fifo_usage(bus))) {
> +             if (bus->wr_ind < bus->wr_size)
> +                     npcm_i2c_wr_byte(bus, bus->wr_buf[bus->wr_ind++]);
> +             else
> +                     npcm_i2c_wr_byte(bus, 0xFF);
> +     }
> +}

...

> +             // Clear stall only after setting STOP
> +             iowrite8(NPCM_I2CST_STASTR, bus->reg + NPCM_I2CST);
> +
> +             ret =  IRQ_HANDLED;

Indentation issue.

...

> +                             if (bus->wr_size)
> +                                     npcm_i2c_set_fifo(bus, -1,
> +                                                       bus->wr_size);
> +                             else
> +                                     npcm_i2c_set_fifo(bus, bus->rd_size,
> +                                                       -1);

These two looks much better on one line.

...

> +                             if (npcm_i2c_is_quick(bus) || bus->wr_size)
> +                                     npcm_i2c_wr_byte(bus, bus->dest_addr);
> +                             else
> +                                     npcm_i2c_wr_byte(bus, bus->dest_addr |
> +                                                           0x01);

0x01 has its definition, hasn't it?

...

> +     // Repeat the following sequence until SDA is released
> +     do {
> +             // Issue a single SCL toggle
> +             iowrite8(NPCM_I2CCST_TGSCL, bus->reg + NPCM_I2CCST);
> +             udelay(20);
> +             // If SDA line is inactive (high), stop
> +             if (npcm_i2c_get_SDA(_adap)) {
> +                     done = true;
> +                     status = 0;
> +             }
> +     } while (!done && iter--);

readx_poll_timeout() ?

...

> +#if IS_ENABLED(CONFIG_DEBUG_FS)
> +     if (!status) {

Why not positive condition?

        if (status) {
                ...
        } else {
                ...
        }

> +     } else {

> +     }
> +#endif

...

> +static int npcm_i2c_init_clk(struct npcm_i2c *bus, u32 bus_freq)
> +{
> +     u32  k1 = 0;
> +     u32  k2 = 0;
> +     u8   dbnct = 0;
> +     u32  sclfrq = 0;
> +     u8   hldt = 7;
> +     bool fast_mode = false;

> +     u32  src_clk_freq; // in KHz

src_clk_khz ?

> +
> +     src_clk_freq = bus->apb_clk / 1000;
> +     bus->bus_freq = bus_freq;
> +
> +     // 100KHz and below:

> +     if (bus_freq <= (I2C_MAX_STANDARD_MODE_FREQ / 1000)) {

Instead of all these / 1000 can't you use bus frequency in Hz and do division
when it's really needed?

> +             sclfrq = src_clk_freq / (bus_freq * 4);
> +
> +             if (sclfrq < SCLFRQ_MIN || sclfrq > SCLFRQ_MAX)
> +                     return -EDOM;
> +
> +             if (src_clk_freq >= 40000)
> +                     hldt = 17;
> +             else if (src_clk_freq >= 12500)
> +                     hldt = 15;
> +             else
> +                     hldt = 7;
> +     }
> +
> +     // 400KHz:
> +     else if (bus_freq <= (I2C_MAX_FAST_MODE_FREQ / 1000)) {
> +             sclfrq = 0;
> +             fast_mode = true;
> +
> +             if (src_clk_freq < 7500)
> +                     // 400KHZ cannot be supported for core clock < 7.5 MHZ
> +                     return -EDOM;
> +
> +             else if (src_clk_freq >= 50000) {
> +                     k1 = 80;
> +                     k2 = 48;
> +                     hldt = 12;
> +                     dbnct = 7;
> +             }
> +
> +             // Master or Slave with frequency > 25 MHZ
> +             else if (src_clk_freq > 25000) {

> +                     hldt = (DIV_ROUND_UP(src_clk_freq * 300,
> +                                                      1000000) + 7) & 0xFF;

How ' & 0xFF' is not no-op here and in the similar cases?

> +
> +                     k1 = DIV_ROUND_UP(src_clk_freq * 1600,
> +                                                1000000);
> +                     k2 = DIV_ROUND_UP(src_clk_freq * 900,
> +                                                1000000);

Perhaps,

#define clk_coef(freq, mul)     DIV_ROUND_UP((freq) * (mul), 1000000)

?

> +                     k1 = round_up(k1, 2);
> +                     k2 = round_up(k2 + 1, 2);
> +                     if (k1 < SCLFRQ_MIN || k1 > SCLFRQ_MAX ||
> +                         k2 < SCLFRQ_MIN || k2 > SCLFRQ_MAX)
> +                             return -EDOM;
> +             }
> +     }
> +
> +     else if (bus_freq <= (I2C_MAX_FAST_MODE_PLUS_FREQ / 1000)) {
> +             sclfrq = 0;
> +             fast_mode = true;
> +
> +             if (src_clk_freq < 24000)
> +             // 1MHZ cannot be supported for master core clock < 15 MHZ
> +             // or slave core clock < 24 MHZ
> +                     return -EDOM;
> +
> +             k1 = round_up((DIV_ROUND_UP(src_clk_freq * 620,
> +                                                  1000000)), 2);
> +             k2 = round_up((DIV_ROUND_UP(src_clk_freq * 380,
> +                                                  1000000) + 1), 2);
> +             if (k1 < SCLFRQ_MIN || k1 > SCLFRQ_MAX ||
> +                 k2 < SCLFRQ_MIN || k2 > SCLFRQ_MAX)
> +                     return -EDOM;
> +
> +             // Core clk > 40 MHZ
> +             if (src_clk_freq > 40000) {
> +                     // Set HLDT:
> +                     // SDA hold time:  (HLDT-7) * T(CLK) >= 120
> +                     // HLDT = 120/T(CLK) + 7 = 120 * FREQ(CLK) + 7
> +                     hldt = (DIV_ROUND_UP(src_clk_freq * 120,
> +                                                      1000000) + 7) & 0xFF;
> +             } else {
> +                     hldt = 7;
> +                     dbnct = 2;
> +             }
> +     }
> +
> +     // Frequency larger than 1 MHZ is not supported
> +     else
> +             return false;

> +     return true;
> +}

...

> +     ret = device_property_read_u32(&pdev->dev, "bus-frequency", &clk_freq);
> +     if (ret < 0) {
> +             dev_info(&pdev->dev, "Could not read bus-frequency property\n");

> +             clk_freq = 100000;

We have define for this, don't we?

> +     }

Wolfram, we discussed this simplified timings property parser,
any news about it?

...

> +static irqreturn_t npcm_i2c_bus_irq(int irq, void *dev_id)
> +{
> +     irqreturn_t ret;
> +     struct npcm_i2c *bus = dev_id;
> +
> +     bus->int_cnt++;
> +
> +     if (npcm_i2c_is_master(bus))
> +             bus->master_or_slave = I2C_MASTER;
> +
> +     if (bus->master_or_slave == I2C_MASTER) {
> +             bus->int_time_stamp = jiffies;
> +             ret = npcm_i2c_int_master_handler(bus);

> +             if (ret == IRQ_HANDLED)
> +                     return ret;

What's the point?

> +     }
> +     return IRQ_HANDLED;
> +}

...

> +     bus->dest_addr = (slave_addr << 1) & 0xFE;

How ' & 0xFE' part is not a no-op?

...

> +     time_left = jiffies +
> +                 msecs_to_jiffies(DEFAULT_STALL_COUNT) + 1;

It's perfectly one line. Fix here and in any other place with similar issue.

...

> +static int i2c_init_debugfs(struct platform_device *pdev, struct npcm_i2c 
> *bus)

Should be void.

...

> +     bus->irq = platform_get_irq(pdev, 0);
> +     if (bus->irq < 0) {

> +             dev_err(&pdev->dev, "failed to get IRQ number\n");

Duplicate message.

> +             return bus->irq;
> +     }

...

> +#if IS_ENABLED(CONFIG_DEBUG_FS)

Why? Okay, why here instead of making a stub?

> +     ret = i2c_init_debugfs(pdev, bus);

> +     if (ret < 0)
> +             return ret;

Wrong. Should not fail the probe.

> +#endif

...

> +#if IS_ENABLED(CONFIG_DEBUG_FS)

Why? Just make it always present in the structure.

> +     if (!!bus->debugfs) {

!! ???

> +             debugfs_remove_recursive(bus->debugfs);
> +             bus->debugfs = NULL;
> +     }
> +#endif

...

> +     npcm_i2c_debugfs_dir = debugfs_create_dir("i2c", NULL);

> +     if (IS_ERR_OR_NULL(npcm_i2c_debugfs_dir)) {
> +             pr_warn("i2c init of debugfs failed\n");
> +             npcm_i2c_debugfs_dir = NULL;
> +             return -ENOMEM;
> +     }

Shouldn't prevent driver to work.

...

> +     if (!!npcm_i2c_debugfs_dir) {

!! ???

> +             debugfs_remove_recursive(npcm_i2c_debugfs_dir);

> +             npcm_i2c_debugfs_dir = NULL;

What's the point?

> +     }

-- 
With Best Regards,
Andy Shevchenko


Reply via email to