int vb2_core_streamon(struct vb2_queue *q, unsigned int type) { int ret; if (type != q->type) { dprintk(1, "invalid stream type\n"); return -EINVAL; } if (q->streaming) { dprintk(3, "already streaming\n"); return 0; } if (!q->num_buffers) { dprintk(1, "no buffers have been allocated\n"); return -EINVAL; } if (q->num_buffers < q->min_buffers_needed) { dprintk(1, "need at least %u allocated buffers\n", q->min_buffers_needed); return -EINVAL; } /* * Tell driver to start streaming provided sufficient buffers * are available. */ if (q->queued_count >= q->min_buffers_needed) { ret = v4l_vb2q_enable_media_source(q); if (ret) return ret; ret = vb2_start_streaming(q); if (ret) return ret; } q->streaming = 1; dprintk(3, "successful\n"); return 0; } EXPORT_SYMBOL_GPL(vb2_core_streamon); /* * vb2_start_streaming() - Attempt to start streaming. * @q: videobuf2 queue * * Attempt to start streaming. When this function is called there must be * at least q->min_buffers_needed buffers queued up (i.e. the minimum * number of buffers required for the DMA engine to function). If the * @start_streaming op fails it is supposed to return all the driver-owned * buffers back to vb2 in state QUEUED. Check if that happened and if * not warn and reclaim them forcefully. */ static int vb2_start_streaming(struct vb2_queue *q) { struct vb2_buffer *vb; int ret; /* * If any buffers were queued before streamon, * we can now pass them to driver for processing. */ list_for_each_entry(vb, &q->queued_list, queued_entry) __enqueue_in_driver(vb); /* Tell the driver to start streaming */ q->start_streaming_called = 1; ret = call_qop(q, start_streaming, q, atomic_read(&q->owned_by_drv_count)); if (!ret) return 0; q->start_streaming_called = 0; dprintk(1, "driver refused to start streaming\n"); /* * If you see this warning, then the driver isn't cleaning up properly * after a failed start_streaming(). See the start_streaming() * documentation in videobuf2-core.h for more information how buffers * should be returned to vb2 in start_streaming(). */ if (WARN_ON(atomic_read(&q->owned_by_drv_count))) { unsigned i; /* * Forcefully reclaim buffers if the driver did not * correctly return them to vb2. */ for (i = 0; i < q->num_buffers; ++i) { vb = q->bufs[i]; if (vb->state == VB2_BUF_STATE_ACTIVE) vb2_buffer_done(vb, VB2_BUF_STATE_QUEUED); } /* Must be zero now */ WARN_ON(atomic_read(&q->owned_by_drv_count)); } /* * If done_list is not empty, then start_streaming() didn't call * vb2_buffer_done(vb, VB2_BUF_STATE_QUEUED) but STATE_ERROR or * STATE_DONE. */ WARN_ON(!list_empty(&q->done_list)); return ret; } static struct vb2_ops rkcif_vb2_ops = { .queue_setup = rkcif_queue_setup, .buf_queue = rkcif_buf_queue, .wait_prepare = vb2_ops_wait_prepare, .wait_finish = vb2_ops_wait_finish, .stop_streaming = rkcif_stop_streaming, .start_streaming = rkcif_start_streaming, }; rkcif_start_streaming()实现如下: static int rkcif_start_streaming(struct vb2_queue *queue, unsigned int count) { struct rkcif_stream *stream = queue->drv_priv; struct rkcif_vdev_node *node = &stream->vnode; struct rkcif_device *dev = stream->cifdev; struct v4l2_device *v4l2_dev = &dev->v4l2_dev; struct rkcif_sensor_info *sensor_info = dev->active_sensor; struct rkcif_sensor_info *terminal_sensor = &dev->terminal_sensor; struct rkmodule_hdr_cfg hdr_cfg; int ret; mutex_lock(&dev->stream_lock); if (WARN_ON(stream->state != RKCIF_STATE_READY)) { ret = -EBUSY; v4l2_err(v4l2_dev, "stream in busy state\n"); goto destroy_buf; } /* * 这里挺奇怪既然active_sensor有值,为什么还要再去检测一遍??? * rkcif_update_sensor_info 之前已经分析过了 * 基于RV1126平台imx291分析 --- open及media graph分析 * https://blog.csdn.net/ldl617/article/details/115862796 */ if (dev->active_sensor) { ret = rkcif_update_sensor_info(stream); if (ret < 0) { v4l2_err(v4l2_dev, "update sensor info failed %d\n", ret); goto out; } } /* * 这里的terminal_sensor->sd 对应的就是imx291的subdev */ if (terminal_sensor->sd) { /* * 调用imx291的core->ioctl */ ret = v4l2_subdev_call(terminal_sensor->sd, core, ioctl, RKMODULE_GET_HDR_CFG, &hdr_cfg); if (!ret) dev->hdr.mode = hdr_cfg.hdr_mode; else dev->hdr.mode = NO_HDR; /* * 调用imx291的video->g_frame_interval */ ret = v4l2_subdev_call(terminal_sensor->sd, video, g_frame_interval, &terminal_sensor->fi); if (ret) terminal_sensor->fi.interval = (struct v4l2_fract) {1, 30}; rkcif_sync_crop_info(stream); } ... /* * 基于RV1126平台imx291分析 --- rkcif_mipi注册 * https://blog.csdn.net/ldl617/article/details/115551981 * 上面的文章中有pipe相关的信息 * cif_dev->pipe.open = rkcif_pipeline_open; * cif_dev->pipe.close = rkcif_pipeline_close; * cif_dev->pipe.set_stream = rkcif_pipeline_set_stream; * 分析一下rkcif_pipeline_open */ ret = dev->pipe.open(&dev->pipe, &node->vdev.entity, true); if (ret < 0) { v4l2_err(v4l2_dev, "open cif pipeline failed %d\n", ret); goto destroy_buf; } ... } rkcif_start_streaming() -> rkcif_pipeline_open() static int __cif_pipeline_prepare(struct rkcif_pipeline *p, struct media_entity *me) { struct v4l2_subdev *sd; int i; p->num_subdevs = 0; memset(p->subdevs, 0, sizeof(p->subdevs)); while (1) { struct media_pad *pad = NULL; /* Find remote source pad */ for (i = 0; i < me->num_pads; i++) { struct media_pad *spad = &me->pads[i]; /* * 注意这里只找sink pad */ if (!(spad->flags & MEDIA_PAD_FL_SINK)) continue; /* * media_entity_remote_pad 分析过了 * 现在这里只有backlink才符合 * 所以这里的pad依次是 * 1. mipi csi source pad * 2. mipi csi phy source pad * 3. imx291 source pad */ pad = media_entity_remote_pad(spad); if (pad) break; } if (!pad) break; /* * sd依次为 * 1. mipi csi subdev * 2. mipi csi phy subdev * 3. imx291 subdev * 保存到p->subdevs中 */ sd = media_entity_to_v4l2_subdev(pad->entity); p->subdevs[p->num_subdevs++] = sd; me = &sd->entity; /* * 对于imx291 只有一个source pad * 所以这里直接退出 */ if (me->num_pads == 1) break; } return 0; } static int rkcif_pipeline_open(struct rkcif_pipeline *p, struct media_entity *me, bool prepare) { int ret; if (WARN_ON(!p || !me)) return -EINVAL; if (atomic_inc_return(&p->power_cnt) > 1) return 0; /* go through media graphic and get subdevs */ if (prepare) /* * 分析看上面 */ __cif_pipeline_prepare(p, me); if (!p->num_subdevs) return -EINVAL; /* * 空函数,return 0 */ ret = __cif_pipeline_s_cif_clk(p); if (ret < 0) return ret; return 0; } 回到 rkcif_start_streaming()继续分析 ret = media_pipeline_start(&node->vdev.entity, &dev->pipe.pipe); if (ret < 0) { v4l2_err(&dev->v4l2_dev, "start pipeline failed %d\n", ret); goto pipe_stream_off; } rkcif_start_streaming() -> media_pipeline_start() __must_check int __media_pipeline_start(struct media_entity *entity, struct media_pipeline *pipe) { struct media_device *mdev = entity->graph_obj.mdev; struct media_graph *graph = &pipe->graph; struct media_entity *entity_err = entity; struct media_link *link; int ret; /* * 首次会进行初始化的操作 * 注意这里初始化的是pipe->graph * 之前已经分析过2个graph了 * 1. 局部变量graph 用于update_info * 2. mdev->pm_count_walk 用于pipeline_pm_power */ if (!pipe->streaming_count++) { ret = media_graph_walk_init(&pipe->graph, mdev); if (ret) goto error_graph_walk_start; } /* * 这里的entity是rkcif_mipi的entity */ media_graph_walk_start(&pipe->graph, entity); while ((entity = media_graph_walk_next(graph))) { /* * #define DECLARE_BITMAP(name,bits) * unsigned long name[BITS_TO_LONGS(bits)] * 注意下面的active和has_no_link都是局部变量 * 每次while时候都会重新定义 * 对于这里的entity看下面文章分析 * 基于RV1126平台imx291分析 --- v4l2_pipeline_pm_use * https://blog.csdn.net/ldl617/article/details/115898236 * 可以知道依次出现的顺序是 * 1. m01_f_imx291 1-001a * 2. rockchip-mipi-dphy-rx * 3. stream_cif_mipi_id1 * 4. stream_cif_mipi_id2 * 5. stream_cif_mipi_id3 * 6. rockchip-mipi-csi2 */ DECLARE_BITMAP(active, MEDIA_ENTITY_MAX_PADS); DECLARE_BITMAP(has_no_links, MEDIA_ENTITY_MAX_PADS); /* * 更新stream流的计数 */ entity->stream_count++; if (WARN_ON(entity->pipe && entity->pipe != pipe)) { ret = -EBUSY; goto error; } /* * 赋值pipe */ entity->pipe = pipe; /* Already streaming --- no need to check. */ if (entity->stream_count > 1) continue; /* * 调用link_validate * 当前的环境没有link_validate * 所以下面的代码都执行不到 */ if (!entity->ops || !entity->ops->link_validate) continue; bitmap_zero(active, entity->num_pads); bitmap_fill(has_no_links, entity->num_pads); list_for_each_entry(link, &entity->links, list) { struct media_pad *pad = link->sink->entity == entity ? link->sink : link->source; /* Mark that a pad is connected by a link. */ bitmap_clear(has_no_links, pad->index, 1); /* * Pads that either do not need to connect or * are connected through an enabled link are * fine. */ if (!(pad->flags & MEDIA_PAD_FL_MUST_CONNECT) || link->flags & MEDIA_LNK_FL_ENABLED) bitmap_set(active, pad->index, 1); /* * Link validation will only take place for * sink ends of the link that are enabled. */ if (link->sink != pad || !(link->flags & MEDIA_LNK_FL_ENABLED)) continue; ret = entity->ops->link_validate(link); if (ret < 0 && ret != -ENOIOCTLCMD) { dev_dbg(entity->graph_obj.mdev->dev, "link validation failed for '%s':%u -> '%s':%u, error %d\n", link->source->entity->name, link->source->index, entity->name, link->sink->index, ret); goto error; } } /* Either no links or validated links are fine. */ bitmap_or(active, active, has_no_links, entity->num_pads); if (!bitmap_full(active, entity->num_pads)) { ret = -ENOLINK; dev_dbg(entity->graph_obj.mdev->dev, "'%s':%u must be connected by an enabled link\n", entity->name, (unsigned)find_first_zero_bit( active, entity->num_pads)); goto error; } } return 0; } __must_check int media_pipeline_start(struct media_entity *entity, struct media_pipeline *pipe) { struct media_device *mdev = entity->graph_obj.mdev; int ret; mutex_lock(&mdev->graph_mutex); ret = __media_pipeline_start(entity, pipe); mutex_unlock(&mdev->graph_mutex); return ret; } rkcif_start_streaming()中我们关注的代码已经分析完成了,可以说VIDIOC_STREAMON这部分就分析到这里了。对于media_pipeline_start() 看到一知半解,现在看来作用就是entity->stream_count++。找找这个平台lvds的一个link_validate看看: static int v4l2_subdev_link_validate_get_format(struct media_pad *pad, struct v4l2_subdev_format *fmt) { if (is_media_entity_v4l2_subdev(pad->entity)) { struct v4l2_subdev *sd = media_entity_to_v4l2_subdev(pad->entity); fmt->which = V4L2_SUBDEV_FORMAT_ACTIVE; fmt->pad = pad->index; return v4l2_subdev_call(sd, pad, get_fmt, NULL, fmt); } WARN(pad->entity->function != MEDIA_ENT_F_IO_V4L, "Driver bug! Wrong media entity type 0x%08x, entity %s\n", pad->entity->function, pad->entity->name); return -EINVAL; } int v4l2_subdev_link_validate(struct media_link *link) { struct v4l2_subdev *sink; struct v4l2_subdev_format sink_fmt, source_fmt; int rval; /* * 找到source entity的subdev,调用pad->get_format获取格式 */ rval = v4l2_subdev_link_validate_get_format( link->source, &source_fmt); if (rval < 0) return 0; /* * 找到sink entity的subdev,调用pad->get_format获取格式 */ rval = v4l2_subdev_link_validate_get_format( link->sink, &sink_fmt); if (rval < 0) return 0; /* sink entity的subdev */ sink = media_entity_to_v4l2_subdev(link->sink->entity); /* * 没有找到合适的link_validate * 所以认为没有 */ rval = v4l2_subdev_call(sink, pad, link_validate, link, &source_fmt, &sink_fmt); if (rval != -ENOIOCTLCMD) return rval; /* * source 和sink的format通过一定的规则进行对比 */ return v4l2_subdev_link_validate_default( sink, link, &source_fmt, &sink_fmt); } 看起来这个平台的link_validate只是进行了sink和source的format对比。感觉还是想的简单了,全局搜索entity->stream_count++。找到了关键代码,之前我们其实分析过。 请参考:基于RV1126平台imx291分析 --- media部件连接 csi2_notifier_bound() -> media_entity_setup_link() int __media_entity_setup_link(struct media_link *link, u32 flags) { const u32 mask = MEDIA_LNK_FL_ENABLED; struct media_device *mdev; struct media_entity *source, *sink; int ret = -EBUSY; if (link == NULL) return -EINVAL; /* The non-modifiable link flags must not be modified. */ /* * link->flags 这里是0 * flag = MEDIA_LNK_FL_ENABLE * 所以if不满足 */ if ((link->flags & ~mask) != (flags & ~mask)) return -EINVAL; if (link->flags & MEDIA_LNK_FL_IMMUTABLE) return link->flags == flags ? 0 : -EINVAL; if (link->flags == flags) return 0; /* * 分别找到source entity * 和sink entity */ source = link->source->entity; sink = link->sink->entity; /* * stream_count值大于0,可以认为启动了数据流传输 * 这里没有,所以为0 */ if (!(link->flags & MEDIA_LNK_FL_DYNAMIC) && (source->stream_count || sink->stream_count)) return -EBUSY; ... } 有判断source->stream_count和sink->stream_count。所以这里可不可以这么认为,就是当前的entity已经在运行了,因为一些原因或者操作,这个entity参与到了别的链路创建。这里加上判断是不允许的,当然前提是当前运行的这个link不支持dynamic动态的。 |
|
来自: charlie_linux > 《V4L2》