Fork me on GitHub

Android Qcom USB Driver学习(十二)

keypad 在suspend的过程中导致Android无法进入suspend的问题,导致整体功耗过高,其实是主机都没有进入睡眠,通过以下打log的方式最终定位到问题,pmic vbus输出的时候会有reverse boost所以pmic侧的功耗也会有增加,当pmic电量低的时候会触发uvlo导致vbus上的电断开,需要需要的话,我们可以降低这个uvlo的触发阈值,是得对应的固件能在低电量使用

usb_enable_autosuspend -> pm_runtime_allow -> rpm_idle -> rpm_suspend -> rpm_callback

.runtime_suspend =      usb_runtime_suspend 入口

usb_runtime_suspend -> autosuspend_check
                    -> usb_suspend_both <return error>
                -> usb_suspend_interface<return error>
                    -> driver->suspend(intf, msg)<return error>
                        -> hid_suspend
                            -> hidinput_count_leds
                            -> if(!test_bit(HID_RESET_PENDING, &usbhid->iofl)
                                        && !test_bit(HID_CLEAR_HALT, &usbhid->iofl)
                                         && !test_bit(HID_OUT_RUNNING, &usbhid->iofl)
                                        && !test_bit(HID_CTRL_RUNNING, &usbhid->iofl)
                                         && !test_bit(HID_KEYS_PRESSED, &usbhid->iofl)
                                         && (!ledcount || ignoreled ))<return error>
                            -> driver_suspend = true

                -> usb_suspend_device
                    -> udriver->suspend(udev, msg)
                        -> generic_suspend
                            -> usb_port_suspend
                                -> usb_enable_remote_wakeup
                                -> usb_disable_usb2_hardware_lpm
                                -> usb_disable_ltm
                                -> set_port_feature(USB_PORT_FEAT_SUSPEND)
                                    -> usb_control_msg(hdev, usb_sndctrlpipe(hdev, 0),
                                                USB_REQ_SET_FEATURE, USB_RT_PORT, feature, port1,
                                                NULL, 0, 1000);
                                -> usb_set_device_state(USB_STATE_SUSPENDED)
                                -> pm_runtime_put_sync

问题分析下来确定在usb_suspend_both 中的ledcount中,ignoreled是Module params中得到

usb suspend/resume是通过setportfeature命令控制的,linux usb驱动也是通过这种port command来控制suspend以及resume状态,如果让一个要让一个设备suspend,则想向其所在的 hub port发送USB_PORT_FEAT_SUSPEND即可让设备进入suspend状态,此时基本主机进入suspend状态,而功耗依旧是高的状态基本可以断掉是设备没有进入suspend状态,那基本可以定位为usb外部设备suspend failed,对于roothub,它没有所在的port,所以没有对应的 setportfeatre的方法,那它使用的是系统的全局调用的bus_suspend,除了以上遇到的led report的问题,还需要注意remote wakeup的功能也会是usb_suspend_both发生failedAndroid

一种方法就是过滤出我们的设备,使其走入正常的suspend流程,另一方方法就是将ignoreled设置true

bool isKeypad(struct hid_device *hid,struct usb_interface *intf){
    if(hid->vendor == 0xXXXX && hid->product == 0xXXXX)
        return true;
    return false;
}

usbhid_probe
    ifisKeypad(hid, intf)){
        usb_enable_autosuspend(dev);
        hid_err(intf, "enable autosuspend++++++\n");
    }

usbhid_disconnect
    ifisKeypad(hid, intf)){
        usb_disable_autosuspend(dev);
        hid_err(intf, "disable autosuspend++++++\n");
    }

hid_suspend
-    && (!ledcount || ignoreled ))
+    && (!ledcount || ignoreled || isCW45Keypad(hid, intf)))

Firmware

HID 报告及报告描述简介这篇文章中可以得知键盘的报告描述符和报告中usage有led相关的,可能是让Android能支持对keypad led的处理,如果不对Android处理,将keypad firmware中的ReportDescriptor中这一段删除或者修改某些值可能就OK了,我猜的,固件代码没分析过

code char KeyBoardReportDescriptor[63] = {
	0x05, 0x01, // USAGE_PAGE (Generic Desktop)  //表示用途页为通用桌面设备
	0x09, 0x06, // USAGE (Keyboard) //表示用途为键盘
	0xa1, 0x01, // COLLECTION (Application)
	...
-	0x05, 0x08, // USAGE_PAGE (LEDs) //用途是 LED,即用来控制键盘上的 LED 用的,因此下面会说明它是输出用
-	0x19, 0x01, // USAGE_MINIMUM (Num Lock) //用途最小值是 Num Lock,即数字键锁定灯
-	0x29, 0x05, // USAGE_MAXIMUM (Kana) //用途最大值是 Kana,这个是什么灯我也不清楚
-	0x91, 0x02, // OUTPUT (Data,Var,Abs) //1 表示灯亮, 0 表示灯灭
-	0x95, 0x01, // REPORT_COUNT (1) //这样的数据段个数为 1
-	0x75, 0x03, // REPORT_SIZE (3) //每个段大小为 3bits
-	0x91, 0x03, // OUTPUT (Cnst,Var,Abs) 输出用,常量,值,绝对 对上面的5字节进行补齐
	...
	0xc0 // END_COLLECTION
};
posted @ 2024-08-29 17:57  yooooooo  阅读(11)  评论(0编辑  收藏  举报