// SPDX-License-Identifier: GPL-2.0 /* * cdns3-starfive.c - StarFive specific Glue layer for Cadence USB Controller * * Copyright (C) 2023 StarFive Technology Co., Ltd. * * Author: Minda Chen */ #include #include #include #include #include #include #include #include #include #include #include #include "core.h" #define JH7100_USB0 0x20 #define JH7100_USB0_MODE_STRAP_MASK GENMASK(2, 0) #define JH7100_USB0_MODE_STRAP_HOST 2 #define JH7100_USB3 0x2c #define JH7100_USB3_UTMI_IDDIG BIT(21) #define JH7100_USB7 0x3c #define JH7100_USB7_SSRX_SEL BIT(18) #define JH7100_USB7_SSTX_SEL BIT(19) #define JH7100_USB7_PLL_EN BIT(23) #define JH7100_USB7_EQ_EN BIT(25) #define JH7110_STRAP_HOST BIT(17) #define JH7110_STRAP_DEVICE BIT(18) #define JH7110_STRAP_MASK GENMASK(18, 16) #define JH7110_SUSPENDM_HOST BIT(19) #define JH7110_SUSPENDM_MASK BIT(19) #define JH7110_MISC_CFG_MASK GENMASK(23, 20) #define JH7110_SUSPENDM_BYPS BIT(20) #define JH7110_PLL_EN BIT(22) #define JH7110_REFCLK_MODE BIT(23) struct cdns_starfive { struct reset_control *resets; struct clk_bulk_data *clks; int num_clks; }; typedef int (cdns_starfive_mode_init_t)(struct device *dev, struct cdns_starfive *data); static int cdns_jh7100_mode_init(struct device *dev, struct cdns_starfive *data) { struct regmap *syscon; enum usb_dr_mode mode; syscon = syscon_regmap_lookup_by_phandle(dev->of_node, "starfive,syscon"); if (IS_ERR(syscon)) return dev_err_probe(dev, PTR_ERR(syscon), "failed to get starfive,syscon\n"); /* dr mode setting */ mode = usb_get_dr_mode(dev); switch (mode) { case USB_DR_MODE_HOST: regmap_update_bits(syscon, JH7100_USB0, JH7100_USB0_MODE_STRAP_MASK, JH7100_USB0_MODE_STRAP_HOST); regmap_update_bits(syscon, JH7100_USB7, JH7100_USB7_PLL_EN, JH7100_USB7_PLL_EN); regmap_update_bits(syscon, JH7100_USB7, JH7100_USB7_EQ_EN, JH7100_USB7_EQ_EN); regmap_update_bits(syscon, JH7100_USB7, JH7100_USB7_SSRX_SEL, JH7100_USB7_SSRX_SEL); regmap_update_bits(syscon, JH7100_USB7, JH7100_USB7_SSTX_SEL, JH7100_USB7_SSTX_SEL); regmap_update_bits(syscon, JH7100_USB3, JH7100_USB3_UTMI_IDDIG, JH7100_USB3_UTMI_IDDIG); break; default: break; } return 0; } static int cdns_jh7110_mode_init(struct device *dev, struct cdns_starfive *data) { struct regmap *syscon; unsigned int usb_mode; enum usb_dr_mode mode; syscon = syscon_regmap_lookup_by_phandle_args(dev->of_node, "starfive,stg-syscon", 1, &usb_mode); if (IS_ERR(syscon)) return dev_err_probe(dev, PTR_ERR(syscon), "failed to parse starfive,stg-syscon\n"); regmap_update_bits(syscon, usb_mode, JH7110_MISC_CFG_MASK, JH7110_SUSPENDM_BYPS | JH7110_PLL_EN | JH7110_REFCLK_MODE); /* dr mode setting */ mode = usb_get_dr_mode(dev); switch (mode) { case USB_DR_MODE_HOST: regmap_update_bits(syscon, usb_mode, JH7110_STRAP_MASK, JH7110_STRAP_HOST); regmap_update_bits(syscon, usb_mode, JH7110_SUSPENDM_MASK, JH7110_SUSPENDM_HOST); break; case USB_DR_MODE_PERIPHERAL: regmap_update_bits(syscon, usb_mode, JH7110_STRAP_MASK, JH7110_STRAP_DEVICE); regmap_update_bits(syscon, usb_mode, JH7110_SUSPENDM_MASK, 0); break; default: break; } return 0; } static int cdns_clk_rst_init(struct device *dev, struct cdns_starfive *data) { int ret; ret = clk_bulk_prepare_enable(data->num_clks, data->clks); if (ret) return dev_err_probe(dev, ret, "failed to enable clocks\n"); ret = reset_control_deassert(data->resets); if (ret) { dev_err(dev, "failed to reset clocks\n"); goto err_clk_init; } return ret; err_clk_init: clk_bulk_disable_unprepare(data->num_clks, data->clks); return ret; } static void cdns_clk_rst_deinit(struct device *dev, struct cdns_starfive *data) { reset_control_assert(data->resets); clk_bulk_disable_unprepare(data->num_clks, data->clks); } static int cdns_starfive_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct cdns_starfive *data; cdns_starfive_mode_init_t *mode_init; int ret; data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL); if (!data) return -ENOMEM; data->num_clks = devm_clk_bulk_get_all(dev, &data->clks); if (data->num_clks < 0) return dev_err_probe(dev, -ENODEV, "failed to get clocks\n"); data->resets = devm_reset_control_array_get_exclusive(dev); if (IS_ERR(data->resets)) return dev_err_probe(dev, PTR_ERR(data->resets), "failed to get resets\n"); mode_init = device_get_match_data(dev); ret = mode_init(dev, data); if (ret) return ret; ret = cdns_clk_rst_init(dev, data); if (ret) return ret; ret = of_platform_populate(dev->of_node, NULL, NULL, dev); if (ret) { cdns_clk_rst_deinit(dev, data); return dev_err_probe(dev, ret, "failed to create children\n"); } device_set_wakeup_capable(dev, true); pm_runtime_set_active(dev); pm_runtime_enable(dev); platform_set_drvdata(pdev, data); return 0; } static int cdns_starfive_remove_core(struct device *dev, void *c) { struct platform_device *pdev = to_platform_device(dev); platform_device_unregister(pdev); return 0; } static void cdns_starfive_remove(struct platform_device *pdev) { struct device *dev = &pdev->dev; struct cdns_starfive *data = dev_get_drvdata(dev); pm_runtime_get_sync(dev); device_for_each_child(dev, NULL, cdns_starfive_remove_core); pm_runtime_disable(dev); pm_runtime_put_noidle(dev); cdns_clk_rst_deinit(dev, data); platform_set_drvdata(pdev, NULL); } #ifdef CONFIG_PM static int cdns_starfive_runtime_resume(struct device *dev) { struct cdns_starfive *data = dev_get_drvdata(dev); return clk_bulk_prepare_enable(data->num_clks, data->clks); } static int cdns_starfive_runtime_suspend(struct device *dev) { struct cdns_starfive *data = dev_get_drvdata(dev); clk_bulk_disable_unprepare(data->num_clks, data->clks); return 0; } #ifdef CONFIG_PM_SLEEP static int cdns_starfive_resume(struct device *dev) { struct cdns_starfive *data = dev_get_drvdata(dev); return cdns_clk_rst_init(dev, data); } static int cdns_starfive_suspend(struct device *dev) { struct cdns_starfive *data = dev_get_drvdata(dev); cdns_clk_rst_deinit(dev, data); return 0; } #endif #endif static const struct dev_pm_ops cdns_starfive_pm_ops = { SET_RUNTIME_PM_OPS(cdns_starfive_runtime_suspend, cdns_starfive_runtime_resume, NULL) SET_SYSTEM_SLEEP_PM_OPS(cdns_starfive_suspend, cdns_starfive_resume) }; static const struct of_device_id cdns_starfive_of_match[] = { { .compatible = "starfive,jh7100-usb", .data = cdns_jh7100_mode_init }, { .compatible = "starfive,jh7110-usb", .data = cdns_jh7110_mode_init }, { /* sentinel */ } }; MODULE_DEVICE_TABLE(of, cdns_starfive_of_match); static struct platform_driver cdns_starfive_driver = { .probe = cdns_starfive_probe, .remove = cdns_starfive_remove, .driver = { .name = "cdns3-starfive", .of_match_table = cdns_starfive_of_match, .pm = &cdns_starfive_pm_ops, }, }; module_platform_driver(cdns_starfive_driver); MODULE_ALIAS("platform:cdns3-starfive"); MODULE_LICENSE("GPL v2"); MODULE_DESCRIPTION("Cadence USB3 StarFive Glue Layer");