23

I understand that unix user IDs (UIDs) are usually 16 or 32 bit unsigned integers but how can I find out for any given system (in a shell)?

Josef
  • 393
  • 1
  • 3
  • 8

4 Answers4

14

You'll need to look in <limits.h> (or one of the files it includes, e.g., sys/syslimits.h on OS X) for the #define of UID_MAX.

Most recent operating systems (Solaris 2.x, OS X, BSD, Linux, HP-UX 11i, AIX 6) can handle up to two billion (2^31-2), so I would assume that and make a workaround for the more obscure systems that don't.

DM.
  • 164
  • 2
  • 1
    Unfortunately, there is no such thing as `UID_MAX`. For example, tools from `shadow-utils` use `(uid_t)-1` to find out the max value for UID. – kirelagin Mar 30 '14 at 09:12
  • 5
    Most systems use /etc/login.defs which does have UID_MAX set for the highest usable UID value, 60000 on any system I've checked. – Ryaner Aug 05 '14 at 11:21
  • 6
    The [manpage](http://linux.die.net/man/5/login.defs) for `login.defs` indicates that in that context, `UID_MAX` only controls the highest uid that will automatically be assigned for new users created with `useradd`. – Stephen Touset Jan 26 '15 at 19:34
  • 2
    It's probably 2^32 (4 billion instead of 2). On RHEL _UID 4,294,967,295 (2^32-1) is often reserved for an "invalid value" UID, and 4,294,967,294 (2^32-2) is reserved for the nfsnobody user in some operating systems. Thus the max non-reserved value is 4,294,967,293 (2^32-3)_ – tehnicaorg Sep 22 '18 at 22:06
6

glibc provides definitions for all those system types.

You can check /usr/include/bits/typesizes.h:

% grep UID_T /usr/include/bits/typesizes.h
#define __UID_T_TYPE            __U32_TYPE

Next you look into /usr/include/bits/types.h:

% grep '#define __U32_TYPE' /usr/include/bits/types.h
#define __U32_TYPE              unsigned int

This lets you find out the C type. Since you need the size in bytes, your best option is parsing the typedef name according to the specification in types.h:

We define __S<SIZE>_TYPE and __U<SIZE>_TYPE for the signed and unsigned
variants of each of the following integer types on this machine.

 16      -- "natural" 16-bit type (always short)
 32      -- "natural" 32-bit type (always int)
 64      -- "natural" 64-bit type (long or long long)
 LONG32      -- 32-bit type, traditionally long
 QUAD        -- 64-bit type, always long long
 WORD        -- natural type of __WORDSIZE bits (int or long)
 LONGWORD    -- type of __WORDSIZE bits, traditionally long

So, here is a one-liner:

% grep '#define __UID_T_TYPE' /usr/include/bits/typesizes.h | cut -f 3 | sed -r 's/__([US])([^_]*)_.*/\1 \2/'
U 32

Here U means unsigned (this can also be S for signed) and 32 is the size (look it up in the list above; I think, most of the time you can assume that that's already size in bytes, but if you want your script to be fully portable it might be better to do case switch on this value).

kirelagin
  • 209
  • 2
  • 7
  • 1
    On my system (Ubuntu 12.04) and other Debian based systems the header file is: `/usr/include/$(gcc -print-multiarch)/bits/typesizes.h` or alternatively: `/usr/include/$(dpkg-architecture -qDEB_HOST_MULTIARCH)/bits/typesizes.h` – pabouk - Ukraine stay strong Jul 07 '14 at 19:03
  • 1
    Having those glibc files would probably mean that there is a compiler available. So one could _#include _ to have access to _uid_t_ and print the result ( _printf("uid_t: %d bytes (%d bits)\n", sizeof(uid_t), sizeof(uid_t) * 8);_ ) – tehnicaorg Sep 22 '18 at 22:10
4

In this link the question is asked and a responder uses a trial & error method to determine the system in question uses a signed long int, leaving 31 bits to store the value, with a max of 2,147,483,647.

# groupadd -g 42949672950 testgrp
# more /etc/group
testgrp:*:2147483647:
Donald Byrd
  • 538
  • 5
  • 12
3

That's an interesting question. I'd be surprised if there was a standard, portable method to determine this.

I don't have a Linux box handy, but the id command on FreeBSD 8.0 wraps back to zero:

# id 4294967296
uid=0(root) gid=0(wheel) groups=0(wheel),5(operator)

I'm sure this is undefined behavior, but I'd wager that most versions of id would either wrap to zero with 65'536 (if 16-bit UID) and 4'294'967'296 or error out if you went beyond the system limit.

Geoff Fritz
  • 1,717
  • 9
  • 11