Commit cf92b9df authored by Dmitry Baryshkov's avatar Dmitry Baryshkov Committed by Greg Kroah-Hartman

usb: typec: qcom-pmic-typec: add support for PMI632 PMIC

The PMI632 PMIC support Type-C port handling, but lacks USB
PowerDelivery support. The TCPM requires all callbacks to be provided
by the implementation. Implement a special, 'stub' Qcom PD PHY
implementation to enable the PMI632 support.
Acked-by: default avatarBryan O'Donoghue <bryan.odonoghue@linaro.org>
Acked-by: default avatarHeikki Krogerus <heikki.krogerus@linux.intel.com>
Tested-by: Luca Weiss <luca.weiss@fairphone.com> # sdm632-fairphone-fp3
Signed-off-by: default avatarDmitry Baryshkov <dmitry.baryshkov@linaro.org>
Link: https://lore.kernel.org/r/20240130-pmi632-typec-v3-3-b05fe44f0a51@linaro.orgSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent f637c0c6
...@@ -3,4 +3,5 @@ ...@@ -3,4 +3,5 @@
obj-$(CONFIG_TYPEC_QCOM_PMIC) += qcom_pmic_tcpm.o obj-$(CONFIG_TYPEC_QCOM_PMIC) += qcom_pmic_tcpm.o
qcom_pmic_tcpm-y += qcom_pmic_typec.o \ qcom_pmic_tcpm-y += qcom_pmic_typec.o \
qcom_pmic_typec_port.o \ qcom_pmic_typec_port.o \
qcom_pmic_typec_pdphy.o qcom_pmic_typec_pdphy.o \
qcom_pmic_typec_pdphy_stub.o \
...@@ -42,7 +42,7 @@ static int qcom_pmic_typec_probe(struct platform_device *pdev) ...@@ -42,7 +42,7 @@ static int qcom_pmic_typec_probe(struct platform_device *pdev)
const struct pmic_typec_resources *res; const struct pmic_typec_resources *res;
struct regmap *regmap; struct regmap *regmap;
struct device *bridge_dev; struct device *bridge_dev;
u32 base[2]; u32 base;
int ret; int ret;
res = of_device_get_match_data(dev); res = of_device_get_match_data(dev);
...@@ -62,19 +62,29 @@ static int qcom_pmic_typec_probe(struct platform_device *pdev) ...@@ -62,19 +62,29 @@ static int qcom_pmic_typec_probe(struct platform_device *pdev)
return -ENODEV; return -ENODEV;
} }
ret = of_property_read_u32_array(np, "reg", base, 2); ret = of_property_read_u32_index(np, "reg", 0, &base);
if (ret) if (ret)
return ret; return ret;
ret = qcom_pmic_typec_port_probe(pdev, tcpm, ret = qcom_pmic_typec_port_probe(pdev, tcpm,
res->port_res, regmap, base[0]); res->port_res, regmap, base);
if (ret) if (ret)
return ret; return ret;
ret = qcom_pmic_typec_pdphy_probe(pdev, tcpm, if (res->pdphy_res) {
res->pdphy_res, regmap, base[1]); ret = of_property_read_u32_index(np, "reg", 1, &base);
if (ret) if (ret)
return ret; return ret;
ret = qcom_pmic_typec_pdphy_probe(pdev, tcpm,
res->pdphy_res, regmap, base);
if (ret)
return ret;
} else {
ret = qcom_pmic_typec_pdphy_stub_probe(pdev, tcpm);
if (ret)
return ret;
}
platform_set_drvdata(pdev, tcpm); platform_set_drvdata(pdev, tcpm);
...@@ -123,8 +133,14 @@ static const struct pmic_typec_resources pm8150b_typec_res = { ...@@ -123,8 +133,14 @@ static const struct pmic_typec_resources pm8150b_typec_res = {
.port_res = &pm8150b_port_res, .port_res = &pm8150b_port_res,
}; };
static const struct pmic_typec_resources pmi632_typec_res = {
/* PD PHY not present */
.port_res = &pm8150b_port_res,
};
static const struct of_device_id qcom_pmic_typec_table[] = { static const struct of_device_id qcom_pmic_typec_table[] = {
{ .compatible = "qcom,pm8150b-typec", .data = &pm8150b_typec_res }, { .compatible = "qcom,pm8150b-typec", .data = &pm8150b_typec_res },
{ .compatible = "qcom,pmi632-typec", .data = &pmi632_typec_res },
{ } { }
}; };
MODULE_DEVICE_TABLE(of, qcom_pmic_typec_table); MODULE_DEVICE_TABLE(of, qcom_pmic_typec_table);
......
...@@ -31,5 +31,7 @@ int qcom_pmic_typec_pdphy_probe(struct platform_device *pdev, ...@@ -31,5 +31,7 @@ int qcom_pmic_typec_pdphy_probe(struct platform_device *pdev,
const struct pmic_typec_pdphy_resources *res, const struct pmic_typec_pdphy_resources *res,
struct regmap *regmap, struct regmap *regmap,
u32 base); u32 base);
int qcom_pmic_typec_pdphy_stub_probe(struct platform_device *pdev,
struct pmic_typec *tcpm);
#endif /* __QCOM_PMIC_TYPEC_PDPHY_H__ */ #endif /* __QCOM_PMIC_TYPEC_PDPHY_H__ */
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (c) 2024, Linaro Ltd. All rights reserved.
*/
#include <linux/err.h>
#include <linux/interrupt.h>
#include <linux/kernel.h>
#include <linux/mod_devicetable.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/regulator/consumer.h>
#include <linux/slab.h>
#include <linux/usb/pd.h>
#include <linux/usb/tcpm.h>
#include "qcom_pmic_typec.h"
#include "qcom_pmic_typec_pdphy.h"
static int qcom_pmic_typec_pdphy_stub_pd_transmit(struct tcpc_dev *tcpc,
enum tcpm_transmit_type type,
const struct pd_message *msg,
unsigned int negotiated_rev)
{
struct pmic_typec *tcpm = tcpc_to_tcpm(tcpc);
struct device *dev = tcpm->dev;
dev_dbg(dev, "pdphy_transmit: type=%d\n", type);
tcpm_pd_transmit_complete(tcpm->tcpm_port,
TCPC_TX_SUCCESS);
return 0;
}
static int qcom_pmic_typec_pdphy_stub_set_pd_rx(struct tcpc_dev *tcpc, bool on)
{
struct pmic_typec *tcpm = tcpc_to_tcpm(tcpc);
struct device *dev = tcpm->dev;
dev_dbg(dev, "set_pd_rx: %s\n", on ? "on" : "off");
return 0;
}
static int qcom_pmic_typec_pdphy_stub_set_roles(struct tcpc_dev *tcpc, bool attached,
enum typec_role power_role,
enum typec_data_role data_role)
{
struct pmic_typec *tcpm = tcpc_to_tcpm(tcpc);
struct device *dev = tcpm->dev;
dev_dbg(dev, "pdphy_set_roles: data_role_host=%d power_role_src=%d\n",
data_role, power_role);
return 0;
}
static int qcom_pmic_typec_pdphy_stub_start(struct pmic_typec *tcpm,
struct tcpm_port *tcpm_port)
{
return 0;
}
static void qcom_pmic_typec_pdphy_stub_stop(struct pmic_typec *tcpm)
{
}
int qcom_pmic_typec_pdphy_stub_probe(struct platform_device *pdev,
struct pmic_typec *tcpm)
{
tcpm->tcpc.set_pd_rx = qcom_pmic_typec_pdphy_stub_set_pd_rx;
tcpm->tcpc.set_roles = qcom_pmic_typec_pdphy_stub_set_roles;
tcpm->tcpc.pd_transmit = qcom_pmic_typec_pdphy_stub_pd_transmit;
tcpm->pdphy_start = qcom_pmic_typec_pdphy_stub_start;
tcpm->pdphy_stop = qcom_pmic_typec_pdphy_stub_stop;
return 0;
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment