2006-03-30 06:23:28 +07:00
|
|
|
/*
|
2008-04-17 11:09:32 +07:00
|
|
|
* Copyright (c) 2006, 2007, 2008 QLogic Corporation. All rights reserved.
|
2006-03-30 06:23:28 +07:00
|
|
|
* Copyright (c) 2003, 2004, 2005, 2006 PathScale, Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* This software is available to you under a choice of one of two
|
|
|
|
* licenses. You may choose to be licensed under the terms of the GNU
|
|
|
|
* General Public License (GPL) Version 2, available from the file
|
|
|
|
* COPYING in the main directory of this source tree, or the
|
|
|
|
* OpenIB.org BSD license below:
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or
|
|
|
|
* without modification, are permitted provided that the following
|
|
|
|
* conditions are met:
|
|
|
|
*
|
|
|
|
* - Redistributions of source code must retain the above
|
|
|
|
* copyright notice, this list of conditions and the following
|
|
|
|
* disclaimer.
|
|
|
|
*
|
|
|
|
* - Redistributions in binary form must reproduce the above
|
|
|
|
* copyright notice, this list of conditions and the following
|
|
|
|
* disclaimer in the documentation and/or other materials
|
|
|
|
* provided with the distribution.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/pci.h>
|
|
|
|
#include <linux/netdevice.h>
|
2011-08-30 23:32:52 +07:00
|
|
|
#include <linux/moduleparam.h>
|
include cleanup: Update gfp.h and slab.h includes to prepare for breaking implicit slab.h inclusion from percpu.h
percpu.h is included by sched.h and module.h and thus ends up being
included when building most .c files. percpu.h includes slab.h which
in turn includes gfp.h making everything defined by the two files
universally available and complicating inclusion dependencies.
percpu.h -> slab.h dependency is about to be removed. Prepare for
this change by updating users of gfp and slab facilities include those
headers directly instead of assuming availability. As this conversion
needs to touch large number of source files, the following script is
used as the basis of conversion.
http://userweb.kernel.org/~tj/misc/slabh-sweep.py
The script does the followings.
* Scan files for gfp and slab usages and update includes such that
only the necessary includes are there. ie. if only gfp is used,
gfp.h, if slab is used, slab.h.
* When the script inserts a new include, it looks at the include
blocks and try to put the new include such that its order conforms
to its surrounding. It's put in the include block which contains
core kernel includes, in the same order that the rest are ordered -
alphabetical, Christmas tree, rev-Xmas-tree or at the end if there
doesn't seem to be any matching order.
* If the script can't find a place to put a new include (mostly
because the file doesn't have fitting include block), it prints out
an error message indicating which .h file needs to be added to the
file.
The conversion was done in the following steps.
1. The initial automatic conversion of all .c files updated slightly
over 4000 files, deleting around 700 includes and adding ~480 gfp.h
and ~3000 slab.h inclusions. The script emitted errors for ~400
files.
2. Each error was manually checked. Some didn't need the inclusion,
some needed manual addition while adding it to implementation .h or
embedding .c file was more appropriate for others. This step added
inclusions to around 150 files.
3. The script was run again and the output was compared to the edits
from #2 to make sure no file was left behind.
4. Several build tests were done and a couple of problems were fixed.
e.g. lib/decompress_*.c used malloc/free() wrappers around slab
APIs requiring slab.h to be added manually.
5. The script was run on all .h files but without automatically
editing them as sprinkling gfp.h and slab.h inclusions around .h
files could easily lead to inclusion dependency hell. Most gfp.h
inclusion directives were ignored as stuff from gfp.h was usually
wildly available and often used in preprocessor macros. Each
slab.h inclusion directive was examined and added manually as
necessary.
6. percpu.h was updated not to include slab.h.
7. Build test were done on the following configurations and failures
were fixed. CONFIG_GCOV_KERNEL was turned off for all tests (as my
distributed build env didn't work with gcov compiles) and a few
more options had to be turned off depending on archs to make things
build (like ipr on powerpc/64 which failed due to missing writeq).
* x86 and x86_64 UP and SMP allmodconfig and a custom test config.
* powerpc and powerpc64 SMP allmodconfig
* sparc and sparc64 SMP allmodconfig
* ia64 SMP allmodconfig
* s390 SMP allmodconfig
* alpha SMP allmodconfig
* um on x86_64 SMP allmodconfig
8. percpu.h modifications were reverted so that it could be applied as
a separate patch and serve as bisection point.
Given the fact that I had only a couple of failures from tests on step
6, I'm fairly confident about the coverage of this conversion patch.
If there is a breakage, it's likely to be something in one of the arch
headers which should be easily discoverable easily on most builds of
the specific arch.
Signed-off-by: Tejun Heo <tj@kernel.org>
Guess-its-ok-by: Christoph Lameter <cl@linux-foundation.org>
Cc: Ingo Molnar <mingo@redhat.com>
Cc: Lee Schermerhorn <Lee.Schermerhorn@hp.com>
2010-03-24 15:04:11 +07:00
|
|
|
#include <linux/slab.h>
|
2011-05-28 00:27:45 +07:00
|
|
|
#include <linux/stat.h>
|
2006-03-30 06:23:28 +07:00
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
|
|
|
|
#include "ipath_kernel.h"
|
2006-07-01 18:36:17 +07:00
|
|
|
#include "ipath_common.h"
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* min buffers we want to have per port, after driver
|
|
|
|
*/
|
2008-05-08 01:00:15 +07:00
|
|
|
#define IPATH_MIN_USER_PORT_BUFCNT 7
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Number of ports we are configured to use (to allow for more pio
|
|
|
|
* buffers per port, etc.) Zero means use chip value.
|
|
|
|
*/
|
|
|
|
static ushort ipath_cfgports;
|
|
|
|
|
|
|
|
module_param_named(cfgports, ipath_cfgports, ushort, S_IRUGO);
|
|
|
|
MODULE_PARM_DESC(cfgports, "Set max number of ports to use");
|
|
|
|
|
|
|
|
/*
|
2006-08-26 01:24:34 +07:00
|
|
|
* Number of buffers reserved for driver (verbs and layered drivers.)
|
2008-05-08 01:00:15 +07:00
|
|
|
* Initialized based on number of PIO buffers if not set via module interface.
|
2006-04-25 04:23:00 +07:00
|
|
|
* The problem with this is that it's global, but we'll use different
|
2008-05-08 01:00:15 +07:00
|
|
|
* numbers for different chip types.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
2006-04-25 04:23:00 +07:00
|
|
|
static ushort ipath_kpiobufs;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
static int ipath_set_kpiobufs(const char *val, struct kernel_param *kp);
|
|
|
|
|
2006-04-25 04:23:00 +07:00
|
|
|
module_param_call(kpiobufs, ipath_set_kpiobufs, param_get_ushort,
|
2006-03-30 06:23:28 +07:00
|
|
|
&ipath_kpiobufs, S_IWUSR | S_IRUGO);
|
|
|
|
MODULE_PARM_DESC(kpiobufs, "Set number of PIO buffers for driver");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* create_port0_egr - allocate the eager TID buffers
|
|
|
|
* @dd: the infinipath device
|
|
|
|
*
|
|
|
|
* This code is now quite different for user and kernel, because
|
|
|
|
* the kernel uses skb's, for the accelerated network performance.
|
|
|
|
* This is the kernel (port0) version.
|
|
|
|
*
|
|
|
|
* Allocate the eager TID buffers and program them into infinipath.
|
|
|
|
* We use the network layer alloc_skb() allocator to allocate the
|
2006-08-26 01:24:34 +07:00
|
|
|
* memory, and either use the buffers as is for things like verbs
|
2006-03-30 06:23:28 +07:00
|
|
|
* packets, or pass the buffers up to the ipath layered driver and
|
|
|
|
* thence the network layer, replacing them as we do so (see
|
|
|
|
* ipath_rcv_layer()).
|
|
|
|
*/
|
|
|
|
static int create_port0_egr(struct ipath_devdata *dd)
|
|
|
|
{
|
|
|
|
unsigned e, egrcnt;
|
2006-09-28 23:00:13 +07:00
|
|
|
struct ipath_skbinfo *skbinfo;
|
2006-03-30 06:23:28 +07:00
|
|
|
int ret;
|
|
|
|
|
2008-01-07 12:02:34 +07:00
|
|
|
egrcnt = dd->ipath_p0_rcvegrcnt;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2006-09-28 23:00:13 +07:00
|
|
|
skbinfo = vmalloc(sizeof(*dd->ipath_port0_skbinfo) * egrcnt);
|
|
|
|
if (skbinfo == NULL) {
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_dev_err(dd, "allocation error for eager TID "
|
|
|
|
"skb array\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
for (e = 0; e < egrcnt; e++) {
|
|
|
|
/*
|
|
|
|
* This is a bit tricky in that we allocate extra
|
|
|
|
* space for 2 bytes of the 14 byte ethernet header.
|
|
|
|
* These two bytes are passed in the ipath header so
|
|
|
|
* the rest of the data is word aligned. We allocate
|
|
|
|
* 4 bytes so that the data buffer stays word aligned.
|
|
|
|
* See ipath_kreceive() for more details.
|
|
|
|
*/
|
2006-09-28 23:00:13 +07:00
|
|
|
skbinfo[e].skb = ipath_alloc_skb(dd, GFP_KERNEL);
|
|
|
|
if (!skbinfo[e].skb) {
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_dev_err(dd, "SKB allocation error for "
|
|
|
|
"eager TID %u\n", e);
|
|
|
|
while (e != 0)
|
2006-09-28 23:00:13 +07:00
|
|
|
dev_kfree_skb(skbinfo[--e].skb);
|
|
|
|
vfree(skbinfo);
|
2006-03-30 06:23:28 +07:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* After loop above, so we can test non-NULL to see if ready
|
|
|
|
* to use at receive, etc.
|
|
|
|
*/
|
2006-09-28 23:00:13 +07:00
|
|
|
dd->ipath_port0_skbinfo = skbinfo;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
for (e = 0; e < egrcnt; e++) {
|
2006-09-28 23:00:13 +07:00
|
|
|
dd->ipath_port0_skbinfo[e].phys =
|
|
|
|
ipath_map_single(dd->pcidev,
|
|
|
|
dd->ipath_port0_skbinfo[e].skb->data,
|
|
|
|
dd->ipath_ibmaxlen, PCI_DMA_FROMDEVICE);
|
2006-03-30 06:23:28 +07:00
|
|
|
dd->ipath_f_put_tid(dd, e + (u64 __iomem *)
|
|
|
|
((char __iomem *) dd->ipath_kregbase +
|
2007-06-19 04:24:39 +07:00
|
|
|
dd->ipath_rcvegrbase),
|
|
|
|
RCVHQ_RCV_TYPE_EAGER,
|
2006-09-28 23:00:13 +07:00
|
|
|
dd->ipath_port0_skbinfo[e].phys);
|
2006-03-30 06:23:28 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
bail:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int bringup_link(struct ipath_devdata *dd)
|
|
|
|
{
|
|
|
|
u64 val, ibc;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/* hold IBC in reset */
|
|
|
|
dd->ipath_control &= ~INFINIPATH_C_LINKENABLE;
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_control,
|
|
|
|
dd->ipath_control);
|
|
|
|
|
|
|
|
/*
|
2008-04-17 11:01:12 +07:00
|
|
|
* set initial max size pkt IBC will send, including ICRC; it's the
|
|
|
|
* PIO buffer size in dwords, less 1; also see ipath_set_mtu()
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
2008-04-17 11:01:12 +07:00
|
|
|
val = (dd->ipath_ibmaxlen >> 2) + 1;
|
|
|
|
ibc = val << dd->ibcc_mpl_shift;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2008-04-17 11:01:12 +07:00
|
|
|
/* flowcontrolwatermark is in units of KBytes */
|
2006-03-30 06:23:28 +07:00
|
|
|
ibc |= 0x5ULL << INFINIPATH_IBCC_FLOWCTRLWATERMARK_SHIFT;
|
|
|
|
/*
|
|
|
|
* How often flowctrl sent. More or less in usecs; balance against
|
|
|
|
* watermark value, so that in theory senders always get a flow
|
|
|
|
* control update in time to not let the IB link go idle.
|
|
|
|
*/
|
|
|
|
ibc |= 0x3ULL << INFINIPATH_IBCC_FLOWCTRLPERIOD_SHIFT;
|
|
|
|
/* max error tolerance */
|
|
|
|
ibc |= 0xfULL << INFINIPATH_IBCC_PHYERRTHRESHOLD_SHIFT;
|
|
|
|
/* use "real" buffer space for */
|
|
|
|
ibc |= 4ULL << INFINIPATH_IBCC_CREDITSCALE_SHIFT;
|
|
|
|
/* IB credit flow control. */
|
|
|
|
ibc |= 0xfULL << INFINIPATH_IBCC_OVERRUNTHRESHOLD_SHIFT;
|
|
|
|
/* initially come up waiting for TS1, without sending anything. */
|
|
|
|
dd->ipath_ibcctrl = ibc;
|
|
|
|
/*
|
|
|
|
* Want to start out with both LINKCMD and LINKINITCMD in NOP
|
|
|
|
* (0 and 0). Don't put linkinitcmd in ipath_ibcctrl, want that
|
2008-04-17 11:09:26 +07:00
|
|
|
* to stay a NOP. Flag that we are disabled, for the (unlikely)
|
|
|
|
* case that some recovery path is trying to bring the link up
|
|
|
|
* before we are ready.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
|
|
|
ibc |= INFINIPATH_IBCC_LINKINITCMD_DISABLE <<
|
|
|
|
INFINIPATH_IBCC_LINKINITCMD_SHIFT;
|
2008-04-17 11:09:26 +07:00
|
|
|
dd->ipath_flags |= IPATH_IB_LINK_DISABLED;
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_cdbg(VERBOSE, "Writing 0x%llx to ibcctrl\n",
|
|
|
|
(unsigned long long) ibc);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_ibcctrl, ibc);
|
|
|
|
|
|
|
|
// be sure chip saw it
|
|
|
|
val = ipath_read_kreg64(dd, dd->ipath_kregs->kr_scratch);
|
|
|
|
|
|
|
|
ret = dd->ipath_f_bringup_serdes(dd);
|
|
|
|
|
|
|
|
if (ret)
|
|
|
|
dev_info(&dd->pcidev->dev, "Could not initialize SerDes, "
|
|
|
|
"not usable\n");
|
|
|
|
else {
|
|
|
|
/* enable IBC */
|
|
|
|
dd->ipath_control |= INFINIPATH_C_LINKENABLE;
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_control,
|
|
|
|
dd->ipath_control);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2007-03-16 04:45:08 +07:00
|
|
|
static struct ipath_portdata *create_portdata0(struct ipath_devdata *dd)
|
|
|
|
{
|
|
|
|
struct ipath_portdata *pd = NULL;
|
|
|
|
|
|
|
|
pd = kzalloc(sizeof(*pd), GFP_KERNEL);
|
|
|
|
if (pd) {
|
|
|
|
pd->port_dd = dd;
|
|
|
|
pd->port_cnt = 1;
|
|
|
|
/* The port 0 pkey table is used by the layer interface. */
|
|
|
|
pd->port_pkeys[0] = IPATH_DEFAULT_P_KEY;
|
2008-04-17 11:09:29 +07:00
|
|
|
pd->port_seq_cnt = 1;
|
2007-03-16 04:45:08 +07:00
|
|
|
}
|
|
|
|
return pd;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
static int init_chip_first(struct ipath_devdata *dd)
|
2006-03-30 06:23:28 +07:00
|
|
|
{
|
2008-04-17 11:09:29 +07:00
|
|
|
struct ipath_portdata *pd;
|
2006-03-30 06:23:28 +07:00
|
|
|
int ret = 0;
|
|
|
|
u64 val;
|
|
|
|
|
2008-04-17 11:09:31 +07:00
|
|
|
spin_lock_init(&dd->ipath_kernel_tid_lock);
|
|
|
|
spin_lock_init(&dd->ipath_user_tid_lock);
|
|
|
|
spin_lock_init(&dd->ipath_sendctrl_lock);
|
2008-12-06 02:14:38 +07:00
|
|
|
spin_lock_init(&dd->ipath_uctxt_lock);
|
2008-04-17 11:09:31 +07:00
|
|
|
spin_lock_init(&dd->ipath_sdma_lock);
|
|
|
|
spin_lock_init(&dd->ipath_gpio_lock);
|
|
|
|
spin_lock_init(&dd->ipath_eep_st_lock);
|
|
|
|
spin_lock_init(&dd->ipath_sdepb_lock);
|
|
|
|
mutex_init(&dd->ipath_eep_lock);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* skip cfgports stuff because we are not allocating memory,
|
|
|
|
* and we don't want problems if the portcnt changed due to
|
|
|
|
* cfgports. We do still check and report a difference, if
|
|
|
|
* not same (should be impossible).
|
|
|
|
*/
|
2008-01-07 12:02:34 +07:00
|
|
|
dd->ipath_f_config_ports(dd, ipath_cfgports);
|
2006-03-30 06:23:28 +07:00
|
|
|
if (!ipath_cfgports)
|
|
|
|
dd->ipath_cfgports = dd->ipath_portcnt;
|
|
|
|
else if (ipath_cfgports <= dd->ipath_portcnt) {
|
|
|
|
dd->ipath_cfgports = ipath_cfgports;
|
|
|
|
ipath_dbg("Configured to use %u ports out of %u in chip\n",
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_cfgports, ipath_read_kreg32(dd,
|
|
|
|
dd->ipath_kregs->kr_portcnt));
|
2006-03-30 06:23:28 +07:00
|
|
|
} else {
|
|
|
|
dd->ipath_cfgports = dd->ipath_portcnt;
|
|
|
|
ipath_dbg("Tried to configured to use %u ports; chip "
|
|
|
|
"only supports %u\n", ipath_cfgports,
|
2008-04-17 11:09:29 +07:00
|
|
|
ipath_read_kreg32(dd,
|
|
|
|
dd->ipath_kregs->kr_portcnt));
|
2006-03-30 06:23:28 +07:00
|
|
|
}
|
2006-08-26 01:24:28 +07:00
|
|
|
/*
|
|
|
|
* Allocate full portcnt array, rather than just cfgports, because
|
|
|
|
* cleanup iterates across all possible ports.
|
|
|
|
*/
|
|
|
|
dd->ipath_pd = kzalloc(sizeof(*dd->ipath_pd) * dd->ipath_portcnt,
|
2006-03-30 06:23:28 +07:00
|
|
|
GFP_KERNEL);
|
|
|
|
|
|
|
|
if (!dd->ipath_pd) {
|
|
|
|
ipath_dev_err(dd, "Unable to allocate portdata array, "
|
|
|
|
"failing\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2007-03-16 04:45:08 +07:00
|
|
|
pd = create_portdata0(dd);
|
|
|
|
if (!pd) {
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_dev_err(dd, "Unable to allocate portdata for port "
|
|
|
|
"0, failing\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto done;
|
|
|
|
}
|
2007-03-16 04:45:08 +07:00
|
|
|
dd->ipath_pd[0] = pd;
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
dd->ipath_rcvtidcnt =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvtidcnt);
|
|
|
|
dd->ipath_rcvtidbase =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvtidbase);
|
|
|
|
dd->ipath_rcvegrcnt =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvegrcnt);
|
|
|
|
dd->ipath_rcvegrbase =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvegrbase);
|
|
|
|
dd->ipath_palign =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_pagealign);
|
|
|
|
dd->ipath_piobufbase =
|
|
|
|
ipath_read_kreg64(dd, dd->ipath_kregs->kr_sendpiobufbase);
|
|
|
|
val = ipath_read_kreg64(dd, dd->ipath_kregs->kr_sendpiosize);
|
|
|
|
dd->ipath_piosize2k = val & ~0U;
|
|
|
|
dd->ipath_piosize4k = val >> 32;
|
2008-04-17 11:01:12 +07:00
|
|
|
if (dd->ipath_piosize4k == 0 && ipath_mtu4096)
|
|
|
|
ipath_mtu4096 = 0; /* 4KB not supported by this chip */
|
|
|
|
dd->ipath_ibmtu = ipath_mtu4096 ? 4096 : 2048;
|
2006-03-30 06:23:28 +07:00
|
|
|
val = ipath_read_kreg64(dd, dd->ipath_kregs->kr_sendpiobufcnt);
|
|
|
|
dd->ipath_piobcnt2k = val & ~0U;
|
|
|
|
dd->ipath_piobcnt4k = val >> 32;
|
|
|
|
dd->ipath_pio2kbase =
|
|
|
|
(u32 __iomem *) (((char __iomem *) dd->ipath_kregbase) +
|
|
|
|
(dd->ipath_piobufbase & 0xffffffff));
|
|
|
|
if (dd->ipath_piobcnt4k) {
|
|
|
|
dd->ipath_pio4kbase = (u32 __iomem *)
|
|
|
|
(((char __iomem *) dd->ipath_kregbase) +
|
|
|
|
(dd->ipath_piobufbase >> 32));
|
|
|
|
/*
|
|
|
|
* 4K buffers take 2 pages; we use roundup just to be
|
|
|
|
* paranoid; we calculate it once here, rather than on
|
|
|
|
* ever buf allocate
|
|
|
|
*/
|
|
|
|
dd->ipath_4kalign = ALIGN(dd->ipath_piosize4k,
|
|
|
|
dd->ipath_palign);
|
|
|
|
ipath_dbg("%u 2k(%x) piobufs @ %p, %u 4k(%x) @ %p "
|
|
|
|
"(%x aligned)\n",
|
|
|
|
dd->ipath_piobcnt2k, dd->ipath_piosize2k,
|
|
|
|
dd->ipath_pio2kbase, dd->ipath_piobcnt4k,
|
|
|
|
dd->ipath_piosize4k, dd->ipath_pio4kbase,
|
|
|
|
dd->ipath_4kalign);
|
|
|
|
}
|
|
|
|
else ipath_dbg("%u 2k piobufs @ %p\n",
|
|
|
|
dd->ipath_piobcnt2k, dd->ipath_pio2kbase);
|
|
|
|
|
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* init_chip_reset - re-initialize after a reset, or enable
|
|
|
|
* @dd: the infinipath device
|
|
|
|
*
|
|
|
|
* sanity check at least some of the values after reset, and
|
2011-03-31 08:57:33 +07:00
|
|
|
* ensure no receive or transmit (explicitly, in case reset
|
2006-03-30 06:23:28 +07:00
|
|
|
* failed
|
|
|
|
*/
|
2008-04-17 11:09:29 +07:00
|
|
|
static int init_chip_reset(struct ipath_devdata *dd)
|
2006-03-30 06:23:28 +07:00
|
|
|
{
|
|
|
|
u32 rtmp;
|
2008-04-17 11:09:29 +07:00
|
|
|
int i;
|
2008-04-17 11:09:30 +07:00
|
|
|
unsigned long flags;
|
2008-04-17 11:09:29 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* ensure chip does no sends or receives, tail updates, or
|
|
|
|
* pioavail updates while we re-initialize
|
|
|
|
*/
|
|
|
|
dd->ipath_rcvctrl &= ~(1ULL << dd->ipath_r_tailupd_shift);
|
|
|
|
for (i = 0; i < dd->ipath_portcnt; i++) {
|
|
|
|
clear_bit(dd->ipath_r_portenable_shift + i,
|
|
|
|
&dd->ipath_rcvctrl);
|
|
|
|
clear_bit(dd->ipath_r_intravail_shift + i,
|
|
|
|
&dd->ipath_rcvctrl);
|
|
|
|
}
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvctrl,
|
|
|
|
dd->ipath_rcvctrl);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2008-04-17 11:09:30 +07:00
|
|
|
spin_lock_irqsave(&dd->ipath_sendctrl_lock, flags);
|
|
|
|
dd->ipath_sendctrl = 0U; /* no sdma, etc */
|
2007-09-05 15:57:14 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_sendctrl, dd->ipath_sendctrl);
|
2008-04-17 11:09:30 +07:00
|
|
|
ipath_read_kreg64(dd, dd->ipath_kregs->kr_scratch);
|
|
|
|
spin_unlock_irqrestore(&dd->ipath_sendctrl_lock, flags);
|
|
|
|
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_control, 0ULL);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
rtmp = ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvtidcnt);
|
|
|
|
if (rtmp != dd->ipath_rcvtidcnt)
|
|
|
|
dev_info(&dd->pcidev->dev, "tidcnt was %u before "
|
|
|
|
"reset, now %u, using original\n",
|
|
|
|
dd->ipath_rcvtidcnt, rtmp);
|
|
|
|
rtmp = ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvtidbase);
|
|
|
|
if (rtmp != dd->ipath_rcvtidbase)
|
|
|
|
dev_info(&dd->pcidev->dev, "tidbase was %u before "
|
|
|
|
"reset, now %u, using original\n",
|
|
|
|
dd->ipath_rcvtidbase, rtmp);
|
|
|
|
rtmp = ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvegrcnt);
|
|
|
|
if (rtmp != dd->ipath_rcvegrcnt)
|
|
|
|
dev_info(&dd->pcidev->dev, "egrcnt was %u before "
|
|
|
|
"reset, now %u, using original\n",
|
|
|
|
dd->ipath_rcvegrcnt, rtmp);
|
|
|
|
rtmp = ipath_read_kreg32(dd, dd->ipath_kregs->kr_rcvegrbase);
|
|
|
|
if (rtmp != dd->ipath_rcvegrbase)
|
|
|
|
dev_info(&dd->pcidev->dev, "egrbase was %u before "
|
|
|
|
"reset, now %u, using original\n",
|
|
|
|
dd->ipath_rcvegrbase, rtmp);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int init_pioavailregs(struct ipath_devdata *dd)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
dd->ipath_pioavailregs_dma = dma_alloc_coherent(
|
|
|
|
&dd->pcidev->dev, PAGE_SIZE, &dd->ipath_pioavailregs_phys,
|
|
|
|
GFP_KERNEL);
|
|
|
|
if (!dd->ipath_pioavailregs_dma) {
|
|
|
|
ipath_dev_err(dd, "failed to allocate PIOavail reg area "
|
|
|
|
"in memory\n");
|
|
|
|
ret = -ENOMEM;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* we really want L2 cache aligned, but for current CPUs of
|
|
|
|
* interest, they are the same.
|
|
|
|
*/
|
|
|
|
dd->ipath_statusp = (u64 *)
|
|
|
|
((char *)dd->ipath_pioavailregs_dma +
|
|
|
|
((2 * L1_CACHE_BYTES +
|
|
|
|
dd->ipath_pioavregs * sizeof(u64)) & ~L1_CACHE_BYTES));
|
|
|
|
/* copy the current value now that it's really allocated */
|
|
|
|
*dd->ipath_statusp = dd->_ipath_status;
|
|
|
|
/*
|
|
|
|
* setup buffer to hold freeze msg, accessible to apps,
|
|
|
|
* following statusp
|
|
|
|
*/
|
|
|
|
dd->ipath_freezemsg = (char *)&dd->ipath_statusp[1];
|
|
|
|
/* and its length */
|
|
|
|
dd->ipath_freezelen = L1_CACHE_BYTES - sizeof(dd->ipath_statusp[0]);
|
|
|
|
|
2006-07-01 18:36:03 +07:00
|
|
|
ret = 0;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* init_shadow_tids - allocate the shadow TID array
|
|
|
|
* @dd: the infinipath device
|
|
|
|
*
|
|
|
|
* allocate the shadow TID array, so we can ipath_munlock previous
|
|
|
|
* entries. It may make more sense to move the pageshadow to the
|
|
|
|
* port data structure, so we only allocate memory for ports actually
|
|
|
|
* in use, since we at 8k per port, now.
|
|
|
|
*/
|
|
|
|
static void init_shadow_tids(struct ipath_devdata *dd)
|
|
|
|
{
|
2006-09-28 23:00:13 +07:00
|
|
|
struct page **pages;
|
|
|
|
dma_addr_t *addrs;
|
|
|
|
|
2010-11-05 10:07:36 +07:00
|
|
|
pages = vzalloc(dd->ipath_cfgports * dd->ipath_rcvtidcnt *
|
2006-03-30 06:23:28 +07:00
|
|
|
sizeof(struct page *));
|
2006-09-28 23:00:13 +07:00
|
|
|
if (!pages) {
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_dev_err(dd, "failed to allocate shadow page * "
|
|
|
|
"array, no expected sends!\n");
|
2006-09-28 23:00:13 +07:00
|
|
|
dd->ipath_pageshadow = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
addrs = vmalloc(dd->ipath_cfgports * dd->ipath_rcvtidcnt *
|
|
|
|
sizeof(dma_addr_t));
|
|
|
|
if (!addrs) {
|
|
|
|
ipath_dev_err(dd, "failed to allocate shadow dma handle "
|
|
|
|
"array, no expected sends!\n");
|
2009-02-23 11:04:34 +07:00
|
|
|
vfree(pages);
|
2006-09-28 23:00:13 +07:00
|
|
|
dd->ipath_pageshadow = NULL;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dd->ipath_pageshadow = pages;
|
|
|
|
dd->ipath_physshadow = addrs;
|
2006-03-30 06:23:28 +07:00
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
static void enable_chip(struct ipath_devdata *dd, int reinit)
|
2006-03-30 06:23:28 +07:00
|
|
|
{
|
|
|
|
u32 val;
|
2008-04-17 11:09:29 +07:00
|
|
|
u64 rcvmask;
|
2007-09-05 15:57:14 +07:00
|
|
|
unsigned long flags;
|
2006-03-30 06:23:28 +07:00
|
|
|
int i;
|
|
|
|
|
2006-08-26 01:24:34 +07:00
|
|
|
if (!reinit)
|
|
|
|
init_waitqueue_head(&ipath_state_wait);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvctrl,
|
|
|
|
dd->ipath_rcvctrl);
|
|
|
|
|
2007-09-05 15:57:14 +07:00
|
|
|
spin_lock_irqsave(&dd->ipath_sendctrl_lock, flags);
|
2006-03-30 06:23:28 +07:00
|
|
|
/* Enable PIO send, and update of PIOavail regs to memory. */
|
|
|
|
dd->ipath_sendctrl = INFINIPATH_S_PIOENABLE |
|
|
|
|
INFINIPATH_S_PIOBUFAVAILUPD;
|
2008-04-17 11:09:30 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the PIO avail update threshold to host memory
|
|
|
|
* on chips that support it.
|
|
|
|
*/
|
|
|
|
if (dd->ipath_pioupd_thresh)
|
|
|
|
dd->ipath_sendctrl |= dd->ipath_pioupd_thresh
|
|
|
|
<< INFINIPATH_S_UPDTHRESH_SHIFT;
|
2007-09-05 15:57:14 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_sendctrl, dd->ipath_sendctrl);
|
|
|
|
ipath_read_kreg64(dd, dd->ipath_kregs->kr_scratch);
|
|
|
|
spin_unlock_irqrestore(&dd->ipath_sendctrl_lock, flags);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/*
|
2008-04-17 11:09:29 +07:00
|
|
|
* Enable kernel ports' receive and receive interrupt.
|
|
|
|
* Other ports done as user opens and inits them.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
2008-04-17 11:09:29 +07:00
|
|
|
rcvmask = 1ULL;
|
|
|
|
dd->ipath_rcvctrl |= (rcvmask << dd->ipath_r_portenable_shift) |
|
|
|
|
(rcvmask << dd->ipath_r_intravail_shift);
|
|
|
|
if (!(dd->ipath_flags & IPATH_NODMA_RTAIL))
|
|
|
|
dd->ipath_rcvctrl |= (1ULL << dd->ipath_r_tailupd_shift);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvctrl,
|
|
|
|
dd->ipath_rcvctrl);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* now ready for use. this should be cleared whenever we
|
|
|
|
* detect a reset, or initiate one.
|
|
|
|
*/
|
|
|
|
dd->ipath_flags |= IPATH_INITTED;
|
|
|
|
|
|
|
|
/*
|
2008-04-17 11:09:29 +07:00
|
|
|
* Init our shadow copies of head from tail values,
|
|
|
|
* and write head values to match.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
|
|
|
val = ipath_read_ureg32(dd, ur_rcvegrindextail, 0);
|
2008-04-17 11:09:26 +07:00
|
|
|
ipath_write_ureg(dd, ur_rcvegrindexhead, val, 0);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/* Initialize so we interrupt on next packet received */
|
2008-04-17 11:09:26 +07:00
|
|
|
ipath_write_ureg(dd, ur_rcvhdrhead,
|
|
|
|
dd->ipath_rhdrhead_intr_off |
|
|
|
|
dd->ipath_pd[0]->port_head, 0);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* by now pioavail updates to memory should have occurred, so
|
|
|
|
* copy them into our working/shadow registers; this is in
|
|
|
|
* case something went wrong with abort, but mostly to get the
|
|
|
|
* initial values of the generation bit correct.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < dd->ipath_pioavregs; i++) {
|
2008-04-17 11:01:07 +07:00
|
|
|
__le64 pioavail;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Chip Errata bug 6641; even and odd qwords>3 are swapped.
|
|
|
|
*/
|
2008-01-07 12:12:38 +07:00
|
|
|
if (i > 3 && (dd->ipath_flags & IPATH_SWAP_PIOBUFS))
|
2008-04-17 11:01:07 +07:00
|
|
|
pioavail = dd->ipath_pioavailregs_dma[i ^ 1];
|
2006-03-30 06:23:28 +07:00
|
|
|
else
|
2008-04-17 11:01:07 +07:00
|
|
|
pioavail = dd->ipath_pioavailregs_dma[i];
|
2008-05-08 01:00:15 +07:00
|
|
|
/*
|
|
|
|
* don't need to worry about ipath_pioavailkernel here
|
|
|
|
* because we will call ipath_chg_pioavailkernel() later
|
|
|
|
* in initialization, to busy out buffers as needed
|
|
|
|
*/
|
|
|
|
dd->ipath_pioavailshadow[i] = le64_to_cpu(pioavail);
|
2006-03-30 06:23:28 +07:00
|
|
|
}
|
|
|
|
/* can get counters, stats, etc. */
|
|
|
|
dd->ipath_flags |= IPATH_PRESENT;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
static int init_housekeeping(struct ipath_devdata *dd, int reinit)
|
2006-03-30 06:23:28 +07:00
|
|
|
{
|
2008-04-17 11:09:31 +07:00
|
|
|
char boardn[40];
|
2006-03-30 06:23:28 +07:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* have to clear shadow copies of registers at init that are
|
|
|
|
* not otherwise set here, or all kinds of bizarre things
|
|
|
|
* happen with driver on chip reset
|
|
|
|
*/
|
|
|
|
dd->ipath_rcvhdrsize = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Don't clear ipath_flags as 8bit mode was set before
|
|
|
|
* entering this func. However, we do set the linkstate to
|
|
|
|
* unknown, so we can watch for a transition.
|
2006-04-25 04:23:00 +07:00
|
|
|
* PRESENT is set because we want register reads to work,
|
|
|
|
* and the kernel infrastructure saw it in config space;
|
|
|
|
* We clear it if we have failures.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
2006-04-25 04:23:00 +07:00
|
|
|
dd->ipath_flags |= IPATH_LINKUNK | IPATH_PRESENT;
|
2006-03-30 06:23:28 +07:00
|
|
|
dd->ipath_flags &= ~(IPATH_LINKACTIVE | IPATH_LINKARMED |
|
|
|
|
IPATH_LINKDOWN | IPATH_LINKINIT);
|
|
|
|
|
|
|
|
ipath_cdbg(VERBOSE, "Try to read spc chip revision\n");
|
|
|
|
dd->ipath_revision =
|
|
|
|
ipath_read_kreg64(dd, dd->ipath_kregs->kr_revision);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* set up fundamental info we need to use the chip; we assume
|
|
|
|
* if the revision reg and these regs are OK, we don't need to
|
|
|
|
* special case the rest
|
|
|
|
*/
|
|
|
|
dd->ipath_sregbase =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_sendregbase);
|
|
|
|
dd->ipath_cregbase =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_counterregbase);
|
|
|
|
dd->ipath_uregbase =
|
|
|
|
ipath_read_kreg32(dd, dd->ipath_kregs->kr_userregbase);
|
|
|
|
ipath_cdbg(VERBOSE, "ipath_kregbase %p, sendbase %x usrbase %x, "
|
|
|
|
"cntrbase %x\n", dd->ipath_kregbase, dd->ipath_sregbase,
|
|
|
|
dd->ipath_uregbase, dd->ipath_cregbase);
|
|
|
|
if ((dd->ipath_revision & 0xffffffff) == 0xffffffff
|
|
|
|
|| (dd->ipath_sregbase & 0xffffffff) == 0xffffffff
|
|
|
|
|| (dd->ipath_cregbase & 0xffffffff) == 0xffffffff
|
|
|
|
|| (dd->ipath_uregbase & 0xffffffff) == 0xffffffff) {
|
|
|
|
ipath_dev_err(dd, "Register read failures from chip, "
|
|
|
|
"giving up initialization\n");
|
2006-04-25 04:23:00 +07:00
|
|
|
dd->ipath_flags &= ~IPATH_PRESENT;
|
2006-03-30 06:23:28 +07:00
|
|
|
ret = -ENODEV;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2007-03-16 04:45:07 +07:00
|
|
|
|
|
|
|
/* clear diagctrl register, in case diags were running and crashed */
|
|
|
|
ipath_write_kreg (dd, dd->ipath_kregs->kr_hwdiagctrl, 0);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/* clear the initial reset flag, in case first driver load */
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_errorclear,
|
|
|
|
INFINIPATH_E_RESET);
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
ipath_cdbg(VERBOSE, "Revision %llx (PCI %x)\n",
|
|
|
|
(unsigned long long) dd->ipath_revision,
|
|
|
|
dd->ipath_pcirev);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
|
|
|
if (((dd->ipath_revision >> INFINIPATH_R_SOFTWARE_SHIFT) &
|
|
|
|
INFINIPATH_R_SOFTWARE_MASK) != IPATH_CHIP_SWVERSION) {
|
|
|
|
ipath_dev_err(dd, "Driver only handles version %d, "
|
|
|
|
"chip swversion is %d (%llx), failng\n",
|
|
|
|
IPATH_CHIP_SWVERSION,
|
|
|
|
(int)(dd->ipath_revision >>
|
|
|
|
INFINIPATH_R_SOFTWARE_SHIFT) &
|
|
|
|
INFINIPATH_R_SOFTWARE_MASK,
|
|
|
|
(unsigned long long) dd->ipath_revision);
|
|
|
|
ret = -ENOSYS;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
dd->ipath_majrev = (u8) ((dd->ipath_revision >>
|
|
|
|
INFINIPATH_R_CHIPREVMAJOR_SHIFT) &
|
|
|
|
INFINIPATH_R_CHIPREVMAJOR_MASK);
|
|
|
|
dd->ipath_minrev = (u8) ((dd->ipath_revision >>
|
|
|
|
INFINIPATH_R_CHIPREVMINOR_SHIFT) &
|
|
|
|
INFINIPATH_R_CHIPREVMINOR_MASK);
|
|
|
|
dd->ipath_boardrev = (u8) ((dd->ipath_revision >>
|
|
|
|
INFINIPATH_R_BOARDID_SHIFT) &
|
|
|
|
INFINIPATH_R_BOARDID_MASK);
|
|
|
|
|
|
|
|
ret = dd->ipath_f_get_boardname(dd, boardn, sizeof boardn);
|
|
|
|
|
|
|
|
snprintf(dd->ipath_boardversion, sizeof(dd->ipath_boardversion),
|
2007-07-07 02:48:43 +07:00
|
|
|
"ChipABI %u.%u, %s, InfiniPath%u %u.%u, PCI %u, "
|
2006-03-30 06:23:28 +07:00
|
|
|
"SW Compat %u\n",
|
|
|
|
IPATH_CHIP_VERS_MAJ, IPATH_CHIP_VERS_MIN, boardn,
|
|
|
|
(unsigned)(dd->ipath_revision >> INFINIPATH_R_ARCH_SHIFT) &
|
|
|
|
INFINIPATH_R_ARCH_MASK,
|
|
|
|
dd->ipath_majrev, dd->ipath_minrev, dd->ipath_pcirev,
|
|
|
|
(unsigned)(dd->ipath_revision >>
|
|
|
|
INFINIPATH_R_SOFTWARE_SHIFT) &
|
|
|
|
INFINIPATH_R_SOFTWARE_MASK);
|
|
|
|
|
|
|
|
ipath_dbg("%s", dd->ipath_boardversion);
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
if (ret)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
if (reinit)
|
|
|
|
ret = init_chip_reset(dd);
|
|
|
|
else
|
|
|
|
ret = init_chip_first(dd);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:30 +07:00
|
|
|
static void verify_interrupt(unsigned long opaque)
|
|
|
|
{
|
|
|
|
struct ipath_devdata *dd = (struct ipath_devdata *) opaque;
|
|
|
|
|
|
|
|
if (!dd)
|
|
|
|
return; /* being torn down */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we don't have any interrupts, let the user know and
|
|
|
|
* don't bother checking again.
|
|
|
|
*/
|
|
|
|
if (dd->ipath_int_counter == 0) {
|
|
|
|
if (!dd->ipath_f_intr_fallback(dd))
|
|
|
|
dev_err(&dd->pcidev->dev, "No interrupts detected, "
|
|
|
|
"not usable.\n");
|
|
|
|
else /* re-arm the timer to see if fallback works */
|
|
|
|
mod_timer(&dd->ipath_intrchk_timer, jiffies + HZ/2);
|
|
|
|
} else
|
|
|
|
ipath_cdbg(VERBOSE, "%u interrupts at timer check\n",
|
|
|
|
dd->ipath_int_counter);
|
|
|
|
}
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/**
|
|
|
|
* ipath_init_chip - do the actual initialization sequence on the chip
|
|
|
|
* @dd: the infinipath device
|
|
|
|
* @reinit: reinitializing, so don't allocate new memory
|
|
|
|
*
|
|
|
|
* Do the actual initialization sequence on the chip. This is done
|
|
|
|
* both from the init routine called from the PCI infrastructure, and
|
|
|
|
* when we reset the chip, or detect that it was reset internally,
|
|
|
|
* or it's administratively re-enabled.
|
|
|
|
*
|
|
|
|
* Memory allocation here and in called routines is only done in
|
|
|
|
* the first case (reinit == 0). We have to be careful, because even
|
|
|
|
* without memory allocation, we need to re-write all the chip registers
|
|
|
|
* TIDs, etc. after the reset or enable has completed.
|
|
|
|
*/
|
|
|
|
int ipath_init_chip(struct ipath_devdata *dd, int reinit)
|
|
|
|
{
|
2007-12-20 17:43:23 +07:00
|
|
|
int ret = 0;
|
2008-05-08 01:00:15 +07:00
|
|
|
u32 kpiobufs, defkbufs;
|
2007-03-16 04:45:02 +07:00
|
|
|
u32 piobufs, uports;
|
2006-07-01 18:36:03 +07:00
|
|
|
u64 val;
|
2008-04-17 11:09:29 +07:00
|
|
|
struct ipath_portdata *pd;
|
2006-07-01 18:36:15 +07:00
|
|
|
gfp_t gfp_flags = GFP_USER | __GFP_COMP;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
ret = init_housekeeping(dd, reinit);
|
2006-03-30 06:23:28 +07:00
|
|
|
if (ret)
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We could bump this to allow for full rcvegrcnt + rcvtidcnt,
|
|
|
|
* but then it no longer nicely fits power of two, and since
|
|
|
|
* we now use routines that backend onto __get_free_pages, the
|
|
|
|
* rest would be wasted.
|
|
|
|
*/
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_rcvhdrcnt = max(dd->ipath_p0_rcvegrcnt, dd->ipath_rcvegrcnt);
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvhdrcnt,
|
|
|
|
dd->ipath_rcvhdrcnt);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set up the shadow copies of the piobufavail registers,
|
|
|
|
* which we compare against the chip registers for now, and
|
|
|
|
* the in memory DMA'ed copies of the registers. This has to
|
|
|
|
* be done early, before we calculate lastport, etc.
|
|
|
|
*/
|
2007-03-16 04:45:02 +07:00
|
|
|
piobufs = dd->ipath_piobcnt2k + dd->ipath_piobcnt4k;
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* calc number of pioavail registers, and save it; we have 2
|
|
|
|
* bits per buffer.
|
|
|
|
*/
|
2007-03-16 04:45:02 +07:00
|
|
|
dd->ipath_pioavregs = ALIGN(piobufs, sizeof(u64) * BITS_PER_BYTE / 2)
|
2006-03-30 06:23:28 +07:00
|
|
|
/ (sizeof(u64) * BITS_PER_BYTE / 2);
|
2007-03-16 04:45:02 +07:00
|
|
|
uports = dd->ipath_cfgports ? dd->ipath_cfgports - 1 : 0;
|
2008-05-08 01:00:15 +07:00
|
|
|
if (piobufs > 144)
|
|
|
|
defkbufs = 32 + dd->ipath_pioreserved;
|
2006-04-25 04:23:00 +07:00
|
|
|
else
|
2008-05-08 01:00:15 +07:00
|
|
|
defkbufs = 16 + dd->ipath_pioreserved;
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2008-05-08 01:00:15 +07:00
|
|
|
if (ipath_kpiobufs && (ipath_kpiobufs +
|
|
|
|
(uports * IPATH_MIN_USER_PORT_BUFCNT)) > piobufs) {
|
2007-12-20 17:43:23 +07:00
|
|
|
int i = (int) piobufs -
|
2007-03-16 04:45:02 +07:00
|
|
|
(int) (uports * IPATH_MIN_USER_PORT_BUFCNT);
|
2008-04-17 11:09:29 +07:00
|
|
|
if (i < 1)
|
|
|
|
i = 1;
|
2007-03-16 04:45:02 +07:00
|
|
|
dev_info(&dd->pcidev->dev, "Allocating %d PIO bufs of "
|
|
|
|
"%d for kernel leaves too few for %d user ports "
|
2008-05-08 01:00:15 +07:00
|
|
|
"(%d each); using %u\n", ipath_kpiobufs,
|
2007-03-16 04:45:02 +07:00
|
|
|
piobufs, uports, IPATH_MIN_USER_PORT_BUFCNT, i);
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* shouldn't change ipath_kpiobufs, because could be
|
|
|
|
* different for different devices...
|
|
|
|
*/
|
|
|
|
kpiobufs = i;
|
2008-05-08 01:00:15 +07:00
|
|
|
} else if (ipath_kpiobufs)
|
|
|
|
kpiobufs = ipath_kpiobufs;
|
|
|
|
else
|
|
|
|
kpiobufs = defkbufs;
|
2007-03-16 04:45:02 +07:00
|
|
|
dd->ipath_lastport_piobuf = piobufs - kpiobufs;
|
|
|
|
dd->ipath_pbufsport =
|
|
|
|
uports ? dd->ipath_lastport_piobuf / uports : 0;
|
2008-05-08 01:00:15 +07:00
|
|
|
/* if not an even divisor, some user ports get extra buffers */
|
|
|
|
dd->ipath_ports_extrabuf = dd->ipath_lastport_piobuf -
|
|
|
|
(dd->ipath_pbufsport * uports);
|
|
|
|
if (dd->ipath_ports_extrabuf)
|
|
|
|
ipath_dbg("%u pbufs/port leaves some unused, add 1 buffer to "
|
|
|
|
"ports <= %u\n", dd->ipath_pbufsport,
|
|
|
|
dd->ipath_ports_extrabuf);
|
2008-04-17 11:09:26 +07:00
|
|
|
dd->ipath_lastpioindex = 0;
|
|
|
|
dd->ipath_lastpioindexl = dd->ipath_piobcnt2k;
|
2008-05-08 01:00:15 +07:00
|
|
|
/* ipath_pioavailshadow initialized earlier */
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_cdbg(VERBOSE, "%d PIO bufs for kernel out of %d total %u "
|
|
|
|
"each for %u user ports\n", kpiobufs,
|
2007-03-16 04:45:02 +07:00
|
|
|
piobufs, dd->ipath_pbufsport, uports);
|
2008-04-17 11:09:31 +07:00
|
|
|
ret = dd->ipath_f_early_init(dd);
|
|
|
|
if (ret) {
|
|
|
|
ipath_dev_err(dd, "Early initialization failure\n");
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
/*
|
|
|
|
* Early_init sets rcvhdrentsize and rcvhdrsize, so this must be
|
|
|
|
* done after early_init.
|
|
|
|
*/
|
2006-03-30 06:23:28 +07:00
|
|
|
dd->ipath_hdrqlast =
|
|
|
|
dd->ipath_rcvhdrentsize * (dd->ipath_rcvhdrcnt - 1);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvhdrentsize,
|
|
|
|
dd->ipath_rcvhdrentsize);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvhdrsize,
|
|
|
|
dd->ipath_rcvhdrsize);
|
|
|
|
|
|
|
|
if (!reinit) {
|
|
|
|
ret = init_pioavailregs(dd);
|
|
|
|
init_shadow_tids(dd);
|
|
|
|
if (ret)
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:26 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_sendpioavailaddr,
|
|
|
|
dd->ipath_pioavailregs_phys);
|
2008-05-08 01:00:15 +07:00
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* this is to detect s/w errors, which the h/w works around by
|
|
|
|
* ignoring the low 6 bits of address, if it wasn't aligned.
|
|
|
|
*/
|
|
|
|
val = ipath_read_kreg64(dd, dd->ipath_kregs->kr_sendpioavailaddr);
|
|
|
|
if (val != dd->ipath_pioavailregs_phys) {
|
|
|
|
ipath_dev_err(dd, "Catastrophic software error, "
|
|
|
|
"SendPIOAvailAddr written as %lx, "
|
|
|
|
"read back as %llx\n",
|
|
|
|
(unsigned long) dd->ipath_pioavailregs_phys,
|
|
|
|
(unsigned long long) val);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_rcvbthqp, IPATH_KD_QP);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* make sure we are not in freeze, and PIO send enabled, so
|
|
|
|
* writes to pbc happen
|
|
|
|
*/
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_hwerrmask, 0ULL);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_hwerrclear,
|
|
|
|
~0ULL&~INFINIPATH_HWE_MEMBISTFAILED);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_control, 0ULL);
|
2007-09-05 15:57:14 +07:00
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* before error clears, since we expect serdes pll errors during
|
|
|
|
* this, the first time after reset
|
|
|
|
*/
|
|
|
|
if (bringup_link(dd)) {
|
|
|
|
dev_info(&dd->pcidev->dev, "Failed to bringup IB link\n");
|
|
|
|
ret = -ENETDOWN;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* clear any "expected" hwerrs from reset and/or initialization
|
|
|
|
* clear any that aren't enabled (at least this once), and then
|
|
|
|
* set the enable mask
|
|
|
|
*/
|
|
|
|
dd->ipath_f_init_hwerrors(dd);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_hwerrclear,
|
|
|
|
~0ULL&~INFINIPATH_HWE_MEMBISTFAILED);
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_hwerrmask,
|
|
|
|
dd->ipath_hwerrmask);
|
|
|
|
|
|
|
|
/* clear all */
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_errorclear, -1LL);
|
|
|
|
/* enable errors that are masked, at least this first time. */
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_errormask,
|
|
|
|
~dd->ipath_maskederrs);
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_maskederrs = 0; /* don't re-enable ignored in timer */
|
|
|
|
dd->ipath_errormask =
|
|
|
|
ipath_read_kreg64(dd, dd->ipath_kregs->kr_errormask);
|
2007-07-21 04:41:26 +07:00
|
|
|
/* clear any interrupts up to this point (ints still not enabled) */
|
2006-03-30 06:23:28 +07:00
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_intclear, -1LL);
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_f_tidtemplate(dd);
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* Set up the port 0 (kernel) rcvhdr q and egr TIDs. If doing
|
|
|
|
* re-init, the simplest way to handle this is to free
|
|
|
|
* existing, and re-allocate.
|
2007-03-16 04:45:08 +07:00
|
|
|
* Need to re-create rest of port 0 portdata as well.
|
2006-03-30 06:23:28 +07:00
|
|
|
*/
|
2008-04-17 11:09:29 +07:00
|
|
|
pd = dd->ipath_pd[0];
|
2006-07-01 18:36:03 +07:00
|
|
|
if (reinit) {
|
2008-04-17 11:09:29 +07:00
|
|
|
struct ipath_portdata *npd;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Alloc and init new ipath_portdata for port0,
|
2007-03-16 04:45:08 +07:00
|
|
|
* Then free old pd. Could lead to fragmentation, but also
|
|
|
|
* makes later support for hot-swap easier.
|
|
|
|
*/
|
|
|
|
npd = create_portdata0(dd);
|
|
|
|
if (npd) {
|
|
|
|
ipath_free_pddata(dd, pd);
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_pd[0] = npd;
|
|
|
|
pd = npd;
|
2007-03-16 04:45:08 +07:00
|
|
|
} else {
|
2008-04-17 11:09:29 +07:00
|
|
|
ipath_dev_err(dd, "Unable to allocate portdata"
|
|
|
|
" for port 0, failing\n");
|
2007-03-16 04:45:08 +07:00
|
|
|
ret = -ENOMEM;
|
|
|
|
goto done;
|
|
|
|
}
|
2006-07-01 18:36:03 +07:00
|
|
|
}
|
2006-03-30 06:23:28 +07:00
|
|
|
ret = ipath_create_rcvhdrq(dd, pd);
|
2008-04-17 11:09:29 +07:00
|
|
|
if (!ret)
|
2006-03-30 06:23:28 +07:00
|
|
|
ret = create_port0_egr(dd);
|
2008-04-17 11:09:29 +07:00
|
|
|
if (ret) {
|
|
|
|
ipath_dev_err(dd, "failed to allocate kernel port's "
|
2006-03-30 06:23:28 +07:00
|
|
|
"rcvhdrq and/or egr bufs\n");
|
2008-04-17 11:09:29 +07:00
|
|
|
goto done;
|
|
|
|
}
|
2006-03-30 06:23:28 +07:00
|
|
|
else
|
2008-04-17 11:09:29 +07:00
|
|
|
enable_chip(dd, reinit);
|
2006-03-30 06:23:28 +07:00
|
|
|
|
2008-05-08 01:00:15 +07:00
|
|
|
/* after enable_chip, so pioavailshadow setup */
|
|
|
|
ipath_chg_pioavailkernel(dd, 0, piobufs, 1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Cancel any possible active sends from early driver load.
|
|
|
|
* Follows early_init because some chips have to initialize
|
|
|
|
* PIO buffers in early_init to avoid false parity errors.
|
|
|
|
* After enable and ipath_chg_pioavailkernel so we can safely
|
|
|
|
* enable pioavail updates and PIOENABLE; packets are now
|
|
|
|
* ready to go out.
|
|
|
|
*/
|
|
|
|
ipath_cancel_sends(dd, 1);
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
if (!reinit) {
|
|
|
|
/*
|
|
|
|
* Used when we close a port, for DMA already in flight
|
|
|
|
* at close.
|
|
|
|
*/
|
2006-07-01 18:36:15 +07:00
|
|
|
dd->ipath_dummy_hdrq = dma_alloc_coherent(
|
2008-04-17 11:09:29 +07:00
|
|
|
&dd->pcidev->dev, dd->ipath_pd[0]->port_rcvhdrq_size,
|
2006-07-01 18:36:15 +07:00
|
|
|
&dd->ipath_dummy_hdrq_phys,
|
|
|
|
gfp_flags);
|
2008-04-17 11:09:29 +07:00
|
|
|
if (!dd->ipath_dummy_hdrq) {
|
2006-07-01 18:36:15 +07:00
|
|
|
dev_info(&dd->pcidev->dev,
|
|
|
|
"Couldn't allocate 0x%lx bytes for dummy hdrq\n",
|
2008-04-17 11:09:29 +07:00
|
|
|
dd->ipath_pd[0]->port_rcvhdrq_size);
|
2006-07-01 18:36:15 +07:00
|
|
|
/* fallback to just 0'ing */
|
|
|
|
dd->ipath_dummy_hdrq_phys = 0UL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* cause retrigger of pending interrupts ignored during init,
|
|
|
|
* even if we had errors
|
|
|
|
*/
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_intclear, 0ULL);
|
|
|
|
|
2008-04-17 11:09:29 +07:00
|
|
|
if (!dd->ipath_stats_timer_active) {
|
2006-03-30 06:23:28 +07:00
|
|
|
/*
|
|
|
|
* first init, or after an admin disable/enable
|
|
|
|
* set up stats retrieval timer, even if we had errors
|
|
|
|
* in last portion of setup
|
|
|
|
*/
|
|
|
|
init_timer(&dd->ipath_stats_timer);
|
|
|
|
dd->ipath_stats_timer.function = ipath_get_faststats;
|
|
|
|
dd->ipath_stats_timer.data = (unsigned long) dd;
|
|
|
|
/* every 5 seconds; */
|
|
|
|
dd->ipath_stats_timer.expires = jiffies + 5 * HZ;
|
|
|
|
/* takes ~16 seconds to overflow at full IB 4x bandwdith */
|
|
|
|
add_timer(&dd->ipath_stats_timer);
|
|
|
|
dd->ipath_stats_timer_active = 1;
|
|
|
|
}
|
|
|
|
|
2008-04-17 11:09:32 +07:00
|
|
|
/* Set up SendDMA if chip supports it */
|
|
|
|
if (dd->ipath_flags & IPATH_HAS_SEND_DMA)
|
|
|
|
ret = setup_sdma(dd);
|
|
|
|
|
2008-04-17 11:09:24 +07:00
|
|
|
/* Set up HoL state */
|
|
|
|
init_timer(&dd->ipath_hol_timer);
|
|
|
|
dd->ipath_hol_timer.function = ipath_hol_event;
|
|
|
|
dd->ipath_hol_timer.data = (unsigned long)dd;
|
|
|
|
dd->ipath_hol_state = IPATH_HOL_UP;
|
|
|
|
|
2006-03-30 06:23:28 +07:00
|
|
|
done:
|
|
|
|
if (!ret) {
|
|
|
|
*dd->ipath_statusp |= IPATH_STATUS_CHIP_PRESENT;
|
|
|
|
if (!dd->ipath_f_intrsetup(dd)) {
|
|
|
|
/* now we can enable all interrupts from the chip */
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_intmask,
|
|
|
|
-1LL);
|
|
|
|
/* force re-interrupt of any pending interrupts. */
|
|
|
|
ipath_write_kreg(dd, dd->ipath_kregs->kr_intclear,
|
|
|
|
0ULL);
|
|
|
|
/* chip is usable; mark it as initialized */
|
|
|
|
*dd->ipath_statusp |= IPATH_STATUS_INITTED;
|
2008-04-17 11:09:30 +07:00
|
|
|
|
|
|
|
/*
|
|
|
|
* setup to verify we get an interrupt, and fallback
|
|
|
|
* to an alternate if necessary and possible
|
|
|
|
*/
|
|
|
|
if (!reinit) {
|
|
|
|
init_timer(&dd->ipath_intrchk_timer);
|
|
|
|
dd->ipath_intrchk_timer.function =
|
|
|
|
verify_interrupt;
|
|
|
|
dd->ipath_intrchk_timer.data =
|
|
|
|
(unsigned long) dd;
|
|
|
|
}
|
|
|
|
dd->ipath_intrchk_timer.expires = jiffies + HZ/2;
|
|
|
|
add_timer(&dd->ipath_intrchk_timer);
|
2006-03-30 06:23:28 +07:00
|
|
|
} else
|
|
|
|
ipath_dev_err(dd, "No interrupts enabled, couldn't "
|
|
|
|
"setup interrupt address\n");
|
|
|
|
|
|
|
|
if (dd->ipath_cfgports > ipath_stats.sps_nports)
|
|
|
|
/*
|
|
|
|
* sps_nports is a global, so, we set it to
|
|
|
|
* the highest number of ports of any of the
|
|
|
|
* chips we find; we never decrement it, at
|
|
|
|
* least for now. Since this might have changed
|
|
|
|
* over disable/enable or prior to reset, always
|
|
|
|
* do the check and potentially adjust.
|
|
|
|
*/
|
|
|
|
ipath_stats.sps_nports = dd->ipath_cfgports;
|
|
|
|
} else
|
|
|
|
ipath_dbg("Failed (%d) to initialize chip\n", ret);
|
|
|
|
|
|
|
|
/* if ret is non-zero, we probably should do some cleanup
|
|
|
|
here... */
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ipath_set_kpiobufs(const char *str, struct kernel_param *kp)
|
|
|
|
{
|
|
|
|
struct ipath_devdata *dd;
|
|
|
|
unsigned long flags;
|
|
|
|
unsigned short val;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = ipath_parse_ushort(str, &val);
|
|
|
|
|
|
|
|
spin_lock_irqsave(&ipath_devs_lock, flags);
|
|
|
|
|
|
|
|
if (ret < 0)
|
|
|
|
goto bail;
|
|
|
|
|
|
|
|
if (val == 0) {
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
|
|
|
|
list_for_each_entry(dd, &ipath_dev_list, ipath_list) {
|
|
|
|
if (dd->ipath_kregbase)
|
|
|
|
continue;
|
|
|
|
if (val > (dd->ipath_piobcnt2k + dd->ipath_piobcnt4k -
|
|
|
|
(dd->ipath_cfgports *
|
|
|
|
IPATH_MIN_USER_PORT_BUFCNT)))
|
|
|
|
{
|
|
|
|
ipath_dev_err(
|
|
|
|
dd,
|
|
|
|
"Allocating %d PIO bufs for kernel leaves "
|
|
|
|
"too few for %d user ports (%d each)\n",
|
|
|
|
val, dd->ipath_cfgports - 1,
|
|
|
|
IPATH_MIN_USER_PORT_BUFCNT);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto bail;
|
|
|
|
}
|
|
|
|
dd->ipath_lastport_piobuf =
|
|
|
|
dd->ipath_piobcnt2k + dd->ipath_piobcnt4k - val;
|
|
|
|
}
|
|
|
|
|
2006-08-26 01:24:29 +07:00
|
|
|
ipath_kpiobufs = val;
|
2006-03-30 06:23:28 +07:00
|
|
|
ret = 0;
|
|
|
|
bail:
|
|
|
|
spin_unlock_irqrestore(&ipath_devs_lock, flags);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|