文章目录
一、问题描述
一个外设的供电由SOC的gpio控制,高电平使能供电。原来的控制方式是上电后直接使能供电。
PD协商时,供电会从5v跃迁到9v,导致外设的供电存在一个明显的上电台阶
二、解决思路
延时700ms之后再使能供电,避开电压波动期
市面上大部分pd适配器协商时间大概在500~600ms,在此基础上再预留100ms
三、实现方案
bash
// 查看每个cable的名字
# for d in /sys/class/extcon/extcon0/cable.*; do echo "$d: $(cat $d/name)"; done
/sys/class/extcon/extcon0/cable.0: USB
/sys/class/extcon/extcon0/cable.1: USB-HOST
/sys/class/extcon/extcon0/cable.2: USB_VBUS_EN
/sys/class/extcon/extcon0/cable.3: SDP
/sys/class/extcon/extcon0/cable.4: CDP
/sys/class/extcon/extcon0/cable.5: DCP
/sys/class/extcon/extcon0/cable.6: SLOW-CHARGER
# cat /sys/class/extcon/extcon0/state
USB=1
USB-HOST=0
USB_VBUS_EN=0
SDP=0
CDP=0
DCP=0
SLOW-CHARGER=0
可知extcon0已正确上报USB插入,USB=1 ---> VBUS检测正常
实现一个简单的驱动程序:
Extcon事件处理函数:通过cable.0判断USB是否插入,触发延迟工作队列
延迟工作函数中实现开启电源的操作
此外,当usb口同时也作为供电口时,此时extcon0已经再kernel早期就上报了USB=1的事件,而驱动在later init阶段才加载probe,这时候extcon状态对于驱动来说"没有变化",notifier将永远不会被触发
因此在probe函数时就不能只依赖notifier变化事件,而需要主动读取当前状态并处理
四、部分代码实现
c
static int usb_a_power_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct usb_a_power_data *data;
bool attached;
int ret;
data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
if (!data)
return -ENOMEM;
data->dev = dev;
data->delay_ms = 700;
of_property_read_u32(dev->of_node, "delay-ms", &data->delay_ms);
data->enable_gpio = devm_gpiod_get(dev, "enable", GPIOD_OUT_LOW);
if (IS_ERR(data->enable_gpio))
return PTR_ERR(data->enable_gpio);
data->edev = extcon_get_edev_by_phandle(dev, 0);
if (IS_ERR(data->edev))
return PTR_ERR(data->edev);
INIT_DELAYED_WORK(&data->enable_work, usb_a_enable_work);
data->extcon_nb.notifier_call = usb_a_extcon_handler;
ret = extcon_register_notifier(data->edev, 0, &data->extcon_nb);
if (ret)
return ret;
//处理开机时已插入的情况
attached = extcon_get_state(data->edev, 0);
dev_info(dev, "USB initial state: %s\n", attached ? "attached" : "detached");
if (attached) {
dev_info(dev, "Scheduling USB-A power-on after %u ms\n", data->delay_ms);
schedule_delayed_work(&data->enable_work,
msecs_to_jiffies(data->delay_ms));
}
platform_set_drvdata(pdev, data);
dev_info(dev, "USB-A power driver probed (delay=%u ms)\n", data->delay_ms);
return 0;
}