| /* | 
 |  * USB cluster support for Armada 375 platform. | 
 |  * | 
 |  * Copyright (C) 2014 Marvell | 
 |  * | 
 |  * Gregory CLEMENT <gregory.clement@free-electrons.com> | 
 |  * | 
 |  * This file is licensed under the terms of the GNU General Public | 
 |  * License version 2 or later. This program is licensed "as is" | 
 |  * without any warranty of any kind, whether express or implied. | 
 |  * | 
 |  * Armada 375 comes with an USB2 host and device controller and an | 
 |  * USB3 controller. The USB cluster control register allows to manage | 
 |  * common features of both USB controllers. | 
 |  */ | 
 |  | 
 | #include <dt-bindings/phy/phy.h> | 
 | #include <linux/init.h> | 
 | #include <linux/io.h> | 
 | #include <linux/kernel.h> | 
 | #include <linux/module.h> | 
 | #include <linux/of_address.h> | 
 | #include <linux/phy/phy.h> | 
 | #include <linux/platform_device.h> | 
 |  | 
 | #define USB2_PHY_CONFIG_DISABLE BIT(0) | 
 |  | 
 | struct armada375_cluster_phy { | 
 | 	struct phy *phy; | 
 | 	void __iomem *reg; | 
 | 	bool use_usb3; | 
 | 	int phy_provided; | 
 | }; | 
 |  | 
 | static int armada375_usb_phy_init(struct phy *phy) | 
 | { | 
 | 	struct armada375_cluster_phy *cluster_phy; | 
 | 	u32 reg; | 
 |  | 
 | 	cluster_phy = phy_get_drvdata(phy); | 
 | 	if (!cluster_phy) | 
 | 		return -ENODEV; | 
 |  | 
 | 	reg = readl(cluster_phy->reg); | 
 | 	if (cluster_phy->use_usb3) | 
 | 		reg |= USB2_PHY_CONFIG_DISABLE; | 
 | 	else | 
 | 		reg &= ~USB2_PHY_CONFIG_DISABLE; | 
 | 	writel(reg, cluster_phy->reg); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static const struct phy_ops armada375_usb_phy_ops = { | 
 | 	.init = armada375_usb_phy_init, | 
 | 	.owner = THIS_MODULE, | 
 | }; | 
 |  | 
 | /* | 
 |  * Only one controller can use this PHY. We shouldn't have the case | 
 |  * when two controllers want to use this PHY. But if this case occurs | 
 |  * then we provide a phy to the first one and return an error for the | 
 |  * next one. This error has also to be an error returned by | 
 |  * devm_phy_optional_get() so different from ENODEV for USB2. In the | 
 |  * USB3 case it still optional and we use ENODEV. | 
 |  */ | 
 | static struct phy *armada375_usb_phy_xlate(struct device *dev, | 
 | 					struct of_phandle_args *args) | 
 | { | 
 | 	struct armada375_cluster_phy *cluster_phy = dev_get_drvdata(dev); | 
 |  | 
 | 	if (!cluster_phy) | 
 | 		return  ERR_PTR(-ENODEV); | 
 |  | 
 | 	/* | 
 | 	 * Either the phy had never been requested and then the first | 
 | 	 * usb claiming it can get it, or it had already been | 
 | 	 * requested in this case, we only allow to use it with the | 
 | 	 * same configuration. | 
 | 	 */ | 
 | 	if (WARN_ON((cluster_phy->phy_provided != PHY_NONE) && | 
 | 			(cluster_phy->phy_provided != args->args[0]))) { | 
 | 		dev_err(dev, "This PHY has already been provided!\n"); | 
 | 		dev_err(dev, "Check your device tree, only one controller can use it\n."); | 
 | 		if (args->args[0] == PHY_TYPE_USB2) | 
 | 			return ERR_PTR(-EBUSY); | 
 | 		else | 
 | 			return ERR_PTR(-ENODEV); | 
 | 	} | 
 |  | 
 | 	if (args->args[0] == PHY_TYPE_USB2) | 
 | 		cluster_phy->use_usb3 = false; | 
 | 	else if (args->args[0] == PHY_TYPE_USB3) | 
 | 		cluster_phy->use_usb3 = true; | 
 | 	else { | 
 | 		dev_err(dev, "Invalid PHY mode\n"); | 
 | 		return ERR_PTR(-ENODEV); | 
 | 	} | 
 |  | 
 | 	/* Store which phy mode is used for next test */ | 
 | 	cluster_phy->phy_provided = args->args[0]; | 
 |  | 
 | 	return cluster_phy->phy; | 
 | } | 
 |  | 
 | static int armada375_usb_phy_probe(struct platform_device *pdev) | 
 | { | 
 | 	struct device *dev = &pdev->dev; | 
 | 	struct phy *phy; | 
 | 	struct phy_provider *phy_provider; | 
 | 	void __iomem *usb_cluster_base; | 
 | 	struct resource *res; | 
 | 	struct armada375_cluster_phy *cluster_phy; | 
 |  | 
 | 	cluster_phy = devm_kzalloc(dev, sizeof(*cluster_phy), GFP_KERNEL); | 
 | 	if (!cluster_phy) | 
 | 		return  -ENOMEM; | 
 |  | 
 | 	res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
 | 	usb_cluster_base = devm_ioremap_resource(&pdev->dev, res); | 
 | 	if (IS_ERR(usb_cluster_base)) | 
 | 		return PTR_ERR(usb_cluster_base); | 
 |  | 
 | 	phy = devm_phy_create(dev, NULL, &armada375_usb_phy_ops); | 
 | 	if (IS_ERR(phy)) { | 
 | 		dev_err(dev, "failed to create PHY\n"); | 
 | 		return PTR_ERR(phy); | 
 | 	} | 
 |  | 
 | 	cluster_phy->phy = phy; | 
 | 	cluster_phy->reg = usb_cluster_base; | 
 |  | 
 | 	dev_set_drvdata(dev, cluster_phy); | 
 | 	phy_set_drvdata(phy, cluster_phy); | 
 |  | 
 | 	phy_provider = devm_of_phy_provider_register(&pdev->dev, | 
 | 						     armada375_usb_phy_xlate); | 
 | 	return PTR_ERR_OR_ZERO(phy_provider); | 
 | } | 
 |  | 
 | static const struct of_device_id of_usb_cluster_table[] = { | 
 | 	{ .compatible = "marvell,armada-375-usb-cluster", }, | 
 | 	{ /* end of list */ }, | 
 | }; | 
 | MODULE_DEVICE_TABLE(of, of_usb_cluster_table); | 
 |  | 
 | static struct platform_driver armada375_usb_phy_driver = { | 
 | 	.probe	= armada375_usb_phy_probe, | 
 | 	.driver = { | 
 | 		.of_match_table	= of_usb_cluster_table, | 
 | 		.name  = "armada-375-usb-cluster", | 
 | 	} | 
 | }; | 
 | module_platform_driver(armada375_usb_phy_driver); | 
 |  | 
 | MODULE_DESCRIPTION("Armada 375 USB cluster driver"); | 
 | MODULE_AUTHOR("Gregory CLEMENT <gregory.clement@free-electrons.com>"); | 
 | MODULE_LICENSE("GPL"); |