专栏名称: 安卓开发精选
伯乐在线旗下账号,分享安卓应用相关内容,包括:安卓应用开发、设计和动态等。
目录
相关文章推荐
鸿洋  ·  Android 认识AMS与App端框架 ·  昨天  
鸿洋  ·  再学安卓 - SystemServer ·  2 天前  
鸿洋  ·  Android性能优化之绑定RenderTh ... ·  3 天前  
鸿洋  ·  未来三年,最好保持随时离职的能力 ·  4 天前  
51好读  ›  专栏  ›  安卓开发精选

Android LowMemoryKiller原理分析(上)

安卓开发精选  · 公众号  · android  · 2016-10-14 08:26

正文

(点击上方公众号,可快速关注)


来源:伯乐在线专栏作者 - gityuan

链接:http://android.jobbole.com/84953/

点击 → 了解如何加入专栏作者


frameworks/base/services/core/java/com/android/server/am/ProcessList.java

platform/system/core/lmkd/lmkd.c

kernel/common/drivers/staging/Android/lowmemorykiller.c


一. 概述


Android的设计理念之一,便是应用程序退出,但进程还会继续存在系统以便再次启动时提高响应时间. 这样的设计会带来一个问题, 每个进程都有自己独立的内存地址空间,随着应用打开数量的增多,系统已使用的内存越来越大,就很有可能导致系统内存不足, 那么需要一个能管理所有进程,根据一定策略来释放进程的策略,这便有了lmk,全称为LowMemoryKiller(低内存杀手),lmkd来决定什么时间杀掉什么进程.


Android基于Linux的系统,其实Linux有类似的内存管理策略——OOM killer,全称(Out Of Memory Killer), OOM的策略更多的是用于分配内存不足时触发,将得分最高的进程杀掉。而lmk则会每隔一段时间检查一次,当系统剩余可用内存较低时,便会触发杀进程的策略,根据不同的剩余内存档位来来选择杀不同优先级的进程,而不是等到OOM时再来杀进程,真正OOM时系统可能已经处于异常状态,系统更希望的是未雨绸缪,在内存很低时来杀掉一些优先级较低的进程来保障后续操作的顺利进行。


二. framework层


位于ProcessList.java中定义了3种命令类型,这些文件的定义必须跟lmkd.c定义完全一致,格式分别如下:


LMK_TARGET ... (up to 6 pairs)

LMK_PROCPRIO

LMK_PROCREMOVE


功能命令对应方法触发时机
更新oom_adjLMK_TARGETupdateOomLevelsAMS.updateConfiguration
设置进程adjLMK_PROCPRIOsetOomAdjAMS.applyOomAdjLocked
移除进程LMK_PROCREMOVEremoveAMS.handleAppDiedLocked/cleanUpApplicationRecordLocked


在前面文章Android进程调度之adj算法中有讲到AMS.applyOomAdjLocked,接下来以这个过程为主线开始分析。


2.1 AMS.applyOomAdjLocked


private final boolean applyOomAdjLocked(ProcessRecord app, boolean doingAll, long now,

        long nowElapsed) {

    ...

    if (app.curAdj != app.setAdj) {

        //【见小节2.2】

        ProcessList.setOomAdj(app.pid, app.info.uid, app.curAdj);

        app.setAdj = app.curAdj;

    }

    ...

}


2.2 PL.setOomAdj


public static final void setOomAdj(int pid, int uid, int amt) {

    //当adj=16,则直接返回

    if (amt == UNKNOWN_ADJ)

        return;

    long start = SystemClock.elapsedRealtime();

    ByteBuffer buf = ByteBuffer.allocate(4 * 4);

    buf.putInt(LMK_PROCPRIO);

    buf.putInt(pid);

    buf.putInt(uid);

    buf.putInt(amt);

    //将16Byte字节写入socket【见小节2.3】

    writeLmkd(buf);

    long now = SystemClock.elapsedRealtime();

    if ((now-start) > 250) {

        Slog.w("ActivityManager", "SLOW OOM ADJ: " + (now-start) + "ms for pid " + pid

                + " = " + amt);

    }

}


buf大小为16个字节,依次写入LMK_PROCPRIO(命令类型), pid(进程pid), uid(进程uid), amt(目标adj),将这些字节通过socket发送给lmkd.


2.3 PL.writeLmkd


private static void writeLmkd(ByteBuffer buf) {

    //当socket打开失败会尝试3次

    for (int i = 0; i 3; i++) {

        if (sLmkdSocket == null) {

                //打开socket 【见小节2.4】

                if (openLmkdSocket() == false) {

                    try {

                        Thread.sleep(1000);

                    } catch (InterruptedException ie) {

                    }

                    continue;

                }

        }

        try {

            //将buf信息写入lmkd socket

            sLmkdOutputStream.write(buf.array(), 0, buf.position());

            return;

        } catch (IOException ex) {

            try {

                sLmkdSocket.close();

            } catch (IOException ex2) {

            }

            sLmkdSocket = null;

        }

    }

}


  • 当sLmkdSocket为空,并且打开失败,重新执行该操作;

  • 当sLmkdOutputStream写入buf信息失败,则会关闭sLmkdSocket,重新执行该操作;


