| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Netlink routines for CIFS | 
 |  * | 
 |  * Copyright (c) 2020 Samuel Cabrero <scabrero@suse.de> | 
 |  */ | 
 |  | 
 | #include <net/genetlink.h> | 
 | #include <uapi/linux/cifs/cifs_netlink.h> | 
 |  | 
 | #include "netlink.h" | 
 | #include "cifsglob.h" | 
 | #include "cifs_debug.h" | 
 | #include "cifs_swn.h" | 
 |  | 
 | static const struct nla_policy cifs_genl_policy[CIFS_GENL_ATTR_MAX + 1] = { | 
 | 	[CIFS_GENL_ATTR_SWN_REGISTRATION_ID]	= { .type = NLA_U32 }, | 
 | 	[CIFS_GENL_ATTR_SWN_NET_NAME]		= { .type = NLA_STRING }, | 
 | 	[CIFS_GENL_ATTR_SWN_SHARE_NAME]		= { .type = NLA_STRING }, | 
 | 	[CIFS_GENL_ATTR_SWN_IP]			= { .len = sizeof(struct sockaddr_storage) }, | 
 | 	[CIFS_GENL_ATTR_SWN_NET_NAME_NOTIFY]	= { .type = NLA_FLAG }, | 
 | 	[CIFS_GENL_ATTR_SWN_SHARE_NAME_NOTIFY]	= { .type = NLA_FLAG }, | 
 | 	[CIFS_GENL_ATTR_SWN_IP_NOTIFY]		= { .type = NLA_FLAG }, | 
 | 	[CIFS_GENL_ATTR_SWN_KRB_AUTH]		= { .type = NLA_FLAG }, | 
 | 	[CIFS_GENL_ATTR_SWN_USER_NAME]		= { .type = NLA_STRING }, | 
 | 	[CIFS_GENL_ATTR_SWN_PASSWORD]		= { .type = NLA_STRING }, | 
 | 	[CIFS_GENL_ATTR_SWN_DOMAIN_NAME]	= { .type = NLA_STRING }, | 
 | 	[CIFS_GENL_ATTR_SWN_NOTIFICATION_TYPE]	= { .type = NLA_U32 }, | 
 | 	[CIFS_GENL_ATTR_SWN_RESOURCE_STATE]	= { .type = NLA_U32 }, | 
 | 	[CIFS_GENL_ATTR_SWN_RESOURCE_NAME]	= { .type = NLA_STRING}, | 
 | }; | 
 |  | 
 | static const struct genl_ops cifs_genl_ops[] = { | 
 | 	{ | 
 | 		.cmd = CIFS_GENL_CMD_SWN_NOTIFY, | 
 | 		.validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP, | 
 | 		.doit = cifs_swn_notify, | 
 | 	}, | 
 | }; | 
 |  | 
 | static const struct genl_multicast_group cifs_genl_mcgrps[] = { | 
 | 	[CIFS_GENL_MCGRP_SWN] = { .name = CIFS_GENL_MCGRP_SWN_NAME }, | 
 | }; | 
 |  | 
 | struct genl_family cifs_genl_family = { | 
 | 	.name		= CIFS_GENL_NAME, | 
 | 	.version	= CIFS_GENL_VERSION, | 
 | 	.hdrsize	= 0, | 
 | 	.maxattr	= CIFS_GENL_ATTR_MAX, | 
 | 	.module		= THIS_MODULE, | 
 | 	.policy		= cifs_genl_policy, | 
 | 	.ops		= cifs_genl_ops, | 
 | 	.n_ops		= ARRAY_SIZE(cifs_genl_ops), | 
 | 	.mcgrps		= cifs_genl_mcgrps, | 
 | 	.n_mcgrps	= ARRAY_SIZE(cifs_genl_mcgrps), | 
 | }; | 
 |  | 
 | /** | 
 |  * cifs_genl_init - Register generic netlink family | 
 |  * | 
 |  * Return zero if initialized successfully, otherwise non-zero. | 
 |  */ | 
 | int cifs_genl_init(void) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	ret = genl_register_family(&cifs_genl_family); | 
 | 	if (ret < 0) { | 
 | 		cifs_dbg(VFS, "%s: failed to register netlink family\n", | 
 | 				__func__); | 
 | 		return ret; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | /** | 
 |  * cifs_genl_exit - Unregister generic netlink family | 
 |  */ | 
 | void cifs_genl_exit(void) | 
 | { | 
 | 	int ret; | 
 |  | 
 | 	ret = genl_unregister_family(&cifs_genl_family); | 
 | 	if (ret < 0) { | 
 | 		cifs_dbg(VFS, "%s: failed to unregister netlink family\n", | 
 | 				__func__); | 
 | 	} | 
 | } |