首页 > 其他分享 >MSM8953/SDM450 去PMI的USB3.0 TYPE-C Micro USB OTG功能适配

MSM8953/SDM450 去PMI的USB3.0 TYPE-C Micro USB OTG功能适配

时间:2022-08-29 14:12:21浏览次数:62  
标签:MSM8953 USB 适配 det dev irq gpio id usb

提前说明一下有哪些“坑”。

1、PM8953 GPIO_8的TZ权限

2、PM8953 GPIO_8寄存器的写入保护

3、去掉高通默认的ID检测

4、增加dwc3的ID检测

5、增加TYPE-C的ID PIN控制

 

设备树配置:

&tlmm {
	usb_id_default: usb_id_default {
		mux {
			pins = "gpio1";
			function = "gpio";
		};

		config {
			pins = "gpio1";
			drive-strength = <2>;
			bias-pull-up;
		};
	};
};

&pm8953_typec {
	qcom,ssmux-gpio = <&tlmm 139 GPIO_ACTIVE_LOW>;
	qcom,id-gpio = <&pm8953_gpios 8 0x0>;
	pinctrl-names = "default";
	pinctrl-0 = <&typec_ssmux_config>;
	status = "ok";
};

&usb3 {
	vbus_dwc3-supply = <&usb_otg_switch>;
	otg,id_det_pin = <&tlmm 1 0x0>;
	extcon = <&usb_detect>;
	pinctrl-names = "default";
	pinctrl-0 = <&usb_id_default>;
};

&usb_otg_switch {
	gpio = <&tlmm 3 0>;
};

&soc {
	usb_detect: usb_detect {
		compatible = "qcom,gpio-usbdetect";
		interrupt-parent = <&spmi_bus>;
		interrupts = <0x0 0xc6 0x0>;
		interrupt-names = "vbus_det_irq";
		qcom,vbus-det-gpio = <&pm8953_gpios 7 0x00>;
		pinctrl-names = "default";
		pinctrl-0 = <&pm8953_gpio7_default>;
	};
};

 

修改PM8953 GPIO8 TZ权限

index e1d339e..dda5a23 100755
--- a/TZ.BF.4.0.5/trustzone_images/core/systemdrivers/pmic/config/msm8953_pm8950/src/pm_spmi_config.c
+++ b/TZ.BF.4.0.5/trustzone_images/core/systemdrivers/pmic/config/msm8953_pm8950/src/pm_spmi_config.c
@@ -167,7 +167,7 @@ SpmiCfg_ChannelCfg pm_spmi_pheriph_cfg [] =
     
     
     {0x0, 0xFE, 0,  PM_RPM_OWNER,         PM_RPM_OWNER},         /* TRIM          */
-    {0x0, 0xC7, 0,  PM_WCONNECT_OWNER,    SPMI_OPEN_OWNER},        /* GPIO8          */
+    {0x0, 0xC7, 0,  PM_APPS_HLOS_OWNER,    SPMI_OPEN_OWNER},        /* GPIO8          */
     {0x0, 0xC6, 0,  PM_APPS_HLOS_OWNER,   SPMI_OPEN_OWNER},       /* GPIO7          */
     {0x0, 0xC5, 0,  PM_WCONNECT_OWNER,    SPMI_OPEN_OWNER},        /* GPIO6          */
     {0x0, 0xC4, 0,  PM_WCONNECT_OWNER,    SPMI_OPEN_OWNER},        /* GPIO5          */

  

去除PM8953 GPIO_8寄存器的写入保护

index 0f0b7ba..f7946e0 100644
--- a/drivers/spmi/spmi-pmic-arb.c
+++ b/drivers/spmi/spmi-pmic-arb.c
@@ -451,6 +451,7 @@ static int pmic_arb_write_cmd(struct spmi_controller *ctrl, u8 opc, u8 sid,
                dev_err(&pa->spmic->dev,
                        "error: impermissible write to peripheral sid:%d addr:0x%x\n",
                        sid, addr);
+               if (!(0xc740 == (addr & 0xc740))) // skip PM8953 GPIO_8 Register protect check
                return -ENODEV;
        }

  

