国产一级a片免费看高清,亚洲熟女中文字幕在线视频,黄三级高清在线播放,免费黄色视频在线看

打開APP
userphoto
未登錄

開通VIP,暢享免費電子書等14項超值服

開通VIP
android 電池(二):android關(guān)機充電流程、充電畫面顯示

文章來自:http://blog.csdn.net/wh_19910525/article/details/17260861

上一篇我們講了鋰電池的充放電的流程和電池的一些特性,這一節(jié)我們重點說一下android關(guān)機充電是怎么、充電畫面顯示是怎么實現(xiàn)的,這個在工作中也比較有用,我們開始做這一塊的時候也走了不少的彎路。我記得我們做adnroid2.3的時候,關(guān)機狀態(tài)和充電logo顯示是在uboot中做的。應(yīng)該是有兩種做法,回頭我再看下uboot中做畫面顯示那一塊是怎么做的,這一節(jié)我們重點說系統(tǒng)中的充電logo顯示。


一、android正常開機流程、關(guān)機充電流程

在寫這篇文章之前我們先看兩個流程:正常開機流程,關(guān)機充電系統(tǒng)啟動流程

1、正常開機流程,按開機鍵。

可大致分成三部分

(1)、OS_level:UBOOT、kenrel、init這三步完成系統(tǒng)啟動;

(2)、Android_level:這部分完成android部的初始化;

(3)、Home Screen:這部分就是我們看到的launcher部分。


2、關(guān)機充電系統(tǒng)啟動流程


       與前面相比,這個流程只走到init這一部分,就沒有往后走了,這部分我們會在后面的代碼中分析。



==========================

二、關(guān)機充電邏輯硬件邏輯


1、插入DC,charger IC從硬件上喚醒系統(tǒng),相當于長按開機鍵開機。


下面這部分是charger IC連接系統(tǒng)的控制部分。



=========================


三、軟件邏輯。

DC插入,其實相當于關(guān)機狀態(tài)下“按開機鍵”開機。第一步要走UBOOT、kernel 、android init這一流程。

1、UBOOT

       UBOOT啟動代碼我們不在這里詳細分析,這里我們只要注意二個問題:

a:如何判斷是DC插入;

b:設(shè)定setenv("bootargs", "androidboot.mode=charger"),androidboot.mode這個參數(shù)相當重要,這個參數(shù)決定系統(tǒng)是正常啟動、還是關(guān)機充電狀態(tài)。

