/* $NetBSD: if_eqos_pci.c,v 1.4 2023/12/19 09:39:55 skrll Exp $ */

/*-
* Copyright (c) 2023 Masanobu SAITOH <[email protected]>
* All rights reserved.
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
*/

/*
* TODO:
*      Use multi vector MSI to support multiqueue.
*
*/

#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: if_eqos_pci.c,v 1.4 2023/12/19 09:39:55 skrll Exp $");

#include <sys/param.h>
#include <sys/bus.h>
#include <sys/device.h>
#include <sys/rndsource.h>

#include <net/if_ether.h>
#include <net/if_media.h>

#include <dev/pci/pcireg.h>
#include <dev/pci/pcivar.h>
#include <dev/pci/pcidevs.h>

#include <dev/mii/miivar.h>
#include <dev/ic/dwc_eqos_var.h>

#define EQOS_PCI_MAX_INTR       1

static int eqos_pci_match(device_t, cfdata_t, void *);
static void eqos_pci_attach(device_t, device_t, void *);

struct eqos_pci_softc {
       struct eqos_softc       sc_eqos;
       pci_chipset_tag_t       sc_pc;
       pcitag_t                sc_tag;
       void                    *sc_ihs[EQOS_PCI_MAX_INTR];
       pci_intr_handle_t       *sc_intrs;
       uint16_t                sc_pcidevid;
};

static const struct device_compatible_entry compat_data[] = {
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_ETH) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_RGMII) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_RGMII) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_SGMII_1G) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_SGMII_1G) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_SGMII_2_5G) },
       { .id = PCI_ID_CODE(PCI_VENDOR_INTEL,
                   PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_SGMII_2_5G) },

       PCI_COMPAT_EOL
};

CFATTACH_DECL3_NEW(eqos_pci, sizeof(struct eqos_pci_softc),
   eqos_pci_match, eqos_pci_attach, NULL, NULL, NULL, NULL,
   0);

static int
eqos_pci_match(device_t parent, cfdata_t match, void *aux)
{
       struct pci_attach_args *pa =aux;

       return pci_compatible_match(pa, compat_data);
}

static void
eqos_pci_attach(device_t parent, device_t self, void *aux)
{
       struct eqos_pci_softc * const psc = device_private(self);
       struct eqos_softc * const sc = &psc->sc_eqos;
       struct pci_attach_args *pa =aux;
       const pci_chipset_tag_t pc = pa->pa_pc;
       const pcitag_t tag = pa->pa_tag;
       prop_dictionary_t prop;
       bus_space_tag_t memt;
       bus_space_handle_t memh;
       int counts[PCI_INTR_TYPE_SIZE];
       char intrbuf[PCI_INTRSTR_LEN];
       bus_size_t memsize;
       pcireg_t memtype;
       const char *intrstr;
       uint32_t dma_pbl = 0;

       psc->sc_pc = pc;
       psc->sc_tag = tag;
       psc->sc_pcidevid = PCI_PRODUCT(pa->pa_id);

       memtype = pci_mapreg_type(pa->pa_pc, pa->pa_tag, PCI_BAR0);
       if (pci_mapreg_map(pa, PCI_BAR0, memtype, 0, &memt, &memh, NULL,
               &memsize) != 0) {
               aprint_error(": can't map mem space\n");
               return;
       }
       sc->sc_dev = self;
       sc->sc_bst = memt;
       sc->sc_bsh = memh;
       prop = device_properties(sc->sc_dev);

       if (pci_dma64_available(pa))
               sc->sc_dmat = pa->pa_dmat64;
       else
               sc->sc_dmat = pa->pa_dmat;

       sc->sc_phy_id = MII_PHY_ANY;
       switch (psc->sc_pcidevid) {
       case PCI_PRODUCT_INTEL_EHL_ETH:
               sc->sc_csr_clock = 204800000;
               dma_pbl = 32;
               break;
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_RGMII:
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_RGMII:
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_SGMII_1G:
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_SGMII_1G:
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_0_SGMII_2_5G:
       case PCI_PRODUCT_INTEL_EHL_PSE_ETH_1_SGMII_2_5G:
               sc->sc_dmat = pa->pa_dmat; /* 32bit DMA only */
               sc->sc_csr_clock = 200000000;
               dma_pbl = 32;
               break;
#if 0
       case PCI_PRODUCT_INTEL_QUARTK_ETH:
               dma_pbl = 16;
#endif
       default:
               sc->sc_csr_clock = 200000000; /* XXX */
       }

       if (sc->sc_dmat == pa->pa_dmat64)
               aprint_verbose(", 64-bit DMA");
       else
               aprint_verbose(", 32-bit DMA");

       /* Defaults */
       if (dma_pbl != 0) {
               prop = device_properties(sc->sc_dev);
               prop_dictionary_set_uint32(prop, "snps,pbl", dma_pbl);
       }

       if (eqos_attach(sc) != 0) {
               aprint_error_dev(sc->sc_dev, "failed in eqos_attach()\n");
               return;
       }

       /* Allocation settings */
       counts[PCI_INTR_TYPE_MSI] = 1;
       counts[PCI_INTR_TYPE_INTX] = 1;
       if (pci_intr_alloc(pa, &psc->sc_intrs, counts, PCI_INTR_TYPE_MSI) != 0)
       {
               aprint_error_dev(sc->sc_dev, "failed to allocate interrupt\n");
               return;
       }
       intrstr = pci_intr_string(pc, psc->sc_intrs[0], intrbuf,
           sizeof(intrbuf));
       pci_intr_setattr(pc, &psc->sc_intrs[0], PCI_INTR_MPSAFE, true);
       psc->sc_ihs[0] = pci_intr_establish_xname(pc, psc->sc_intrs[0],
           IPL_NET, eqos_intr, sc, device_xname(self));

       aprint_normal_dev(self, "interrupting on %s\n", intrstr);

       if (pmf_device_register(self, NULL, NULL))
               pmf_class_network_register(self, &sc->sc_ec.ec_if);
       else
               aprint_error_dev(self, "couldn't establish power handler\n");
}