2018-08-30 17:20:04 +03:00
.. SPDX-License-Identifier: GPL-2.0
2016-07-22 16:15:42 +03:00
V4L2 device instance
--------------------
2016-07-20 21:07:56 +03:00
2020-09-25 13:01:25 +03:00
Each device instance is represented by a struct v4l2_device.
2016-07-20 21:07:56 +03:00
Very simple devices can just allocate this struct, but most of the time you
would embed this struct inside a larger struct.
2016-07-20 21:20:02 +03:00
You must register the device instance by calling:
2016-07-20 21:07:56 +03:00
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_register <v4l2_device_register>`
2016-07-20 21:20:02 +03:00
(dev, :c:type: `v4l2_dev <v4l2_device>` ).
2016-07-20 21:07:56 +03:00
2016-07-20 21:20:02 +03:00
Registration will initialize the :c:type: `v4l2_device` struct. If the
dev->driver_data field is `` NULL `` , it will be linked to
:c:type: `v4l2_dev <v4l2_device>` argument.
2016-07-20 21:07:56 +03:00
Drivers that want integration with the media device framework need to set
dev->driver_data manually to point to the driver-specific device structure
2020-09-25 13:01:25 +03:00
that embed the struct v4l2_device instance. This is achieved by a
2016-07-20 21:20:02 +03:00
`` dev_set_drvdata() `` call before registering the V4L2 device instance.
2020-09-25 13:01:25 +03:00
They must also set the struct v4l2_device mdev field to point to a
2016-07-20 21:20:02 +03:00
properly initialized and registered :c:type: `media_device` instance.
If :c:type: `v4l2_dev <v4l2_device>` \ ->name is empty then it will be set to a
value derived from dev (driver name followed by the bus_id, to be precise).
2016-07-23 13:21:06 +03:00
If you set it up before calling :c:func: `v4l2_device_register` then it will
2016-07-20 21:20:02 +03:00
be untouched. If dev is `` NULL `` , then you **must** setup
:c:type: `v4l2_dev <v4l2_device>` \ ->name before calling
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_register` .
2016-07-20 21:20:02 +03:00
2016-07-23 13:21:06 +03:00
You can use :c:func: `v4l2_device_set_name` to set the name based on a driver
2016-07-20 21:20:02 +03:00
name and a driver-global atomic_t instance. This will generate names like
`` ivtv0 `` , `` ivtv1 `` , etc. If the name ends with a digit, then it will insert
a dash: `` cx18-0 `` , `` cx18-1 `` , etc. This function returns the instance number.
The first `` dev `` argument is normally the `` struct device `` pointer of a
`` pci_dev `` , `` usb_interface `` or `` platform_device `` . It is rare for dev to
be `` NULL `` , but it happens with ISA devices or when one device creates
multiple PCI devices, thus making it impossible to associate
:c:type: `v4l2_dev <v4l2_device>` with a particular parent.
You can also supply a `` notify() `` callback that can be called by sub-devices
to notify you of events. Whether you need to set this depends on the
sub-device. Any notifications a sub-device supports must be defined in a header
in `` include/media/subdevice.h `` .
V4L2 devices are unregistered by calling:
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_unregister`
2016-07-20 21:20:02 +03:00
(:c:type: `v4l2_dev <v4l2_device>` ).
If the dev->driver_data field points to :c:type: `v4l2_dev <v4l2_device>` ,
it will be reset to `` NULL `` . Unregistering will also automatically unregister
all subdevs from the device.
2016-07-20 21:07:56 +03:00
If you have a hotpluggable device (e.g. a USB device), then when a disconnect
2016-07-20 21:20:02 +03:00
happens the parent device becomes invalid. Since :c:type: `v4l2_device` has a
pointer to that parent device it has to be cleared as well to mark that the
parent is gone. To do this call:
2016-07-20 21:07:56 +03:00
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_disconnect`
2016-07-20 21:20:02 +03:00
(:c:type: `v4l2_dev <v4l2_device>` ).
2016-07-20 21:07:56 +03:00
This does *not* unregister the subdevs, so you still need to call the
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_unregister` function for that. If your driver is not
hotpluggable, then there is no need to call :c:func: `v4l2_device_disconnect` .
2016-07-20 21:07:56 +03:00
Sometimes you need to iterate over all devices registered by a specific
driver. This is usually the case if multiple device drivers use the same
hardware. E.g. the ivtvfb driver is a framebuffer driver that uses the ivtv
hardware. The same is true for alsa drivers for example.
You can iterate over all registered devices as follows:
2016-07-20 21:20:02 +03:00
.. code-block :: c
2016-07-20 21:07:56 +03:00
static int callback(struct device *dev, void * p)
{
struct v4l2_device *v4l2_dev = dev_get_drvdata(dev);
/* test if this device was inited * /
if (v4l2_dev == NULL)
return 0;
...
return 0;
}
int iterate(void *p)
{
struct device_driver *drv;
int err;
/* Find driver 'ivtv' on the PCI bus.
2019-02-18 22:29:07 +03:00
pci_bus_type is a global. For USB buses use usb_bus_type. */
2016-07-20 21:07:56 +03:00
drv = driver_find("ivtv", &pci_bus_type);
/* iterate over all ivtv device instances * /
err = driver_for_each_device(drv, NULL, p, callback);
put_driver(drv);
return err;
}
Sometimes you need to keep a running counter of the device instance. This is
commonly used to map a device instance to an index of a module option array.
The recommended approach is as follows:
2016-07-20 21:20:02 +03:00
.. code-block :: c
2016-07-20 21:07:56 +03:00
static atomic_t drv_instance = ATOMIC_INIT(0);
static int drv_probe(struct pci_dev *pdev, const struct pci_device_id * pci_id)
{
...
state->instance = atomic_inc_return(&drv_instance) - 1;
}
If you have multiple device nodes then it can be difficult to know when it is
2016-07-20 21:20:02 +03:00
safe to unregister :c:type: `v4l2_device` for hotpluggable devices. For this
purpose :c:type: `v4l2_device` has refcounting support. The refcount is
2016-07-23 13:21:06 +03:00
increased whenever :c:func: `video_register_device` is called and it is
2016-07-20 21:20:02 +03:00
decreased whenever that device node is released. When the refcount reaches
zero, then the :c:type: `v4l2_device` release() callback is called. You can
do your final cleanup there.
2016-07-20 21:07:56 +03:00
If other device nodes (e.g. ALSA) are created, then you can increase and
decrease the refcount manually as well by calling:
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_get`
2016-07-20 21:20:02 +03:00
(:c:type: `v4l2_dev <v4l2_device>` ).
2016-07-20 21:07:56 +03:00
or:
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_put`
2016-07-20 21:20:02 +03:00
(:c:type: `v4l2_dev <v4l2_device>` ).
2016-07-20 21:07:56 +03:00
2016-07-20 21:20:02 +03:00
Since the initial refcount is 1 you also need to call
2016-07-23 13:21:06 +03:00
:c:func: `v4l2_device_put` in the `` disconnect() `` callback (for USB devices)
2016-07-20 21:20:02 +03:00
or in the `` remove() `` callback (for e.g. PCI devices), otherwise the refcount
will never reach 0.
2016-07-20 21:07:56 +03:00
2016-07-22 16:15:42 +03:00
v4l2_device functions and data structures
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
2016-07-20 20:14:37 +03:00
.. kernel-doc :: include/media/v4l2-device.h