Re: [RFC][PATCH 5/5] Add pm8001 SAS/SATA HBA driver resend

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

 



On Thu, Jul 02, 2009 at 09:43:57AM +0800, jack wang wrote:
> diff --git a/drivers/scsi/pm8001/Kconfig b/drivers/scsi/pm8001/Kconfig
> new file mode 100644
> index 0000000..4db6021
> --- /dev/null
> +++ b/drivers/scsi/pm8001/Kconfig
> @@ -0,0 +1,48 @@
> + config SCSI_PM8001
> + 	tristate "PMC-Sierra SPC 8001 SAS/SATA Based Host Adapter driver"
> + 	depends on PCI && SCSI
> + 	select SCSI_SAS_LIBSAS
> + 	help
> +		This driver supports PMC-Sierra PCIE SAS/SATA 8x6G SPC 8001
> chip
> +		based host adapters.

Not really enough here to justify a separate Kconfig file.  How about
putting this bit in drivers/scsi/Kconfig instead of in its own file?

> diff --git a/drivers/scsi/pm8001/Makefile b/drivers/scsi/pm8001/Makefile
> new file mode 100644
> index 0000000..6e668ae
> --- /dev/null
> +++ b/drivers/scsi/pm8001/Makefile
> @@ -0,0 +1,47 @@
> +#
> +# Kernel configuration file for the PM8001 SAS/SATA 8x6G based HBA driver
> +#
> +# Copyright (C) 2008-2009  USI Co., Ltd.
> +# Copyright (C) 2009 Jack Wang <jack_wang@xxxxxxxxx>
> +
> +# 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 the Free Software Foundation; either version 2
> +# of the License, or (at your option) any later version.
> +
> +# This program is distributed in the hope that it will be useful,
> +# but WITHOUT ANY WARRANTY; without even the implied warranty of
> +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
> +# GNU General Public License for more details.
> +
> +# NO WARRANTY
> +# THE PROGRAM IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OR
> +# CONDITIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED INCLUDING, WITHOUT
> +# LIMITATION, ANY WARRANTIES OR CONDITIONS OF TITLE, NON-INFRINGEMENT,
> +# MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. Each Recipient is
> +# solely responsible for determining the appropriateness of using and
> +# distributing the Program and assumes all risks associated with its
> +# exercise of rights under this Agreement, including but not limited to
> +# the risks and costs of program errors, damage to or loss of data,
> +# programs or equipment, and unavailability or interruption of operations.
> +
> +# DISCLAIMER OF LIABILITY
> +# NEITHER RECIPIENT NOR ANY CONTRIBUTORS SHALL HAVE ANY LIABILITY FOR ANY
> +# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
> +# DAMAGES (INCLUDING WITHOUT LIMITATION LOST PROFITS), HOWEVER CAUSED AND
> +# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
> +# TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
> +# USE OR DISTRIBUTION OF THE PROGRAM OR THE EXERCISE OF ANY RIGHTS GRANTED
> +# HEREUNDER, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGES
> +
> +# You should have received a copy of the GNU General Public License
> +# along with this program; if not, write to the Free Software
> +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
> 02110-1301,
> +# USA.

I'm not sure you need a copyright notice in the Makefile.  I don't think
what you have here is even copyrightable:

> +obj-$(CONFIG_SCSI_PM8001) += pm8001.o
> +pm8001-y += pm8001_init.o \
> +	     pm8001_sas.o  \
> +	     pm8001_ctl.o  \
> +	     pm8001_hwi.o
> \ No newline at end of file

And you should look into putting a newline at the end of files.

