Re: [PATCH] scsi:stex.c Support Pegasus 3 product

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Charles,

On Mon, Jun 6, 2016 at 5:53 PM, Charles Chiou <ch1102chiou@xxxxxxxxx> wrote:
> From: Charles <charles.chiou@xxxxxxxxxxxxxx>
>
> Pegasus series is a RAID support product by using Thunderbolt technology.
>
> The newest product, Pegasus 3 is support Thunderbolt 3 technology with another chip.
>
> 1.Change driver version.
>
> 2.Add Pegasus 3 VID, DID and define it's device address.
>
> 3.Pegasus 3 use msi interrupt, so stex_request_irq P3 type enable msi.
>
> 4.For hibernation, use msi_lock in stex_ss_handshake to prevent msi register write again when handshaking.
>
> 5.Pegasus 3 don't need read() as flush.
>
> 6.In stex_ss_intr & stex_abort, P3 only clear interrupt register when getting vendor defined interrupt.
>
> 7.Add reboot notifier and register it in stex_probe for all supported device.
>
> 8.For all supported device in restart flow, we get a callback from notifier and set S6flag for stex_shutdown & stex_hba_stop to send restart command to FW.
>
> Signed-off-by: Charles <charles.chiou@xxxxxxxxxxxxxx>
> Signed-off-by: Paul <paul.lyu@xxxxxxxxxxxxxx>
> ---
>  drivers/scsi/stex.c | 282 +++++++++++++++++++++++++++++++++++++++-------------
>  1 file changed, 214 insertions(+), 68 deletions(-)
>
> diff --git a/drivers/scsi/stex.c b/drivers/scsi/stex.c
> index 5b23175..9de2de2 100644
> --- a/drivers/scsi/stex.c
> +++ b/drivers/scsi/stex.c
> @@ -87,7 +95,7 @@ enum {
>         MU_STATE_STOP                           = 5,
>         MU_STATE_NOCONNECT                      = 6,
>
> -       MU_MAX_DELAY                            = 120,
> +       MU_MAX_DELAY                            = 50,

This won't cause problems for older adapters, right?

>         MU_HANDSHAKE_SIGNATURE                  = 0x55aaaa55,
>         MU_HANDSHAKE_SIGNATURE_HALF             = 0x5a5a0000,
>         MU_HARD_RESET_WAIT                      = 30000,
> @@ -540,11 +556,15 @@ stex_ss_send_cmd(struct st_hba *hba, struct req_msg *req, u16 tag)
>
>         ++hba->req_head;
>         hba->req_head %= hba->rq_count+1;
> -
> -       writel((addr >> 16) >> 16, hba->mmio_base + YH2I_REQ_HI);
> -       readl(hba->mmio_base + YH2I_REQ_HI); /* flush */
> -       writel(addr, hba->mmio_base + YH2I_REQ);
> -       readl(hba->mmio_base + YH2I_REQ); /* flush */
> +       if (hba->cardtype == st_P3) {
> +               writel((addr >> 16) >> 16, hba->mmio_base + YH2I_REQ_HI);
> +               writel(addr, hba->mmio_base + YH2I_REQ);
> +       } else {
> +               writel((addr >> 16) >> 16, hba->mmio_base + YH2I_REQ_HI);
> +               readl(hba->mmio_base + YH2I_REQ_HI); /* flush */
> +               writel(addr, hba->mmio_base + YH2I_REQ);
> +               readl(hba->mmio_base + YH2I_REQ); /* flush */
> +       }

The first writel() lines in each branch of the if statement are
identical, so they could be outside of it.

Would it make sense to add a helper that does the readl() flush only
for non-st_P3? This could be a function pointer in the hba structure
which shouldn't slow stuff down.

>  }
>
>  static void return_abnormal_state(struct st_hba *hba, int status)
> @@ -974,15 +994,31 @@ static irqreturn_t stex_ss_intr(int irq, void *__hba)
>
>         spin_lock_irqsave(hba->host->host_lock, flags);
>
> -       data = readl(base + YI2H_INT);
> -       if (data && data != 0xffffffff) {
> -               /* clear the interrupt */
> -               writel(data, base + YI2H_INT_C);
> -               stex_ss_mu_intr(hba);
> -               spin_unlock_irqrestore(hba->host->host_lock, flags);
> -               if (unlikely(data & SS_I2H_REQUEST_RESET))
> -                       queue_work(hba->work_q, &hba->reset_work);
> -               return IRQ_HANDLED;
> +       if (hba->cardtype == st_yel) {

I note that there's a few different card types beyond sd_yel and
st_P3. Does this function only get called for st_yel and st_P3?

> +               data = readl(base + YI2H_INT);
> +               if (data && data != 0xffffffff) {
> +                       /* clear the interrupt */
> +                       writel(data, base + YI2H_INT_C);
> +                       stex_ss_mu_intr(hba);
> +                       spin_unlock_irqrestore(hba->host->host_lock, flags);
> +                       if (unlikely(data & SS_I2H_REQUEST_RESET))
> +                               queue_work(hba->work_q, &hba->reset_work);
> +                       return IRQ_HANDLED;
> +               }
> +       } else {
> +               data = readl(base + PSCRATCH4);
> +               if (data != 0xffffffff) {
> +                       if (data != 0) {
> +                               /* clear the interrupt */
> +                               writel(data, base + PSCRATCH1);
> +                               writel((1 << 22), base + YH2I_INT);
> +                       }
> +                       stex_ss_mu_intr(hba);
> +                       spin_unlock_irqrestore(hba->host->host_lock, flags);
> +                       if (unlikely(data & SS_I2H_REQUEST_RESET))
> +                               queue_work(hba->work_q, &hba->reset_work);
> +                       return IRQ_HANDLED;
> +               }
>         }
>
>         spin_unlock_irqrestore(hba->host->host_lock, flags);
> @@ -1085,14 +1121,27 @@ static int stex_ss_handshake(struct st_hba *hba)
>         int ret = 0;
>
>         before = jiffies;
> -       while ((readl(base + YIOA_STATUS) & SS_MU_OPERATIONAL) == 0) {
> -               if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> -                       printk(KERN_ERR DRV_NAME
> -                               "(%s): firmware not operational\n",
> -                               pci_name(hba->pdev));
> -                       return -1;
> +
> +       if (hba->cardtype == st_yel) {

Same question as above. Does this only get called for st_yel and st_P3?

> +               while ((readl(base + YIOA_STATUS) & SS_MU_OPERATIONAL) == 0) {
> +                       if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> +                               printk(KERN_ERR DRV_NAME
> +                                       "(%s): firmware not operational\n",
> +                                       pci_name(hba->pdev));
> +                               return -1;
> +                       }
> +                       msleep(1);
> +               }
> +       } else if (hba->cardtype == st_P3) {

If it does only get called for st_yel and st_P3, then the if part of
this else-if is redundant.

> +               while ((readl(base + PSCRATCH3) & SS_MU_OPERATIONAL) == 0) {
> +                       if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> +                               printk(KERN_ERR DRV_NAME
> +                                       "(%s): firmware not operational\n",
> +                                       pci_name(hba->pdev));
> +                               return -1;
> +                       }
> +                       msleep(1);
>                 }
> -               msleep(1);
>         }
>
>         msg_h = (struct st_msg_header *)hba->dma_mem;
> @@ -1111,30 +1160,63 @@ static int stex_ss_handshake(struct st_hba *hba)
>         scratch_size = (hba->sts_count+1)*sizeof(u32);
>         h->scratch_size = cpu_to_le32(scratch_size);
>
> -       data = readl(base + YINT_EN);
> -       data &= ~4;
> -       writel(data, base + YINT_EN);
> -       writel((hba->dma_handle >> 16) >> 16, base + YH2I_REQ_HI);
> -       readl(base + YH2I_REQ_HI);
> -       writel(hba->dma_handle, base + YH2I_REQ);
> -       readl(base + YH2I_REQ); /* flush */
> +       if (hba->cardtype == st_yel) {

Same question again.

> +               data = readl(base + YINT_EN);
> +               data &= ~4;
> +               writel(data, base + YINT_EN);
> +               writel((hba->dma_handle >> 16) >> 16, base + YH2I_REQ_HI);
> +               readl(base + YH2I_REQ_HI);
> +               writel(hba->dma_handle, base + YH2I_REQ);
> +               readl(base + YH2I_REQ); /* flush */
> +       } else if (hba->cardtype == st_P3) {
> +               data = readl(base + YINT_EN);
> +               data &= ~(1 << 0);
> +               data &= ~(1 << 2);
> +               writel(data, base + YINT_EN);
> +               if (hba->msi_lock == 0) {
> +                       /* P3 MSI Register cannot access twice */
> +                       writel((1 << 6), base + YH2I_INT);
> +                       hba->msi_lock  = 1;
> +               }
> +               writel((hba->dma_handle >> 16) >> 16, base + YH2I_REQ_HI);
> +               writel(hba->dma_handle, base + YH2I_REQ);
> +       }

