【Camera专题】mm-qcamera-daemon浅析

一、随便聊聊

  • 国庆时,去了重庆游玩,可太有意思了,某天去姐夫的奶茶店喝奶茶,就听到了忽然之间这首歌,阿虾和崔铭嘉的男生合唱版,觉得真的好听,于是最近都在疯狂循环。

  • 就像这首歌的名称,很多想法都是忽然之间,我有了新的想法,希望未来几个月能实现!!!

  • 本文基本参考前辈总结的文档,加自己一丢丢的思考和改动。

  • 推荐文章
    camera daemon进程
    Qualcomm 8X camera daemon进程浅析

二、老版mm-camera框架图

  • mm-camera架构有2个版本,最老的版本是有一个守护进程mm-qcamera-daemon的,
    如msm8909平台,后来新版的架构改过,移除了这个守护进程,如msm8937(sdm429)平台。

  • camera一直在发展,现在最新的架构是CamX,camera架构的变化,是否可以给我们带来一些思考?

  • 2个进程通信
  • 最老版本的mm-camera架构,实际上有2个进程,cameraservermm-qcamera-deamon
  • 这2个进程通信的方式:Domain socket(本地套接字)进行通信的

2.1 mm-qcamera-daemon是什么

  • mm-qcamera-daemon是一个守护进程
    Linux下的daemon进程是是运行在后台的一种特殊进程。
    它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。
    守护进程常常在系统引导装入时启动,在系统关闭时终止。
    由于它真正的父进程在它被fork出来之后就退出了,所以守护进程会被init进程收养、管理,init进程成为它的父进程。
    Linux下大多数系统服务都是由守护进程实现的。

2.2 mm-qcamera-daemon的作用

mm-qcamera-daemo进程是hal和kernel沟通的桥梁。
mm-qcamera-daemon处于hal与kernel之间,进行hal与kernel的通信,

例如:
应用在发出操作camera的执行命令之后,通过framwork、hal,
之后通过ioctl调用到kernel中,kernel则发送消息到daemon中,
daemon接到消息之后再发送消息到各个module中,
各个module操作实际硬件完成相应操作。

2.3、mm-qcamera-daemon的启动

device/qcom/msm8909w/init.target.rc


整个系统在启动的时候会读取、装载 .rc文件,该文件以AIL脚本编写(Android init Lang),
系统启动会解析该文件来启动一些需要初始化启动的服务,
例如:在8909平台下的一个init.target.rc文件,

 service qcamerasvr /system/bin/mm-qcamera-daemon

该行代码要表述的是一个系统服务,服务名为qcamerasvr,
服务程序所在路径为 /system/bin/mm-qcamera-daemon。
system/core/init/init.cpp在解析.rc文件之后就会到对应的路径下启动该服务

2.4、 mm-qcamera-daemon程序入口和流程

vendor/qcom/proprietary/mm-camera/mm-camera2/server-imaging/server.c

该服务主要完成三大工作:

  • (1)、找到服务节点dev/video0,并打开该节点,用来接收消息,提供服务。
  • (2)、初始化各个模块,这里有sensor、iface、isp、stats、pproc以及imagelib。
  • (3)、订阅V4L2事件之后,进入do...while(1)循环,循环中select监听服务节点,接收三类消息:
    • a、来自hal通过kernel传来的消息。(RD_DS_FD_HAL )——Buffer mapping
    • b、hal直接传来的消息。(RD_FD_HAL)——Video node updates from kernel
    • c、来自于mct的消息。(RD_PIPE_FD_MCT)——Updates from MCT (over Unix pipe)
