Re: [PATCH 3/8] usb: gadget: uvc: configfs: Allocate groups dynamically

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Laurent,

Thank you for the patch,

On 01/08/18 01:29, Laurent Pinchart wrote:
> The UVC configfs implementation creates all groups as global static
> variables. This prevents creationg of multiple UVC function instances,

/creationg/creation/

> as they would all require their own configfs group instances.
> 
> Fix this by allocating all groups dynamically. To avoid duplicating code
> around, extend the config_item_type structure with group name and
> children, and implement helper functions to create children
> automatically for most groups.
> 
> Signed-off-by: Laurent Pinchart <laurent.pinchart@xxxxxxxxxxxxxxxx>

I'm struggling to see what paths free all of the dynamically created
children in this patch.

Is this already supported by the config_group framework?

I see a reference to config_group_put(&opts->func_inst.group); in one
error path - but that's about it.

Am I missing something nice and obvious? (or is it already handled by
framework code not in this patch)


In fact, I can't see how it could be handled by core - as the children
are added to a new structure you have created.

I'll let you look into this :)



> ---
>  drivers/usb/gadget/function/f_uvc.c        |   8 +-
>  drivers/usb/gadget/function/uvc_configfs.c | 480 +++++++++++++++++------------
>  2 files changed, 282 insertions(+), 206 deletions(-)
> 
> diff --git a/drivers/usb/gadget/function/f_uvc.c b/drivers/usb/gadget/function/f_uvc.c
> index d8ce7868fe22..95cb1b5f5ffe 100644
> --- a/drivers/usb/gadget/function/f_uvc.c
> +++ b/drivers/usb/gadget/function/f_uvc.c
> @@ -792,6 +792,7 @@ static struct usb_function_instance *uvc_alloc_inst(void)
>  	struct uvc_output_terminal_descriptor *od;
>  	struct uvc_color_matching_descriptor *md;
>  	struct uvc_descriptor_header **ctl_cls;
> +	int ret;
>  
>  	opts = kzalloc(sizeof(*opts), GFP_KERNEL);
>  	if (!opts)
> @@ -868,7 +869,12 @@ static struct usb_function_instance *uvc_alloc_inst(void)
>  	opts->streaming_interval = 1;
>  	opts->streaming_maxpacket = 1024;
>  
> -	uvcg_attach_configfs(opts);
> +	ret = uvcg_attach_configfs(opts);
> +	if (ret < 0) {
> +		kfree(opts);
> +		return ERR_PTR(ret);
> +	}
> +
>  	return &opts->func_inst;
>  }
>  
> diff --git a/drivers/usb/gadget/function/uvc_configfs.c b/drivers/usb/gadget/function/uvc_configfs.c
> index dbc95c9558de..e019ea317c7a 100644
> --- a/drivers/usb/gadget/function/uvc_configfs.c
> +++ b/drivers/usb/gadget/function/uvc_configfs.c
> @@ -41,6 +41,49 @@ static inline struct f_uvc_opts *to_f_uvc_opts(struct config_item *item)
>  			    func_inst.group);
>  }
>  
> +struct uvcg_config_group_type {
> +	struct config_item_type type;
> +	const char *name;
> +	const struct uvcg_config_group_type **children;
> +	int (*create_children)(struct config_group *group);
> +};
> +
> +static int uvcg_config_create_group(struct config_group *parent,
> +				    const struct uvcg_config_group_type *type);
> +
> +static int uvcg_config_create_children(struct config_group *group,
> +				const struct uvcg_config_group_type *type)
> +{
> +	const struct uvcg_config_group_type **child;
> +	int ret;
> +
> +	if (type->create_children)
> +		return type->create_children(group);
> +
> +	for (child = type->children; child && *child; ++child) {
> +		ret = uvcg_config_create_group(group, *child);
> +		if (ret < 0)
> +			return ret;
> +	}
> +
> +	return 0;
> +}
> +
> +static int uvcg_config_create_group(struct config_group *parent,
> +				    const struct uvcg_config_group_type *type)
> +{
> +	struct config_group *group;
> +
> +	group = kzalloc(sizeof(*group), GFP_KERNEL);
> +	if (!group)
> +		return -ENOMEM;
> +
> +	config_group_init_type_name(group, type->name, &type->type);
> +	configfs_add_default_group(group, parent);
> +
> +	return uvcg_config_create_children(group, type);
> +}
> +
>  /* -----------------------------------------------------------------------------
>   * control/header/<NAME>
>   * control/header
> @@ -169,24 +212,23 @@ static void uvcg_control_header_drop(struct config_group *group,
>  	kfree(h);
>  }
>  
> -static struct config_group uvcg_control_header_grp;
> -
>  static struct configfs_group_operations uvcg_control_header_grp_ops = {
>  	.make_item		= uvcg_control_header_make,
>  	.drop_item		= uvcg_control_header_drop,
>  };
>  
> -static const struct config_item_type uvcg_control_header_grp_type = {
> -	.ct_group_ops	= &uvcg_control_header_grp_ops,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_control_header_grp_type = {
> +	.type = {
> +		.ct_group_ops	= &uvcg_control_header_grp_ops,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "header",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/processing/default
>   */
>  
> -static struct config_group uvcg_default_processing_grp;
> -
>  #define UVCG_DEFAULT_PROCESSING_ATTR(cname, aname, conv)		\
>  static ssize_t uvcg_default_processing_##cname##_show(			\
>  	struct config_item *item, char *page)				\
> @@ -265,27 +307,33 @@ static struct configfs_attribute *uvcg_default_processing_attrs[] = {
>  	NULL,
>  };
>  
> -static const struct config_item_type uvcg_default_processing_type = {
> -	.ct_attrs	= uvcg_default_processing_attrs,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_default_processing_type = {
> +	.type = {
> +		.ct_attrs	= uvcg_default_processing_attrs,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "default",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/processing
>   */
>  
> -static struct config_group uvcg_processing_grp;
> -
> -static const struct config_item_type uvcg_processing_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_processing_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "processing",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_default_processing_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/terminal/camera/default
>   */
>  
> -static struct config_group uvcg_default_camera_grp;
> -
>  #define UVCG_DEFAULT_CAMERA_ATTR(cname, aname, conv)			\
>  static ssize_t uvcg_default_camera_##cname##_show(			\
>  	struct config_item *item, char *page)				\
> @@ -375,27 +423,33 @@ static struct configfs_attribute *uvcg_default_camera_attrs[] = {
>  	NULL,
>  };
>  
> -static const struct config_item_type uvcg_default_camera_type = {
> -	.ct_attrs	= uvcg_default_camera_attrs,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_default_camera_type = {
> +	.type = {
> +		.ct_attrs	= uvcg_default_camera_attrs,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "default",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/terminal/camera
>   */
>  
> -static struct config_group uvcg_camera_grp;
> -
> -static const struct config_item_type uvcg_camera_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_camera_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "camera",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_default_camera_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/terminal/output/default
>   */
>  
> -static struct config_group uvcg_default_output_grp;
> -
>  #define UVCG_DEFAULT_OUTPUT_ATTR(cname, aname, conv)			\
>  static ssize_t uvcg_default_output_##cname##_show(			\
>  	struct config_item *item, char *page)				\
> @@ -446,47 +500,65 @@ static struct configfs_attribute *uvcg_default_output_attrs[] = {
>  	NULL,
>  };
>  
> -static const struct config_item_type uvcg_default_output_type = {
> -	.ct_attrs	= uvcg_default_output_attrs,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_default_output_type = {
> +	.type = {
> +		.ct_attrs	= uvcg_default_output_attrs,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "default",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/terminal/output
>   */
>  
> -static struct config_group uvcg_output_grp;
> -
> -static const struct config_item_type uvcg_output_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_output_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "output",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_default_output_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/terminal
>   */
>  
> -static struct config_group uvcg_terminal_grp;
> -
> -static const struct config_item_type uvcg_terminal_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_terminal_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "terminal",
> +	.children = (const struct uvcg_config_group_type*[]) {

Is this cast really needed? Or is it just to constify the array ?

> +		&uvcg_camera_grp_type,
> +		&uvcg_output_grp_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control/class/{fs|ss}
>   */
>  
> -static struct config_group uvcg_control_class_fs_grp;
> -static struct config_group uvcg_control_class_ss_grp;
> +struct uvcg_control_class_group {
> +	struct config_group group;
> +	const char *name;
> +};
>  
>  static inline struct uvc_descriptor_header
>  **uvcg_get_ctl_class_arr(struct config_item *i, struct f_uvc_opts *o)
>  {
> -	struct config_group *group = to_config_group(i);
> +	struct uvcg_control_class_group *group =
> +		container_of(i, struct uvcg_control_class_group,
> +			     group.cg_item);
>  
> -	if (group == &uvcg_control_class_fs_grp)
> +	if (!strcmp(group->name, "fs"))
>  		return o->uvc_fs_control_cls;
>  
> -	if (group == &uvcg_control_class_ss_grp)
> +	if (!strcmp(group->name, "ss"))
>  		return o->uvc_ss_control_cls;
>  
>  	return NULL;
> @@ -581,20 +653,52 @@ static const struct config_item_type uvcg_control_class_type = {
>   * control/class
>   */
>  
> -static struct config_group uvcg_control_class_grp;
> +static int uvcg_control_class_create_children(struct config_group *parent)
> +{
> +	static const char * const names[] = { "fs", "ss" };
> +	unsigned int i;
> +
> +	for (i = 0; i < ARRAY_SIZE(names); ++i) {
> +		struct uvcg_control_class_group *group;
> +
> +		group = kzalloc(sizeof(*group), GFP_KERNEL);
> +		if (!group)
> +			return -ENOMEM;
> +
> +		group->name = names[i];
>  
> -static const struct config_item_type uvcg_control_class_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +		config_group_init_type_name(&group->group, group->name,
> +					    &uvcg_control_class_type);
> +		configfs_add_default_group(&group->group, parent);
> +	}
> +
> +	return 0;
> +}
> +
> +static const struct uvcg_config_group_type uvcg_control_class_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "class",
> +	.create_children = uvcg_control_class_create_children,
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * control
>   */
>  
> -static struct config_group uvcg_control_grp;
> -
> -static const struct config_item_type uvcg_control_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_control_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "control",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_control_header_grp_type,
> +		&uvcg_processing_grp_type,
> +		&uvcg_terminal_grp_type,
> +		&uvcg_control_class_grp_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
> @@ -602,12 +706,9 @@ static const struct config_item_type uvcg_control_grp_type = {
>   * streaming/mjpeg
>   */
>  
> -static struct config_group uvcg_uncompressed_grp;
> -static struct config_group uvcg_mjpeg_grp;
> -
> -static struct config_item *fmt_parent[] = {
> -	&uvcg_uncompressed_grp.cg_item,
> -	&uvcg_mjpeg_grp.cg_item,
> +static const char * const uvcg_format_names[] = {
> +	"uncompressed",
> +	"mjpeg",
>  };
>  
>  enum uvcg_format_type {
> @@ -733,10 +834,22 @@ static int uvcg_streaming_header_allow_link(struct config_item *src,
>  		goto out;
>  	}
>  
> -	for (i = 0; i < ARRAY_SIZE(fmt_parent); ++i)
> -		if (target->ci_parent == fmt_parent[i])
> +	/*
> +	 * Linking is only allowed to direct children of the format nodes
> +	 * (streaming/uncompressed or streaming/mjpeg nodes). First check that
> +	 * the grand-parent of the target matches the grand-parent of the source
> +	 * (the streaming node), and then verify that the target parent is a
> +	 * format node.
> +	 */
> +	if (src->ci_parent->ci_parent != target->ci_parent->ci_parent)
> +		goto out;
> +
> +	for (i = 0; i < ARRAY_SIZE(uvcg_format_names); ++i) {
> +		if (!strcmp(target->ci_parent->ci_name, uvcg_format_names[i]))
>  			break;
> -	if (i == ARRAY_SIZE(fmt_parent))
> +	}
> +
> +	if (i == ARRAY_SIZE(uvcg_format_names))
>  		goto out;
>  
>  	target_fmt = container_of(to_config_group(target), struct uvcg_format,
> @@ -881,16 +994,17 @@ static void uvcg_streaming_header_drop(struct config_group *group,
>  	kfree(h);
>  }
>  
> -static struct config_group uvcg_streaming_header_grp;
> -
>  static struct configfs_group_operations uvcg_streaming_header_grp_ops = {
>  	.make_item		= uvcg_streaming_header_make,
>  	.drop_item		= uvcg_streaming_header_drop,
>  };
>  
> -static const struct config_item_type uvcg_streaming_header_grp_type = {
> -	.ct_group_ops	= &uvcg_streaming_header_grp_ops,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_streaming_header_grp_type = {
> +	.type = {
> +		.ct_group_ops	= &uvcg_streaming_header_grp_ops,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "header",
>  };
>  
>  /* -----------------------------------------------------------------------------
> @@ -1462,9 +1576,12 @@ static struct configfs_group_operations uvcg_uncompressed_grp_ops = {
>  	.drop_item		= uvcg_uncompressed_drop,
>  };
>  
> -static const struct config_item_type uvcg_uncompressed_grp_type = {
> -	.ct_group_ops	= &uvcg_uncompressed_grp_ops,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_uncompressed_grp_type = {
> +	.type = {
> +		.ct_group_ops	= &uvcg_uncompressed_grp_ops,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "uncompressed",
>  };
>  
>  /* -----------------------------------------------------------------------------
> @@ -1659,17 +1776,18 @@ static struct configfs_group_operations uvcg_mjpeg_grp_ops = {
>  	.drop_item		= uvcg_mjpeg_drop,
>  };
>  
> -static const struct config_item_type uvcg_mjpeg_grp_type = {
> -	.ct_group_ops	= &uvcg_mjpeg_grp_ops,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_mjpeg_grp_type = {
> +	.type = {
> +		.ct_group_ops	= &uvcg_mjpeg_grp_ops,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "mjpeg",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * streaming/color_matching/default
>   */
>  
> -static struct config_group uvcg_default_color_matching_grp;
> -
>  #define UVCG_DEFAULT_COLOR_MATCHING_ATTR(cname, aname, conv)		\
>  static ssize_t uvcg_default_color_matching_##cname##_show(		\
>  	struct config_item *item, char *page)			\
> @@ -1717,41 +1835,52 @@ static struct configfs_attribute *uvcg_default_color_matching_attrs[] = {
>  	NULL,
>  };
>  
> -static const struct config_item_type uvcg_default_color_matching_type = {
> -	.ct_attrs	= uvcg_default_color_matching_attrs,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_default_color_matching_type = {
> +	.type = {
> +		.ct_attrs	= uvcg_default_color_matching_attrs,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "default",
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * streaming/color_matching
>   */
>  
> -static struct config_group uvcg_color_matching_grp;
> -
> -static const struct config_item_type uvcg_color_matching_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_color_matching_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "color_matching",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_default_color_matching_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * streaming/class/{fs|hs|ss}
>   */
>  
> -static struct config_group uvcg_streaming_class_fs_grp;
> -static struct config_group uvcg_streaming_class_hs_grp;
> -static struct config_group uvcg_streaming_class_ss_grp;
> +struct uvcg_streaming_class_group {
> +	struct config_group group;
> +	const char *name;
> +};
>  
>  static inline struct uvc_descriptor_header
>  ***__uvcg_get_stream_class_arr(struct config_item *i, struct f_uvc_opts *o)
>  {
> -	struct config_group *group = to_config_group(i);
> +	struct uvcg_streaming_class_group *group =
> +		container_of(i, struct uvcg_streaming_class_group,
> +			     group.cg_item);
>  
> -	if (group == &uvcg_streaming_class_fs_grp)
> +	if (!strcmp(group->name, "fs"))
>  		return &o->uvc_fs_streaming_cls;
>  
> -	if (group == &uvcg_streaming_class_hs_grp)
> +	if (!strcmp(group->name, "hs"))
>  		return &o->uvc_hs_streaming_cls;
>  
> -	if (group == &uvcg_streaming_class_ss_grp)
> +	if (!strcmp(group->name, "ss"))
>  		return &o->uvc_ss_streaming_cls;
>  
>  	return NULL;
> @@ -2083,20 +2212,53 @@ static const struct config_item_type uvcg_streaming_class_type = {
>   * streaming/class
>   */
>  
> -static struct config_group uvcg_streaming_class_grp;
> +static int uvcg_streaming_class_create_children(struct config_group *parent)
> +{
> +	static const char * const names[] = { "fs", "hs", "ss" };
> +	unsigned int i;
> +
> +	for (i = 0; i < ARRAY_SIZE(names); ++i) {
> +		struct uvcg_streaming_class_group *group;
> +
> +		group = kzalloc(sizeof(*group), GFP_KERNEL);
> +		if (!group)
> +			return -ENOMEM;
> +
> +		group->name = names[i];
> +
> +		config_group_init_type_name(&group->group, group->name,
> +					    &uvcg_streaming_class_type);
> +		configfs_add_default_group(&group->group, parent);
> +	}
> +
> +	return 0;
> +}
>  
> -static const struct config_item_type uvcg_streaming_class_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_streaming_class_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "class",
> +	.create_children = uvcg_streaming_class_create_children,
>  };
>  
>  /* -----------------------------------------------------------------------------
>   * streaming
>   */
>  
> -static struct config_group uvcg_streaming_grp;
> -
> -static const struct config_item_type uvcg_streaming_grp_type = {
> -	.ct_owner = THIS_MODULE,
> +static const struct uvcg_config_group_type uvcg_streaming_grp_type = {
> +	.type = {
> +		.ct_owner = THIS_MODULE,
> +	},
> +	.name = "streaming",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_streaming_header_grp_type,
> +		&uvcg_uncompressed_grp_type,
> +		&uvcg_mjpeg_grp_type,
> +		&uvcg_color_matching_grp_type,
> +		&uvcg_streaming_class_grp_type,
> +		NULL,
> +	},
>  };
>  
>  /* -----------------------------------------------------------------------------
> @@ -2179,123 +2341,31 @@ static struct configfs_attribute *uvc_attrs[] = {
>  	NULL,
>  };
>  
> -static const struct config_item_type uvc_func_type = {
> -	.ct_item_ops	= &uvc_item_ops,
> -	.ct_attrs	= uvc_attrs,
> -	.ct_owner	= THIS_MODULE,
> +static const struct uvcg_config_group_type uvc_func_type = {
> +	.type = {
> +		.ct_item_ops	= &uvc_item_ops,
> +		.ct_attrs	= uvc_attrs,
> +		.ct_owner	= THIS_MODULE,
> +	},
> +	.name = "",
> +	.children = (const struct uvcg_config_group_type*[]) {
> +		&uvcg_control_grp_type,
> +		&uvcg_streaming_grp_type,
> +		NULL,
> +	},
>  };
>  
>  int uvcg_attach_configfs(struct f_uvc_opts *opts)
>  {
> -	config_group_init_type_name(&uvcg_control_header_grp,
> -				    "header",
> -				    &uvcg_control_header_grp_type);
> -
> -	config_group_init_type_name(&uvcg_default_processing_grp,
> -			"default", &uvcg_default_processing_type);
> -	config_group_init_type_name(&uvcg_processing_grp,
> -			"processing", &uvcg_processing_grp_type);
> -	configfs_add_default_group(&uvcg_default_processing_grp,
> -			&uvcg_processing_grp);
> -
> -	config_group_init_type_name(&uvcg_default_camera_grp,
> -			"default", &uvcg_default_camera_type);
> -	config_group_init_type_name(&uvcg_camera_grp,
> -			"camera", &uvcg_camera_grp_type);
> -	configfs_add_default_group(&uvcg_default_camera_grp,
> -			&uvcg_camera_grp);
> -
> -	config_group_init_type_name(&uvcg_default_output_grp,
> -			"default", &uvcg_default_output_type);
> -	config_group_init_type_name(&uvcg_output_grp,
> -			"output", &uvcg_output_grp_type);
> -	configfs_add_default_group(&uvcg_default_output_grp,
> -			&uvcg_output_grp);
> -
> -	config_group_init_type_name(&uvcg_terminal_grp,
> -			"terminal", &uvcg_terminal_grp_type);
> -	configfs_add_default_group(&uvcg_camera_grp,
> -			&uvcg_terminal_grp);
> -	configfs_add_default_group(&uvcg_output_grp,
> -			&uvcg_terminal_grp);
> -
> -	config_group_init_type_name(&uvcg_control_class_fs_grp,
> -			"fs", &uvcg_control_class_type);
> -	config_group_init_type_name(&uvcg_control_class_ss_grp,
> -			"ss", &uvcg_control_class_type);
> -	config_group_init_type_name(&uvcg_control_class_grp,
> -			"class",
> -			&uvcg_control_class_grp_type);
> -	configfs_add_default_group(&uvcg_control_class_fs_grp,
> -			&uvcg_control_class_grp);
> -	configfs_add_default_group(&uvcg_control_class_ss_grp,
> -			&uvcg_control_class_grp);
> -
> -	config_group_init_type_name(&uvcg_control_grp,
> -			"control",
> -			&uvcg_control_grp_type);
> -	configfs_add_default_group(&uvcg_control_header_grp,
> -			&uvcg_control_grp);
> -	configfs_add_default_group(&uvcg_processing_grp,
> -			&uvcg_control_grp);
> -	configfs_add_default_group(&uvcg_terminal_grp,
> -			&uvcg_control_grp);
> -	configfs_add_default_group(&uvcg_control_class_grp,
> -			&uvcg_control_grp);
> -
> -	config_group_init_type_name(&uvcg_streaming_header_grp,
> -				    "header",
> -				    &uvcg_streaming_header_grp_type);
> -	config_group_init_type_name(&uvcg_uncompressed_grp,
> -				    "uncompressed",
> -				    &uvcg_uncompressed_grp_type);
> -	config_group_init_type_name(&uvcg_mjpeg_grp,
> -				    "mjpeg",
> -				    &uvcg_mjpeg_grp_type);
> -	config_group_init_type_name(&uvcg_default_color_matching_grp,
> -				    "default",
> -				    &uvcg_default_color_matching_type);
> -	config_group_init_type_name(&uvcg_color_matching_grp,
> -			"color_matching",
> -			&uvcg_color_matching_grp_type);
> -	configfs_add_default_group(&uvcg_default_color_matching_grp,
> -			&uvcg_color_matching_grp);
> -
> -	config_group_init_type_name(&uvcg_streaming_class_fs_grp,
> -			"fs", &uvcg_streaming_class_type);
> -	config_group_init_type_name(&uvcg_streaming_class_hs_grp,
> -			"hs", &uvcg_streaming_class_type);
> -	config_group_init_type_name(&uvcg_streaming_class_ss_grp,
> -			"ss", &uvcg_streaming_class_type);
> -	config_group_init_type_name(&uvcg_streaming_class_grp,
> -			"class", &uvcg_streaming_class_grp_type);
> -	configfs_add_default_group(&uvcg_streaming_class_fs_grp,
> -			&uvcg_streaming_class_grp);
> -	configfs_add_default_group(&uvcg_streaming_class_hs_grp,
> -			&uvcg_streaming_class_grp);
> -	configfs_add_default_group(&uvcg_streaming_class_ss_grp,
> -			&uvcg_streaming_class_grp);
> -
> -	config_group_init_type_name(&uvcg_streaming_grp,
> -			"streaming", &uvcg_streaming_grp_type);
> -	configfs_add_default_group(&uvcg_streaming_header_grp,
> -			&uvcg_streaming_grp);
> -	configfs_add_default_group(&uvcg_uncompressed_grp,
> -			&uvcg_streaming_grp);
> -	configfs_add_default_group(&uvcg_mjpeg_grp,
> -			&uvcg_streaming_grp);
> -	configfs_add_default_group(&uvcg_color_matching_grp,
> -			&uvcg_streaming_grp);
> -	configfs_add_default_group(&uvcg_streaming_class_grp,
> -			&uvcg_streaming_grp);
> -
> -	config_group_init_type_name(&opts->func_inst.group,
> -			"",
> -			&uvc_func_type);
> -	configfs_add_default_group(&uvcg_control_grp,
> -			&opts->func_inst.group);
> -	configfs_add_default_group(&uvcg_streaming_grp,
> -			&opts->func_inst.group);
> +	int ret;
>  
> -	return 0;
> +	config_group_init_type_name(&opts->func_inst.group, uvc_func_type.name,
> +				    &uvc_func_type.type);
> +
> +	ret = uvcg_config_create_children(&opts->func_inst.group,
> +					  &uvc_func_type);
> +	if (ret < 0)
> +		config_group_put(&opts->func_inst.group);
> +
> +	return ret;
>  }
> 

-- 
Regards
--
Kieran
--
To unsubscribe from this list: send the line "unsubscribe linux-usb" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html



[Index of Archives]     [Linux Media]     [Linux Input]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]     [Old Linux USB Devel Archive]

  Powered by Linux