Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Abel Vesa | 1077 | 100.00% | 1 | 100.00% |
Total | 1077 | 1 |
// SPDX-License-Identifier: GPL-2.0 /* * Copyright (c) 2023, Linaro Limited */ #include <linux/module.h> #include <linux/platform_device.h> #include <linux/regulator/consumer.h> #include <linux/regmap.h> #include <linux/of.h> #include <linux/of_device.h> #include <linux/phy/phy.h> /* eUSB2 status registers */ #define EUSB2_RPTR_STATUS 0x08 #define RPTR_OK BIT(7) /* eUSB2 control registers */ #define EUSB2_EN_CTL1 0x46 #define EUSB2_RPTR_EN BIT(7) #define EUSB2_FORCE_EN_5 0xe8 #define F_CLK_19P2M_EN BIT(6) #define EUSB2_FORCE_VAL_5 0xeD #define V_CLK_19P2M_EN BIT(6) #define EUSB2_TUNE_IUSB2 0x51 #define EUSB2_TUNE_SQUELCH_U 0x54 #define EUSB2_TUNE_USB2_PREEM 0x57 #define QCOM_EUSB2_REPEATER_INIT_CFG(o, v) \ { \ .offset = o, \ .val = v, \ } struct eusb2_repeater_init_tbl { unsigned int offset; unsigned int val; }; struct eusb2_repeater_cfg { const struct eusb2_repeater_init_tbl *init_tbl; int init_tbl_num; const char * const *vreg_list; int num_vregs; }; struct eusb2_repeater { struct device *dev; struct regmap *regmap; struct phy *phy; struct regulator_bulk_data *vregs; const struct eusb2_repeater_cfg *cfg; u16 base; enum phy_mode mode; }; static const char * const pm8550b_vreg_l[] = { "vdd18", "vdd3", }; static const struct eusb2_repeater_init_tbl pm8550b_init_tbl[] = { QCOM_EUSB2_REPEATER_INIT_CFG(EUSB2_TUNE_IUSB2, 0x8), QCOM_EUSB2_REPEATER_INIT_CFG(EUSB2_TUNE_SQUELCH_U, 0x3), QCOM_EUSB2_REPEATER_INIT_CFG(EUSB2_TUNE_USB2_PREEM, 0x5), }; static const struct eusb2_repeater_cfg pm8550b_eusb2_cfg = { .init_tbl = pm8550b_init_tbl, .init_tbl_num = ARRAY_SIZE(pm8550b_init_tbl), .vreg_list = pm8550b_vreg_l, .num_vregs = ARRAY_SIZE(pm8550b_vreg_l), }; static int eusb2_repeater_init_vregs(struct eusb2_repeater *rptr) { int num = rptr->cfg->num_vregs; struct device *dev = rptr->dev; int i; rptr->vregs = devm_kcalloc(dev, num, sizeof(*rptr->vregs), GFP_KERNEL); if (!rptr->vregs) return -ENOMEM; for (i = 0; i < num; i++) rptr->vregs[i].supply = rptr->cfg->vreg_list[i]; return devm_regulator_bulk_get(dev, num, rptr->vregs); } static int eusb2_repeater_init(struct phy *phy) { struct eusb2_repeater *rptr = phy_get_drvdata(phy); const struct eusb2_repeater_init_tbl *init_tbl = rptr->cfg->init_tbl; int num = rptr->cfg->init_tbl_num; u32 val; int ret; int i; ret = regulator_bulk_enable(rptr->cfg->num_vregs, rptr->vregs); if (ret) return ret; regmap_update_bits(rptr->regmap, rptr->base + EUSB2_EN_CTL1, EUSB2_RPTR_EN, EUSB2_RPTR_EN); for (i = 0; i < num; i++) regmap_update_bits(rptr->regmap, rptr->base + init_tbl[i].offset, init_tbl[i].val, init_tbl[i].val); ret = regmap_read_poll_timeout(rptr->regmap, rptr->base + EUSB2_RPTR_STATUS, val, val & RPTR_OK, 10, 5); if (ret) dev_err(rptr->dev, "initialization timed-out\n"); return ret; } static int eusb2_repeater_set_mode(struct phy *phy, enum phy_mode mode, int submode) { struct eusb2_repeater *rptr = phy_get_drvdata(phy); switch (mode) { case PHY_MODE_USB_HOST: /* * CM.Lx is prohibited when repeater is already into Lx state as * per eUSB 1.2 Spec. Below implement software workaround until * PHY and controller is fixing seen observation. */ regmap_update_bits(rptr->regmap, rptr->base + EUSB2_FORCE_EN_5, F_CLK_19P2M_EN, F_CLK_19P2M_EN); regmap_update_bits(rptr->regmap, rptr->base + EUSB2_FORCE_VAL_5, V_CLK_19P2M_EN, V_CLK_19P2M_EN); break; case PHY_MODE_USB_DEVICE: /* * In device mode clear host mode related workaround as there * is no repeater reset available, and enable/disable of * repeater doesn't clear previous value due to shared * regulators (say host <-> device mode switch). */ regmap_update_bits(rptr->regmap, rptr->base + EUSB2_FORCE_EN_5, F_CLK_19P2M_EN, 0); regmap_update_bits(rptr->regmap, rptr->base + EUSB2_FORCE_VAL_5, V_CLK_19P2M_EN, 0); break; default: return -EINVAL; } return 0; } static int eusb2_repeater_exit(struct phy *phy) { struct eusb2_repeater *rptr = phy_get_drvdata(phy); return regulator_bulk_disable(rptr->cfg->num_vregs, rptr->vregs); } static const struct phy_ops eusb2_repeater_ops = { .init = eusb2_repeater_init, .exit = eusb2_repeater_exit, .set_mode = eusb2_repeater_set_mode, .owner = THIS_MODULE, }; static int eusb2_repeater_probe(struct platform_device *pdev) { struct eusb2_repeater *rptr; struct device *dev = &pdev->dev; struct phy_provider *phy_provider; struct device_node *np = dev->of_node; u32 res; int ret; rptr = devm_kzalloc(dev, sizeof(*rptr), GFP_KERNEL); if (!rptr) return -ENOMEM; rptr->dev = dev; dev_set_drvdata(dev, rptr); rptr->cfg = of_device_get_match_data(dev); if (!rptr->cfg) return -EINVAL; rptr->regmap = dev_get_regmap(dev->parent, NULL); if (!rptr->regmap) return -ENODEV; ret = of_property_read_u32(np, "reg", &res); if (ret < 0) return ret; rptr->base = res; ret = eusb2_repeater_init_vregs(rptr); if (ret < 0) { dev_err(dev, "unable to get supplies\n"); return ret; } rptr->phy = devm_phy_create(dev, np, &eusb2_repeater_ops); if (IS_ERR(rptr->phy)) { dev_err(dev, "failed to create PHY: %d\n", ret); return PTR_ERR(rptr->phy); } phy_set_drvdata(rptr->phy, rptr); phy_provider = devm_of_phy_provider_register(dev, of_phy_simple_xlate); if (IS_ERR(phy_provider)) return PTR_ERR(phy_provider); dev_info(dev, "Registered Qcom-eUSB2 repeater\n"); return 0; } static int eusb2_repeater_remove(struct platform_device *pdev) { struct eusb2_repeater *rptr = platform_get_drvdata(pdev); if (!rptr) return 0; eusb2_repeater_exit(rptr->phy); return 0; } static const struct of_device_id eusb2_repeater_of_match_table[] = { { .compatible = "qcom,pm8550b-eusb2-repeater", .data = &pm8550b_eusb2_cfg, }, { }, }; MODULE_DEVICE_TABLE(of, eusb2_repeater_of_match_table); static struct platform_driver eusb2_repeater_driver = { .probe = eusb2_repeater_probe, .remove = eusb2_repeater_remove, .driver = { .name = "qcom-eusb2-repeater", .of_match_table = eusb2_repeater_of_match_table, }, }; module_platform_driver(eusb2_repeater_driver); MODULE_DESCRIPTION("Qualcomm PMIC eUSB2 Repeater driver"); MODULE_LICENSE("GPL");
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1