这个重新执行操作最多3次,如果3次后还失败,则writeLmkd操作会直接结束。尝试3次,则不管结果如何都将退出该操作,可见writeLmkd写入操作还有可能失败的。


2.4 PL.openLmkdSocket


private static boolean openLmkdSocket() {

    try {

        sLmkdSocket = new LocalSocket(LocalSocket.SOCKET_SEQPACKET);

        //与远程lmkd守护进程建立socket连接

        sLmkdSocket.connect(

            new LocalSocketAddress("lmkd",

                    LocalSocketAddress.Namespace.RESERVED));

        sLmkdOutputStream = sLmkdSocket.getOutputStream();

    } catch (IOException ex) {

        Slog.w(TAG, "lowmemorykiller daemon socket open failed");

        sLmkdSocket = null;

        return false;

    }

    return true;

}


sLmkdSocket采用的是SOCK_SEQPACKET,这是类型的socket能提供顺序确定的,可靠的,双向基于连接的socket endpoint,与类型SOCK_STREAM很相似,唯一不同的是SEQPACKET保留消息的边界,而SOCK_STREAM是基于字节流,并不会记录边界。


举例:本地通过write()系统调用向远程先后发送两组数据:一组4字节,一组8字节;对于SOCK_SEQPACKET类型通过read()能获知这是两组数据以及大小,而对于SOCK_STREAM类型,通过read()一次性读取到12个字节,并不知道数据包的边界情况。


常见的数据类型还有SOCK_DGRAM,提供数据报形式,用于udp这样不可靠的通信过程。


再回到openLmkdSocket()方法,该方法是打开一个名为lmkd的socket,类型为LocalSocket.SOCKET_SEQPACKET,这只是一个封装,真实类型就是SOCK_SEQPACKET。先跟远程lmkd守护进程建立连接,再向其通过write()将数据写入该socket,再接下来进入lmkd过程。


三. lmkd


lmkd是由init进程,通过解析init.rc文件来启动的lmkd守护进程,lmkd会创建名为lmkd的socket,节点位于/dev/socket/lmkd,该socket用于跟上层framework交互。


service lmkd /system/bin/lmkd

    class core

    critical

    socket lmkd seqpacket 0660 system system

    writepid /dev/cpuset/system-background/tasks


lmkd启动后,接下里的操作都在platform/system/core/lmkd/lmkd.c文件,首先进入main()方法


3.1 main


int main(int argc __unused, char **argv __unused) {

    struct sched_param param = {

            .sched_priority = 1,

    };

    mlockall(MCL_FUTURE);

    sched_setscheduler(0, SCHED_FIFO, &param);

    //初始化【见小节3.2】

    if (!init())

        mainloop(); //成功后进入loop [见小节3.3]

    ALOGI("exiting");

    return 0;

}


3.2 init


static int init(void) {

    struct epoll_event epev;

    int i;

    int ret;

    page_k = sysconf(_SC_PAGESIZE);

    if (page_k == -1)

        page_k = PAGE_SIZE;

    page_k /= 1024;

    //创建epoll监听文件句柄

    epollfd = epoll_create(MAX_EPOLL_EVENTS);

 

    //获取lmkd控制描述符

    ctrl_lfd = android_get_control_socket("lmkd");

    //监听lmkd socket

    ret = listen(ctrl_lfd, 1);

 

    epev.events = EPOLLIN;

    epev.data.ptr = (void *)ctrl_connect_handler;

 

    //将文件句柄ctrl_lfd,加入epoll句柄

    if (epoll_ctl(epollfd, EPOLL_CTL_ADD, ctrl_lfd, &epev) == -1) {

        return -1;

    }

 

    maxevents++;

    //该路径是否具有可写的权限

    use_inkernel_interface = !access(INKERNEL_MINFREE_PATH, W_OK);

    if (use_inkernel_interface) {

        ALOGI("Using in-kernel low memory killer interface");

    } else {

        ret = init_mp(MEMPRESSURE_WATCH_LEVEL, (void *)&mp_event);

        if (ret)

            ALOGE("Kernel does not support memory pressure events or in-kernel low memory killer");

    }

 

    for (i = 0; i ADJTOSLOT(OOM_SCORE_ADJ_MAX); i++) {

        procadjslot_list[i].next = &procadjslot_list[i];

        procadjslot_list[i].prev = &procadjslot_list[i];

    }

    return 0;

}


这里,通过检验/sys/module/lowmemorykiller/parameters/minfree节点是否具有可写权限来判断是否使用kernel接口来管理lmk事件。默认该节点是具有系统可写的权限,也就意味着use_inkernel_interface=1.


3.3 mainloop


static void mainloop(void) {

    while (1) {

        struct epoll_event events[maxevents];

        int nevents;

        int i;

        ctrl_dfd_reopened = 0;

 

        //等待epollfd上的事件

        nevents = epoll_wait(epollfd, events, maxevents, -1);

        if (nevents == -1) {

            if (errno == EINTR)

                continue;

            continue;

        }

        for (i = 0; i nevents; ++i) {

            if (events[i].events & EPOLLERR)

                ALOGD("EPOLLERR on event #%d", i);

            // 当事件到来,则调用ctrl_connect_handler方法 【见小节3.4】

            if (events[i].data.ptr)

                (*(void (*)(uint32_t))events[i].data.ptr)(events[i].events);

        }

    }

}


