/* * Copyright 2023 NXP * * SPDX-License-Identifier: Apache-2.0 */ #define DT_DRV_COMPAT nxp_enet_mdio #include #include #include #include #include #include #include #include #include struct nxp_enet_mdio_config { ENET_Type *base; const struct pinctrl_dev_config *pincfg; const struct device *clock_dev; clock_control_subsys_t clock_subsys; uint32_t mdc_freq; uint16_t timeout; bool disable_preamble; }; struct nxp_enet_mdio_data { struct k_mutex mdio_mutex; struct k_sem mdio_sem; bool interrupt_up; }; /* * This function is used for both read and write operations * in order to wait for the completion of an MDIO transaction. * It returns -ETIMEDOUT if timeout occurs as specified in DT, * otherwise returns 0 if EIR MII bit is set indicting completed * operation, otherwise -EIO. */ static int nxp_enet_mdio_wait_xfer(const struct device *dev) { const struct nxp_enet_mdio_config *config = dev->config; struct nxp_enet_mdio_data *data = dev->data; ENET_Type *base = config->base; int ret = 0; /* This function will not make sense from IRQ context */ if (k_is_in_isr()) { return -EWOULDBLOCK; } /* Enable the interrupt */ base->EIMR |= ENET_EIMR_MII_MASK; /* Wait for operation to complete or time out */ if (!data->interrupt_up) { /* In the case where the interrupt has not been enabled yet because * ethernet driver has not initiaized, just do a busy wait */ k_busy_wait(USEC_PER_MSEC * config->timeout); if (base->EIR & ENET_EIR_MII_MASK) { ret = 0; } else { ret = -ETIMEDOUT; } } else if (k_sem_take(&data->mdio_sem, K_MSEC(config->timeout))) { /* Interrupt was enabled but did not occur in time */ ret = -ETIMEDOUT; } else if (base->EIR & ENET_EIR_MII_MASK) { /* Interrupt happened meaning mdio transaction completed */ ret = 0; } else { /* No idea what happened */ ret = -EIO; } return ret; } /* MDIO Read API implementation */ static int nxp_enet_mdio_read(const struct device *dev, uint8_t prtad, uint8_t regad, uint16_t *read_data) { const struct nxp_enet_mdio_config *config = dev->config; struct nxp_enet_mdio_data *data = dev->data; int ret; /* Only one MDIO bus operation attempt at a time */ (void)k_mutex_lock(&data->mdio_mutex, K_FOREVER); /* * Clear the bit (W1C) that indicates MDIO transfer is ready to * prepare to wait for it to be set once this read is done */ config->base->EIR |= ENET_EIR_MII_MASK; /* * Write MDIO frame to MII management register which will * send the read command and data out to the MDIO bus as this frame: * ST = start, 1 means start * OP = operation, 2 means read * PA = PHY/Port address * RA = Register/Device Address * TA = Turnaround, must be 2 to be valid * data = data to be written to the PHY register */ config->base->MMFR = ENET_MMFR_ST(0x1U) | ENET_MMFR_OP(MDIO_OP_C22_READ) | ENET_MMFR_PA(prtad) | ENET_MMFR_RA(regad) | ENET_MMFR_TA(0x2U); ret = nxp_enet_mdio_wait_xfer(dev); if (ret) { (void)k_mutex_unlock(&data->mdio_mutex); return ret; } /* The data is received in the same register that we wrote the command to */ *read_data = (config->base->MMFR & ENET_MMFR_DATA_MASK) >> ENET_MMFR_DATA_SHIFT; /* Clear the same bit as before because the event has been handled */ config->base->EIR |= ENET_EIR_MII_MASK; /* This MDIO interaction is finished */ (void)k_mutex_unlock(&data->mdio_mutex); return ret; } /* MDIO Write API implementation */ static int nxp_enet_mdio_write(const struct device *dev, uint8_t prtad, uint8_t regad, uint16_t write_data) { const struct nxp_enet_mdio_config *config = dev->config; struct nxp_enet_mdio_data *data = dev->data; int ret; /* Only one MDIO bus operation attempt at a time */ (void)k_mutex_lock(&data->mdio_mutex, K_FOREVER); /* * Clear the bit (W1C) that indicates MDIO transfer is ready to * prepare to wait for it to be set once this write is done */ config->base->EIR |= ENET_EIR_MII_MASK; /* * Write MDIO frame to MII management register which will * send the write command and data out to the MDIO bus as this frame: * ST = start, 1 means start * OP = operation, 1 means write * PA = PHY/Port address * RA = Register/Device Address * TA = Turnaround, must be 2 to be valid * data = data to be written to the PHY register */ config->base->MMFR = ENET_MMFR_ST(0x1U) | ENET_MMFR_OP(MDIO_OP_C22_WRITE) | ENET_MMFR_PA(prtad) | ENET_MMFR_RA(regad) | ENET_MMFR_TA(0x2U) | write_data; ret = nxp_enet_mdio_wait_xfer(dev); if (ret) { (void)k_mutex_unlock(&data->mdio_mutex); return ret; } /* Clear the same bit as before because the event has been handled */ config->base->EIR |= ENET_EIR_MII_MASK; /* This MDIO interaction is finished */ (void)k_mutex_unlock(&data->mdio_mutex); return ret; } static const struct mdio_driver_api nxp_enet_mdio_api = { .read = nxp_enet_mdio_read, .write = nxp_enet_mdio_write, }; static void nxp_enet_mdio_isr_cb(const struct device *dev) { const struct nxp_enet_mdio_config *config = dev->config; struct nxp_enet_mdio_data *data = dev->data; /* Signal that operation finished */ k_sem_give(&data->mdio_sem); /* Disable the interrupt */ config->base->EIMR &= ~ENET_EIMR_MII_MASK; } static void nxp_enet_mdio_post_module_reset_init(const struct device *dev) { const struct nxp_enet_mdio_config *config = dev->config; uint32_t enet_module_clock_rate; /* Set up MSCR register */ (void) clock_control_get_rate(config->clock_dev, config->clock_subsys, &enet_module_clock_rate); uint32_t mii_speed = (enet_module_clock_rate + 2 * config->mdc_freq - 1) / (2 * config->mdc_freq) - 1; uint32_t holdtime = (10 + NSEC_PER_SEC / enet_module_clock_rate - 1) / (NSEC_PER_SEC / enet_module_clock_rate) - 1; uint32_t mscr = ENET_MSCR_MII_SPEED(mii_speed) | ENET_MSCR_HOLDTIME(holdtime) | (config->disable_preamble ? ENET_MSCR_DIS_PRE_MASK : 0); config->base->MSCR = mscr; } void nxp_enet_mdio_callback(const struct device *dev, enum nxp_enet_callback_reason event, void *cb_data) { struct nxp_enet_mdio_data *data = dev->data; ARG_UNUSED(cb_data); switch (event) { case NXP_ENET_MODULE_RESET: nxp_enet_mdio_post_module_reset_init(dev); break; case NXP_ENET_INTERRUPT: nxp_enet_mdio_isr_cb(dev); break; case NXP_ENET_INTERRUPT_ENABLED: data->interrupt_up = true; break; default: break; } } static int nxp_enet_mdio_init(const struct device *dev) { const struct nxp_enet_mdio_config *config = dev->config; struct nxp_enet_mdio_data *data = dev->data; int ret = 0; ret = pinctrl_apply_state(config->pincfg, PINCTRL_STATE_DEFAULT); if (ret) { return ret; } ret = k_mutex_init(&data->mdio_mutex); if (ret) { return ret; } ret = k_sem_init(&data->mdio_sem, 0, 1); if (ret) { return ret; } /* All operations done after module reset should be done during device init too */ nxp_enet_mdio_post_module_reset_init(dev); return ret; } #define NXP_ENET_MDIO_INIT(inst) \ PINCTRL_DT_INST_DEFINE(inst); \ \ static const struct nxp_enet_mdio_config nxp_enet_mdio_cfg_##inst = { \ .base = (ENET_Type *) DT_REG_ADDR(DT_INST_PARENT(inst)), \ .pincfg = PINCTRL_DT_INST_DEV_CONFIG_GET(inst), \ .timeout = CONFIG_MDIO_NXP_ENET_TIMEOUT, \ .clock_dev = DEVICE_DT_GET(DT_CLOCKS_CTLR(DT_INST_PARENT(inst))), \ .clock_subsys = (void *) DT_CLOCKS_CELL_BY_IDX( \ DT_INST_PARENT(inst), 0, name), \ .disable_preamble = DT_INST_PROP(inst, suppress_preamble), \ .mdc_freq = DT_INST_PROP(inst, clock_frequency), \ }; \ \ static struct nxp_enet_mdio_data nxp_enet_mdio_data_##inst; \ \ DEVICE_DT_INST_DEFINE(inst, &nxp_enet_mdio_init, NULL, \ &nxp_enet_mdio_data_##inst, &nxp_enet_mdio_cfg_##inst, \ POST_KERNEL, CONFIG_MDIO_INIT_PRIORITY, \ &nxp_enet_mdio_api); DT_INST_FOREACH_STATUS_OKAY(NXP_ENET_MDIO_INIT)