一般bq25700等充电芯片可以通过读取状态寄存器的值判断当前AC接入的状态,若ACOK引脚电平状态为高电平,寄存器AC插入状态位的值为1,反之则为0。驱动中先读取寄存器状态判断当前适配器的插入状态,再设置ACOK中断的有效触发电平。如下代码:
static irqreturn_t bq25700_irq_handler_thread(int irq, void *private)
{
struct bq25700_device *charger = private;
int irq_flag;
struct bq25700_state state;
if (bq25700_field_read(charger, AC_STAT)) {
irq_flag = IRQF_TRIGGER_LOW;
charger->typec0_status = USB_STATUS_AC;
} else {
irq_flag = IRQF_TRIGGER_HIGH;
bq25700_disable_charge(charger);
bq25700_get_chip_state(charger, &state);
charger->state = state;
power_supply_changed(charger->supply_charger);
charger->typec0_status = USB_STATUS_NONE;
charger->typec1_status = USB_STATUS_NONE;
}
irq_set_irq_type(irq, irq_flag | IRQF_ONESHOT);
return IRQ_HANDLED;
}
当读取AC_STAT为1时说明适配器处于插入状态,此时ACOK引脚为高电平,所以设置下次中断触发的有效电平为低电平。
如果硬件上对ACOK的电平状态做了反向设计(比如增加了MOS),那么读取AC_STAT为1时适配器处于插入状态,但CPU端的ACOK对应GPIO为低电平,此时驱动中仍设置下次中断触发的有效电平为低电平就会出现中断不停触发的现象,且代码只会一直跑中断函数中的if和else中其中1个。
解决方法很简单,将驱动代码中的下次有效触发电平反向就可以了。改成如下代码:
if (bq25700_field_read(charger, AC_STAT)) {
irq_flag = IRQF_TRIGGER_HIGH;
charger->typec0_status = USB_STATUS_AC;
} else {
irq_flag = IRQF_TRIGGER_LOW;
bq25700_disable_charge(charger);
bq25700_get_chip_state(charger, &state);
charger->state = state;
power_supply_changed(charger->supply_charger);
charger->typec0_status = USB_STATUS_NONE;
charger->typec1_status = USB_STATUS_NONE;
}
若中断不停上报但在中断函数的if和else两段代码中切换,那么很可能是ACOK引脚电平状态异常,处于一种反复高低变化的状态,可以通过示波器抓取ACOK电平状态判断,若不停高低变化,很可能触发了充电芯片的某种保护机制,需电源工程师进一步分析排查。