1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-27 14:04:05 +03:00

merge some of the nsswitch code from tng to head

the libnss_winbind.so from head now works with
the winbindd from tng
This commit is contained in:
Andrew Tridgell -
parent b9137b613d
commit 67ccfd2826
5 changed files with 701 additions and 268 deletions

View File

@ -384,9 +384,9 @@ WINBINDD_OBJ = \
$(LIBNMB_OBJ) $(PARAM_OBJ) $(UBIQX_OBJ) $(LIB_OBJ) \
$(NSSWINS_OBJ) $(SIDDB_OBJ) $(LIBSMB_OBJ)
WBINFO_OBJ = nsswitch/wbinfo.o nsswitch/wb_common.o
WBINFO_OBJ = nsswitch/wbinfo.o
WINBIND_NSS_OBJ = nsswitch/winbind.o nsswitch/wb_common.o
WINBIND_NSS_OBJ = nsswitch/winbind_nss.o nsswitch/wb_common.o
WINBIND_NSS_PICOBJS = $(WINBIND_NSS_OBJ:.o=.po)

View File

@ -37,7 +37,7 @@ void init_request(struct winbindd_request *request, int request_type)
static char *domain_env;
static BOOL initialised;
request->cmd = (enum winbindd_cmd)request_type;
request->cmd = request_type;
request->pid = getpid();
request->domain[0] = '\0';
@ -59,7 +59,7 @@ void init_response(struct winbindd_response *response)
{
/* Initialise return value */
response->result = (enum winbindd_result)NSS_STATUS_UNAVAIL;
response->result = WINBINDD_ERROR;
}
/* Close established socket */
@ -141,6 +141,7 @@ static int open_pipe_sock(void)
if (connect(established_socket, (struct sockaddr *)&sunaddr,
sizeof(sunaddr)) == -1) {
close_sock();
established_socket = -1;
return -1;
}
@ -304,7 +305,7 @@ void free_response(struct winbindd_response *response)
/* Handle simple types of requests */
enum nss_status winbindd_request(int req_type,
NSS_STATUS winbindd_request(int req_type,
struct winbindd_request *request,
struct winbindd_response *response)
{

View File

@ -101,10 +101,11 @@ static BOOL wbinfo_check_secret(void)
if (result) {
if (response.data.num_entries) {
if (response.data.num_entries == 0) {
printf("Secret is good\n");
} else {
printf("Secret is bad\n");
printf("Secret is bad\n0x%08x\n",
response.data.num_entries);
}
return True;

View File

@ -25,11 +25,15 @@
#include "winbind_nss_config.h"
#include "winbindd_nss.h"
/* prototypes from common.c */
/* Prototypes from common.c */
void init_request(struct winbindd_request *req,int rq_type);
NSS_STATUS winbindd_request(int req_type,
struct winbindd_request *request,
struct winbindd_response *response);
int write_sock(void *buffer, int count);
int read_reply(struct winbindd_response *response);
void free_response(struct winbindd_response *response);
/* Allocate some space from the nss static buffer. The buffer and buflen
are the pointers passed in by the C library to the _nss_ntdom_*
@ -46,6 +50,16 @@ static char *get_static(char **buffer, int *buflen, int len)
return NULL;
}
/* Some architectures, like Sparc, need pointers aligned on
boundaries */
#if _ALIGNMENT_REQUIRED
{
int mod = len % _MAX_ALIGNMENT;
if(mod != 0)
len += _MAX_ALIGNMENT - mod;
}
#endif
/* Return an index into the static buffer */
result = *buffer;
@ -59,10 +73,6 @@ static char *get_static(char **buffer, int *buflen, int len)
lib/util_str.c as I really don't want to have to link in any other
objects if I can possibly avoid it. */
#ifdef strchr /* Aargh! This points at multibyte_strchr(). )-: */
#undef strchr
#endif
static char *last_ptr = NULL;
BOOL next_token(char **ptr, char *buff, char *sep, size_t bufsize)
@ -105,49 +115,14 @@ BOOL next_token(char **ptr, char *buff, char *sep, size_t bufsize)
return(True);
}
/* handle simple types of requests */
static enum nss_status generic_request(int req_type,
struct winbindd_request *request,
struct winbindd_response *response)
{
struct winbindd_request lrequest;
struct winbindd_response lresponse;
if (!response) response = &lresponse;
if (!request) request = &lrequest;
/* Fill in request and send down pipe */
init_request(request, req_type);
if (write_sock(request, sizeof(*request)) == -1) {
return NSS_STATUS_UNAVAIL;
}
/* Wait for reply */
if (read_reply(response) == -1) {
return NSS_STATUS_UNAVAIL;
}
/* Copy reply data from socket */
if (response->result != WINBINDD_OK) {
return NSS_STATUS_NOTFOUND;
}
return NSS_STATUS_SUCCESS;
}
/* Fill a pwent structure from a winbindd_response structure. We use
the static data passed to us by libc to put strings and stuff in.
Return errno = ERANGE and NSS_STATUS_TRYAGAIN if we run out of
memory. */
Return NSS_STATUS_TRYAGAIN if we run out of memory. */
static enum nss_status fill_pwent(struct passwd *result,
struct winbindd_response *response,
char **buffer, int *buflen, int *errnop)
static NSS_STATUS fill_pwent(struct passwd *result,
struct winbindd_pw *pw,
char **buffer, int *buflen)
{
struct winbindd_pw *pw = &response->data.pw;
/* User name */
if ((result->pw_name =
@ -155,7 +130,6 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -168,7 +142,6 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -186,7 +159,6 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -199,7 +171,6 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -212,7 +183,6 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -223,14 +193,11 @@ static enum nss_status fill_pwent(struct passwd *result,
/* Fill a grent structure from a winbindd_response structure. We use
the static data passed to us by libc to put strings and stuff in.
Return errno = ERANGE and NSS_STATUS_TRYAGAIN if we run out of
memory. */
Return NSS_STATUS_TRYAGAIN if we run out of memory. */
static int fill_grent(struct group *result,
struct winbindd_response *response,
char **buffer, int *buflen, int *errnop)
static int fill_grent(struct group *result, struct winbindd_gr *gr,
char *gr_mem, char **buffer, int *buflen)
{
struct winbindd_gr *gr = &response->data.gr;
fstring name;
int i;
@ -241,7 +208,6 @@ static int fill_grent(struct group *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -254,7 +220,6 @@ static int fill_grent(struct group *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -266,7 +231,7 @@ static int fill_grent(struct group *result,
/* Group membership */
if ((gr->num_gr_mem < 0) || !response->extra_data) {
if ((gr->num_gr_mem < 0) || !gr_mem) {
gr->num_gr_mem = 0;
}
@ -276,7 +241,6 @@ static int fill_grent(struct group *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -292,7 +256,7 @@ static int fill_grent(struct group *result,
i = 0;
while(next_token(&response->extra_data, name, ",", sizeof(fstring))) {
while(next_token((char **)&gr_mem, name, ",", sizeof(fstring))) {
/* Allocate space for member */
@ -301,7 +265,6 @@ static int fill_grent(struct group *result,
/* Out of memory */
*errnop = ERANGE;
return NSS_STATUS_TRYAGAIN;
}
@ -320,146 +283,568 @@ static int fill_grent(struct group *result,
* NSS user functions
*/
static struct winbindd_response getpwent_response;
static int ndx_pw_cache; /* Current index into pwd cache */
static int num_pw_cache; /* Current size of pwd cache */
/* Rewind "file pointer" to start of ntdom password database */
enum nss_status
NSS_STATUS
_nss_winbind_setpwent(void)
{
return generic_request(WINBINDD_SETPWENT, NULL, NULL);
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: setpwent\n", getpid());
#endif
if (num_pw_cache > 0) {
ndx_pw_cache = num_pw_cache = 0;
free_response(&getpwent_response);
}
return winbindd_request(WINBINDD_SETPWENT, NULL, NULL);
}
/* Close ntdom password database "file pointer" */
enum nss_status
NSS_STATUS
_nss_winbind_endpwent(void)
{
return generic_request(WINBINDD_ENDPWENT, NULL, NULL);
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: endpwent\n", getpid());
#endif
if (num_pw_cache > 0) {
ndx_pw_cache = num_pw_cache = 0;
free_response(&getpwent_response);
}
return winbindd_request(WINBINDD_ENDPWENT, NULL, NULL);
}
/* Fetch the next password entry from ntdom password database */
enum nss_status
#define MAX_GETPWENT_USERS 250
NSS_STATUS
_nss_winbind_getpwent_r(struct passwd *result, char *buffer,
size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
struct winbindd_request request;
static int called_again;
ret = generic_request(WINBINDD_GETPWENT, NULL, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: getpwent\n", getpid());
#endif
return fill_pwent(result, &response, &buffer, &buflen, errnop);
/* Return an entry from the cache if we have one, or if we are
called again because we exceeded our static buffer. */
if ((ndx_pw_cache < num_pw_cache) || called_again) {
goto return_result;
}
/* Else call winbindd to get a bunch of entries */
if (num_pw_cache > 0) {
free_response(&getpwent_response);
}
ZERO_STRUCT(request);
ZERO_STRUCT(getpwent_response);
request.data.num_entries = MAX_GETPWENT_USERS;
ret = winbindd_request(WINBINDD_GETPWENT, &request,
&getpwent_response);
if (ret == NSS_STATUS_SUCCESS) {
struct winbindd_pw *pw_cache;
/* Fill cache */
ndx_pw_cache = 0;
num_pw_cache = getpwent_response.data.num_entries;
/* Return a result */
return_result:
pw_cache = getpwent_response.extra_data;
/* Check data is valid */
if (pw_cache == NULL) {
return NSS_STATUS_NOTFOUND;
}
ret = fill_pwent(result, &pw_cache[ndx_pw_cache],
&buffer, &buflen);
/* Out of memory - try again */
if (ret == NSS_STATUS_TRYAGAIN) {
called_again = True;
*errnop = errno = ERANGE;
return ret;
}
*errnop = errno = 0;
called_again = False;
ndx_pw_cache++;
/* If we've finished with this lot of results free cache */
if (ndx_pw_cache == num_pw_cache) {
ndx_pw_cache = num_pw_cache = 0;
free_response(&getpwent_response);
}
}
return ret;
}
/* Return passwd struct from uid */
enum nss_status
NSS_STATUS
_nss_winbind_getpwuid_r(uid_t uid, struct passwd *result, char *buffer,
size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
static struct winbindd_response response;
struct winbindd_request request;
static int keep_response=0;
/* If our static buffer needs to be expanded we are called again */
if (!keep_response) {
/* Call for the first time */
ZERO_STRUCT(response);
ZERO_STRUCT(request);
request.data.uid = uid;
ret = generic_request(WINBINDD_GETPWNAM_FROM_UID, &request, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
ret = winbindd_request(WINBINDD_GETPWNAM_FROM_UID, &request,
&response);
return fill_pwent(result, &response, &buffer, &buflen, errnop);
if (ret == NSS_STATUS_SUCCESS) {
ret = fill_pwent(result, &response.data.pw,
&buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
}
} else {
/* We've been called again */
ret = fill_pwent(result, &response.data.pw, &buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
keep_response = False;
*errnop = errno = 0;
}
free_response(&response);
return ret;
}
/* Return passwd struct from username */
enum nss_status
NSS_STATUS
_nss_winbind_getpwnam_r(const char *name, struct passwd *result, char *buffer,
size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
static struct winbindd_response response;
struct winbindd_request request;
static int keep_response;
strncpy(request.data.username, name, sizeof(request.data.username) - 1);
request.data.username[sizeof(request.data.username) - 1] = '\0';
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: getpwnam %s\n", getpid(), name);
#endif
ret = generic_request(WINBINDD_GETPWNAM_FROM_USER, &request, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
/* If our static buffer needs to be expanded we are called again */
return fill_pwent(result, &response, &buffer, &buflen, errnop);
if (!keep_response) {
/* Call for the first time */
ZERO_STRUCT(response);
ZERO_STRUCT(request);
strncpy(request.data.username, name,
sizeof(request.data.username) - 1);
request.data.username
[sizeof(request.data.username) - 1] = '\0';
ret = winbindd_request(WINBINDD_GETPWNAM_FROM_USER, &request,
&response);
if (ret == NSS_STATUS_SUCCESS) {
ret = fill_pwent(result, &response.data.pw, &buffer,
&buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
}
} else {
/* We've been called again */
ret = fill_pwent(result, &response.data.pw, &buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
keep_response = False;
*errnop = errno = 0;
}
free_response(&response);
return ret;
}
/*
* NSS group functions
*/
static struct winbindd_response getgrent_response;
static int ndx_gr_cache; /* Current index into grp cache */
static int num_gr_cache; /* Current size of grp cache */
/* Rewind "file pointer" to start of ntdom group database */
enum nss_status
NSS_STATUS
_nss_winbind_setgrent(void)
{
return generic_request(WINBINDD_SETGRENT, NULL, NULL);
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: setgrent\n", getpid());
#endif
if (num_gr_cache > 0) {
ndx_gr_cache = num_gr_cache = 0;
free_response(&getgrent_response);
}
return winbindd_request(WINBINDD_SETGRENT, NULL, NULL);
}
/* Close "file pointer" for ntdom group database */
enum nss_status
NSS_STATUS
_nss_winbind_endgrent(void)
{
return generic_request(WINBINDD_ENDGRENT, NULL, NULL);
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: endgrent\n", getpid());
#endif
if (num_gr_cache > 0) {
ndx_gr_cache = num_gr_cache = 0;
free_response(&getgrent_response);
}
return winbindd_request(WINBINDD_ENDGRENT, NULL, NULL);
}
/* Get next entry from ntdom group database */
enum nss_status
#define MAX_GETGRENT_USERS 250
NSS_STATUS
_nss_winbind_getgrent_r(struct group *result,
char *buffer, size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
static struct winbindd_request request;
static int called_again;
ret = generic_request(WINBINDD_GETGRENT, NULL, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: getgrent\n", getpid());
#endif
return fill_grent(result, &response, &buffer, &buflen, errnop);
/* Return an entry from the cache if we have one, or if we are
called again because we exceeded our static buffer. */
if ((ndx_gr_cache < num_gr_cache) || called_again) {
goto return_result;
}
/* Else call winbindd to get a bunch of entries */
if (num_gr_cache > 0) {
free_response(&getgrent_response);
}
ZERO_STRUCT(request);
ZERO_STRUCT(getgrent_response);
request.data.num_entries = MAX_GETGRENT_USERS;
ret = winbindd_request(WINBINDD_GETGRENT, &request,
&getgrent_response);
if (ret == NSS_STATUS_SUCCESS) {
struct winbindd_gr *gr_cache;
int mem_ofs;
/* Fill cache */
ndx_gr_cache = 0;
num_gr_cache = getgrent_response.data.num_entries;
/* Return a result */
return_result:
gr_cache = getgrent_response.extra_data;
/* Check data is valid */
if (gr_cache == NULL) {
return NSS_STATUS_NOTFOUND;
}
/* Fill group membership. The offset into the extra data
for the group membership is the reported offset plus the
size of all the winbindd_gr records returned. */
mem_ofs = gr_cache[ndx_gr_cache].gr_mem_ofs +
num_gr_cache * sizeof(struct winbindd_gr);
ret = fill_grent(result, &gr_cache[ndx_gr_cache],
(char *)(getgrent_response.extra_data +
mem_ofs), &buffer, &buflen);
/* Out of memory - try again */
if (ret == NSS_STATUS_TRYAGAIN) {
called_again = True;
*errnop = errno = ERANGE;
return ret;
}
*errnop = 0;
called_again = False;
ndx_gr_cache++;
/* If we've finished with this lot of results free cache */
if (ndx_gr_cache == num_gr_cache) {
ndx_gr_cache = num_gr_cache = 0;
free_response(&getgrent_response);
}
}
return ret;
}
/* Return group struct from group name */
enum nss_status
NSS_STATUS
_nss_winbind_getgrnam_r(const char *name,
struct group *result, char *buffer,
size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
static struct winbindd_response response;
struct winbindd_request request;
static int keep_response;
strncpy(request.data.groupname, name, sizeof(request.data.groupname));
request.data.groupname[sizeof(request.data.groupname) - 1] = '\0';
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: getgrnam %s\n", getpid(), name);
#endif
ret = generic_request(WINBINDD_GETGRNAM_FROM_GROUP, &request, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
/* If our static buffer needs to be expanded we are called again */
return fill_grent(result, &response, &buffer, &buflen, errnop);
if (!keep_response) {
/* Call for the first time */
ZERO_STRUCT(request);
ZERO_STRUCT(response);
strncpy(request.data.groupname, name,
sizeof(request.data.groupname));
request.data.groupname
[sizeof(request.data.groupname) - 1] = '\0';
ret = winbindd_request(WINBINDD_GETGRNAM_FROM_GROUP,
&request, &response);
if (ret == NSS_STATUS_SUCCESS) {
ret = fill_grent(result, &response.data.gr,
response.extra_data,
&buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
}
} else {
/* We've been called again */
ret = fill_grent(result, &response.data.gr,
response.extra_data, &buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
keep_response = False;
*errnop = 0;
}
free_response(&response);
return ret;
}
/* Return group struct from gid */
enum nss_status
NSS_STATUS
_nss_winbind_getgrgid_r(gid_t gid,
struct group *result, char *buffer,
size_t buflen, int *errnop)
{
enum nss_status ret;
struct winbindd_response response;
NSS_STATUS ret;
static struct winbindd_response response;
struct winbindd_request request;
static int keep_response;
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: getgrgid %d\n", getpid(), gid);
#endif
/* If our static buffer needs to be expanded we are called again */
if (!keep_response) {
/* Call for the first time */
ZERO_STRUCT(request);
ZERO_STRUCT(response);
request.data.gid = gid;
ret = generic_request(WINBINDD_GETGRNAM_FROM_GID, &request, &response);
if (ret != NSS_STATUS_SUCCESS) return ret;
ret = winbindd_request(WINBINDD_GETGRNAM_FROM_GID, &request,
&response);
return fill_grent(result, &response, &buffer, &buflen, errnop);
if (ret == NSS_STATUS_SUCCESS) {
ret = fill_grent(result, &response.data.gr,
response.extra_data,
&buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
}
} else {
/* We've been called again */
ret = fill_grent(result, &response.data.gr,
response.extra_data, &buffer, &buflen);
if (ret == NSS_STATUS_TRYAGAIN) {
keep_response = True;
*errnop = errno = ERANGE;
return ret;
}
keep_response = False;
*errnop = 0;
}
free_response(&response);
return ret;
}
/* Initialise supplementary groups */
NSS_STATUS
_nss_winbind_initgroups(char *user, gid_t group, long int *start,
long int *size, gid_t *groups, long int limit,
int *errnop)
{
NSS_STATUS ret;
struct winbindd_request request;
struct winbindd_response response;
int i;
#ifdef DEBUG_NSS
fprintf(stderr, "[%5d]: initgroups %s (%d)\n", getpid(),
user, group);
#endif
ZERO_STRUCT(request);
ZERO_STRUCT(response);
strncpy(request.data.username, user,
sizeof(request.data.username) - 1);
ret = winbindd_request(WINBINDD_GETGROUPS, &request, &response);
if (ret == NSS_STATUS_SUCCESS) {
int num_gids = response.data.num_entries;
gid_t *gid_list = (gid_t *)response.extra_data;
/* Copy group list to client */
for (i = 0; i < num_gids; i++) {
/* Skip primary group */
if (gid_list[i] == group) continue;
/* Add to buffer */
if (*start == *size && limit <= 0) {
groups = realloc(
groups, 2 * (*size) * sizeof(*groups));
if (!groups) goto done;
*size *= 2;
}
groups[*start] = gid_list[i];
*start += 1;
/* Filled buffer? */
if (*start == limit) goto done;
}
}
/* Back to your regularly scheduled programming */
done:
return ret;
}

View File

@ -70,17 +70,62 @@
#include <errno.h>
#include <pwd.h>
#ifdef HAVE_NSS_H
#ifdef HAVE_NSS_COMMON_H
/* Sun Solaris */
#include <nss_common.h>
#include <nss_dbdefs.h>
#include <nsswitch.h>
typedef nss_status_t NSS_STATUS;
#define NSS_STATUS_SUCCESS NSS_SUCCESS
#define NSS_STATUS_NOTFOUND NSS_NOTFOUND
#define NSS_STATUS_UNAVAIL NSS_UNAVAIL
#define NSS_STATUS_TRYAGAIN NSS_TRYAGAIN
#elif HAVE_NSS_H
/* GNU */
#include <nss.h>
#else
/* Minimal needed to compile.. */
enum nss_status {
typedef enum nss_status NSS_STATUS;
#else /* Nothing's defined. Neither gnu nor sun */
typedef enum
{
NSS_STATUS_SUCCESS,
NSS_STATUS_NOTFOUND,
NSS_STATUS_UNAVAIL
};
NSS_STATUS_UNAVAIL,
NSS_STATUS_TRYAGAIN
} NSS_STATUS;
#endif
/* Declarations for functions in winbind_nss.c
needed in winbind_nss_solaris.c (solaris wrapper to nss) */
NSS_STATUS _nss_winbind_setpwent(void);
NSS_STATUS _nss_winbind_endpwent(void);
NSS_STATUS _nss_winbind_getpwent_r(struct passwd* result, char* buffer,
size_t buflen, int* errnop);
NSS_STATUS _nss_winbind_getpwuid_r(uid_t, struct passwd*, char* buffer,
size_t buflen, int* errnop);
NSS_STATUS _nss_winbind_getpwnam_r(const char* name, struct passwd* result,
char* buffer, size_t buflen, int* errnop);
NSS_STATUS _nss_winbind_setgrent(void);
NSS_STATUS _nss_winbind_endgrent(void);
NSS_STATUS _nss_winbind_getgrent_r(struct group* result, char* buffer,
size_t buflen, int* errnop);
NSS_STATUS _nss_winbind_getgrnam_r(const char *name,
struct group *result, char *buffer,
size_t buflen, int *errnop);
NSS_STATUS _nss_winbind_getgrgid_r(gid_t gid,
struct group *result, char *buffer,
size_t buflen, int *errnop);
/* I'm trying really hard not to include anything from smb.h with the
result of some silly looking redeclaration of structures. */
@ -127,6 +172,7 @@ typedef int BOOL;
/* zero a structure given a pointer to the structure */
#define ZERO_STRUCTP(x) { if ((x) != NULL) memset((char *)(x), 0, sizeof(*(x))); }
/* Some systems (SCO) treat UNIX domain sockets as FIFOs */
#ifndef S_IFSOCK