xref: /titanic_52/usr/src/uts/common/io/fibre-channel/fca/oce/oce_main.c (revision 3abb112f8485b33b6b9b52b340bede0a333c10bf)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /* Copyright © 2003-2011 Emulex. All rights reserved.  */
23 
24 
25 /*
26  * Source file containing the implementation of the driver entry points
27  * and related helper functions
28  */
29 
30 #include <oce_impl.h>
31 #include <oce_stat.h>
32 #include <oce_ioctl.h>
33 
34 #define	ATTACH_DEV_INIT 	0x1
35 #define	ATTACH_FM_INIT		0x2
36 #define	ATTACH_PCI_CFG		0x4
37 #define	ATTACH_LOCK_INIT	0x8
38 #define	ATTACH_PCI_INIT 	0x10
39 #define	ATTACH_HW_INIT		0x20
40 #define	ATTACH_SETUP_TXRX 	0x40
41 #define	ATTACH_SETUP_ADAP	0x80
42 #define	ATTACH_SETUP_INTR	0x100
43 #define	ATTACH_STAT_INIT	0x200
44 #define	ATTACH_MAC_REG		0x400
45 
46 /* ---[ globals and externs ]-------------------------------------------- */
47 const char oce_ident_string[] = OCE_IDENT_STRING;
48 const char oce_mod_name[] = OCE_MOD_NAME;
49 struct oce_dev *oce_dev_list[MAX_DEVS + 1];	/* Last entry is invalid */
50 
51 /* driver properties */
52 static const char flow_control[]	 = "flow_control";
53 static const char mtu_prop_name[]	 = "oce_default_mtu";
54 static const char tx_ring_size_name[]	 = "tx_ring_size";
55 static const char tx_bcopy_limit_name[]	 = "tx_bcopy_limit";
56 static const char rx_bcopy_limit_name[]	 = "rx_bcopy_limit";
57 static const char rx_frag_size_name[]	 = "rx_frag_size";
58 static const char rx_max_bufs_name[]	 = "rx_max_bufs";
59 static const char fm_cap_name[]		 = "oce_fm_capability";
60 static const char log_level_name[]	 = "oce_log_level";
61 static const char lso_capable_name[]	 = "lso_capable";
62 static const char rx_pkt_per_intr_name[] = "rx_pkts_per_intr";
63 static const char tx_reclaim_threshold_name[] = "tx_reclaim_threshold";
64 static const char rx_rings_name[]	 = "max_rx_rings";
65 static const char tx_rings_name[]	 = "max_tx_rings";
66 
67 /* --[ static function prototypes here ]------------------------------- */
68 static int oce_attach(dev_info_t *devinfo, ddi_attach_cmd_t cmd);
69 static int oce_detach(dev_info_t *dip, ddi_detach_cmd_t cmd);
70 static int oce_quiesce(dev_info_t *dip);
71 static int oce_suspend(dev_info_t *dip);
72 static int oce_resume(dev_info_t *dip);
73 static void oce_unconfigure(struct oce_dev *dev);
74 static void oce_init_locks(struct oce_dev *dev);
75 static void oce_destroy_locks(struct oce_dev *dev);
76 static void oce_get_params(struct oce_dev *dev);
77 static int oce_get_prop(struct oce_dev *dev, char *propname, int minval,
78     int maxval, int defval, uint32_t *values);
79 
80 static struct cb_ops oce_cb_ops = {
81 	nulldev,		/* cb_open */
82 	nulldev,		/* cb_close */
83 	nodev,			/* cb_strategy */
84 	nodev,			/* cb_print */
85 	nodev,			/* cb_dump */
86 	nodev,			/* cb_read */
87 	nodev,			/* cb_write */
88 	nodev,			/* cb_ioctl */
89 	nodev,			/* cb_devmap */
90 	nodev,			/* cb_mmap */
91 	nodev,			/* cb_segmap */
92 	nochpoll,		/* cb_chpoll */
93 	ddi_prop_op,	/* cb_prop_op */
94 	NULL,			/* cb_stream */
95 	D_MP,			/* cb_flag */
96 	CB_REV,			/* cb_rev */
97 	nodev,			/* cb_aread */
98 	nodev			/* cb_awrite */
99 };
100 
101 static struct dev_ops oce_dev_ops = {
102 	DEVO_REV,	/* devo_rev */
103 	0,		/* devo_refcnt */
104 	NULL,		/* devo_getinfo */
105 	NULL,		/* devo_identify */
106 	nulldev,	/* devo_probe */
107 	oce_attach,	/* devo_attach */
108 	oce_detach,	/* devo_detach */
109 	nodev,		/* devo_reset */
110 	&oce_cb_ops,	/* devo_cb_ops */
111 	NULL,		/* devo_bus_ops */
112 	nodev,		/* devo_power */
113 	oce_quiesce	/* devo_quiesce */
114 };
115 
116 static struct modldrv oce_drv = {
117 	&mod_driverops,	/* Type of module.  This one is a driver */
118 	(char *)oce_ident_string, /* Description string */
119 	&oce_dev_ops,	/* driver ops */
120 };
121 
122 static struct modlinkage oce_mod_linkage = {
123 	MODREV_1, &oce_drv, NULL
124 };
125 
126 #define	OCE_M_CB_FLAGS	(MC_IOCTL | MC_GETCAPAB | MC_SETPROP | MC_GETPROP | \
127     MC_PROPINFO)
128 static mac_callbacks_t oce_mac_cb = {
129 	OCE_M_CB_FLAGS,		/* mc_callbacks */
130 	oce_m_stat,		/* mc_getstat */
131 	oce_m_start,		/* mc_start */
132 	oce_m_stop,		/* mc_stop */
133 	oce_m_promiscuous,	/* mc_setpromisc */
134 	oce_m_multicast,	/* mc_multicast */
135 	oce_m_unicast,		/* mc_unicast */
136 	oce_m_send,		/* mc_tx */
137 	NULL,			/* mc_reserve */
138 	oce_m_ioctl,		/* mc_ioctl */
139 	oce_m_getcap,		/* mc_getcapab */
140 	NULL,			/* open */
141 	NULL,			/* close */
142 	oce_m_setprop,		/* set properties */
143 	oce_m_getprop,		/* get properties */
144 	oce_m_propinfo		/* properties info */
145 };
146 
147 extern char *oce_priv_props[];
148 
149 /* Module Init */
150 int
151 _info(struct modinfo *modinfop)
152 {
153 	return (mod_info(&oce_mod_linkage, modinfop));
154 } /* _info */
155 
156 int
157 _init(void)
158 {
159 	int ret = 0;
160 
161 	/* install the module */
162 	mac_init_ops(&oce_dev_ops, "oce");
163 
164 	ret = mod_install(&oce_mod_linkage);
165 	if (ret) {
166 		cmn_err(CE_WARN, "mod_install failed  rval=%x", ret);
167 	}
168 
169 	return (ret);
170 } /* _init */
171 
172 
173 int
174 _fini(void)
175 {
176 	int ret = 0;
177 	/* remove the module */
178 	ret = mod_remove(&oce_mod_linkage);
179 	if (ret != 0) {
180 		return (ret);
181 	}
182 
183 	mac_fini_ops(&oce_dev_ops);
184 
185 	return (ret);
186 } /* _fini */
187 
188 
189 static int
190 oce_attach(dev_info_t *dip, ddi_attach_cmd_t cmd)
191 {
192 	int ret = 0;
193 	struct oce_dev *dev = NULL;
194 	mac_register_t *mac;
195 	uint8_t dev_index = 0;
196 
197 	switch (cmd) {
198 	case DDI_RESUME:
199 		return (oce_resume(dip));
200 	default:
201 		return (DDI_FAILURE);
202 
203 	case DDI_ATTACH:
204 		break;
205 	}
206 
207 	/* allocate dev */
208 	dev = kmem_zalloc(sizeof (struct oce_dev), KM_SLEEP);
209 
210 	/* populate the dev structure */
211 	dev->dip = dip;
212 	dev->dev_id = ddi_get_instance(dip);
213 	dev->suspended = B_FALSE;
214 
215 	dev->dev_list_index = MAX_DEVS;
216 	while (dev_index < MAX_DEVS) {
217 		(void) atomic_cas_ptr(&oce_dev_list[dev_index], NULL, dev);
218 		if (oce_dev_list[dev_index] == dev) {
219 			break;
220 		}
221 		dev_index++;
222 	}
223 	if (dev_index == MAX_DEVS) {
224 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
225 		    "Too many oce devices on the system. Failed to attach.");
226 		goto attach_fail;
227 	}
228 	dev->dev_list_index = dev_index;
229 
230 	/* get the parameters */
231 	oce_get_params(dev);
232 
233 	/*
234 	 * set the ddi driver private data pointer. This is
235 	 * sent to all mac callback entry points
236 	 */
237 	ddi_set_driver_private(dip, dev);
238 
239 	dev->attach_state |= ATTACH_DEV_INIT;
240 
241 	oce_fm_init(dev);
242 	dev->attach_state |= ATTACH_FM_INIT;
243 
244 	ret = pci_config_setup(dev->dip, &dev->pci_cfg_handle);
245 	if (ret != DDI_SUCCESS) {
246 		oce_log(dev, CE_WARN, MOD_CONFIG,
247 		    "Map PCI config failed with  %d", ret);
248 		goto attach_fail;
249 	}
250 	dev->attach_state |= ATTACH_PCI_CFG;
251 
252 	ret = oce_identify_hw(dev);
253 
254 	if (ret != DDI_SUCCESS) {
255 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
256 		    "Device Unknown");
257 		goto attach_fail;
258 	}
259 
260 	ret = oce_get_bdf(dev);
261 	if (ret != DDI_SUCCESS) {
262 		oce_log(dev, CE_WARN, MOD_CONFIG,
263 		    "Failed to read BDF, status = 0x%x", ret);
264 		goto attach_fail;
265 	}
266 	/* Update the dev->rss */
267 	oce_dev_rss_ready(dev);
268 
269 	/* setup PCI bars */
270 	ret = oce_pci_init(dev);
271 	if (ret != DDI_SUCCESS) {
272 		oce_log(dev, CE_WARN, MOD_CONFIG,
273 		    "PCI initialization failed with %d", ret);
274 		goto attach_fail;
275 	}
276 	dev->attach_state |= ATTACH_PCI_INIT;
277 
278 	ret = oce_setup_intr(dev);
279 	if (ret != DDI_SUCCESS) {
280 		oce_log(dev, CE_WARN, MOD_CONFIG,
281 		    "Interrupt setup failed with %d", ret);
282 		goto attach_fail;
283 
284 	}
285 	dev->attach_state |= ATTACH_SETUP_INTR;
286 
287 	/* initialize locks */
288 	oce_init_locks(dev);
289 	dev->attach_state |= ATTACH_LOCK_INIT;
290 
291 
292 	/* HW init */
293 	ret = oce_hw_init(dev);
294 	if (ret != DDI_SUCCESS) {
295 		oce_log(dev, CE_WARN, MOD_CONFIG,
296 		    "HW initialization failed with %d", ret);
297 		goto attach_fail;
298 	}
299 	dev->attach_state |= ATTACH_HW_INIT;
300 
301 	ret = oce_init_txrx(dev);
302 	if (ret  != DDI_SUCCESS) {
303 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
304 		    "Failed to init rings");
305 		goto attach_fail;
306 	}
307 	dev->attach_state |= ATTACH_SETUP_TXRX;
308 
309 	ret = oce_setup_adapter(dev);
310 	if (ret != DDI_SUCCESS) {
311 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
312 		    "Failed to setup adapter");
313 		goto attach_fail;
314 	}
315 	dev->attach_state |=  ATTACH_SETUP_ADAP;
316 
317 
318 	ret = oce_stat_init(dev);
319 	if (ret != DDI_SUCCESS) {
320 		oce_log(dev, CE_WARN, MOD_CONFIG,
321 		    "kstat setup Failed with %d", ret);
322 		goto attach_fail;
323 	}
324 	dev->attach_state |= ATTACH_STAT_INIT;
325 
326 	/* mac_register_t */
327 	oce_log(dev, CE_NOTE, MOD_CONFIG,
328 	    "MAC_VERSION = 0x%x", MAC_VERSION);
329 	mac = mac_alloc(MAC_VERSION);
330 	if (mac == NULL) {
331 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
332 		    "MAC allocation Failed");
333 		goto attach_fail;
334 	}
335 	/*
336 	 * fill the mac structure before calling mac_register
337 	 */
338 	mac->m_type_ident = MAC_PLUGIN_IDENT_ETHER;
339 	mac->m_driver = dev;
340 	mac->m_dip = dip;
341 	mac->m_src_addr = dev->mac_addr;
342 	mac->m_callbacks = &oce_mac_cb;
343 	mac->m_min_sdu = 0;
344 	mac->m_max_sdu = dev->mtu;
345 	mac->m_margin = VTAG_SIZE;
346 	mac->m_priv_props = oce_priv_props;
347 
348 	oce_log(dev, CE_NOTE, MOD_CONFIG,
349 	    "Driver Private structure = 0x%p", (void *)dev);
350 
351 	/* now register with GLDv3 */
352 	ret = mac_register(mac, (mac_handle_t *)&dev->mac_handle);
353 	/* regardless of the status, free mac_register */
354 	mac_free(mac);
355 	mac = NULL;
356 	if (ret != DDI_SUCCESS) {
357 		oce_log(dev, CE_WARN, MOD_CONFIG,
358 		    "MAC registration failed :0x%x", ret);
359 		goto attach_fail;
360 
361 	}
362 
363 	/* correct link status only after start */
364 	dev->link_status = LINK_STATE_UNKNOWN;
365 	mac_link_update(dev->mac_handle, dev->link_status);
366 
367 	dev->attach_state |= ATTACH_MAC_REG;
368 	dev->state |= STATE_INIT;
369 
370 	oce_log(dev, CE_NOTE, MOD_CONFIG, "%s",
371 	    "ATTACH SUCCESS");
372 
373 	return (DDI_SUCCESS);
374 
375 attach_fail:
376 	oce_unconfigure(dev);
377 	return (DDI_FAILURE);
378 } /* oce_attach */
379 
380 static int
381 oce_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
382 {
383 	struct oce_dev *dev;
384 	int pcnt = 0;
385 	int qid;
386 
387 	dev = ddi_get_driver_private(dip);
388 	if (dev == NULL) {
389 		return (DDI_FAILURE);
390 	}
391 	oce_log(dev, CE_NOTE, MOD_CONFIG,
392 	    "Detaching driver: cmd = 0x%x", cmd);
393 
394 	switch (cmd) {
395 	default:
396 		return (DDI_FAILURE);
397 	case DDI_SUSPEND:
398 		return (oce_suspend(dip));
399 	case DDI_DETACH:
400 		break;
401 	} /* switch cmd */
402 
403 	/* Fail detach if MAC unregister is unsuccessfule */
404 	if (mac_unregister(dev->mac_handle) != 0) {
405 		oce_log(dev, CE_WARN, MOD_CONFIG, "%s",
406 		    "Failed to unregister MAC ");
407 	}
408 	dev->attach_state &= ~ATTACH_MAC_REG;
409 
410 	/* check if the detach is called with out stopping */
411 	DEV_LOCK(dev);
412 	if (dev->state & STATE_MAC_STARTED) {
413 		dev->state &= ~STATE_MAC_STARTED;
414 		oce_stop(dev);
415 		DEV_UNLOCK(dev);
416 	} else
417 		DEV_UNLOCK(dev);
418 
419 	/*
420 	 * Wait for Packets sent up to be freed
421 	 */
422 	for (qid = 0; qid < dev->rx_rings; qid++) {
423 		pcnt = oce_rx_pending(dev, dev->rq[qid], DEFAULT_DRAIN_TIME);
424 		if (pcnt != 0) {
425 			oce_log(dev, CE_WARN, MOD_CONFIG,
426 			    "%d Pending Buffers Detach failed", pcnt);
427 			return (DDI_FAILURE);
428 		}
429 	}
430 	oce_unconfigure(dev);
431 
432 	return (DDI_SUCCESS);
433 } /* oce_detach */
434 
435 static int
436 oce_quiesce(dev_info_t *dip)
437 {
438 	int ret = DDI_SUCCESS;
439 	struct oce_dev *dev = ddi_get_driver_private(dip);
440 
441 	if (dev == NULL) {
442 		return (DDI_FAILURE);
443 	}
444 	if (dev->suspended) {
445 		return (DDI_SUCCESS);
446 	}
447 
448 	oce_chip_di(dev);
449 
450 	ret = oce_reset_fun(dev);
451 
452 	return (ret);
453 }
454 
455 static int
456 oce_suspend(dev_info_t *dip)
457 {
458 	struct oce_dev *dev = ddi_get_driver_private(dip);
459 
460 	mutex_enter(&dev->dev_lock);
461 	/* Suspend the card */
462 	dev->suspended = B_TRUE;
463 	/* stop the adapter */
464 	if (dev->state & STATE_MAC_STARTED) {
465 		oce_stop(dev);
466 		oce_unsetup_adapter(dev);
467 	}
468 	dev->state &= ~STATE_MAC_STARTED;
469 	mutex_exit(&dev->dev_lock);
470 	return (DDI_SUCCESS);
471 } /* oce_suspend */
472 
473 static int
474 oce_resume(dev_info_t *dip)
475 {
476 	struct oce_dev *dev;
477 	int ret;
478 
479 	/* get the dev pointer from dip */
480 	dev = ddi_get_driver_private(dip);
481 	mutex_enter(&dev->dev_lock);
482 	if (!dev->suspended) {
483 		mutex_exit(&dev->dev_lock);
484 		return (DDI_SUCCESS);
485 	}
486 	if (!(dev->state & STATE_MAC_STARTED)) {
487 		ret = oce_setup_adapter(dev);
488 		if (ret != DDI_SUCCESS) {
489 			mutex_exit(&dev->dev_lock);
490 			return (DDI_FAILURE);
491 		}
492 		ret = oce_start(dev);
493 		if (ret != DDI_SUCCESS) {
494 			mutex_exit(&dev->dev_lock);
495 			return (DDI_FAILURE);
496 		}
497 	}
498 	dev->suspended = B_FALSE;
499 	dev->state |= STATE_MAC_STARTED;
500 	mutex_exit(&dev->dev_lock);
501 	return (ret);
502 } /* oce_resume */
503 
504 static void
505 oce_init_locks(struct oce_dev *dev)
506 {
507 	/* initialize locks */
508 	mutex_init(&dev->dev_lock, NULL, MUTEX_DRIVER,
509 	    DDI_INTR_PRI(dev->intr_pri));
510 	mutex_init(&dev->bmbx_lock, NULL, MUTEX_DRIVER,
511 	    DDI_INTR_PRI(dev->intr_pri));
512 } /* oce_init_locks */
513 
514 static void
515 oce_destroy_locks(struct oce_dev *dev)
516 {
517 	mutex_destroy(&dev->dev_lock);
518 	mutex_destroy(&dev->bmbx_lock);
519 } /* oce_destroy_locks */
520 
521 static void
522 oce_unconfigure(struct oce_dev *dev)
523 {
524 	uint32_t state = dev->attach_state;
525 
526 	if (state & ATTACH_MAC_REG) {
527 		(void) mac_unregister(dev->mac_handle);
528 	}
529 	if (state & ATTACH_STAT_INIT) {
530 		oce_stat_fini(dev);
531 	}
532 	if (state & ATTACH_SETUP_ADAP) {
533 		oce_unsetup_adapter(dev);
534 	}
535 	if (state & ATTACH_SETUP_TXRX) {
536 		oce_fini_txrx(dev);
537 	}
538 	if (state & ATTACH_HW_INIT) {
539 		oce_hw_fini(dev);
540 	}
541 	if (state & ATTACH_LOCK_INIT) {
542 		oce_destroy_locks(dev);
543 	}
544 	if (state & ATTACH_SETUP_INTR) {
545 		(void) oce_teardown_intr(dev);
546 	}
547 	if (state & ATTACH_PCI_INIT) {
548 		oce_pci_fini(dev);
549 	}
550 	if (state & ATTACH_PCI_CFG) {
551 		pci_config_teardown(&dev->pci_cfg_handle);
552 	}
553 	if (state & ATTACH_FM_INIT) {
554 		oce_fm_fini(dev);
555 	}
556 	if (state & ATTACH_DEV_INIT) {
557 		ddi_set_driver_private(dev->dip, NULL);
558 		oce_dev_list[dev->dev_list_index] = NULL;
559 		kmem_free(dev, sizeof (struct oce_dev));
560 	}
561 } /* oce_unconfigure */
562 
563 static void
564 oce_get_params(struct oce_dev *dev)
565 {
566 	uint32_t log_level;
567 	uint16_t mod_mask;
568 	uint16_t severity;
569 	/*
570 	 * Allowed values for the driver parameters. If all values in a range
571 	 * is allowed, the the array has only one value.
572 	 */
573 	uint32_t fc_values[] = {OCE_FC_NONE, OCE_FC_TX, OCE_FC_RX,
574 	    OCE_DEFAULT_FLOW_CONTROL, END};
575 	uint32_t mtu_values[] = {OCE_MIN_MTU, OCE_MAX_MTU, END};
576 	uint32_t tx_rs_values[] = {SIZE_256, SIZE_512, SIZE_1K, SIZE_2K, END};
577 	uint32_t tx_bcl_values[] = {SIZE_128, SIZE_256, SIZE_512, SIZE_1K,
578 	    SIZE_2K, END};
579 	uint32_t rx_bcl_values[] = {SIZE_128, SIZE_256, SIZE_512, SIZE_1K,
580 	    SIZE_2K, END};
581 	uint32_t rq_fs_values[] = {SIZE_2K, SIZE_4K, SIZE_8K, END};
582 	uint32_t rq_mb_values[] = {SIZE_2K, SIZE_4K, SIZE_8K, END};
583 	uint32_t lso_capable_values[] = {0, 1, END};
584 	uint32_t fm_caps_values[] = {DDI_FM_NOT_CAPABLE, OCE_FM_CAPABILITY,
585 	    END};
586 	uint32_t tx_rt_values[] = {END};
587 	uint32_t rx_ppi_values[] = {END};
588 	uint32_t rx_rings_values[] = {END};
589 	uint32_t tx_rings_values[] = {END};
590 	uint32_t log_level_values[] = {END};
591 
592 	/* non tunables  */
593 	dev->rx_ring_size = OCE_DEFAULT_RX_RING_SIZE;
594 
595 	/* configurable parameters */
596 	dev->flow_control = oce_get_prop(dev, (char *)flow_control, OCE_FC_NONE,
597 	    OCE_DEFAULT_FLOW_CONTROL, OCE_DEFAULT_FLOW_CONTROL, fc_values);
598 
599 	dev->mtu = oce_get_prop(dev, (char *)mtu_prop_name, OCE_MIN_MTU,
600 	    OCE_MAX_MTU, OCE_MIN_MTU, mtu_values);
601 
602 	dev->tx_ring_size = oce_get_prop(dev, (char *)tx_ring_size_name,
603 	    SIZE_256, SIZE_2K, OCE_DEFAULT_TX_RING_SIZE, tx_rs_values);
604 
605 	dev->tx_bcopy_limit = oce_get_prop(dev, (char *)tx_bcopy_limit_name,
606 	    SIZE_128, SIZE_2K, OCE_DEFAULT_TX_BCOPY_LIMIT, tx_bcl_values);
607 
608 	dev->rx_bcopy_limit = oce_get_prop(dev, (char *)rx_bcopy_limit_name,
609 	    SIZE_128, SIZE_2K, OCE_DEFAULT_RX_BCOPY_LIMIT, rx_bcl_values);
610 
611 	dev->rq_frag_size = oce_get_prop(dev, (char *)rx_frag_size_name,
612 	    SIZE_2K, SIZE_8K, OCE_RQ_BUF_SIZE, rq_fs_values);
613 
614 	dev->rq_max_bufs = oce_get_prop(dev, (char *)rx_max_bufs_name, SIZE_2K,
615 	    SIZE_8K, OCE_RQ_NUM_BUFFERS, rq_mb_values);
616 
617 	dev->lso_capable = oce_get_prop(dev, (char *)lso_capable_name, 0,
618 	    1, 1, lso_capable_values);
619 
620 	dev->fm_caps = oce_get_prop(dev, (char *)fm_cap_name,
621 	    DDI_FM_NOT_CAPABLE, OCE_FM_CAPABILITY, OCE_FM_CAPABILITY,
622 	    fm_caps_values);
623 
624 	dev->tx_reclaim_threshold = oce_get_prop(dev,
625 	    (char *)tx_reclaim_threshold_name, 0, dev->tx_ring_size/2,
626 	    OCE_DEFAULT_TX_RECLAIM_THRESHOLD, tx_rt_values);
627 
628 	dev->rx_pkt_per_intr = oce_get_prop(dev, (char *)rx_pkt_per_intr_name,
629 	    0, dev->rx_ring_size/2, OCE_DEFAULT_RX_PKT_PER_INTR, rx_ppi_values);
630 
631 	dev->rx_rings = oce_get_prop(dev, (char *)rx_rings_name,
632 	    OCE_MIN_RQ, OCE_MAX_RQ, OCE_DEFAULT_RQS, rx_rings_values);
633 
634 	dev->tx_rings = oce_get_prop(dev, (char *)tx_rings_name,
635 	    OCE_DEFAULT_WQS, OCE_DEFAULT_WQS, OCE_DEFAULT_WQS, tx_rings_values);
636 
637 	log_level = oce_get_prop(dev, (char *)log_level_name, 0,
638 	    OCE_MAX_LOG_SETTINGS, OCE_DEFAULT_LOG_SETTINGS, log_level_values);
639 
640 	severity = (uint16_t)(log_level & 0xffff);
641 	mod_mask = (uint16_t)(log_level >> 16);
642 	if (mod_mask > MOD_ISR) {
643 		mod_mask = 0;
644 	}
645 	if (severity > CE_IGNORE) {
646 		severity = 0;
647 	}
648 
649 	dev->mod_mask = mod_mask;
650 	dev->severity = severity;
651 } /* oce_get_params */
652 
653 static int
654 oce_get_prop(struct oce_dev *dev, char *propname, int minval, int maxval,
655     int defval, uint32_t *values)
656 {
657 	int value = 0;
658 	int i = 0;
659 
660 	value = ddi_prop_get_int(DDI_DEV_T_ANY, dev->dip,
661 	    DDI_PROP_DONTPASS, propname, defval);
662 
663 	if (value > maxval)
664 		value = maxval;
665 
666 	if (value < minval)
667 		value = minval;
668 
669 	while (values[i] != 0xdeadface) {
670 		if (values[i] == value) {
671 			break;
672 		}
673 		i++;
674 	}
675 
676 	if ((i != 0) && (values[i] == 0xdeadface)) {
677 		value = defval;
678 	}
679 
680 	return (value);
681 }
682