主循环调用epoll_wait(),等待epollfd上的事件,当接收到中断或者不存在事件,则执行continue操作。当事件到来,则 调用的ctrl_connect_handler方法,该方法是由init()过程中设定的方法。


3.4 ctrl_connect_handler


static void ctrl_connect_handler(uint32_t events __unused) {

    struct epoll_event epev;

    if (ctrl_dfd >= 0) {

        ctrl_data_close();

        ctrl_dfd_reopened = 1;

    }

    ctrl_dfd = accept(ctrl_lfd, NULL, NULL);

    if (ctrl_dfd 0) {

        ALOGE("lmkd control socket accept failed; errno=%d", errno);

        return;

    }

    ALOGI("ActivityManager connected");

    maxevents++;

    epev.events = EPOLLIN;

    epev.data.ptr = (void *)ctrl_data_handler;

 

    //将ctrl_lfd添加到epollfd

    if (epoll_ctl(epollfd, EPOLL_CTL_ADD, ctrl_dfd, &epev) == -1) {

        ALOGE("epoll_ctl for data connection socket failed; errno=%d", errno);

        ctrl_data_close();

        return;

    }

}


当事件触发,则调用ctrl_data_handler


3.5 ctrl_data_handler


static void ctrl_data_handler(uint32_t events) {

    if (events & EPOLLHUP) {

        //ActivityManager 连接已断开

        if (!ctrl_dfd_reopened)

            ctrl_data_close();

    } else if (events & EPOLLIN) {

        //[见小节3.6]

        ctrl_command_handler();

    }

}


3.6 ctrl_command_handler


static void ctrl_command_handler(void) {

    int ibuf[CTRL_PACKET_MAX / sizeof(int)];

    int len;

    int cmd = -1;

    int nargs;

    int targets;

    len = ctrl_data_read((char *)ibuf, CTRL_PACKET_MAX);

    if (len 0)

        return;

    nargs = len / sizeof(int) - 1;

    if (nargs 0)

        goto wronglen;

    //将网络字节顺序转换为主机字节顺序

    cmd = ntohl(ibuf[0]);

    switch(cmd) {

    case LMK_TARGET:

        targets = nargs / 2;

        if (nargs & 0x1 || targets > (int)ARRAY_SIZE(lowmem_adj))

            goto wronglen;

        cmd_target(targets, &ibuf[1]);

        break;

    case LMK_PROCPRIO:

        if (nargs != 3)

            goto wronglen;

        //设置进程adj【见小节3.7】

        cmd_procprio(ntohl(ibuf[1]), ntohl(ibuf[2]), ntohl(ibuf[3]));

        break;

    case LMK_PROCREMOVE:

        if (nargs != 1)

            goto wronglen;

        cmd_procremove(ntohl(ibuf[1]));

        break;

    default:

        ALOGE("Received unknown command code %d", cmd);

        return;

    }

    return;

wronglen:

    ALOGE("Wrong control socket read length cmd=%d len=%d", cmd, len);

}


CTRL_PACKET_MAX 大小等于 (sizeof(int) * (MAX_TARGETS * 2 + 1));而MAX_TARGETS=6,对于sizeof(int)=4的系统,则CTRL_PACKET_MAX=52。 获取framework传递过来的buf数据后,根据3种不同的命令,进入不同的分支。 接下来,继续以前面传递过来的LMK_PROCPRIO命令来往下讲解,进入cmd_procprio过程。


3.7 cmd_procprio


static void cmd_procprio(int pid, int uid, int oomadj) {

    struct proc *procp;

    char path[80];

    char val[20];

    ...

    snprintf(path, sizeof(path), "/proc/%d/oom_score_adj", pid);

    snprintf(val, sizeof(val), "%d", oomadj);

    //向节点/proc//oom_score_adj写入oomadj

    writefilestring(path, val);

 

    //当使用kernel方式则直接返回

    if (use_inkernel_interface)

        return;

    procp = pid_lookup(pid);

    if (!procp) {

            procp = malloc(sizeof(struct proc));

            if (!procp) {

                // Oh, the irony.  May need to rebuild our state.

                return;

            }

            procp->pid = pid;

            procp->uid = uid;

            procp->oomadj = oomadj;

            proc_insert(procp);

    } else {

        proc_unslot(procp);

        procp->oomadj = oomadj;

        proc_slot(procp);

    }

}


向节点/proc//oom_score_adj写入oomadj。由于use_inkernel_interface=1,那么再接下里需要看看kernel的情况


接下文


专栏作者简介( 点击 → 加入专栏作者 


gityuan:Android全栈工程师:上至能写App,中间能改framework和Native代码,下至能调驱动,全栈能解决性能与稳定性。(新浪微博:@Gityuan)

打赏支持作者写出更多好文章,谢谢!



 关注「安卓开发精选

看更多精选安卓技术文章
↓↓↓