|
27 | 27 | #include <linux/types.h>
|
28 | 28 | #include <linux/interrupt.h>
|
29 | 29 | #include <linux/reset.h>
|
| 30 | +#include <linux/pm_domain.h> |
| 31 | +#include <linux/pm_runtime.h> |
30 | 32 |
|
31 | 33 | #include "pcie-designware.h"
|
32 | 34 |
|
@@ -59,6 +61,11 @@ struct imx6_pcie {
|
59 | 61 | u32 tx_swing_low;
|
60 | 62 | int link_gen;
|
61 | 63 | struct regulator *vpcie;
|
| 64 | + |
| 65 | + /* power domain for pcie */ |
| 66 | + struct device *pd_pcie; |
| 67 | + /* power domain for pcie phy */ |
| 68 | + struct device *pd_pcie_phy; |
62 | 69 | };
|
63 | 70 |
|
64 | 71 | /* Parameters for the waiting for PCIe PHY PLL to lock on i.MX7 */
|
@@ -292,6 +299,43 @@ static int imx6q_pcie_abort_handler(unsigned long addr,
|
292 | 299 | return 1;
|
293 | 300 | }
|
294 | 301 |
|
| 302 | +static int imx6_pcie_attach_pd(struct device *dev) |
| 303 | +{ |
| 304 | + struct imx6_pcie *imx6_pcie = dev_get_drvdata(dev); |
| 305 | + struct device_link *link; |
| 306 | + |
| 307 | + /* Do nothing when in a single power domain */ |
| 308 | + if (dev->pm_domain) |
| 309 | + return 0; |
| 310 | + |
| 311 | + imx6_pcie->pd_pcie = dev_pm_domain_attach_by_name(dev, "pcie"); |
| 312 | + if (IS_ERR(imx6_pcie->pd_pcie)) |
| 313 | + return PTR_ERR(imx6_pcie->pd_pcie); |
| 314 | + link = device_link_add(dev, imx6_pcie->pd_pcie, |
| 315 | + DL_FLAG_STATELESS | |
| 316 | + DL_FLAG_PM_RUNTIME | |
| 317 | + DL_FLAG_RPM_ACTIVE); |
| 318 | + if (!link) { |
| 319 | + dev_err(dev, "Failed to add device_link to pcie pd.\n"); |
| 320 | + return -EINVAL; |
| 321 | + } |
| 322 | + |
| 323 | + imx6_pcie->pd_pcie_phy = dev_pm_domain_attach_by_name(dev, "pcie_phy"); |
| 324 | + if (IS_ERR(imx6_pcie->pd_pcie_phy)) |
| 325 | + return PTR_ERR(imx6_pcie->pd_pcie_phy); |
| 326 | + |
| 327 | + device_link_add(dev, imx6_pcie->pd_pcie_phy, |
| 328 | + DL_FLAG_STATELESS | |
| 329 | + DL_FLAG_PM_RUNTIME | |
| 330 | + DL_FLAG_RPM_ACTIVE); |
| 331 | + if (IS_ERR(link)) { |
| 332 | + dev_err(dev, "Failed to add device_link to pcie_phy pd: %ld\n", PTR_ERR(link)); |
| 333 | + return PTR_ERR(link); |
| 334 | + } |
| 335 | + |
| 336 | + return 0; |
| 337 | +} |
| 338 | + |
295 | 339 | static void imx6_pcie_assert_core_reset(struct imx6_pcie *imx6_pcie)
|
296 | 340 | {
|
297 | 341 | struct device *dev = imx6_pcie->pci->dev;
|
@@ -985,6 +1029,10 @@ static int imx6_pcie_probe(struct platform_device *pdev)
|
985 | 1029 |
|
986 | 1030 | platform_set_drvdata(pdev, imx6_pcie);
|
987 | 1031 |
|
| 1032 | + ret = imx6_pcie_attach_pd(dev); |
| 1033 | + if (ret) |
| 1034 | + return ret; |
| 1035 | + |
988 | 1036 | ret = imx6_add_pcie_port(imx6_pcie, pdev);
|
989 | 1037 | if (ret < 0)
|
990 | 1038 | return ret;
|
|
0 commit comments