mirror of
https://github.com/samba-team/samba.git
synced 2025-01-20 14:03:59 +03:00
844e1bdc6d
In order to use SASL authentitation within a TLS connection we now provide "client ldap sasl wrapping = starttls" or "client ldap sasl wrapping = ldaps". Signed-off-by: Stefan Metzmacher <metze@samba.org> Reviewed-by: Andrew Bartlett <abartlet@samba.org>
468 lines
14 KiB
C
468 lines
14 KiB
C
/*
|
|
Unix SMB/CIFS implementation.
|
|
Parameter loading functions
|
|
Copyright (C) Karl Auer 1993-1998
|
|
|
|
Largely re-written by Andrew Tridgell, September 1994
|
|
|
|
Copyright (C) Simo Sorce 2001
|
|
Copyright (C) Alexander Bokovoy 2002
|
|
Copyright (C) Stefan (metze) Metzmacher 2002
|
|
Copyright (C) Jim McDonough <jmcd@us.ibm.com> 2003
|
|
Copyright (C) Michael Adam 2008
|
|
Copyright (C) Jelmer Vernooij <jelmer@samba.org> 2007
|
|
Copyright (C) Andrew Bartlett 2011
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "lib/param/param.h"
|
|
#include "lib/param/loadparm.h"
|
|
#include "lib/param/param_global.h"
|
|
#include "libcli/auth/ntlm_check.h"
|
|
#include "libcli/smb/smb_constants.h"
|
|
#include "libds/common/roles.h"
|
|
#include "libds/common/flags.h"
|
|
#include "source4/lib/tls/tls.h"
|
|
#include "auth/credentials/credentials.h"
|
|
#include "source3/librpc/gen_ndr/ads.h"
|
|
|
|
#ifndef N_
|
|
#define N_(x) x
|
|
#endif
|
|
|
|
static const struct enum_list enum_protocol[] = {
|
|
{PROTOCOL_DEFAULT, "default"}, /* the caller decides what this means */
|
|
{PROTOCOL_SMB2_10, "SMB2"}, /* for now keep PROTOCOL_SMB2_10 */
|
|
{PROTOCOL_SMB3_11, "SMB3"}, /* for now keep PROTOCOL_SMB3_11 */
|
|
{PROTOCOL_SMB3_11, "SMB3_11"},
|
|
{PROTOCOL_SMB3_02, "SMB3_02"},
|
|
{PROTOCOL_SMB3_00, "SMB3_00"},
|
|
{PROTOCOL_SMB2_10, "SMB2_10"},
|
|
{PROTOCOL_SMB2_02, "SMB2_02"},
|
|
{PROTOCOL_NT1, "NT1"},
|
|
{PROTOCOL_LANMAN2, "LANMAN2"},
|
|
{PROTOCOL_LANMAN1, "LANMAN1"},
|
|
{PROTOCOL_CORE, "CORE"},
|
|
{PROTOCOL_COREPLUS, "COREPLUS"},
|
|
{PROTOCOL_COREPLUS, "CORE+"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
const char* lpcfg_get_smb_protocol(int type)
|
|
{
|
|
int i;
|
|
for (i = 1; enum_protocol[i].value != -1; i++) {
|
|
if (enum_protocol[i].value == type) {
|
|
return enum_protocol[i].name;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static const struct enum_list enum_security[] = {
|
|
{SEC_AUTO, "AUTO"},
|
|
{SEC_USER, "USER"},
|
|
{SEC_DOMAIN, "DOMAIN"},
|
|
{SEC_ADS, "ADS"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_bool_auto[] = {
|
|
{false, "No"},
|
|
{false, "False"},
|
|
{false, "0"},
|
|
{true, "Yes"},
|
|
{true, "True"},
|
|
{true, "1"},
|
|
{Auto, "Auto"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_csc_policy[] = {
|
|
{CSC_POLICY_MANUAL, "manual"},
|
|
{CSC_POLICY_DOCUMENTS, "documents"},
|
|
{CSC_POLICY_PROGRAMS, "programs"},
|
|
{CSC_POLICY_DISABLE, "disable"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* Server role options */
|
|
static const struct enum_list enum_server_role[] = {
|
|
{ROLE_AUTO, "auto"},
|
|
{ROLE_STANDALONE, "standalone server"},
|
|
{ROLE_STANDALONE, "standalone"},
|
|
{ROLE_DOMAIN_MEMBER, "member server"},
|
|
{ROLE_DOMAIN_MEMBER, "member"},
|
|
{ROLE_DOMAIN_PDC, "classic primary domain controller"},
|
|
{ROLE_DOMAIN_BDC, "classic backup domain controller"},
|
|
{ROLE_ACTIVE_DIRECTORY_DC, "active directory domain controller"},
|
|
{ROLE_ACTIVE_DIRECTORY_DC, "domain controller"},
|
|
{ROLE_ACTIVE_DIRECTORY_DC, "dc"},
|
|
{ROLE_IPA_DC, "IPA primary domain controller"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* SMB signing types. */
|
|
static const struct enum_list enum_smb_signing_vals[] = {
|
|
{SMB_SIGNING_DEFAULT, "default"},
|
|
{SMB_SIGNING_OFF, "No"},
|
|
{SMB_SIGNING_OFF, "False"},
|
|
{SMB_SIGNING_OFF, "0"},
|
|
{SMB_SIGNING_OFF, "Off"},
|
|
{SMB_SIGNING_OFF, "disabled"},
|
|
{SMB_SIGNING_IF_REQUIRED, "if_required"},
|
|
{SMB_SIGNING_IF_REQUIRED, "Yes"},
|
|
{SMB_SIGNING_IF_REQUIRED, "True"},
|
|
{SMB_SIGNING_IF_REQUIRED, "1"},
|
|
{SMB_SIGNING_IF_REQUIRED, "On"},
|
|
{SMB_SIGNING_IF_REQUIRED, "enabled"},
|
|
{SMB_SIGNING_IF_REQUIRED, "auto"},
|
|
{SMB_SIGNING_DESIRED, "desired"},
|
|
{SMB_SIGNING_REQUIRED, "required"},
|
|
{SMB_SIGNING_REQUIRED, "mandatory"},
|
|
{SMB_SIGNING_REQUIRED, "force"},
|
|
{SMB_SIGNING_REQUIRED, "forced"},
|
|
{SMB_SIGNING_REQUIRED, "enforced"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_smb_encryption_vals[] = {
|
|
{SMB_ENCRYPTION_DEFAULT, "default"},
|
|
{SMB_ENCRYPTION_OFF, "No"},
|
|
{SMB_ENCRYPTION_OFF, "False"},
|
|
{SMB_ENCRYPTION_OFF, "0"},
|
|
{SMB_ENCRYPTION_OFF, "Off"},
|
|
{SMB_ENCRYPTION_OFF, "disabled"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "if_required"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "Yes"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "True"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "1"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "On"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "enabled"},
|
|
{SMB_ENCRYPTION_IF_REQUIRED, "auto"},
|
|
{SMB_ENCRYPTION_DESIRED, "desired"},
|
|
{SMB_ENCRYPTION_REQUIRED, "required"},
|
|
{SMB_ENCRYPTION_REQUIRED, "mandatory"},
|
|
{SMB_ENCRYPTION_REQUIRED, "force"},
|
|
{SMB_ENCRYPTION_REQUIRED, "forced"},
|
|
{SMB_ENCRYPTION_REQUIRED, "enforced"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_use_kerberos_vals[] = {
|
|
{CRED_USE_KERBEROS_DESIRED, "desired"},
|
|
{CRED_USE_KERBEROS_DESIRED, "auto"},
|
|
{CRED_USE_KERBEROS_REQUIRED, "yes"},
|
|
{CRED_USE_KERBEROS_REQUIRED, "required"},
|
|
{CRED_USE_KERBEROS_DISABLED, "no"},
|
|
{CRED_USE_KERBEROS_DISABLED, "disabled"},
|
|
{CRED_USE_KERBEROS_DISABLED, "off"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_client_protection_vals[] = {
|
|
{CRED_CLIENT_PROTECTION_DEFAULT, "default"},
|
|
{CRED_CLIENT_PROTECTION_PLAIN, "plain"},
|
|
{CRED_CLIENT_PROTECTION_SIGN, "sign"},
|
|
{CRED_CLIENT_PROTECTION_ENCRYPT, "encrypt"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_mdns_name_values[] = {
|
|
{MDNS_NAME_NETBIOS, "netbios"},
|
|
{MDNS_NAME_MDNS, "mdns"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_tls_verify_peer_vals[] = {
|
|
{TLS_VERIFY_PEER_NO_CHECK,
|
|
TLS_VERIFY_PEER_NO_CHECK_STRING},
|
|
{TLS_VERIFY_PEER_CA_ONLY,
|
|
TLS_VERIFY_PEER_CA_ONLY_STRING},
|
|
{TLS_VERIFY_PEER_CA_AND_NAME_IF_AVAILABLE,
|
|
TLS_VERIFY_PEER_CA_AND_NAME_IF_AVAILABLE_STRING},
|
|
{TLS_VERIFY_PEER_CA_AND_NAME,
|
|
TLS_VERIFY_PEER_CA_AND_NAME_STRING},
|
|
{TLS_VERIFY_PEER_AS_STRICT_AS_POSSIBLE,
|
|
TLS_VERIFY_PEER_AS_STRICT_AS_POSSIBLE_STRING},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* DNS update options. */
|
|
static const struct enum_list enum_dns_update_settings[] = {
|
|
{DNS_UPDATE_OFF, "disabled"},
|
|
{DNS_UPDATE_OFF, "No"},
|
|
{DNS_UPDATE_OFF, "False"},
|
|
{DNS_UPDATE_OFF, "0"},
|
|
{DNS_UPDATE_OFF, "Off"},
|
|
{DNS_UPDATE_ON, "nonsecure and secure"},
|
|
{DNS_UPDATE_ON, "nonsecure"},
|
|
{DNS_UPDATE_SIGNED, "secure only"},
|
|
{DNS_UPDATE_SIGNED, "secure"},
|
|
{DNS_UPDATE_SIGNED, "signed"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/*
|
|
Do you want session setups at user level security with a invalid
|
|
password to be rejected or allowed in as guest? WinNT rejects them
|
|
but it can be a pain as it means "net view" needs to use a password
|
|
|
|
You have 3 choices in the setting of map_to_guest:
|
|
|
|
"Never" means session setups with an invalid password
|
|
are rejected. This is the default.
|
|
|
|
"Bad User" means session setups with an invalid password
|
|
are rejected, unless the username does not exist, in which case it
|
|
is treated as a guest login
|
|
|
|
"Bad Password" means session setups with an invalid password
|
|
are treated as a guest login
|
|
|
|
Note that map_to_guest only has an effect in user or server
|
|
level security.
|
|
*/
|
|
|
|
static const struct enum_list enum_map_to_guest[] = {
|
|
{NEVER_MAP_TO_GUEST, "Never"},
|
|
{MAP_TO_GUEST_ON_BAD_USER, "Bad User"},
|
|
{MAP_TO_GUEST_ON_BAD_PASSWORD, "Bad Password"},
|
|
{MAP_TO_GUEST_ON_BAD_UID, "Bad Uid"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* Config backend options */
|
|
|
|
static const struct enum_list enum_config_backend[] = {
|
|
{CONFIG_BACKEND_FILE, "file"},
|
|
{CONFIG_BACKEND_REGISTRY, "registry"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_smbd_profiling_level[] = {
|
|
{0, "off"}, {1, "count"}, {2, "on"}, {-1, NULL}
|
|
};
|
|
|
|
|
|
/* ADS kerberos ticket verification options */
|
|
|
|
static const struct enum_list enum_kerberos_method[] = {
|
|
{KERBEROS_VERIFY_SECRETS, "default"},
|
|
{KERBEROS_VERIFY_SECRETS, "secrets only"},
|
|
{KERBEROS_VERIFY_SECRETS, "secretsonly"},
|
|
{KERBEROS_VERIFY_SYSTEM_KEYTAB, "system keytab"},
|
|
{KERBEROS_VERIFY_SYSTEM_KEYTAB, "systemkeytab"},
|
|
{KERBEROS_VERIFY_DEDICATED_KEYTAB, "dedicated keytab"},
|
|
{KERBEROS_VERIFY_DEDICATED_KEYTAB, "dedicatedkeytab"},
|
|
{KERBEROS_VERIFY_SECRETS_AND_KEYTAB, "secrets and keytab"},
|
|
{KERBEROS_VERIFY_SECRETS_AND_KEYTAB, "secretsandkeytab"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* Kerberos encryption types selection options */
|
|
|
|
static const struct enum_list enum_kerberos_encryption_types_vals[] = {
|
|
{KERBEROS_ETYPES_ALL, "all"},
|
|
{KERBEROS_ETYPES_STRONG, "strong"},
|
|
{KERBEROS_ETYPES_LEGACY, "legacy"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_printing[] = {
|
|
{PRINT_SYSV, "sysv"},
|
|
{PRINT_AIX, "aix"},
|
|
{PRINT_HPUX, "hpux"},
|
|
{PRINT_BSD, "bsd"},
|
|
{PRINT_QNX, "qnx"},
|
|
{PRINT_PLP, "plp"},
|
|
{PRINT_LPRNG, "lprng"},
|
|
#ifdef HAVE_CUPS
|
|
{PRINT_CUPS, "cups"},
|
|
#endif
|
|
#ifdef HAVE_IPRINT
|
|
{PRINT_IPRINT, "iprint"},
|
|
#endif
|
|
{PRINT_LPRNT, "nt"},
|
|
{PRINT_LPROS2, "os2"},
|
|
#if defined(DEVELOPER) || defined(ENABLE_SELFTEST)
|
|
{PRINT_TEST, "test"},
|
|
{PRINT_VLP, "vlp"},
|
|
#endif /* DEVELOPER */
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_ldap_sasl_wrapping[] = {
|
|
{0, "plain"},
|
|
{ADS_AUTH_SASL_SIGN, "sign"},
|
|
{ADS_AUTH_SASL_SEAL, "seal"},
|
|
{ADS_AUTH_SASL_STARTTLS, "starttls"},
|
|
{ADS_AUTH_SASL_LDAPS, "ldaps"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_ldap_server_require_strong_auth_vals[] = {
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_NO, "No" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_NO, "False" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_NO, "0" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_ALLOW_SASL_OVER_TLS,
|
|
"allow_sasl_over_tls" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_ALLOW_SASL_WITHOUT_TLS_CB,
|
|
"allow_sasl_without_tls_channel_bindings" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_YES, "Yes" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_YES, "True" },
|
|
{ LDAP_SERVER_REQUIRE_STRONG_AUTH_YES, "1" },
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_ldap_ssl[] = {
|
|
{LDAP_SSL_OFF, "no"},
|
|
{LDAP_SSL_OFF, "off"},
|
|
{LDAP_SSL_START_TLS, "start tls"},
|
|
{LDAP_SSL_START_TLS, "start_tls"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* LDAP Dereferencing Alias types */
|
|
#define SAMBA_LDAP_DEREF_NEVER 0
|
|
#define SAMBA_LDAP_DEREF_SEARCHING 1
|
|
#define SAMBA_LDAP_DEREF_FINDING 2
|
|
#define SAMBA_LDAP_DEREF_ALWAYS 3
|
|
|
|
static const struct enum_list enum_ldap_deref[] = {
|
|
{SAMBA_LDAP_DEREF_NEVER, "never"},
|
|
{SAMBA_LDAP_DEREF_SEARCHING, "searching"},
|
|
{SAMBA_LDAP_DEREF_FINDING, "finding"},
|
|
{SAMBA_LDAP_DEREF_ALWAYS, "always"},
|
|
{-1, "auto"}
|
|
};
|
|
|
|
static const struct enum_list enum_ldap_passwd_sync[] = {
|
|
{LDAP_PASSWD_SYNC_OFF, "no"},
|
|
{LDAP_PASSWD_SYNC_OFF, "off"},
|
|
{LDAP_PASSWD_SYNC_ON, "yes"},
|
|
{LDAP_PASSWD_SYNC_ON, "on"},
|
|
{LDAP_PASSWD_SYNC_ONLY, "only"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_map_readonly[] = {
|
|
{MAP_READONLY_NO, "no"},
|
|
{MAP_READONLY_NO, "false"},
|
|
{MAP_READONLY_NO, "0"},
|
|
{MAP_READONLY_YES, "yes"},
|
|
{MAP_READONLY_YES, "true"},
|
|
{MAP_READONLY_YES, "1"},
|
|
{MAP_READONLY_PERMISSIONS, "permissions"},
|
|
{MAP_READONLY_PERMISSIONS, "perms"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_case[] = {
|
|
{CASE_LOWER, "lower"},
|
|
{CASE_UPPER, "upper"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_inherit_owner_vals[] = {
|
|
{INHERIT_OWNER_NO, "no"},
|
|
{INHERIT_OWNER_WINDOWS_AND_UNIX, "windows and unix"},
|
|
{INHERIT_OWNER_WINDOWS_AND_UNIX, "yes"},
|
|
{INHERIT_OWNER_UNIX_ONLY, "unix only"},
|
|
{-1, NULL}};
|
|
|
|
static const struct enum_list enum_mangled_names[] = {
|
|
{MANGLED_NAMES_NO, "no"},
|
|
{MANGLED_NAMES_NO, "false"},
|
|
{MANGLED_NAMES_NO, "0"},
|
|
{MANGLED_NAMES_ILLEGAL, "illegal"},
|
|
{MANGLED_NAMES_YES, "yes"},
|
|
{MANGLED_NAMES_YES, "true"},
|
|
{MANGLED_NAMES_YES, "1"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_ntlm_auth[] = {
|
|
{NTLM_AUTH_DISABLED, "disabled"},
|
|
{NTLM_AUTH_NTLMV2_ONLY, "ntlmv2-only"},
|
|
{NTLM_AUTH_NTLMV2_ONLY, "no"},
|
|
{NTLM_AUTH_NTLMV2_ONLY, "false"},
|
|
{NTLM_AUTH_NTLMV2_ONLY, "0"},
|
|
{NTLM_AUTH_ON, "ntlmv1-permitted"},
|
|
{NTLM_AUTH_ON, "yes"},
|
|
{NTLM_AUTH_ON, "true"},
|
|
{NTLM_AUTH_ON, "1"},
|
|
{NTLM_AUTH_MSCHAPv2_NTLMV2_ONLY, "mschapv2-and-ntlmv2-only"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_nt_hash_store[] = {
|
|
{NT_HASH_STORE_AUTO, "auto"},
|
|
{NT_HASH_STORE_NEVER, "never"},
|
|
{NT_HASH_STORE_ALWAYS, "always"},
|
|
};
|
|
|
|
|
|
static const struct enum_list enum_spotlight_backend[] = {
|
|
{SPOTLIGHT_BACKEND_NOINDEX, "noindex"},
|
|
{SPOTLIGHT_BACKEND_TRACKER, "tracker"},
|
|
{SPOTLIGHT_BACKEND_ES, "elasticsearch"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_debug_syslog_format[] = {
|
|
{DEBUG_SYSLOG_FORMAT_NO, "No"},
|
|
{DEBUG_SYSLOG_FORMAT_NO, "False"},
|
|
{DEBUG_SYSLOG_FORMAT_NO, "0"},
|
|
{DEBUG_SYSLOG_FORMAT_IN_LOGS, "in_logs"},
|
|
{DEBUG_SYSLOG_FORMAT_IN_LOGS, "Yes"},
|
|
{DEBUG_SYSLOG_FORMAT_IN_LOGS, "True"},
|
|
{DEBUG_SYSLOG_FORMAT_IN_LOGS, "1"},
|
|
{DEBUG_SYSLOG_FORMAT_ALWAYS, "always"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_ad_functional_level[] = {
|
|
{DS_DOMAIN_FUNCTION_2008_R2, "2008_R2"},
|
|
{DS_DOMAIN_FUNCTION_2012, "2012"},
|
|
{DS_DOMAIN_FUNCTION_2012_R2, "2012_R2"},
|
|
{DS_DOMAIN_FUNCTION_2016, "2016"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
static const struct enum_list enum_acl_claims_evaluation[] = {
|
|
{ACL_CLAIMS_EVALUATION_AD_DC_ONLY, "AD DC only"},
|
|
{ACL_CLAIMS_EVALUATION_NEVER, "never"},
|
|
{-1, NULL}
|
|
};
|
|
|
|
/* Note: We do not initialise the defaults union - it is not allowed in ANSI C
|
|
*
|
|
* NOTE: Handling of duplicated (synonym) parameters:
|
|
* Parameters that are synonymous are stored in the same variable.
|
|
* All but the default spelling carry the flag FLAG_SYNONYM.
|
|
*/
|
|
|
|
#define GLOBAL_VAR(name) offsetof(struct loadparm_global, name)
|
|
#define LOCAL_VAR(name) offsetof(struct loadparm_service, name)
|
|
|
|
#include "lib/param/param_table_gen.c"
|
|
|
|
int num_parameters(void)
|
|
{
|
|
return (sizeof(parm_table) / sizeof(struct parm_struct));
|
|
}
|