/* SPDX-License-Identifier: GPL-2.0-only */ /* * Mediated device definition * * Copyright (c) 2016, NVIDIA CORPORATION. All rights reserved. * Author: Neo Jia * Kirti Wankhede */ #ifndef MDEV_H #define MDEV_H #include #include struct mdev_type; struct mdev_device { struct device dev; guid_t uuid; struct list_head next; struct mdev_type *type; bool active; }; struct mdev_type { /* set by the driver before calling mdev_register parent: */ const char *sysfs_name; const char *pretty_name; /* set by the core, can be used drivers */ struct mdev_parent *parent; /* internal only */ struct kobject kobj; struct kobject *devices_kobj; }; /* embedded into the struct device that the mdev devices hang off */ struct mdev_parent { struct device *dev; struct mdev_driver *mdev_driver; struct kset *mdev_types_kset; /* Synchronize device creation/removal with parent unregistration */ struct rw_semaphore unreg_sem; struct mdev_type **types; unsigned int nr_types; }; static inline struct mdev_device *to_mdev_device(struct device *dev) { return container_of(dev, struct mdev_device, dev); } /* interface for exporting mdev supported type attributes */ struct mdev_type_attribute { struct attribute attr; ssize_t (*show)(struct mdev_type *mtype, struct mdev_type_attribute *attr, char *buf); ssize_t (*store)(struct mdev_type *mtype, struct mdev_type_attribute *attr, const char *buf, size_t count); }; #define MDEV_TYPE_ATTR_RO(_name) \ struct mdev_type_attribute mdev_type_attr_##_name = __ATTR_RO(_name) #define MDEV_TYPE_ATTR_WO(_name) \ struct mdev_type_attribute mdev_type_attr_##_name = __ATTR_WO(_name) /** * struct mdev_driver - Mediated device driver * @device_api: string to return for the device_api sysfs * @probe: called when new device created * @remove: called when device removed * @get_available: Return the max number of instances that can be created * @types_attrs: attributes to the type kobjects. * @driver: device driver structure **/ struct mdev_driver { const char *device_api; int (*probe)(struct mdev_device *dev); void (*remove)(struct mdev_device *dev); unsigned int (*get_available)(struct mdev_type *mtype); const struct attribute * const *types_attrs; struct device_driver driver; }; int mdev_register_parent(struct mdev_parent *parent, struct device *dev, struct mdev_driver *mdev_driver, struct mdev_type **types, unsigned int nr_types); void mdev_unregister_parent(struct mdev_parent *parent); int mdev_register_driver(struct mdev_driver *drv); void mdev_unregister_driver(struct mdev_driver *drv); static inline struct device *mdev_dev(struct mdev_device *mdev) { return &mdev->dev; } #endif /* MDEV_H */