forked from mirrors/linux
'struct config_item_type' is not modified in this file. Apparently, these structures are only used with config_group_init_type_name() which takes a const struct config_item_type* as a 3rd argument. Constifying this structure moves some data to a read-only section, so increase overall security, especially when the structure holds some function pointers. On a x86_64, with allmodconfig: Before: ====== text data bss dec hex filename 40834 5112 64 46010 b3ba drivers/usb/gadget/configfs.o After: ===== text data bss dec hex filename 41218 4728 64 46010 b3ba drivers/usb/gadget/configfs.o Signed-off-by: Christophe JAILLET <christophe.jaillet@wanadoo.fr> Link: https://lore.kernel.org/r/513223e97082e1bb758e36d55c175ec9ea34a71c.1723323896.git.christophe.jaillet@wanadoo.fr Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
99 lines
2.9 KiB
C
99 lines
2.9 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef __GADGET_CONFIGFS__
|
|
#define __GADGET_CONFIGFS__
|
|
|
|
#include <linux/configfs.h>
|
|
|
|
#define GS_STRINGS_W(__struct, __name) \
|
|
static ssize_t __struct##_##__name##_store(struct config_item *item, \
|
|
const char *page, size_t len) \
|
|
{ \
|
|
struct __struct *gs = to_##__struct(item); \
|
|
int ret; \
|
|
\
|
|
ret = usb_string_copy(page, &gs->__name); \
|
|
if (ret) \
|
|
return ret; \
|
|
return len; \
|
|
}
|
|
|
|
#define GS_STRINGS_R(__struct, __name) \
|
|
static ssize_t __struct##_##__name##_show(struct config_item *item, char *page) \
|
|
{ \
|
|
struct __struct *gs = to_##__struct(item); \
|
|
return sprintf(page, "%s\n", gs->__name ?: ""); \
|
|
}
|
|
|
|
#define GS_STRINGS_RW(struct_name, _name) \
|
|
GS_STRINGS_R(struct_name, _name) \
|
|
GS_STRINGS_W(struct_name, _name) \
|
|
CONFIGFS_ATTR(struct_name##_, _name)
|
|
|
|
#define USB_CONFIG_STRING_RW_OPS(struct_in) \
|
|
static struct configfs_item_operations struct_in##_langid_item_ops = { \
|
|
.release = struct_in##_attr_release, \
|
|
}; \
|
|
\
|
|
static const struct config_item_type struct_in##_langid_type = { \
|
|
.ct_item_ops = &struct_in##_langid_item_ops, \
|
|
.ct_attrs = struct_in##_langid_attrs, \
|
|
.ct_owner = THIS_MODULE, \
|
|
}
|
|
|
|
#define USB_CONFIG_STRINGS_LANG(struct_in, struct_member) \
|
|
static struct config_group *struct_in##_strings_make( \
|
|
struct config_group *group, \
|
|
const char *name) \
|
|
{ \
|
|
struct struct_member *gi; \
|
|
struct struct_in *gs; \
|
|
struct struct_in *new; \
|
|
int langs = 0; \
|
|
int ret; \
|
|
\
|
|
new = kzalloc(sizeof(*new), GFP_KERNEL); \
|
|
if (!new) \
|
|
return ERR_PTR(-ENOMEM); \
|
|
\
|
|
ret = check_user_usb_string(name, &new->stringtab_dev); \
|
|
if (ret) \
|
|
goto err; \
|
|
config_group_init_type_name(&new->group, name, \
|
|
&struct_in##_langid_type); \
|
|
\
|
|
gi = container_of(group, struct struct_member, strings_group); \
|
|
ret = -EEXIST; \
|
|
list_for_each_entry(gs, &gi->string_list, list) { \
|
|
if (gs->stringtab_dev.language == new->stringtab_dev.language) \
|
|
goto err; \
|
|
langs++; \
|
|
} \
|
|
ret = -EOVERFLOW; \
|
|
if (langs >= MAX_USB_STRING_LANGS) \
|
|
goto err; \
|
|
\
|
|
list_add_tail(&new->list, &gi->string_list); \
|
|
return &new->group; \
|
|
err: \
|
|
kfree(new); \
|
|
return ERR_PTR(ret); \
|
|
} \
|
|
\
|
|
static void struct_in##_strings_drop( \
|
|
struct config_group *group, \
|
|
struct config_item *item) \
|
|
{ \
|
|
config_item_put(item); \
|
|
} \
|
|
\
|
|
static struct configfs_group_operations struct_in##_strings_ops = { \
|
|
.make_group = &struct_in##_strings_make, \
|
|
.drop_item = &struct_in##_strings_drop, \
|
|
}; \
|
|
\
|
|
static const struct config_item_type struct_in##_strings_type = { \
|
|
.ct_group_ops = &struct_in##_strings_ops, \
|
|
.ct_owner = THIS_MODULE, \
|
|
}
|
|
|
|
#endif
|