Ps:关于b中的DS指domain socket,是在socket架构上发展起来的用于同一台主机的进程间通讯(IPC),
它不需要经过网络协议栈,不需要打包拆包、计算校验和、维护序号和应答等,
只是将数据从一个进程拷贝到另一个进程。
UNIX Domain Socket有SOCK_DGRAM或SOCK_STREAM两种工作模式,类似于UDP和TCP,
但是面向消息的UNIX Domain Socket也是可靠的,消息既不会丢失也不会顺序错乱。
这里主要用作Buffer mapping.
int main(int argc __unused, char *argv[] __unused)
{

  CLOGD(CAM_MCT_MODULE, "CAMERA_DAEMON: start of camera Daemon.");
  /* 1. find server node name and open the node */

  if (get_server_node_name(serv_hal_node_name) == FALSE){
    CLOGE(CAM_MCT_MODULE, "Going to bad node");
    goto bad_node_fd;
  }

  CLOGD(CAM_MCT_MODULE, "after get_server_node_name");
···
  snprintf(dev_name, sizeof(dev_name), "/dev/%s", serv_hal_node_name);
  hal_fd->fd[0] = open(dev_name, O_RDWR | O_NONBLOCK);

  hal_fd->type = RD_FD_HAL;
···

  CLOGD(CAM_MCT_MODULE, "CAMERA_DAEMON: start all modules init");
  /* 2. after open node, initialize modules */
  if(server_process_module_sensor_init() == FALSE)
    goto module_init_fail;
  CLOGD(CAM_MCT_MODULE, "CAMERA_DAEMON:End of all modules init");

  /* 3. Subcribe V4L2 event */
  memset(&subscribe, 0, sizeof(struct v4l2_event_subscription));
  subscribe.type = MSM_CAMERA_V4L2_EVENT_TYPE;
  for (i = MSM_CAMERA_EVENT_MIN + 1; i < MSM_CAMERA_EVENT_MAX; i++) {
    subscribe.id = i;
    if (ioctl(hal_fd->fd[0], VIDIOC_SUBSCRIBE_EVENT, &subscribe) < 0)
      goto subscribe_failed;
  }

···

  CLOGD(CAM_MCT_MODULE, "CAMERA_DAEMON:waiting for camera to open");
  do {

    FD_ZERO(&(select_fds.fds));
    mct_list_traverse(listen_fd_list, server_reset_select_fd, &select_fds);

    /* no timeout */
    ret = select(select_fds.select_fd + 1, &(select_fds.fds), NULL, NULL, NULL);

    if (ret > 0) {
     ···
      switch (fd_info->type) {
      case RD_FD_HAL: {

        if (ioctl(fd_info->fd[0], VIDIOC_DQEVENT, &event) < 0) {
          continue;
        }
        /* server process HAL event:*/
      ···
        break;

      case RD_DS_FD_HAL:
      ···
        break;

      case RD_PIPE_FD_MCT:
        /* server process message sent by media controller
         * through pipe: */
        proc_ret = server_process_mct_msg(fd_info->fd[0],
          fd_info->session);
        break;

      default:
        continue;
      } /* switch (fd_info->type) */

流程图如下:


三、总结

  • 1 、在Server.c中从mian()函数进入
    (1) find server node name and open the node,找到server节点并且打开节点
    (2) after open node, initialize modules,打开节点后初始化模块
  • 2、在初始化模块中
    (1) 调用server_process_module_sensor_init() 进入server_process.c中的boolean server_process_module_sensor_init(void)函数
    (2) 根据modules_list[0]初始化list数组中的模块,初始化模块包含sensor,iface,isp,stats,pproc和imglib
static mct_module_init_name_t modules_list[] = {
 {"sensor", module_sensor_init,   module_sensor_deinit, NULL},
 {"iface",  module_iface_init,   module_iface_deinit, NULL},
 {"isp",    module_isp_init,      module_isp_deinit, NULL},
 {"stats",  stats_module_init,    stats_module_deinit, NULL},
 {"pproc",  pproc_module_init,    pproc_module_deinit, NULL},
 {"imglib", module_imglib_init, module_imglib_deinit, NULL},
};
  • 3、Sensor模块初始化过程
    (1) 调用module_sensor_init函数进入module_sensor.c中的mct_module_t *module_sensor_init(const char *name)函数
    (2) Create MCT module for sensor,为sensor创建MCT Module
    (3) Fill function table in MCT module,填充功能表
    (4) Create sensor module control structure that consists of bundle information,创建sensor module control结构体:module_ctrl = malloc(sizeof(module_sensor_ctrl_t))
    (5) module_sensor_probe_sensors,sensor识过程,在server_init.c中调用sensor_init_probe函数
    1 > 打开设备节点"/dev/video0"
    2 > Open sensor_init subdev,找到subdev并且打开
    3 > sensor识别过程
    (6) module_sensor_find_other_subdev(module_ctrl),找到sensor外的其他子设备,为每一个自设备创建sensor_bundle,并填充设备信息到sensor_bundle
    (7) Create ports based on CID info,为sensor_bundle创建端口
    (8) intiialize the eeprom,初始化eeprom,module_sensor_init_eeprom
  • 4、iface、isp、stats、pproc和imglib相继进行初始化操作
  • 5、进入主循环来处理来自HAL及MCT的事件及消息,处理完之后的结果反馈给kernel

Stay hungry,Stay foolish!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • ![Flask](...
    极客学院Wiki阅读 7,229评论 0 3
  • 不知不觉易趣客已经在路上走了快一年了,感觉也该让更多朋友认识知道易趣客,所以就谢了这篇简介,已做创业记事。 易趣客...
    Physher阅读 3,407评论 1 2
  • 双胎妊娠有家族遗传倾向,随母系遗传。有研究表明,如果孕妇本人是双胎之一,她生双胎的机率为1/58;若孕妇的父亲或母...
    邺水芙蓉hibiscus阅读 3,694评论 0 2
  • 晴天,拥抱阳光,拥抱你。雨天,想念雨滴,想念你。 我可以喜欢你吗可以啊 我还可以喜欢你吗可以,可是你要知道我们不可...
    露薇霜凝阅读 1,202评论 1 2