> +++ b/drivers/scsi/pm8001/pm8001_chips.h
> +#define READ_LE_32(ADDR32, DMA_ADDR, OFFSET)	\
> +	(*((u32 *)ADDR32)) = (*((u32 *)(((u8 *)DMA_ADDR) + (OFFSET))))
> +
> +#define WRITE_LE_32(DMA_ADDR, OFFSET, VALUE32)	\
> +	(*((u32 *)(((u8 *)DMA_ADDR)+(OFFSET)))) = (u32)(VALUE32)
> +
> +#define READ_LE_16(ADDR16, DMA_ADDR, OFFSET)	\
> +	(*((u16 *)ADDR16)) = (*((u16 *)(((u8 *)DMA_ADDR) + (OFFSET))))

Don't these macros assume that the processor is little-endian?  Have you
tested this driver on a big-endian platform, like powerpc, for example?

> +/**
> + * pm8001_ctl_verify_adapter - validates chip_number passed from
> application
> + * @ppm8001_ha: per adapter object
> + * @chip_number: chip id.
> + *
> + * Return (-1) means error, else chip_number.
> + */
> +static int
> +pm8001_ctl_verify_adapter(int chip_number, struct pm8001_hba_info
> **ppm8001_ha)
> +{
> +	struct pm8001_hba_info *pm8001_ha;
> +	printk(KERN_INFO "chip_number = %d \n", chip_number);
> +
> +	list_for_each_entry(pm8001_ha, &hba_list, list) {
> +		printk(KERN_INFO "pm8001_ha->id = %x \n", pm8001_ha->id);

Is this a debugging printk left in, or do you really intend this to be
emitted to the kernel log?

> +		if (pm8001_ha->id != chip_number)
> +			continue;
> +		*ppm8001_ha = pm8001_ha;
> +		return chip_number;
> +	}
> +	*ppm8001_ha = NULL;
> +	return -1;
> +}
> +
> +/**
> + * pm8001_ctl_ioctl_main - main ioctl entry point
> + * @file - (struct file)
> + * @cmd - ioctl opcode
> + * @arg -
> + */
> +static long
> +pm8001_ctl_ioctl_main(struct file *file, unsigned int cmd, void __user
> *arg)
> +{
> +	long ret = 0;
> +	u16 arg_length;
> +	struct pm8001_ioctl_payload __user *payload = (void __user *) arg;
> +	struct pm8001_ioctl_payload	*ioctl_payload ;
> +	DECLARE_COMPLETION_ONSTACK(completion);
> +	struct pm8001_hba_info *pm8001_ha = NULL;
> +	arg_length = sizeof(struct pm8001_ioctl_payload) + payload->length;
> +	ioctl_payload = kmalloc(arg_length, GFP_KERNEL);
> +
> +	if (copy_from_user(ioctl_payload, payload, arg_length)) {
> +		pm8001_printk("copy data from user space @ %p failed\n",
> +		payload);
> +		return -EFAULT;
> +	}
> +
> +	if (pm8001_ctl_verify_adapter(ioctl_payload->id, &pm8001_ha) == -1
> ||
> +		!pm8001_ha) {
> +		pm8001_printk("find the chip fail \n");
> +		return -ENODEV;
> +	}
> +
> +	pm8001_ha->nvmd_completion = &completion;
> +	switch (cmd) {
> +	case IOCTL_GET_EVENT_LOG1:
> +		PM8001_IOCTL_DBG(pm8001_ha,
> +			pm8001_printk("cmd:IOCTL_GET_EVENT_LOG1 \n"));
> +		ioctl_payload->status = IOCTL_STATUS_OK;
> +		if ((4096 + 32) < ioctl_payload->offset) {
> +			ioctl_payload->status = IOCTL_STATUS_NO_MORE_DATA;
> +			ioctl_payload->length = 0;
> +			ret = IOCTL_CALL_FAIL;
> +		}
> +		memcpy(ioctl_payload->func_specific,
> +		pm8001_ha->memoryMap.region[AAP1].virt_ptr +
> +		ioctl_payload->offset, ioctl_payload->length);
> +		if (copy_to_user((void *)arg, (void *)ioctl_payload,
> +			arg_length)) {
> +			pm8001_printk("copy to user ERROR\n");
> +			ret = -EFAULT;
> +		}
> +		kfree(ioctl_payload);
> +		return ret;
> +	case IOCTL_GET_EVENT_LOG2:
> +		PM8001_IOCTL_DBG(pm8001_ha,
> +			pm8001_printk(" cmd: IOCTL_GET_EVENT_LOG2 \n"));
> +		ioctl_payload->status = IOCTL_STATUS_OK;
> +		if ((4096+32) < ioctl_payload->offset) {
> +			ioctl_payload->status = IOCTL_STATUS_NO_MORE_DATA;
> +			ioctl_payload->length = 0;
> +			ret = IOCTL_CALL_FAIL;
> +		}
> +		memcpy(ioctl_payload->func_specific,
> +			(u8 *)pm8001_ha->memoryMap.region[IOP].virt_ptr,
> +			ioctl_payload->length);
> +		if (copy_to_user((void *)arg,
> +			(void *)ioctl_payload, arg_length)) {
> +			pm8001_printk("copy to user ERROR\n");
> +			ret = -EFAULT;
> +		}
> +		kfree(ioctl_payload);
> +		return ret;
> +	case IOCTL_FW_CONTROL:
> +		PM8001_IOCTL_DBG(pm8001_ha,
> +			pm8001_printk(" cmd: IOCTL_FW_CONTROL \n"));
> +		ret = PM8001_CHIP_DISP->fw_flash_update_req(pm8001_ha,
> +			0, ioctl_payload);
> +		break;
> +	case IOCTL_NVMD_GET:
> +		PM8001_IOCTL_DBG(pm8001_ha,
> +			pm8001_printk(" cmd: IOCTL_NVMD_GET \n"));
> +		ret = PM8001_CHIP_DISP->get_nvmd_req(pm8001_ha,
> +			0, ioctl_payload);
> +		break;
> +	case IOCTL_NVMD_SET:
> +		PM8001_IOCTL_DBG(pm8001_ha,
> +			pm8001_printk(" cmd: IOCTL_NVMD_SET \n"));
> +		ret = PM8001_CHIP_DISP->set_nvmd_req(pm8001_ha,
> +			0, ioctl_payload);
> +		break;
> +	default:
> +		return IOCTL_CALL_INVALID_CODE;
> +	}
> +	wait_for_completion(&completion);
> +	if (copy_to_user((void *)arg, (void *)ioctl_payload, arg_length)) {
> +		printk(KERN_INFO "copy to user ERROR\n");
> +		ret = -EFAULT;
> +	}
> +	kfree(ioctl_payload);
> +	return ret;
> +}
> +
> +/**
> + * pm8001_ctl_ioctl - main ioctl entry point (unlocked)
> + * @file - (struct file)
> + * @cmd - ioctl opcode
> + * @arg -
> + */
> +static long
> +pm8001_ctl_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
> +{
> +	long ret;
> +	lock_kernel();
> +	ret = pm8001_ctl_ioctl_main(file, cmd, (void __user *)arg);
> +	unlock_kernel();
> +	return ret;
> +}

Umm ... you probably don't need lock_kernel here.  And ioctls are
generally discouraged.  We've been encouraging people to use SG_IO to
send commands down like this.

> +++ b/drivers/scsi/pm8001/pm8001_defs.h
> +#define PCI_VENDOR_ID_PMC_Sierra	0x11f8
> +#define PCI_DEVICE_ID_8001		0x8001

It should be in all caps ... and it should be in include/linux/pci_ids.h,
between PCI_VENDOR_ID_COMPEX and PCI_VENDOR_ID_RP.

There's undoubtably more things to comment on ...

-- 
Matthew Wilcox				Intel Open Source Technology Centre
"Bill, look, we understand that you're interested in selling us this
operating system, but compare it to ours.  We can't possibly take such
a retrograde step."
--
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