162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/** 362306a36Sopenharmony_ci * DOC: emac_arc.c - ARC EMAC specific glue layer 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2014 Romain Perier 662306a36Sopenharmony_ci * 762306a36Sopenharmony_ci * Romain Perier <romain.perier@gmail.com> 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <linux/etherdevice.h> 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/of_net.h> 1362306a36Sopenharmony_ci#include <linux/platform_device.h> 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#include "emac.h" 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#define DRV_NAME "emac_arc" 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_cistatic int emac_arc_probe(struct platform_device *pdev) 2062306a36Sopenharmony_ci{ 2162306a36Sopenharmony_ci struct device *dev = &pdev->dev; 2262306a36Sopenharmony_ci struct arc_emac_priv *priv; 2362306a36Sopenharmony_ci phy_interface_t interface; 2462306a36Sopenharmony_ci struct net_device *ndev; 2562306a36Sopenharmony_ci int err; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci if (!dev->of_node) 2862306a36Sopenharmony_ci return -ENODEV; 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci ndev = alloc_etherdev(sizeof(struct arc_emac_priv)); 3162306a36Sopenharmony_ci if (!ndev) 3262306a36Sopenharmony_ci return -ENOMEM; 3362306a36Sopenharmony_ci platform_set_drvdata(pdev, ndev); 3462306a36Sopenharmony_ci SET_NETDEV_DEV(ndev, dev); 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci priv = netdev_priv(ndev); 3762306a36Sopenharmony_ci priv->drv_name = DRV_NAME; 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_ci err = of_get_phy_mode(dev->of_node, &interface); 4062306a36Sopenharmony_ci if (err) { 4162306a36Sopenharmony_ci if (err == -ENODEV) 4262306a36Sopenharmony_ci interface = PHY_INTERFACE_MODE_MII; 4362306a36Sopenharmony_ci else 4462306a36Sopenharmony_ci goto out_netdev; 4562306a36Sopenharmony_ci } 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ci priv->clk = devm_clk_get(dev, "hclk"); 4862306a36Sopenharmony_ci if (IS_ERR(priv->clk)) { 4962306a36Sopenharmony_ci dev_err(dev, "failed to retrieve host clock from device tree\n"); 5062306a36Sopenharmony_ci err = -EINVAL; 5162306a36Sopenharmony_ci goto out_netdev; 5262306a36Sopenharmony_ci } 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_ci err = arc_emac_probe(ndev, interface); 5562306a36Sopenharmony_ciout_netdev: 5662306a36Sopenharmony_ci if (err) 5762306a36Sopenharmony_ci free_netdev(ndev); 5862306a36Sopenharmony_ci return err; 5962306a36Sopenharmony_ci} 6062306a36Sopenharmony_ci 6162306a36Sopenharmony_cistatic int emac_arc_remove(struct platform_device *pdev) 6262306a36Sopenharmony_ci{ 6362306a36Sopenharmony_ci struct net_device *ndev = platform_get_drvdata(pdev); 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci arc_emac_remove(ndev); 6662306a36Sopenharmony_ci free_netdev(ndev); 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci return 0; 6962306a36Sopenharmony_ci} 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_cistatic const struct of_device_id emac_arc_dt_ids[] = { 7262306a36Sopenharmony_ci { .compatible = "snps,arc-emac" }, 7362306a36Sopenharmony_ci { /* Sentinel */ } 7462306a36Sopenharmony_ci}; 7562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, emac_arc_dt_ids); 7662306a36Sopenharmony_ci 7762306a36Sopenharmony_cistatic struct platform_driver emac_arc_driver = { 7862306a36Sopenharmony_ci .probe = emac_arc_probe, 7962306a36Sopenharmony_ci .remove = emac_arc_remove, 8062306a36Sopenharmony_ci .driver = { 8162306a36Sopenharmony_ci .name = DRV_NAME, 8262306a36Sopenharmony_ci .of_match_table = emac_arc_dt_ids, 8362306a36Sopenharmony_ci }, 8462306a36Sopenharmony_ci}; 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cimodule_platform_driver(emac_arc_driver); 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ciMODULE_AUTHOR("Romain Perier <romain.perier@gmail.com>"); 8962306a36Sopenharmony_ciMODULE_DESCRIPTION("ARC EMAC platform driver"); 9062306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 91