去掉高通默认的ID检测

diff --git a/drivers/platform/msm/gpio-usbdetect.c b/drivers/platform/msm/gpio-usbdetect.c
index 6730d4a..d2cbdea 100644
--- a/drivers/platform/msm/gpio-usbdetect.c
+++ b/drivers/platform/msm/gpio-usbdetect.c
@@ -22,6 +22,7 @@
 #include <linux/gpio.h>
 #include <linux/extcon.h>
 #include <linux/regulator/consumer.h>
+#include <linux/delay.h>
 
 struct gpio_usbdetect {
 	struct platform_device	*pdev;
@@ -45,66 +46,21 @@ static irqreturn_t gpio_usbdetect_vbus_irq(int irq, void *data)
 	struct gpio_usbdetect *usb = data;
 	union extcon_property_value val;
 
-	usb->vbus_state = gpio_get_value(usb->gpio);
+	usb->vbus_state = !!gpio_get_value(usb->gpio);
+
 	if (usb->vbus_state) {
-		dev_dbg(&usb->pdev->dev, "setting vbus notification\n");
+		dev_err(&usb->pdev->dev, "xcz setting vbus notification\n");
 		val.intval = true;
-		extcon_set_property(usb->extcon_dev, EXTCON_USB,
-					EXTCON_PROP_USB_SS, val);
+
 		extcon_set_cable_state_(usb->extcon_dev, EXTCON_USB, 1);
 	} else {
-		dev_dbg(&usb->pdev->dev, "setting vbus removed notification\n");
+		dev_err(&usb->pdev->dev, "xcz setting vbus removed notification\n");
 		extcon_set_cable_state_(usb->extcon_dev, EXTCON_USB, 0);
 	}
 
 	return IRQ_HANDLED;
 }
 
-static irqreturn_t gpio_usbdetect_id_irq(int irq, void *data)
-{
-	struct gpio_usbdetect *usb = data;
-	int ret;
-
-	ret = irq_get_irqchip_state(irq, IRQCHIP_STATE_LINE_LEVEL,
-			&usb->id_state);
-	if (ret < 0) {
-		dev_err(&usb->pdev->dev, "unable to read ID IRQ LINE\n");
-		return IRQ_HANDLED;
-	}
-
-	return IRQ_WAKE_THREAD;
-}
-
-static irqreturn_t gpio_usbdetect_id_irq_thread(int irq, void *data)
-{
-	struct gpio_usbdetect *usb = data;
-	bool curr_id_state;
-	static int prev_id_state = -EINVAL;
-	union extcon_property_value val;
-
-	curr_id_state = usb->id_state;
-	if (curr_id_state == prev_id_state) {
-		dev_dbg(&usb->pdev->dev, "no change in ID state\n");
-		return IRQ_HANDLED;
-	}
-
-	if (curr_id_state) {
-		dev_dbg(&usb->pdev->dev, "stopping usb host\n");
-		extcon_set_cable_state_(usb->extcon_dev, EXTCON_USB_HOST, 0);
-		enable_irq(usb->vbus_det_irq);
-	} else {
-		dev_dbg(&usb->pdev->dev, "starting usb HOST\n");
-		disable_irq(usb->vbus_det_irq);
-		val.intval = true;
-		extcon_set_property(usb->extcon_dev, EXTCON_USB_HOST,
-					EXTCON_PROP_USB_SS, val);
-		extcon_set_cable_state_(usb->extcon_dev, EXTCON_USB_HOST, 1);
-	}
-
-	prev_id_state = curr_id_state;
-	return IRQ_HANDLED;
-}
-
 static const u32 gpio_usb_extcon_exclusive[] = {0x3, 0};
 
 static int gpio_usbdetect_probe(struct platform_device *pdev)
