In case of set-user-ID and set-group-ID programs, when the effective user-ID and group-ID are different from those of the real user, it is important to drop not only the user level privileges but also the group privileges. While doing so, the order of revocation must be correct.
| POSIX defines {{setgid()}} to have the following behaviour \[[Open Group 04|AA. C References#Open Group 04]\] | 
If the process has appropriate privileges,
setgid()shall set the real group ID, effective group ID, and the saved set-group-ID of the calling process togid.If the process does not have appropriate privileges, but
gidis equal to the real group ID or the saved set-group-ID,setgid()shall set the effective group ID togid; the real group ID and saved set-group-ID shall remain unchanged.
This non-compliant code example drops privileges to those of the real user and similarly also accounts for dropping the group privileges. However, the specified order is incorrect as the call to setuid() will leave the effective user ID as non zero. The setgid() system call in the next line should be run with superuser privileges, however, this call fails to behave as expected since the effective user ID is no longer that of the superuser (now non zero after the privilege drop in the previous line). In effect, if another flaw that allows execution of a setegid(0) or a setregid(-1,0) is found in the program, the attacker can regain the original group privileges because setgid(getgid()) tends to leave the saved set-group-ID intact under the conditions discussed.
| 
/* Drop superuser privileges in incorrect order */
if (setuid(getuid()) == -1) {
  /* handle error condition */
}
if (setgid(getgid()) == -1) {
  /* handle error condition */
}
/* It is still possible to regain group privileges due to incorrect relinquishment order */
 | 
Relinquish group privileges before taking away the user level privileges so that both operations execute as intended.
| 
/*  Drop superuser privileges in correct order */
if (setgid(getgid()) == -1) {
  /* handle error condition */
}
if (setuid(getuid()) == -1) {
  /* handle error condition */
}
/*  Not possible to regain group privileges due to correct relinquishment order  */
 | 
This rule captures avoidable mistakes which may otherwise lead to a false sense of code security and unintended privilege escalation.
| Rule | Severity | Likelihood | Remediation Cost | Priority | Level | 
|---|---|---|---|---|---|
| POS36-C | high | probable | medium | P12 | L1 | 
Search for vulnerabilities resulting from the violation of this rule on the CERT website.
| \[[Chen 02|AA. C References#Chen 02]\] Setuid Demystified
\[[Dowd 06|AA. C References#Dowd 06]\] Chapter 9, "Unix I: Privileges and Files"
\[[Open Group 04|AA. C References#Open Group 04]\] [{{setuid()}}|http://www.opengroup.org/onlinepubs/009695399/functions/setuid.html], [{{setgid()}}|http://www.opengroup.org/onlinepubs/009695399/functions/setgid.html] |