154dfc97bSShailend Chand /*-
254dfc97bSShailend Chand * SPDX-License-Identifier: BSD-3-Clause
354dfc97bSShailend Chand *
4d438b4efSShailend Chand * Copyright (c) 2023-2024 Google LLC
554dfc97bSShailend Chand *
654dfc97bSShailend Chand * Redistribution and use in source and binary forms, with or without modification,
754dfc97bSShailend Chand * are permitted provided that the following conditions are met:
854dfc97bSShailend Chand *
954dfc97bSShailend Chand * 1. Redistributions of source code must retain the above copyright notice, this
1054dfc97bSShailend Chand * list of conditions and the following disclaimer.
1154dfc97bSShailend Chand *
1254dfc97bSShailend Chand * 2. Redistributions in binary form must reproduce the above copyright notice,
1354dfc97bSShailend Chand * this list of conditions and the following disclaimer in the documentation
1454dfc97bSShailend Chand * and/or other materials provided with the distribution.
1554dfc97bSShailend Chand *
1654dfc97bSShailend Chand * 3. Neither the name of the copyright holder nor the names of its contributors
1754dfc97bSShailend Chand * may be used to endorse or promote products derived from this software without
1854dfc97bSShailend Chand * specific prior written permission.
1954dfc97bSShailend Chand *
2054dfc97bSShailend Chand * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
2154dfc97bSShailend Chand * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
2254dfc97bSShailend Chand * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
2354dfc97bSShailend Chand * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
2454dfc97bSShailend Chand * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
2554dfc97bSShailend Chand * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
2654dfc97bSShailend Chand * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
2754dfc97bSShailend Chand * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
2854dfc97bSShailend Chand * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
2954dfc97bSShailend Chand * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
3054dfc97bSShailend Chand */
3154dfc97bSShailend Chand #include "gve.h"
3254dfc97bSShailend Chand #include "gve_adminq.h"
33d438b4efSShailend Chand #include "gve_dqo.h"
3454dfc97bSShailend Chand
35*22fe926aSVee Agarwal #define GVE_DRIVER_VERSION "GVE-FBSD-1.3.3\n"
365f62584aSShailend Chand #define GVE_VERSION_MAJOR 1
372348ac89SShailend Chand #define GVE_VERSION_MINOR 3
38*22fe926aSVee Agarwal #define GVE_VERSION_SUB 3
3954dfc97bSShailend Chand
4054dfc97bSShailend Chand #define GVE_DEFAULT_RX_COPYBREAK 256
4154dfc97bSShailend Chand
421bbdfb0bSXin LI /* Devices supported by this driver. */
431bbdfb0bSXin LI static struct gve_dev {
441bbdfb0bSXin LI uint16_t vendor_id;
451bbdfb0bSXin LI uint16_t device_id;
461bbdfb0bSXin LI const char *name;
471bbdfb0bSXin LI } gve_devs[] = {
481bbdfb0bSXin LI { PCI_VENDOR_ID_GOOGLE, PCI_DEV_ID_GVNIC, "gVNIC" }
491bbdfb0bSXin LI };
501bbdfb0bSXin LI
5154dfc97bSShailend Chand struct sx gve_global_lock;
5254dfc97bSShailend Chand
5354dfc97bSShailend Chand static int
gve_verify_driver_compatibility(struct gve_priv * priv)5454dfc97bSShailend Chand gve_verify_driver_compatibility(struct gve_priv *priv)
5554dfc97bSShailend Chand {
5654dfc97bSShailend Chand int err;
5754dfc97bSShailend Chand struct gve_driver_info *driver_info;
5854dfc97bSShailend Chand struct gve_dma_handle driver_info_mem;
5954dfc97bSShailend Chand
6054dfc97bSShailend Chand err = gve_dma_alloc_coherent(priv, sizeof(struct gve_driver_info),
6154dfc97bSShailend Chand PAGE_SIZE, &driver_info_mem);
6254dfc97bSShailend Chand
6354dfc97bSShailend Chand if (err != 0)
6454dfc97bSShailend Chand return (ENOMEM);
6554dfc97bSShailend Chand
6654dfc97bSShailend Chand driver_info = driver_info_mem.cpu_addr;
6754dfc97bSShailend Chand
6854dfc97bSShailend Chand *driver_info = (struct gve_driver_info) {
6954dfc97bSShailend Chand .os_type = 3, /* Freebsd */
7054dfc97bSShailend Chand .driver_major = GVE_VERSION_MAJOR,
7154dfc97bSShailend Chand .driver_minor = GVE_VERSION_MINOR,
7254dfc97bSShailend Chand .driver_sub = GVE_VERSION_SUB,
7354dfc97bSShailend Chand .os_version_major = htobe32(FBSD_VERSION_MAJOR),
7454dfc97bSShailend Chand .os_version_minor = htobe32(FBSD_VERSION_MINOR),
7554dfc97bSShailend Chand .os_version_sub = htobe32(FBSD_VERSION_PATCH),
7654dfc97bSShailend Chand .driver_capability_flags = {
7754dfc97bSShailend Chand htobe64(GVE_DRIVER_CAPABILITY_FLAGS1),
7854dfc97bSShailend Chand htobe64(GVE_DRIVER_CAPABILITY_FLAGS2),
7954dfc97bSShailend Chand htobe64(GVE_DRIVER_CAPABILITY_FLAGS3),
8054dfc97bSShailend Chand htobe64(GVE_DRIVER_CAPABILITY_FLAGS4),
8154dfc97bSShailend Chand },
8254dfc97bSShailend Chand };
8354dfc97bSShailend Chand
8454dfc97bSShailend Chand snprintf(driver_info->os_version_str1, sizeof(driver_info->os_version_str1),
8554dfc97bSShailend Chand "FreeBSD %u", __FreeBSD_version);
8654dfc97bSShailend Chand
8754dfc97bSShailend Chand bus_dmamap_sync(driver_info_mem.tag, driver_info_mem.map,
8854dfc97bSShailend Chand BUS_DMASYNC_PREREAD);
8954dfc97bSShailend Chand
9054dfc97bSShailend Chand err = gve_adminq_verify_driver_compatibility(priv,
9154dfc97bSShailend Chand sizeof(struct gve_driver_info), driver_info_mem.bus_addr);
9254dfc97bSShailend Chand
9354dfc97bSShailend Chand /* It's ok if the device doesn't support this */
9454dfc97bSShailend Chand if (err == EOPNOTSUPP)
9554dfc97bSShailend Chand err = 0;
9654dfc97bSShailend Chand
9754dfc97bSShailend Chand gve_dma_free_coherent(&driver_info_mem);
9854dfc97bSShailend Chand
9954dfc97bSShailend Chand return (err);
10054dfc97bSShailend Chand }
10154dfc97bSShailend Chand
10254dfc97bSShailend Chand static int
gve_up(struct gve_priv * priv)10354dfc97bSShailend Chand gve_up(struct gve_priv *priv)
10454dfc97bSShailend Chand {
10554dfc97bSShailend Chand if_t ifp = priv->ifp;
10654dfc97bSShailend Chand int err;
10754dfc97bSShailend Chand
10854dfc97bSShailend Chand GVE_IFACE_LOCK_ASSERT(priv->gve_iface_lock);
10954dfc97bSShailend Chand
11054dfc97bSShailend Chand if (device_is_attached(priv->dev) == 0) {
11154dfc97bSShailend Chand device_printf(priv->dev, "Cannot bring the iface up when detached\n");
11254dfc97bSShailend Chand return (ENXIO);
11354dfc97bSShailend Chand }
11454dfc97bSShailend Chand
11554dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_QUEUES_UP))
11654dfc97bSShailend Chand return (0);
11754dfc97bSShailend Chand
11854dfc97bSShailend Chand if_clearhwassist(ifp);
11954dfc97bSShailend Chand if (if_getcapenable(ifp) & IFCAP_TXCSUM)
12054dfc97bSShailend Chand if_sethwassistbits(ifp, CSUM_TCP | CSUM_UDP, 0);
12154dfc97bSShailend Chand if (if_getcapenable(ifp) & IFCAP_TXCSUM_IPV6)
12254dfc97bSShailend Chand if_sethwassistbits(ifp, CSUM_IP6_TCP | CSUM_IP6_UDP, 0);
12354dfc97bSShailend Chand if (if_getcapenable(ifp) & IFCAP_TSO4)
12454dfc97bSShailend Chand if_sethwassistbits(ifp, CSUM_IP_TSO, 0);
12554dfc97bSShailend Chand if (if_getcapenable(ifp) & IFCAP_TSO6)
12654dfc97bSShailend Chand if_sethwassistbits(ifp, CSUM_IP6_TSO, 0);
12754dfc97bSShailend Chand
1282348ac89SShailend Chand if (gve_is_qpl(priv)) {
12954dfc97bSShailend Chand err = gve_register_qpls(priv);
13054dfc97bSShailend Chand if (err != 0)
13154dfc97bSShailend Chand goto reset;
132d438b4efSShailend Chand }
13354dfc97bSShailend Chand
13454dfc97bSShailend Chand err = gve_create_rx_rings(priv);
13554dfc97bSShailend Chand if (err != 0)
13654dfc97bSShailend Chand goto reset;
13754dfc97bSShailend Chand
13854dfc97bSShailend Chand err = gve_create_tx_rings(priv);
13954dfc97bSShailend Chand if (err != 0)
14054dfc97bSShailend Chand goto reset;
14154dfc97bSShailend Chand
14254dfc97bSShailend Chand if_setdrvflagbits(ifp, IFF_DRV_RUNNING, IFF_DRV_OACTIVE);
14354dfc97bSShailend Chand
14454dfc97bSShailend Chand if (!gve_get_state_flag(priv, GVE_STATE_FLAG_LINK_UP)) {
14554dfc97bSShailend Chand if_link_state_change(ifp, LINK_STATE_UP);
14654dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_LINK_UP);
14754dfc97bSShailend Chand }
14854dfc97bSShailend Chand
14954dfc97bSShailend Chand gve_unmask_all_queue_irqs(priv);
15054dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_QUEUES_UP);
15154dfc97bSShailend Chand priv->interface_up_cnt++;
15254dfc97bSShailend Chand return (0);
15354dfc97bSShailend Chand
15454dfc97bSShailend Chand reset:
15554dfc97bSShailend Chand gve_schedule_reset(priv);
15654dfc97bSShailend Chand return (err);
15754dfc97bSShailend Chand }
15854dfc97bSShailend Chand
15954dfc97bSShailend Chand static void
gve_down(struct gve_priv * priv)16054dfc97bSShailend Chand gve_down(struct gve_priv *priv)
16154dfc97bSShailend Chand {
16254dfc97bSShailend Chand GVE_IFACE_LOCK_ASSERT(priv->gve_iface_lock);
16354dfc97bSShailend Chand
16454dfc97bSShailend Chand if (!gve_get_state_flag(priv, GVE_STATE_FLAG_QUEUES_UP))
16554dfc97bSShailend Chand return;
16654dfc97bSShailend Chand
16754dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_LINK_UP)) {
16854dfc97bSShailend Chand if_link_state_change(priv->ifp, LINK_STATE_DOWN);
16954dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_LINK_UP);
17054dfc97bSShailend Chand }
17154dfc97bSShailend Chand
17254dfc97bSShailend Chand if_setdrvflagbits(priv->ifp, IFF_DRV_OACTIVE, IFF_DRV_RUNNING);
17354dfc97bSShailend Chand
17454dfc97bSShailend Chand if (gve_destroy_rx_rings(priv) != 0)
17554dfc97bSShailend Chand goto reset;
17654dfc97bSShailend Chand
17754dfc97bSShailend Chand if (gve_destroy_tx_rings(priv) != 0)
17854dfc97bSShailend Chand goto reset;
17954dfc97bSShailend Chand
1802348ac89SShailend Chand if (gve_is_qpl(priv)) {
18154dfc97bSShailend Chand if (gve_unregister_qpls(priv) != 0)
18254dfc97bSShailend Chand goto reset;
183d438b4efSShailend Chand }
18454dfc97bSShailend Chand
185d438b4efSShailend Chand if (gve_is_gqi(priv))
18654dfc97bSShailend Chand gve_mask_all_queue_irqs(priv);
18754dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_QUEUES_UP);
18854dfc97bSShailend Chand priv->interface_down_cnt++;
18954dfc97bSShailend Chand return;
19054dfc97bSShailend Chand
19154dfc97bSShailend Chand reset:
19254dfc97bSShailend Chand gve_schedule_reset(priv);
19354dfc97bSShailend Chand }
19454dfc97bSShailend Chand
195e0464f74SVee Agarwal int
gve_adjust_rx_queues(struct gve_priv * priv,uint16_t new_queue_cnt)196e0464f74SVee Agarwal gve_adjust_rx_queues(struct gve_priv *priv, uint16_t new_queue_cnt)
197e0464f74SVee Agarwal {
198e0464f74SVee Agarwal int err;
199e0464f74SVee Agarwal
200e0464f74SVee Agarwal GVE_IFACE_LOCK_ASSERT(priv->gve_iface_lock);
201e0464f74SVee Agarwal
202e0464f74SVee Agarwal gve_down(priv);
203e0464f74SVee Agarwal
204e0464f74SVee Agarwal if (new_queue_cnt < priv->rx_cfg.num_queues) {
205e0464f74SVee Agarwal /*
206e0464f74SVee Agarwal * Freeing a ring still preserves its ntfy_id,
207e0464f74SVee Agarwal * which is needed if we create the ring again.
208e0464f74SVee Agarwal */
209e0464f74SVee Agarwal gve_free_rx_rings(priv, new_queue_cnt, priv->rx_cfg.num_queues);
210e0464f74SVee Agarwal } else {
211e0464f74SVee Agarwal err = gve_alloc_rx_rings(priv, priv->rx_cfg.num_queues, new_queue_cnt);
212e0464f74SVee Agarwal if (err != 0) {
213e0464f74SVee Agarwal device_printf(priv->dev, "Failed to allocate new queues");
214e0464f74SVee Agarwal /* Failed to allocate rings, start back up with old ones */
215e0464f74SVee Agarwal gve_up(priv);
216e0464f74SVee Agarwal return (err);
217e0464f74SVee Agarwal
218e0464f74SVee Agarwal }
219e0464f74SVee Agarwal }
220e0464f74SVee Agarwal priv->rx_cfg.num_queues = new_queue_cnt;
221e0464f74SVee Agarwal
222e0464f74SVee Agarwal err = gve_up(priv);
223e0464f74SVee Agarwal if (err != 0)
224e0464f74SVee Agarwal gve_schedule_reset(priv);
225e0464f74SVee Agarwal
226e0464f74SVee Agarwal return (err);
227e0464f74SVee Agarwal }
228e0464f74SVee Agarwal
229e0464f74SVee Agarwal int
gve_adjust_tx_queues(struct gve_priv * priv,uint16_t new_queue_cnt)230e0464f74SVee Agarwal gve_adjust_tx_queues(struct gve_priv *priv, uint16_t new_queue_cnt)
231e0464f74SVee Agarwal {
232e0464f74SVee Agarwal int err;
233e0464f74SVee Agarwal
234e0464f74SVee Agarwal GVE_IFACE_LOCK_ASSERT(priv->gve_iface_lock);
235e0464f74SVee Agarwal
236e0464f74SVee Agarwal gve_down(priv);
237e0464f74SVee Agarwal
238e0464f74SVee Agarwal if (new_queue_cnt < priv->tx_cfg.num_queues) {
239e0464f74SVee Agarwal /*
240e0464f74SVee Agarwal * Freeing a ring still preserves its ntfy_id,
241e0464f74SVee Agarwal * which is needed if we create the ring again.
242e0464f74SVee Agarwal */
243e0464f74SVee Agarwal gve_free_tx_rings(priv, new_queue_cnt, priv->tx_cfg.num_queues);
244e0464f74SVee Agarwal } else {
245e0464f74SVee Agarwal err = gve_alloc_tx_rings(priv, priv->tx_cfg.num_queues, new_queue_cnt);
246e0464f74SVee Agarwal if (err != 0) {
247e0464f74SVee Agarwal device_printf(priv->dev, "Failed to allocate new queues");
248e0464f74SVee Agarwal /* Failed to allocate rings, start back up with old ones */
249e0464f74SVee Agarwal gve_up(priv);
250e0464f74SVee Agarwal return (err);
251e0464f74SVee Agarwal
252e0464f74SVee Agarwal }
253e0464f74SVee Agarwal }
254e0464f74SVee Agarwal priv->tx_cfg.num_queues = new_queue_cnt;
255e0464f74SVee Agarwal
256e0464f74SVee Agarwal err = gve_up(priv);
257e0464f74SVee Agarwal if (err != 0)
258e0464f74SVee Agarwal gve_schedule_reset(priv);
259e0464f74SVee Agarwal
260e0464f74SVee Agarwal return (err);
261e0464f74SVee Agarwal }
262e0464f74SVee Agarwal
263*22fe926aSVee Agarwal int
gve_adjust_ring_sizes(struct gve_priv * priv,uint16_t new_desc_cnt,bool is_rx)264*22fe926aSVee Agarwal gve_adjust_ring_sizes(struct gve_priv *priv, uint16_t new_desc_cnt, bool is_rx)
265*22fe926aSVee Agarwal {
266*22fe926aSVee Agarwal int err;
267*22fe926aSVee Agarwal uint16_t prev_desc_cnt;
268*22fe926aSVee Agarwal
269*22fe926aSVee Agarwal GVE_IFACE_LOCK_ASSERT(priv->gve_iface_lock);
270*22fe926aSVee Agarwal
271*22fe926aSVee Agarwal gve_down(priv);
272*22fe926aSVee Agarwal
273*22fe926aSVee Agarwal if (is_rx) {
274*22fe926aSVee Agarwal gve_free_rx_rings(priv, 0, priv->rx_cfg.num_queues);
275*22fe926aSVee Agarwal prev_desc_cnt = priv->rx_desc_cnt;
276*22fe926aSVee Agarwal priv->rx_desc_cnt = new_desc_cnt;
277*22fe926aSVee Agarwal err = gve_alloc_rx_rings(priv, 0, priv->rx_cfg.num_queues);
278*22fe926aSVee Agarwal if (err != 0) {
279*22fe926aSVee Agarwal device_printf(priv->dev,
280*22fe926aSVee Agarwal "Failed to allocate rings. Trying to start back up with previous ring size.");
281*22fe926aSVee Agarwal priv->rx_desc_cnt = prev_desc_cnt;
282*22fe926aSVee Agarwal err = gve_alloc_rx_rings(priv, 0, priv->rx_cfg.num_queues);
283*22fe926aSVee Agarwal }
284*22fe926aSVee Agarwal } else {
285*22fe926aSVee Agarwal gve_free_tx_rings(priv, 0, priv->tx_cfg.num_queues);
286*22fe926aSVee Agarwal prev_desc_cnt = priv->tx_desc_cnt;
287*22fe926aSVee Agarwal priv->tx_desc_cnt = new_desc_cnt;
288*22fe926aSVee Agarwal err = gve_alloc_tx_rings(priv, 0, priv->tx_cfg.num_queues);
289*22fe926aSVee Agarwal if (err != 0) {
290*22fe926aSVee Agarwal device_printf(priv->dev,
291*22fe926aSVee Agarwal "Failed to allocate rings. Trying to start back up with previous ring size.");
292*22fe926aSVee Agarwal priv->tx_desc_cnt = prev_desc_cnt;
293*22fe926aSVee Agarwal err = gve_alloc_tx_rings(priv, 0, priv->tx_cfg.num_queues);
294*22fe926aSVee Agarwal }
295*22fe926aSVee Agarwal }
296*22fe926aSVee Agarwal
297*22fe926aSVee Agarwal if (err != 0) {
298*22fe926aSVee Agarwal device_printf(priv->dev, "Failed to allocate rings! Cannot start device back up!");
299*22fe926aSVee Agarwal return (err);
300*22fe926aSVee Agarwal }
301*22fe926aSVee Agarwal
302*22fe926aSVee Agarwal err = gve_up(priv);
303*22fe926aSVee Agarwal if (err != 0) {
304*22fe926aSVee Agarwal gve_schedule_reset(priv);
305*22fe926aSVee Agarwal return (err);
306*22fe926aSVee Agarwal }
307*22fe926aSVee Agarwal
308*22fe926aSVee Agarwal return (0);
309*22fe926aSVee Agarwal }
310*22fe926aSVee Agarwal
31154dfc97bSShailend Chand static int
gve_set_mtu(if_t ifp,uint32_t new_mtu)31254dfc97bSShailend Chand gve_set_mtu(if_t ifp, uint32_t new_mtu)
31354dfc97bSShailend Chand {
31454dfc97bSShailend Chand struct gve_priv *priv = if_getsoftc(ifp);
315909e2d7bSJasper Tran O'Leary const uint32_t max_problem_range = 8227;
316909e2d7bSJasper Tran O'Leary const uint32_t min_problem_range = 7822;
31754dfc97bSShailend Chand int err;
31854dfc97bSShailend Chand
31954dfc97bSShailend Chand if ((new_mtu > priv->max_mtu) || (new_mtu < ETHERMIN)) {
32054dfc97bSShailend Chand device_printf(priv->dev, "Invalid new MTU setting. new mtu: %d max mtu: %d min mtu: %d\n",
32154dfc97bSShailend Chand new_mtu, priv->max_mtu, ETHERMIN);
32254dfc97bSShailend Chand return (EINVAL);
32354dfc97bSShailend Chand }
32454dfc97bSShailend Chand
325909e2d7bSJasper Tran O'Leary /*
326909e2d7bSJasper Tran O'Leary * When hardware LRO is enabled in DQ mode, MTUs within the range
327909e2d7bSJasper Tran O'Leary * [7822, 8227] trigger hardware issues which cause a drastic drop
328909e2d7bSJasper Tran O'Leary * in throughput.
329909e2d7bSJasper Tran O'Leary */
330909e2d7bSJasper Tran O'Leary if (!gve_is_gqi(priv) && !gve_disable_hw_lro &&
331909e2d7bSJasper Tran O'Leary new_mtu >= min_problem_range && new_mtu <= max_problem_range) {
332909e2d7bSJasper Tran O'Leary device_printf(priv->dev,
333909e2d7bSJasper Tran O'Leary "Cannot set to MTU to %d within the range [%d, %d] while hardware LRO is enabled\n",
334909e2d7bSJasper Tran O'Leary new_mtu, min_problem_range, max_problem_range);
335909e2d7bSJasper Tran O'Leary return (EINVAL);
336909e2d7bSJasper Tran O'Leary }
337909e2d7bSJasper Tran O'Leary
33854dfc97bSShailend Chand err = gve_adminq_set_mtu(priv, new_mtu);
33954dfc97bSShailend Chand if (err == 0) {
34054dfc97bSShailend Chand if (bootverbose)
34154dfc97bSShailend Chand device_printf(priv->dev, "MTU set to %d\n", new_mtu);
34254dfc97bSShailend Chand if_setmtu(ifp, new_mtu);
34354dfc97bSShailend Chand } else {
34454dfc97bSShailend Chand device_printf(priv->dev, "Failed to set MTU to %d\n", new_mtu);
34554dfc97bSShailend Chand }
34654dfc97bSShailend Chand
34754dfc97bSShailend Chand return (err);
34854dfc97bSShailend Chand }
34954dfc97bSShailend Chand
35054dfc97bSShailend Chand static void
gve_init(void * arg)35154dfc97bSShailend Chand gve_init(void *arg)
35254dfc97bSShailend Chand {
35354dfc97bSShailend Chand struct gve_priv *priv = (struct gve_priv *)arg;
35454dfc97bSShailend Chand
35554dfc97bSShailend Chand if (!gve_get_state_flag(priv, GVE_STATE_FLAG_QUEUES_UP)) {
35654dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
35754dfc97bSShailend Chand gve_up(priv);
35854dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
35954dfc97bSShailend Chand }
36054dfc97bSShailend Chand }
36154dfc97bSShailend Chand
36254dfc97bSShailend Chand static int
gve_ioctl(if_t ifp,u_long command,caddr_t data)36354dfc97bSShailend Chand gve_ioctl(if_t ifp, u_long command, caddr_t data)
36454dfc97bSShailend Chand {
36554dfc97bSShailend Chand struct gve_priv *priv;
36654dfc97bSShailend Chand struct ifreq *ifr;
36754dfc97bSShailend Chand int rc = 0;
36854dfc97bSShailend Chand
36954dfc97bSShailend Chand priv = if_getsoftc(ifp);
37054dfc97bSShailend Chand ifr = (struct ifreq *)data;
37154dfc97bSShailend Chand
37254dfc97bSShailend Chand switch (command) {
37354dfc97bSShailend Chand case SIOCSIFMTU:
37454dfc97bSShailend Chand if (if_getmtu(ifp) == ifr->ifr_mtu)
37554dfc97bSShailend Chand break;
37654dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
37754dfc97bSShailend Chand gve_down(priv);
37854dfc97bSShailend Chand gve_set_mtu(ifp, ifr->ifr_mtu);
37954dfc97bSShailend Chand rc = gve_up(priv);
38054dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
38154dfc97bSShailend Chand break;
38254dfc97bSShailend Chand
38354dfc97bSShailend Chand case SIOCSIFFLAGS:
38454dfc97bSShailend Chand if ((if_getflags(ifp) & IFF_UP) != 0) {
38554dfc97bSShailend Chand if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) == 0) {
38654dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
38754dfc97bSShailend Chand rc = gve_up(priv);
38854dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
38954dfc97bSShailend Chand }
39054dfc97bSShailend Chand } else {
39154dfc97bSShailend Chand if ((if_getdrvflags(ifp) & IFF_DRV_RUNNING) != 0) {
39254dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
39354dfc97bSShailend Chand gve_down(priv);
39454dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
39554dfc97bSShailend Chand }
39654dfc97bSShailend Chand }
39754dfc97bSShailend Chand break;
39854dfc97bSShailend Chand
39954dfc97bSShailend Chand case SIOCSIFCAP:
40054dfc97bSShailend Chand if (ifr->ifr_reqcap == if_getcapenable(ifp))
40154dfc97bSShailend Chand break;
40254dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
40354dfc97bSShailend Chand gve_down(priv);
40454dfc97bSShailend Chand if_setcapenable(ifp, ifr->ifr_reqcap);
40554dfc97bSShailend Chand rc = gve_up(priv);
40654dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
40754dfc97bSShailend Chand break;
40854dfc97bSShailend Chand
40954dfc97bSShailend Chand case SIOCSIFMEDIA:
41054dfc97bSShailend Chand /* FALLTHROUGH */
41154dfc97bSShailend Chand case SIOCGIFMEDIA:
41254dfc97bSShailend Chand rc = ifmedia_ioctl(ifp, ifr, &priv->media, command);
41354dfc97bSShailend Chand break;
41454dfc97bSShailend Chand
41554dfc97bSShailend Chand default:
41654dfc97bSShailend Chand rc = ether_ioctl(ifp, command, data);
41754dfc97bSShailend Chand break;
41854dfc97bSShailend Chand }
41954dfc97bSShailend Chand
42054dfc97bSShailend Chand return (rc);
42154dfc97bSShailend Chand }
42254dfc97bSShailend Chand
42354dfc97bSShailend Chand static int
gve_media_change(if_t ifp)42454dfc97bSShailend Chand gve_media_change(if_t ifp)
42554dfc97bSShailend Chand {
42654dfc97bSShailend Chand struct gve_priv *priv = if_getsoftc(ifp);
42754dfc97bSShailend Chand
42854dfc97bSShailend Chand device_printf(priv->dev, "Media change not supported\n");
42954dfc97bSShailend Chand return (0);
43054dfc97bSShailend Chand }
43154dfc97bSShailend Chand
43254dfc97bSShailend Chand static void
gve_media_status(if_t ifp,struct ifmediareq * ifmr)43354dfc97bSShailend Chand gve_media_status(if_t ifp, struct ifmediareq *ifmr)
43454dfc97bSShailend Chand {
43554dfc97bSShailend Chand struct gve_priv *priv = if_getsoftc(ifp);
43654dfc97bSShailend Chand
43754dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
43854dfc97bSShailend Chand
43954dfc97bSShailend Chand ifmr->ifm_status = IFM_AVALID;
44054dfc97bSShailend Chand ifmr->ifm_active = IFM_ETHER;
44154dfc97bSShailend Chand
44254dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_LINK_UP)) {
44354dfc97bSShailend Chand ifmr->ifm_status |= IFM_ACTIVE;
44454dfc97bSShailend Chand ifmr->ifm_active |= IFM_AUTO;
44554dfc97bSShailend Chand } else {
44654dfc97bSShailend Chand ifmr->ifm_active |= IFM_NONE;
44754dfc97bSShailend Chand }
44854dfc97bSShailend Chand
44954dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
45054dfc97bSShailend Chand }
45154dfc97bSShailend Chand
45254dfc97bSShailend Chand static uint64_t
gve_get_counter(if_t ifp,ift_counter cnt)45354dfc97bSShailend Chand gve_get_counter(if_t ifp, ift_counter cnt)
45454dfc97bSShailend Chand {
45554dfc97bSShailend Chand struct gve_priv *priv;
45654dfc97bSShailend Chand uint64_t rpackets = 0;
45754dfc97bSShailend Chand uint64_t tpackets = 0;
45854dfc97bSShailend Chand uint64_t rbytes = 0;
45954dfc97bSShailend Chand uint64_t tbytes = 0;
46054dfc97bSShailend Chand uint64_t rx_dropped_pkt = 0;
46154dfc97bSShailend Chand uint64_t tx_dropped_pkt = 0;
46254dfc97bSShailend Chand
46354dfc97bSShailend Chand priv = if_getsoftc(ifp);
46454dfc97bSShailend Chand
46554dfc97bSShailend Chand gve_accum_stats(priv, &rpackets, &rbytes, &rx_dropped_pkt, &tpackets,
46654dfc97bSShailend Chand &tbytes, &tx_dropped_pkt);
46754dfc97bSShailend Chand
46854dfc97bSShailend Chand switch (cnt) {
46954dfc97bSShailend Chand case IFCOUNTER_IPACKETS:
47054dfc97bSShailend Chand return (rpackets);
47154dfc97bSShailend Chand
47254dfc97bSShailend Chand case IFCOUNTER_OPACKETS:
47354dfc97bSShailend Chand return (tpackets);
47454dfc97bSShailend Chand
47554dfc97bSShailend Chand case IFCOUNTER_IBYTES:
47654dfc97bSShailend Chand return (rbytes);
47754dfc97bSShailend Chand
47854dfc97bSShailend Chand case IFCOUNTER_OBYTES:
47954dfc97bSShailend Chand return (tbytes);
48054dfc97bSShailend Chand
48154dfc97bSShailend Chand case IFCOUNTER_IQDROPS:
48254dfc97bSShailend Chand return (rx_dropped_pkt);
48354dfc97bSShailend Chand
48454dfc97bSShailend Chand case IFCOUNTER_OQDROPS:
48554dfc97bSShailend Chand return (tx_dropped_pkt);
48654dfc97bSShailend Chand
48754dfc97bSShailend Chand default:
48854dfc97bSShailend Chand return (if_get_counter_default(ifp, cnt));
48954dfc97bSShailend Chand }
49054dfc97bSShailend Chand }
49154dfc97bSShailend Chand
492aa386085SZhenlei Huang static void
gve_setup_ifnet(device_t dev,struct gve_priv * priv)49354dfc97bSShailend Chand gve_setup_ifnet(device_t dev, struct gve_priv *priv)
49454dfc97bSShailend Chand {
49554dfc97bSShailend Chand int caps = 0;
49654dfc97bSShailend Chand if_t ifp;
49754dfc97bSShailend Chand
49854dfc97bSShailend Chand ifp = priv->ifp = if_alloc(IFT_ETHER);
49954dfc97bSShailend Chand if_initname(ifp, device_get_name(dev), device_get_unit(dev));
50054dfc97bSShailend Chand if_setsoftc(ifp, priv);
50154dfc97bSShailend Chand if_setdev(ifp, dev);
50254dfc97bSShailend Chand if_setinitfn(ifp, gve_init);
50354dfc97bSShailend Chand if_setioctlfn(ifp, gve_ioctl);
50454dfc97bSShailend Chand if_settransmitfn(ifp, gve_xmit_ifp);
50554dfc97bSShailend Chand if_setqflushfn(ifp, gve_qflush);
50654dfc97bSShailend Chand
507d438b4efSShailend Chand /*
508d438b4efSShailend Chand * Set TSO limits, must match the arguments to bus_dma_tag_create
5092348ac89SShailend Chand * when creating tx->dqo.buf_dmatag. Only applies to the RDA mode
510031800c7SJasper Tran O'Leary * because in QPL we copy the entire packet into the bounce buffer
5112348ac89SShailend Chand * and thus it does not matter how fragmented the mbuf is.
512d438b4efSShailend Chand */
5132348ac89SShailend Chand if (!gve_is_gqi(priv) && !gve_is_qpl(priv)) {
514d438b4efSShailend Chand if_sethwtsomaxsegcount(ifp, GVE_TX_MAX_DATA_DESCS_DQO);
515d438b4efSShailend Chand if_sethwtsomaxsegsize(ifp, GVE_TX_MAX_BUF_SIZE_DQO);
516d438b4efSShailend Chand }
5172348ac89SShailend Chand if_sethwtsomax(ifp, GVE_TSO_MAXSIZE_DQO);
518d438b4efSShailend Chand
51954dfc97bSShailend Chand #if __FreeBSD_version >= 1400086
52054dfc97bSShailend Chand if_setflags(ifp, IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST);
52154dfc97bSShailend Chand #else
52254dfc97bSShailend Chand if_setflags(ifp, IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST | IFF_KNOWSEPOCH);
52354dfc97bSShailend Chand #endif
52454dfc97bSShailend Chand
52554dfc97bSShailend Chand ifmedia_init(&priv->media, IFM_IMASK, gve_media_change, gve_media_status);
52654dfc97bSShailend Chand if_setgetcounterfn(ifp, gve_get_counter);
52754dfc97bSShailend Chand
52854dfc97bSShailend Chand caps = IFCAP_RXCSUM |
52954dfc97bSShailend Chand IFCAP_TXCSUM |
53054dfc97bSShailend Chand IFCAP_TXCSUM_IPV6 |
53154dfc97bSShailend Chand IFCAP_TSO |
53254dfc97bSShailend Chand IFCAP_LRO;
53354dfc97bSShailend Chand
53454dfc97bSShailend Chand if ((priv->supported_features & GVE_SUP_JUMBO_FRAMES_MASK) != 0)
53554dfc97bSShailend Chand caps |= IFCAP_JUMBO_MTU;
53654dfc97bSShailend Chand
53754dfc97bSShailend Chand if_setcapabilities(ifp, caps);
53854dfc97bSShailend Chand if_setcapenable(ifp, caps);
53954dfc97bSShailend Chand
54054dfc97bSShailend Chand if (bootverbose)
54154dfc97bSShailend Chand device_printf(priv->dev, "Setting initial MTU to %d\n", priv->max_mtu);
54254dfc97bSShailend Chand if_setmtu(ifp, priv->max_mtu);
54354dfc97bSShailend Chand
54454dfc97bSShailend Chand ether_ifattach(ifp, priv->mac);
54554dfc97bSShailend Chand
54654dfc97bSShailend Chand ifmedia_add(&priv->media, IFM_ETHER | IFM_AUTO, 0, NULL);
54754dfc97bSShailend Chand ifmedia_set(&priv->media, IFM_ETHER | IFM_AUTO);
54854dfc97bSShailend Chand }
54954dfc97bSShailend Chand
55054dfc97bSShailend Chand static int
gve_alloc_counter_array(struct gve_priv * priv)55154dfc97bSShailend Chand gve_alloc_counter_array(struct gve_priv *priv)
55254dfc97bSShailend Chand {
55354dfc97bSShailend Chand int err;
55454dfc97bSShailend Chand
55554dfc97bSShailend Chand err = gve_dma_alloc_coherent(priv, sizeof(uint32_t) * priv->num_event_counters,
55654dfc97bSShailend Chand PAGE_SIZE, &priv->counter_array_mem);
55754dfc97bSShailend Chand if (err != 0)
55854dfc97bSShailend Chand return (err);
55954dfc97bSShailend Chand
56054dfc97bSShailend Chand priv->counters = priv->counter_array_mem.cpu_addr;
56154dfc97bSShailend Chand return (0);
56254dfc97bSShailend Chand }
56354dfc97bSShailend Chand
56454dfc97bSShailend Chand static void
gve_free_counter_array(struct gve_priv * priv)56554dfc97bSShailend Chand gve_free_counter_array(struct gve_priv *priv)
56654dfc97bSShailend Chand {
56754dfc97bSShailend Chand if (priv->counters != NULL)
56854dfc97bSShailend Chand gve_dma_free_coherent(&priv->counter_array_mem);
56954dfc97bSShailend Chand priv->counter_array_mem = (struct gve_dma_handle){};
57054dfc97bSShailend Chand }
57154dfc97bSShailend Chand
57254dfc97bSShailend Chand static int
gve_alloc_irq_db_array(struct gve_priv * priv)57354dfc97bSShailend Chand gve_alloc_irq_db_array(struct gve_priv *priv)
57454dfc97bSShailend Chand {
57554dfc97bSShailend Chand int err;
57654dfc97bSShailend Chand
57754dfc97bSShailend Chand err = gve_dma_alloc_coherent(priv,
57854dfc97bSShailend Chand sizeof(struct gve_irq_db) * (priv->num_queues), PAGE_SIZE,
57954dfc97bSShailend Chand &priv->irqs_db_mem);
58054dfc97bSShailend Chand if (err != 0)
58154dfc97bSShailend Chand return (err);
58254dfc97bSShailend Chand
58354dfc97bSShailend Chand priv->irq_db_indices = priv->irqs_db_mem.cpu_addr;
58454dfc97bSShailend Chand return (0);
58554dfc97bSShailend Chand }
58654dfc97bSShailend Chand
58754dfc97bSShailend Chand static void
gve_free_irq_db_array(struct gve_priv * priv)58854dfc97bSShailend Chand gve_free_irq_db_array(struct gve_priv *priv)
58954dfc97bSShailend Chand {
59054dfc97bSShailend Chand if (priv->irq_db_indices != NULL)
59154dfc97bSShailend Chand gve_dma_free_coherent(&priv->irqs_db_mem);
59254dfc97bSShailend Chand priv->irqs_db_mem = (struct gve_dma_handle){};
59354dfc97bSShailend Chand }
59454dfc97bSShailend Chand
59554dfc97bSShailend Chand static void
gve_free_rings(struct gve_priv * priv)59654dfc97bSShailend Chand gve_free_rings(struct gve_priv *priv)
59754dfc97bSShailend Chand {
59854dfc97bSShailend Chand gve_free_irqs(priv);
599e0464f74SVee Agarwal
600e0464f74SVee Agarwal gve_free_tx_rings(priv, 0, priv->tx_cfg.num_queues);
601e0464f74SVee Agarwal free(priv->tx, M_GVE);
602e0464f74SVee Agarwal priv->tx = NULL;
603e0464f74SVee Agarwal
604e0464f74SVee Agarwal gve_free_rx_rings(priv, 0, priv->rx_cfg.num_queues);
605e0464f74SVee Agarwal free(priv->rx, M_GVE);
606e0464f74SVee Agarwal priv->rx = NULL;
60754dfc97bSShailend Chand }
60854dfc97bSShailend Chand
60954dfc97bSShailend Chand static int
gve_alloc_rings(struct gve_priv * priv)61054dfc97bSShailend Chand gve_alloc_rings(struct gve_priv *priv)
61154dfc97bSShailend Chand {
61254dfc97bSShailend Chand int err;
61354dfc97bSShailend Chand
614e0464f74SVee Agarwal priv->rx = malloc(sizeof(struct gve_rx_ring) * priv->rx_cfg.max_queues,
615e0464f74SVee Agarwal M_GVE, M_WAITOK | M_ZERO);
616e0464f74SVee Agarwal err = gve_alloc_rx_rings(priv, 0, priv->rx_cfg.num_queues);
61754dfc97bSShailend Chand if (err != 0)
61854dfc97bSShailend Chand goto abort;
61954dfc97bSShailend Chand
620e0464f74SVee Agarwal priv->tx = malloc(sizeof(struct gve_tx_ring) * priv->tx_cfg.max_queues,
621e0464f74SVee Agarwal M_GVE, M_WAITOK | M_ZERO);
622e0464f74SVee Agarwal err = gve_alloc_tx_rings(priv, 0, priv->tx_cfg.num_queues);
62354dfc97bSShailend Chand if (err != 0)
62454dfc97bSShailend Chand goto abort;
62554dfc97bSShailend Chand
62654dfc97bSShailend Chand err = gve_alloc_irqs(priv);
62754dfc97bSShailend Chand if (err != 0)
62854dfc97bSShailend Chand goto abort;
62954dfc97bSShailend Chand
63054dfc97bSShailend Chand return (0);
63154dfc97bSShailend Chand
63254dfc97bSShailend Chand abort:
63354dfc97bSShailend Chand gve_free_rings(priv);
63454dfc97bSShailend Chand return (err);
63554dfc97bSShailend Chand }
63654dfc97bSShailend Chand
63754dfc97bSShailend Chand static void
gve_deconfigure_and_free_device_resources(struct gve_priv * priv)63862b2d0c3SJasper Tran O'Leary gve_deconfigure_and_free_device_resources(struct gve_priv *priv)
63954dfc97bSShailend Chand {
64054dfc97bSShailend Chand int err;
64154dfc97bSShailend Chand
64254dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_RESOURCES_OK)) {
64354dfc97bSShailend Chand err = gve_adminq_deconfigure_device_resources(priv);
64454dfc97bSShailend Chand if (err != 0) {
64554dfc97bSShailend Chand device_printf(priv->dev, "Failed to deconfigure device resources: err=%d\n",
64654dfc97bSShailend Chand err);
64754dfc97bSShailend Chand return;
64854dfc97bSShailend Chand }
64954dfc97bSShailend Chand if (bootverbose)
65054dfc97bSShailend Chand device_printf(priv->dev, "Deconfigured device resources\n");
65154dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_RESOURCES_OK);
65254dfc97bSShailend Chand }
65354dfc97bSShailend Chand
65454dfc97bSShailend Chand gve_free_irq_db_array(priv);
65554dfc97bSShailend Chand gve_free_counter_array(priv);
656d438b4efSShailend Chand
657d438b4efSShailend Chand if (priv->ptype_lut_dqo) {
658d438b4efSShailend Chand free(priv->ptype_lut_dqo, M_GVE);
659d438b4efSShailend Chand priv->ptype_lut_dqo = NULL;
660d438b4efSShailend Chand }
66154dfc97bSShailend Chand }
66254dfc97bSShailend Chand
66354dfc97bSShailend Chand static int
gve_alloc_and_configure_device_resources(struct gve_priv * priv)66462b2d0c3SJasper Tran O'Leary gve_alloc_and_configure_device_resources(struct gve_priv *priv)
66554dfc97bSShailend Chand {
66654dfc97bSShailend Chand int err;
66754dfc97bSShailend Chand
66854dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_RESOURCES_OK))
66954dfc97bSShailend Chand return (0);
67054dfc97bSShailend Chand
67154dfc97bSShailend Chand err = gve_alloc_counter_array(priv);
67254dfc97bSShailend Chand if (err != 0)
67354dfc97bSShailend Chand return (err);
67454dfc97bSShailend Chand
67554dfc97bSShailend Chand err = gve_alloc_irq_db_array(priv);
67654dfc97bSShailend Chand if (err != 0)
67754dfc97bSShailend Chand goto abort;
67854dfc97bSShailend Chand
67954dfc97bSShailend Chand err = gve_adminq_configure_device_resources(priv);
68054dfc97bSShailend Chand if (err != 0) {
68154dfc97bSShailend Chand device_printf(priv->dev, "Failed to configure device resources: err=%d\n",
68254dfc97bSShailend Chand err);
68354dfc97bSShailend Chand err = (ENXIO);
68454dfc97bSShailend Chand goto abort;
68554dfc97bSShailend Chand }
68654dfc97bSShailend Chand
687d438b4efSShailend Chand if (!gve_is_gqi(priv)) {
688d438b4efSShailend Chand priv->ptype_lut_dqo = malloc(sizeof(*priv->ptype_lut_dqo), M_GVE,
689d438b4efSShailend Chand M_WAITOK | M_ZERO);
690d438b4efSShailend Chand
691d438b4efSShailend Chand err = gve_adminq_get_ptype_map_dqo(priv, priv->ptype_lut_dqo);
692d438b4efSShailend Chand if (err != 0) {
693d438b4efSShailend Chand device_printf(priv->dev, "Failed to configure ptype lut: err=%d\n",
694d438b4efSShailend Chand err);
695d438b4efSShailend Chand goto abort;
696d438b4efSShailend Chand }
697d438b4efSShailend Chand }
698d438b4efSShailend Chand
69954dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_RESOURCES_OK);
70054dfc97bSShailend Chand if (bootverbose)
70154dfc97bSShailend Chand device_printf(priv->dev, "Configured device resources\n");
70254dfc97bSShailend Chand return (0);
70354dfc97bSShailend Chand
70454dfc97bSShailend Chand abort:
70562b2d0c3SJasper Tran O'Leary gve_deconfigure_and_free_device_resources(priv);
70654dfc97bSShailend Chand return (err);
70754dfc97bSShailend Chand }
70854dfc97bSShailend Chand
70954dfc97bSShailend Chand static void
gve_set_queue_cnts(struct gve_priv * priv)71054dfc97bSShailend Chand gve_set_queue_cnts(struct gve_priv *priv)
71154dfc97bSShailend Chand {
71254dfc97bSShailend Chand priv->tx_cfg.max_queues = gve_reg_bar_read_4(priv, MAX_TX_QUEUES);
71354dfc97bSShailend Chand priv->rx_cfg.max_queues = gve_reg_bar_read_4(priv, MAX_RX_QUEUES);
71454dfc97bSShailend Chand priv->tx_cfg.num_queues = priv->tx_cfg.max_queues;
71554dfc97bSShailend Chand priv->rx_cfg.num_queues = priv->rx_cfg.max_queues;
71654dfc97bSShailend Chand
71754dfc97bSShailend Chand if (priv->default_num_queues > 0) {
71854dfc97bSShailend Chand priv->tx_cfg.num_queues = MIN(priv->default_num_queues,
71954dfc97bSShailend Chand priv->tx_cfg.num_queues);
72054dfc97bSShailend Chand priv->rx_cfg.num_queues = MIN(priv->default_num_queues,
72154dfc97bSShailend Chand priv->rx_cfg.num_queues);
72254dfc97bSShailend Chand }
72354dfc97bSShailend Chand
724e0464f74SVee Agarwal priv->num_queues = priv->tx_cfg.max_queues + priv->rx_cfg.max_queues;
72554dfc97bSShailend Chand priv->mgmt_msix_idx = priv->num_queues;
72654dfc97bSShailend Chand }
72754dfc97bSShailend Chand
72854dfc97bSShailend Chand static int
gve_alloc_adminq_and_describe_device(struct gve_priv * priv)72954dfc97bSShailend Chand gve_alloc_adminq_and_describe_device(struct gve_priv *priv)
73054dfc97bSShailend Chand {
73154dfc97bSShailend Chand int err;
73254dfc97bSShailend Chand
73354dfc97bSShailend Chand if ((err = gve_adminq_alloc(priv)) != 0)
73454dfc97bSShailend Chand return (err);
73554dfc97bSShailend Chand
73654dfc97bSShailend Chand if ((err = gve_verify_driver_compatibility(priv)) != 0) {
73754dfc97bSShailend Chand device_printf(priv->dev,
73854dfc97bSShailend Chand "Failed to verify driver compatibility: err=%d\n", err);
73954dfc97bSShailend Chand goto abort;
74054dfc97bSShailend Chand }
74154dfc97bSShailend Chand
74254dfc97bSShailend Chand if ((err = gve_adminq_describe_device(priv)) != 0)
74354dfc97bSShailend Chand goto abort;
74454dfc97bSShailend Chand
74554dfc97bSShailend Chand gve_set_queue_cnts(priv);
74654dfc97bSShailend Chand
74754dfc97bSShailend Chand priv->num_registered_pages = 0;
74854dfc97bSShailend Chand return (0);
74954dfc97bSShailend Chand
75054dfc97bSShailend Chand abort:
75154dfc97bSShailend Chand gve_release_adminq(priv);
75254dfc97bSShailend Chand return (err);
75354dfc97bSShailend Chand }
75454dfc97bSShailend Chand
75554dfc97bSShailend Chand void
gve_schedule_reset(struct gve_priv * priv)75654dfc97bSShailend Chand gve_schedule_reset(struct gve_priv *priv)
75754dfc97bSShailend Chand {
75854dfc97bSShailend Chand if (gve_get_state_flag(priv, GVE_STATE_FLAG_IN_RESET))
75954dfc97bSShailend Chand return;
76054dfc97bSShailend Chand
76154dfc97bSShailend Chand device_printf(priv->dev, "Scheduling reset task!\n");
76254dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_DO_RESET);
76354dfc97bSShailend Chand taskqueue_enqueue(priv->service_tq, &priv->service_task);
76454dfc97bSShailend Chand }
76554dfc97bSShailend Chand
76654dfc97bSShailend Chand static void
gve_destroy(struct gve_priv * priv)76754dfc97bSShailend Chand gve_destroy(struct gve_priv *priv)
76854dfc97bSShailend Chand {
76954dfc97bSShailend Chand gve_down(priv);
77062b2d0c3SJasper Tran O'Leary gve_deconfigure_and_free_device_resources(priv);
77154dfc97bSShailend Chand gve_release_adminq(priv);
77254dfc97bSShailend Chand }
77354dfc97bSShailend Chand
77454dfc97bSShailend Chand static void
gve_restore(struct gve_priv * priv)77554dfc97bSShailend Chand gve_restore(struct gve_priv *priv)
77654dfc97bSShailend Chand {
77754dfc97bSShailend Chand int err;
77854dfc97bSShailend Chand
77954dfc97bSShailend Chand err = gve_adminq_alloc(priv);
78054dfc97bSShailend Chand if (err != 0)
78154dfc97bSShailend Chand goto abort;
78254dfc97bSShailend Chand
78362b2d0c3SJasper Tran O'Leary err = gve_adminq_configure_device_resources(priv);
78462b2d0c3SJasper Tran O'Leary if (err != 0) {
78562b2d0c3SJasper Tran O'Leary device_printf(priv->dev, "Failed to configure device resources: err=%d\n",
78662b2d0c3SJasper Tran O'Leary err);
78762b2d0c3SJasper Tran O'Leary err = (ENXIO);
78854dfc97bSShailend Chand goto abort;
78962b2d0c3SJasper Tran O'Leary }
79062b2d0c3SJasper Tran O'Leary if (!gve_is_gqi(priv)) {
79162b2d0c3SJasper Tran O'Leary err = gve_adminq_get_ptype_map_dqo(priv, priv->ptype_lut_dqo);
79262b2d0c3SJasper Tran O'Leary if (err != 0) {
79362b2d0c3SJasper Tran O'Leary device_printf(priv->dev, "Failed to configure ptype lut: err=%d\n",
79462b2d0c3SJasper Tran O'Leary err);
79562b2d0c3SJasper Tran O'Leary goto abort;
79662b2d0c3SJasper Tran O'Leary }
79762b2d0c3SJasper Tran O'Leary }
79854dfc97bSShailend Chand
79954dfc97bSShailend Chand err = gve_up(priv);
80054dfc97bSShailend Chand if (err != 0)
80154dfc97bSShailend Chand goto abort;
80254dfc97bSShailend Chand
80354dfc97bSShailend Chand return;
80454dfc97bSShailend Chand
80554dfc97bSShailend Chand abort:
80654dfc97bSShailend Chand device_printf(priv->dev, "Restore failed!\n");
80754dfc97bSShailend Chand return;
80854dfc97bSShailend Chand }
80954dfc97bSShailend Chand
81054dfc97bSShailend Chand static void
gve_clear_device_resources(struct gve_priv * priv)81162b2d0c3SJasper Tran O'Leary gve_clear_device_resources(struct gve_priv *priv)
81262b2d0c3SJasper Tran O'Leary {
81362b2d0c3SJasper Tran O'Leary int i;
81462b2d0c3SJasper Tran O'Leary
81562b2d0c3SJasper Tran O'Leary for (i = 0; i < priv->num_event_counters; i++)
81662b2d0c3SJasper Tran O'Leary priv->counters[i] = 0;
81762b2d0c3SJasper Tran O'Leary bus_dmamap_sync(priv->counter_array_mem.tag, priv->counter_array_mem.map,
81862b2d0c3SJasper Tran O'Leary BUS_DMASYNC_PREWRITE);
81962b2d0c3SJasper Tran O'Leary
82062b2d0c3SJasper Tran O'Leary for (i = 0; i < priv->num_queues; i++)
82162b2d0c3SJasper Tran O'Leary priv->irq_db_indices[i] = (struct gve_irq_db){};
82262b2d0c3SJasper Tran O'Leary bus_dmamap_sync(priv->irqs_db_mem.tag, priv->irqs_db_mem.map,
82362b2d0c3SJasper Tran O'Leary BUS_DMASYNC_PREWRITE);
82462b2d0c3SJasper Tran O'Leary
82562b2d0c3SJasper Tran O'Leary if (priv->ptype_lut_dqo)
82662b2d0c3SJasper Tran O'Leary *priv->ptype_lut_dqo = (struct gve_ptype_lut){0};
82762b2d0c3SJasper Tran O'Leary }
82862b2d0c3SJasper Tran O'Leary
82962b2d0c3SJasper Tran O'Leary static void
gve_handle_reset(struct gve_priv * priv)83054dfc97bSShailend Chand gve_handle_reset(struct gve_priv *priv)
83154dfc97bSShailend Chand {
83254dfc97bSShailend Chand if (!gve_get_state_flag(priv, GVE_STATE_FLAG_DO_RESET))
83354dfc97bSShailend Chand return;
83454dfc97bSShailend Chand
83554dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_DO_RESET);
83654dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_IN_RESET);
83754dfc97bSShailend Chand
83854dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
83954dfc97bSShailend Chand
84054dfc97bSShailend Chand if_setdrvflagbits(priv->ifp, IFF_DRV_OACTIVE, IFF_DRV_RUNNING);
84154dfc97bSShailend Chand if_link_state_change(priv->ifp, LINK_STATE_DOWN);
84254dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_LINK_UP);
84354dfc97bSShailend Chand
84454dfc97bSShailend Chand /*
84554dfc97bSShailend Chand * Releasing the adminq causes the NIC to destroy all resources
84654dfc97bSShailend Chand * registered with it, so by clearing the flags beneath we cause
84754dfc97bSShailend Chand * the subsequent gve_down call below to not attempt to tell the
84854dfc97bSShailend Chand * NIC to destroy these resources again.
84954dfc97bSShailend Chand *
85054dfc97bSShailend Chand * The call to gve_down is needed in the first place to refresh
85154dfc97bSShailend Chand * the state and the DMA-able memory within each driver ring.
85254dfc97bSShailend Chand */
85354dfc97bSShailend Chand gve_release_adminq(priv);
85454dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_RESOURCES_OK);
85554dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_QPLREG_OK);
85654dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_RX_RINGS_OK);
85754dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_TX_RINGS_OK);
85854dfc97bSShailend Chand
85954dfc97bSShailend Chand gve_down(priv);
86062b2d0c3SJasper Tran O'Leary gve_clear_device_resources(priv);
86162b2d0c3SJasper Tran O'Leary
86254dfc97bSShailend Chand gve_restore(priv);
86354dfc97bSShailend Chand
86454dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
86554dfc97bSShailend Chand
86654dfc97bSShailend Chand priv->reset_cnt++;
86754dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_IN_RESET);
86854dfc97bSShailend Chand }
86954dfc97bSShailend Chand
87054dfc97bSShailend Chand static void
gve_handle_link_status(struct gve_priv * priv)87154dfc97bSShailend Chand gve_handle_link_status(struct gve_priv *priv)
87254dfc97bSShailend Chand {
87354dfc97bSShailend Chand uint32_t status = gve_reg_bar_read_4(priv, DEVICE_STATUS);
87454dfc97bSShailend Chand bool link_up = status & GVE_DEVICE_STATUS_LINK_STATUS;
87554dfc97bSShailend Chand
87654dfc97bSShailend Chand if (link_up == gve_get_state_flag(priv, GVE_STATE_FLAG_LINK_UP))
87754dfc97bSShailend Chand return;
87854dfc97bSShailend Chand
87954dfc97bSShailend Chand if (link_up) {
88054dfc97bSShailend Chand if (bootverbose)
88154dfc97bSShailend Chand device_printf(priv->dev, "Device link is up.\n");
88254dfc97bSShailend Chand if_link_state_change(priv->ifp, LINK_STATE_UP);
88354dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_LINK_UP);
88454dfc97bSShailend Chand } else {
88554dfc97bSShailend Chand device_printf(priv->dev, "Device link is down.\n");
88654dfc97bSShailend Chand if_link_state_change(priv->ifp, LINK_STATE_DOWN);
88754dfc97bSShailend Chand gve_clear_state_flag(priv, GVE_STATE_FLAG_LINK_UP);
88854dfc97bSShailend Chand }
88954dfc97bSShailend Chand }
89054dfc97bSShailend Chand
89154dfc97bSShailend Chand static void
gve_service_task(void * arg,int pending)89254dfc97bSShailend Chand gve_service_task(void *arg, int pending)
89354dfc97bSShailend Chand {
89454dfc97bSShailend Chand struct gve_priv *priv = (struct gve_priv *)arg;
89554dfc97bSShailend Chand uint32_t status = gve_reg_bar_read_4(priv, DEVICE_STATUS);
89654dfc97bSShailend Chand
89754dfc97bSShailend Chand if (((GVE_DEVICE_STATUS_RESET_MASK & status) != 0) &&
89854dfc97bSShailend Chand !gve_get_state_flag(priv, GVE_STATE_FLAG_IN_RESET)) {
89954dfc97bSShailend Chand device_printf(priv->dev, "Device requested reset\n");
90054dfc97bSShailend Chand gve_set_state_flag(priv, GVE_STATE_FLAG_DO_RESET);
90154dfc97bSShailend Chand }
90254dfc97bSShailend Chand
90354dfc97bSShailend Chand gve_handle_reset(priv);
90454dfc97bSShailend Chand gve_handle_link_status(priv);
90554dfc97bSShailend Chand }
90654dfc97bSShailend Chand
90754dfc97bSShailend Chand static int
gve_probe(device_t dev)90854dfc97bSShailend Chand gve_probe(device_t dev)
90954dfc97bSShailend Chand {
9101bbdfb0bSXin LI uint16_t deviceid, vendorid;
9111bbdfb0bSXin LI int i;
9121bbdfb0bSXin LI
9131bbdfb0bSXin LI vendorid = pci_get_vendor(dev);
9141bbdfb0bSXin LI deviceid = pci_get_device(dev);
9151bbdfb0bSXin LI
9161177a6c8SXin LI for (i = 0; i < nitems(gve_devs); i++) {
9171bbdfb0bSXin LI if (vendorid == gve_devs[i].vendor_id &&
9181bbdfb0bSXin LI deviceid == gve_devs[i].device_id) {
9191bbdfb0bSXin LI device_set_desc(dev, gve_devs[i].name);
92054dfc97bSShailend Chand return (BUS_PROBE_DEFAULT);
92154dfc97bSShailend Chand }
9221bbdfb0bSXin LI }
92354dfc97bSShailend Chand return (ENXIO);
92454dfc97bSShailend Chand }
92554dfc97bSShailend Chand
92654dfc97bSShailend Chand static void
gve_free_sys_res_mem(struct gve_priv * priv)92754dfc97bSShailend Chand gve_free_sys_res_mem(struct gve_priv *priv)
92854dfc97bSShailend Chand {
92954dfc97bSShailend Chand if (priv->msix_table != NULL)
93054dfc97bSShailend Chand bus_release_resource(priv->dev, SYS_RES_MEMORY,
93154dfc97bSShailend Chand rman_get_rid(priv->msix_table), priv->msix_table);
93254dfc97bSShailend Chand
93354dfc97bSShailend Chand if (priv->db_bar != NULL)
93454dfc97bSShailend Chand bus_release_resource(priv->dev, SYS_RES_MEMORY,
93554dfc97bSShailend Chand rman_get_rid(priv->db_bar), priv->db_bar);
93654dfc97bSShailend Chand
93754dfc97bSShailend Chand if (priv->reg_bar != NULL)
93854dfc97bSShailend Chand bus_release_resource(priv->dev, SYS_RES_MEMORY,
93954dfc97bSShailend Chand rman_get_rid(priv->reg_bar), priv->reg_bar);
94054dfc97bSShailend Chand }
94154dfc97bSShailend Chand
94254dfc97bSShailend Chand static int
gve_attach(device_t dev)94354dfc97bSShailend Chand gve_attach(device_t dev)
94454dfc97bSShailend Chand {
94554dfc97bSShailend Chand struct gve_priv *priv;
94654dfc97bSShailend Chand int rid;
94754dfc97bSShailend Chand int err;
94854dfc97bSShailend Chand
949d438b4efSShailend Chand snprintf(gve_version, sizeof(gve_version), "%d.%d.%d",
950d438b4efSShailend Chand GVE_VERSION_MAJOR, GVE_VERSION_MINOR, GVE_VERSION_SUB);
951d438b4efSShailend Chand
95254dfc97bSShailend Chand priv = device_get_softc(dev);
95354dfc97bSShailend Chand priv->dev = dev;
95454dfc97bSShailend Chand GVE_IFACE_LOCK_INIT(priv->gve_iface_lock);
95554dfc97bSShailend Chand
95654dfc97bSShailend Chand pci_enable_busmaster(dev);
95754dfc97bSShailend Chand
95854dfc97bSShailend Chand rid = PCIR_BAR(GVE_REGISTER_BAR);
95954dfc97bSShailend Chand priv->reg_bar = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
96054dfc97bSShailend Chand &rid, RF_ACTIVE);
96154dfc97bSShailend Chand if (priv->reg_bar == NULL) {
96254dfc97bSShailend Chand device_printf(dev, "Failed to allocate BAR0\n");
96354dfc97bSShailend Chand err = ENXIO;
96454dfc97bSShailend Chand goto abort;
96554dfc97bSShailend Chand }
96654dfc97bSShailend Chand
96754dfc97bSShailend Chand rid = PCIR_BAR(GVE_DOORBELL_BAR);
96854dfc97bSShailend Chand priv->db_bar = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
96954dfc97bSShailend Chand &rid, RF_ACTIVE);
97054dfc97bSShailend Chand if (priv->db_bar == NULL) {
97154dfc97bSShailend Chand device_printf(dev, "Failed to allocate BAR2\n");
97254dfc97bSShailend Chand err = ENXIO;
97354dfc97bSShailend Chand goto abort;
97454dfc97bSShailend Chand }
97554dfc97bSShailend Chand
97654dfc97bSShailend Chand rid = pci_msix_table_bar(priv->dev);
97754dfc97bSShailend Chand priv->msix_table = bus_alloc_resource_any(dev, SYS_RES_MEMORY,
97854dfc97bSShailend Chand &rid, RF_ACTIVE);
97954dfc97bSShailend Chand if (priv->msix_table == NULL) {
98054dfc97bSShailend Chand device_printf(dev, "Failed to allocate msix table\n");
98154dfc97bSShailend Chand err = ENXIO;
98254dfc97bSShailend Chand goto abort;
98354dfc97bSShailend Chand }
98454dfc97bSShailend Chand
98554dfc97bSShailend Chand err = gve_alloc_adminq_and_describe_device(priv);
98654dfc97bSShailend Chand if (err != 0)
98754dfc97bSShailend Chand goto abort;
98854dfc97bSShailend Chand
98962b2d0c3SJasper Tran O'Leary err = gve_alloc_and_configure_device_resources(priv);
99054dfc97bSShailend Chand if (err != 0)
99154dfc97bSShailend Chand goto abort;
99254dfc97bSShailend Chand
99354dfc97bSShailend Chand err = gve_alloc_rings(priv);
99454dfc97bSShailend Chand if (err != 0)
99554dfc97bSShailend Chand goto abort;
99654dfc97bSShailend Chand
997aa386085SZhenlei Huang gve_setup_ifnet(dev, priv);
99854dfc97bSShailend Chand
99954dfc97bSShailend Chand priv->rx_copybreak = GVE_DEFAULT_RX_COPYBREAK;
100054dfc97bSShailend Chand
100154dfc97bSShailend Chand bus_write_multi_1(priv->reg_bar, DRIVER_VERSION, GVE_DRIVER_VERSION,
100254dfc97bSShailend Chand sizeof(GVE_DRIVER_VERSION) - 1);
100354dfc97bSShailend Chand
100454dfc97bSShailend Chand TASK_INIT(&priv->service_task, 0, gve_service_task, priv);
100554dfc97bSShailend Chand priv->service_tq = taskqueue_create("gve service", M_WAITOK | M_ZERO,
100654dfc97bSShailend Chand taskqueue_thread_enqueue, &priv->service_tq);
100754dfc97bSShailend Chand taskqueue_start_threads(&priv->service_tq, 1, PI_NET, "%s service tq",
100854dfc97bSShailend Chand device_get_nameunit(priv->dev));
100954dfc97bSShailend Chand
101054dfc97bSShailend Chand gve_setup_sysctl(priv);
101154dfc97bSShailend Chand
101254dfc97bSShailend Chand if (bootverbose)
101354dfc97bSShailend Chand device_printf(priv->dev, "Successfully attached %s", GVE_DRIVER_VERSION);
101454dfc97bSShailend Chand return (0);
101554dfc97bSShailend Chand
101654dfc97bSShailend Chand abort:
101754dfc97bSShailend Chand gve_free_rings(priv);
101862b2d0c3SJasper Tran O'Leary gve_deconfigure_and_free_device_resources(priv);
101954dfc97bSShailend Chand gve_release_adminq(priv);
102054dfc97bSShailend Chand gve_free_sys_res_mem(priv);
102154dfc97bSShailend Chand GVE_IFACE_LOCK_DESTROY(priv->gve_iface_lock);
102254dfc97bSShailend Chand return (err);
102354dfc97bSShailend Chand }
102454dfc97bSShailend Chand
102554dfc97bSShailend Chand static int
gve_detach(device_t dev)102654dfc97bSShailend Chand gve_detach(device_t dev)
102754dfc97bSShailend Chand {
102854dfc97bSShailend Chand struct gve_priv *priv = device_get_softc(dev);
102954dfc97bSShailend Chand if_t ifp = priv->ifp;
1030d412c076SJohn Baldwin int error;
1031d412c076SJohn Baldwin
1032d412c076SJohn Baldwin error = bus_generic_detach(dev);
1033d412c076SJohn Baldwin if (error != 0)
1034d412c076SJohn Baldwin return (error);
103554dfc97bSShailend Chand
103654dfc97bSShailend Chand ether_ifdetach(ifp);
103754dfc97bSShailend Chand
103854dfc97bSShailend Chand GVE_IFACE_LOCK_LOCK(priv->gve_iface_lock);
103954dfc97bSShailend Chand gve_destroy(priv);
104054dfc97bSShailend Chand GVE_IFACE_LOCK_UNLOCK(priv->gve_iface_lock);
104154dfc97bSShailend Chand
104254dfc97bSShailend Chand gve_free_rings(priv);
104354dfc97bSShailend Chand gve_free_sys_res_mem(priv);
104454dfc97bSShailend Chand GVE_IFACE_LOCK_DESTROY(priv->gve_iface_lock);
104554dfc97bSShailend Chand
104654dfc97bSShailend Chand while (taskqueue_cancel(priv->service_tq, &priv->service_task, NULL))
104754dfc97bSShailend Chand taskqueue_drain(priv->service_tq, &priv->service_task);
104854dfc97bSShailend Chand taskqueue_free(priv->service_tq);
104954dfc97bSShailend Chand
105054dfc97bSShailend Chand if_free(ifp);
1051d412c076SJohn Baldwin return (0);
105254dfc97bSShailend Chand }
105354dfc97bSShailend Chand
105454dfc97bSShailend Chand static device_method_t gve_methods[] = {
105554dfc97bSShailend Chand DEVMETHOD(device_probe, gve_probe),
105654dfc97bSShailend Chand DEVMETHOD(device_attach, gve_attach),
105754dfc97bSShailend Chand DEVMETHOD(device_detach, gve_detach),
105854dfc97bSShailend Chand DEVMETHOD_END
105954dfc97bSShailend Chand };
106054dfc97bSShailend Chand
106154dfc97bSShailend Chand static driver_t gve_driver = {
106254dfc97bSShailend Chand "gve",
106354dfc97bSShailend Chand gve_methods,
106454dfc97bSShailend Chand sizeof(struct gve_priv)
106554dfc97bSShailend Chand };
106654dfc97bSShailend Chand
106754dfc97bSShailend Chand #if __FreeBSD_version < 1301503
106854dfc97bSShailend Chand static devclass_t gve_devclass;
106954dfc97bSShailend Chand
107054dfc97bSShailend Chand DRIVER_MODULE(gve, pci, gve_driver, gve_devclass, 0, 0);
107154dfc97bSShailend Chand #else
107254dfc97bSShailend Chand DRIVER_MODULE(gve, pci, gve_driver, 0, 0);
107354dfc97bSShailend Chand #endif
10741177a6c8SXin LI MODULE_PNP_INFO("U16:vendor;U16:device;D:#", pci, gve, gve_devs,
10751177a6c8SXin LI nitems(gve_devs));
1076