@@ -164,8 +120,6 @@ static int gpio_usbdetect_probe(struct platform_device *pdev)
 				"qcom,vbus-det-gpio", 0);
 	if (usb->gpio < 0) {
 		dev_err(&pdev->dev, "Failed to get gpio: %d\n", usb->gpio);
-		rc = usb->gpio;
-		goto error;
 	}
 
 	rc = gpio_request(usb->gpio, "vbus-det-gpio");
@@ -174,7 +128,7 @@ static int gpio_usbdetect_probe(struct platform_device *pdev)
 		goto error;
 	}
 
-	usb->vbus_det_irq = gpio_to_irq(usb->gpio);
+	usb->vbus_det_irq = gpio_to_irq(usb->gpio);//0xc6;//= gpio_to_irq(usb->gpio);
 	if (usb->vbus_det_irq < 0) {
 		dev_err(&pdev->dev, "get vbus_det_irq failed\n");
 		rc = usb->vbus_det_irq;
@@ -191,35 +145,11 @@ static int gpio_usbdetect_probe(struct platform_device *pdev)
 		goto error;
 	}
 
-	usb->id_det_irq = platform_get_irq_byname(pdev, "pmic_id_irq");
-	if (usb->id_det_irq < 0) {
-		dev_err(&pdev->dev, "get id_det_irq failed\n");
-		rc = usb->id_det_irq;
-		goto error;
-	}
-
-	rc = devm_request_threaded_irq(&pdev->dev, usb->id_det_irq,
-				gpio_usbdetect_id_irq,
-				gpio_usbdetect_id_irq_thread,
-				IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING |
-				IRQF_ONESHOT, "id_det_irq", usb);
-	if (rc) {
-		dev_err(&pdev->dev, "request for id_det_irq failed: %d\n", rc);
-		goto error;
-	}
+	usb->id_det_irq = platform_get_irq_byname(pdev, "vbus_det_irq");
 
 	enable_irq_wake(usb->vbus_det_irq);
-	enable_irq_wake(usb->id_det_irq);
 	dev_set_drvdata(&pdev->dev, usb);
 
-	if (usb->id_det_irq) {
-		gpio_usbdetect_id_irq(usb->id_det_irq, usb);
-		if (!usb->id_state) {
-			gpio_usbdetect_id_irq_thread(usb->id_det_irq, usb);
-			return 0;
-		}
-	}
-
 	/* Read and report initial VBUS state */
 	gpio_usbdetect_vbus_irq(usb->vbus_det_irq, usb);
 

  

增加dwc3的ID检测

diff --git a/drivers/usb/dwc3/dwc3-msm.c b/drivers/usb/dwc3/dwc3-msm.c
index f549047..cc54672 100644
--- a/drivers/usb/dwc3/dwc3-msm.c
+++ b/drivers/usb/dwc3/dwc3-msm.c
@@ -1,4 +1,4 @@
-/* Copyright (c) 2012-2019, The Linux Foundation. All rights reserved.
+/* Copyright (c) 2012-2018, The Linux Foundation. All rights reserved.
  *
  * This program is free software; you can redistribute it and/or modify
  * it under the terms of the GNU General Public License version 2 and
@@ -55,6 +55,8 @@
 #include "debug.h"
 #include "xhci.h"
 
+#define DEBUG
+
 #define SDP_CONNETION_CHECK_TIME 10000 /* in ms */
 
 /* time out to wait for USB cable status notification (in ms)*/
