mirror of https://gitee.com/openkylin/linux.git
242 lines
9.7 KiB
ReStructuredText
242 lines
9.7 KiB
ReStructuredText
.. SPDX-License-Identifier: GFDL-1.1-no-invariants-or-later
|
|
|
|
.. _open:
|
|
|
|
***************************
|
|
Opening and Closing Devices
|
|
***************************
|
|
|
|
.. _v4l2_hardware_control:
|
|
|
|
Controlling a hardware peripheral via V4L2
|
|
==========================================
|
|
|
|
Hardware that is supported using the V4L2 uAPI often consists of multiple
|
|
devices or peripherals, each of which have their own driver.
|
|
|
|
The bridge driver exposes one or more V4L2 device nodes
|
|
(see :ref:`v4l2_device_naming`).
|
|
|
|
There are other drivers providing support for other components of
|
|
the hardware, which may also expose device nodes, called V4L2 sub-devices.
|
|
|
|
When such V4L2 sub-devices are exposed, they allow controlling those
|
|
other hardware components - usually connected via a serial bus (like
|
|
I²C, SMBus or SPI). Depending on the bridge driver, those sub-devices
|
|
can be controlled indirectly via the bridge driver or explicitly via
|
|
the :ref:`Media Controller <media_controller>` and via the
|
|
:ref:`V4L2 sub-devices <subdev>`.
|
|
|
|
The devices that require the use of the
|
|
:ref:`Media Controller <media_controller>` are called **MC-centric**
|
|
devices. The devices that are fully controlled via V4L2 device nodes
|
|
are called **video-node-centric**.
|
|
|
|
Userspace can check if a V4L2 hardware peripheral is MC-centric by
|
|
calling :ref:`VIDIOC_QUERYCAP` and checking the
|
|
:ref:`device_caps field <device-capabilities>`.
|
|
|
|
If the device returns ``V4L2_CAP_IO_MC`` flag at ``device_caps``,
|
|
then it is MC-centric, otherwise, it is video-node-centric.
|
|
|
|
It is required for MC-centric drivers to identify the V4L2
|
|
sub-devices and to configure the pipelines via the
|
|
:ref:`media controller API <media_controller>` before using the peripheral.
|
|
Also, the sub-devices' configuration shall be controlled via the
|
|
:ref:`sub-device API <subdev>`.
|
|
|
|
.. note::
|
|
|
|
A video-node-centric may still provide media-controller and
|
|
sub-device interfaces as well.
|
|
|
|
However, in that case the media-controller and the sub-device
|
|
interfaces are read-only and just provide information about the
|
|
device. The actual configuration is done via the video nodes.
|
|
|
|
.. _v4l2_device_naming:
|
|
|
|
V4L2 Device Node Naming
|
|
=======================
|
|
|
|
V4L2 drivers are implemented as kernel modules, loaded manually by the
|
|
system administrator or automatically when a device is first discovered.
|
|
The driver modules plug into the ``videodev`` kernel module. It provides
|
|
helper functions and a common application interface specified in this
|
|
document.
|
|
|
|
Each driver thus loaded registers one or more device nodes with major
|
|
number 81. Minor numbers are allocated dynamically unless the kernel
|
|
is compiled with the kernel option CONFIG_VIDEO_FIXED_MINOR_RANGES.
|
|
In that case minor numbers are allocated in ranges depending on the
|
|
device node type.
|
|
|
|
The device nodes supported by the Video4Linux subsystem are:
|
|
|
|
======================== ====================================================
|
|
Default device node name Usage
|
|
======================== ====================================================
|
|
``/dev/videoX`` Video and metadata for capture/output devices
|
|
``/dev/vbiX`` Vertical blank data (i.e. closed captions, teletext)
|
|
``/dev/radioX`` Radio tuners and modulators
|
|
``/dev/swradioX`` Software Defined Radio tuners and modulators
|
|
``/dev/v4l-touchX`` Touch sensors
|
|
``/dev/v4l-subdevX`` Video sub-devices (used by sensors and other
|
|
components of the hardware peripheral)\ [#]_
|
|
======================== ====================================================
|
|
|
|
Where ``X`` is a non-negative integer.
|
|
|
|
.. note::
|
|
|
|
1. The actual device node name is system-dependent, as udev rules may apply.
|
|
2. There is no guarantee that ``X`` will remain the same for the same
|
|
device, as the number depends on the device driver's probe order.
|
|
If you need an unique name, udev default rules produce
|
|
``/dev/v4l/by-id/`` and ``/dev/v4l/by-path/`` directories containing
|
|
links that can be used uniquely to identify a V4L2 device node::
|
|
|
|
$ tree /dev/v4l
|
|
/dev/v4l
|
|
├── by-id
|
|
│ └── usb-OmniVision._USB_Camera-B4.04.27.1-video-index0 -> ../../video0
|
|
└── by-path
|
|
└── pci-0000:00:14.0-usb-0:2:1.0-video-index0 -> ../../video0
|
|
|
|
.. [#] **V4L2 sub-device nodes** (e. g. ``/dev/v4l-subdevX``) use a different
|
|
set of system calls, as covered at :ref:`subdev`.
|
|
|
|
Many drivers support "video_nr", "radio_nr" or "vbi_nr" module
|
|
options to select specific video/radio/vbi node numbers. This allows the
|
|
user to request that the device node is named e.g. /dev/video5 instead
|
|
of leaving it to chance. When the driver supports multiple devices of
|
|
the same type more than one device node number can be assigned,
|
|
separated by commas:
|
|
|
|
.. code-block:: none
|
|
|
|
# modprobe mydriver video_nr=0,1 radio_nr=0,1
|
|
|
|
In ``/etc/modules.conf`` this may be written as:
|
|
|
|
::
|
|
|
|
options mydriver video_nr=0,1 radio_nr=0,1
|
|
|
|
When no device node number is given as module option the driver supplies
|
|
a default.
|
|
|
|
Normally udev will create the device nodes in /dev automatically for
|
|
you. If udev is not installed, then you need to enable the
|
|
CONFIG_VIDEO_FIXED_MINOR_RANGES kernel option in order to be able to
|
|
correctly relate a minor number to a device node number. I.e., you need
|
|
to be certain that minor number 5 maps to device node name video5. With
|
|
this kernel option different device types have different minor number
|
|
ranges. These ranges are listed in :ref:`devices`.
|
|
|
|
The creation of character special files (with mknod) is a privileged
|
|
operation and devices cannot be opened by major and minor number. That
|
|
means applications cannot *reliably* scan for loaded or installed
|
|
drivers. The user must enter a device name, or the application can try
|
|
the conventional device names.
|
|
|
|
|
|
.. _related:
|
|
|
|
Related Devices
|
|
===============
|
|
|
|
Devices can support several functions. For example video capturing, VBI
|
|
capturing and radio support.
|
|
|
|
The V4L2 API creates different V4L2 device nodes for each of these functions.
|
|
|
|
The V4L2 API was designed with the idea that one device node could
|
|
support all functions. However, in practice this never worked: this
|
|
'feature' was never used by applications and many drivers did not
|
|
support it and if they did it was certainly never tested. In addition,
|
|
switching a device node between different functions only works when
|
|
using the streaming I/O API, not with the
|
|
:ref:`read() <func-read>`/\ :ref:`write() <func-write>` API.
|
|
|
|
Today each V4L2 device node supports just one function.
|
|
|
|
Besides video input or output the hardware may also support audio
|
|
sampling or playback. If so, these functions are implemented as ALSA PCM
|
|
devices with optional ALSA audio mixer devices.
|
|
|
|
One problem with all these devices is that the V4L2 API makes no
|
|
provisions to find these related V4L2 device nodes. Some really complex
|
|
hardware use the Media Controller (see :ref:`media_controller`) which can
|
|
be used for this purpose. But several drivers do not use it, and while some
|
|
code exists that uses sysfs to discover related V4L2 device nodes (see
|
|
libmedia_dev in the
|
|
`v4l-utils <http://git.linuxtv.org/cgit.cgi/v4l-utils.git/>`__ git
|
|
repository), there is no library yet that can provide a single API
|
|
towards both Media Controller-based devices and devices that do not use
|
|
the Media Controller. If you want to work on this please write to the
|
|
linux-media mailing list:
|
|
`https://linuxtv.org/lists.php <https://linuxtv.org/lists.php>`__.
|
|
|
|
|
|
Multiple Opens
|
|
==============
|
|
|
|
V4L2 devices can be opened more than once. [#f1]_ When this is supported
|
|
by the driver, users can for example start a "panel" application to
|
|
change controls like brightness or audio volume, while another
|
|
application captures video and audio. In other words, panel applications
|
|
are comparable to an ALSA audio mixer application. Just opening a V4L2
|
|
device should not change the state of the device. [#f2]_
|
|
|
|
Once an application has allocated the memory buffers needed for
|
|
streaming data (by calling the :ref:`VIDIOC_REQBUFS`
|
|
or :ref:`VIDIOC_CREATE_BUFS` ioctls, or
|
|
implicitly by calling the :ref:`read() <func-read>` or
|
|
:ref:`write() <func-write>` functions) that application (filehandle)
|
|
becomes the owner of the device. It is no longer allowed to make changes
|
|
that would affect the buffer sizes (e.g. by calling the
|
|
:ref:`VIDIOC_S_FMT <VIDIOC_G_FMT>` ioctl) and other applications are
|
|
no longer allowed to allocate buffers or start or stop streaming. The
|
|
EBUSY error code will be returned instead.
|
|
|
|
Merely opening a V4L2 device does not grant exclusive access. [#f3]_
|
|
Initiating data exchange however assigns the right to read or write the
|
|
requested type of data, and to change related properties, to this file
|
|
descriptor. Applications can request additional access privileges using
|
|
the priority mechanism described in :ref:`app-pri`.
|
|
|
|
|
|
Shared Data Streams
|
|
===================
|
|
|
|
V4L2 drivers should not support multiple applications reading or writing
|
|
the same data stream on a device by copying buffers, time multiplexing
|
|
or similar means. This is better handled by a proxy application in user
|
|
space.
|
|
|
|
|
|
Functions
|
|
=========
|
|
|
|
To open and close V4L2 devices applications use the
|
|
:ref:`open() <func-open>` and :ref:`close() <func-close>` function,
|
|
respectively. Devices are programmed using the
|
|
:ref:`ioctl() <func-ioctl>` function as explained in the following
|
|
sections.
|
|
|
|
.. [#f1]
|
|
There are still some old and obscure drivers that have not been
|
|
updated to allow for multiple opens. This implies that for such
|
|
drivers :ref:`open() <func-open>` can return an ``EBUSY`` error code
|
|
when the device is already in use.
|
|
|
|
.. [#f2]
|
|
Unfortunately, opening a radio device often switches the state of the
|
|
device to radio mode in many drivers. This behavior should be fixed
|
|
eventually as it violates the V4L2 specification.
|
|
|
|
.. [#f3]
|
|
Drivers could recognize the ``O_EXCL`` open flag. Presently this is
|
|
not required, so applications cannot know if it really works.
|