239 lines
9.6 KiB
ReStructuredText
239 lines
9.6 KiB
ReStructuredText
|
.. SPDX-License-Identifier: GFDL-1.1-no-invariants-or-later
|
||
|
.. c:namespace:: V4L
|
||
|
|
||
|
.. _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
|
||
|
:c:func:`read()`/\ :c: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 :c:func:`read()` or
|
||
|
:c: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
|
||
|
:c:func:`open()` and :c: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 :c: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.
|