blob: 5553e3985d42bcb4bbc81943b53aa053ecd999e4 [file] [log] [blame]
/* main.c */
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <dirent.h>
#include "porting.h"
#define LOG_TAG "USBPower"
#include "usbctrl.h"
#define TOLOWER(x) ((x) | 0x20)
//driver support 0-kernel2.6 1-kernel3.0(before M6) 2-kernel3.0(M6&later)
#define DWC_DRIVER_1 0
#define DWC_DRIVER_2 1
#define DWC_DRIVER_3 2
#define DWC_DRIVER_4 3
#define DWC_DRIVER_MAX 4
#define DWC_DRIVER_VERSION_1 "2.60a"
#define DWC_DRIVER_VERSION_2 "2.20a"
#define DWC_DRIVER_VERSION_3 "2.94a"
#define DWC_DRIVER_VERSION_4 "3.10a"
#define DWC_DRIVER_VERSION_LEN 5
#define DWC_OTG_VERSION_DIR "/sys/bus/logicmodule/drivers/dwc_otg/version"
#define IDX_ATTR_FILENAME "/sys/devices/platform/usb_phy_control/index"
#define POWER_ATTR_FILENAME_1 "/sys/devices/platform/usb_phy_control/por"
#define POWER_ATTR_FILENAME_2 "/sys/devices/lm0/peri_sleepm"
#define POWER_ATTR_FILENAME_3 "/sys/devices/lm0/peri_sleepm"
#define POWER_ATTR_FILENAME_4 "/sys/devices/lm0/peri_sleepm"
//#define POWER_ATTR_FILENAME_2 "/sys/devices/lm0/peri_power"
//#define POWER_ATTR_FILENAME_3 "/sys/devices/lm0/peri_power"
#define OTG_DISABLE_FILE_NAME_1 "/sys/devices/platform/usb_phy_control/otgdisable"
#define OTG_DISABLE_FILE_NAME_2 "/sys/devices/lm0/peri_otg_disable"
#define OTG_DISABLE_FILE_NAME_3 "/sys/devices/lm0/peri_otg_disable"
#define OTG_DISABLE_FILE_NAME_4 "/sys/devices/lm0/peri_otg_disable"
#define CONNECT_STR "Bus Connected = 0x"
#define CONNECT_FILE_NAME "/sys/devices/lm0/busconnected"
#define PULLUP_FILE_NAME "/sys/devices/lm0/pullup"
#define GOTGCTL_STR "GOTGCTL = 0x"
#define GOTGCTL_FILE_NAME "/sys/devices/lm0/gotgctl"
#define USB_GATE_OFF_FILE_NAME "/sys/class/aml_mod/mod_off"
#define USB_GATE_ON_FILE_NAME "/sys/class/aml_mod/mod_on"
#define USB_ID 16
#define USB_ID_HOST 0x0
#define USB_ID_DEVICE 0x1
#define USB_SES 18
#define USB_SES_VALID 0x3
int dwc_driver_version=-1; //if dwc_driver_version == -1(not support) 0--2(driver version)
char dwc_driver_version_str[DWC_DRIVER_MAX][DWC_DRIVER_VERSION_LEN+1] =
{
DWC_DRIVER_VERSION_1,
DWC_DRIVER_VERSION_2,
DWC_DRIVER_VERSION_3,
DWC_DRIVER_VERSION_4
};
char power_attr_filename_str[DWC_DRIVER_MAX][64] =
{
POWER_ATTR_FILENAME_1,
POWER_ATTR_FILENAME_2,
POWER_ATTR_FILENAME_3,
POWER_ATTR_FILENAME_4
};
char otg_disable_filename_str[DWC_DRIVER_MAX][64]=
{
OTG_DISABLE_FILE_NAME_1,
OTG_DISABLE_FILE_NAME_2,
OTG_DISABLE_FILE_NAME_3,
OTG_DISABLE_FILE_NAME_4
};
char usb_index_str[USB_IDX_MAX][2]=
{
"A","B"
};
char usb_state_str[USB_CMD_MAX][8]=
{
"on","off","if"
};
char usb_state_val[USB_CMD_MAX][2]=
{
"0","1","2"
};
char pullup_filename_str[32];
char usb_gate_str[8]={"usb0"};
static void usage(void)
{
printf("usbpower USAGE:\n");
printf("usbpower <portindex> <cmd> \n");
printf(" index: A or B ; state: on/off/if(find device exist)\n");
printf("for example: usbpower A on\n");
exit(1);
}
static int get_device_if(int idx)
{
int ret = 0;
int err = 0;
char line[32];
char filename[32];
FILE *fp;
unsigned int busconnect,gotgctl;
strcpy(filename,GOTGCTL_FILE_NAME);
filename[15] = idx + '0';
if((fp = fopen(filename,"r"))) {
if (fgets(line, 32, fp)) {
if (!strncmp(line, GOTGCTL_STR, strlen(GOTGCTL_STR))) {
sscanf(line+strlen(GOTGCTL_STR),"%x",&gotgctl);
}
else
{
SLOGE("gotgctl txt is error\n");
err =1;
}
} else {
SLOGE("Failed to read gotgctl\n");
err=2;
}
fclose(fp);
} else {
//SLOGW("No usb device\n");
err=3;
}
if(err)
return 0;
strcpy(filename,CONNECT_FILE_NAME);
filename[15] = idx + '0';
if((fp = fopen(filename,"r"))) {
if (fgets(line, 32, fp)) {
if (!strncmp(line, CONNECT_STR, strlen(CONNECT_STR))) {
sscanf(line+strlen(CONNECT_STR),"%x",&busconnect);
}else{
SLOGE("busconnected txt is error\n");
err=4;
}
} else {
SLOGE("Failed to read busconnected\n");
err=5;
}
fclose(fp);
} else {
//SLOGW("No usb device\n");
err=6;
}
if(err)
return 0;
if(((gotgctl>>USB_ID)&0x1)== USB_ID_HOST)
{
if(idx==0)
{
if(((gotgctl>>USB_SES)&0x3)== USB_SES_VALID)
{
// this case,check busconnected
if(busconnect==1)
ret = 1;
}
else
{
//this case ,can't power of,so we look it as connect
ret = 1;
}
}
else
{
if(busconnect==1)
ret = 1;
}
}
else
{
// this case,check gotgctl USB_SES_VALID
if(((gotgctl>>USB_SES)&0x3)== USB_SES_VALID)
ret = 1;
}
return ret;
}
static int set_power_ctl(int idx,int cmd)
{
int ret = 0;
int err = 0;
FILE *fp,*fpp = NULL,*fpo = NULL, *fp_gotgctl,*fp_gate;
unsigned int gotgctl;
char filename[32];
char line[32];
int version = dwc_driver_version;
if(cmd == USB_CMD_OFF)
{
ret = get_device_if(idx);
if(ret == 1)
{
printf("Has devices on this port,can't power off!\n");
return ret;
}
}
strcpy(filename,GOTGCTL_FILE_NAME);
filename[15] = idx + '0';
if((fp_gotgctl = fopen(filename,"r"))) {
if (fgets(line, 32, fp_gotgctl)) {
if (!strncmp(line, GOTGCTL_STR, strlen(GOTGCTL_STR))) {
sscanf(line+strlen(GOTGCTL_STR),"%x",&gotgctl);
}
else
{
SLOGE("gotgctl txt is error\n");
err =1;
}
} else {
SLOGE("Failed to read gotgctl\n");
err=2;
}
fclose(fp_gotgctl);
} else {
//SLOGW("No usb device\n");
err=3;
}
if(err)
return -1;
if(version == DWC_DRIVER_1)
{
if((fp = fopen(IDX_ATTR_FILENAME,"w"))){
fwrite(usb_index_str[idx], 1, strlen(usb_index_str[idx]),fp);
fclose(fp);
}
else
{
ret = -1;
}
if(ret == 0)
{
if(((gotgctl>>USB_ID)&0x1)== USB_ID_HOST)
{
if((fp = fopen(power_attr_filename_str[version],"w"))){
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fp); //power
}
else
{
ret = -2;
}
if(fp) fclose(fp);
}
else
{
strcpy(pullup_filename_str,PULLUP_FILE_NAME);
pullup_filename_str[15] = idx + '0';
if((fp = fopen(power_attr_filename_str[version],"w")) && (fpp = fopen(pullup_filename_str,"w"))
&& (fpo = fopen(otg_disable_filename_str[version],"w"))){
if(cmd == USB_CMD_OFF)
{
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fpo); //otg
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fpp); //pullup
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fp); //power
}
else
{
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fp); //power
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fpo); //otg
fwrite(usb_state_str[cmd], 1, strlen(usb_state_str[cmd]),fpp); //pullup
}
}
else
{
ret = -2;
}
if(fp) fclose(fp);
if(fpp) fclose(fpp);
if(fpo) fclose(fpo);
}
}
}
else
{
power_attr_filename_str[version][15] = idx + '0';
otg_disable_filename_str[version][15]= idx + '0';
usb_gate_str[3] = idx + '0';
if(((gotgctl>>USB_ID)&0x1)== USB_ID_HOST)
{
if((fp = fopen(power_attr_filename_str[version],"w"))){
if(cmd == USB_CMD_OFF)
{
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fp); //power
fp_gate = fopen(USB_GATE_OFF_FILE_NAME,"w");
if(fp_gate){
fwrite(usb_gate_str,1,strlen(usb_gate_str),fp_gate);
fclose(fp_gate);
}
}
else
{
fp_gate = fopen(USB_GATE_ON_FILE_NAME,"w");
if(fp_gate){
fwrite(usb_gate_str,1,strlen(usb_gate_str),fp_gate);
fclose(fp_gate);
}
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fp); //power
}
}
else
{
ret = -2;
}
if(fp) fclose(fp);
}
else
{
if((fp = fopen(power_attr_filename_str[version],"w"))
&& (fpo = fopen(otg_disable_filename_str[version],"w"))){
if(cmd == USB_CMD_OFF)
{
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fpo); //otg
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fp); //power
fp_gate = fopen(USB_GATE_OFF_FILE_NAME,"w");
if(fp_gate){
fwrite(usb_gate_str,1,strlen(usb_gate_str),fp_gate);
fclose(fp_gate);
}
}
else
{
fp_gate = fopen(USB_GATE_ON_FILE_NAME,"w");
if(fp_gate){
fwrite(usb_gate_str,1,strlen(usb_gate_str),fp_gate);
fclose(fp_gate);
}
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fp); //power
fwrite(usb_state_val[cmd], 1, strlen(usb_state_val[cmd]),fpo); //otg
}
}
else
{
ret = -2;
}
if(fp) fclose(fp);
if(fpo) fclose(fpo);
}
}
return ret;
}
char ver_buf[DWC_DRIVER_VERSION_LEN + 1];
int get_dwc_driver_version(void)
{
FILE *fpv;
int i;
dwc_driver_version = -1;
fpv = fopen(DWC_OTG_VERSION_DIR,"r");
if(fpv){
if (fread(ver_buf,1,DWC_DRIVER_VERSION_LEN + 1,fpv) == DWC_DRIVER_VERSION_LEN + 1 ) {
for (i=DWC_DRIVER_1; i<DWC_DRIVER_MAX; i++) {
if(strncmp(ver_buf,dwc_driver_version_str[i],DWC_DRIVER_VERSION_LEN)==0){
dwc_driver_version = i;
break;
}
}
}
fclose(fpv);
}
if(dwc_driver_version==-1){
//used other process for the version different from DWC_DRIVER_VERSION
return -1;
}
if(dwc_driver_version == DWC_DRIVER_2)
{
//now we have not supported this version.It will be supported later.
return -1;
}
return 0;
}
int usbpower(int index,int cmd)
{
int ret=0;
if((cmd==USB_CMD_ON)||(cmd==USB_CMD_OFF))
{
ret = set_power_ctl(index,cmd);
}
else if(cmd == USB_CMD_IF)
{
ret = get_device_if(index);
#if 0
if(ret == 1 )
printf("Has Device\n");
else
printf("No device\n");
#endif
}
return ret;
}