| // 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 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__); |
| } |
| } |