The two writel()s at the end of each branch of the if statement are
identical except for the readl() calls to flush the data in the non-P3
case. This would be simplified by adding a helper as discussed above.

> -       scratch = hba->scratch;
>         before = jiffies;
> -       while (!(le32_to_cpu(*scratch) & SS_STS_HANDSHAKE)) {
> -               if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> -                       printk(KERN_ERR DRV_NAME
> -                               "(%s): no signature after handshake frame\n",
> -                               pci_name(hba->pdev));
> -                       ret = -1;
> -                       break;
> +
> +       if (hba->cardtype == st_yel) {

Again, is this only called for st_yel and st_P3?

> +               scratch = hba->scratch;
> +
> +               while (!(le32_to_cpu(*scratch) & SS_STS_HANDSHAKE)) {
> +                       if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> +                               printk(KERN_ERR DRV_NAME
> +                                       "(%s): no signature after handshake frame\n",
> +                                       pci_name(hba->pdev));
> +                               ret = -1;
> +                               break;
> +                       }
> +                       rmb();
> +                       msleep(1);
>                 }
> -               rmb();
> -               msleep(1);
> +               memset(scratch, 0, scratch_size);
> +       } else if (hba->cardtype == st_P3) {
> +               while ((readl(base + MAILBOX_BASE + MAILBOX_HNDSHK_STS)
> +                & SS_STS_HANDSHAKE) == 0) {
> +                       if (time_after(jiffies, before + MU_MAX_DELAY * HZ)) {
> +                               printk(KERN_ERR DRV_NAME
> +                                       "(%s): no signature after handshake frame\n",
> +                                       pci_name(hba->pdev));
> +                               ret = -1;
> +                               break;
> +                       }
> +                       rmb();
> +                       msleep(1);
> +               }
> +               memset(hba->scratch, 0, scratch_size);

The memsets at the end of each branch of the if statement are identical.

>         }
>
> -       memset(scratch, 0, scratch_size);
>         msg_h->flag = 0;
> +
>         return ret;
>  }
>
> @@ -1144,7 +1226,7 @@ static int stex_handshake(struct st_hba *hba)
>         unsigned long flags;
>         unsigned int mu_status;
>
> -       err = (hba->cardtype == st_yel) ?
> +       err = (hba->cardtype == st_yel || hba->cardtype == st_P3) ?
>                 stex_ss_handshake(hba) : stex_common_handshake(hba);

