>-----Original Message-----
>From: Daniel Lezcano [mailto:daniel.lezc...@linaro.org]
>Sent: Tuesday, May 21, 2013 6:14 PM
>To: Lu Jingchang-B35083
>Cc: linux-kernel@vger.kernel.org; linux-arm-ker...@lists.infradead.org;
>john.stu...@linaro.org; t...@linutronix.de; shawn....@linaro.org;
>s.ha...@pengutronix.de
>Subject: Re: [PATCH v4] clocksource: add Vybrid Family pit timer support
>
>On 05/21/2013 09:27 AM, Jingchang Lu wrote:
>> Add Freescale Vybrid Family period interrupt timer support.
>>
>> Signed-off-by: Jingchang Lu <b35...@freescale.com>
>> ---
>> v4:
>>   use family name names driver and symbol instead of SoC name.
>>   remove redundant code.
>>   use BUG_ON instead of WARN_ON.
>>   add necessory comment information.
>>
>>  drivers/clocksource/Kconfig         |   5 +
>>  drivers/clocksource/Makefile        |   1 +
>>  drivers/clocksource/mvf_pit_timer.c | 187
>> ++++++++++++++++++++++++++++++++++++
>>  3 files changed, 193 insertions(+)
>>  create mode 100644 drivers/clocksource/mvf_pit_timer.c
>
>[ ... ]
>
>> +
>> +static int pit_set_next_event(unsigned long delta,
>> +                            struct clock_event_device *unused) {
>> +    /*
>> +     * set a new value to PITLDVAL register will not restart the timer,
>> +     * to abort the current cycle and start a timer period with the new
>> +     * value, the timer must be disabled and enabled again.
>> +     * and the PITLAVAL should be set to delta minus one.
>
>Why delta "minus one" ?
[Lu Jingchang-B35083] 
This is required by the PIT hardware, it is a down counter, the PITLAVAL 
register should be set to (delta-1), it will raise an interrupt when it counts 
down to 0. Thanks!
>
>> +     */
>> +    pit_timer_disable();
>> +    __raw_writel(delta - 1, clkevt_base + PITLDVAL);
>> +    pit_timer_enable();
>> +
>> +    return 0;
>> +}
>> +
>> +static void pit_set_mode(enum clock_event_mode mode,
>> +                            struct clock_event_device *evt)
>> +{
>> +    switch (mode) {
>> +    case CLOCK_EVT_MODE_PERIODIC:
>> +            pit_timer_disable();
>> +            __raw_writel(cycle_per_jiffy - 1, clkevt_base + PITLDVAL);
>> +            pit_timer_enable();
>
>You can call pit_set_next_event(cycle_per_jiffy, evt) instead of adding
>redundant code, no ?
[Lu Jingchang-B35083] 
 Yes, I will do that. Thanks!
>
>> +            break;
>> +    default:
>> +            break;
>> +    }
>> +}
>
>[ ... ]
>
>> +
>> +static struct clock_event_device clockevent_pit = {
>> +    .name           = "MVF pit timer",
>> +    .features       = CLOCK_EVT_FEAT_PERIODIC | CLOCK_EVT_FEAT_ONESHOT,
>> +    .set_mode       = pit_set_mode,
>> +    .set_next_event = pit_set_next_event,
>> +    .rating         = 300,
>> +};
>> +
>> +static struct irqaction pit_timer_irq = {
>> +    .name           = "MVF pit timer",
>> +    .flags          = IRQF_DISABLED | IRQF_TIMER | IRQF_IRQPOLL,
>
>Did you take into account Thomas's comment ?
[Lu Jingchang-B35083] 
Sorry, I misunderstand Thomas's meaning, I will remove IRQF_DISABLED flag. 
Thanks! 
>
>> +    .handler        = pit_timer_interrupt,
>> +    .dev_id         = &clockevent_pit,
>> +};
>> +
>> +static int __init pit_clockevent_init(struct clk *pit_clk, int irq) {
>> +    unsigned int c = clk_get_rate(pit_clk);
>> +
>> +    __raw_writel(0, clkevt_base + PITTCTRL);
>> +    __raw_writel(PITTFLG_TIF, clkevt_base + PITTFLG);
>> +
>> +    clockevent_pit.cpumask = cpumask_of(0);
>
>The irq initialization is missing.
>
>       clockevent_pit.irq = irq;
>
[Lu Jingchang-B35083] 
Yes, I will add this. Thanks!

>> +    clockevents_config_and_register(&clockevent_pit, c, 2, 0xffffffff);
>
>Is it possible to have a comment with the justification for these values ?
[Lu Jingchang-B35083] 
Yes, I will add a comment for these values. Thanks!
>
>> +
>> +    BUG_ON(setup_irq(irq, &pit_timer_irq));
>> +    return 0;
>
>Everything is ok if you can't setup your timer ?
>
>> +}
>> +
>> +static void __init pit_timer_init(struct device_node *np) {
>> +    struct clk *pit_clk;
>> +    void __iomem *timer_base;
>> +    int irq;
>> +
>> +    timer_base = of_iomap(np, 0);
>> +    BUG_ON(!timer_base);
>
>You raise a bug and then you go ahead setting up the address with an
>invalid value, leading to a random crash.
[Lu Jingchang-B35083] 
The BUG_ON() will call BUG() if condition is true. It will print the stack 
trace and the current process will die, it won't go ahead, won't it? Thanks! 
>
>> +    /*
>> +     * PIT0 and PIT1 can be chained to build a 64-bit timer,
>> +     * so choose PIT2 as clocksource, PIT3 as clockevent device,
>> +     * and leave PIT0 and PIT1 unused for anyone else who needs them.
>> +     */
>> +    clksrc_base = timer_base + PITn_OFFSET(2);
>> +    clkevt_base = timer_base + PITn_OFFSET(3);
>> +
>> +    irq = irq_of_parse_and_map(np, 0);
>> +
>> +    pit_clk = of_clk_get(np, 0);
>> +    BUG_ON(IS_ERR(pit_clk));
>> +
>> +    BUG_ON(clk_prepare_enable(pit_clk));
>
>Same here.
>
>> +    cycle_per_jiffy = clk_get_rate(pit_clk)/(HZ);
>> +
>> +    /* enable the pit module */
>> +    __raw_writel(~PITMCR_MDIS, timer_base + PITMCR);
>> +
>> +    BUG_ON(pit_clocksource_init(pit_clk));
>> +
>> +    pit_clockevent_init(pit_clk, irq);
>
>Please, remove these BUG_ON, this is inconsistent especially with a one
>call init function. If pit_timer_init can't be initialized, just pr_err
>+ BUG.
[Lu Jingchang-B35083] 
Do you mean that I should not use the BUG_ON or I need print an error 
information by pr_err before BUG. Thanks!
>
>Thanks
>  -- Daniel
>
>
>--
> <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs
>
>Follow Linaro:  <http://www.facebook.com/pages/Linaro> Facebook |
><http://twitter.com/#!/linaroorg> Twitter | <http://www.linaro.org/linaro-
>blog/> Blog
>

Reply via email to