@@ -235,6 +237,8 @@ struct dwc3_msm {
 	bool			use_pdc_interrupts;
 	enum dwc3_id_state	id_state;
 	unsigned long		lpm_flags;
+	int                     id_det_pin;
+	int                     id_det_irq;
 #define MDWC3_SS_PHY_SUSPEND		BIT(0)
 #define MDWC3_ASYNC_IRQ_WAKE_CAPABILITY	BIT(1)
 #define MDWC3_POWER_COLLAPSE		BIT(2)
@@ -290,7 +294,7 @@ struct dwc3_msm {
 
 #define DSTS_CONNECTSPD_SS		0x4
 
-
+static int dwc3_msm_id_notifier(struct notifier_block *nb, unsigned long event, void *ptr);
 static void dwc3_pwr_event_handler(struct dwc3_msm *mdwc);
 static int dwc3_msm_gadget_vbus_draw(struct dwc3_msm *mdwc, unsigned int mA);
 static void dwc3_msm_notify_event(struct dwc3 *dwc, unsigned int event,
@@ -1860,13 +1864,8 @@ static void dwc3_msm_notify_event(struct dwc3 *dwc, unsigned int event,
 		reg |= DWC3_GCTL_CORESOFTRESET;
 		dwc3_msm_write_reg(mdwc->base, DWC3_GCTL, reg);
 
-		/*
-		 * If the core could not recover after MAX_ERROR_RECOVERY_TRIES,
-		 * skip the restart USB work and keep the core in softreset
-		 * state.
-		 */
-		if (dwc->retries_on_error < MAX_ERROR_RECOVERY_TRIES)
-			schedule_work(&mdwc->restart_usb_work);
+		/* restart USB which performs full reset and reconnect */
+		schedule_work(&mdwc->restart_usb_work);
 		break;
 	case DWC3_CONTROLLER_RESET_EVENT:
 		dev_dbg(mdwc->dev, "DWC3_CONTROLLER_RESET_EVENT received\n");
@@ -2828,6 +2827,17 @@ static irqreturn_t msm_dwc3_pwr_irq(int irq, void *data)
 	return IRQ_HANDLED;
 }
 
+// add 2020-02-24
+static irqreturn_t id_det_irq_handle(int irq, void *data)
+{
+	struct dwc3_msm *mdwc = data;
+
+	dwc3_msm_id_notifier(&mdwc->id_nb, gpio_get_value(mdwc->id_det_pin) ? 0 : 1, mdwc->extcon_id);
+
+	return IRQ_HANDLED;
+}
+// add 2020-02-24 end
+
 static int dwc3_cpu_notifier_cb(struct notifier_block *nfb,
 		unsigned long action, void *hcpu)
 {
@@ -2953,7 +2963,7 @@ static int dwc3_msm_get_clk_gdsc(struct dwc3_msm *mdwc)
 	return 0;
 }
 
-static int dwc3_msm_id_notifier(struct notifier_block *nb,
+static int dwc3_msm_id_notifier(struct notifier_block *nb, //xcz
 	unsigned long event, void *ptr)
 {
 	struct dwc3_msm *mdwc = container_of(nb, struct dwc3_msm, id_nb);
@@ -3005,7 +3015,10 @@ static int dwc3_msm_vbus_notifier(struct notifier_block *nb,
 	struct dwc3_msm *mdwc = container_of(nb, struct dwc3_msm, vbus_nb);
 	struct dwc3 *dwc = platform_get_drvdata(mdwc->dwc3);
 
-	dev_dbg(mdwc->dev, "vbus:%ld event received\n", event);
+	dev_err(mdwc->dev, "vbus:%ld event received\n", event);
+
+	printk("xcz haha ===== vbus:%ld event received\n", event);
+
 
 	if (mdwc->vbus_active == event)
 		return NOTIFY_DONE;
@@ -3442,6 +3455,30 @@ static int dwc3_msm_probe(struct platform_device *pdev)
 		}
 	}
 
+	// add 2020-02-24
+	mdwc->id_det_pin = of_get_named_gpio(pdev->dev.of_node, "otg,id_det_pin", 0);
+	if (mdwc->id_det_pin > 0) {
+		ret = gpio_request(mdwc->id_det_pin, "id_det_pin");
+		if(0 != ret) {
+			dev_err(&pdev->dev, "xcz Otg id_det_pin request %d failed.\n", mdwc->id_det_pin);
+		}
+
+		mdwc->id_det_irq = gpio_to_irq(mdwc->id_det_pin);
+		dev_err(&pdev->dev, "xcz Otg id_det_pin: %d, irq: %d\n",
+						mdwc->id_det_pin, mdwc->id_det_irq);
+
+		ret = devm_request_irq(&pdev->dev,
+				       mdwc->id_det_irq,
+				       id_det_irq_handle,
+				       IRQF_TRIGGER_RISING |
+				       IRQF_TRIGGER_FALLING,
+				       "id_det_irq",
+				       mdwc);
+	} else {
+		dev_err(&pdev->dev, "xcz Otg id_det_pin NUll\n");
+	}
+	// add 2020-02-24 end
+
 	res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "tcsr_base");
 	if (!res) {
 		dev_dbg(&pdev->dev, "missing TCSR memory resource\n");
@@ -3701,7 +3738,8 @@ static int dwc3_msm_probe(struct platform_device *pdev)
 		if (pval.intval > 0)
 			dev_info(mdwc->dev, "charger detection in progress\n");
 	}
-
+	if(mdwc->id_det_pin > 0)
+		id_det_irq_handle(mdwc->id_det_irq, mdwc);
 	device_create_file(&pdev->dev, &dev_attr_mode);
 	device_create_file(&pdev->dev, &dev_attr_speed);
 	device_create_file(&pdev->dev, &dev_attr_usb_compliance_mode);

  

增加TYPE-C的ID PIN控制

diff --git a/drivers/power/supply/qcom/qpnp-typec.c b/drivers/power/supply/qcom/qpnp-typec.c
index 12aa16b..20f1a71 100644
--- a/drivers/power/supply/qcom/qpnp-typec.c
+++ b/drivers/power/supply/qcom/qpnp-typec.c
@@ -113,6 +113,7 @@ struct qpnp_typec_chip {
 	int			ssmux_gpio;
 	enum of_gpio_flags	gpio_flag;
 	int			typec_state;
+	int			id_gpio;
 
 	/* Dual role support */
 	bool				role_reversal_supported;
@@ -280,6 +281,15 @@ static int qpnp_typec_configure_ssmux(struct qpnp_typec_chip *chip,
 					return rc;
 				}
 			}
+
+			if (chip->id_gpio) {
+				rc = gpio_direction_input(chip->id_gpio);
+				if (rc) {
+					pr_err("failed to configure usb id gpio rc=%d\n",
+							rc);
+					return rc;
+				}
+			}
 			break;
 		case CC_1:
 		case CC_2:
@@ -526,6 +536,8 @@ static irqreturn_t dfp_detect_handler(int irq, void *_chip)
 	pr_debug("UFP status reg = 0x%x DFP status reg = 0x%x\n",
 			reg[0], reg[1]);
 
+	// pull down USB ID pin to start host
+	gpio_direction_output(chip->id_gpio, 0);
 out:
 	mutex_unlock(&chip->typec_lock);
 	return IRQ_HANDLED;
@@ -545,6 +557,8 @@ static irqreturn_t dfp_detach_handler(int irq, void *_chip)
 
 	mutex_unlock(&chip->typec_lock);
 
+	// pull up USB ID pin to start device
+	gpio_direction_output(chip->id_gpio, 1);
 	return IRQ_HANDLED;
 }
 
@@ -589,6 +603,27 @@ static int qpnp_typec_parse_dt(struct qpnp_typec_chip *chip)
 		}
 	}
 
