Added a little more compatibility code for non-standard systems.
[rsync/rsync.git] / uidlist.c
index bee3538..d4cb7a0 100644 (file)
--- a/uidlist.c
+++ b/uidlist.c
 
 #include "rsync.h"
 
+#ifdef GETGROUPS_T
+# ifndef NGROUPS_MAX
+/* It ought to be defined, but just in case. */
+#  define NGROUPS_MAX 32
+# endif
+#endif
+
 extern int preserve_uid;
 extern int preserve_gid;
 extern int numeric_ids;
@@ -90,7 +97,8 @@ static uid_t match_uid(uid_t uid)
        static uid_t last_in, last_out;
        struct idlist *list = uidlist;
 
-       if (uid == last_in) return last_out;
+       if (uid == last_in)
+               return last_out;
 
        last_in = uid;
 
@@ -109,7 +117,7 @@ static uid_t match_uid(uid_t uid)
 static int is_in_group(gid_t gid)
 {
 #ifdef GETGROUPS_T
-       static gid_t last_in = (gid_t) -2, last_out;
+       static gid_t last_in = GID_NONE, last_out;
        static int ngroups = -2;
        static GETGROUPS_T *gidset;
        int n;
@@ -117,35 +125,45 @@ static int is_in_group(gid_t gid)
        if (gid == last_in)
                return last_out;
        if (ngroups < -1) {
-               /* treat failure (-1) as if not member of any group */
+               gid_t mygid = getgid();
                ngroups = getgroups(0, 0);
-               if (ngroups > 0) {
-                       gidset = new_array(GETGROUPS_T, ngroups);
+               /* If that didn't work, perhaps 0 isn't treated specially? */
+               if (ngroups <= 0)
+                       ngroups = NGROUPS_MAX;
+               gidset = new_array(GETGROUPS_T, ngroups+1);
+               if (ngroups > 0)
                        ngroups = getgroups(ngroups, gidset);
+               /* The default gid might not be in the list on some systems. */
+               for (n = 0; n < ngroups; n++) {
+                       if (gidset[n] == mygid)
+                               break;
                }
+               if (n == ngroups)
+                       gidset[ngroups++] = mygid;
        }
 
        last_in = gid;
-       last_out = 0;
        for (n = 0; n < ngroups; n++) {
-               if (gidset[n] == gid) {
-                       last_out = 1;
-                       break;
-               }
+               if (gidset[n] == gid)
+                       return last_out = 1;
        }
-       return last_out;
+       return last_out = 0;
 
 #else
-       return 0;
+       static gid_t mygid = GID_NONE;
+       if (mygid == GID_NONE)
+               mygid = getgid();
+       return gid == mygid;
 #endif
 }
 
 static gid_t match_gid(gid_t gid)
 {
-       static gid_t last_in, last_out;
+       static gid_t last_in = GID_NONE, last_out = GID_NONE;
        struct idlist *list = gidlist;
 
-       if (gid == last_in) return last_out;
+       if (gid == last_in)
+               return last_out;
 
        last_in = gid;