> -----Original Message-----
> From: tiejun.chen [mailto:tiejun.c...@windriver.com]
> Sent: Monday, October 18, 2010 16:56 PM
> To: Zang Roy-R61911
> Cc: linux-...@lists.infradead.org; Wood Scott-B07421; dedeki...@gmail.com; Lan
> Chunhe-B25806; linuxppc-...@ozlabs.org; a...@linux-foundation.org;
> dw...@infradead.org; Gala Kumar-B11780
> Subject: Re: [PATCH 1/2] P4080/eLBC: Make Freescale elbc interrupt common to
> elbc devices
> 
> Roy Zang wrote:
> > Move Freescale elbc interrupt from nand dirver to elbc driver.
> > Then all elbc devices can use the interrupt instead of ONLY nand.
> >
> > For former nand driver, it had the two functions:
> >
> > 1. detecting nand flash partitions;
> > 2. registering elbc interrupt.
> >
> > Now, second function is removed to fsl_lbc.c.
> >
> > Signed-off-by: Lan Chunhe-B25806 <b25...@freescale.com>
> > Signed-off-by: Roy Zang <tie-fei.z...@freescale.com>
> > Reviewed-by: Anton Vorontsov <cbouatmai...@gmail.com>
> > Cc: Wood Scott-B07421 <b07...@freescale.com>
> > ---
> >
> > These two patches are based on the following commits:
> > 1.  http://lists.infradead.org/pipermail/linux-mtd/2010-
> September/032112.html
> > 2.  http://lists.infradead.org/pipermail/linux-mtd/2010-
> September/032110.html
> > 3.  http://lists.infradead.org/pipermail/linux-mtd/2010-
> September/032111.html
> > According to Anton's comment, I merge 1 & 2 together and start a new thread.
> > Comparing the provided link:
> > 1.  Merge 1 & 2 together.
> > 2.  Some code style updates
> > 3.  Add counter protect for elbc driver remove
> > 4.  Rebase to 2.6.36-rc7
> >
> > Other histories from the links:
> > V2: Comparing with v1, according to the feedback, add some decorations.
> >
> > V3: Comparing with v2:
> > 1.  according to the feedback, add some decorations.
> > 2.  change of_platform_driver to platform_driver
> > 3.  rebase to 2.6.36-rc4
> >
> > V4: Comparing with v3
> > 1.  minor fix from type unsigned int to u32
> > 2.  fix platform_driver issue.
> > 3.  add mutex for nand probe
> >
> >  arch/powerpc/Kconfig               |    7 +-
> >  arch/powerpc/include/asm/fsl_lbc.h |   33 +++-
> >  arch/powerpc/sysdev/fsl_lbc.c      |  229 ++++++++++++++---
> >  drivers/mtd/nand/Kconfig           |    1 +
> >  drivers/mtd/nand/fsl_elbc_nand.c   |  482 +++++++++++++++------------------
> ---
> >  5 files changed, 425 insertions(+), 327 deletions(-)
> >
> > diff --git a/arch/powerpc/Kconfig b/arch/powerpc/Kconfig
> > index 631e5a0..44df1ba 100644
> > --- a/arch/powerpc/Kconfig
> > +++ b/arch/powerpc/Kconfig
> > @@ -687,9 +687,12 @@ config 4xx_SOC
> >     bool
> >
> >  config FSL_LBC
> > -   bool
> > +   bool "Freescale Local Bus support"
> > +   depends on FSL_SOC
> >     help
> > -     Freescale Localbus support
> > +     Enables reporting of errors from the Freescale local bus
> > +     controller.  Also contains some common code used by
> > +     drivers for specific local bus peripherals.
> >
> >  config FSL_GTM
> >     bool
> > diff --git a/arch/powerpc/include/asm/fsl_lbc.h
> b/arch/powerpc/include/asm/fsl_lbc.h
> > index 1b5a210..0c40c05 100644
> > --- a/arch/powerpc/include/asm/fsl_lbc.h
> > +++ b/arch/powerpc/include/asm/fsl_lbc.h
> > @@ -1,9 +1,10 @@
> >  /* Freescale Local Bus Controller
> >   *
> > - * Copyright (c) 2006-2007 Freescale Semiconductor
> > + * Copyright (c) 2006-2007, 2010 Freescale Semiconductor
> >   *
> >   * Authors: Nick Spence <nick.spe...@freescale.com>,
> >   *          Scott Wood <scottw...@freescale.com>
> > + *          Jack Lan <jack....@freescale.com>
> >   *
> >   * This program is free software; you can redistribute it and/or modify
> >   * it under the terms of the GNU General Public License as published by
> > @@ -26,6 +27,8 @@
> >  #include <linux/compiler.h>
> >  #include <linux/types.h>
> >  #include <linux/io.h>
> > +#include <linux/device.h>
> > +#include <linux/spinlock.h>
> >
> >  struct fsl_lbc_bank {
> >     __be32 br;             /**< Base Register  */
> > @@ -125,13 +128,23 @@ struct fsl_lbc_regs {
> >  #define LTESR_ATMW 0x00800000
> >  #define LTESR_ATMR 0x00400000
> >  #define LTESR_CS   0x00080000
> > +#define LTESR_UPM  0x00000002
> >  #define LTESR_CC   0x00000001
> >  #define LTESR_NAND_MASK (LTESR_FCT | LTESR_PAR | LTESR_CC)
> > +#define LTESR_MASK      (LTESR_BM | LTESR_FCT | LTESR_PAR | LTESR_WP \
> > +                    | LTESR_ATMW | LTESR_ATMR | LTESR_CS | LTESR_UPM \
> > +                    | LTESR_CC)
> > +#define LTESR_CLEAR        0xFFFFFFFF
> > +#define LTECCR_CLEAR       0xFFFFFFFF
> > +#define LTESR_STATUS       LTESR_MASK
> > +#define LTEIR_ENABLE       LTESR_MASK
> > +#define LTEDR_ENABLE       0x00000000
> >     __be32 ltedr;           /**< Transfer Error Disable Register */
> >     __be32 lteir;           /**< Transfer Error Interrupt Register */
> >     __be32 lteatr;          /**< Transfer Error Attributes Register */
> >     __be32 ltear;           /**< Transfer Error Address Register */
> > -   u8 res6[0xC];
> > +   __be32 lteccr;          /**< Transfer Error ECC Register */
> > +   u8 res6[0x8];
> >     __be32 lbcr;            /**< Configuration Register */
> >  #define LBCR_LDIS  0x80000000
> >  #define LBCR_LDIS_SHIFT    31
> > @@ -265,7 +278,23 @@ static inline void fsl_upm_end_pattern(struct fsl_upm
> *upm)
> >             cpu_relax();
> >  }
> >
> > +/* overview of the fsl lbc controller */
> > +
> > +struct fsl_lbc_ctrl {
> > +   /* device info */
> > +   struct device                   *dev;
> > +   struct fsl_lbc_regs __iomem     *regs;
> > +   int                             irq;
> > +   wait_queue_head_t               irq_wait;
> > +   spinlock_t                      lock;
> > +   void                            *nand;
> > +
> > +   /* status read from LTESR by irq handler */
> > +   unsigned int                    irq_status;
> > +};
> > +
> >  extern int fsl_upm_run_pattern(struct fsl_upm *upm, void __iomem *io_base,
> >                            u32 mar);
> > +extern struct fsl_lbc_ctrl *fsl_lbc_ctrl_dev;
> >
> >  #endif /* __ASM_FSL_LBC_H */
> > diff --git a/arch/powerpc/sysdev/fsl_lbc.c b/arch/powerpc/sysdev/fsl_lbc.c
> > index dceb8d1..4bb0336 100644
> > --- a/arch/powerpc/sysdev/fsl_lbc.c
> > +++ b/arch/powerpc/sysdev/fsl_lbc.c
> > @@ -2,8 +2,11 @@
> >   * Freescale LBC and UPM routines.
> >   *
> >   * Copyright (c) 2007-2008  MontaVista Software, Inc.
> > + * Copyright (c) 2010 Freescale Semiconductor
> >   *
> >   * Author: Anton Vorontsov <avoront...@ru.mvista.com>
> > + * Author: Jack Lan <jack....@freescale.com>
> > + * Author: Roy Zang <tie-fei.z...@freescale.com>
> >   *
> >   * This program is free software; you can redistribute it and/or modify
> >   * it under the terms of the GNU General Public License as published by
> > @@ -19,39 +22,16 @@
> >  #include <linux/types.h>
> >  #include <linux/io.h>
> >  #include <linux/of.h>
> > +#include <linux/slab.h>
> > +#include <linux/platform_device.h>
> > +#include <linux/interrupt.h>
> > +#include <linux/mod_devicetable.h>
> >  #include <asm/prom.h>
> >  #include <asm/fsl_lbc.h>
> >
> >  static spinlock_t fsl_lbc_lock = __SPIN_LOCK_UNLOCKED(fsl_lbc_lock);
> > -static struct fsl_lbc_regs __iomem *fsl_lbc_regs;
> > -
> > -static char __initdata *compat_lbc[] = {
> > -   "fsl,pq2-localbus",
> > -   "fsl,pq2pro-localbus",
> > -   "fsl,pq3-localbus",
> > -   "fsl,elbc",
> > -};
> > -
> > -static int __init fsl_lbc_init(void)
> > -{
> > -   struct device_node *lbus;
> > -   int i;
> > -
> > -   for (i = 0; i < ARRAY_SIZE(compat_lbc); i++) {
> > -           lbus = of_find_compatible_node(NULL, NULL, compat_lbc[i]);
> > -           if (lbus)
> > -                   goto found;
> > -   }
> > -   return -ENODEV;
> > -
> > -found:
> > -   fsl_lbc_regs = of_iomap(lbus, 0);
> > -   of_node_put(lbus);
> > -   if (!fsl_lbc_regs)
> > -           return -ENOMEM;
> > -   return 0;
> > -}
> > -arch_initcall(fsl_lbc_init);
> > +struct fsl_lbc_ctrl *fsl_lbc_ctrl_dev;
> > +EXPORT_SYMBOL(fsl_lbc_ctrl_dev);
> >
> >  /**
> >   * fsl_lbc_find - find Localbus bank
> > @@ -65,13 +45,15 @@ arch_initcall(fsl_lbc_init);
> >  int fsl_lbc_find(phys_addr_t addr_base)
> >  {
> >     int i;
> > +   struct fsl_lbc_regs __iomem *lbc;
> >
> > -   if (!fsl_lbc_regs)
> > +   if (!fsl_lbc_ctrl_dev || !fsl_lbc_ctrl_dev->regs)
> >             return -ENODEV;
> >
> > -   for (i = 0; i < ARRAY_SIZE(fsl_lbc_regs->bank); i++) {
> > -           __be32 br = in_be32(&fsl_lbc_regs->bank[i].br);
> > -           __be32 or = in_be32(&fsl_lbc_regs->bank[i].or);
> > +   lbc = fsl_lbc_ctrl_dev->regs;
> > +   for (i = 0; i < ARRAY_SIZE(lbc->bank); i++) {
> > +           __be32 br = in_be32(&lbc->bank[i].br);
> > +           __be32 or = in_be32(&lbc->bank[i].or);
> >
> >             if (br & BR_V && (br & or & BR_BA) == addr_base)
> >                     return i;
> > @@ -94,22 +76,27 @@ int fsl_upm_find(phys_addr_t addr_base, struct fsl_upm
> *upm)
> >  {
> >     int bank;
> >     __be32 br;
> > +   struct fsl_lbc_regs __iomem *lbc;
> >
> >     bank = fsl_lbc_find(addr_base);
> >     if (bank < 0)
> >             return bank;
> >
> > -   br = in_be32(&fsl_lbc_regs->bank[bank].br);
> > +   if (!fsl_lbc_ctrl_dev || !fsl_lbc_ctrl_dev->regs)
> > +           return -ENODEV;
> > +
> > +   lbc = fsl_lbc_ctrl_dev->regs;
> > +   br = in_be32(&lbc->bank[bank].br);
> >
> >     switch (br & BR_MSEL) {
> >     case BR_MS_UPMA:
> > -           upm->mxmr = &fsl_lbc_regs->mamr;
> > +           upm->mxmr = &lbc->mamr;
> >             break;
> >     case BR_MS_UPMB:
> > -           upm->mxmr = &fsl_lbc_regs->mbmr;
> > +           upm->mxmr = &lbc->mbmr;
> >             break;
> >     case BR_MS_UPMC:
> > -           upm->mxmr = &fsl_lbc_regs->mcmr;
> > +           upm->mxmr = &lbc->mcmr;
> >             break;
> >     default:
> >             return -EINVAL;
> > @@ -148,9 +135,12 @@ int fsl_upm_run_pattern(struct fsl_upm *upm, void
> __iomem *io_base, u32 mar)
> >     int ret = 0;
> >     unsigned long flags;
> >
> > +   if (!fsl_lbc_ctrl_dev || !fsl_lbc_ctrl_dev->regs)
> > +           return -ENODEV;
> > +
> >     spin_lock_irqsave(&fsl_lbc_lock, flags);
> >
> > -   out_be32(&fsl_lbc_regs->mar, mar);
> > +   out_be32(&fsl_lbc_ctrl_dev->regs->mar, mar);
> >
> >     switch (upm->width) {
> >     case 8:
> > @@ -172,3 +162,166 @@ int fsl_upm_run_pattern(struct fsl_upm *upm, void
> __iomem *io_base, u32 mar)
> >     return ret;
> >  }
> >  EXPORT_SYMBOL(fsl_upm_run_pattern);
> > +
> > +static int __devinit fsl_lbc_ctrl_init(struct fsl_lbc_ctrl *ctrl)
> > +{
> > +   struct fsl_lbc_regs __iomem *lbc = ctrl->regs;
> > +
> > +   /* clear event registers */
> > +   setbits32(&lbc->ltesr, LTESR_CLEAR);
> > +   out_be32(&lbc->lteatr, 0);
> > +   out_be32(&lbc->ltear, 0);
> > +   out_be32(&lbc->lteccr, LTECCR_CLEAR);
> > +   out_be32(&lbc->ltedr, LTEDR_ENABLE);
> > +
> > +   /* Enable interrupts for any detected events */
> > +   out_be32(&lbc->lteir, LTEIR_ENABLE);
> > +
> > +   return 0;
> > +}
> > +
> > +/*
> > + * NOTE: This interrupt is used to report localbus events of various kinds,
> > + * such as transaction errors on the chipselects.
> > + */
> > +
> > +static irqreturn_t fsl_lbc_ctrl_irq(int irqno, void *data)
> > +{
> > +   struct fsl_lbc_ctrl *ctrl = data;
> > +   struct fsl_lbc_regs __iomem *lbc = ctrl->regs;
> > +   u32 status;
> > +
> > +   status = in_be32(&lbc->ltesr);
> > +   if (!status)
> > +           return IRQ_NONE;
> > +
> > +   out_be32(&lbc->ltesr, LTESR_CLEAR);
> > +   out_be32(&lbc->lteatr, 0);
> > +   out_be32(&lbc->ltear, 0);
> > +   ctrl->irq_status = status;
> > +
> > +   if (status & LTESR_BM)
> > +           dev_err(ctrl->dev, "Local bus monitor time-out: "
> > +                   "LTESR 0x%08X\n", status);
> > +   if (status & LTESR_WP)
> > +           dev_err(ctrl->dev, "Write protect error: "
> > +                   "LTESR 0x%08X\n", status);
> > +   if (status & LTESR_ATMW)
> > +           dev_err(ctrl->dev, "Atomic write error: "
> > +                   "LTESR 0x%08X\n", status);
> > +   if (status & LTESR_ATMR)
> > +           dev_err(ctrl->dev, "Atomic read error: "
> > +                   "LTESR 0x%08X\n", status);
> > +   if (status & LTESR_CS)
> > +           dev_err(ctrl->dev, "Chip select error: "
> > +                   "LTESR 0x%08X\n", status);
> > +   if (status & LTESR_UPM)
> > +           ;
> > +   if (status & LTESR_FCT) {
> > +           dev_err(ctrl->dev, "FCM command time-out: "
> > +                   "LTESR 0x%08X\n", status);
> > +           smp_wmb();
> > +           wake_up(&ctrl->irq_wait);
> > +   }
> > +   if (status & LTESR_PAR) {
> > +           dev_err(ctrl->dev, "Parity or Uncorrectable ECC error: "
> > +                   "LTESR 0x%08X\n", status);
> > +           smp_wmb();
> > +           wake_up(&ctrl->irq_wait);
> > +   }
> > +   if (status & LTESR_CC) {
> > +           smp_wmb();
> > +           wake_up(&ctrl->irq_wait);
> > +   }
> > +   if (status & ~LTESR_MASK)
> > +           dev_err(ctrl->dev, "Unknown error: "
> > +                   "LTESR 0x%08X\n", status);
> > +   return IRQ_HANDLED;
> > +}
> > +
> > +/*
> > + * fsl_lbc_ctrl_probe
> > + *
> > + * called by device layer when it finds a device matching
> > + * one our driver can handled. This code allocates all of
> > + * the resources needed for the controller only.  The
> > + * resources for the NAND banks themselves are allocated
> > + * in the chip probe function.
> > +*/
> > +
> > +static int __devinit fsl_lbc_ctrl_probe(struct platform_device *dev)
> > +{
> > +   int ret;
> > +
> > +   if (!dev->dev.of_node) {
> > +           dev_err(&dev->dev, "Device OF-Node is NULL");
> > +           return -EFAULT;
> > +   }
> > +
> > +   fsl_lbc_ctrl_dev = kzalloc(sizeof(*fsl_lbc_ctrl_dev), GFP_KERNEL);
> > +   if (!fsl_lbc_ctrl_dev)
> > +           return -ENOMEM;
> > +
> > +   dev_set_drvdata(&dev->dev, fsl_lbc_ctrl_dev);
> > +
> > +   spin_lock_init(&fsl_lbc_ctrl_dev->lock);
> > +   init_waitqueue_head(&fsl_lbc_ctrl_dev->irq_wait);
> > +
> > +   fsl_lbc_ctrl_dev->regs = of_iomap(dev->dev.of_node, 0);
> > +   if (!fsl_lbc_ctrl_dev->regs) {
> > +           dev_err(&dev->dev, "failed to get memory region\n");
> > +           ret = -ENODEV;
> > +           goto err;
> 
> Looks you always iounmap(fsl_lbc_ctrl_dev->regs) on position 'err' but here
> of_iomap() is already failed you should skip iounmap() fsl_lbc_ctrl_dev->regs
> again. So you should improve that as the following on 'err', or layout 'err'
> in
> gain.
> ------
>       if(fsl_lbc_ctrl_dev->regs)
>               iounmap(fsl_lbc_ctrl_dev->regs);
> 
> Tiejun

You are right!
How about
 
        if (!fsl_lbc_ctrl_dev->regs) {
                dev_err(&dev->dev, "failed to get memory region\n");
                kfree(fsl_lbc_ctrl_dev);
                return -ENOMEM;
        }

...

Thanks.
Roy
_______________________________________________
Linuxppc-dev mailing list
Linuxppc-dev@lists.ozlabs.org
https://lists.ozlabs.org/listinfo/linuxppc-dev

Reply via email to