Re: [PATCH 1/2] cciss: kernel thread to detect changes on MSA2012

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

 



On Thu, 5 Mar 2009 17:26:47 -0600
Mike Miller <mike.miller@xxxxxx> wrote:

> PATCH 1 of 2
> 
> The MSA2000 the firmware cannot tell the driver to rescan when logical
> drives are added or deleted. This patch adds a check for unit attentions and
> if a change in the topology is detected a kernel thread will fire off and
> call rebuild_lun_table to update the drivers view of the world.
> 
> The MSA2012 uses out of band management for configuration unlike any other
> storage box we support. This is the reason for this patch.
> 
> Please consider this for inclusion.
> 
> Signed-off-by: Mike Miller <mike.miller@xxxxxx>
> 
> diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c
> index d2cb67b..d745d8c 100644
> --- a/drivers/block/cciss.c
> +++ b/drivers/block/cciss.c
> @@ -186,6 +186,7 @@ static int sendcmd_withirq(__u8 cmd, int ctlr, void *buff, size_t size,
>  			   __u8 page_code, int cmd_type);
>  
>  static void fail_all_cmds(unsigned long ctlr);
> +static int scan_thread(ctlr_info_t *h);
>  
>  #ifdef CONFIG_PROC_FS
>  static void cciss_procinit(int i);
> @@ -3008,6 +3009,69 @@ static irqreturn_t do_cciss_intr(int irq, void *dev_id)
>  	return IRQ_HANDLED;
>  }
>  
> +static int scan_thread(ctlr_info_t *h)
> +{
> +	int rc;
> +
> +	DECLARE_COMPLETION(wait);
> +	INIT_COMPLETION(wait);
> +	h->rescan_wait = &wait;

The blank line is in the wrong place.

This should use DECLARE_COMPLETION_ONSTACK() to keep lockdep happy.

Please test patches with lockdep enabled.

INIT_COMPLATION() isn't needed - DECLARE_COMPLETION[_ONSTACK]() already
did that.

> +	for (;;) {
> +		rc = wait_for_completion_timeout(&wait);

that won't compile.

> +		if (!rc)
> +			continue;
> +		else
> +			rebuild_lun_table(h, 0);
> +
> +		INIT_COMPLETION(wait);

No need to reinitialise it here.

> +	}
> +
> +	return 0;
> +}
> +
> +static int check_for_unit_attention(ctlr_info_t *h, CommandList_struct *c)
> +{
> +	if (CMD_TARGET_STATUS && (c->err_info->ScsiStatus ==
> +			SAM_STAT_CHECK_CONDITION) &&
> +			c->err_info->SenseInfo[2] == UNIT_ATTENTION) {

You could do

	if (!<that expression>)
		return 0;

> +		switch (c->err_info->SenseInfo[12]) {
> +		case STATE_CHANGED:
> +			printk(KERN_WARNING "cciss%d: a state change "
> +				"detected, command retried\n", h->ctlr);
> +			return 1;
> +		break;
> +		case LUN_FAILED:
> +			printk(KERN_WARNING "cciss%d: LUN failure "
> +				"detected, action required\n", h->ctlr);
> +			return 1;
> +		break;
> +		case REPORT_LUNS_CHANGED:
> +			printk(KERN_WARNING "cciss%d: report LUN data "
> +				"changed\n", h->ctlr);
> +			if (h->rescan_wait)
> +				complete(h->rescan_wait);
> +			return 1;
> +		break;
> +		case POWER_OR_RESET:
> +			printk(KERN_WARNING "cciss%d: a power on "
> +				"or device reset detected\n", h->ctlr);
> +			return 1;
> +		break;
> +		case UNIT_ATTENTION_CLEARED:
> +			printk(KERN_WARNING "cciss%d: unit attention "
> +			    "cleared by another initiator\n", h->ctlr);
> +			return 1;
> +		break;
> +		default:
> +			printk(KERN_WARNING "cciss%d: unknown "
> +				"unit attention detected\n", h->ctlr);
> +				return 1;
> +		}
> +	}

and then save a tabstop for all the above.

> +	return 0;
> +}
> +
>  /*
>   *  We cannot read the structure directly, for portability we must use
>   *   the io functions.
> @@ -3600,6 +3664,7 @@ static int __devinit cciss_init_one(struct pci_dev *pdev,
>  	int rc;
>  	int dac, return_code;
>  	InquiryData_struct *inq_buff = NULL;
> +	struct task_struct *cciss_scan_thread;
>  
>  	if (reset_devices) {
>  		/* Reset the controller with a PCI power-cycle */
> @@ -3751,6 +3816,11 @@ static int __devinit cciss_init_one(struct pci_dev *pdev,
>  	hba[i]->busy_initializing = 0;
>  
>  	rebuild_lun_table(hba[i], 1);
> +	cciss_scan_thread = kthread((void *)scan_thread, hba[i],
> +				"scan_thread");

This will appears in `ps' output as "scan_thread".  Nobody will know
where it came from.  I suggest that it have "cciss" in its name.

Do we really need one of these per controller?  Can't arrange for a
single thread kernel-wide?

Do we really need the thread to be running all the time?  Can't create
it on demand?  Perhaps use the kernel/async.c infrastructure?  Or
perhaps schedule_work() or schedule_delayed_work()?


> +	if (IS_ERR(cciss_scan_thread))
> +		return PTR_ERR(cciss_scan_thread);
> +
>  	return 1;
>  
>  clean4:
> diff --git a/drivers/block/cciss.h b/drivers/block/cciss.h
> index 15e2b84..c85783f 100644
> --- a/drivers/block/cciss.h
> +++ b/drivers/block/cciss.h
> @@ -121,6 +121,7 @@ struct ctlr_info
>  	struct sendcmd_reject_list scsi_rejects;
>  #endif
>  	unsigned char alive;
> +	struct completion *rescan_wait;
>  };

--
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