/* $NetBSD: if_enet_imx7.c,v 1.4 2019/07/30 06:26:31 hkenken Exp $ */
/*
* Copyright (c) 2014 Ryo Shimizu <ryo@nerv.org>
* 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.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: if_enet_imx7.c,v 1.4 2019/07/30 06:26:31 hkenken Exp $");
#include "locators.h"
#include "imxccm.h"
#include "imxocotp.h"
#include <sys/param.h>
#include <sys/device.h>
#include <sys/bus.h>
#include <arm/imx/imx7var.h>
#include <arm/imx/imx7reg.h>
#include <arm/imx/imx7_ccmreg.h>
#include <arm/imx/imx7_ccmvar.h>
#include <arm/imx/imx7_ocotpreg.h>
#include <arm/imx/imx7_ocotpvar.h>
#include <arm/imx/if_enetreg.h>
#include <arm/imx/if_enetvar.h>
CFATTACH_DECL_NEW(enet, sizeof(struct enet_softc),
enet_match, enet_attach, NULL, NULL);
static void get_mac_from_ocotp(struct enet_softc *, device_t self,
const char *);
int
enet_match(device_t parent __unused, struct cfdata *match __unused, void *aux)
{
struct axi_attach_args *aa;
aa = aux;
switch (aa->aa_addr) {
case (IMX7_AIPS_BASE + AIPS3_ENET1_BASE):
return 1;
case (IMX7_AIPS_BASE + AIPS3_ENET2_BASE):
return 1;
}
return 0;
}
void
enet_attach(device_t parent, device_t self, void *aux)
{
struct enet_softc *sc = device_private(self);
struct axi_attach_args *aa = aux;
if (aa->aa_size == AXICF_SIZE_DEFAULT)
aa->aa_size = AIPS_ENET_SIZE;
sc->sc_dev = self;
sc->sc_iot = aa->aa_iot;
sc->sc_dmat = aa->aa_dmat;
sc->sc_imxtype = 7; /* i.MX7 */
sc->sc_rgmii = 1;
switch (aa->aa_addr) {
case (IMX7_AIPS_BASE + AIPS3_ENET1_BASE):
sc->sc_unit = 0;
get_mac_from_ocotp(sc, self, "enet1-ocotp-mac");
break;
case (IMX7_AIPS_BASE + AIPS3_ENET2_BASE):
sc->sc_unit = 1;
get_mac_from_ocotp(sc, self, "enet2-ocotp-mac");
break;
}
#if NIMXCCM > 0
/* PLL power up */
if (imx7_pll_power(CCM_ANALOG_PLL_ENET, 1) != 0) {
aprint_error_dev(sc->sc_dev,
"couldn't enable CCM_ANALOG_PLL_ENET\n");
return;
}
sc->sc_pllclock = imx7_get_clock(IMX7CLK_ENET_PLL);
#else
sc->sc_pllclock = 1000000000;
#endif
if (bus_space_map(sc->sc_iot, aa->aa_addr, aa->aa_size, 0,
&sc->sc_ioh)) {
aprint_error_dev(self, "cannot map registers\n");
return;
}
aprint_naive("\n");
aprint_normal(": Gigabit Ethernet Controller\n");
/* setup interrupt handlers */
if ((sc->sc_ih = intr_establish(aa->aa_irq, IPL_NET,
IST_LEVEL, enet_intr, sc)) == NULL) {
aprint_error_dev(self, "unable to establish interrupt\n");
goto failure;
}
/* i.MX7 use 3 interrupts */
if ((sc->sc_ih2 = intr_establish(aa->aa_irq + 1, IPL_NET,
IST_LEVEL, enet_intr, sc)) == NULL) {
aprint_error_dev(self,
"unable to establish 2nd interrupt\n");
intr_disestablish(sc->sc_ih);
goto failure;
}
if ((sc->sc_ih3 = intr_establish(aa->aa_irq + 2, IPL_NET,
IST_LEVEL, enet_intr, sc)) == NULL) {
aprint_error_dev(self,
"unable to establish 3rd interrupt\n");
intr_disestablish(sc->sc_ih2);
intr_disestablish(sc->sc_ih);
goto failure;
}
if (enet_attach_common(self) != 0)
goto failure;
return;
failure:
bus_space_unmap(sc->sc_iot, sc->sc_ioh, aa->aa_size);
return;
}
static void
get_mac_from_ocotp(struct enet_softc *sc, device_t self, const char *key)
{
#if NIMXOCOTP > 0
char ocotp_mac[12];
prop_dictionary_t dict;
const char *ocotp_mac_layout;
uint32_t eaddr;
int i;
#define HTOI(c) \
((((c) >= '0') && ((c) <= '9')) ? (c) - '0' : \
(((c) >= 'a') && ((c) <= 'f')) ? (c) - 'a' + 10: -1)
/*
* some board has illegal layout of macaddr in OCOTP.
* so you can configure sequence of macaddr by device property.
*
* OCOTP[MAC_ADDR0]: [0][1][2][3]
* OCOTP[MAC_ADDR1]: [4][5][6][7]
* OCOTP[MAC_ADDR2]: [8][9][a][b]
*
* default layout, ENET1 macaddr is [6,7,0,1,2,3]
* ENET2 macaddr is [8,9,a,b,4,5]
*/
dict = device_properties(self);
if (!prop_dictionary_get_cstring_nocopy(dict, key, &ocotp_mac_layout)) {
if (sc->sc_unit == 0)
ocotp_mac_layout = "670123";
else if (sc->sc_unit == 1)
ocotp_mac_layout = "89ab45";
else
return;
}
eaddr = imxocotp_read(OCOTP_MAC_ADDR0);
ocotp_mac[0] = eaddr >> 24;
ocotp_mac[1] = eaddr >> 16;
ocotp_mac[2] = eaddr >> 8;
ocotp_mac[3] = eaddr;
eaddr = imxocotp_read(OCOTP_MAC_ADDR1);
ocotp_mac[4] = eaddr >> 24;
ocotp_mac[5] = eaddr >> 16;
ocotp_mac[6] = eaddr >> 8;
ocotp_mac[7] = eaddr;
eaddr = imxocotp_read(OCOTP_MAC_ADDR2);
ocotp_mac[8] = eaddr >> 24;
ocotp_mac[9] = eaddr >> 16;
ocotp_mac[10] = eaddr >> 8;
ocotp_mac[11] = eaddr;
for (i = 0; i < ETHER_ADDR_LEN; i++) {
int offs;
offs = HTOI(ocotp_mac_layout[i]);
KASSERT((offs >= 0) && (offs < 12));
sc->sc_enaddr[i] = ocotp_mac[offs];
}
#endif
}