diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2015-09-17 22:23:53 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2015-09-17 22:23:53 -0700 |
commit | 86d3b7542e1dffa525f0d131d1ef26440055f39e (patch) | |
tree | 6b8ffa84dedef15ed83b15b872cf7b376ffb1cb3 /sysif.c | |
parent | 7959a22f6f5434dcbc505c1a8f7b560c5b621077 (diff) | |
download | txr-86d3b7542e1dffa525f0d131d1ef26440055f39e.tar.gz txr-86d3b7542e1dffa525f0d131d1ef26440055f39e.tar.bz2 txr-86d3b7542e1dffa525f0d131d1ef26440055f39e.zip |
Add unix group database functions.
* configure (have_grgid): New variable.
New tests added for getgrent and the rest.
(HAVE_GRGID, HAVE_GRGID_R): New preprocessor symbols
conditionally deposited into config/config.h.
* sysif.c (group_s, mem_s): New global symbol variables.
(setgrent_wrap, endgrent_wrap, fill_group, make_grstruct,
get_grent_wrap, getgrgid_wrap, getgrnam_wrap): New static
functions.
(sysif_init): New global symbol variables initialized.
New group struct type instantiated.
Intrinsic functions setgrent, endgrent, getgrent,
getgrgid and getgrnam registered.
* txr.1: Documented group structure and functions.
Diffstat (limited to 'sysif.c')
-rw-r--r-- | sysif.c | 110 |
1 files changed, 110 insertions, 0 deletions
@@ -59,6 +59,9 @@ #if HAVE_PWUID #include <pwd.h> #endif +#if HAVE_GRGID +#include <grp.h> +#endif #include ALLOCA_H #include "lib.h" #include "stream.h" @@ -85,6 +88,10 @@ val atime_s, mtime_s, ctime_s; val passwd_s, name_s, gecos_s, dir_s, shell_s; #endif +#if HAVE_GRGID +val group_s, mem_s; +#endif + static val errno_wrap(val newval) { val oldval = num(errno); @@ -944,6 +951,93 @@ static val getpwnam_wrap(val wname) #endif +#if HAVE_GRGID + +static val setgrent_wrap(void) +{ + setgrent(); + return nil; +} + +static val endgrent_wrap(void) +{ + endgrent(); + return nil; +} + +static void fill_group(val to, struct group *from) +{ + list_collect_decl (out, ptail); + int i; + + slotset(to, name_s, string_utf8(from->gr_name)); + slotset(to, passwd_s, string_utf8(from->gr_passwd)); + slotset(to, gid_s, num(from->gr_gid)); + + for (i = 0; from->gr_mem[i] != 0; i++) + ptail = list_collect(ptail, string_utf8(from->gr_mem[i])); + + slotset(to, mem_s, out); +} + +static val make_grstruct(struct group *g) +{ + args_decl(args, ARGS_MIN); + val out = make_struct(group_s, nil, args); + fill_group(out, g); + return out; +} + + +static val getgrent_wrap(void) +{ + struct group *g = getgrent(); + return (g != 0) ? make_grstruct(g) : nil; +} + + +#endif + +#if HAVE_GRGID_R + +static val getgrgid_wrap(val uid) +{ + char buf[1024]; + struct group gr, *g; + int res = getgrgid_r(c_num(uid), &gr, buf, sizeof buf, &g); + + return (res == 0 && g != 0) ? make_grstruct(&gr) : nil; +} + +static val getgrnam_wrap(val wname) +{ + char buf[1024]; + struct group gr, *g; + char *name = utf8_dup_to(c_str(wname)); + int res = getgrnam_r(name, &gr, buf, sizeof buf, &g); + + free(name); + return (res == 0 && g != 0) ? make_grstruct(&gr) : nil; +} + +#elif HAVE_GRGID + +static val getgrgid_wrap(val uid) +{ + struct group *g = getgrgid(c_num(uid)); + return (g != 0) ? make_grstruct(g) : nil; +} + +static val getgrnam_wrap(val wname) +{ + char *name = utf8_dup_to(c_str(wname)); + struct group *g = getgrnam(name); + free(name); + return (g != 0) ? make_grstruct(g) : nil; +} + +#endif + void sysif_init(void) { stat_s = intern(lit("stat"), user_package); @@ -980,6 +1074,10 @@ void sysif_init(void) dir_s = intern(lit("dir"), user_package); shell_s = intern(lit("shell"), user_package); #endif +#if HAVE_GRGID + group_s = intern(lit("group"), user_package); + mem_s = intern(lit("mem"), user_package); +#endif make_struct_type(stat_s, nil, list(dev_s, ino_s, mode_s, nlink_s, uid_s, gid_s, @@ -990,6 +1088,10 @@ void sysif_init(void) list(name_s, passwd_s, uid_s, gid_s, gecos_s, dir_s, shell_s, nao), nil, nil); #endif +#if HAVE_GRGID + make_struct_type(group_s, nil, + list(name_s, passwd_s, gid_s, mem_s, nao), nil, nil); +#endif reg_fun(intern(lit("errno"), user_package), func_n1o(errno_wrap, 0)); reg_fun(intern(lit("exit"), user_package), func_n1(exit_wrap)); @@ -1196,6 +1298,14 @@ void sysif_init(void) reg_fun(intern(lit("getpwnam"), user_package), func_n1(getpwnam_wrap)); #endif +#if HAVE_GRGID + reg_fun(intern(lit("setgrent"), user_package), func_n0(setgrent_wrap)); + reg_fun(intern(lit("endgrent"), user_package), func_n0(endgrent_wrap)); + reg_fun(intern(lit("getgrent"), user_package), func_n0(getgrent_wrap)); + reg_fun(intern(lit("getgrgid"), user_package), func_n1(getgrgid_wrap)); + reg_fun(intern(lit("getgrnam"), user_package), func_n1(getgrnam_wrap)); +#endif + #if HAVE_POLL reg_fun(intern(lit("poll"), user_package), func_n2o(poll_wrap, 1)); #endif |