+	/* USB ID configuration gpio */
+	if (of_find_property(node, "qcom,id-gpio", NULL)) {
+		chip->id_gpio = of_get_named_gpio(node,
+				"qcom,id-gpio", 0);
+		if (!gpio_is_valid(chip->id_gpio)) {
+			if (chip->id_gpio != -EPROBE_DEFER)
+				pr_err("failed to get usb id config gpio=%d\n",
+						chip->id_gpio);
+			return chip->id_gpio;
+		}
+
+		rc = devm_gpio_request(chip->dev, chip->id_gpio,
+				"typec_usb_id_gpio");
+		if (rc) {
+			pr_err("failed to request usb id gpio rc=%d\n", rc);
+			chip->id_gpio = 0;
+		} else {
+			gpio_direction_input(chip->id_gpio);
+		}
+	}
+
 	/* SS-Mux regulator */
 	if (of_find_property(node, "ss-mux-supply", NULL)) {
 		chip->ss_mux_vreg = devm_regulator_get(chip->dev, "ss-mux");

  

标签:MSM8953,USB,适配,det,dev,irq,gpio,id,usb
From: https://www.cnblogs.com/pypyn/p/16635766.html

相关文章

  • Pclint的适配以及使用方法
    Pclint的适配以及使用方法一.简介C/C++语言的语法拥有其它语言所没有的灵活性,这种灵活性带来了代码效率的提升,但相应也使得代码编写具有很大的随意性,另外C/C++编译器......
  • 设计模式10 - 设计模式 - 适配器设计模式(高频-结构型)
    一、定义适配器模式将一个类的接口转换成客户希望的另外一个接口。适配器模式使得原本由于接口不兼容而不能在一起工作的那些类可以一起工作。    适配器模式......
  • 可编程USB转 UART/I2C /SMBusS/SPI/CAN/1 -Wire适配器USB2S 常见问题及注意事项
    河北稳控科技可编程USB转UART/I2C/SMBusS/SPI/CAN/1-Wire适配器USB2S常见问题及注意事项 (1)外接引线长度当使用导线连接外部设备或芯片时,导线不可过长,一般控制在2......
  • vue 大屏等比缩放适配方法
    大屏适配方案之一,根据设计稿尺寸,固定容器尺寸,内容完全按设计稿大小开发,可保持比例居中显示。//windowScale.jsexportfunctionuseIndex(appRef,fill){//*appRef......
  • Mac 与 iPad USB 方式传输文件 All In One
    Mac与iPadUSB-C方式传输文件AllInOneMac使用USB连接后看不到iPad选项/Mac使用数据线连接iPad无法识别solution如果您的电脑无法识别iPhone、iPad......
  • Day16-响应式布局+移动端适配
    0821:Day16响应式布局:响应式布局:响应式布局 特点: 面对不同的分辨率设备灵活性强 能够快捷解决多设备显示适应问题 缺点:媒体查询:媒体查询:根据显示器的特性,为其设......
  • 共享USB打印机设置方法
    打印机共享一、准备所有计算机在同一个网段。所有计算机在同一个工作组,组名可以自定义,默认WORKGROUP。使用超级管理员用户,目的是为了激活guest用户。验证之后可......
  • codesys作为本机ethernet/ip适配器
    摘要:codesys作为本机EIP适配器,也就是说在Codesys中自己配置变量,然后导出为EDS文件,其它设备导入这个EDS文件,就可以和codesys通过Ethernet/IP的方式通信。以欧姆龙PLCCP1H......
  • Nginx代理:通过同个域名同个端口分别在PC端和手机端访问不同的适配页面
    一、nginx配置1、传递请求头最终目的是要使用UserAgent头来识别用户的客户端,然后返回不同的内容给不同的UA用户。而CDN(内容分发网络)缓存并不会区分UA(UserAgent),只......
  • 2022-08-19 记录一下 奥睿科 2.5/3.5英寸双盘位USB3.0硬盘底座 使用感受
    什么?电脑识别不了硬盘???我把京东客服给骂了,再到我写这个随笔的时候,有点心疼那个京东客服。为了扩容,昨天入手了希捷的2t机械家用盘,以及这次的主角奥睿科硬盘底座,简称硬盘盒......