Uboot/board/samsung/smdk4212/smkd4212.c


  1. int board_late_init (void)  
  2. {  
  3.     int keystate = 0;  
  4.     printf("check start mode\n");  
  5.   if ((*(int *)0x10020800==0x19721212) || (*(int *)0x10020804==0x19721212)  
  6. || (*(int *)0x10020808==0x19721212)) //(1)、檢查是否有DC插入;  
  7. {  
  8.     setenv ("bootargs", "");//(2)、沒有DC插入;  
  9.   } else  {//DC插入  
  10.         int tmp=*(int *)0x11000c08;  
  11.     *(int *)0x10020800=*(int *)0x10020804=0x19721212;  
  12.     *(int *)0x11000c08=(tmp&(~0xc000))|0xc000;  
  13.     udelay(10000);  
  14.     if ((*(int *)0x11000c04 & 0x80)!=0x80 && INF_REG4_REG != 0xf) {  
  15.         setenv ("bootargs", "androidboot.mode=charger");//(3)、設(shè)定bootargs為charger狀態(tài)  
  16.         printf("charger mode\n");  
  17.     } else {  
  18.         setenv ("bootargs", "");  
  19.     }  
  20.     *(int *)0x11000c08=tmp;  
  21.   }  
  22. #ifdef CONFIG_CPU_EXYNOS4X12  
  23.     int charge_status=CheckBatteryLow();//(4)、檢查電池電量,如果檢測電量太低,則不開機  
  24.     keystate=board_key_check();//(5)、檢查按鍵狀態(tài);  
  25.     // fuse bootloader  
  26.     if(second_boot_info != 0) {  
  27.         boot_symbol=1;  
  28.         INF_REG2_REG =0x8;  
  29.         run_command(CONFIG_BOOTCMD_FUSE_BOOTLOADER, NULL);  
  30.     }  
  31.     if((INF_REG4_REG == 0xd)) {  
  32.         // reboot default  
  33.         char buf[10];  
  34.         sprintf(buf, "%d", CONFIG_BOOTDELAY);  
  35.         setenv ("bootdelay", buf);  
  36.         setenv ("reserved", NULL);  
  37.         saveenv();  
  38.     } else if((INF_REG4_REG == 0xe) || keystate == (0x1 | 0x2)) {//(6)、按鍵進入fastboot模式;  
  39.         // reboot bootloader  
  40.         boot_symbol=1;  
  41.         INF_REG2_REG =0x8;  
  42.         printf("BOOTLOADER - FASTBOOT\n");  
  43.         setenv ("reserved", "fastboot");  
  44.         setenv ("bootdelay", "0");  
  45.     } else if((INF_REG4_REG == 0xf) || keystate == (0x1 | 0x2 | 0x4)) {//(7)、按鍵進入recovery模式;  
  46.         // reboot recovery  
  47.         printf("BOOTLOADER - RECOVERY\n");  
  48.         boot_symbol=1;  
  49.         INF_REG2_REG =0x8;  
  50.         setenv ("reserved", CONFIG_BOOTCMD_RECOVERY);  
  51.         setenv ("bootdelay", "0");  
  52.     } else  
  53.     if(keystate == (0x1 | 0x4) || second_boot_info != 0 || partition_check()) {//(8)、按鍵進入卡升級模式;  
  54.         // 2nd boot  
  55.         printf("BOOTLOADER - 2ND BOOT DEVICE\n");  
  56.         boot_symbol=1;  
  57.         INF_REG2_REG =0x8;  
  58.         setenv ("bootcmd", CONFIG_BOOTCOMMAND);  
  59.         setenv ("reserved", CONFIG_BOOTCMD_FUSE_RELEASE);  
  60.         setenv ("bootdelay", "0");  
  61.     } else {//(9)、正常啟動;  
  62.         // normal case  
  63.         char buf[10];  
  64.         sprintf(buf, "%d", CONFIG_BOOTDELAY);  
  65.         setenv ("bootdelay", buf);  
  66.     }  
  67.     INF_REG4_REG = 0;  
  68.     return 0;  


(1)、檢查是否有DC插入;

  1. if ((*(int *)0x10020800==0x19721212) || (*(int *)0x10020804==0x19721212)  
  2.  (*(int *)0x10020808==0x19721212))   

這部分檢查寄存器的值。

(2)、沒有DC插入;

(3)、設(shè)定bootargs為charger狀態(tài)


  1. if ((*(int *)0x11000c04 & 0x80)!=0x80 && INF_REG4_REG != 0xf) {  
  2.         setenv ("bootargs", "androidboot.mode=charger");  

這是這部分的重點,如果能過寄存器判斷是DC插入,把androidboot.mode設(shè)定為charger狀態(tài)。

以下這部分根據(jù)需要加入,通過判斷不同的情況進入不同的功能,如fastboot\revovery…………,這部分不做詳細解釋。

(4)、檢查電池電量;

    這個在正常開機狀態(tài)下,如果檢測電量太低,則不開機,這部分代碼就不做分析。

(5)、檢查按鍵狀態(tài);

      我們這個平臺有幾種模式:fastboot\recovery\卡升級等……

(6)、按鍵進入fastboot模式;

(7)、按鍵進入recovery模式;

(8)、按鍵進入卡升級模式

(9)、正常啟動;


2、kernel

這部分和正常啟動是一樣的。


3、init

前面所有的描述其實只有一點和正常啟動不太一樣,那就是在UBOOT中把androidboot.mode設(shè)定為charger狀態(tài),內(nèi)核正常啟動,然后到init時要對charger這種狀態(tài)處理


system\core\init\init.c


  1. int main(int argc, char **argv)  
  2. {  
  3.     ………………  
  4.     action_for_each_trigger("early-init", action_add_queue_tail);  
  5.   
  6.     queue_builtin_action(wait_for_coldboot_done_action, "wait_for_coldboot_done");  
  7.     queue_builtin_action(property_init_action, "property_init");  
  8.     queue_builtin_action(keychord_init_action, "keychord_init");  
  9.     queue_builtin_action(console_init_action, "console_init");  //(1)、顯示initlogo.rle,也就是android第二張圖片;  
  10.     queue_builtin_action(set_init_properties_action, "set_init_properties");  
  11.   
  12.     /* execute all the boot actions to get us started */  
  13.     action_for_each_trigger("init", action_add_queue_tail);  
  14.   
  15.     /* skip mounting filesystems in charger mode */  
  16.     if (strcmp(bootmode, "charger") != 0) {//(2)、這里就是UBOOT中設(shè)定的bootmode,如果是charger模式,跳過下面初始化;  
  17.         action_for_each_trigger("early-fs", action_add_queue_tail);  
  18.         action_for_each_trigger("fs", action_add_queue_tail);  
  19.         action_for_each_trigger("post-fs", action_add_queue_tail);  
  20.         action_for_each_trigger("post-fs-data", action_add_queue_tail);  
  21.     }  
  22.   
  23.     queue_builtin_action(property_service_init_action, "property_service_init");  
  24.     queue_builtin_action(signal_init_action, "signal_init");  
  25.     queue_builtin_action(check_startup_action, "check_startup");  
  26.   
  27.     if (!strcmp(bootmode, "charger")) {//(3)、如果為charger,則調(diào)用charger.c。  
  28.         action_for_each_trigger("charger", action_add_queue_tail);  
  29.     } else {  
  30.         action_for_each_trigger("early-boot", action_add_queue_tail);  
  31.         action_for_each_trigger("boot", action_add_queue_tail);  
  32.     }  
  33. ……………………  
  34. }  

(1)、顯示initlogo.rle,也就是android第二張圖片;

queue_builtin_action(console_init_action,"console_init");調(diào)用console_init_action

  1. static int console_init_action(int nargs, char **args)  
  2. {  
  3.     int fd;  
  4.     char tmp[PROP_VALUE_MAX];  
  5.     if (console[0]) {  
  6.         snprintf(tmp, sizeof(tmp), "/dev/%s", console);  
  7.         console_name = strdup(tmp);  
  8.     }  
  9.     fd = open(console_name, O_RDWR);  
  10.     if (fd >= 0)  
  11.         have_console = 1;  
  12.     close(fd);  
  13.     if( load_565rle_image(INIT_IMAGE_FILE) ) {//這里定義rle文件的名稱#define INIT_IMAGE_FILE "/initlogo.rle"  
  14.         fd = open("/dev/tty0", O_WRONLY);  
  15.         if (fd >= 0) {//如果沒有這張圖片,就顯示android字樣,在屏幕左上角;  
  16.             const char *msg;  
  17.                 msg = "\n"  
  18.             "\n"  
  19.             "\n"  // console is 40 cols x 30 lines  
  20.             "\n"  
  21.             "\n"  
  22.             "\n"  
  23.             "\n"  
  24.             "\n"  
  25.             "\n"  
  26.             "\n"  
  27.             "             A N D R O I D ";  
  28.             write(fd, msg, strlen(msg));  
  29.             close(fd);  
  30.         }  
  31.     }  
  32.     return 0;  
  33. }  

(2)、這里就是UBOOT中設(shè)定的bootmode,如果是charger模式,跳過下面初始化;

  1. /* skip mounting filesystems in charger mode */  
  2. if (strcmp(bootmode, "charger") != 0) {  
  3.     action_for_each_trigger("early-fs", action_add_queue_tail);  
  4.     action_for_each_trigger("fs", action_add_queue_tail);  
  5.     action_for_each_trigger("post-fs", action_add_queue_tail);  
  6.     action_for_each_trigger("post-fs-data", action_add_queue_tail);  
  7. }  

(3)、如果為charger,則調(diào)用charger.c

  1. action_for_each_trigger("charger", action_add_queue_tail);  

我們在后面細分charger這部分。


4、charger.c

這部分就是我們充電部分,充電畫面顯示的實現(xiàn)。


system\core\charger\charger.c


  1. int main(int argc, char **argv)  
  2. {  
  3. ………………  
  4.     klog_set_level(CHARGER_KLOG_LEVEL);  
  5.     dump_last_kmsg();  
  6.     LOGI("--------------- STARTING CHARGER MODE ---------------\n");  
  7.   
  8.     gr_init();  
  9.     gr_font_size(&char_width, &char_height); //(1)、初始化graphics,包括buf大小;  
  10.   
  11.     ev_init(input_callback, charger);//(2)初始化按鍵;  
  12.      
  13. fd = uevent_open_socket(64*1024, true);  
  14.     if (fd >= 0) {  
  15.         fcntl(fd, F_SETFL, O_NONBLOCK);  
  16.         ev_add_fd(fd, uevent_callback, charger);  
  17.     }  
  18.   
  19.     charger->uevent_fd = fd;  
  20.     coldboot(charger, "/sys/class/power_supply", "add");//(3)、創(chuàng)建/sys/class/power_supply結(jié)點,把socket信息通知應(yīng)用層;  
  21.       
  22. ret = res_create_surface("charger/battery_fail", &charger->surf_unknown);  
  23.     if (ret < 0) {  
  24.         LOGE("Cannot load image\n");  
  25.         charger->surf_unknown = NULL;  
  26.     }  
  27.     for (i = 0; i < charger->batt_anim->num_frames; i++) {//(4)、這里是顯示charger logo,res_create_surface顯示圖片函數(shù);  
  28.         struct frame *frame = &charger->batt_anim->frames[i];  
  29.         ret = res_create_surface(frame->name, &frame->surface);  
  30.         if (ret < 0) {  
  31.             LOGE("Cannot load image %s\n", frame->name);  
  32.             /* TODO: free the already allocated surfaces... */  
  33.             charger->batt_anim->num_frames = 0;  
  34.             charger->batt_anim->num_cycles = 1;  
  35.             break;  
  36.         }  
  37.     }  
  38. ev_sync_key_state(set_key_callback, charger);  
  39.     gr_fb_blank(true);  
  40.   
  41.     charger->next_screen_transition = now - 1;  
  42.     charger->next_key_check = -1;  
  43.     charger->next_pwr_check = -1;  
  44.     reset_animation(charger->batt_anim);  
  45.     kick_animation(charger->batt_anim);  
  46.     event_loop(charger);//(5)、event_loop循環(huán),電池狀態(tài),檢測按鍵是否按下;  
  47.     return 0;  
  48.   
  49. }  

(1)、初始化graphics,包括buf大小

android/bootable/recovery/minui/graphics.c

gr_init():minui/graphics.c[settty0 to graphic mode, open fb0],設(shè)制tty0為圖形模式,打開fb0;

  1. int gr_init(void)  
  2. {  
  3.     gglInit(&gr_context);  
  4.     GGLContext *gl = gr_context;  
  5.     gr_init_font();  
  6.     gr_vt_fd = open("/dev/tty0", O_RDWR | O_SYNC);  
  7.     if (gr_vt_fd < 0) {  
  8.         // This is non-fatal; post-Cupcake kernels don't have tty0.  
  9.         perror("can't open /dev/tty0");  
  10.   
  11.     } else if (ioctl(gr_vt_fd, KDSETMODE, (void*) KD_GRAPHICS)) {  
  12.         // However, if we do open tty0, we expect the ioctl to work.  
  13.         perror("failed KDSETMODE to KD_GRAPHICS on tty0");  
  14.         gr_exit();  
  15.         return -1;  
  16.     }  
  17.     gr_fb_fd = get_framebuffer(gr_framebuffer);  
  18.     if (gr_fb_fd < 0) {  
  19.         gr_exit();  
  20.         return -1;  
  21.     }  
  22.     get_memory_surface(&gr_mem_surface);  
  23.     fprintf(stderr, "framebuffer: fd %d (%d x %d)\n",  
  24.             gr_fb_fd, gr_framebuffer[0].width, gr_framebuffer[0].height);  
  25.         /* start with 0 as front (displayed) and 1 as back (drawing) */  
  26.     gr_active_fb = 0;  
  27.     set_active_framebuffer(0);  
  28.     gl->colorBuffer(gl, &gr_mem_surface);  
  29.     gl->activeTexture(gl, 0);  
  30.     gl->enable(gl, GGL_BLEND);  
  31.     gl->blendFunc(gl, GGL_SRC_ALPHA, GGL_ONE_MINUS_SRC_ALPHA);  
  32.     gr_fb_blank(true);  
  33.     gr_fb_blank(false);  
  34.     return 0;  
  35.   
  36. }  

(2)android/bootable/recovery/minui/events.c

ev_init():minui/events.c[open /dev/input/event*]打開 /dev/input/event*

這部分是在,充電狀態(tài)下,按鍵操作的初始化,比如:短按顯示充電logo,長按開機,初始化代碼如下。

  1. int ev_init(ev_callback input_cb, void *data)  
  2. {  
  3.     DIR *dir;  
  4.     struct dirent *de;  
  5.     int fd;  
  6.     dir = opendir("/dev/input");//打開驅(qū)動結(jié)點;  
  7.     if(dir != 0) {  
  8.         while((de = readdir(dir))) {  
  9.             unsigned long ev_bits[BITS_TO_LONGS(EV_MAX)];  
  10. //            fprintf(stderr,"/dev/input/%s\n", de->d_name);  
  11.             if(strncmp(de->d_name,"event",5)) continue;  
  12.             fd = openat(dirfd(dir), de->d_name, O_RDONLY);  
  13.             if(fd < 0) continue;  
  14.             /* read the evbits of the input device */  
  15.             if (ioctl(fd, EVIOCGBIT(0, sizeof(ev_bits)), ev_bits) < 0) {  
  16.                 close(fd);  
  17.                 continue;  
  18.             }  
  19.             /* TODO: add ability to specify event masks. For now, just assume 
  20.              * that only EV_KEY and EV_REL event types are ever needed. */  
  21.             if (!test_bit(EV_KEY, ev_bits) && !test_bit(EV_REL, ev_bits)) {  
  22.                 close(fd);  
  23.                 continue;  
  24.             }  
  25.             ev_fds[ev_count].fd = fd;  
  26.             ev_fds[ev_count].events = POLLIN;  
  27.             ev_fdinfo[ev_count].cb = input_cb;  
  28.             ev_fdinfo[ev_count].data = data;  
  29.             ev_count++;  
  30.             ev_dev_count++;  
  31.             if(ev_dev_count == MAX_DEVICES) break;  
  32.         }  
  33.     }  
  34.     return 0;  
  35. }  

(3)、創(chuàng)建/sys/class/power_supply結(jié)點,把socket信息通知應(yīng)用層

uevent_open_socket這個函數(shù)是通過kobject_uevent的方式通知的應(yīng)用層,就是往一個socket廣播一個消息,只需要在應(yīng)用層打開socket監(jiān)聽NETLINK_KOBJECT_UEVENT組的消息,就可以收到了,主要是創(chuàng)建了socket接口獲得uevent的文件描述符,然后觸發(fā)/sys/class/power_supply目錄及其子目錄下的uevent,然后接受并創(chuàng)建設(shè)備節(jié)點,至此設(shè)備節(jié)點才算創(chuàng)建。

(4)、這里顯示charger logo,res_create_surface顯示圖片函數(shù);

res_create_surface:minui/resource.c[create surfaces for all bitmaps used later, include icons, bmps]

創(chuàng)建surface為所以的位圖,包括圖標、位圖。  這些圖片的位置為:system\core\charger\images

(5)、event_loop循環(huán),電池狀態(tài),檢測按鍵是否按下;

5、event_loop

       這個函數(shù)判斷按鍵狀態(tài),DC是否插拔。如果長按開機:執(zhí)行android_reboot(ANDROID_RB_RESTART,0, 0);如果拔出DC:執(zhí)行android_reboot(ANDROID_RB_POWEROFF,0, 0);

  1. static void event_loop(struct charger *charger)  
  2. {  
  3.     int ret;  
  4.     while (true) {  
  5.         int64_t now = curr_time_ms();//(1)、獲得當前時間;  
  6.         LOGV("[%lld] event_loop()\n", now);  
  7.         handle_input_state(charger, now);//(2)、檢查按鍵狀態(tài);  
  8.         handle_power_supply_state(charger, now);// (3)、檢查DC是否拔出;   
  9.         /* do screen update last in case any of the above want to start 
  10.          * screen transitions (animations, etc) 
  11.          */  
  12.         update_screen_state(charger, now);//(4)、對按鍵時間狀態(tài)標志位的判斷,顯示不同電量的充電logo;   
  13.         wait_next_event(charger, now);  
  14.     }  
  15. }  

(1)、獲得當前時間;

   int64_t now = curr_time_ms();

       這個時間來判斷,有沒有屏幕超時,如果超時關(guān)閉屏幕充電logo顯示。

(2)、檢查按鍵狀態(tài);

  1. static void handle_input_state(struct charger *charger, int64_t now)  
  2. {  
  3.     process_key(charger, KEY_POWER, now);  
  4.     if (charger->next_key_check != -1 && now > charger->next_key_check)  
  5.         charger->next_key_check = -1;  
  6. }  
  7. 我們再看下:process_key(charger, KEY_POWER, now);  
  8. static void process_key(struct charger *charger, int code, int64_t now)  
  9. {  
  10. ………………  
  11.     if (code == KEY_POWER) {  
  12.         if (key->down) {  
  13.             int64_t reboot_timeout = key->timestamp + POWER_ON_KEY_TIME;  
  14.             if (now >= reboot_timeout) {//如果長按power鍵,就重新啟動,也就是重啟開機;  
  15.                 LOGI("[%lld] rebooting\n", now);  
  16.                 android_reboot(ANDROID_RB_RESTART, 0, 0);//重啟命令;  
  17.             }  
  18.     ………………  
  19.     }  
  20.   
  21.     key->pending = false;  
  22. }  

(3)、檢查DC是否拔出;

handle_power_supply_state(charger, now); 

  1. static void handle_power_supply_state(struct charger *charger, int64_t now)  
  2. {  
  3.     if (charger->num_supplies_online == 0) {  
  4.         if (charger->next_pwr_check == -1) {  
  5.             charger->next_pwr_check = now + UNPLUGGED_SHUTDOWN_TIME;  
  6.             LOGI("[%lld] device unplugged: shutting down in %lld (@ %lld)\n",  
  7.                  now, UNPLUGGED_SHUTDOWN_TIME, charger->next_pwr_check);  
  8.         } else if (now >= charger->next_pwr_check) {  
  9.             LOGI("[%lld] shutting down\n", now);  
  10.             android_reboot(ANDROID_RB_POWEROFF, 0, 0);//如果DC拔出,則關(guān)機;  
  11.         }   
  12. ………………  
  13. }  

(4)、對按鍵時間狀態(tài)標志位的判斷,顯示不同電量的充電logo;

  update_screen_state(charger, now);

這個函數(shù)比較長了,其實做用就是:我們在狀態(tài)的過程中,充電logo的電量是要增加的,比如電量是20%時,要從第一格開始閃爍;如果是80%時,則要從第三格開始閃爍,電量顯示就是通過這個函數(shù)來計算實現(xiàn)的。

  1. static void update_screen_state(struct charger *charger, int64_t now)  
  2. {  
  3.     struct animation *batt_anim = charger->batt_anim;  
  4.     int cur_frame;  
  5.     int disp_time;  
  6.   
  7.     if (!batt_anim->run || now < charger->next_screen_transition)  
  8.         return;  
  9.   
  10.     /* animation is over, blank screen and leave */  
  11.     if (batt_anim->cur_cycle == batt_anim->num_cycles) {  
  12.         reset_animation(batt_anim);  
  13.         charger->next_screen_transition = -1;  
  14.         gr_fb_blank(true);  
  15.         LOGV("[%lld] animation done\n", now);  
  16.         return;  
  17.     }  
  18.   
  19.     disp_time = batt_anim->frames[batt_anim->cur_frame].disp_time;  
  20.   
  21.     /* animation starting, set up the animation */  
  22.     if (batt_anim->cur_frame == 0) {  
  23.         int batt_cap;  
  24.         int ret;  
  25.   
  26.         LOGV("[%lld] animation starting\n", now);  
  27.         batt_cap = get_battery_capacity(charger);  
  28.         if (batt_cap >= 0 && batt_anim->num_frames != 0) {  
  29.             int i;  
  30.   
  31.             /* find first frame given current capacity */  
  32.             for (i = 1; i < batt_anim->num_frames; i++) {  
  33.                 if (batt_cap < batt_anim->frames[i].min_capacity)  
  34.                     break;  
  35.             }  
  36.             batt_anim->cur_frame = i - 1;  
  37.   
  38.             /* show the first frame for twice as long */  
  39.             disp_time = batt_anim->frames[batt_anim->cur_frame].disp_time * 2;  
  40.         }  
  41.   
  42.         batt_anim->capacity = batt_cap;  
  43.     }  
  44.   
  45.     /* unblank the screen  on first cycle */  
  46.     if (batt_anim->cur_cycle == 0)  
  47.         gr_fb_blank(false);  
  48.   
  49.     /* draw the new frame (@ cur_frame) */  
  50.     redraw_screen(charger);  
  51.   
  52.     /* if we don't have anim frames, we only have one image, so just bump 
  53.      * the cycle counter and exit 
  54.      */  
  55.     if (batt_anim->num_frames == 0 || batt_anim->capacity < 0) {  
  56.         LOGV("[%lld] animation missing or unknown battery status\n", now);  
  57.         charger->next_screen_transition = now + BATTERY_UNKNOWN_TIME;  
  58.         batt_anim->cur_cycle++;  
  59.         return;  
  60.     }  
  61.   
  62.     /* schedule next screen transition */  
  63.     charger->next_screen_transition = now + disp_time;  
  64.   
  65.     /* advance frame cntr to the next valid frame 
  66.      * if necessary, advance cycle cntr, and reset frame cntr 
  67.      */  
  68.     batt_anim->cur_frame++;  
  69.   
  70.     /* if the frame is used for level-only, that is only show it when it's 
  71.      * the current level, skip it during the animation. 
  72.      */  
  73.     while (batt_anim->cur_frame < batt_anim->num_frames &&  
  74.            batt_anim->frames[batt_anim->cur_frame].level_only)  
  75.         batt_anim->cur_frame++;  
  76.     if (batt_anim->cur_frame >= batt_anim->num_frames) {  
  77.         batt_anim->cur_cycle++;  
  78.         batt_anim->cur_frame = 0;  
  79.   
  80.         /* don't reset the cycle counter, since we use that as a signal 
  81.          * in a test above to check if animation is over 
  82.          */  
  83.     }  
  84. }  

下面是 不同 電量時 顯示logo的函數(shù):

  1. static struct frame batt_anim_frames[] = {  
  2.     {  
  3.         .name = "charger/battery_0",  
  4.         .disp_time = 750,  
  5.         .min_capacity = 0,  
  6.     },  
  7.     {  
  8.         .name = "charger/battery_1",  
  9.         .disp_time = 750,  
  10.         .min_capacity = 20,  
  11.     },  
  12.     {  
  13.         .name = "charger/battery_2",  
  14.         .disp_time = 750,  
  15.         .min_capacity = 40,  
  16.     },  
  17.     {  
  18.         .name = "charger/battery_3",  
  19.         .disp_time = 750,  
  20.         .min_capacity = 60,  
  21.     },  
  22.     {  
  23.         .name = "charger/battery_4",  
  24.         .disp_time = 750,  
  25.         .min_capacity = 80,  
  26.         .level_only = true,  
  27.     },  
  28.     {  
  29.         .name = "charger/battery_5",  
  30.         .disp_time = 750,  
  31.         .min_capacity = BATTERY_FULL_THRESH,  
  32.     },  
  33. };  
更多0



















本站僅提供存儲服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊舉報。
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
Android init進程中鮮為人知的charger mode 模式
android動畫簡介
詳解Android動畫之Frame Animation
Android 動畫使用心得 - 軟酷快訊
CocoStudio下載地址【持續(xù)更新】 | CocoStudio中文社區(qū)
Android 手機開機問題分析流程
更多類似文章 >>
生活服務(wù)
分享 收藏 導(dǎo)長圖 關(guān)注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權(quán)!
如果VIP功能使用有故障,
可點擊這里聯(lián)系客服!

聯(lián)系客服