2001-09-09 18:52:37 +02:00
|
|
|
/* pwdgrp.h
|
|
|
|
|
2015-02-20 16:13:46 +01:00
|
|
|
Copyright 2001, 2002, 2003, 2014, 2015 Red Hat inc.
|
2001-09-09 18:52:37 +02:00
|
|
|
|
|
|
|
Stuff common to pwd and grp handling.
|
|
|
|
|
|
|
|
This file is part of Cygwin.
|
|
|
|
|
|
|
|
This software is a copyrighted work licensed under the terms of the
|
|
|
|
Cygwin license. Please consult the file "CYGWIN_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
2014-02-09 20:44:56 +01:00
|
|
|
#pragma once
|
|
|
|
|
2014-02-27 13:57:27 +01:00
|
|
|
#include "sync.h"
|
|
|
|
#include "ldap.h"
|
|
|
|
#include "miscfuncs.h"
|
2015-02-23 21:51:12 +01:00
|
|
|
#include "userinfo.h"
|
2014-02-27 13:57:27 +01:00
|
|
|
|
2002-12-10 Pierre Humblet <pierre.humblet@ieee.org>
* pwdgrp.h (pwdgrp_check::pwdgrp_state): Replace by
pwdgrp_check::isinitializing ().
(pwdgrp_check::isinitializing): Create.
* passwd.cc (grab_int): Change type to unsigned, use strtoul and
set the pointer content to 0 if the field is invalid.
(parse_pwd): Move validity test after getting pw_gid.
(read_etc_passwd): Replace "passwd_state <= " by
passwd_state::isinitializing ().
(internal_getpwuid): Ditto.
(internal_getpwnam): Ditto.
(getpwent): Ditto.
(getpass): Ditto.
* grp.cc (parse_grp): Use strtoul for gr_gid and verify the validity.
(read_etc_group): Replace "group_state <= " by
group_state::isinitializing ().
(internal_getgrgid): Ditto.
(getgrent32): Ditto.
(internal_getgrent): Ditto.
2002-12-10 Pierre Humblet <pierre.humblet@ieee.org>
* security.h: Move declarations of internal_getgrent,
internal_getpwsid and internal_getgrsid to pwdgrp.h.
* pwdgrp.h: Declare internal_getpwsid, internal_getpwnam,
internal_getpwuid, internal_getgrsid, internal_getgrgid,
internal_getgrnam, internal_getgrent and internal_getgroups.
Delete "emulated" from enum pwdgrp_state.
(pwdgrp_check::isuninitialized): Create.
(pwdgrp_check::pwdgrp_state): Change state to initializing
rather than to uninitialized.
(pwdgrp_read::gets): Remove trailing CRs.
* passwd.cc (grab_string): Don't look for NLs.
(grab_int): Ditto.
(parse_pwd): Don't look for CRs. Return 0 if entry is too short.
(search_for): Delete.
(read_etc_passwd): Simplify tests to actually read the file.
Set state to loaded before making internal_getpwXX calls.
Replace search_for calls by equivalent internal_pwgetXX calls.
(internal_getpwsid): Use passwd_state.isuninitialized to decide
to call read_etc_passwd.
(internal_getpwuid): Create.
(internal_getpwnam): Create.
(getpwuid32): Simply call internal_getpwuid.
(getpwuid_r32): Call internal_getpwuid.
(getpwnam): Simply call internal_getpwnam.
(getpwnam_r): Call internal_getpwnam.
* grp.cc (parse_grp): Don't look for CRs. Adjust blank space.
(add_grp_line): Adjust blank space.
(class group_lock): Ditto.
(read_etc_group): Simplify tests to actually read the file.
Set state to loaded before making internal_getgrXX calls.
Replace getgrXX calls by equivalent internal calls.
(internal_getgrsid): Use group_state.isuninitialized to decide
to call read_etc_group.
(internal_getgrgid): Create.
(internal_getgrnam): Create.
(getgroups32): Simply call internal_getgrgid.
(getgrnam32): Simply call internal_getgrnam.
(internal_getgrent): Call group_state.isuninitialized.
(internal_getgroups): Create from the former getgroups32, using
two of the four arguments. Set gid to myself->gid and username
to cygheap->user.name ().
(getgroups32): Simply call internal_getgroup.
(getgroups): Call internal_getgroup instead of getgroups32.
(setgroups32): Call internal versions of get{pw,gr}XX.
* sec_helper.cc: Include pwdgrp.h.
(is_grp_member): Call internal versions of get{pw,gr}XX.
* security.cc: Include pwdgrp.h.
(alloc_sd): Call internal versions of get{pw,gr}XX.
* syscalls.cc: Include pwdgrp.h.
(seteuid32): Call internal versions of get{pw,gr}XX.
(setegid32): Ditto.
* uinfo.cc: Include pwdgrp.h.
(internal_getlogin): Call internal versions of get{pw,gr}XX.
(cygheap_user::ontherange): Ditto.
* sec_acl.cc: Include pwdgrp.h.
(setacl): Call internal versions of get{pw,gr}XX.
(acl_access): Ditto and simplify logic.
(aclfromtext): Ditto.
2002-12-10 13:43:49 +01:00
|
|
|
/* These functions are needed to allow searching and walking through
|
|
|
|
the passwd and group lists */
|
2014-02-27 13:57:27 +01:00
|
|
|
extern struct passwd *internal_getpwsid (cygpsid &, cyg_ldap * = NULL);
|
2014-02-22 20:38:12 +01:00
|
|
|
extern struct passwd *internal_getpwsid_from_db (cygpsid &sid);
|
2014-02-27 13:57:27 +01:00
|
|
|
extern struct passwd *internal_getpwnam (const char *, cyg_ldap * = NULL);
|
|
|
|
extern struct passwd *internal_getpwuid (uid_t, cyg_ldap * = NULL);
|
|
|
|
extern struct group *internal_getgrsid (cygpsid &, cyg_ldap * = NULL);
|
2014-02-22 20:38:12 +01:00
|
|
|
extern struct group *internal_getgrsid_from_db (cygpsid &sid);
|
2014-02-27 13:57:27 +01:00
|
|
|
extern struct group *internal_getgrgid (gid_t, cyg_ldap * = NULL);
|
|
|
|
extern struct group *internal_getgrnam (const char *, cyg_ldap * = NULL);
|
2002-12-10 Pierre Humblet <pierre.humblet@ieee.org>
* pwdgrp.h (pwdgrp_check::pwdgrp_state): Replace by
pwdgrp_check::isinitializing ().
(pwdgrp_check::isinitializing): Create.
* passwd.cc (grab_int): Change type to unsigned, use strtoul and
set the pointer content to 0 if the field is invalid.
(parse_pwd): Move validity test after getting pw_gid.
(read_etc_passwd): Replace "passwd_state <= " by
passwd_state::isinitializing ().
(internal_getpwuid): Ditto.
(internal_getpwnam): Ditto.
(getpwent): Ditto.
(getpass): Ditto.
* grp.cc (parse_grp): Use strtoul for gr_gid and verify the validity.
(read_etc_group): Replace "group_state <= " by
group_state::isinitializing ().
(internal_getgrgid): Ditto.
(getgrent32): Ditto.
(internal_getgrent): Ditto.
2002-12-10 Pierre Humblet <pierre.humblet@ieee.org>
* security.h: Move declarations of internal_getgrent,
internal_getpwsid and internal_getgrsid to pwdgrp.h.
* pwdgrp.h: Declare internal_getpwsid, internal_getpwnam,
internal_getpwuid, internal_getgrsid, internal_getgrgid,
internal_getgrnam, internal_getgrent and internal_getgroups.
Delete "emulated" from enum pwdgrp_state.
(pwdgrp_check::isuninitialized): Create.
(pwdgrp_check::pwdgrp_state): Change state to initializing
rather than to uninitialized.
(pwdgrp_read::gets): Remove trailing CRs.
* passwd.cc (grab_string): Don't look for NLs.
(grab_int): Ditto.
(parse_pwd): Don't look for CRs. Return 0 if entry is too short.
(search_for): Delete.
(read_etc_passwd): Simplify tests to actually read the file.
Set state to loaded before making internal_getpwXX calls.
Replace search_for calls by equivalent internal_pwgetXX calls.
(internal_getpwsid): Use passwd_state.isuninitialized to decide
to call read_etc_passwd.
(internal_getpwuid): Create.
(internal_getpwnam): Create.
(getpwuid32): Simply call internal_getpwuid.
(getpwuid_r32): Call internal_getpwuid.
(getpwnam): Simply call internal_getpwnam.
(getpwnam_r): Call internal_getpwnam.
* grp.cc (parse_grp): Don't look for CRs. Adjust blank space.
(add_grp_line): Adjust blank space.
(class group_lock): Ditto.
(read_etc_group): Simplify tests to actually read the file.
Set state to loaded before making internal_getgrXX calls.
Replace getgrXX calls by equivalent internal calls.
(internal_getgrsid): Use group_state.isuninitialized to decide
to call read_etc_group.
(internal_getgrgid): Create.
(internal_getgrnam): Create.
(getgroups32): Simply call internal_getgrgid.
(getgrnam32): Simply call internal_getgrnam.
(internal_getgrent): Call group_state.isuninitialized.
(internal_getgroups): Create from the former getgroups32, using
two of the four arguments. Set gid to myself->gid and username
to cygheap->user.name ().
(getgroups32): Simply call internal_getgroup.
(getgroups): Call internal_getgroup instead of getgroups32.
(setgroups32): Call internal versions of get{pw,gr}XX.
* sec_helper.cc: Include pwdgrp.h.
(is_grp_member): Call internal versions of get{pw,gr}XX.
* security.cc: Include pwdgrp.h.
(alloc_sd): Call internal versions of get{pw,gr}XX.
* syscalls.cc: Include pwdgrp.h.
(seteuid32): Call internal versions of get{pw,gr}XX.
(setegid32): Ditto.
* uinfo.cc: Include pwdgrp.h.
(internal_getlogin): Call internal versions of get{pw,gr}XX.
(cygheap_user::ontherange): Ditto.
* sec_acl.cc: Include pwdgrp.h.
(setacl): Call internal versions of get{pw,gr}XX.
(acl_access): Ditto and simplify logic.
(aclfromtext): Ditto.
2002-12-10 13:43:49 +01:00
|
|
|
|
2015-02-20 16:13:46 +01:00
|
|
|
extern int internal_getgroups (int, gid_t *, cyg_ldap *, const DWORD = 0);
|
2014-02-28 12:37:02 +01:00
|
|
|
|
2014-02-18 20:39:48 +01:00
|
|
|
/* These functions are called from mkpasswd/mkgroup via cygwin_internal. */
|
|
|
|
void *setpwent_filtered (int enums, PCWSTR enum_tdoms);
|
|
|
|
void *getpwent_filtered (void *gr);
|
|
|
|
void endpwent_filtered (void *gr);
|
|
|
|
void *setgrent_filtered (int enums, PCWSTR enum_tdoms);
|
|
|
|
void *getgrent_filtered (void *gr);
|
|
|
|
void endgrent_filtered (void *gr);
|
|
|
|
|
2014-02-09 20:44:56 +01:00
|
|
|
struct pg_pwd
|
|
|
|
{
|
|
|
|
struct passwd p;
|
|
|
|
cygsid sid;
|
2014-05-07 13:00:00 +02:00
|
|
|
size_t len;
|
2014-02-09 20:44:56 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct pg_grp
|
|
|
|
{
|
|
|
|
struct group g;
|
|
|
|
cygsid sid;
|
2014-05-07 13:00:00 +02:00
|
|
|
size_t len;
|
2014-02-09 20:44:56 +01:00
|
|
|
};
|
|
|
|
|
2003-01-17 06:18:30 +01:00
|
|
|
class pwdgrp
|
|
|
|
{
|
2014-02-17 16:36:33 +01:00
|
|
|
friend class pg_ent;
|
|
|
|
friend class pw_ent;
|
|
|
|
friend class gr_ent;
|
|
|
|
|
2003-01-21 07:58:11 +01:00
|
|
|
unsigned pwdgrp_buf_elem_size;
|
2014-02-09 20:44:56 +01:00
|
|
|
void *pwdgrp_buf;
|
2003-01-24 04:53:46 +01:00
|
|
|
bool (pwdgrp::*parse) ();
|
2014-02-09 20:44:56 +01:00
|
|
|
UNICODE_STRING path;
|
|
|
|
OBJECT_ATTRIBUTES attr;
|
|
|
|
LARGE_INTEGER last_modified;
|
|
|
|
char *lptr;
|
|
|
|
ULONG curr_lines;
|
|
|
|
ULONG max_lines;
|
2005-04-05 08:04:57 +02:00
|
|
|
static muto pglock;
|
2003-01-20 03:57:54 +01:00
|
|
|
|
2003-01-24 04:53:46 +01:00
|
|
|
bool parse_passwd ();
|
|
|
|
bool parse_group ();
|
|
|
|
char *add_line (char *);
|
2003-01-26 06:38:38 +01:00
|
|
|
char *raw_ptr () const {return lptr;}
|
2003-01-27 01:16:01 +01:00
|
|
|
char *next_str (char);
|
2003-01-26 06:38:38 +01:00
|
|
|
bool next_num (unsigned long&);
|
|
|
|
bool next_num (unsigned int& i)
|
|
|
|
{
|
|
|
|
unsigned long x;
|
|
|
|
bool res = next_num (x);
|
|
|
|
i = (unsigned int) x;
|
|
|
|
return res;
|
|
|
|
}
|
2003-04-10 07:27:34 +02:00
|
|
|
inline bool next_num (int& i)
|
2003-01-26 06:38:38 +01:00
|
|
|
{
|
|
|
|
unsigned long x;
|
|
|
|
bool res = next_num (x);
|
|
|
|
i = (int) x;
|
|
|
|
return res;
|
|
|
|
}
|
2014-02-17 16:36:33 +01:00
|
|
|
void *add_account_post_fetch (char *line, bool lock);
|
2014-02-09 20:44:56 +01:00
|
|
|
void *add_account_from_file (cygpsid &sid);
|
|
|
|
void *add_account_from_file (const char *name);
|
|
|
|
void *add_account_from_file (uint32_t id);
|
2014-03-07 21:38:48 +01:00
|
|
|
void *add_account_from_windows (cygpsid &sid, cyg_ldap *pldap = NULL);
|
|
|
|
void *add_account_from_windows (const char *name, cyg_ldap *pldap = NULL);
|
|
|
|
void *add_account_from_windows (uint32_t id, cyg_ldap *pldap = NULL);
|
2014-03-12 18:36:56 +01:00
|
|
|
void *add_account_from_cygserver (cygpsid &sid);
|
|
|
|
void *add_account_from_cygserver (const char *name);
|
|
|
|
void *add_account_from_cygserver (uint32_t id);
|
2014-02-09 20:44:56 +01:00
|
|
|
char *fetch_account_from_line (fetch_user_arg_t &arg, const char *line);
|
|
|
|
char *fetch_account_from_file (fetch_user_arg_t &arg);
|
2014-03-07 21:38:48 +01:00
|
|
|
char *fetch_account_from_windows (fetch_user_arg_t &arg,
|
2014-02-27 13:57:27 +01:00
|
|
|
cyg_ldap *pldap = NULL);
|
2014-03-12 18:36:56 +01:00
|
|
|
char *fetch_account_from_cygserver (fetch_user_arg_t &arg);
|
2001-09-09 18:52:37 +02:00
|
|
|
|
|
|
|
public:
|
2014-02-09 20:44:56 +01:00
|
|
|
ULONG cached_users () const { return curr_lines; }
|
|
|
|
ULONG cached_groups () const { return curr_lines; }
|
2014-02-17 16:36:33 +01:00
|
|
|
POBJECT_ATTRIBUTES file_attr () { return &attr; }
|
2014-03-07 21:38:48 +01:00
|
|
|
bool check_file ();
|
2003-01-20 03:57:54 +01:00
|
|
|
|
2014-02-09 20:44:56 +01:00
|
|
|
void init_pwd ();
|
2014-03-07 21:38:48 +01:00
|
|
|
bool is_passwd () const { return pwdgrp_buf_elem_size == sizeof (pg_pwd); }
|
2014-02-09 20:44:56 +01:00
|
|
|
pg_pwd *passwd () const { return (pg_pwd *) pwdgrp_buf; };
|
2014-03-12 18:36:56 +01:00
|
|
|
struct passwd *add_user_from_cygserver (cygpsid &sid)
|
|
|
|
{ return (struct passwd *) add_account_from_cygserver (sid); }
|
|
|
|
struct passwd *add_user_from_cygserver (const char *name)
|
|
|
|
{ return (struct passwd *) add_account_from_cygserver (name); }
|
|
|
|
struct passwd *add_user_from_cygserver (uint32_t id)
|
|
|
|
{ return (struct passwd *) add_account_from_cygserver (id); }
|
|
|
|
struct passwd *add_user_from_file (cygpsid &sid)
|
2014-02-09 20:44:56 +01:00
|
|
|
{ return (struct passwd *) add_account_from_file (sid); }
|
|
|
|
struct passwd *add_user_from_file (const char *name)
|
|
|
|
{ return (struct passwd *) add_account_from_file (name); }
|
|
|
|
struct passwd *add_user_from_file (uint32_t id)
|
|
|
|
{ return (struct passwd *) add_account_from_file (id); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct passwd *add_user_from_windows (cygpsid &sid, cyg_ldap *pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct passwd *) add_account_from_windows (sid, pldap); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct passwd *add_user_from_windows (const char *name,
|
|
|
|
cyg_ldap* pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct passwd *) add_account_from_windows (name, pldap); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct passwd *add_user_from_windows (uint32_t id, cyg_ldap *pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct passwd *) add_account_from_windows (id, pldap); }
|
2014-02-09 20:44:56 +01:00
|
|
|
struct passwd *find_user (cygpsid &sid);
|
|
|
|
struct passwd *find_user (const char *name);
|
|
|
|
struct passwd *find_user (uid_t uid);
|
|
|
|
|
|
|
|
void init_grp ();
|
2014-03-07 21:38:48 +01:00
|
|
|
bool is_group () const { return pwdgrp_buf_elem_size == sizeof (pg_grp); }
|
2014-02-09 20:44:56 +01:00
|
|
|
pg_grp *group () const { return (pg_grp *) pwdgrp_buf; };
|
2014-03-12 18:36:56 +01:00
|
|
|
struct group *add_group_from_cygserver (cygpsid &sid)
|
|
|
|
{ return (struct group *) add_account_from_cygserver (sid); }
|
|
|
|
struct group *add_group_from_cygserver (const char *name)
|
|
|
|
{ return (struct group *) add_account_from_cygserver (name); }
|
|
|
|
struct group *add_group_from_cygserver (uint32_t id)
|
|
|
|
{ return (struct group *) add_account_from_cygserver (id); }
|
2014-02-09 20:44:56 +01:00
|
|
|
struct group *add_group_from_file (cygpsid &sid)
|
|
|
|
{ return (struct group *) add_account_from_file (sid); }
|
|
|
|
struct group *add_group_from_file (const char *name)
|
|
|
|
{ return (struct group *) add_account_from_file (name); }
|
|
|
|
struct group *add_group_from_file (uint32_t id)
|
|
|
|
{ return (struct group *) add_account_from_file (id); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct group *add_group_from_windows (cygpsid &sid, cyg_ldap *pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct group *) add_account_from_windows (sid, pldap); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct group *add_group_from_windows (const char *name,
|
|
|
|
cyg_ldap *pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct group *) add_account_from_windows (name, pldap); }
|
2014-02-27 13:57:27 +01:00
|
|
|
struct group *add_group_from_windows (uint32_t id, cyg_ldap *pldap = NULL)
|
2014-03-07 21:38:48 +01:00
|
|
|
{ return (struct group *) add_account_from_windows (id, pldap); }
|
2015-02-23 21:51:12 +01:00
|
|
|
struct group *add_group_from_windows (fetch_full_grp_t &full_grp,
|
|
|
|
cyg_ldap *pldap = NULL);
|
2014-02-09 20:44:56 +01:00
|
|
|
struct group *find_group (cygpsid &sid);
|
|
|
|
struct group *find_group (const char *name);
|
|
|
|
struct group *find_group (gid_t gid);
|
|
|
|
};
|
2014-02-17 16:36:33 +01:00
|
|
|
|
|
|
|
class pg_ent
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
pwdgrp pg;
|
|
|
|
bool group;
|
|
|
|
pg_pwd pwd;
|
|
|
|
pg_grp grp;
|
|
|
|
NT_readline rl;
|
|
|
|
cyg_ldap cldap;
|
|
|
|
PCHAR buf;
|
|
|
|
ULONG cnt;
|
|
|
|
ULONG max;
|
|
|
|
ULONG_PTR resume;
|
2014-02-22 20:38:12 +01:00
|
|
|
int enums; /* ENUM_xxx values defined in sys/cygwin.h. */
|
2014-02-17 16:36:33 +01:00
|
|
|
PCWSTR enum_tdoms;
|
|
|
|
bool from_files;
|
|
|
|
bool from_db;
|
|
|
|
enum {
|
|
|
|
rewound = 0,
|
|
|
|
from_cache,
|
|
|
|
from_file,
|
|
|
|
from_builtin,
|
|
|
|
from_local,
|
|
|
|
from_sam,
|
|
|
|
from_ad,
|
|
|
|
finished
|
|
|
|
} state;
|
|
|
|
|
|
|
|
void clear_cache ();
|
|
|
|
inline bool nss_db_enum_caches () const { return !!(enums & ENUM_CACHE); }
|
|
|
|
inline bool nss_db_enum_files () const { return !!(enums & ENUM_FILES); }
|
|
|
|
inline bool nss_db_enum_builtin () const { return !!(enums & ENUM_BUILTIN); }
|
|
|
|
inline bool nss_db_enum_local () const { return !!(enums & ENUM_LOCAL); }
|
|
|
|
inline bool nss_db_enum_primary () const { return !!(enums & ENUM_PRIMARY); }
|
|
|
|
inline bool nss_db_enum_tdom (PWCHAR domain)
|
|
|
|
{
|
|
|
|
if (enums & ENUM_TDOMS_ALL)
|
|
|
|
return true;
|
|
|
|
if (!(enums & ENUM_TDOMS) || !enum_tdoms || !domain)
|
|
|
|
return false;
|
|
|
|
for (PCWSTR td = enum_tdoms; td && *td; td = wcschr (td, L'\0'))
|
|
|
|
if (!wcscasecmp (td, domain))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
virtual void *enumerate_caches () = 0;
|
|
|
|
virtual void *enumerate_file ();
|
|
|
|
virtual void *enumerate_builtin ();
|
|
|
|
virtual void *enumerate_local () = 0;
|
|
|
|
virtual void *enumerate_sam ();
|
|
|
|
virtual void *enumerate_ad ();
|
|
|
|
|
|
|
|
public:
|
|
|
|
void setent (bool _group, int _enums = 0, PCWSTR _enum_tdoms = NULL);
|
|
|
|
void *getent ();
|
|
|
|
void endent (bool _group);
|
|
|
|
};
|
|
|
|
|
|
|
|
class pw_ent : public pg_ent
|
|
|
|
{
|
|
|
|
void *enumerate_caches ();
|
|
|
|
void *enumerate_local ();
|
|
|
|
public:
|
|
|
|
inline void setpwent (int _enums = 0, PCWSTR _enum_tdoms = NULL)
|
|
|
|
{ setent (false, _enums, _enum_tdoms); }
|
|
|
|
struct passwd *getpwent ();
|
|
|
|
inline void endpwent () { endent (false); }
|
|
|
|
};
|
|
|
|
|
|
|
|
class gr_ent : public pg_ent
|
|
|
|
{
|
|
|
|
void *enumerate_caches ();
|
|
|
|
void *enumerate_local ();
|
|
|
|
public:
|
|
|
|
inline void setgrent (int _enums = 0, PCWSTR _enum_tdoms = NULL)
|
|
|
|
{ setent (true, _enums, _enum_tdoms); }
|
|
|
|
struct group *getgrent ();
|
|
|
|
inline void endgrent () { endent (true); }
|
|
|
|
};
|