PCI: qcom: Allow ->post_init() to fail
authorBjorn Andersson <bjorn.andersson@linaro.org>
Sun, 16 Jul 2017 06:42:03 +0000 (23:42 -0700)
committerBjorn Helgaas <bhelgaas@google.com>
Thu, 3 Aug 2017 21:55:43 +0000 (16:55 -0500)
host_init() should detect and propagate errors from post_init().

In addition, by acknowledging that post_init() can fail we must disable the
post_init() resources in a step separate from the deinit, so that we don't
try to disable the post_init() resources a second time.

Signed-off-by: Bjorn Andersson <bjorn.andersson@linaro.org>
Signed-off-by: Bjorn Helgaas <bhelgaas@google.com>
Acked-by: Stanimir Varbanov <svarbanov@mm-sol.com>
drivers/pci/dwc/pcie-qcom.c

index 7b70374..26e84a9 100644 (file)
@@ -124,6 +124,7 @@ struct qcom_pcie_ops {
        int (*init)(struct qcom_pcie *pcie);
        int (*post_init)(struct qcom_pcie *pcie);
        void (*deinit)(struct qcom_pcie *pcie);
+       void (*post_deinit)(struct qcom_pcie *pcie);
        void (*ltssm_enable)(struct qcom_pcie *pcie);
 };
 
@@ -517,13 +518,19 @@ static void qcom_pcie_deinit_v2(struct qcom_pcie *pcie)
 {
        struct qcom_pcie_resources_v2 *res = &pcie->res.v2;
 
-       clk_disable_unprepare(res->pipe_clk);
        clk_disable_unprepare(res->slave_clk);
        clk_disable_unprepare(res->master_clk);
        clk_disable_unprepare(res->cfg_clk);
        clk_disable_unprepare(res->aux_clk);
 }
 
+static void qcom_pcie_post_deinit_v2(struct qcom_pcie *pcie)
+{
+       struct qcom_pcie_resources_v2 *res = &pcie->res.v2;
+
+       clk_disable_unprepare(res->pipe_clk);
+}
+
 static int qcom_pcie_init_v2(struct qcom_pcie *pcie)
 {
        struct qcom_pcie_resources_v2 *res = &pcie->res.v2;
@@ -907,8 +914,11 @@ static int qcom_pcie_host_init(struct pcie_port *pp)
        if (ret)
                goto err_deinit;
 
-       if (pcie->ops->post_init)
-               pcie->ops->post_init(pcie);
+       if (pcie->ops->post_init) {
+               ret = pcie->ops->post_init(pcie);
+               if (ret)
+                       goto err_disable_phy;
+       }
 
        dw_pcie_setup_rc(pp);
 
@@ -924,6 +934,9 @@ static int qcom_pcie_host_init(struct pcie_port *pp)
        return 0;
 err:
        qcom_ep_reset_assert(pcie);
+       if (pcie->ops->post_deinit)
+               pcie->ops->post_deinit(pcie);
+err_disable_phy:
        phy_power_off(pcie->phy);
 err_deinit:
        pcie->ops->deinit(pcie);
@@ -971,6 +984,7 @@ static const struct qcom_pcie_ops ops_v2 = {
        .init = qcom_pcie_init_v2,
        .post_init = qcom_pcie_post_init_v2,
        .deinit = qcom_pcie_deinit_v2,
+       .post_deinit = qcom_pcie_post_deinit_v2,
        .ltssm_enable = qcom_pcie_v2_ltssm_enable,
 };