This might be cleaner as an if statement.

>         spin_lock_irqsave(hba->host->host_lock, flags);
>         mu_status = hba->mu_status;
> @@ -1197,9 +1288,9 @@ static int stex_abort(struct scsi_cmnd *cmd)
>
>                 writel(data, base + ODBL);
>                 readl(base + ODBL); /* flush */
> -
>                 stex_mu_intr(hba, data);
>         }
> +

Unrelated whitespace change.

>         if (hba->wait_ccb == NULL) {
>                 printk(KERN_WARNING DRV_NAME
>                         "(%s): lost interrupt\n", pci_name(hba->pdev));
> @@ -1502,18 +1620,30 @@ static int stex_request_irq(struct st_hba *hba)
>         struct pci_dev *pdev = hba->pdev;
>         int status;
>
> -       if (msi) {
> +       if (hba->cardtype == st_yel) {

Again, is this only run for st_yel or st_P3?

Why not simplify this to:

-        if (msi) {
+        if (msi || hba->cardtype == st_P3) {

> +               if (msi) {
> +                       status = pci_enable_msi(pdev);
> +                       if (status != 0)
> +                               printk(KERN_ERR DRV_NAME
> +                                       "(%s): error %d setting up MSI\n",
> +                                        pci_name(pdev), status);
> +                       else
> +                               hba->msi_enabled = 1;
> +               } else
> +                       hba->msi_enabled = 0;
> +       } else if (hba->cardtype == st_P3) {
>                 status = pci_enable_msi(pdev);
>                 if (status != 0)
>                         printk(KERN_ERR DRV_NAME
>                                 "(%s): error %d setting up MSI\n",
> -                               pci_name(pdev), status);
> +                                pci_name(pdev), status);
>                 else
>                         hba->msi_enabled = 1;
>         } else
>                 hba->msi_enabled = 0;
>
> -       status = request_irq(pdev->irq, hba->cardtype == st_yel ?
> +       status = request_irq(pdev->irq,
> +               (hba->cardtype == st_yel || hba->cardtype == st_P3) ?
>                 stex_ss_intr : stex_intr, IRQF_SHARED, DRV_NAME, hba);
>
>         if (status != 0) {
> @@ -1546,6 +1676,9 @@ static int stex_probe(struct pci_dev *pdev, const struct pci_device_id *id)
>
>         pci_set_master(pdev);
>
> +       S6flag = 0;
> +       register_reboot_notifier(&stex_notifier);
> +

Adding the reboot notifier applies to all cards, so it should probably
be a separate patch.

>         host = scsi_host_alloc(&driver_template, sizeof(struct st_hba));
>
>         if (!host) {
> @@ -1736,28 +1870,29 @@ static void stex_hba_stop(struct st_hba *hba, int st_sleep_mic)
>
>         spin_lock_irqsave(hba->host->host_lock, flags);
>
> -       if (hba->cardtype == st_yel && hba->supports_pm == 1)
> -       {
> -               if(st_sleep_mic == ST_NOTHANDLED)
> -               {
> +       if ((hba->cardtype == st_yel && hba->supports_pm == 1)
> +               || (hba->cardtype == st_P3 && hba->supports_pm == 1)) {

if ((hba->cardtype == st_yel || hba->cardtype == st_P3) &&
hba->supports_pm == 1) {

is simpler.

> +               if (st_sleep_mic == ST_NOTHANDLED) {
>                         spin_unlock_irqrestore(hba->host->host_lock, flags);
>                         return;
>                 }
>         }
>         req = hba->alloc_rq(hba);
> -       if (hba->cardtype == st_yel) {
> +       if (hba->cardtype == st_yel || hba->cardtype == st_P3) {
>                 msg_h = (struct st_msg_header *)req - 1;
>                 memset(msg_h, 0, hba->rq_size);
>         } else
>                 memset(req, 0, hba->rq_size);
>
> -       if ((hba->cardtype == st_yosemite || hba->cardtype == st_yel)
> +       if ((hba->cardtype == st_yosemite || hba->cardtype == st_yel
> +               || hba->cardtype == st_P3)
>                 && st_sleep_mic == ST_IGNORED) {
>                 req->cdb[0] = MGT_CMD;
>                 req->cdb[1] = MGT_CMD_SIGNATURE;
>                 req->cdb[2] = CTLR_CONFIG_CMD;
>                 req->cdb[3] = CTLR_SHUTDOWN;
> -       } else if (hba->cardtype == st_yel && st_sleep_mic != ST_IGNORED) {
> +       } else if ((hba->cardtype == st_yel || hba->cardtype == st_P3)
> +               && st_sleep_mic != ST_IGNORED) {

Er, this will never get run.

We have:

if (hba->cardtype == st_yosemite || hba->cardtype == st_yel ||
hba->cardtype == st_P3) {
    // stuff
} else if ((hba->cardtype == st_yel || hba->cardtype == st_P3) &&
st_sleep_mic != ST_IGNORED) {
    // stuff
}

Should the two branches of the if statement be reversed or should the
first one be written like:

if (hba->cardtype == st_yosemite || ((hba->cardtype == st_yel ||
hba->cardtype == st_P3) && st_sleep_mic == ST_IGNORED)) {

>                 req->cdb[0] = MGT_CMD;
>                 req->cdb[1] = MGT_CMD_SIGNATURE;
>                 req->cdb[2] = CTLR_CONFIG_CMD;
> @@ -1768,16 +1903,14 @@ static void stex_hba_stop(struct st_hba *hba, int st_sleep_mic)
>                 req->cdb[1] = CTLR_POWER_STATE_CHANGE;
>                 req->cdb[2] = CTLR_POWER_SAVING;
>         }
> -
>         hba->ccb[tag].cmd = NULL;
>         hba->ccb[tag].sg_count = 0;
>         hba->ccb[tag].sense_bufflen = 0;
>         hba->ccb[tag].sense_buffer = NULL;
>         hba->ccb[tag].req_type = PASSTHRU_REQ_TYPE;
> -
>         hba->send(hba, req, tag);
> -       spin_unlock_irqrestore(hba->host->host_lock, flags);
>
> +       spin_unlock_irqrestore(hba->host->host_lock, flags);

More unrelated whitespace changes.

>         before = jiffies;
>         while (hba->ccb[tag].req_type & PASSTHRU_REQ_TYPE) {
>                 if (time_after(jiffies, before + ST_INTERNAL_TIMEOUT * HZ)) {
> @@ -1821,24 +1954,29 @@ static void stex_remove(struct pci_dev *pdev)
>         scsi_host_put(hba->host);
>
>         pci_disable_device(pdev);
> +
> +       unregister_reboot_notifier(&stex_notifier);

Again, not P3 specific.

>  }
>
>  static void stex_shutdown(struct pci_dev *pdev)
>  {
>         struct st_hba *hba = pci_get_drvdata(pdev);
> -
> -       if (hba->supports_pm == 0)
> +       if (hba->supports_pm == 0) {
>                 stex_hba_stop(hba, ST_IGNORED);
> -       else
> +       } else if (hba->supports_pm == 1 && S6flag) {
> +               unregister_reboot_notifier(&stex_notifier);
> +               stex_hba_stop(hba, ST_S6);
> +       } else

Also not P3 specific.

>                 stex_hba_stop(hba, ST_S5);
>  }
>
> -static int stex_choice_sleep_mic(pm_message_t state)
> +static int stex_choice_sleep_mic(struct st_hba *hba, pm_message_t state)
>  {
>         switch (state.event) {
>         case PM_EVENT_SUSPEND:
>                 return ST_S3;
>         case PM_EVENT_HIBERNATE:
> +               hba->msi_lock = 0;
>                 return ST_S4;
>         default:
>                 return ST_NOTHANDLED;
> @@ -1864,6 +2003,13 @@ static int stex_resume(struct pci_dev *pdev)
>         stex_handshake(hba);
>         return 0;
>  }
> +
> +static int stex_halt(struct notifier_block *nb, unsigned long event, void *buf)
> +{
> +       S6flag = 1;
> +       return NOTIFY_OK;
> +}
> +

And again.

Why is this needed?

>  MODULE_DEVICE_TABLE(pci, stex_pci_tbl);
>
>  static struct pci_driver stex_pci_driver = {
> --
> 1.9.1
>
> --
> To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
> the body of a message to majordomo@xxxxxxxxxxxxxxx
> More majordomo info at  http://vger.kernel.org/majordomo-info.html

Thanks,

-- 
Julian Calaby

Email: julian.calaby@xxxxxxxxx
Profile: http://www.google.com/profiles/julian.calaby/
--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Index of Archives]     [SCSI Target Devel]     [Linux SCSI Target Infrastructure]     [Kernel Newbies]     [IDE]     [Security]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux ATA RAID]     [Linux IIO]     [Samba]     [Device Mapper]
  Powered by Linux