diff options
Diffstat (limited to 'usr/src/uts/intel/io')
| -rw-r--r-- | usr/src/uts/intel/io/dktp/dcdev/dadk.c | 48 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/hotplug/pcicfg/pcicfg.c | 12 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/ipmi/ipmivars.h | 1 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/pci/pci_boot.c | 2 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/buildNumber.h | 12 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/includeCheck.h | 159 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/net.h | 220 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/net_sg.h | 84 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vm_basic_types.h | 1037 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vm_device_version.h | 246 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vmnet_def.h | 91 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vmxnet.c | 2442 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vmxnet.conf | 24 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vmxnet2_def.h | 436 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet/vmxnet_def.h | 184 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet3s/vmxnet3_rx.c | 3 | ||||
| -rw-r--r-- | usr/src/uts/intel/io/vmxnet3s/vmxnet3_tx.c | 3 | 
17 files changed, 4996 insertions, 8 deletions
| diff --git a/usr/src/uts/intel/io/dktp/dcdev/dadk.c b/usr/src/uts/intel/io/dktp/dcdev/dadk.c index 35f97482b8..f74a0d4137 100644 --- a/usr/src/uts/intel/io/dktp/dcdev/dadk.c +++ b/usr/src/uts/intel/io/dktp/dcdev/dadk.c @@ -21,6 +21,7 @@  /*   * Copyright (c) 2006, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2011, Joyent, Inc. All rights reserved.   */  /* @@ -170,6 +171,8 @@ static	int	dadk_debug = DGEOM;  #endif	/* DADK_DEBUG */ +#define ONE_MIN	((longlong_t)60 * NANOSEC) +  static int dadk_check_media_time = 3000000;	/* 3 Second State Check */  static int dadk_dk_maxphys = 0x80000; @@ -1376,6 +1379,47 @@ static struct dadkio_derr dadk_errtab[] = {  	{COMMAND_DONE_ERROR, GDA_FATAL},	/* 23 DERR_RESV		*/  }; +/* + * A bad disk can result in a large number of errors spewed to the log. + * This can in turn lead to /var/adm/messages filling up the file system on + * a machine with a small root or /var file system. + * + * Instead of logging every error, if we're seeing repeated errors on a disk + * only log them periodically. + */ +static void +dadk_logerr(struct dadk *dadkp, struct cmpkt *pktp, char *label, +    int severity, daddr_t blkno, daddr_t err_blkno, +    char **cmdvec, char **senvec) +{ +	hrtime_t now; + +	now = gethrtime(); +	if ((now - dadkp->dad_last_log) < ONE_MIN) { +		atomic_add_32(&dadkp->dad_err_cnt, 1); +		return; +	} + +	if (dadkp->dad_err_cnt > 0) { +		dev_info_t *dev = dadkp->dad_sd->sd_dev; +		char name[256], buf[256]; + +		if (dev) +			(void) snprintf(name, sizeof (name), "%s (%s%d)", +                            ddi_pathname(dev, buf), label, +                            ddi_get_instance(dev)); +		else +			(void) strlcpy(name, label, sizeof (name)); +		cmn_err(CE_WARN, "%s: %d additional unlogged errors\n", +		    name, dadkp->dad_err_cnt); +	} + +	gda_errmsg(dadkp->dad_sd, pktp, label, severity, blkno, err_blkno, +	    cmdvec, senvec); +	dadkp->dad_err_cnt = 0; +	dadkp->dad_last_log = now; +} +  static int  dadk_chkerr(struct cmpkt *pktp)  { @@ -1462,7 +1506,7 @@ dadk_chkerr(struct cmpkt *pktp)  		return (COMMAND_DONE);  	}  	if (pktp->cp_passthru == NULL) { -		gda_errmsg(dadkp->dad_sd, pktp, dadk_name, +		dadk_logerr(dadkp, pktp, dadk_name,  		    dadk_errtab[scb].d_severity, pktp->cp_srtsec,  		    err_blkno, dadk_cmds, dadk_sense);  	} @@ -1519,7 +1563,7 @@ dadk_recorderr(struct cmpkt *pktp, struct dadkio_rwcmd *rwcmdp)  	if (rwcmdp->flags & DADKIO_FLAG_SILENT)  		return; -	gda_errmsg(dadkp->dad_sd, pktp, dadk_name, dadk_errtab[scb].d_severity, +	dadk_logerr(dadkp, pktp, dadk_name, dadk_errtab[scb].d_severity,  	    rwcmdp->blkaddr, rwcmdp->status.failed_blk,  	    dadk_cmds, dadk_sense);  } diff --git a/usr/src/uts/intel/io/hotplug/pcicfg/pcicfg.c b/usr/src/uts/intel/io/hotplug/pcicfg/pcicfg.c index 09cf261d9b..b482117c7c 100644 --- a/usr/src/uts/intel/io/hotplug/pcicfg/pcicfg.c +++ b/usr/src/uts/intel/io/hotplug/pcicfg/pcicfg.c @@ -20,6 +20,7 @@   */  /*   * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2019, Joyent, Inc.   */  /* @@ -557,7 +558,7 @@ pcicfg_configure(dev_info_t *devi, uint_t device, uint_t function,  	pci_bus_range_t pci_bus_range;  	int rv;  	int circ; -	uint_t highest_bus; +	uint_t highest_bus, visited = 0;  	int ari_mode = B_FALSE;  	int max_function = PCI_MAX_FUNCTIONS;  	int trans_device; @@ -669,6 +670,11 @@ pcicfg_configure(dev_info_t *devi, uint_t device, uint_t function,  				goto cleanup;  		} +		/* +		 * Note that we've successfully gone through and visited at +		 * least one node. +		 */ +		visited++;  next:  		/*  		 * Determine if ARI Forwarding should be enabled. @@ -696,7 +702,7 @@ next:  				goto cleanup;  			/* -			 * Check if there are more fucntions to probe. +			 * Check if there are more functions to probe.  			 */  			if (next_function == 0) {  				DEBUG0("Next Function - " @@ -712,7 +718,7 @@ next:  	ndi_devi_exit(devi, circ); -	if (func == 0) +	if (visited == 0)  		return (PCICFG_FAILURE);	/* probe failed */  	else  		return (PCICFG_SUCCESS); diff --git a/usr/src/uts/intel/io/ipmi/ipmivars.h b/usr/src/uts/intel/io/ipmi/ipmivars.h index 7fd819cd3d..f547d6f043 100644 --- a/usr/src/uts/intel/io/ipmi/ipmivars.h +++ b/usr/src/uts/intel/io/ipmi/ipmivars.h @@ -78,6 +78,7 @@ struct ipmi_request {  #define	SMIC_CTL_STS			1  #define	SMIC_FLAGS			2 +struct ipmi_softc;  #define	IPMI_BUSY	0x1  #define	IPMI_CLOSING	0x2 diff --git a/usr/src/uts/intel/io/pci/pci_boot.c b/usr/src/uts/intel/io/pci/pci_boot.c index 601153d612..d2961338b6 100644 --- a/usr/src/uts/intel/io/pci/pci_boot.c +++ b/usr/src/uts/intel/io/pci/pci_boot.c @@ -2873,7 +2873,7 @@ add_ppb_props(dev_info_t *dip, uchar_t bus, uchar_t dev, uchar_t func,  	 * If it is unset, we disable i/o and mark it for reconfiguration in  	 * later passes by setting the base > limit  	 */ -	val = (uint_t)pci_getw(bus, dev, func, PCI_CONF_COMM); +	val = (uint64_t)pci_getw(bus, dev, func, PCI_CONF_COMM);  	if (val & PCI_COMM_IO) {  		val = (uint_t)pci_getb(bus, dev, func, PCI_BCNF_IO_BASE_LOW);  		io_range[0] = ((val & PCI_BCNF_IO_MASK) << PCI_BCNF_IO_SHIFT); diff --git a/usr/src/uts/intel/io/vmxnet/buildNumber.h b/usr/src/uts/intel/io/vmxnet/buildNumber.h new file mode 100644 index 0000000000..97f18a3cbc --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/buildNumber.h @@ -0,0 +1,12 @@ +#define BUILD_NUMBER \ +	"build-425873" +#define BUILD_NUMBER_NUMERIC \ +	425873 +#define BUILD_NUMBER_NUMERIC_STRING \ +	"425873" +#define PRODUCT_BUILD_NUMBER \ +	"product-build-6261" +#define PRODUCT_BUILD_NUMBER_NUMERIC \ +	6261 +#define PRODUCT_BUILD_NUMBER_NUMERIC_STRING \ +	"6261" diff --git a/usr/src/uts/intel/io/vmxnet/includeCheck.h b/usr/src/uts/intel/io/vmxnet/includeCheck.h new file mode 100644 index 0000000000..c414d6daf5 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/includeCheck.h @@ -0,0 +1,159 @@ +/********************************************************* + * Copyright (C) 1998 VMware, Inc. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published + * by the Free Software Foundation version 2.1 and no 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 Lesser GNU General Public + * License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA. + * + *********************************************************/ + +/********************************************************* + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. 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. + * 3. Neither the name of VMware Inc. nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission of VMware Inc. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * 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 OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/* + * includeCheck.h -- + * + *	Restrict include file use. + * + * In every .h file, define one or more of these + * + *	INCLUDE_ALLOW_VMX  + *	INCLUDE_ALLOW_USERLEVEL  + *	INCLUDE_ALLOW_VMCORE + *	INCLUDE_ALLOW_MODULE + *	INCLUDE_ALLOW_VMKERNEL  + *	INCLUDE_ALLOW_DISTRIBUTE + *	INCLUDE_ALLOW_VMK_MODULE + *      INCLUDE_ALLOW_VMKDRIVERS + *      INCLUDE_ALLOW_VMIROM + * + * Then include this file. + * + * Any file that has INCLUDE_ALLOW_DISTRIBUTE defined will potentially + * be distributed in source form along with GPLed code.  Ensure + * that this is acceptable. + */ + + +/* + * Declare a VMCORE-only variable to help classify object + * files.  The variable goes in the common block and does + * not create multiple definition link-time conflicts. + */ + +#if defined VMCORE && defined VMX86_DEVEL && defined VMX86_DEBUG && \ +    defined linux && !defined MODULE && \ +    !defined COMPILED_WITH_VMCORE +#define COMPILED_WITH_VMCORE compiled_with_vmcore +#ifdef ASM +        .comm   compiled_with_vmcore, 0 +#else +        asm(".comm compiled_with_vmcore, 0"); +#endif /* ASM */ +#endif + + +#if defined VMCORE && \ +    !(defined VMX86_VMX || defined VMM || \ +      defined MONITOR_APP || defined VMMON) +#error "Makefile problem: VMCORE without VMX86_VMX or \ +        VMM or MONITOR_APP or MODULE." +#endif + +#if defined VMCORE && !defined INCLUDE_ALLOW_VMCORE +#error "The surrounding include file is not allowed in vmcore." +#endif +#undef INCLUDE_ALLOW_VMCORE + +#if defined VMX86_VMX && !defined VMCORE && \ +    !(defined INCLUDE_ALLOW_VMX || defined INCLUDE_ALLOW_USERLEVEL) +#error "The surrounding include file is not allowed in the VMX." +#endif +#undef INCLUDE_ALLOW_VMX + +#if defined USERLEVEL && !defined VMX86_VMX && !defined VMCORE && \ +    !defined INCLUDE_ALLOW_USERLEVEL +#error "The surrounding include file is not allowed at userlevel." +#endif +#undef INCLUDE_ALLOW_USERLEVEL + +#if defined MODULE && !defined VMKERNEL_MODULE && \ +    !defined VMMON && !defined INCLUDE_ALLOW_MODULE +#error "The surrounding include file is not allowed in driver modules." +#endif +#undef INCLUDE_ALLOW_MODULE + +#if defined VMMON && !defined INCLUDE_ALLOW_VMMON +#error "The surrounding include file is not allowed in vmmon." +#endif +#undef INCLUDE_ALLOW_VMMON + +#if defined VMKERNEL && !defined INCLUDE_ALLOW_VMKERNEL +#error "The surrounding include file is not allowed in the vmkernel." +#endif +#undef INCLUDE_ALLOW_VMKERNEL + +#if defined GPLED_CODE && !defined INCLUDE_ALLOW_DISTRIBUTE +#error "The surrounding include file is not allowed in GPL code." +#endif +#undef INCLUDE_ALLOW_DISTRIBUTE + +#if defined VMKERNEL_MODULE && !defined VMKERNEL && \ +    !defined INCLUDE_ALLOW_VMK_MODULE && !defined INCLUDE_ALLOW_VMKDRIVERS +#error "The surrounding include file is not allowed in vmkernel modules." +#endif +#undef INCLUDE_ALLOW_VMK_MODULE +#undef INCLUDE_ALLOW_VMKDRIVERS + +#if defined VMIROM && ! defined INCLUDE_ALLOW_VMIROM +#error "The surrounding include file is not allowed in vmirom." +#endif +#undef INCLUDE_ALLOW_VMIROM diff --git a/usr/src/uts/intel/io/vmxnet/net.h b/usr/src/uts/intel/io/vmxnet/net.h new file mode 100644 index 0000000000..41b6eb1d14 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/net.h @@ -0,0 +1,220 @@ +/********************************************************* + * Copyright (C) 1998 VMware, Inc. All rights reserved. + * + * 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 version 2 and no 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. + * + * 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 St, Fifth Floor, Boston, MA  02110-1301 USA + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/************************************************************ + * + *   net.h + * + *   This file should contain all network global defines. + *   No vlance/vmxnet/vnet/vmknet specific stuff should be + *   put here only defines used/usable by all network code. + *   --gustav + * + ************************************************************/ + +#ifndef VMWARE_DEVICES_NET_H +#define VMWARE_DEVICES_NET_H + +#define INCLUDE_ALLOW_USERLEVEL +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMCORE + +#include "includeCheck.h" +#include "vm_device_version.h" + +#ifdef VMCORE +#include "config.h" +#include "str.h" +#include "strutil.h" +#endif + +#define ETHERNET_MTU         1518 +#define ETH_MIN_FRAME_LEN      60 + +#ifndef ETHER_ADDR_LEN +#define ETHER_ADDR_LEN          6  /* length of MAC address */ +#endif +#define ETH_HEADER_LEN	       14  /* length of Ethernet header */ +#define IP_ADDR_LEN	        4  /* length of IPv4 address */ +#define IP_HEADER_LEN	       20  /* minimum length of IPv4 header */ + +#define ETHER_MAX_QUEUED_PACKET 1600 + + +/* + * State's that a NIC can be in currently we only use this + * in VLance but if we implement/emulate new adapters that + * we also want to be able to morph a new corresponding + * state should be added. + */ + +#define LANCE_CHIP  0x2934 +#define VMXNET_CHIP 0x4392 + +/* + * Size of reserved IO space needed by the LANCE adapter and + * the VMXNET adapter. If you add more ports to Vmxnet than + * there is reserved space you must bump VMXNET_CHIP_IO_RESV_SIZE. + * The sizes must be powers of 2. + */ + +#define LANCE_CHIP_IO_RESV_SIZE  0x20 +#define VMXNET_CHIP_IO_RESV_SIZE 0x40 + +#define MORPH_PORT_SIZE 4 + +#ifdef VMCORE +typedef struct Net_AdapterCount { +   uint8 vlance; +   uint8 vmxnet2; +   uint8 vmxnet3; +   uint8 e1000; +   uint8 e1000e; +} Net_AdapterCount; +#endif + +#ifdef USERLEVEL + +/* + *---------------------------------------------------------------------------- + * + * Net_AddAddrToLADRF -- + * + *      Given a MAC address, sets the corresponding bit in the LANCE style + *      Logical Address Filter 'ladrf'. + *      The caller should have initialized the ladrf to all 0's, as this + *      function only ORs on a bit in the array. + *      'addr' is presumed to be ETHER_ADDR_LEN in size; + *      'ladrf' is presumed to point to a 64-bit vector. + * + *      Derived from a long history of derivations, originally inspired by + *      sample code from the AMD "Network Products: Ethernet Controllers 1998 + *      Data Book, Book 2", pages 1-53..1-55. + * + * Returns: + *      None. + * + * Side effects: + *      Updates 'ladrf'. + * + *---------------------------------------------------------------------------- + */ + +static INLINE void +Net_AddAddrToLadrf(const uint8 *addr,  // IN: pointer to MAC address +                   uint8 *ladrf)       // IN/OUT: pointer to ladrf +{ +#define CRC_POLYNOMIAL_BE 0x04c11db7UL	/* Ethernet CRC, big endian */ + +   uint16 hashcode; +   int32 crc = 0xffffffff;		/* init CRC for each address */ +   int32 j; +   int32 bit; +   int32 byte; + +   ASSERT(addr); +   ASSERT(ladrf); + +   for (byte = 0; byte < ETHER_ADDR_LEN; byte++) {  /* for each address byte */ +      /* process each address bit */ +      for (bit = *addr++, j = 0; +           j < 8; +           j++, bit >>= 1) { +	 crc = (crc << 1) ^ ((((crc < 0 ? 1 : 0) ^ bit) & 0x01) ? +               CRC_POLYNOMIAL_BE : 0); +      } +   } +   hashcode = (crc & 1);	       /* hashcode is 6 LSb of CRC ... */ +   for (j = 0; j < 5; j++) {	       /* ... in reverse order. */ +      hashcode = (hashcode << 1) | ((crc>>=1) & 1); +   } + +   ladrf[hashcode >> 3] |= 1 << (hashcode & 0x07); +} +#endif // USERLEVEL + +#ifdef VMCORE +/* + *---------------------------------------------------------------------- + * + * Net_GetNumAdapters -- + * + *      Returns the number of each type of network adapter configured in this  + *      VM. + * + * Results: + *      None. + * + * Side effects: + *      None. + * + *---------------------------------------------------------------------- + */ + +static INLINE void +Net_GetNumAdapters(Net_AdapterCount *counts) +{ +   uint32 i; + +   counts->vlance = 0; +   counts->vmxnet2 = 0; +   counts->vmxnet3 = 0; +   counts->e1000 = 0; +   counts->e1000e = 0; + +   for (i = 0; i < MAX_ETHERNET_CARDS; i++) { +      char* adapterStr; + +      if (!Config_GetBool(FALSE, "ethernet%d.present", i)) { +	 continue; +      } +      adapterStr = Config_GetString("vlance", "ethernet%d.virtualDev", i); +      if (Str_Strcasecmp(adapterStr, "vmxnet3") == 0) { +         counts->vmxnet3++; +      } else if (Str_Strcasecmp(adapterStr, "vlance") == 0) { +         counts->vlance++; +      } else if (Str_Strcasecmp(adapterStr, "vmxnet") == 0) { +         counts->vmxnet2++; +      } else if (Str_Strcasecmp(adapterStr, "e1000") == 0) { +         counts->e1000++; +      } else if (Str_Strcasecmp(adapterStr, "e1000e") == 0) { +         counts->e1000e++; +      } else { +         LOG_ONCE(("%s: unknown adapter: %s\n", __FUNCTION__, adapterStr)); +      } +      free(adapterStr);       +   } +} + +#endif // VMCORE + +#endif // VMWARE_DEVICES_NET_H diff --git a/usr/src/uts/intel/io/vmxnet/net_sg.h b/usr/src/uts/intel/io/vmxnet/net_sg.h new file mode 100644 index 0000000000..f6c30fb2b5 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/net_sg.h @@ -0,0 +1,84 @@ +/********************************************************* + * Copyright (C) 2000 VMware, Inc. All rights reserved. + * + * 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 version 2 and no 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. + * + * 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 St, Fifth Floor, Boston, MA  02110-1301 USA + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/* + * net_sg.h -- + * + *	Network packet scatter gather structure. + */ + + +#ifndef _NET_SG_H +#define _NET_SG_H + +#define INCLUDE_ALLOW_USERLEVEL + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMK_MODULE +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_DISTRIBUTE +#include "includeCheck.h" + +#define NET_SG_DEFAULT_LENGTH	16 + +/* + * A single scatter-gather element for a network packet. + * The address is split into low and high to save space. + * If we make it 64 bits then Windows pads things out such that + * we lose a lot of space for each scatter gather array. + * This adds up when you have embedded scatter-gather  + * arrays for transmit and receive ring buffers. + */ +typedef struct NetSG_Elem { +   uint32 	addrLow; +   uint16	addrHi; +   uint16	length; +} NetSG_Elem; + +typedef enum NetSG_AddrType { +   NET_SG_MACH_ADDR, +   NET_SG_PHYS_ADDR, +   NET_SG_VIRT_ADDR, +} NetSG_AddrType; + +typedef struct NetSG_Array { +   uint16	addrType; +   uint16	length; +   NetSG_Elem	sg[NET_SG_DEFAULT_LENGTH]; +} NetSG_Array; + +#define NET_SG_SIZE(len) (sizeof(NetSG_Array) + (len - NET_SG_DEFAULT_LENGTH) * sizeof(NetSG_Elem)) + +#define NET_SG_MAKE_PA(elem)                 (PA)QWORD(elem.addrHi, elem.addrLow) +#define NET_SG_MAKE_PTR(elem) (char *)(uintptr_t)QWORD(elem.addrHi, elem.addrLow) + +#endif diff --git a/usr/src/uts/intel/io/vmxnet/vm_basic_types.h b/usr/src/uts/intel/io/vmxnet/vm_basic_types.h new file mode 100644 index 0000000000..adeac1b708 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vm_basic_types.h @@ -0,0 +1,1037 @@ +/********************************************************* + * Copyright (C) 1998-2009 VMware, Inc. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published + * by the Free Software Foundation version 2.1 and no 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 Lesser GNU General Public + * License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA. + * + *********************************************************/ + +/********************************************************* + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + *    notice, this list of conditions and the following disclaimer. + * 2. 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. + * 3. Neither the name of VMware Inc. nor the names of its contributors + *    may be used to endorse or promote products derived from this software + *    without specific prior written permission of VMware Inc. + * + * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE + * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL + * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS + * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) + * 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 OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF + * SUCH DAMAGE. + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/* + * + * vm_basic_types.h -- + * + *    basic data types. + */ + + +#ifndef _VM_BASIC_TYPES_H_ +#define _VM_BASIC_TYPES_H_ + +#define INCLUDE_ALLOW_USERLEVEL + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMMON +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_VMKDRIVERS +#define INCLUDE_ALLOW_VMK_MODULE +#define INCLUDE_ALLOW_DISTRIBUTE +#define INCLUDE_ALLOW_VMCORE +#define INCLUDE_ALLOW_VMIROM +#include "includeCheck.h" + +/* STRICT ANSI means the Xserver build and X defines Bool differently. */ +#if !defined(_XTYPEDEF_BOOL) && \ +    (!defined(__STRICT_ANSI__) || defined(__FreeBSD__) || defined(__MINGW32__)) +#define _XTYPEDEF_BOOL +typedef char           Bool; +#endif + +#ifndef FALSE +#define FALSE          0 +#endif + +#ifndef TRUE +#define TRUE           1 +#endif + +#define IsBool(x)      (((x) & ~1) == 0) +#define IsBool2(x, y)  ((((x) | (y)) & ~1) == 0) + +/* + * Macros __i386__ and __ia64 are intrinsically defined by GCC + */ +#if defined _MSC_VER && defined _M_X64 +#  define __x86_64__ +#elif defined _MSC_VER && defined _M_IX86 +#  define __i386__ +#endif + +#ifdef __i386__ +#define VM_I386 +#endif + +#ifdef __x86_64__ +#define VM_X86_64 +#define VM_I386 +#define vm_x86_64 (1) +#else +#define vm_x86_64 (0) +#endif + + +#ifdef _MSC_VER + +#pragma warning (3 :4505) // unreferenced local function +#pragma warning (disable :4018) // signed/unsigned mismatch +#pragma warning (disable :4761) // integral size mismatch in argument; conversion supplied +#pragma warning (disable :4305) // truncation from 'const int' to 'short' +#pragma warning (disable :4244) // conversion from 'unsigned short' to 'unsigned char' +#pragma warning (disable :4267) // truncation of 'size_t' +#pragma warning (disable :4146) // unary minus operator applied to unsigned type, result still unsigned +#pragma warning (disable :4142) // benign redefinition of type + +#endif + +#if defined(__APPLE__) || defined(HAVE_STDINT_H) + +/* + * TODO: This is a C99 standard header.  We should be able to test for + * #if __STDC_VERSION__ >= 199901L, but that breaks the Netware build + * (which doesn't have stdint.h). + */ + +#include <stdint.h> + +typedef uint64_t    uint64; +typedef  int64_t     int64; +typedef uint32_t    uint32; +typedef  int32_t     int32; +typedef uint16_t    uint16; +typedef  int16_t     int16; +typedef  uint8_t    uint8; +typedef   int8_t     int8; + +/* + * Note: C does not specify whether char is signed or unsigned, and + * both gcc and msvc implement processor-specific signedness.  With + * three types: + * typeof(char) != typeof(signed char) != typeof(unsigned char) + * + * Be careful here, because gcc (4.0.1 and others) likes to warn about + * conversions between signed char * and char *. + */ + +#else /* !HAVE_STDINT_H */ + +#ifdef _MSC_VER + +typedef unsigned __int64 uint64; +typedef signed __int64 int64; + +#elif defined(__GNUC__) || defined(__SUNPRO_C) +/* The Xserver source compiles with -ansi -pendantic */ +#   if !defined(__STRICT_ANSI__) || defined(__FreeBSD__) +#      if defined(VM_X86_64) +typedef unsigned long uint64; +typedef long int64; +#      else +typedef unsigned long long uint64; +typedef long long int64; +#      endif +#   endif +#else +#   error - Need compiler define for int64/uint64 +#endif /* _MSC_VER */ + +typedef unsigned int       uint32; +typedef unsigned short     uint16; +typedef unsigned char      uint8; + +typedef int                int32; +typedef short              int16; +typedef signed char        int8; + +#endif /* HAVE_STDINT_H */ + +/* + * FreeBSD (for the tools build) unconditionally defines these in + * sys/inttypes.h so don't redefine them if this file has already + * been included. [greg] + * + * This applies to Solaris as well. + */ + +/* + * Before trying to do the includes based on OS defines, see if we can use + * feature-based defines to get as much functionality as possible + */ + +#ifdef HAVE_INTTYPES_H +#include <inttypes.h> +#endif +#ifdef HAVE_SYS_TYPES_H +#include <sys/types.h> +#endif +#ifdef HAVE_SYS_INTTYPES_H +#include <sys/inttypes.h> +#endif +#ifdef HAVE_STDLIB_H +#include <stdlib.h> +#endif + +#ifdef __FreeBSD__ +#include <sys/param.h> /* For __FreeBSD_version */ +#endif + +#if !defined(USING_AUTOCONF) +#   if defined(__FreeBSD__) || defined(sun) +#      ifdef KLD_MODULE +#         include <sys/types.h> +#      else +#         if __FreeBSD_version >= 500043 +#            if !defined(VMKERNEL) +#               include <inttypes.h> +#            endif +#            include <sys/types.h> +#         else +#            include <sys/inttypes.h> +#         endif +#      endif +#   elif defined __APPLE__ +#      if KERNEL +#         include <sys/unistd.h> +#         include <sys/types.h> /* mostly for size_t */ +#         include <stdint.h> +#      else +#         include <unistd.h> +#         include <inttypes.h> +#         include <stdlib.h> +#         include <stdint.h> +#      endif +#   else +#      if !defined(__intptr_t_defined) && !defined(intptr_t) +#         ifdef VM_I386 +#         define __intptr_t_defined +#            ifdef VM_X86_64 +typedef int64     intptr_t; +#            else +typedef int32     intptr_t; +#            endif +#         elif defined(__arm__) +typedef int32     intptr_t; +#         endif +#      endif + +#      ifndef _STDINT_H +#         ifdef VM_I386 +#            ifdef VM_X86_64 +typedef uint64    uintptr_t; +#            else +typedef uint32    uintptr_t; +#            endif +#         elif defined(__arm__) +typedef uint32    uintptr_t; +#         endif +#      endif +#   endif +#endif + + +/* + * Time + * XXX These should be cleaned up.  -- edward + */ + +typedef int64 VmTimeType;          /* Time in microseconds */ +typedef int64 VmTimeRealClock;     /* Real clock kept in microseconds */ +typedef int64 VmTimeVirtualClock;  /* Virtual Clock kept in CPU cycles */ + +/* + * Printf format specifiers for size_t and 64-bit number. + * Use them like this: + *    printf("%"FMT64"d\n", big); + * + * FMTH is for handles/fds. + */ + +#ifdef _MSC_VER +   #define FMT64      "I64" +   #ifdef VM_X86_64 +      #define FMTSZ      "I64" +      #define FMTPD      "I64" +      #define FMTH       "I64" +   #else +      #define FMTSZ      "I" +      #define FMTPD      "I" +      #define FMTH       "I" +   #endif +#elif defined __APPLE__ +   /* Mac OS hosts use the same formatters for 32- and 64-bit. */ +   #define FMT64 "ll" +   #if KERNEL +      #define FMTSZ "l" +   #else +      #define FMTSZ "z" +   #endif +   #define FMTPD "l" +   #define FMTH "" +#elif defined(__GNUC__) || defined(__SUNPRO_C) +   #define FMTH "" +   #if defined(N_PLAT_NLM) || defined(sun) || \ +       (defined(__FreeBSD__) && (__FreeBSD__ + 0) && ((__FreeBSD__ + 0) < 5)) +      /* +       * Why (__FreeBSD__ + 0)?  See bug 141008. +       * Yes, we really need to test both (__FreeBSD__ + 0) and +       * ((__FreeBSD__ + 0) < 5).  No, we can't remove "+ 0" from +       * ((__FreeBSD__ + 0) < 5). +       */ +      #ifdef VM_X86_64 +         #define FMTSZ  "l" +         #define FMTPD  "l" +      #else +         #define FMTSZ  "" +         #define FMTPD  "" +      #endif +   #elif defined(__linux__) \ +      || (defined(_POSIX_C_SOURCE) && _POSIX_C_SOURCE >= 200112L) \ +      || (defined(_POSIX_VERSION) && _POSIX_VERSION >= 200112L) \ +      || (defined(_POSIX2_VERSION) && _POSIX2_VERSION >= 200112L) +      /* BSD, Linux */ +      #define FMTSZ     "z" + +      #if defined(VM_X86_64) +         #define FMTPD  "l" +      #else +         #define FMTPD  "" +      #endif +   #else +      /* Systems with a pre-C99 libc */ +      #define FMTSZ     "Z" +      #ifdef VM_X86_64 +         #define FMTPD  "l" +      #else +         #define FMTPD  "" +      #endif +   #endif +   #ifdef VM_X86_64 +      #define FMT64     "l" +   #elif defined(sun) || defined(__FreeBSD__) +      #define FMT64     "ll" +   #else +      #define FMT64     "L" +   #endif +#else +   #error - Need compiler define for FMT64 and FMTSZ +#endif + +/* + * Suffix for 64-bit constants.  Use it like this: + *    CONST64(0x7fffffffffffffff) for signed or + *    CONST64U(0x7fffffffffffffff) for unsigned. + * + * 2004.08.30(thutt): + *   The vmcore/asm64/gen* programs are compiled as 32-bit + *   applications, but must handle 64 bit constants.  If the + *   64-bit-constant defining macros are already defined, the + *   definition will not be overwritten. + */ + +#if !defined(CONST64) || !defined(CONST64U) +#ifdef _MSC_VER +#define CONST64(c) c##I64 +#define CONST64U(c) c##uI64 +#elif defined __APPLE__ +#define CONST64(c) c##LL +#define CONST64U(c) c##uLL +#elif defined(__GNUC__) || defined(__SUNPRO_C) +#ifdef VM_X86_64 +#define CONST64(c) c##L +#define CONST64U(c) c##uL +#else +#define CONST64(c) c##LL +#define CONST64U(c) c##uLL +#endif +#else +#error - Need compiler define for CONST64 +#endif +#endif + +/* + * Use CONST3264/CONST3264U if you want a constant to be + * treated as a 32-bit number on 32-bit compiles and + * a 64-bit number on 64-bit compiles. Useful in the case + * of shifts, like (CONST3264U(1) << x), where x could be + * more than 31 on a 64-bit compile. + */ + +#ifdef VM_X86_64 +    #define CONST3264(a) CONST64(a) +    #define CONST3264U(a) CONST64U(a) +#else +    #define CONST3264(a) (a) +    #define CONST3264U(a) (a) +#endif + +#define MIN_INT8   ((int8)0x80) +#define MAX_INT8   ((int8)0x7f) + +#define MIN_UINT8  ((uint8)0) +#define MAX_UINT8  ((uint8)0xff) + +#define MIN_INT16  ((int16)0x8000) +#define MAX_INT16  ((int16)0x7fff) + +#define MIN_UINT16 ((uint16)0) +#define MAX_UINT16 ((uint16)0xffff) + +#define MIN_INT32  ((int32)0x80000000) +#define MAX_INT32  ((int32)0x7fffffff) + +#define MIN_UINT32 ((uint32)0) +#define MAX_UINT32 ((uint32)0xffffffff) + +#define MIN_INT64  (CONST64(0x8000000000000000)) +#define MAX_INT64  (CONST64(0x7fffffffffffffff)) + +#define MIN_UINT64 (CONST64U(0)) +#define MAX_UINT64 (CONST64U(0xffffffffffffffff)) + +typedef uint8 *TCA;  /* Pointer into TC (usually). */ + +/* + * Type big enough to hold an integer between 0..100 + */ +typedef uint8 Percent; +#define AsPercent(v)	((Percent)(v)) +#define CHOOSE_PERCENT  AsPercent(101) + + +typedef uintptr_t VA; +typedef uintptr_t VPN; + +typedef uint64    PA; +typedef uint32    PPN; + +typedef uint64    PhysMemOff; +typedef uint64    PhysMemSize; + +/* The Xserver source compiles with -ansi -pendantic */ +#ifndef __STRICT_ANSI__ +typedef uint64    BA; +#endif +typedef uint32    BPN; +typedef uint32    PageNum; +typedef unsigned  MemHandle; +typedef int32     World_ID; + +/* !! do not alter the definition of INVALID_WORLD_ID without ensuring + * that the values defined in both bora/public/vm_basic_types.h and + * lib/vprobe/vm_basic_types.h are the same.  Additionally, the definition + * of VMK_INVALID_WORLD_ID in vmkapi_world.h also must be defined with + * the same value + */ + +#define INVALID_WORLD_ID ((World_ID)0) + +typedef World_ID User_CartelID; +#define INVALID_CARTEL_ID INVALID_WORLD_ID + +typedef User_CartelID User_SessionID; +#define INVALID_SESSION_ID INVALID_CARTEL_ID + +typedef User_CartelID User_CartelGroupID; +#define INVALID_CARTELGROUP_ID INVALID_CARTEL_ID + +typedef uint32 Worldlet_ID; +#define INVALID_WORLDLET_ID ((Worldlet_ID)-1) + +/* The Xserver source compiles with -ansi -pendantic */ +#ifndef __STRICT_ANSI__ +typedef uint64 MA; +typedef uint32 MPN; +#endif + +/* + * This type should be used for variables that contain sector + * position/quantity. + */ +typedef uint64 SectorType; + +/* + * Linear address + */ + +typedef uintptr_t LA; +typedef uintptr_t LPN; +#define LA_2_LPN(_la)     ((_la) >> PAGE_SHIFT) +#define LPN_2_LA(_lpn)    ((_lpn) << PAGE_SHIFT) + +#define LAST_LPN   ((((LA)  1) << (8 * sizeof(LA)   - PAGE_SHIFT)) - 1) +#define LAST_LPN32 ((((LA32)1) << (8 * sizeof(LA32) - PAGE_SHIFT)) - 1) +#define LAST_LPN64 ((((LA64)1) << (8 * sizeof(LA64) - PAGE_SHIFT)) - 1) + +/* Valid bits in a LPN. */ +#define LPN_MASK   LAST_LPN +#define LPN_MASK32 LAST_LPN32 +#define LPN_MASK64 LAST_LPN64 + +/* + * On 64 bit platform, address and page number types default + * to 64 bit. When we need to represent a 32 bit address, we use + * types defined below. + * + * On 32 bit platform, the following types are the same as the + * default types. + */ +typedef uint32 VA32; +typedef uint32 VPN32; +typedef uint32 LA32; +typedef uint32 LPN32; +typedef uint32 PA32; +typedef uint32 PPN32; +typedef uint32 MA32; +typedef uint32 MPN32; + +/* + * On 64 bit platform, the following types are the same as the + * default types. + */ +typedef uint64 VA64; +typedef uint64 VPN64; +typedef uint64 LA64; +typedef uint64 LPN64; +typedef uint64 PA64; +typedef uint64 PPN64; +typedef uint64 MA64; +typedef uint64 MPN64; + +/* + * VA typedefs for user world apps. + */ +typedef VA32 UserVA32; +typedef VA64 UserVA64; +typedef UserVA64 UserVAConst; /* Userspace ptr to data that we may only read. */ +typedef UserVA32 UserVA32Const; /* Userspace ptr to data that we may only read. */ +typedef UserVA64 UserVA64Const; /* Used by 64-bit syscalls until conversion is finished. */ +#ifdef VMKERNEL +typedef UserVA64 UserVA; +#else +typedef void * UserVA; +#endif + + +/* + * Maximal possible PPN value (errors too) that PhysMem can handle. + * Must be at least as large as MAX_PPN which is the maximum PPN + * for any region other than buserror. + */ +#define PHYSMEM_MAX_PPN ((PPN)0xffffffff) +#define MAX_PPN         ((PPN)0x1fffffff)   /* Maximal observable PPN value. */ +#define INVALID_PPN     ((PPN)0xffffffff) + +#define INVALID_BPN     ((BPN)0x1fffffff) + +#define RESERVED_MPN    ((MPN) 0) +#define INVALID_MPN     ((MPN)-1) +#define MEMREF_MPN      ((MPN)-2) +#define RELEASED_MPN    ((MPN)-3) +#define MAX_MPN         ((MPN)0x7fffffff)  /* 43 bits of address space. */ + +#define INVALID_LPN     ((LPN)-1) +#define INVALID_VPN     ((VPN)-1) +#define INVALID_LPN64   ((LPN64)-1) +#define INVALID_PAGENUM ((PageNum)-1) + + +/* + * Format modifier for printing VA, LA, and VPN. + * Use them like this: Log("%#"FMTLA"x\n", laddr) + */ + +#if defined(VMM) || defined(FROBOS64) || vm_x86_64 || defined __APPLE__ +#   define FMTLA "l" +#   define FMTVA "l" +#   define FMTVPN "l" +#else +#   define FMTLA "" +#   define FMTVA "" +#   define FMTVPN "" +#endif + +#ifndef EXTERN +#define EXTERN        extern +#endif +#define CONST         const + + +#ifndef INLINE +#   ifdef _MSC_VER +#      define INLINE        __inline +#   else +#      define INLINE        inline +#   endif +#endif + + +/* + * Annotation for data that may be exported into a DLL and used by other + * apps that load that DLL and import the data. + */ +#if defined(_WIN32) && defined(VMX86_IMPORT_DLLDATA) +#  define VMX86_EXTERN_DATA       extern __declspec(dllimport) +#else // !_WIN32 +#  define VMX86_EXTERN_DATA       extern +#endif + +#if defined(_WIN32) && !defined(VMX86_NO_THREADS) +#define THREADSPECIFIC __declspec(thread) +#else +#define THREADSPECIFIC +#endif + +/* + * Due to the wonderful "registry redirection" feature introduced in + * 64-bit Windows, if you access any key under HKLM\Software in 64-bit + * code, you need to open/create/delete that key with + * VMKEY_WOW64_32KEY if you want a consistent view with 32-bit code. + */ + +#ifdef _WIN32 +#ifdef _WIN64 +#define VMW_KEY_WOW64_32KEY KEY_WOW64_32KEY +#else +#define VMW_KEY_WOW64_32KEY 0x0 +#endif +#endif + + +/* + * Consider the following reasons functions are inlined: + * + *  1) inlined for performance reasons + *  2) inlined because it's a single-use function + * + * Functions which meet only condition 2 should be marked with this + * inline macro; It is not critical to be inlined (but there is a + * code-space & runtime savings by doing so), so when other callers + * are added the inline-ness should be removed. + */ + +#if __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 3) +/* + * Starting at version 3.3, gcc does not always inline functions marked + * 'inline' (it depends on their size). To force gcc to do so, one must use the + * extra __always_inline__ attribute. + */ +#   define INLINE_SINGLE_CALLER INLINE __attribute__((__always_inline__)) +#else +#   define INLINE_SINGLE_CALLER INLINE +#endif + +/* + * Used when a hard guaranteed of no inlining is needed. Very few + * instances need this since the absence of INLINE is a good hint + * that gcc will not do inlining. + */ + +#if defined(__GNUC__) && defined(VMM) +#define ABSOLUTELY_NOINLINE __attribute__((__noinline__)) +#endif + +/* + * Attributes placed on function declarations to tell the compiler + * that the function never returns. + */ + +#ifdef _MSC_VER +#define NORETURN __declspec(noreturn) +#elif __GNUC__ > 2 || (__GNUC__ == 2 && __GNUC_MINOR__ >= 9) +#define NORETURN __attribute__((__noreturn__)) +#else +#define NORETURN +#endif + +/* + * GCC 3.2 inline asm needs the + constraint for input/ouput memory operands. + * Older GCCs don't know about it --hpreg + */ + +#if __GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 2) +#   define VM_ASM_PLUS 1 +#else +#   define VM_ASM_PLUS 0 +#endif + +/* + * Branch prediction hints: + *     LIKELY(exp)   - Expression exp is likely TRUE. + *     UNLIKELY(exp) - Expression exp is likely FALSE. + *   Usage example: + *        if (LIKELY(excCode == EXC_NONE)) { + *               or + *        if (UNLIKELY(REAL_MODE(vc))) { + * + * We know how to predict branches on gcc3 and later (hopefully), + * all others we don't so we do nothing. + */ + +#if (__GNUC__ >= 3) +/* + * gcc3 uses __builtin_expect() to inform the compiler of an expected value. + * We use this to inform the static branch predictor. The '!!' in LIKELY + * will convert any !=0 to a 1. + */ +#define LIKELY(_exp)     __builtin_expect(!!(_exp), 1) +#define UNLIKELY(_exp)   __builtin_expect((_exp), 0) +#else +#define LIKELY(_exp)      (_exp) +#define UNLIKELY(_exp)    (_exp) +#endif + +/* + * GCC's argument checking for printf-like functions + * This is conditional until we have replaced all `"%x", void *' + * with `"0x%08x", (uint32) void *'. Note that %p prints different things + * on different platforms.  Argument checking is enabled for the + * vmkernel, which has already been cleansed. + * + * fmtPos is the position of the format string argument, beginning at 1 + * varPos is the position of the variable argument, beginning at 1 + */ + +#if defined(__GNUC__) +# define PRINTF_DECL(fmtPos, varPos) __attribute__((__format__(__printf__, fmtPos, varPos))) +#else +# define PRINTF_DECL(fmtPos, varPos) +#endif + +#if defined(__GNUC__) +# define SCANF_DECL(fmtPos, varPos) __attribute__((__format__(__scanf__, fmtPos, varPos))) +#else +# define SCANF_DECL(fmtPos, varPos) +#endif + +/* + * UNUSED_PARAM should surround the parameter name and type declaration, + * e.g. "int MyFunction(int var1, UNUSED_PARAM(int var2))" + * + */ + +#ifndef UNUSED_PARAM +# if defined(__GNUC__) +#  define UNUSED_PARAM(_parm) _parm  __attribute__((__unused__)) +# else +#  define UNUSED_PARAM(_parm) _parm +# endif +#endif + +/* + * REGPARM defaults to REGPARM3; i.e., a request that gcc + * put the first three arguments in registers.  (It is fine + * if the function has fewer than three arguments.)  Gcc only. + * Syntactically, put REGPARM where you'd put INLINE or NORETURN. + * + * Note that 64-bit code already puts the first six arguments in + * registers, so these attributes are only useful for 32-bit code. + */ + +#if defined(__GNUC__) +# define REGPARM0 __attribute__((regparm(0))) +# define REGPARM1 __attribute__((regparm(1))) +# define REGPARM2 __attribute__((regparm(2))) +# define REGPARM3 __attribute__((regparm(3))) +# define REGPARM REGPARM3 +#else +# define REGPARM0 +# define REGPARM1 +# define REGPARM2 +# define REGPARM3 +# define REGPARM +#endif + +/* + * ALIGNED specifies minimum alignment in "n" bytes. + */ + +#ifdef __GNUC__ +#define ALIGNED(n) __attribute__((__aligned__(n))) +#else +#define ALIGNED(n) +#endif + +/* + * __func__ is a stringified function name that is part of the C99 standard. The block + * below defines __func__ on older systems where the compiler does not support that + * macro. + */ +#if defined(__GNUC__) \ +   && ((__GNUC__ == 2 && __GNUC_MINOR < 96) \ +       || (__GNUC__ < 2)) +#   define __func__ __FUNCTION__ +#endif + +/* + * Once upon a time, this was used to silence compiler warnings that + * get generated when the compiler thinks that a function returns + * when it is marked noreturn.  Don't do it.  Use NOT_REACHED(). + */ + +#define INFINITE_LOOP()           do { } while (1) + +/* + * On FreeBSD (for the tools build), size_t is typedef'd if _BSD_SIZE_T_ + * is defined. Use the same logic here so we don't define it twice. [greg] + */ +#ifdef __FreeBSD__ +#   ifdef _BSD_SIZE_T_ +#      undef _BSD_SIZE_T_ +#      ifdef VM_I386 +#         ifdef VM_X86_64 +             typedef uint64 size_t; +#         else +             typedef uint32 size_t; +#         endif +#      endif /* VM_I386 */ +#   endif + +#   ifdef _BSD_SSIZE_T_ +#      undef _BSD_SSIZE_T_ +#      ifdef VM_I386 +#         ifdef VM_X86_64 +             typedef int64 ssize_t; +#         else +             typedef int32 ssize_t; +#         endif +#      endif /* VM_I386 */ +#   endif + +#else +#   ifndef _SIZE_T +#      ifdef VM_I386 +#         define _SIZE_T +#         ifdef VM_X86_64 +             typedef uint64 size_t; +#         else +             typedef uint32 size_t; +#         endif +#      elif defined(__arm__) +#         define _SIZE_T +          typedef uint32 size_t; +#      endif +#   endif + +#   if !defined(FROBOS) && !defined(_SSIZE_T) && !defined(_SSIZE_T_) && \ +       !defined(ssize_t) && !defined(__ssize_t_defined) && \ +       !defined(_SSIZE_T_DECLARED) +#      ifdef VM_I386 +#         define _SSIZE_T +#         define __ssize_t_defined +#         define _SSIZE_T_DECLARED +#         ifdef VM_X86_64 +             typedef int64 ssize_t; +#         else +             typedef int32 ssize_t; +#         endif +#      elif defined(__arm__) +#         define _SSIZE_T +#         define __ssize_t_defined +#         define _SSIZE_T_DECLARED +          typedef int32 ssize_t; +#      endif +#   endif + +#endif + +/* + * Format modifier for printing pid_t.  On sun the pid_t is a ulong, but on + * Linux it's an int. + * Use this like this: printf("The pid is %"FMTPID".\n", pid); + */ +#ifdef sun +#   ifdef VM_X86_64 +#      define FMTPID "d" +#   else +#      define FMTPID "lu" +#   endif +#else +# define FMTPID "d" +#endif + +/* + * Format modifier for printing uid_t.  On Solaris 10 and earlier, uid_t + * is a ulong, but on other platforms it's an unsigned int. + * Use this like this: printf("The uid is %"FMTUID".\n", uid); + */ +#if defined(sun) && !defined(SOL11) +#   ifdef VM_X86_64 +#      define FMTUID "u" +#   else +#      define FMTUID "lu" +#   endif +#else +# define FMTUID "u" +#endif + +/* + * Format modifier for printing mode_t.  On sun the mode_t is a ulong, but on + * Linux it's an int. + * Use this like this: printf("The mode is %"FMTMODE".\n", mode); + */ +#ifdef sun +#   ifdef VM_X86_64 +#      define FMTMODE "o" +#   else +#      define FMTMODE "lo" +#   endif +#else +# define FMTMODE "o" +#endif + +/* + * Format modifier for printing time_t. Most platforms define a time_t to be + * a long int, but on FreeBSD (as of 5.0, it seems), the time_t is a signed + * size quantity. Refer to the definition of FMTSZ to see why we need silly + * preprocessor arithmetic. + * Use this like this: printf("The mode is %"FMTTIME".\n", time); + */ +#if defined(__FreeBSD__) && (__FreeBSD__ + 0) && ((__FreeBSD__ + 0) >= 5) +#   define FMTTIME FMTSZ"d" +#else +#   if defined(_MSC_VER) +#      ifndef _SAFETIME_H_ +#         if (_MSC_VER < 1400) || defined(_USE_32BIT_TIME_T) +#             define FMTTIME "ld" +#         else +#             define FMTTIME FMT64"d" +#         endif +#      else +#         ifndef FMTTIME +#            error "safetime.h did not define FMTTIME" +#         endif +#      endif +#   else +#      define FMTTIME "ld" +#   endif +#endif + +#ifdef __APPLE__ +/* + * Format specifier for all these annoying types such as {S,U}Int32 + * which are 'long' in 32-bit builds + *       and  'int' in 64-bit builds. + */ +#   ifdef __LP64__ +#      define FMTLI "" +#   else +#      define FMTLI "l" +#   endif + +/* + * Format specifier for all these annoying types such as NS[U]Integer + * which are  'int' in 32-bit builds + *       and 'long' in 64-bit builds. + */ +#   ifdef __LP64__ +#      define FMTIL "l" +#   else +#      define FMTIL "" +#   endif +#endif + + +/* + * Define MXSemaHandle here so both vmmon and vmx see this definition. + */ + +#ifdef _WIN32 +typedef uintptr_t MXSemaHandle; +#else +typedef int MXSemaHandle; +#endif + +/* + * Define type for poll device handles. + */ + +typedef int64 PollDevHandle; + +/* + * Define the utf16_t type. + */ + +#if defined(_WIN32) && defined(_NATIVE_WCHAR_T_DEFINED) +typedef wchar_t utf16_t; +#else +typedef uint16 utf16_t; +#endif + +/* + * Define for point and rectangle types.  Defined here so they + * can be used by other externally facing headers in bora/public. + */ + +typedef struct VMPoint { +   int x, y; +} VMPoint; + +#if defined _WIN32 && defined USERLEVEL +struct tagRECT; +typedef struct tagRECT VMRect; +#else +typedef struct VMRect { +   int left; +   int top; +   int right; +   int bottom; +} VMRect; +#endif + +/* + * ranked locks "everywhere" + */ + +typedef uint32 MX_Rank; + +#endif  /* _VM_BASIC_TYPES_H_ */ diff --git a/usr/src/uts/intel/io/vmxnet/vm_device_version.h b/usr/src/uts/intel/io/vmxnet/vm_device_version.h new file mode 100644 index 0000000000..7046594a6c --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vm_device_version.h @@ -0,0 +1,246 @@ +/********************************************************* + * Copyright (C) 1998 VMware, Inc. All rights reserved. + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as published + * by the Free Software Foundation version 2.1 and no 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 Lesser GNU General Public + * License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA. + * + *********************************************************/ + +#ifndef VM_DEVICE_VERSION_H +#define VM_DEVICE_VERSION_H + +#define INCLUDE_ALLOW_USERLEVEL + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_VMCORE +#include "includeCheck.h" + +#ifdef _WIN32 +#ifdef __MINGW32__ +#include "initguid.h" +#else +#include "guiddef.h" +#endif +#endif + +/* LSILogic 53C1030 Parallel SCSI controller + * LSILogic SAS1068 SAS controller + */ +#define PCI_VENDOR_ID_LSILOGIC          0x1000 +#define PCI_DEVICE_ID_LSI53C1030        0x0030 +#define PCI_DEVICE_ID_LSISAS1068        0x0054 + +/* Our own PCI IDs + *    VMware SVGA II (Unified VGA) + *    VMware SVGA (PCI Accelerator) + *    VMware vmxnet (Idealized NIC) + *    VMware vmxscsi (Abortive idealized SCSI controller) + *    VMware chipset (Subsystem ID for our motherboards) + *    VMware e1000 (Subsystem ID) + *    VMware vmxnet3 (Uniform Pass Through NIC) + *    VMware HD Audio codec + *    VMware HD Audio controller + */ +#define PCI_VENDOR_ID_VMWARE                    0x15AD +#define PCI_DEVICE_ID_VMWARE_SVGA2              0x0405 +#define PCI_DEVICE_ID_VMWARE_SVGA               0x0710 +#define PCI_DEVICE_ID_VMWARE_NET                0x0720 +#define PCI_DEVICE_ID_VMWARE_SCSI               0x0730 +#define PCI_DEVICE_ID_VMWARE_VMCI               0x0740 +#define PCI_DEVICE_ID_VMWARE_CHIPSET            0x1976 +#define PCI_DEVICE_ID_VMWARE_82545EM            0x0750 /* single port */ +#define PCI_DEVICE_ID_VMWARE_82546EB            0x0760 /* dual port   */ +#define PCI_DEVICE_ID_VMWARE_EHCI               0x0770 +#define PCI_DEVICE_ID_VMWARE_UHCI               0x0774 +#define PCI_DEVICE_ID_VMWARE_XHCI               0x0778 +#define PCI_DEVICE_ID_VMWARE_1394               0x0780 +#define PCI_DEVICE_ID_VMWARE_BRIDGE             0x0790 +#define PCI_DEVICE_ID_VMWARE_ROOTPORT           0x07A0 +#define PCI_DEVICE_ID_VMWARE_VMXNET3            0x07B0 +#define PCI_DEVICE_ID_VMWARE_VMXWIFI            0x07B8 +#define PCI_DEVICE_ID_VMWARE_PVSCSI             0x07C0 +#define PCI_DEVICE_ID_VMWARE_82574              0x07D0 +#define PCI_DEVICE_ID_VMWARE_HDAUDIO_CODEC      0x1975 +#define PCI_DEVICE_ID_VMWARE_HDAUDIO_CONTROLLER 0x1977 + +/* The hypervisor device might grow.  Please leave room + * for 7 more subfunctions. + */ +#define PCI_DEVICE_ID_VMWARE_HYPER      0x0800 +#define PCI_DEVICE_ID_VMWARE_VMI        0x0801 + +#define PCI_DEVICE_VMI_CLASS            0x05 +#define PCI_DEVICE_VMI_SUBCLASS         0x80 +#define PCI_DEVICE_VMI_INTERFACE        0x00 +#define PCI_DEVICE_VMI_REVISION         0x01 + +/* From linux/pci_ids.h: + *   AMD Lance Ethernet controller + *   BusLogic SCSI controller + *   Ensoniq ES1371 sound controller + */ +#define PCI_VENDOR_ID_AMD               0x1022 +#define PCI_DEVICE_ID_AMD_VLANCE        0x2000 +#define PCI_VENDOR_ID_BUSLOGIC			0x104B +#define PCI_DEVICE_ID_BUSLOGIC_MULTIMASTER_NC	0x0140 +#define PCI_DEVICE_ID_BUSLOGIC_MULTIMASTER	0x1040 +#define PCI_VENDOR_ID_ENSONIQ           0x1274 +#define PCI_DEVICE_ID_ENSONIQ_ES1371    0x1371 + +/* From linux/pci_ids.h: + *    Intel 82439TX (430 HX North Bridge) + *    Intel 82371AB (PIIX4 South Bridge) + *    Intel 82443BX (440 BX North Bridge and AGP Bridge) + *    Intel 82545EM (e1000, server adapter, single port) + *    Intel 82546EB (e1000, server adapter, dual port) + *    Intel HECI (as embedded in ich9m) + */ +#define PCI_VENDOR_ID_INTEL             0x8086 +#define PCI_DEVICE_ID_INTEL_82439TX     0x7100 +#define PCI_DEVICE_ID_INTEL_82371AB_0   0x7110 +#define PCI_DEVICE_ID_INTEL_82371AB_2   0x7112 +#define PCI_DEVICE_ID_INTEL_82371AB_3   0x7113 +#define PCI_DEVICE_ID_INTEL_82371AB     0x7111 +#define PCI_DEVICE_ID_INTEL_82443BX     0x7190 +#define PCI_DEVICE_ID_INTEL_82443BX_1   0x7191 +#define PCI_DEVICE_ID_INTEL_82443BX_2   0x7192 /* Used when no AGP support */ +#define PCI_DEVICE_ID_INTEL_82545EM     0x100f +#define PCI_DEVICE_ID_INTEL_82546EB     0x1010 +#define PCI_DEVICE_ID_INTEL_82574       0x10d3 +#define PCI_DEVICE_ID_INTEL_82574_APPLE 0x10f6 +#define PCI_DEVICE_ID_INTEL_HECI        0x2a74 + +#define E1000E_PCI_DEVICE_ID_CONFIG_STR "e1000e.pci.deviceID" +#define E1000E_PCI_SUB_VENDOR_ID_CONFIG_STR "e1000e.pci.subVendorID" +#define E1000E_PCI_SUB_DEVICE_ID_CONFIG_STR "e1000e.pci.subDeviceID" + +/* + * Intel HD Audio controller and Realtek ALC885 codec. + */ +#define PCI_DEVICE_ID_INTEL_631XESB_632XESB  0x269a +#define PCI_VENDOR_ID_REALTEK                0x10ec +#define PCI_DEVICE_ID_REALTEK_ALC885         0x0885 + + +/* + * Fresco Logic xHCI (USB 3.0) Controller + */ +#define PCI_VENDOR_ID_FRESCO            0x1B73 +#define PCI_DEVICE_ID_FRESCO_FL1000     0x1000   // Original 1-port chip +#define PCI_DEVICE_ID_FRESCO_FL1009     0x1009   // New 2-port chip (Driver 3.0.98+) +#define PCI_DEVICE_ID_FRESCO_FL1400     0x1400   // Unknown (4-port? Dev hardware?) + +/* + * NEC/Renesas xHCI (USB 3.0) Controller + */ +#define PCI_VENDOR_ID_NEC               0x1033 +#define PCI_DEVICE_ID_NEC_UPD720200     0x0194 +#define PCI_REVISION_NEC_UPD720200      0x03 +#define PCI_FIRMWARE_NEC_UPD720200      0x3015 + + +/************* Strings for IDE Identity Fields **************************/ +#define VIDE_ID_SERIAL_STR	"00000000000000000001"	/* Must be 20 Bytes */ +#define VIDE_ID_FIRMWARE_STR	"00000001"		/* Must be 8 Bytes */ + +/* No longer than 40 Bytes */ +#define VIDE_ATA_MODEL_STR PRODUCT_GENERIC_NAME " Virtual IDE Hard Drive" +#define VIDE_ATAPI_MODEL_STR PRODUCT_GENERIC_NAME " Virtual IDE CDROM Drive" + +#define ATAPI_VENDOR_ID	"NECVMWar"		/* Must be 8 Bytes */ +#define ATAPI_PRODUCT_ID PRODUCT_GENERIC_NAME " IDE CDROM"	/* Must be 16 Bytes */ +#define ATAPI_REV_LEVEL	"1.00"			/* Must be 4 Bytes */ + +#define IDE_NUM_INTERFACES   2	/* support for two interfaces */ +#define IDE_DRIVES_PER_IF    2 + +/************* Strings for SCSI Identity Fields **************************/ +#define SCSI_DISK_MODEL_STR PRODUCT_GENERIC_NAME " Virtual SCSI Hard Drive" +#define SCSI_DISK_VENDOR_NAME COMPANY_NAME +#define SCSI_DISK_REV_LEVEL "1.0" +#define SCSI_CDROM_MODEL_STR PRODUCT_GENERIC_NAME " Virtual SCSI CDROM Drive" +#define SCSI_CDROM_VENDOR_NAME COMPANY_NAME +#define SCSI_CDROM_REV_LEVEL "1.0" + +/************* SCSI implementation limits ********************************/ +#define SCSI_MAX_CONTROLLERS	 4	  // Need more than 1 for MSCS clustering +#define	SCSI_MAX_DEVICES	 16	  // BT-958 emulates only 16 +#define PVSCSI_MAX_DEVICES       255      // 255 (including the controller) +/* + * VSCSI_BV_INTS is the number of uint32's needed for a bit vector  + * to cover all scsi devices per target. + */ +#define VSCSI_BV_INTS            CEILING(PVSCSI_MAX_DEVICES, 8 * sizeof (uint32)) +#define SCSI_IDE_CHANNEL         SCSI_MAX_CONTROLLERS +#define SCSI_IDE_HOSTED_CHANNEL  (SCSI_MAX_CONTROLLERS + 1) +#define SCSI_MAX_CHANNELS        (SCSI_MAX_CONTROLLERS + 2) + +/************* Strings for the VESA BIOS Identity Fields *****************/ +#define VBE_OEM_STRING COMPANY_NAME " SVGA" +#define VBE_VENDOR_NAME COMPANY_NAME +#define VBE_PRODUCT_NAME PRODUCT_GENERIC_NAME + +/************* PCI implementation limits ********************************/ +#define PCI_MAX_BRIDGES         15 + +/************* Ethernet implementation limits ***************************/ +#define MAX_ETHERNET_CARDS      10 + +/********************** Floppy limits ***********************************/ +#define MAX_FLOPPY_DRIVES      2 + +/************* PCI Passthrough implementation limits ********************/ +#define MAX_PCI_PASSTHRU_DEVICES 6 + +/************* USB implementation limits ********************************/ +#define MAX_USB_DEVICES_PER_HOST_CONTROLLER 127 + +/************* Strings for Host USB Driver *******************************/ + +#ifdef _WIN32 + +/* + * Globally unique ID for the VMware device interface. Define INITGUID before including + * this header file to instantiate the variable. + */ +DEFINE_GUID(GUID_DEVICE_INTERFACE_VMWARE_USB_DEVICES,  +0x2da1fe75, 0xaab3, 0x4d2c, 0xac, 0xdf, 0x39, 0x8, 0x8c, 0xad, 0xa6, 0x65); + +/* + * Globally unique ID for the VMware device setup class. + */ +DEFINE_GUID(GUID_CLASS_VMWARE_USB_DEVICES,  +0x3b3e62a5, 0x3556, 0x4d7e, 0xad, 0xad, 0xf5, 0xfa, 0x3a, 0x71, 0x2b, 0x56); + +/* + * This string defines the device ID string of a VMware USB device. + * The format is USB\Vid_XXXX&Pid_YYYY, where XXXX and YYYY are the + * hexadecimal representations of the vendor and product ids, respectively. + * + * The official vendor ID for VMware, Inc. is 0x0E0F. + * The product id for USB generic devices is 0x0001. + */ +#define USB_VMWARE_DEVICE_ID_WIDE L"USB\\Vid_0E0F&Pid_0001" +#define USB_DEVICE_ID_LENGTH (sizeof(USB_VMWARE_DEVICE_ID_WIDE) / sizeof(WCHAR)) + +#ifdef UNICODE +#define USB_PNP_SETUP_CLASS_NAME L"VMwareUSBDevices" +#define USB_PNP_DRIVER_NAME L"vmusb" +#else +#define USB_PNP_SETUP_CLASS_NAME "VMwareUSBDevices" +#define USB_PNP_DRIVER_NAME "vmusb" +#endif +#endif + +#endif /* VM_DEVICE_VERSION_H */ diff --git a/usr/src/uts/intel/io/vmxnet/vmnet_def.h b/usr/src/uts/intel/io/vmxnet/vmnet_def.h new file mode 100644 index 0000000000..6e44aea2bb --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vmnet_def.h @@ -0,0 +1,91 @@ +/********************************************************* + * Copyright (C) 2004 VMware, Inc. All rights reserved. + * + * 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 version 2 and no 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. + * + * 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 St, Fifth Floor, Boston, MA  02110-1301 USA + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/* + * vmnet_def.h  + * + *     - definitions which are (mostly) not vmxnet or vlance specific + */ + +#ifndef _VMNET_DEF_H_ +#define _VMNET_DEF_H_ + +#define INCLUDE_ALLOW_USERLEVEL +#define INCLUDE_ALLOW_VMCORE + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMK_MODULE +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_DISTRIBUTE +#include "includeCheck.h" + +#define VMNET_NAME_BUFFER_LEN  128 /* Increased for i18n. */ +#define VMNET_COAL_SCHEME_NAME_LEN 16 + + +/* + * capabilities - not all of these are implemented in the virtual HW + *                (eg VLAN support is in the virtual switch)  so even vlance  + *                can use them + */ +#define VMNET_CAP_SG	        0x0001	/* Can do scatter-gather transmits. */ +#define VMNET_CAP_IP4_CSUM      0x0002	/* Can checksum only TCP/UDP over IPv4. */ +#define VMNET_CAP_HW_CSUM       0x0004	/* Can checksum all packets. */ +#define VMNET_CAP_HIGH_DMA      0x0008	/* Can DMA to high memory. */ +#define VMNET_CAP_TOE	        0x0010	/* Supports TCP/IP offload. */ +#define VMNET_CAP_TSO	        0x0020	/* Supports TCP Segmentation offload */ +#define VMNET_CAP_SW_TSO        0x0040	/* Supports SW TCP Segmentation */ +#define VMNET_CAP_VMXNET_APROM  0x0080	/* Vmxnet APROM support */ +#define VMNET_CAP_HW_TX_VLAN    0x0100  /* Can we do VLAN tagging in HW */ +#define VMNET_CAP_HW_RX_VLAN    0x0200  /* Can we do VLAN untagging in HW */ +#define VMNET_CAP_SW_VLAN       0x0400  /* Can we do VLAN tagging/untagging in SW */ +#define VMNET_CAP_WAKE_PCKT_RCV 0x0800  /* Can wake on network packet recv? */ +#define VMNET_CAP_ENABLE_INT_INLINE 0x1000  /* Enable Interrupt Inline */ +#define VMNET_CAP_ENABLE_HEADER_COPY 0x2000  /* copy header for vmkernel */ +#define VMNET_CAP_TX_CHAIN      0x4000  /* Guest can use multiple tx entries for a pkt */ +#define VMNET_CAP_RX_CHAIN      0x8000  /* a pkt can span multiple rx entries */ +#define VMNET_CAP_LPD           0x10000 /* large pkt delivery */ +#define VMNET_CAP_BPF           0x20000 /* BPF Support in VMXNET Virtual Hardware */ +#define VMNET_CAP_SG_SPAN_PAGES 0x40000	/* Can do scatter-gather span multiple pages transmits. */ +#define VMNET_CAP_IP6_CSUM      0x80000	/* Can do IPv6 csum offload. */ +#define VMNET_CAP_TSO6         0x100000	/* Can do TSO segmentation offload for IPv6 pkts. */ +#define VMNET_CAP_TSO256k      0x200000	/* Can do TSO segmentation offload for pkts up to 256kB. */ +#define VMNET_CAP_UPT          0x400000	/* Support UPT */ +#define VMNET_CAP_RDONLY_INETHDRS 0x800000 /* Modifies inet headers for TSO/CSUm */ +#define VMNET_CAP_NPA         0x1000000	/* Support NPA */ +#define VMNET_CAP_DCB         0x2000000	/* Support DCB */ +#define VMNET_CAP_OFFLOAD_8OFFSET 0x4000000 /* supports 8bit parameterized offsets */  +#define VMNET_CAP_OFFLOAD_16OFFSET 0x8000000 /* supports 16bit parameterized offsets */  +#define VMNET_CAP_IP6_CSUM_EXT_HDRS 0x10000000 /* support csum of ip6 ext hdrs */ +#define VMNET_CAP_TSO6_EXT_HDRS 0x20000000 /* support TSO for ip6 ext hdrs */ +#define VMNET_CAP_SCHED       0x40000000 /* compliant with network scheduling */ +#endif // _VMNET_DEF_H_ diff --git a/usr/src/uts/intel/io/vmxnet/vmxnet.c b/usr/src/uts/intel/io/vmxnet/vmxnet.c new file mode 100644 index 0000000000..e170f049d9 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vmxnet.c @@ -0,0 +1,2442 @@ +/********************************************************* + * Copyright (C) 2004 VMware, Inc. All rights reserved. + * + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +/* + * Copyright 2018 Joyent, Inc. + */ + +#include <sys/types.h> +#include <sys/conf.h> +#include <sys/debug.h> +#include <sys/stropts.h> +#include <sys/stream.h> +#include <sys/strlog.h> +#include <sys/kmem.h> +#include <sys/stat.h> +#include <sys/kstat.h> +#include <sys/vtrace.h> +#include <sys/dlpi.h> +#include <sys/strsun.h> +#include <sys/ethernet.h> +#include <sys/modctl.h> +#include <sys/errno.h> +#include <sys/ddi.h> +#include <sys/sunddi.h> +#include <sys/ddi_impldefs.h> +#include <sys/gld.h> +#include <sys/pci.h> +#include <sys/strsubr.h> + +/* + * This used to be defined in sys/gld.h, but was flagged as private, + * and we used it anyway.  Now it no longer exists, and we're stuck + * with it for the time being. + */ +#ifndef GLD_MAX_MULTICAST +#define GLD_MAX_MULTICAST 64 +#endif + +#define __intptr_t_defined +#define _STDINT_H +#include "vm_basic_types.h" +#include "vmxnet2_def.h" +#include "vm_device_version.h" +#include "net.h" +#include "buildNumber.h" + +#define SOLVMXNET_SUCCESS 1 +#define SOLVMXNET_FAILURE 0 + +#ifdef SOLVMXNET_DEBUG_LEVEL +static int vxn_debug = SOLVMXNET_DEBUG_LEVEL; +#define  DPRINTF(n, args)  if (vxn_debug>(n)) cmn_err args +#else +#define  DPRINTF(n, args) +#endif + +static char ident[] = "VMware Ethernet Adapter b" BUILD_NUMBER_NUMERIC_STRING; +char _depends_on[] = {"misc/gld"}; + +#define MAX_NUM_RECV_BUFFERS            128 +#define DEFAULT_NUM_RECV_BUFFERS        100 +#define MAX_NUM_XMIT_BUFFERS            128 +#define DEFAULT_NUM_XMIT_BUFFERS        100 +#define CRC_POLYNOMIAL_LE               0xedb88320UL +#define SOLVMXNET_MAXNAME               20 +#define MAX_TX_WAIT_ON_STOP             2000 + +#define ETHERALIGN  2 +#define SLACKBYTES  4 +#define MAXPKTBUF   (14 + ETHERALIGN + ETHERMTU + SLACKBYTES) + + +#define QHIWATER (MAX_NUM_RECV_BUFFERS*ETHERMTU) + +#define OUTB(dp, p, v)  \ +        ddi_put8((dp)->vxnIOHdl, \ +                (uint8_t *)((caddr_t)((dp)->vxnIOp) + (p)), v) +#define OUTW(dp, p, v)  \ +        ddi_put16((dp)->vxnIOHdl, \ +                (uint16_t *)((caddr_t)((dp)->vxnIOp) + (p)), v) +#define OUTL(dp, p, v)  \ +        ddi_put32((dp)->vxnIOHdl, \ +                (uint32_t *)((caddr_t)((dp)->vxnIOp) + (p)), v) +#define INB(dp, p)      \ +        ddi_get8((dp)->vxnIOHdl, \ +                (uint8_t *)(((caddr_t)(dp)->vxnIOp) + (p))) +#define INW(dp, p)      \ +        ddi_get16((dp)->vxnIOHdl, \ +                (uint16_t *)(((caddr_t)(dp)->vxnIOp) + (p))) +#define INL(dp, p)      \ +        ddi_get32((dp)->vxnIOHdl, \ +                (uint32_t *)(((caddr_t)(dp)->vxnIOp) + (p))) + +#define VMXNET_INC(val, max) \ +   val++; \ +   if (UNLIKELY(val == max)) { \ +      val = 0; \ +   } + +#define TX_RINGBUF_MBLK(dp, idx) (dp->txRingBuf[idx].mblk) +#define TX_RINGBUF_DMAMEM(dp, idx) (dp->txRingBuf[idx].dmaMem) + +typedef struct { +   caddr_t           buf;             /* Virtual address */ +   uint32_t          phyBuf;          /* Physical address */ +   size_t            bufLen;          /* Buffer length */ +   ddi_dma_cookie_t  cookie;          /* Dma cookie */ +   uint_t            cookieCount;     /* Cookie count */ +   ddi_dma_handle_t  dmaHdl;          /* Dma handle */  +   ddi_acc_handle_t  dataAccHdl;      /* Dada access handle */ +} dma_buf_t; + +typedef struct rx_dma_buf { +   dma_buf_t           dmaDesc;       /* Dma descriptor */ +   mblk_t              *mblk;         /* Streams message block */ +   frtn_t              freeCB;        /* Free callback */ +   struct vxn_softc    *softc;        /* Back pointer to softc */ +   struct rx_dma_buf   *next;         /* Next one in list */ +} rx_dma_buf_t; + +typedef struct vxn_stats { +   uint32_t    errxmt;               /* Transmit errors */ +   uint32_t    errrcv;               /* Receive errors */ +   uint32_t    runt;                 /* Runt packets */ +   uint32_t    norcvbuf;             /* Buffer alloc errors */ +   uint32_t    interrupts;	     /* Interrupts */ +   uint32_t    defer;		     /* Deferred transmits */ +} vxn_stats_t; + +typedef struct tx_ring_buf { +   mblk_t      *mblk; +   dma_buf_t   dmaMem; +} tx_ring_buf_t; + +typedef struct vxn_softc { +   char                    drvName[SOLVMXNET_MAXNAME]; /* Driver name string */ +   int                     unit;               /* Driver instance */ +   vxn_stats_t             stats;              /* Stats */ + +   dev_info_t              *dip;               /* Info pointer */ +   ddi_iblock_cookie_t     iblockCookie;       /* Interrupt block cookie */ +   gld_mac_info_t          *macInfo;           /* GLD mac info */ +   ddi_acc_handle_t        confHdl;            /* Configuration space handle */ +   ddi_acc_handle_t        vxnIOHdl;           /* I/O space handle */ +   caddr_t                 vxnIOp;             /* I/O space pointer */ +   boolean_t               morphed;            /* Adapter morphed ? */ + +   kmutex_t                intrlock;           /* Interrupt lock */ +   kmutex_t                xmitlock;           /* Transmit lock */ +   kmutex_t                rxlistlock;         /* Rx free pool lock */ + +   boolean_t               nicActive;          /* NIC active flag */ +   boolean_t               inIntr;             /* Interrupt processing flag */ + +   struct ether_addr       devAddr;            /* MAC address */ + +   uint32_t                vxnNumRxBufs;       /* Number of reveice buffers */ +   uint32_t                vxnNumTxBufs;       /* Number of transmit buffers */ + +   dma_buf_t               driverDataDmaMem;   /* Driver Data (dma handle) */ +   Vmxnet2_DriverData      *driverData;        /* Driver Data */ +   void                    *driverDataPhy;     /* Driver Data busaddr pointer */ +   Vmxnet2_RxRingEntry     *rxRing;            /* Receive ring */ +   Vmxnet2_TxRingEntry     *txRing;            /* Transmit ring */ +   ddi_dma_handle_t        txDmaHdl;           /* Tx buffers dma handle */ +   rx_dma_buf_t            *rxRingBuffPtr[MAX_NUM_RECV_BUFFERS]; +                                               /* DMA buffers associated with rxRing */ +   tx_ring_buf_t           txRingBuf[MAX_NUM_XMIT_BUFFERS]; /* tx Ring buffers */ + +   rx_dma_buf_t            *rxFreeBufList; +   uint32_t                rxNumFreeBufs;      /* current # of buffers in pool */ +   uint32_t                rxMaxFreeBufs;      /* max # of buffers in pool */ + +   uint32_t                txPending;          /* Pending transmits */ +   uint32_t                maxTxFrags;         /* Max Tx fragments */ + +   int                     multiCount;         /* Multicast address count */ +   struct ether_addr       multicastList[GLD_MAX_MULTICAST]; /* Multicast list */ + +   struct vxn_softc        *next;              /* Circular list of instances */ +   struct vxn_softc        *prev; +} vxn_softc_t; + +/* used for rx buffers or buffers allocated by ddi_dma_mem_alloc() */ +static ddi_dma_attr_t vxn_dma_attrs = { +        DMA_ATTR_V0,            /* dma_attr version */ +        0,                      /* dma_attr_addr_lo */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_addr_hi */ +        0x7FFFFFFF,             /* dma_attr_count_max */ +        4,                      /* dma_attr_align */ +        0x3F,                   /* dma_attr_burstsizes */ +        1,                      /* dma_attr_minxfer */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_maxxfer */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_seg */ +        1,                      /* dma_attr_sgllen */ +        1,                      /* dma_attr_granular */ +        0,                      /* dma_attr_flags */ +}; + +/* used for tx buffers */ +static ddi_dma_attr_t vxn_dma_attrs_tx = { +        DMA_ATTR_V0,            /* dma_attr version */ +        0,                      /* dma_attr_addr_lo */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_addr_hi */ +        0x7FFFFFFF,             /* dma_attr_count_max */ +        1,                      /* dma_attr_align */ +        0x3F,                   /* dma_attr_burstsizes */ +        1,                      /* dma_attr_minxfer */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_maxxfer */ +        (uint64_t)0xFFFFFFFF,   /* dma_attr_seg */ +        1,                      /* dma_attr_sgllen */ +        1,                      /* dma_attr_granular */ +        0,                      /* dma_attr_flags */ +}; + + +static   struct ether_addr etherbroadcastaddr = { +   {0xff, 0xff, 0xff, 0xff, 0xff, 0xff} +}; + +static struct ddi_device_acc_attr vxn_buf_attrs = { +        DDI_DEVICE_ATTR_V0, +        DDI_STRUCTURE_LE_ACC, +        DDI_STRICTORDER_ACC +}; + +static struct ddi_device_acc_attr dev_attr = { +        DDI_DEVICE_ATTR_V0, +        DDI_STRUCTURE_LE_ACC, +        DDI_STRICTORDER_ACC +}; + +static vxn_softc_t vxnList;		/* for debugging */ +static kmutex_t vxnListLock; + +static void *Vxn_Memset(void *s, int c, size_t n); +static int Vxn_Reset(gld_mac_info_t *macInfo); +static int Vxn_SetPromiscuous(gld_mac_info_t *macInfo, int flag); +static int Vxn_GetStats(gld_mac_info_t *macInfo, struct gld_stats *gs); +static void Vxn_ApplyAddressFilter(vxn_softc_t *dp); +static int Vxn_SetMulticast(gld_mac_info_t *macinfo, uint8_t *ep, int flag); +static int Vxn_SetMacAddress(gld_mac_info_t *macInfo, uint8_t *mac); +static int Vxn_Start(gld_mac_info_t *macInfo); +static int Vxn_Stop(gld_mac_info_t *macInfo); +static void Vxn_FreeTxBuf(vxn_softc_t *dp, int idx); +static int Vxn_EncapTxBuf(vxn_softc_t *dp, mblk_t *mp, Vmxnet2_TxRingEntry *xre, +                          tx_ring_buf_t *txBuf); +static int Vxn_Send(gld_mac_info_t *macinfo, mblk_t *mp); +static boolean_t Vxn_TxComplete(vxn_softc_t *dp, boolean_t *reschedp); +static boolean_t Vxn_Receive(vxn_softc_t *dp); +static u_int Vxn_Interrupt(gld_mac_info_t *macInfo); +static void Vxn_ReclaimRxBuf(rx_dma_buf_t *rxDesc); +static void Vxn_FreeRxBuf(rx_dma_buf_t *rxDesc); +static rx_dma_buf_t *Vxn_AllocRxBuf(vxn_softc_t *dp, int cansleep); +static void Vxn_FreeInitBuffers(vxn_softc_t *dp); +static int Vxn_AllocInitBuffers(vxn_softc_t *dp); +static void Vxn_FreeDmaMem(dma_buf_t *dma); +static int Vxn_AllocDmaMem(vxn_softc_t *dp, int size, int cansleep, dma_buf_t *dma); +static void Vxn_FreeDriverData(vxn_softc_t *dp); +static int Vxn_AllocDriverData(vxn_softc_t *dp); +static int Vxn_Attach(dev_info_t *dip, ddi_attach_cmd_t cmd); +static int Vxn_Detach(dev_info_t *dip, ddi_detach_cmd_t cmd); +static int Vxn_AllocRxBufPool(vxn_softc_t *dp); +static void Vxn_FreeRxBufPool(vxn_softc_t *dp); +static rx_dma_buf_t * Vxn_AllocRxBufFromPool(vxn_softc_t *dp); +static void Vxn_FreeRxBufToPool(rx_dma_buf_t *rxDesc); + +/* + *----------------------------------------------------------------------------- + * Vxn_Memset -- + *    memset() (Because bzero does not get resolved by module loader)  + * + * Results: + *    pointer to the memory area s + * + * Side effects: + *    None   + *----------------------------------------------------------------------------- + */ +static void * +Vxn_Memset(void *s, int c, size_t n) +{ +   while (n--) { +      ((uint8_t *)s)[n] = c; +   } + +   return s; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Reset --  + *    Stub routine to reset hardware. Presently does nothing. Start/Stop should + *    take care of resets. + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_Reset(gld_mac_info_t *macInfo) +{ +   return GLD_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_SetPromiscuous -- + *    Set/Reset NIC to/from promiscuous mode + * + * Results: + *    GLD_SUCCESS + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_SetPromiscuous(gld_mac_info_t *macInfo, int flag) +{ +   vxn_softc_t *dp = (vxn_softc_t *)macInfo->gldm_private; +   Vmxnet2_DriverData *dd = dp->driverData; + +   mutex_enter(&dp->intrlock); +   if (flag == GLD_MAC_PROMISC_PHYS) { +      dd->ifflags |= VMXNET_IFF_PROMISC; +   } else if (flag == GLD_MAC_PROMISC_MULTI) { +     /* +      * This should really set VMXNET_IFF_ALLMULTI, +      * but unfortunately it doesn't exist.  The next +      * best thing would be to set the LADRFs to all +      * 0xFFs and set VMXNET_IFF_MULTICAST, but that +      * opens up a whole new set of potential pitfalls, +      * so this is a reasonable temporary solution. +      */ +      dd->ifflags |= VMXNET_IFF_PROMISC; +   } else if (flag == GLD_MAC_PROMISC_NONE) { +      dd->ifflags &= ~VMXNET_IFF_PROMISC; +   } else { +     /* This could be GLD_MAC_PROMISC_NOOP? */ +     mutex_exit(&dp->intrlock); +     cmn_err(CE_WARN, "%s%d: Vxn_SetPromiscuous: Unexpected mode flag: 0x%x",  +             dp->drvName, dp->unit, flag); + +     return GLD_FAILURE; +   } + +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_UPDATE_IFF); +   mutex_exit(&dp->intrlock); + +   return GLD_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_GetStats -- + *    Get driver specific stats + * + * Results: + *    GLD_SUCCESS + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_GetStats(gld_mac_info_t *macInfo, struct gld_stats *gs) +{ +   vxn_softc_t *dp = (vxn_softc_t *)macInfo->gldm_private; + +   gs->glds_errxmt    = dp->stats.errxmt; +   gs->glds_errrcv    = dp->stats.errrcv; +   gs->glds_short     = dp->stats.runt; +   gs->glds_norcvbuf  = dp->stats.norcvbuf; +   gs->glds_intr      = dp->stats.interrupts; +   gs->glds_defer     = dp->stats.defer; + +   return GLD_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_ApplyAddressFilter -- + *    Go over multicast list and compute/apply address filter + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_ApplyAddressFilter(vxn_softc_t *dp) +{ +   uint8_t *ep; +   int i, j, bit, byte; +   uint32_t crc, poly = CRC_POLYNOMIAL_LE; +   Vmxnet2_DriverData *dd = dp->driverData; +   volatile uint16_t *mcastTable = (uint16_t *)dd->LADRF; + +   ASSERT(MUTEX_HELD(&dp->intrlock)); + +   /* clear the multicast filter */ +   dd->LADRF[0] = 0; +   dd->LADRF[1] = 0; + +   for (i = 0; i < dp->multiCount; i++) { +      crc = 0xffffffff; +      ep = (uint8_t *)&dp->multicastList[i].ether_addr_octet; + +      for (byte = 0; byte < 6; byte++) { +         for (bit = *ep++, j = 0; j < 8; j++, bit >>= 1) { +            int test; + +            test = ((bit ^ crc) & 0x01); +            crc >>= 1; + +            if (test) { +               crc = crc ^ poly; +            } +         } +      } + +      crc = crc >> 26; +      mcastTable[crc >> 4] |= 1 << (crc & 0xf); +   } +} + +/* + *----------------------------------------------------------------------------- + * Vxn_SetMulticast -- + *    Add delete entry from multicast list + * + * Results: + *    GLD_FAILURE on failure + *    GLD_SUCCESS on success + * + * Side effects: + *    None   + *----------------------------------------------------------------------------- + */ +static int +Vxn_SetMulticast(gld_mac_info_t *macinfo, uint8_t *ep, int flag) +{ +   int i; +   int copyLen; +   vxn_softc_t *dp = (vxn_softc_t *)macinfo->gldm_private; +   Vmxnet2_DriverData *dd = dp->driverData; + +   if (flag == GLD_MULTI_ENABLE) { +      /* +       * Exceeded multicast address limit +       */ +      if (dp->multiCount >= GLD_MAX_MULTICAST) { +         return GLD_FAILURE; +      } + +      /*  +       * Add mac address to multicast list +       */ +      bcopy(ep, dp->multicastList[dp->multiCount].ether_addr_octet,  +                ETHERADDRL); +      dp->multiCount++; +   } +   else { +      for (i=0; i<dp->multiCount; i++) { +         if (bcmp(ep, dp->multicastList[i].ether_addr_octet, ETHERADDRL) == 0) { +            goto found; +         } +      } +      return GLD_FAILURE; + +   found: +      /* +       * Delete mac address from multicast list +       */ +      copyLen = (dp->multiCount - (i+1)) * sizeof(struct ether_addr); +      if (copyLen > 0) { +        bcopy(&dp->multicastList[i+1], &dp->multicastList[i], copyLen); +      } +      dp->multiCount--; +   } + +   /* +    * Compute address filter from list of addressed and apply it +    */ +   mutex_enter(&dp->intrlock); +   Vxn_ApplyAddressFilter(dp); + +   if (dp->multiCount) { +     ASSERT(dd->LADRF[0] || dd->LADRF[1]); +     dd->ifflags |= VMXNET_IFF_MULTICAST; +   } else { +     ASSERT(!(dd->LADRF[0] || dd->LADRF[1])); +     dd->ifflags &= ~VMXNET_IFF_MULTICAST; +   } + +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_UPDATE_IFF); +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_UPDATE_LADRF); +   mutex_exit(&dp->intrlock);    + +   return GLD_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_SetMacAddress -- + *    Change device MAC address   + * + * Results: + *    GLD_SUCCESS + *    GLD_FAILURE + * + * Side effects: + *    None  + *----------------------------------------------------------------------------- + */ +static int +Vxn_SetMacAddress(gld_mac_info_t *macInfo, uint8_t *mac) +{ +   int i; +   int err = GLD_SUCCESS; +   vxn_softc_t * dp = (vxn_softc_t *)macInfo->gldm_private; + +   mutex_enter(&dp->intrlock); +   mutex_enter(&dp->xmitlock); + +   /* +    * Don't change MAC address on a running NIC +    */ +   if (dp->nicActive) { +      err = GLD_FAILURE; +      goto out; +   } + +   /* +    * Save new MAC address  +    */ +   for (i = 0; i < 6; i++) { +      dp->devAddr.ether_addr_octet[i] = mac[i]; +   } +    +   /* +    * Push new MAC address down into hardware +    */ +   for (i = 0; i < 6; i++) { +      OUTB(dp, VMXNET_MAC_ADDR + i, mac[i]); +   } + +out: +   mutex_exit(&dp->xmitlock); +   mutex_exit(&dp->intrlock); +   return err; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Start -- + *    Device start routine. Called on "ifconfig plumb" + * + * Results: + *    GLD_SUCCESS + *    GLD_FAILURE + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_Start(gld_mac_info_t *macInfo) +{ +   int err = GLD_SUCCESS; +   uint32_t r, capabilities, features; +   vxn_softc_t * dp = (vxn_softc_t *)macInfo->gldm_private; + +   mutex_enter(&dp->intrlock); +   mutex_enter(&dp->xmitlock); + +   if (!dp->nicActive) { +      /* +       * Register ring structure with hardware +       * +       * This downcast is OK because we requested a 32-bit physical address +       */ +      OUTL(dp, VMXNET_INIT_ADDR, (uint32_t)(uintptr_t)dp->driverDataPhy); +      OUTL(dp, VMXNET_INIT_LENGTH, dp->driverData->length); + +      /*  +       * Make sure registeration succeded  +       */ +      r = INL(dp, VMXNET_INIT_LENGTH); +      if (!r) { +         cmn_err(CE_WARN, "%s%d: Vxn_Start: failed to register ring",  +                          dp->drvName, dp->unit); +         err = GLD_FAILURE; +         goto out; +      } + +      /* +       * Get maximum tx fragments supported +       */ +      OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_GET_CAPABILITIES); +      capabilities = INL(dp, VMXNET_COMMAND_ADDR); + +      OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_GET_FEATURES); +      features = INL(dp, VMXNET_COMMAND_ADDR); + +      DPRINTF(3, (CE_CONT, "%s%d: chip capabilities=0x%x features=0x%x\n",  +              dp->drvName, dp->unit, capabilities, features)); + +      if ((capabilities & VMNET_CAP_SG) && +          (features & VMXNET_FEATURE_ZERO_COPY_TX)) { +         dp->maxTxFrags = VMXNET2_SG_DEFAULT_LENGTH; +      } else { +         dp->maxTxFrags = 1; +      } +      ASSERT(dp->maxTxFrags >= 1); + +      /* +       * Alloc Tx DMA handle +       */ +      vxn_dma_attrs_tx.dma_attr_sgllen = dp->maxTxFrags; +      if (ddi_dma_alloc_handle(dp->dip, &vxn_dma_attrs_tx, DDI_DMA_SLEEP, +                               NULL, &dp->txDmaHdl) != DDI_SUCCESS)  { +         cmn_err(CE_WARN, "%s%d: Vxn_Start: failed to alloc tx dma handle",  +                 dp->drvName, dp->unit); +         err = GLD_FAILURE; +         goto out; +      } + +      /* +       * Enable interrupts on the card +       */ +      dp->driverData->ifflags |= VMXNET_IFF_BROADCAST | VMXNET_IFF_DIRECTED; + +      OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_INTR_ENABLE); +      OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_UPDATE_IFF); +      OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_UPDATE_LADRF); + +      dp->nicActive = TRUE; +  } + +out: +   mutex_exit(&dp->xmitlock); +   mutex_exit(&dp->intrlock); +   return err; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Stop -- + *    Device stop routine. Called on "ifconfig unplumb" + * + * Results: + *    GLD_SUCCESS + *    GLD_FAILURE + * + * Side effects: + *    None    + *----------------------------------------------------------------------------- + */ +static int +Vxn_Stop(gld_mac_info_t *macInfo) +{ +   int i; +   int err = GLD_SUCCESS; +   vxn_softc_t * dp = (vxn_softc_t *)macInfo->gldm_private; +   boolean_t resched; + +   mutex_enter(&dp->intrlock); +   mutex_enter(&dp->xmitlock); + +   if (!dp->nicActive) { +      goto out; +   } + +   /* +    * Disable interrupts +    */ +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_INTR_DISABLE); + +   /* +    * Wait for pending transmits +    */ +   if (dp->txPending) { +      for (i=0; i < MAX_TX_WAIT_ON_STOP && dp->txPending; i++) { +         delay(drv_usectohz(1000)); +         OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_CHECK_TX_DONE); +         (void) Vxn_TxComplete(dp, &resched); +         /* +          * Don't worry about rescheduling transmits - GLD handles +          * this automatically. +          */ +      } +   } +   if (dp->txPending) { +      cmn_err(CE_WARN, "%s%d: Vxn_Stop: giving up on %d pending transmits",  +                       dp->drvName, dp->unit, dp->txPending);       +   } + +   OUTL(dp, VMXNET_INIT_ADDR, 0); +   dp->nicActive = FALSE; + +   /* +    * Free Tx DMA handle +    * +    * The ddi_dma_free_handle() man page says that ddi_dma_unbind_handle() must be called +    * prior to calling ddi_dma_free_handle(). +    * However, call to ddi_dma_unbind_handle() is not required here, because +    * ddi_dma_addr_bind_handle() and matching ddi_dma_unbind_handle() are called from +    * Vxn_EncapTxBuf(). +    * xmitlock is held in Vxn_EncapTxBuf() as well as acquired above in Vxn_Stop(). +    */ +   ddi_dma_free_handle(&dp->txDmaHdl); +   dp->txDmaHdl = NULL; + +out: +   mutex_exit(&dp->xmitlock); +   mutex_exit(&dp->intrlock); +   return err; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeTxBuf -- + *    Free transmit buffer + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeTxBuf(vxn_softc_t *dp, int idx) +{ +   mblk_t **txMblkp = &TX_RINGBUF_MBLK(dp, idx); +   dma_buf_t *dmaMem = &TX_RINGBUF_DMAMEM(dp, idx); + +   if (*txMblkp) { +      freemsg(*txMblkp); +      *txMblkp = NULL; +   } + +   if (dmaMem->buf) { +      Vxn_FreeDmaMem(dmaMem); +      ASSERT(dmaMem->buf == NULL); +   } +} + +/* + *----------------------------------------------------------------------------- + * Vxn_EncapTxBuf --  + *    Go over dma mappings of Tx buffers and drop buffer physical address  + *    into ring entry + * + * Results: + *    SOLVMXNET_SUCCESS on success + *    SOLVMXNET_FAILURE on failure + * + * Side effects: + *    None   + *---------------- ------------------------------------------------------------- + */ +static int +Vxn_EncapTxBuf(vxn_softc_t *dp,  +               mblk_t *mp,  +               Vmxnet2_TxRingEntry *xre,  +               tx_ring_buf_t *txBuf) +{ +   int frag; +   int fragcount; +   int rval; +   mblk_t *tp; +   mblk_t *mblk; +   boolean_t needPullup = FALSE; +   boolean_t dmaMemAlloced = FALSE; + +   ASSERT(txBuf); +   ASSERT(txBuf->mblk == NULL); +   ASSERT(MUTEX_HELD(&dp->xmitlock)); + +   xre->sg.length = 0; +   xre->flags = 0; + +   fragcount = 0; +   for (tp = mp; tp != NULL; tp = tp->b_cont) { +      fragcount++; +   } +   if (fragcount > dp->maxTxFrags) { +      needPullup = TRUE; +   } + +pullup: +   frag = 0; +   if (needPullup) { +      if (!(mblk = msgpullup(mp, -1))) { +         cmn_err(CE_WARN, "%s%d: Vxn_EncapTxBuf: msgpullup failed", +                          dp->drvName, dp->unit); +         goto err; +      } +   } else { +      mblk = mp; +   } + +   /* +    * Go through message chain and drop packet pointers into ring +    * scatter/gather array +    */ +   for (tp = mblk; tp != NULL; tp = tp->b_cont) { + +      uint_t nCookies; +      ddi_dma_cookie_t dmaCookie; +      int len = tp->b_wptr - tp->b_rptr; + +      if (len) { +         /* +          * Associate tx buffer with dma handle +          */ +         ASSERT(dp->txDmaHdl); +         if ((rval = ddi_dma_addr_bind_handle(dp->txDmaHdl, NULL, (caddr_t)tp->b_rptr, +                                      len, DDI_DMA_RDWR | DDI_DMA_STREAMING, +                                      DDI_DMA_DONTWAIT, NULL, +                                      &dmaCookie, &nCookies)) +             != DDI_DMA_MAPPED) { + +            /* +             *  Try to handle bind failure caused by a page boundary spill +             *  by allocating a private dma buffer and copying data into it +             */ +            if ((rval == DDI_DMA_TOOBIG) && !dmaMemAlloced ) { +               /* +                * Force pullup  +                */ +               if (!needPullup && (dp->maxTxFrags > 1)) { +                  needPullup = TRUE; +                  goto pullup; +               } + +               if (Vxn_AllocDmaMem(dp, len, FALSE, &txBuf->dmaMem)  +                                             != SOLVMXNET_SUCCESS) { +                  goto err; +               } + +               dmaMemAlloced = TRUE; + +               /* +                * Copy data into DMA capable buffer +                */ +               bcopy(tp->b_rptr, txBuf->dmaMem.buf, len); + +               /* +                * Stick buffer physical addr in the ring +                */ +               xre->sg.sg[frag].addrLow = txBuf->dmaMem.phyBuf; +               xre->sg.sg[frag].length = len; +               frag++; + +               continue; + +            } else { +               cmn_err(CE_WARN, "%s%d: Vxn_EncapTxBuf: failed (%d) to bind dma " +                                "handle for len %d. [dmaMemAlloced=%d]",  +                                dp->drvName, dp->unit, rval, len, dmaMemAlloced); +               goto err; +            } +         } + +         /* +          * Extract tx buffer physical addresses from cookie +          */ +         while (nCookies) { +            if (UNLIKELY(frag == dp->maxTxFrags)) { +               (void)ddi_dma_unbind_handle(dp->txDmaHdl); + +               if (!needPullup) { +                  ASSERT(!dmaMemAlloced); +                  needPullup = TRUE; +                  goto pullup; +               } else { +                  cmn_err(CE_WARN, "%s%d: Vxn_EncapTxBuf: " +                          "exceeded max (%d) fragments in message", +                          dp->drvName, dp->unit, dp->maxTxFrags); +                  goto err; +               } +            } + +            /* +             * Stick it in the ring +             */ +            xre->sg.sg[frag].addrLow = dmaCookie.dmac_address; +            xre->sg.sg[frag].length = dmaCookie.dmac_size; +            frag++; + +            if (--nCookies) { +               ddi_dma_nextcookie(dp->txDmaHdl, &dmaCookie); +            } +         } + +         (void)ddi_dma_unbind_handle(dp->txDmaHdl); +      } +   } + +   if (frag > 0) { +      xre->sg.length = frag; + +      /* Give ownership to NIC */ +      xre->sg.addrType = NET_SG_PHYS_ADDR; +      xre->ownership = VMXNET2_OWNERSHIP_NIC; +      xre->flags |= VMXNET2_TX_CAN_KEEP; +      txBuf->mblk = mblk; + +      /* +       * If we called msgpullup to concatenate fragments, free +       * original mblk now since we're going to return success. +       */ +      if (mblk != mp) { +         freemsg(mp); +      } + +      return SOLVMXNET_SUCCESS; +   } + +err: +   if (mblk != NULL && mblk != mp) { +      /* +       * Free mblk allocated by msgpullup. +       */ +      freemsg(mblk); +   } + +   if (dmaMemAlloced) { +      ASSERT(txBuf->dmaMem.buf); +      Vxn_FreeDmaMem(&txBuf->dmaMem); +   } + +   return SOLVMXNET_FAILURE; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Send -- + *    GLD Transmit routine. Starts packet hard tx. + * + * Results: + *    GLD_SUCCESS on success + *    GLD_FAILURE on failure + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_Send(gld_mac_info_t *macinfo, mblk_t *mp) +{ +   Vmxnet2_TxRingEntry *xre; +   int err = GLD_SUCCESS; +   vxn_softc_t *dp = (vxn_softc_t *)macinfo->gldm_private; +   Vmxnet2_DriverData *dd = dp->driverData; +   boolean_t resched = FALSE; + +   mutex_enter(&dp->xmitlock); + +   /* +    * Check if ring entry at drop pointer is available +    */ +   if (TX_RINGBUF_MBLK(dp, dd->txDriverNext) != NULL) { +      DPRINTF(3, (CE_NOTE, "%s%d: Vxn_Send: tx ring full", +                  dp->drvName, dp->unit)); +      err = GLD_NORESOURCES; +      dd->txStopped = TRUE; +      dp->stats.defer++; +      goto out; +   } + +   xre = &dp->txRing[dd->txDriverNext]; + +   /* +    * Drop packet into ring entry +    */ +   if (Vxn_EncapTxBuf(dp, mp, xre, &dp->txRingBuf[dd->txDriverNext]) +       != SOLVMXNET_SUCCESS) { +      err = GLD_FAILURE; +      dp->stats.errxmt++; +      goto out; +   } + +   /* +    * Increment drop pointer +    */ +   VMXNET_INC(dd->txDriverNext, dd->txRingLength); +   dd->txNumDeferred++; +   dp->txPending++; + +   /* +    * Transmit, if number of pending packets > tx cluster length +    */ +   if (dd->txNumDeferred >= dd->txClusterLength) { +      dd->txNumDeferred = 0; + +      /*  +       * Call hardware transmit  +       */ +      INL(dp, VMXNET_TX_ADDR); +   } + +   /* +    * Clean up transmit ring. TX completion interrupts are not guaranteed +    */ +   (void) Vxn_TxComplete(dp, &resched); + +out: +   mutex_exit(&dp->xmitlock); +   if (resched) { +      /* Tell GLD to retry any deferred packets */ +      gld_sched(dp->macInfo); +   } +   return err; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_TxComplete -- + *    Scan Tx ring for completed transmits. Reclaim Tx buffers. + * + * Results: + *    Returns TRUE if it found a completed transmit, FALSE otherwise. + *    Also sets *reschedp to TRUE if the caller should call gld_sched + *    to reschedule transmits (once all locks are dropped). + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static boolean_t +Vxn_TxComplete(vxn_softc_t *dp, boolean_t *reschedp) +{ +   Vmxnet2_DriverData *dd = dp->driverData; +   boolean_t found = FALSE; +   boolean_t needresched = FALSE; + +   ASSERT(MUTEX_HELD(&dp->xmitlock)); + +   while (1) { +      Vmxnet2_TxRingEntry *xre = &dp->txRing[dd->txDriverCur]; + +      if (xre->ownership != VMXNET2_OWNERSHIP_DRIVER ||  +          (TX_RINGBUF_MBLK(dp, dd->txDriverCur) == NULL)) { +         break; +      } + +      found = TRUE; +      Vxn_FreeTxBuf(dp, dd->txDriverCur); + +      dp->txPending--; +      VMXNET_INC(dd->txDriverCur, dd->txRingLength); +      if (dd->txStopped) { +         needresched = TRUE; +         dd->txStopped = FALSE; +      } +   } + +   *reschedp = needresched; +   return found; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Receive -- + *    Rx handler. First assembles the packets into a chain of mblks, + *    then drops locks and passes them up the stack to GLD. + * + * Results: + *    Returns TRUE if it find a packet ready for processing, FALSE + *    otherwise. + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static boolean_t +Vxn_Receive(vxn_softc_t *dp) +{ +   int ringnext; +   short pktlen; +   Vmxnet2_DriverData *dd = dp->driverData;    +   rx_dma_buf_t *rxDesc; +   rx_dma_buf_t *newRxDesc; +   mblk_t *mblk; +   mblk_t *head = NULL; +   mblk_t **tail = &head; +   mblk_t *next; +   boolean_t found = FALSE;	/* Did we find at least one packet? */ + +   ASSERT(MUTEX_HELD(&dp->intrlock)); + +   /* +    * Walk receive ring looking for entries with ownership  +    * reverted back to driver +    */ +   while (1) { +      Vmxnet2_RxRingEntry *rre; +      rx_dma_buf_t **rbuf; + +      ringnext = dd->rxDriverNext; +      rre = &dp->rxRing[ringnext]; +      rbuf = &dp->rxRingBuffPtr[ringnext]; +       +      if (rre->ownership != VMXNET2_OWNERSHIP_DRIVER) { +         break; +      } + +      found = TRUE; + +      pktlen = rre->actualLength; + +      if (pktlen < (60 - 4)) { +         /* +          * Ethernet header vlan tags are 4 bytes.  Some vendors generate +          *  60byte frames including vlan tags.  When vlan tag +          *  is stripped, such frames become 60 - 4. (PR106153) +          */ +         dp->stats.errrcv++; +         if (pktlen != 0) { +            DPRINTF(3, (CE_CONT, "%s%d: runt packet\n", dp->drvName, dp->unit)); +            dp->stats.runt++; +         } +      } else { +         /* +          * Alloc new Rx buffer to replace current one +          */ +         newRxDesc = Vxn_AllocRxBufFromPool(dp); + +         if (newRxDesc) { +            rxDesc = *rbuf; +            mblk = rxDesc->mblk; +             +            *rbuf = newRxDesc; +            rre->paddr = newRxDesc->dmaDesc.phyBuf + ETHERALIGN; +            rre->bufferLength = MAXPKTBUF - ETHERALIGN; +            rre->actualLength = 0; + +            /* +             * Advance write pointer past packet length +             */ +            mblk->b_wptr = mblk->b_rptr + pktlen; +             +            /* +             * Add to end of chain. +             */ +            mblk->b_next = NULL; +            *tail = mblk; +            tail = &mblk->b_next; +         } else { +            dp->stats.errrcv++; +            dp->stats.norcvbuf++; +         } +      } + +      /* Give the descriptor back to NIC */ +      rre->ownership = VMXNET2_OWNERSHIP_NIC; +      VMXNET_INC(dd->rxDriverNext, dd->rxRingLength); +   } + +   /* +    * Walk chain and pass mblks up to gld_recv one by one. +    */ +   mutex_exit(&dp->intrlock); +   for (mblk = head; mblk != NULL; mblk = next) { +      next = mblk->b_next; +      mblk->b_next = NULL; +      gld_recv(dp->macInfo, mblk); +   } +   mutex_enter(&dp->intrlock); + +   return (found); +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Interrupt -- + *    GLD interrupt handler. Scan: Rx ring for received packets, Tx ring for + *    completed transmits + * + * Results: + *    - DDI_INTR_CLAIMED (if we found something to do) + *    - DDI_INTR_UNCLAIMED (if not) + * + * Side effects: + *    None      + *----------------------------------------------------------------------------- + */ +static u_int +Vxn_Interrupt(gld_mac_info_t *macInfo) +{ +   u_int ret = DDI_INTR_UNCLAIMED; +   vxn_softc_t *dp = (vxn_softc_t *)macInfo->gldm_private; +   boolean_t foundRx, foundTx; +   boolean_t resched = FALSE; + +   mutex_enter(&dp->intrlock); +   dp->inIntr = TRUE; + +   if (!dp->nicActive) { +      goto out; +   } + +   /* +    * Ack interrupt +    */ +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_INTR_ACK); + +   foundRx = Vxn_Receive(dp); + +   mutex_enter(&dp->xmitlock); +   foundTx = Vxn_TxComplete(dp, &resched); +   mutex_exit(&dp->xmitlock); + +   if (foundRx || foundTx) { +      ret = DDI_INTR_CLAIMED; +      dp->stats.interrupts++; +   } + +out: +   dp->inIntr = FALSE; +   mutex_exit(&dp->intrlock); + +   if (resched) { +      gld_sched(dp->macInfo); +   } + +   return ret;  +} + + +/* + *----------------------------------------------------------------------------- + * Vxn_ReclaimRxBuf -- + *    Callback handler invoked by freemsg(). Frees Rx buffer memory and mappings + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_ReclaimRxBuf(rx_dma_buf_t *rxDesc) +{ +   Vxn_FreeRxBufToPool(rxDesc); +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeRxBuf -- + *    Free allocated Rx buffer + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeRxBuf(rx_dma_buf_t *rxDesc) +{ +   ASSERT(rxDesc); + +   if (rxDesc->mblk) { +      freemsg(rxDesc->mblk); +   } else { +      Vxn_FreeDmaMem(&rxDesc->dmaDesc); +      kmem_free(rxDesc, sizeof(rx_dma_buf_t)); +   } +} + + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocRxBuf -- + *    Allocate Rx buffer + * + * Results: + *    Pointer to Rx buffer descriptor - on success + *    NULL - on failure + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static rx_dma_buf_t * +Vxn_AllocRxBuf(vxn_softc_t *dp, int cansleep) +{ +   rx_dma_buf_t *rxDesc; + +   rxDesc = (rx_dma_buf_t *)kmem_zalloc(sizeof(rx_dma_buf_t),  +                                  cansleep ? KM_SLEEP : KM_NOSLEEP); +   if (!rxDesc) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocRxBuf: kmem_zalloc failed",  +                       dp->drvName, dp->unit); +      return NULL; +   } + +   rxDesc->softc = dp; + +   /* +    * Alloc dma-able packet memory +    */ +   if (Vxn_AllocDmaMem(dp, MAXPKTBUF, cansleep, &rxDesc->dmaDesc)  +         != SOLVMXNET_SUCCESS) { +      kmem_free(rxDesc, sizeof(rx_dma_buf_t)); +      return NULL; +   } + +   /* +    * Fill in free callback; fired by freemsg() +    */ +   rxDesc->freeCB.free_func = &Vxn_ReclaimRxBuf; +   rxDesc->freeCB.free_arg = (caddr_t) rxDesc; + +   rxDesc->mblk = NULL; +   return rxDesc; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeInitBuffers -- + *    Free allocated Tx and Rx buffers  + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeInitBuffers(vxn_softc_t *dp) +{ +   int i; + +   for (i=0; i<dp->vxnNumRxBufs; i++) { +      if (dp->rxRingBuffPtr[i]) { +         Vxn_FreeRxBuf(dp->rxRingBuffPtr[i]); +         dp->rxRingBuffPtr[i] = NULL; +      } +   } + +   for (i=0; i<dp->vxnNumTxBufs; i++) { +      if (TX_RINGBUF_MBLK(dp, i)) { +         Vxn_FreeTxBuf(dp, i); +      } +   } + +   /* +    * Rx pool must get freed last. Rx buffers above will +    * show up on the pool when freemsg callback fires. +    */ +   Vxn_FreeRxBufPool(dp); +} + + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocRxBufPool -- + *    Allocate pool of rx buffers - 3 * configured Rx buffers + * + * Results: + *    SOLVMXNET_SUCCESS/SOLVMXNET_FAILURE + *     + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_AllocRxBufPool(vxn_softc_t *dp) +{ +   int i; + +   dp->rxFreeBufList = NULL; + +   // Allow list to double in size if needed.  Any additional buffers +   // that are allocated on the fly will be freed back to main memory. +   dp->rxMaxFreeBufs = dp->vxnNumRxBufs * 6; + +   for (i = 0; i < dp->vxnNumRxBufs * 3; i++) { +      rx_dma_buf_t *rxDesc; + +      /* +       * Alloc rx buffer +       */ +      if (!(rxDesc = Vxn_AllocRxBuf(dp, TRUE))) { +         cmn_err(CE_WARN, "%s%d: Vxn_AllocRxBufPool: failed to allocate memory", +                 dp->drvName, dp->unit); +         dp->rxNumFreeBufs = i; +         return SOLVMXNET_FAILURE; +      } +      /* +       * Add to free list +       */ +      rxDesc->next = dp->rxFreeBufList; +      dp->rxFreeBufList = rxDesc; +   } + +   dp->rxNumFreeBufs = i; +   return SOLVMXNET_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeRxBufPool -- + *    Free rx buffers pool + * + * Results: + *    None     + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeRxBufPool(vxn_softc_t *dp) +{ +   while (dp->rxFreeBufList) { +      rx_dma_buf_t *rxDesc = dp->rxFreeBufList; + +      /* unlink */ +      dp->rxFreeBufList = rxDesc->next; + +      ASSERT(rxDesc->mblk == NULL); +      Vxn_FreeDmaMem(&rxDesc->dmaDesc); +      kmem_free(rxDesc, sizeof(rx_dma_buf_t)); +   } +   dp->rxNumFreeBufs = 0; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocRxBufFromPool -- + *    Allocate Rx buffer from free pool + * + * Results: + *    Pointer to Rx buffer descriptor - on success + *    NULL - on failure + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static rx_dma_buf_t * +Vxn_AllocRxBufFromPool(vxn_softc_t *dp) +{ +   rx_dma_buf_t *rxDesc = NULL; + +   mutex_enter(&dp->rxlistlock); +   if (dp->rxFreeBufList) { +      rxDesc = dp->rxFreeBufList; +      dp->rxFreeBufList = rxDesc->next; +      ASSERT(dp->rxNumFreeBufs >= 1); +      dp->rxNumFreeBufs--; +   } +   mutex_exit(&dp->rxlistlock); + +   if (!rxDesc) { +      /* +       * Try to allocate new descriptor from memory.  Can't block here +       * since we could be being called from interrupt context. +       */ +      DPRINTF(5, (CE_NOTE, "%s%d: allocating rx buf from memory", +                  dp->drvName, dp->unit));  +      if (!(rxDesc = Vxn_AllocRxBuf(dp, FALSE))) { +         cmn_err(CE_WARN, +                 "%s%d: Vxn_AllocRxBufFromPool : pool rx alloc failed", +                 dp->drvName, dp->unit); +         return NULL; +      } +   } + +   /* +    * Allocate new message block for this buffer +    */ +   rxDesc->mblk = desballoc((uchar_t *)rxDesc->dmaDesc.buf + ETHERALIGN, +                           rxDesc->dmaDesc.bufLen - ETHERALIGN,  +                           BPRI_MED, &rxDesc->freeCB); +   if (!rxDesc->mblk) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocRxBufFromPool : desballoc failed",  +                        dp->drvName, dp->unit); + +      /* put back on free list */ +      Vxn_FreeRxBufToPool(rxDesc); +      return NULL; +   } + +   return rxDesc; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeRxBufToPool -- + *    Return rx buffer to free pool + * + * Results: + *    None    + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeRxBufToPool(rx_dma_buf_t *rxDesc) +{ +   vxn_softc_t *dp = rxDesc->softc; + +   rxDesc->mblk = NULL; + +   /* +    * Insert on free list, or free if the list is full +    */ +   mutex_enter(&dp->rxlistlock); +   if (dp->rxNumFreeBufs >= dp->rxMaxFreeBufs) { +      DPRINTF(5, (CE_NOTE, "%s%d: freeing rx buf to memory",  +                  dp->drvName, dp->unit)); +      Vxn_FreeRxBuf(rxDesc); +   } else { +      rxDesc->next = dp->rxFreeBufList; +      dp->rxFreeBufList = rxDesc; +      dp->rxNumFreeBufs++; +   } +   mutex_exit(&dp->rxlistlock); +} + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocInitBuffers -- + *    Allocated Rx buffers and init ring entries + * + * Results: + *    SOLVMXNET_SUCCESS - on success + *    SOLVMXNET_FAILURE - on failure + * + * Side effects: + *    None       + *----------------------------------------------------------------------------- + */ +static int +Vxn_AllocInitBuffers(vxn_softc_t *dp) +{ +   Vmxnet2_DriverData  *dd; +   uint32_t            i, offset; + +   dd = dp->driverData; +   offset = sizeof(*dd); + +   /* +    * Init shared structures +    */ +   dd->rxRingLength = dp->vxnNumRxBufs; +   dd->rxRingOffset = offset; +   dp->rxRing = (Vmxnet2_RxRingEntry *)((uintptr_t)dd + offset); +   offset += dp->vxnNumRxBufs * sizeof(Vmxnet2_RxRingEntry); + +   dd->rxRingLength2 = 1; +   dd->rxRingOffset2 = offset; +   offset += sizeof(Vmxnet2_RxRingEntry); + +   dd->txRingLength = dp->vxnNumTxBufs; +   dd->txRingOffset = offset; +   dp->txRing = (Vmxnet2_TxRingEntry *)((uintptr_t)dd + offset); +   offset += dp->vxnNumTxBufs * sizeof(Vmxnet2_TxRingEntry); + +   /* +    * Alloc Rx buffers pool +    */ +   if ( Vxn_AllocRxBufPool(dp) != SOLVMXNET_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocInitBuffers: failed to alloc buf pool",  +                       dp->drvName, dp->unit); +      return SOLVMXNET_FAILURE; +   } + +   /* +    * Allocate receive buffers +    */ +   for (i = 0; i < dp->vxnNumRxBufs; i++) { +      rx_dma_buf_t *rxDesc; +      Vmxnet2_RxRingEntry *rre = &dp->rxRing[i]; + +      if (!(rxDesc = Vxn_AllocRxBufFromPool(dp))) { +         cmn_err(CE_WARN, "%s%d: Vxn_AllocInitBuffers: " +                          "failed to alloc buf from pool", dp->drvName, dp->unit); +         goto err; +      } + +      /* +       * Init ring entries +       */ +      rre->paddr = rxDesc->dmaDesc.phyBuf + ETHERALIGN; +      rre->bufferLength = MAXPKTBUF - ETHERALIGN; +      rre->actualLength = 0; +      dp->rxRingBuffPtr[i] = rxDesc; +      rre->ownership = VMXNET2_OWNERSHIP_NIC; +   } + +   dp->txDmaHdl = NULL; + +   /*  +    * Dummy recvRing2 tacked on to the end, with a single unusable entry  +    */ +   dp->rxRing[i].paddr = 0; +   dp->rxRing[i].bufferLength = 0; +   dp->rxRing[i].actualLength = 0; +   dp->rxRingBuffPtr[i] = NULL; +   dp->rxRing[i].ownership = VMXNET2_OWNERSHIP_DRIVER; +    +   dd->rxDriverNext = 0; +                                                                                +   /* +    * Give xmit ring ownership to DRIVER +    */ +   for (i = 0; i < dp->vxnNumTxBufs; i++) { +      dp->txRing[i].ownership = VMXNET2_OWNERSHIP_DRIVER; +      dp->txRingBuf[i].mblk = NULL; +      dp->txRingBuf[i].dmaMem.buf = NULL; +      dp->txRing[i].sg.sg[0].addrHi = 0; +   } +    +   dd->txDriverCur = dd->txDriverNext = 0; +   dd->txStopped = FALSE; + +   return SOLVMXNET_SUCCESS; + +err: +   for (i=0; i<dp->vxnNumRxBufs; i++) { +      if (dp->rxRingBuffPtr[i]) { +         Vxn_FreeRxBuf(dp->rxRingBuffPtr[i]); +         dp->rxRingBuffPtr[i] = NULL; +      } +   } + +   Vxn_FreeRxBufPool(dp); +   return SOLVMXNET_FAILURE; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeDmaMem -- + *    Free allocated dma memory + * + * Results: + *    None + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeDmaMem(dma_buf_t *dma) +{ +   ddi_dma_unbind_handle(dma->dmaHdl); +   ddi_dma_mem_free(&dma->dataAccHdl); +   ddi_dma_free_handle(&dma->dmaHdl); + +   dma->buf = NULL; +   dma->phyBuf = NULL; +   dma->bufLen = 0; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocDmaMem -- + *    Allocate dma-able memory and fill passed in dma descriptor pointer  + *    if successful + * + * Results: + *    SOLVMXNET_SUCCESS on success + *    SOLVMXNET_FAILURE on failure + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_AllocDmaMem(vxn_softc_t *dp, int size, int cansleep, dma_buf_t *dma) +{ +   /*  +    * Allocate handle +    */ +   if (ddi_dma_alloc_handle(dp->dip, &vxn_dma_attrs, +                            cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT,  +                            NULL, &dma->dmaHdl) != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocDmaMem: failed to allocate handle",  +              dp->drvName, dp->unit); +      return SOLVMXNET_FAILURE; +   } + +   /*  +    * Allocate memory  +    */ +   if (ddi_dma_mem_alloc(dma->dmaHdl, size, &vxn_buf_attrs, DDI_DMA_CONSISTENT, +                         cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT, NULL,  +                         &dma->buf, &dma->bufLen, &dma->dataAccHdl)  +       != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocDmaMem: " +                       "ddi_dma_mem_alloc %d bytes failed",  +                       dp->drvName, dp->unit, size); +      ddi_dma_free_handle(&dma->dmaHdl); +      return SOLVMXNET_FAILURE; +   } + +   /* +    * Mapin memory +    */ +   if (ddi_dma_addr_bind_handle(dma->dmaHdl, NULL, dma->buf, dma->bufLen, +                                DDI_DMA_RDWR | DDI_DMA_STREAMING, +                                cansleep ? DDI_DMA_SLEEP : DDI_DMA_DONTWAIT,  +                                NULL, &dma->cookie, &dma->cookieCount)  +       != DDI_DMA_MAPPED) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocDmaMem: failed to bind handle", +              dp->drvName, dp->unit); +      ddi_dma_mem_free(&dma->dataAccHdl); +      ddi_dma_free_handle(&dma->dmaHdl); +      return SOLVMXNET_FAILURE; +   } + +   if (dma->cookieCount != 1) { +      cmn_err(CE_WARN, "%s%d: Vxn_AllocDmaMem: too many DMA cookies",  +                       dp->drvName, dp->unit); +      Vxn_FreeDmaMem(dma); +      return SOLVMXNET_FAILURE; +   } + +   /* +    * Save physical address (for easy use) +    */ +   dma->phyBuf = dma->cookie.dmac_address; + +   return SOLVMXNET_SUCCESS; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_FreeDriverData -- + *   Free driver data structures and Tx Rx buffers + * + * Results: + *   None + * + * Side effects: + *   None + *----------------------------------------------------------------------------- + */ +static void +Vxn_FreeDriverData(vxn_softc_t *dp) +{ +   Vxn_FreeInitBuffers(dp); +   Vxn_FreeDmaMem(&dp->driverDataDmaMem); +} + +/* + *----------------------------------------------------------------------------- + * Vxn_AllocDriverData -- + *    Allocate driver data structures and Tx Rx buffers on init + * + * Results: + *    SOLVMXNET_SUCCESS on success + *    SOLVMXNET_FAILURE on failure + * + * Side effects: + *    None + *----------------------------------------------------------------------------- + */ +static int +Vxn_AllocDriverData(vxn_softc_t *dp) +{ +   uint32_t r, driverDataSize; + +   /* +    * Get configured receive buffers +    */ +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_GET_NUM_RX_BUFFERS); +   r = INL(dp, VMXNET_COMMAND_ADDR); +   if (r == 0 || r > MAX_NUM_RECV_BUFFERS) { +      r = DEFAULT_NUM_RECV_BUFFERS; +   } +   dp->vxnNumRxBufs = r; + +   /* +    * Get configured transmit buffers +    */ +   OUTL(dp, VMXNET_COMMAND_ADDR, VMXNET_CMD_GET_NUM_TX_BUFFERS); +   r = INL(dp, VMXNET_COMMAND_ADDR); +   if (r == 0 || r > MAX_NUM_XMIT_BUFFERS) { +      r = DEFAULT_NUM_XMIT_BUFFERS; +   } +   dp->vxnNumTxBufs = r; + +   /* +    * Calculate shared data size and allocate memory for it +    */ +   driverDataSize = +      sizeof(Vmxnet2_DriverData) + +      /* numRecvBuffers + 1 for the dummy recvRing2 (used only by Windows) */ +      (dp->vxnNumRxBufs + 1) * sizeof(Vmxnet2_RxRingEntry) + +      dp->vxnNumTxBufs * sizeof(Vmxnet2_TxRingEntry); + +   if (Vxn_AllocDmaMem(dp, driverDataSize, TRUE, &dp->driverDataDmaMem)  +        != SOLVMXNET_SUCCESS) { +      return SOLVMXNET_FAILURE; +   } + +   /* +    * Clear memory (bzero isn't resolved by module loader for some reason)  +    */ +   ASSERT(dp->driverDataDmaMem.buf && dp->driverDataDmaMem.bufLen); +   Vxn_Memset(dp->driverDataDmaMem.buf, 0, dp->driverDataDmaMem.bufLen); + +   dp->driverData = (Vmxnet2_DriverData *)dp->driverDataDmaMem.buf; +   dp->driverDataPhy = (void *)(uintptr_t)dp->driverDataDmaMem.phyBuf; + +   /* So that the vmkernel can check it is compatible */ +   dp->driverData->magic = VMXNET2_MAGIC; +   dp->driverData->length = driverDataSize; + +   /* +    * Alloc rx/tx buffers, init ring, register with hardware etc. +    */ +   if (Vxn_AllocInitBuffers(dp) != SOLVMXNET_SUCCESS) { +      Vxn_FreeDmaMem(&dp->driverDataDmaMem); +      return SOLVMXNET_FAILURE; +   } + +   DPRINTF(3, (CE_CONT, "%s%d: numRxBufs=(%d*%"FMT64"d) numTxBufs=(%d*%"FMT64"d)"  +               " driverDataSize=%d driverDataPhy=0x%p\n",  +               dp->drvName, dp->unit,  +               dp->vxnNumRxBufs, (uint64_t)sizeof(Vmxnet2_RxRingEntry), +               dp->vxnNumTxBufs, (uint64_t)sizeof(Vmxnet2_TxRingEntry), +               driverDataSize, dp->driverDataPhy)); + +   return SOLVMXNET_SUCCESS; +} + + +/* + *----------------------------------------------------------------------------- + * Vxn_Attach -- + *    Probe and attach driver to stack + * + * Results: + *    DDI_SUCCESS + *    DDI_FAILURE + * + * Side effects: + *    None  + *----------------------------------------------------------------------------- + */ +static int +Vxn_Attach(dev_info_t *dip, ddi_attach_cmd_t cmd) +{ +   int                  i, ret, len, unit; +   const char           *drvName; +   ddi_acc_handle_t     confHdl; +   uint16_t             vid, did; +   uint8_t              revid __unused; +   struct pci_phys_spec *regs; +   caddr_t              vxnIOp; +   ddi_acc_handle_t     vxnIOHdl; +   uint32_t             vLow, vHigh; +   gld_mac_info_t       *macInfo; +   vxn_softc_t          *dp; +   boolean_t            morphed = FALSE; +   uint_t               regSpaceSize; +   uint_t               chip; +   uint_t               vxnIOSize; + +   if (cmd != DDI_ATTACH) { +      return DDI_FAILURE; +   } + +   unit = ddi_get_instance(dip); +   drvName = ddi_driver_name(dip); + +   /* +    * Check if chip is supported. +    */ +   if (pci_config_setup(dip, &confHdl) != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: pci_config_setup() failed", drvName, unit); +      return DDI_FAILURE; +   } + +   vid   = pci_config_get16(confHdl, PCI_CONF_VENID); +   did   = pci_config_get16(confHdl, PCI_CONF_DEVID); +   revid = pci_config_get8(confHdl, PCI_CONF_REVID); + +   if (vid == PCI_VENDOR_ID_VMWARE && did == PCI_DEVICE_ID_VMWARE_NET) { +      /* Found vmxnet */ +      chip = VMXNET_CHIP; +   } +   else if (vid == PCI_VENDOR_ID_AMD && did == PCI_DEVICE_ID_AMD_VLANCE) { +      /* Found vlance (maybe a vmxnet disguise) */ +      chip = LANCE_CHIP; +   } +   else { +      /* Not Found */ +      DPRINTF(3, (CE_WARN, "%s: Vxn_Attach: wrong PCI venid/devid (0x%x, 0x%x)", +              drvName, vid, did)); +      goto err; +   } + +   DPRINTF(3, (CE_CONT, "%s%d: (vid: 0x%04x, did: 0x%04x, revid: 0x%02x)\n", +           drvName, unit, vid, did, revid)); + +   /*  +    * Get device properties +    */ +   regs = NULL; +   len  = 0; +   if (ddi_getlongprop(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS, +                       "reg", (caddr_t)®s, &len) != DDI_PROP_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: failed to get reg property",  +              drvName, unit); +      goto err; +   } + +   ASSERT(regs != NULL && len > 0); + +   /*  +    * Search device properties for IO-space  +    */ +   for (i = 0; i <len / sizeof(struct pci_phys_spec); i++) { +      if ((regs[i].pci_phys_hi & PCI_REG_ADDR_M) == PCI_ADDR_IO) { +         regSpaceSize = regs[i].pci_size_low; +         DPRINTF(5, (CE_CONT, "%s%d: Vxn_Attach: regSpaceSize=%d\n",  +                     drvName, unit, regSpaceSize)); +         kmem_free(regs, len); +         goto map_space_found; +      } +   } + +   cmn_err(CE_WARN, "%s%d: Vxn_Attach: failed to find IO space", drvName, unit); +   kmem_free(regs, len); +   goto err; + +map_space_found: + +   /*  +    * Ensure we can access registers through IO space.  +    */ +   ret = pci_config_get16(confHdl, PCI_CONF_COMM); +   ret |= PCI_COMM_IO | PCI_COMM_ME; +   pci_config_put16(confHdl, PCI_CONF_COMM, ret); + +   if (ddi_regs_map_setup(dip, i, (caddr_t *)&vxnIOp, 0, 0, &dev_attr,  +                          &vxnIOHdl) != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: ddi_regs_map_setup failed",  +              drvName, unit); +      goto err; +   } + +   if (chip == VMXNET_CHIP) { +      vxnIOSize = VMXNET_CHIP_IO_RESV_SIZE; +   } +   else { +      /* +       * Since this is a vlance adapter we can only use it if +       * its I/0 space is big enough for the adapter to be +       * capable of morphing. This is the first requirement +       * for this adapter to potentially be morphable. The +       * layout of a morphable LANCE adapter is +       * +       * I/O space: +       * +       * |------------------|  +       * | LANCE IO PORTS   | +       * |------------------| +       * | MORPH PORT       | +       * |------------------| +       * | VMXNET IO PORTS  | +       * |------------------| +       * +       * VLance has 8 ports of size 4 bytes, the morph port is 4 bytes, and +       * Vmxnet has 10 ports of size 4 bytes. +       * +       * We shift up the ioaddr with the size of the LANCE I/O space since +       * we want to access the vmxnet ports. We also shift the ioaddr up by +       * the MORPH_PORT_SIZE so other port access can be independent of +       * whether we are Vmxnet or a morphed VLance. This means that when +       * we want to access the MORPH port we need to subtract the size +       * from ioaddr to get to it. +       */ +      vxnIOp += LANCE_CHIP_IO_RESV_SIZE + MORPH_PORT_SIZE; +      vxnIOSize = LANCE_CHIP_IO_RESV_SIZE + MORPH_PORT_SIZE + +                  VMXNET_CHIP_IO_RESV_SIZE; +   } + +   /* +    * Do not attempt to morph non-morphable AMD PCnet +    */ +   if (vxnIOSize > regSpaceSize) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: " +              "vlance device is not supported by this driver", drvName, unit); +      goto err_free_regs_map; +   } + +   /* +    * Morph, if we found a vlance adapter +    */ +   if (chip == LANCE_CHIP) { +      uint16_t magic; + +      /* Read morph port to verify that we can morph the adapter */ +      magic = ddi_get16(vxnIOHdl, (uint16_t *)(vxnIOp - MORPH_PORT_SIZE)); +      if (magic != LANCE_CHIP && magic != VMXNET_CHIP) { +         cmn_err(CE_WARN, "%s%d: Vxn_Attach: Invalid magic, read: 0x%08X", +                 drvName, unit, magic); +         goto err_free_regs_map; +      } + +      /* Morph */ +      ddi_put16(vxnIOHdl, (uint16_t *)(vxnIOp - MORPH_PORT_SIZE), VMXNET_CHIP); +      morphed = TRUE; + +      /* Verify that we morphed correctly */ +      magic = ddi_get16(vxnIOHdl, (uint16_t *)(vxnIOp - MORPH_PORT_SIZE)); +      if (magic != VMXNET_CHIP) { +         cmn_err(CE_WARN, "%s%d: Vxn_Attach: Couldn't morph adapter." +                 " Invalid magic, read:: 0x%08X", drvName, unit, magic); +         goto err_morph_back; +      } +   } + +   /* +    * Check the version number of the device implementation +    */  +   vLow  = (uint32_t)ddi_get32(vxnIOHdl,  +                   (uint32_t *)(vxnIOp+VMXNET_LOW_VERSION)); +   vHigh = (uint32_t)ddi_get32(vxnIOHdl,  +                   (uint32_t *)(vxnIOp+VMXNET_HIGH_VERSION)); + +   if ((vLow & 0xffff0000) != (VMXNET2_MAGIC & 0xffff0000) ||  +       ((VMXNET2_MAGIC < vLow) || (VMXNET2_MAGIC > vHigh))) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: driver version 0x%08X doesn't " +                       "match device 0x%08X:0x%08X",  +              drvName, unit, VMXNET2_MAGIC, vLow, vHigh); +      goto err_version_mismatch; +   } + +   /* +    * Alloc soft state +    */ +   macInfo = gld_mac_alloc(dip); +   if (!macInfo) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: gld_mac_alloc failed",  +              drvName, unit); +      goto err_gld_mac_alloc; +   } + +   dp = (vxn_softc_t *) kmem_zalloc(sizeof(vxn_softc_t), KM_SLEEP); +   ASSERT(dp); + +   /* +    * Get interrupt cookie +    */ +   if (ddi_get_iblock_cookie(dip, 0, &dp->iblockCookie) != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: Vxn_Attach: ddi_get_iblock_cookie failed",  +              drvName, unit); +      goto err_get_iblock_cookie; +   } + +   strncpy(dp->drvName, drvName, SOLVMXNET_MAXNAME); +   dp->unit = unit; +   dp->dip = dip; +   dp->macInfo = macInfo; +   dp->confHdl = confHdl; +   dp->vxnIOHdl = vxnIOHdl; +   dp->vxnIOp = vxnIOp; +   dp->morphed = morphed; +   dp->nicActive = FALSE; +   dp->txPending = 0; +   dp->maxTxFrags = 1; + +   /* +    * Initialize mutexes +    */ +   mutex_init(&dp->intrlock, NULL, MUTEX_DRIVER, (void *)dp->iblockCookie); +   mutex_init(&dp->xmitlock, NULL, MUTEX_DRIVER, (void *)dp->iblockCookie); +   mutex_init(&dp->rxlistlock, NULL, MUTEX_DRIVER, (void *)dp->iblockCookie); + +   /*  +    * Allocate and initialize our private and shared data structures +    */ +   if (Vxn_AllocDriverData(dp) != SOLVMXNET_SUCCESS) { +      goto err_alloc_driverdata; +   } + +   /* +    * Read the MAC address from the device +    */ +   for (i = 0; i < 6; i++) { +      dp->devAddr.ether_addr_octet[i] =  +         (uint8_t)ddi_get8(vxnIOHdl, (uint8_t *)(vxnIOp + VMXNET_MAC_ADDR + i)); +   } +   macInfo->gldm_vendor_addr = dp->devAddr.ether_addr_octet; +   macInfo->gldm_broadcast_addr = etherbroadcastaddr.ether_addr_octet; + +   DPRINTF(3, (CE_CONT, +           "MAC address: %02x:%02x:%02x:%02x:%02x:%02x\n", +           dp->devAddr.ether_addr_octet[0], +           dp->devAddr.ether_addr_octet[1], +           dp->devAddr.ether_addr_octet[2], +           dp->devAddr.ether_addr_octet[3], +           dp->devAddr.ether_addr_octet[4], +           dp->devAddr.ether_addr_octet[5])); + +   /* +    * Configure GLD entry points +    */ +   macInfo->gldm_devinfo      = dip; +   macInfo->gldm_private      = (caddr_t)dp; +   macInfo->gldm_cookie       = dp->iblockCookie; +   macInfo->gldm_reset        = Vxn_Reset; +   macInfo->gldm_start        = Vxn_Start; +   macInfo->gldm_stop         = Vxn_Stop; +   macInfo->gldm_set_mac_addr = Vxn_SetMacAddress; +   macInfo->gldm_send         = Vxn_Send; +   macInfo->gldm_set_promiscuous = Vxn_SetPromiscuous; +   macInfo->gldm_get_stats    = Vxn_GetStats; +   macInfo->gldm_ioctl        = NULL; +   macInfo->gldm_set_multicast= Vxn_SetMulticast; +   macInfo->gldm_intr         = Vxn_Interrupt; +   macInfo->gldm_mctl         = NULL; +    +   macInfo->gldm_ident        = (char *)ddi_driver_name(dip); +   macInfo->gldm_type         = DL_ETHER; +   macInfo->gldm_minpkt       = 0; +   macInfo->gldm_maxpkt       = ETHERMTU; +   macInfo->gldm_addrlen      = ETHERADDRL; +   macInfo->gldm_saplen       = -2; +   macInfo->gldm_ppa          = unit; + +   /* +    * Register with GLD (Generic Lan Driver) framework +    */ +   if (gld_register(dip, +            (char *)ddi_driver_name(dip), macInfo) != DDI_SUCCESS) { +      goto err_gld_register; +   } + +   /* +    * Add interrupt to system. +    */ +   if (ddi_add_intr(dip, 0, NULL, NULL, gld_intr, +                    (caddr_t)macInfo) != DDI_SUCCESS) { +      cmn_err(CE_WARN, "%s%d: ddi_add_intr failed", drvName, unit); +      goto err_ddi_add_intr; +   } + +   /* +    * Add to list of interfaces. +    */ +   mutex_enter(&vxnListLock); +   dp->next = &vxnList; +   dp->prev = vxnList.prev; +   vxnList.prev->next = dp; +   vxnList.prev = dp; +   mutex_exit(&vxnListLock); + +   /* +    * Success +    */ +   return DDI_SUCCESS; + +err_ddi_add_intr: +   gld_unregister(macInfo); + +err_gld_register: +   Vxn_FreeDriverData(dp); + +err_alloc_driverdata: +   mutex_destroy(&dp->intrlock); +   mutex_destroy(&dp->xmitlock); + +err_get_iblock_cookie: +   kmem_free(dp, sizeof(*dp)); +   gld_mac_free(macInfo); + +err_gld_mac_alloc: +err_version_mismatch: +err_morph_back: +   if (morphed) { +      ddi_put16(vxnIOHdl, (uint16_t *)(vxnIOp - MORPH_PORT_SIZE), LANCE_CHIP); +   } + +err_free_regs_map: +   ddi_regs_map_free(&vxnIOHdl); + +err: +   pci_config_teardown(&confHdl); +   return DDI_FAILURE; +} + +/* + *----------------------------------------------------------------------------- + * Vxn_Detach -- + *    Called on module unload + * + * Results: + *    DDI_SUCCESS + *    DDI_FAILURE + * + * Side effects: + *    None    + *----------------------------------------------------------------------------- + */ +static int +Vxn_Detach(dev_info_t *dip, ddi_detach_cmd_t cmd) +{ +   gld_mac_info_t *macInfo; +   vxn_softc_t    *dp; + +   macInfo = (gld_mac_info_t *)ddi_get_driver_private(dip); +   dp = (vxn_softc_t *)macInfo->gldm_private; + +   if (cmd == DDI_DETACH) { +      /* +       * Tear down interrupt +       */ +      ddi_remove_intr(dip, 0,  macInfo->gldm_cookie); +      gld_unregister(macInfo); + +      /* +       * Quiesce hardware +       */ +      Vxn_Stop(macInfo); + +      /* +       * Free driver-data, tx/rx buffers etc +       */ +      Vxn_FreeDriverData(dp); + +      /* +       * Destroy locks +       */ +      mutex_destroy(&dp->intrlock); +      mutex_destroy(&dp->xmitlock); + +      /* +       * Unmorph +       */ +      if (dp->morphed) { +         uint16_t magic; + +         /* Verify that we had morphed earlier */ +         magic = ddi_get16(dp->vxnIOHdl,  +                           (uint16_t *)(dp->vxnIOp - MORPH_PORT_SIZE)); +         if (magic != VMXNET_CHIP) { +            cmn_err(CE_WARN, "%s%d: Vxn_Detach: Adapter not morphed" +                             " magic=0x%08X", dp->drvName, dp->unit, magic); +         } +         else { +            /* Unmorph */ +            ddi_put16(dp->vxnIOHdl,  +                      (uint16_t *)(dp->vxnIOp - MORPH_PORT_SIZE), LANCE_CHIP); + +            /* Verify */ +            magic = ddi_get16(dp->vxnIOHdl,  +                              (uint16_t *)(dp->vxnIOp - MORPH_PORT_SIZE)); +            if (magic != LANCE_CHIP) { +               cmn_err(CE_WARN, "%s%d: Vxn_Detach: Unable to unmorph adapter" +                             " magic=0x%08X", dp->drvName, dp->unit, magic); +            } +         } +      } + +      /* +       * Release resister mappings +       */  +      ddi_regs_map_free(&dp->vxnIOHdl); +      pci_config_teardown(&dp->confHdl); + +      /* +       * Remove from list of interfaces. +       */ +      mutex_enter(&vxnListLock); +      ASSERT(dp != &vxnList); +      dp->prev->next = dp->next; +      dp->next->prev = dp->prev; +      mutex_exit(&vxnListLock); + +      /* +       * Release memory +       */ +      kmem_free(dp, sizeof(*dp)); +      gld_mac_free(macInfo); + +      return DDI_SUCCESS; +   } +   else { +      return DDI_FAILURE; +   } +} + +static   struct module_info vxnminfo = { +  0,                    /* mi_idnum */ +  "vmxnet",             /* mi_idname */ +  0,                    /* mi_minpsz */ +  ETHERMTU,             /* mi_maxpsz */ +  QHIWATER,             /* mi_hiwat */ +  1,                    /* mi_lowat */ +}; + +static   struct qinit vxnrinit = { +   NULL,                /* qi_putp */ +   gld_rsrv,            /* qi_srvp */ +   gld_open,            /* qi_qopen */ +   gld_close,           /* qi_qclose */ +   NULL,                /* qi_qadmin */ +   &vxnminfo,           /* qi_minfo */ +   NULL                 /* qi_mstat */ +}; + +static   struct qinit vxnwinit = { +   gld_wput,            /* qi_putp */ +   gld_wsrv,            /* qi_srvp */ +   NULL,                /* qi_qopen */ +   NULL,                /* qi_qclose */ +   NULL,                /* qi_qadmin */ +   &vxnminfo,           /* qi_minfo */ +   NULL                 /* qi_mstat */ +}; + +static struct streamtab vxn_info = { +   &vxnrinit,           /* st_rdinit */ +   &vxnwinit,           /* st_wrinit */ +   NULL,                /* st_muxrinit */ +   NULL                 /* st_muxwrinit */ +}; + +static   struct cb_ops cb_vxn_ops = { +   nulldev,             /* cb_open */ +   nulldev,             /* cb_close */ +   nodev,               /* cb_strategy */ +   nodev,               /* cb_print */ +   nodev,               /* cb_dump */ +   nodev,               /* cb_read */ +   nodev,               /* cb_write */ +   nodev,               /* cb_ioctl */ +   nodev,               /* cb_devmap */ +   nodev,               /* cb_mmap */ +   nodev,               /* cb_segmap */ +   nochpoll,            /* cb_chpoll */ +   ddi_prop_op,         /* cb_prop_op */ +   &vxn_info,           /* cb_stream */ +   D_NEW|D_MP           /* cb_flag */ +}; + +static   struct dev_ops vxn_ops = { +   DEVO_REV,            /* devo_rev */ +   0,                   /* devo_refcnt */ +   gld_getinfo,         /* devo_getinfo */ +   nulldev,             /* devo_identify */ +   nulldev,             /* devo_probe */ +   Vxn_Attach,          /* devo_attach */ +   Vxn_Detach,          /* devo_detach */ +   nodev,               /* devo_reset */ +   &cb_vxn_ops,         /* devo_cb_ops */ +   NULL,                /* devo_bus_ops */ +   ddi_power            /* devo_power */ +}; + +static struct modldrv modldrv = { +   &mod_driverops, +   ident, +   &vxn_ops, +}; + +static struct modlinkage modlinkage = { +   MODREV_1, {&modldrv, NULL,} +}; + + +/* + * Module load entry point + */ +int +_init(void) +{ +   int err; + +   DPRINTF(5, (CE_CONT, "vxn: _init:\n")); +   /* Initialize interface list */ +   vxnList.next = vxnList.prev = &vxnList; +   mutex_init(&vxnListLock, NULL, MUTEX_DRIVER, NULL); +   if ((err = mod_install(&modlinkage)) != 0) { +      mutex_destroy(&vxnListLock); +   } +   return err; +} + +/* + * Module unload entry point + */ +int +_fini(void) +{ +   int err; + +   DPRINTF(5, (CE_CONT, "vxn: _fini:\n")); +   if ((err = mod_remove(&modlinkage)) == 0) { +      mutex_destroy(&vxnListLock); +   } +   return err; +} + +/* + * Module info entry point + */ +int +_info(struct modinfo *modinfop) +{ +   return (mod_info(&modlinkage, modinfop)); +} + diff --git a/usr/src/uts/intel/io/vmxnet/vmxnet.conf b/usr/src/uts/intel/io/vmxnet/vmxnet.conf new file mode 100644 index 0000000000..eb3b160412 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vmxnet.conf @@ -0,0 +1,24 @@ +# +# CDDL HEADER START +# +# The contents of this file are subject to the terms of the +# Common Development and Distribution License (the "License"). +# You may not use this file except in compliance with the License. +# +# You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE +# or http://www.opensolaris.org/os/licensing. +# See the License for the specific language governing permissions +# and limitations under the License. +# +# When distributing Covered Code, include this CDDL HEADER in each +# file and include the License file at usr/src/OPENSOLARIS.LICENSE. +# If applicable, add the following below this CDDL HEADER, with the +# fields enclosed by brackets "[]" replaced with your own identifying +# information: Portions Copyright [yyyy] [name of copyright owner] +# +# CDDL HEADER END +# +# +# Copyright (c) 2012, Joyent, Inc.  All rights reserved. +# Use is subject to license terms. +# diff --git a/usr/src/uts/intel/io/vmxnet/vmxnet2_def.h b/usr/src/uts/intel/io/vmxnet/vmxnet2_def.h new file mode 100644 index 0000000000..5ea437df72 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vmxnet2_def.h @@ -0,0 +1,436 @@ +/********************************************************* + * Copyright (C) 2004 VMware, Inc. All rights reserved. + * + * 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 version 2 and no 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. + * + * 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 St, Fifth Floor, Boston, MA  02110-1301 USA + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +#ifndef _VMXNET2_DEF_H_ +#define _VMXNET2_DEF_H_ + +#define INCLUDE_ALLOW_USERLEVEL + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMK_MODULE +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_DISTRIBUTE +#include "includeCheck.h" + +#include "net_sg.h" +#include "vmxnet_def.h" + + +/* + * Magic number that identifies this version of the vmxnet protocol. + */ +#define VMXNET2_MAGIC			0xbabe864f + +/* size of the rx ring */ +#define VMXNET2_MAX_NUM_RX_BUFFERS		128 +#define VMXNET2_DEFAULT_NUM_RX_BUFFERS	        100 + + +/* size of the rx ring when enhanced vmxnet is used */ +#define ENHANCED_VMXNET2_MAX_NUM_RX_BUFFERS     512  +#define ENHANCED_VMXNET2_DEFAULT_NUM_RX_BUFFERS 150  + +/* size of the 2nd rx ring */ +#define VMXNET2_MAX_NUM_RX_BUFFERS2             2048  +#define VMXNET2_DEFAULT_NUM_RX_BUFFERS2	        512 + +/* size of the tx ring */ +#define VMXNET2_MAX_NUM_TX_BUFFERS		128 +#define VMXNET2_DEFAULT_NUM_TX_BUFFERS	        100 + +/* size of the tx ring when tso/jf is used */ +#define VMXNET2_MAX_NUM_TX_BUFFERS_TSO          512 +#define VMXNET2_DEFAULT_NUM_TX_BUFFERS_TSO	256 + +enum { +   VMXNET2_OWNERSHIP_DRIVER, +   VMXNET2_OWNERSHIP_DRIVER_PENDING, +   VMXNET2_OWNERSHIP_NIC, +   VMXNET2_OWNERSHIP_NIC_PENDING, +   VMXNET2_OWNERSHIP_NIC_FRAG, +   VMXNET2_OWNERSHIP_DRIVER_FRAG, +}; + +#define VMXNET2_SG_DEFAULT_LENGTH	6 + +typedef struct Vmxnet2_SG_Array { +   uint16	addrType; +   uint16	length; +   NetSG_Elem	sg[VMXNET2_SG_DEFAULT_LENGTH]; +} Vmxnet2_SG_Array; + +typedef struct Vmxnet2_RxRingEntry { +   uint64		paddr;		/* Physical address of the packet data. */ +   uint32		bufferLength;	/* The length of the data at paddr. */ +   uint32		actualLength;	/* The actual length of the received data. */ +   uint16               ownership;	/* Who owns the packet. */ +   uint16		flags;		/* Flags as defined below. */ +   uint32               index;          /*  +                                         * Currently: +                                         * +                                         * This is being used as an packet index to +                                         * rx buffers. +                                         * +                                         * Originally:  +                                         * +					 * was void* driverData ("Driver specific data.") +					 * which was used for sk_buf**s in Linux and +                                         * VmxnetRxBuff*s in Windows.  It could not be +					 * here because the structure needs to be the +					 * same size between architectures, and it was +					 * not used on the device side, anyway.  Look +					 * for its replacement in +					 * Vmxnet_Private.rxRingBuffPtr on Linux and +					 * VmxnetAdapter.rxRingBuffPtr on Windows. +					 */ +} Vmxnet2_RxRingEntry; + +/* + * Vmxnet2_RxRingEntry flags: + *  + * VMXNET2_RX_HW_XSUM_OK       The hardware verified the TCP/UDP checksum. + * VMXNET2_RX_WITH_FRAG        More data is in the 2nd ring + * VMXNET2_RX_FRAG_EOP         This is the last frag, the only valid flag for + *                             2nd ring entry + * + */ +#define VMXNET2_RX_HW_XSUM_OK  0x01 +#define VMXNET2_RX_WITH_FRAG   0x02 +#define VMXNET2_RX_FRAG_EOP    0x04 + +typedef struct Vmxnet2_TxRingEntry { +   uint16		flags;		/* Flags as defined below. */ +   uint16 	        ownership;	/* Who owns this packet. */ +   uint32               extra;          /* +					 * was void* driverData ("Driver specific data.") +					 * which was used for sk_buf*s in Linux and +                                         * VmxnetTxInfo*s in Windows.  It could not be +					 * here because the structure needs to be the +					 * same size between architectures, and it was +					 * not used on the device side, anyway.  Look +					 * for its replacement in +					 * Vmxnet_Private.txRingBuffPtr on Linux and +					 * VmxnetAdapter.txRingBuffPtr on Windows. +					 */ +   uint32               tsoMss;         /* TSO pkt MSS */ +   Vmxnet2_SG_Array	sg;		/* Packet data. */ +} Vmxnet2_TxRingEntry; + +/* + * Vmxnet2_TxRingEntry flags: + * + *   VMXNET2_TX_CAN_KEEP	The implementation can return the tx ring entry  + *				to the driver when it is ready as opposed to  + *				before the transmit call from the driver completes. + *   VMXNET2_TX_RING_LOW	The driver's transmit ring buffer is low on free + *				slots. + *   VMXNET2_TX_HW_XSUM         The hardware should perform the TCP/UDP checksum + *   VMXNET2_TX_TSO             The hardware should do TCP segmentation. + *   VMXNET2_TX_PINNED_BUFFER   The driver used one of the preallocated vmkernel + *                              buffers *and* it has been pinned with Net_PinTxBuffers. + *   VMXNET2_TX_MORE            This is *not* the last tx entry for the pkt. + *                              All flags except VMXNET2_TX_MORE are ignored + *                              for the subsequent tx entries. + */ +#define VMXNET2_TX_CAN_KEEP	     0x0001 +#define VMXNET2_TX_RING_LOW	     0x0002 +#define VMXNET2_TX_HW_XSUM           0x0004 +#define VMXNET2_TX_TSO	             0x0008 +#define VMXNET2_TX_PINNED_BUFFER     0x0010 +#define VMXNET2_TX_MORE              0x0020 + +/* + * Structure used by implementations.  This structure allows the inline + * functions below to be used. + */ +typedef struct Vmxnet2_RxRingInfo { +   Vmxnet2_RxRingEntry    *base;       /* starting addr of the ring */ +   uint32                  nicNext;    /* next entry to use in the ring */ +   uint32                  ringLength; /* # of entries in the ring */ +   PA                      startPA;    /* PA of the starting addr of the ring */ +#ifdef VMX86_DEBUG +   const char             *name; +#endif +} Vmxnet2_RxRingInfo; + +typedef struct Vmxnet2_TxRingInfo { +   Vmxnet2_TxRingEntry    *base;       /* starting addr of the ring */ +   uint32                  nicNext;    /* next entry to use in the ring */ +   uint32                  ringLength; /* # of entries in the ring */ +   PA                      startPA;    /* PA of the starting addr of the ring */ +#ifdef VMX86_DEBUG +   const char             *name; +#endif +} Vmxnet2_TxRingInfo; + +typedef struct Vmxnet2_ImplData { +   Vmxnet2_RxRingInfo    rxRing; +   Vmxnet2_RxRingInfo    rxRing2; +   Vmxnet2_TxRingInfo    txRing; + +   struct PhysMem_Token	  *ddPhysMemToken; +} Vmxnet2_ImplData; + +/*  + * Used internally for performance studies. By default this will be off so there  + * should be no compatibilty or other interferences. + */ + +/* #define ENABLE_VMXNET2_PROFILING    */ + + +#ifdef ENABLE_VMXNET2_PROFILING +typedef struct Vmxnet2_VmmStats { +   uint64      vIntTSC;             /* the time that virtual int was posted */ +   uint64      actionsCount;        /* Number of actions received */ +   uint64      numWasteActions;     /* Number of non-productive actions */ +}  Vmxnet2_VmmStats; +#endif + +typedef struct Vmxnet2_DriverStats { +   uint32	transmits;	   /* # of times that the drivers transmit function */ +				   /*   is called. The driver could transmit more */ +				   /*   than one packet per call. */ +   uint32	pktsTransmitted;   /* # of packets transmitted. */ +   uint32	noCopyTransmits;   /* # of packets that are transmitted without */ +				   /*   copying any data. */ +   uint32	copyTransmits;	   /* # of packets that are transmittted by copying */ +				   /*   the data into a buffer. */ +   uint32	maxTxsPending;	   /* Max # of transmits outstanding. */ +   uint32	txStopped;	   /* # of times that transmits got stopped because */ +				   /*   the tx ring was full. */ +   uint32	txRingOverflow;	   /* # of times that transmits got deferred bc */ +				   /*   the tx ring was full.  This must be >= */ +				   /*   txStopped since there will be one */ +				   /*   txStopped when the ring fills up and then */ +				   /*   one txsRingOverflow for each packet that */ +				   /*   that gets deferred until there is space. */ +   uint32	interrupts;	   /* # of times interrupted. */ +   uint32	pktsReceived;	   /* # of packets received. */ +   uint32	rxBuffersLow;	   /* # of times that the driver was low on */ +				   /*   receive buffers. */ +#ifdef ENABLE_VMXNET2_PROFILING +    Vmxnet2_VmmStats  vmmStats;     /* vmm related stats for perf study */ +#endif +} Vmxnet2_DriverStats; + +/* + * Shared data structure between the vm, the vmm, and the vmkernel. + * This structure was originally arranged to try to group common data  + * on 32-byte cache lines, but bit rot and the fact that we no longer + * run on many CPUs with that cacheline size killed that optimization. + * vmxnet3 should target 128 byte sizes and alignments to optimize for + * the 64 byte cacheline pairs on P4. + */ +typedef struct Vmxnet2_DriverData { +   /* +    * Magic must be first. +    */ +   Vmxnet_DDMagic       magic; + +   /* +    * Receive fields.  +    */ +   uint32		rxRingLength;		/* Length of the receive ring. */ +   uint32		rxDriverNext;		/* Index of the next packet that will */ +						/*   be filled in by the impl */ + +   uint32		rxRingLength2;	        /* Length of the 2nd receive ring. */ +   uint32		rxDriverNext2;	        /* Index of the next packet that will */ +						/*   be filled in by the impl */ + +   uint32		notUsed1;               /* was "irq" */ + +   /* +    * Interface flags and multicast filter. +    */ +   uint32		ifflags; +   uint32		LADRF[VMXNET_MAX_LADRF]; + +   /* +    * Transmit fields +    */ +   uint32               txDontClusterSize;      /* All packets <= this will be transmitted */ +                                                /* immediately, regardless of clustering */ +                                                /* settings [was fill[1]] */ +   uint32		txRingLength;		/* Length of the transmit ring. */ +   uint32		txDriverCur;		/* Index of the next packet to be */ +						/*   returned by the implementation.*/ +   uint32		txDriverNext;		/* Index of the entry in the ring */ +						/*   buffer to use for the next packet.*/ +   uint32		txStopped;  		/* The driver has stopped transmitting */ +						/*   because its ring buffer is full.*/ +   uint32		txClusterLength;	/* Maximum number of packets to */ +						/*   put in the ring buffer before */ +						/*   asking the implementation to */ +						/*   transmit the packets in the buffer.*/ +   uint32		txNumDeferred;          /* Number of packets that have been */ +						/*   queued in the ring buffer since */ +						/*   the last time the implementation */ +						/*   was asked to transmit. */ +   uint32		notUsed3;               /* This field is deprecated but still used */ +                                                /* as minXmitPhysLength on the escher branch. */ +                                                /* It cannot be used for other purposes */ +                                                /* until escher vms no longer are allowed */ +                                                /* to install this driver. */ + +   uint32              totalRxBuffers;          /* used by esx for max rx buffers */ +   uint64              rxBufferPhysStart;       /* used by esx for pinng rx buffers */ +   /* +    * Extra fields for future expansion. +    */ +   uint32		extra[2]; + +   uint16               maxFrags;               /* # of frags the driver can handle */ +   uint16               featureCtl;             /* for driver to enable some feature */ + +   /* +    * The following fields are used to save the nicNext indexes part +    * of implData in the vmkernel when disconnecting the adapter, we +    * need them when we reconnect.  This mechanism is used for +    * checkpointing as well. +    */ +   uint32               savedRxNICNext; +   uint32               savedRxNICNext2; +   uint32               savedTxNICNext; + +   /* +    * Fields used during initialization or debugging. +    */ +   uint32		length; +   uint32		rxRingOffset; +   uint32		rxRingOffset2; +   uint32		txRingOffset;    +   uint32		debugLevel; +   uint32		txBufferPhysStart; +   uint32		txBufferPhysLength; +   uint32		txPktMaxSize; + +   /* +    * Driver statistics. +    */ +   Vmxnet2_DriverStats	stats; +} Vmxnet2_DriverData; + +/*  + * Shared between VMM and Vmkernel part of vmxnet2 to optimize action posting + * VMM writes 1 (don't post) or 0 (okay to post) and vmk reads this. + */ +typedef struct VmxnetVMKShared { +   uint32  dontPostActions;   +} VmxnetVMKShared; + +#if defined VMX86_VMX || defined VMKERNEL + +/* + * Inline functions used to assist the implementation of the vmxnet interface. + */ + +/* + * Get the next empty packet out of the receive ring and move to  + * the next packet. + */ +static INLINE Vmxnet2_RxRingEntry * +Vmxnet2_GetNextRx(Vmxnet2_RxRingInfo *ri, uint16 ownership) +{ +   Vmxnet2_RxRingEntry *rre = ri->base + ri->nicNext; +   if (rre->ownership == ownership) { +      VMXNET_INC(ri->nicNext, ri->ringLength); +   } else { +      rre = NULL; +   } + +   return rre; +} + +/* + * Return ownership of a packet in the receive ring to the driver. + */ +static INLINE void +Vmxnet2_PutRx(Vmxnet2_RxRingEntry *rre, uint32 pktLength, uint16 ownership) +{ +   rre->actualLength = pktLength; +   COMPILER_MEM_BARRIER(); +   rre->ownership = ownership; +} + +/* + * Get the next pending packet out of the transmit ring. + */ +static INLINE Vmxnet2_TxRingEntry * +Vmxnet2_GetNextTx(Vmxnet2_TxRingInfo *ri) +{ +   Vmxnet2_TxRingEntry *txre = ri->base + ri->nicNext; +   if (txre->ownership == VMXNET2_OWNERSHIP_NIC) { +      return txre; +   } else { +      return NULL; +   } +} + +/* + * Move to the next entry in the transmit ring. + */ +static INLINE unsigned int +Vmxnet2_IncNextTx(Vmxnet2_TxRingInfo *ri) +{ +   unsigned int prev = ri->nicNext; +   Vmxnet2_TxRingEntry *txre = ri->base + ri->nicNext; +    +   txre->ownership = VMXNET2_OWNERSHIP_NIC_PENDING; + +   VMXNET_INC(ri->nicNext, ri->ringLength); +   return prev; +} + +/* + * Get the indicated entry from transmit ring. + */ +static INLINE Vmxnet2_TxRingEntry * +Vmxnet2_GetTxEntry(Vmxnet2_TxRingInfo *ri, unsigned int idx) +{ +   return ri->base + idx; +} + +/* + * Get the indicated entry from the given rx ring + */ +static INLINE Vmxnet2_RxRingEntry * +Vmxnet2_GetRxEntry(Vmxnet2_RxRingInfo *ri, unsigned int idx) +{ +   return ri->base + idx; +} + +#endif /* defined VMX86_VMX || defined VMKERNEL */ + +#endif + diff --git a/usr/src/uts/intel/io/vmxnet/vmxnet_def.h b/usr/src/uts/intel/io/vmxnet/vmxnet_def.h new file mode 100644 index 0000000000..703466c995 --- /dev/null +++ b/usr/src/uts/intel/io/vmxnet/vmxnet_def.h @@ -0,0 +1,184 @@ +/********************************************************* + * Copyright (C) 1999 VMware, Inc. All rights reserved. + * + * 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 version 2 and no 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. + * + * 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 St, Fifth Floor, Boston, MA  02110-1301 USA + * + *********************************************************/ + +/********************************************************* + * The contents of this file are subject to the terms of the Common + * Development and Distribution License (the "License") version 1.0 + * and no later version.  You may not use this file except in + * compliance with the License. + * + * You can obtain a copy of the License at + *         http://www.opensource.org/licenses/cddl1.php + * + * See the License for the specific language governing permissions + * and limitations under the License. + * + *********************************************************/ + +#ifndef _VMXNET_DEF_H_ +#define _VMXNET_DEF_H_ + +#define INCLUDE_ALLOW_USERLEVEL + +#define INCLUDE_ALLOW_MODULE +#define INCLUDE_ALLOW_VMK_MODULE +#define INCLUDE_ALLOW_VMKERNEL +#define INCLUDE_ALLOW_DISTRIBUTE +#include "includeCheck.h" + +#include "net_sg.h" +#include "vmnet_def.h" + + +/* + *   Vmxnet I/O ports, used by both the vmxnet driver and  + *   the device emulation code. + */ + +#define VMXNET_INIT_ADDR		0x00 +#define VMXNET_INIT_LENGTH		0x04 +#define VMXNET_TX_ADDR		        0x08 +#define VMXNET_COMMAND_ADDR		0x0c +#define VMXNET_MAC_ADDR			0x10 +#define VMXNET_LOW_VERSION		0x18 +#define VMXNET_HIGH_VERSION		0x1c +#define VMXNET_STATUS_ADDR		0x20 +#define VMXNET_TOE_INIT_ADDR            0x24 +#define VMXNET_APROM_ADDR               0x28 +#define VMXNET_INT_ENABLE_ADDR          0x30 +#define VMXNET_WAKE_PKT_PATTERNS        0x34 + +/* + * Vmxnet command register values. + */ +#define VMXNET_CMD_INTR_ACK		0x0001 +#define VMXNET_CMD_UPDATE_LADRF		0x0002 +#define VMXNET_CMD_UPDATE_IFF		0x0004 +#define VMXNET_CMD_UNUSED 1		0x0008 +#define VMXNET_CMD_UNUSED_2		0x0010 +#define VMXNET_CMD_INTR_DISABLE  	0x0020 +#define VMXNET_CMD_INTR_ENABLE   	0x0040 +#define VMXNET_CMD_UNUSED_3		0x0080 +#define VMXNET_CMD_CHECK_TX_DONE	0x0100 +#define VMXNET_CMD_GET_NUM_RX_BUFFERS	0x0200 +#define VMXNET_CMD_GET_NUM_TX_BUFFERS	0x0400 +#define VMXNET_CMD_PIN_TX_BUFFERS	0x0800 +#define VMXNET_CMD_GET_CAPABILITIES	0x1000 +#define VMXNET_CMD_GET_FEATURES		0x2000 +#define VMXNET_CMD_SET_POWER_FULL       0x4000 +#define VMXNET_CMD_SET_POWER_LOW        0x8000 + +/* + * Vmxnet status register values. + */ +#define VMXNET_STATUS_CONNECTED		0x0001 +#define VMXNET_STATUS_ENABLED		0x0002 +#define VMXNET_STATUS_TX_PINNED         0x0004 + +/* + * Values for the interface flags. + */ +#define VMXNET_IFF_PROMISC		0x01 +#define VMXNET_IFF_BROADCAST		0x02 +#define VMXNET_IFF_MULTICAST		0x04 +#define VMXNET_IFF_DIRECTED             0x08 + +/* + * Length of the multicast address filter. + */ +#define VMXNET_MAX_LADRF		2 + +/* + * Size of Vmxnet APROM.  + */ +#define VMXNET_APROM_SIZE 6 + +/* + * An invalid ring index. + */ +#define VMXNET_INVALID_RING_INDEX	(-1) + +/* + * Features that are implemented by the driver.  These are driver + * specific so not all features will be listed here.  In addition not all + * drivers have to pay attention to these feature flags. + * + *  VMXNET_FEATURE_ZERO_COPY_TX 	The driver won't do any copies as long as + *					the packet length is >  + *					Vmxnet_DriverData.minTxPhysLength. + *  + *  VMXNET_FEATURE_TSO                  The driver will use the TSO capabilities + *                                      of the underlying hardware if available  + *                                      and enabled. + * + *  VMXNET_FEATURE_JUMBO_FRAME          The driver can send/rcv jumbo frame  + * + *  VMXNET_FEATURE_LPD                  The backend can deliver large pkts + */ +#define VMXNET_FEATURE_ZERO_COPY_TX             0x01 +#define VMXNET_FEATURE_TSO                      0x02 +#define VMXNET_FEATURE_JUMBO_FRAME              0x04 +#define VMXNET_FEATURE_LPD                      0x08 + +/* + * Define the set of capabilities required by each feature above + */ +#define VMXNET_FEATURE_ZERO_COPY_TX_CAPS        VMXNET_CAP_SG +#define VMXNET_FEATURE_TSO_CAPS                 VMXNET_CAP_TSO +#define VMXNET_HIGHEST_FEATURE_BIT              VMXNET_FEATURE_TSO + +#define VMXNET_INC(val, max)     \ +   val++;                        \ +   if (UNLIKELY(val == max)) {   \ +      val = 0;                   \ +   } + +/* + * code that just wants to switch on the different versions of the + * guest<->implementation protocol can cast driver data to this. + */ +typedef uint32 Vmxnet_DDMagic; + +/* + * Wake packet pattern commands sent through VMXNET_WAKE_PKT_PATTERNS port + */ + +#define VMXNET_PM_OPCODE_START 3 /* args: cnt of wake packet patterns */ +#define VMXNET_PM_OPCODE_LEN   2 /* args: index of wake packet pattern */ +                                 /*       number of pattern byte values */ +#define VMXNET_PM_OPCODE_DATA  1 /* args: index of wake packet pattern */ +                                 /*       offset in pattern byte values list */ +                                 /*       packet byte offset */ +                                 /*       packet byte value */ +#define VMXNET_PM_OPCODE_END   0 /* args: <none> */ + +typedef union Vmxnet_WakePktCmd { +   uint32 pktData : 32; +   struct { +      unsigned cmd : 2; /* wake packet pattern cmd [from list above] */ +      unsigned cnt : 3; /* cnt wk pkt pttrns 1..MAX_NUM_FILTER_PTTRNS */ +      unsigned ind : 3; /* ind wk pkt pttrn 0..MAX_NUM_FILTER_PTTRNS-1 */ +      unsigned lenOff : 8; /* num pttrn byte vals 1..MAX_PKT_FILTER_SIZE */ +                           /* OR offset in pattern byte values list */ +                           /* 0..MAX_PKT_FILTER_SIZE-1 */ +      unsigned byteOff : 8; /* pkt byte offset 0..MAX_PKT_FILTER_SIZE-1 */ +      unsigned byteVal : 8; /* packet byte value 0..255 */ +   } pktPttrn; +} Vmxnet_WakePktCmd; + +#endif /* _VMXNET_DEF_H_ */ diff --git a/usr/src/uts/intel/io/vmxnet3s/vmxnet3_rx.c b/usr/src/uts/intel/io/vmxnet3s/vmxnet3_rx.c index 8004a3eb3e..8dd039e8cc 100644 --- a/usr/src/uts/intel/io/vmxnet3s/vmxnet3_rx.c +++ b/usr/src/uts/intel/io/vmxnet3s/vmxnet3_rx.c @@ -14,6 +14,7 @@   */  /*   * Copyright (c) 2013, 2016 by Delphix. All rights reserved. + * Copyright 2018 Joyent, Inc.   */  #include <vmxnet3.h> @@ -322,7 +323,7 @@ vmxnet3_rx_hwcksum(vmxnet3_softc_t *dp, mblk_t *mp,  		VMXNET3_DEBUG(dp, 3, "rx cksum flags = 0x%x\n", flags); -		(void) hcksum_assoc(mp, NULL, NULL, 0, 0, 0, 0, flags, 0); +		mac_hcksum_set(mp, 0, 0, 0, 0, flags);  	}  } diff --git a/usr/src/uts/intel/io/vmxnet3s/vmxnet3_tx.c b/usr/src/uts/intel/io/vmxnet3s/vmxnet3_tx.c index 8a9f05e690..8769d938ab 100644 --- a/usr/src/uts/intel/io/vmxnet3s/vmxnet3_tx.c +++ b/usr/src/uts/intel/io/vmxnet3s/vmxnet3_tx.c @@ -15,6 +15,7 @@  /*   * Copyright (c) 2012, 2016 by Delphix. All rights reserved. + * Copyright 2018 Joyent, Inc.   */  #include <vmxnet3.h> @@ -79,7 +80,7 @@ vmxnet3_tx_prepare_offload(vmxnet3_softc_t *dp, vmxnet3_offload_t *ol,  	ol->hlen = 0;  	ol->msscof = 0; -	hcksum_retrieve(mp, NULL, NULL, &start, &stuff, NULL, &value, &flags); +	mac_hcksum_get(mp, &start, &stuff, NULL, &value, &flags);  	mac_lso_get(mp, &mss, &lso_flag); | 
