4 * Copyright (C) 2010 Nokia Corporation
6 * Contacts: Laurent Pinchart <laurent.pinchart@ideasonboard.com>
7 * Sakari Ailus <sakari.ailus@iki.fi>
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License version 2 as
11 * published by the Free Software Foundation.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
23 #include <linux/compat.h>
24 #include <linux/export.h>
25 #include <linux/ioctl.h>
26 #include <linux/media.h>
27 #include <linux/types.h>
29 #include <media/media-device.h>
30 #include <media/media-devnode.h>
31 #include <media/media-entity.h>
33 /* -----------------------------------------------------------------------------
37 static int media_device_open(struct file
*filp
)
42 static int media_device_close(struct file
*filp
)
47 static int media_device_get_info(struct media_device
*dev
,
48 struct media_device_info __user
*__info
)
50 struct media_device_info info
;
52 memset(&info
, 0, sizeof(info
));
54 strlcpy(info
.driver
, dev
->dev
->driver
->name
, sizeof(info
.driver
));
55 strlcpy(info
.model
, dev
->model
, sizeof(info
.model
));
56 strlcpy(info
.serial
, dev
->serial
, sizeof(info
.serial
));
57 strlcpy(info
.bus_info
, dev
->bus_info
, sizeof(info
.bus_info
));
59 info
.media_version
= MEDIA_API_VERSION
;
60 info
.hw_revision
= dev
->hw_revision
;
61 info
.driver_version
= dev
->driver_version
;
63 if (copy_to_user(__info
, &info
, sizeof(*__info
)))
68 static struct media_entity
*find_entity(struct media_device
*mdev
, u32 id
)
70 struct media_entity
*entity
;
71 int next
= id
& MEDIA_ENT_ID_FLAG_NEXT
;
73 id
&= ~MEDIA_ENT_ID_FLAG_NEXT
;
75 spin_lock(&mdev
->lock
);
77 media_device_for_each_entity(entity
, mdev
) {
78 if ((entity
->id
== id
&& !next
) ||
79 (entity
->id
> id
&& next
)) {
80 spin_unlock(&mdev
->lock
);
85 spin_unlock(&mdev
->lock
);
90 static long media_device_enum_entities(struct media_device
*mdev
,
91 struct media_entity_desc __user
*uent
)
93 struct media_entity
*ent
;
94 struct media_entity_desc u_ent
;
96 memset(&u_ent
, 0, sizeof(u_ent
));
97 if (copy_from_user(&u_ent
.id
, &uent
->id
, sizeof(u_ent
.id
)))
100 ent
= find_entity(mdev
, u_ent
.id
);
107 strncpy(u_ent
.name
, ent
->name
, sizeof(u_ent
.name
));
108 u_ent
.name
[sizeof(u_ent
.name
) - 1] = '\0';
110 memset(u_ent
.name
, 0, sizeof(u_ent
.name
));
112 u_ent
.type
= ent
->type
;
113 u_ent
.revision
= ent
->revision
;
114 u_ent
.flags
= ent
->flags
;
115 u_ent
.group_id
= ent
->group_id
;
116 u_ent
.pads
= ent
->num_pads
;
117 u_ent
.links
= ent
->num_links
- ent
->num_backlinks
;
118 memcpy(&u_ent
.raw
, &ent
->info
, sizeof(ent
->info
));
119 if (copy_to_user(uent
, &u_ent
, sizeof(u_ent
)))
124 static void media_device_kpad_to_upad(const struct media_pad
*kpad
,
125 struct media_pad_desc
*upad
)
127 upad
->entity
= kpad
->entity
->id
;
128 upad
->index
= kpad
->index
;
129 upad
->flags
= kpad
->flags
;
132 static long __media_device_enum_links(struct media_device
*mdev
,
133 struct media_links_enum
*links
)
135 struct media_entity
*entity
;
137 entity
= find_entity(mdev
, links
->entity
);
144 for (p
= 0; p
< entity
->num_pads
; p
++) {
145 struct media_pad_desc pad
;
147 memset(&pad
, 0, sizeof(pad
));
148 media_device_kpad_to_upad(&entity
->pads
[p
], &pad
);
149 if (copy_to_user(&links
->pads
[p
], &pad
, sizeof(pad
)))
155 struct media_link_desc __user
*ulink
;
158 for (l
= 0, ulink
= links
->links
; l
< entity
->num_links
; l
++) {
159 struct media_link_desc link
;
161 /* Ignore backlinks. */
162 if (entity
->links
[l
].source
->entity
!= entity
)
165 memset(&link
, 0, sizeof(link
));
166 media_device_kpad_to_upad(entity
->links
[l
].source
,
168 media_device_kpad_to_upad(entity
->links
[l
].sink
,
170 link
.flags
= entity
->links
[l
].flags
;
171 if (copy_to_user(ulink
, &link
, sizeof(*ulink
)))
180 static long media_device_enum_links(struct media_device
*mdev
,
181 struct media_links_enum __user
*ulinks
)
183 struct media_links_enum links
;
186 if (copy_from_user(&links
, ulinks
, sizeof(links
)))
189 rval
= __media_device_enum_links(mdev
, &links
);
193 if (copy_to_user(ulinks
, &links
, sizeof(*ulinks
)))
199 static long media_device_setup_link(struct media_device
*mdev
,
200 struct media_link_desc __user
*_ulink
)
202 struct media_link
*link
= NULL
;
203 struct media_link_desc ulink
;
204 struct media_entity
*source
;
205 struct media_entity
*sink
;
208 if (copy_from_user(&ulink
, _ulink
, sizeof(ulink
)))
211 /* Find the source and sink entities and link.
213 source
= find_entity(mdev
, ulink
.source
.entity
);
214 sink
= find_entity(mdev
, ulink
.sink
.entity
);
216 if (source
== NULL
|| sink
== NULL
)
219 if (ulink
.source
.index
>= source
->num_pads
||
220 ulink
.sink
.index
>= sink
->num_pads
)
223 link
= media_entity_find_link(&source
->pads
[ulink
.source
.index
],
224 &sink
->pads
[ulink
.sink
.index
]);
228 /* Setup the link on both entities. */
229 ret
= __media_entity_setup_link(link
, ulink
.flags
);
231 if (copy_to_user(_ulink
, &ulink
, sizeof(ulink
)))
237 static long media_device_ioctl(struct file
*filp
, unsigned int cmd
,
240 struct media_devnode
*devnode
= media_devnode_data(filp
);
241 struct media_device
*dev
= to_media_device(devnode
);
245 case MEDIA_IOC_DEVICE_INFO
:
246 ret
= media_device_get_info(dev
,
247 (struct media_device_info __user
*)arg
);
250 case MEDIA_IOC_ENUM_ENTITIES
:
251 ret
= media_device_enum_entities(dev
,
252 (struct media_entity_desc __user
*)arg
);
255 case MEDIA_IOC_ENUM_LINKS
:
256 mutex_lock(&dev
->graph_mutex
);
257 ret
= media_device_enum_links(dev
,
258 (struct media_links_enum __user
*)arg
);
259 mutex_unlock(&dev
->graph_mutex
);
262 case MEDIA_IOC_SETUP_LINK
:
263 mutex_lock(&dev
->graph_mutex
);
264 ret
= media_device_setup_link(dev
,
265 (struct media_link_desc __user
*)arg
);
266 mutex_unlock(&dev
->graph_mutex
);
278 struct media_links_enum32
{
280 compat_uptr_t pads
; /* struct media_pad_desc * */
281 compat_uptr_t links
; /* struct media_link_desc * */
285 static long media_device_enum_links32(struct media_device
*mdev
,
286 struct media_links_enum32 __user
*ulinks
)
288 struct media_links_enum links
;
289 compat_uptr_t pads_ptr
, links_ptr
;
291 memset(&links
, 0, sizeof(links
));
293 if (get_user(links
.entity
, &ulinks
->entity
)
294 || get_user(pads_ptr
, &ulinks
->pads
)
295 || get_user(links_ptr
, &ulinks
->links
))
298 links
.pads
= compat_ptr(pads_ptr
);
299 links
.links
= compat_ptr(links_ptr
);
301 return __media_device_enum_links(mdev
, &links
);
304 #define MEDIA_IOC_ENUM_LINKS32 _IOWR('|', 0x02, struct media_links_enum32)
306 static long media_device_compat_ioctl(struct file
*filp
, unsigned int cmd
,
309 struct media_devnode
*devnode
= media_devnode_data(filp
);
310 struct media_device
*dev
= to_media_device(devnode
);
314 case MEDIA_IOC_DEVICE_INFO
:
315 case MEDIA_IOC_ENUM_ENTITIES
:
316 case MEDIA_IOC_SETUP_LINK
:
317 return media_device_ioctl(filp
, cmd
, arg
);
319 case MEDIA_IOC_ENUM_LINKS32
:
320 mutex_lock(&dev
->graph_mutex
);
321 ret
= media_device_enum_links32(dev
,
322 (struct media_links_enum32 __user
*)arg
);
323 mutex_unlock(&dev
->graph_mutex
);
332 #endif /* CONFIG_COMPAT */
334 static const struct media_file_operations media_device_fops
= {
335 .owner
= THIS_MODULE
,
336 .open
= media_device_open
,
337 .ioctl
= media_device_ioctl
,
339 .compat_ioctl
= media_device_compat_ioctl
,
340 #endif /* CONFIG_COMPAT */
341 .release
= media_device_close
,
344 /* -----------------------------------------------------------------------------
348 static ssize_t
show_model(struct device
*cd
,
349 struct device_attribute
*attr
, char *buf
)
351 struct media_device
*mdev
= to_media_device(to_media_devnode(cd
));
353 return sprintf(buf
, "%.*s\n", (int)sizeof(mdev
->model
), mdev
->model
);
356 static DEVICE_ATTR(model
, S_IRUGO
, show_model
, NULL
);
358 /* -----------------------------------------------------------------------------
359 * Registration/unregistration
362 static void media_device_release(struct media_devnode
*mdev
)
367 * media_device_register - register a media device
368 * @mdev: The media device
370 * The caller is responsible for initializing the media device before
371 * registration. The following fields must be set:
373 * - dev must point to the parent device
374 * - model must be filled with the device model name
376 int __must_check
media_device_register(struct media_device
*mdev
)
380 if (WARN_ON(mdev
->dev
== NULL
|| mdev
->model
[0] == 0))
384 INIT_LIST_HEAD(&mdev
->entities
);
385 spin_lock_init(&mdev
->lock
);
386 mutex_init(&mdev
->graph_mutex
);
388 /* Register the device node. */
389 mdev
->devnode
.fops
= &media_device_fops
;
390 mdev
->devnode
.parent
= mdev
->dev
;
391 mdev
->devnode
.release
= media_device_release
;
392 ret
= media_devnode_register(&mdev
->devnode
);
396 ret
= device_create_file(&mdev
->devnode
.dev
, &dev_attr_model
);
398 media_devnode_unregister(&mdev
->devnode
);
404 EXPORT_SYMBOL_GPL(media_device_register
);
407 * media_device_unregister - unregister a media device
408 * @mdev: The media device
411 void media_device_unregister(struct media_device
*mdev
)
413 struct media_entity
*entity
;
414 struct media_entity
*next
;
416 list_for_each_entry_safe(entity
, next
, &mdev
->entities
, list
)
417 media_device_unregister_entity(entity
);
419 device_remove_file(&mdev
->devnode
.dev
, &dev_attr_model
);
420 media_devnode_unregister(&mdev
->devnode
);
422 EXPORT_SYMBOL_GPL(media_device_unregister
);
425 * media_device_register_entity - Register an entity with a media device
426 * @mdev: The media device
427 * @entity: The entity
429 int __must_check
media_device_register_entity(struct media_device
*mdev
,
430 struct media_entity
*entity
)
432 /* Warn if we apparently re-register an entity */
433 WARN_ON(entity
->parent
!= NULL
);
434 entity
->parent
= mdev
;
436 spin_lock(&mdev
->lock
);
438 entity
->id
= mdev
->entity_id
++;
440 mdev
->entity_id
= max(entity
->id
+ 1, mdev
->entity_id
);
441 list_add_tail(&entity
->list
, &mdev
->entities
);
442 spin_unlock(&mdev
->lock
);
446 EXPORT_SYMBOL_GPL(media_device_register_entity
);
449 * media_device_unregister_entity - Unregister an entity
450 * @entity: The entity
452 * If the entity has never been registered this function will return
455 void media_device_unregister_entity(struct media_entity
*entity
)
457 struct media_device
*mdev
= entity
->parent
;
462 spin_lock(&mdev
->lock
);
463 list_del(&entity
->list
);
464 spin_unlock(&mdev
->lock
);
465 entity
->parent
= NULL
;
467 EXPORT_SYMBOL_GPL(media_device_unregister_entity
);