I'm writing some highly portable security code. I'm trying to avoid security flaw in a utility program such as this one found in some versions of sudo
:
... it is possible to become the super user by running
sudo -k
and then resetting the system clock to 01-01-1970.
This happens because sudo
relies on absolute (aka calendar) time to determine whether or not access has timed out.
My idea is to use CLOCK_MONOTONIC
defined in time.h.
From the POSIX standard,
[CLOCK_MONOTONIC is] defined as a clock whose value cannot be set via clock_settime() and which cannot have backward clock jumps. The maximum possible clock jump shall be implementation-defined.
Problem is, on many (most?) systems, CLOCK_MONOTONIC
resets on reboot. Is there any guaranteed POSIX-compliant way to determine whether or not the system has rebooted since a program last ran?
One (bad) way is to check whether or not the stored clock value is greater than the current clock value, however this just shifts the problem. On systems where CLOCK_MONOTONIC
resets on reboot, there could be a short window of length TIMEOUT
where access would be permitted.
What am I missing that would avoid this problem?
It seems to me this is straightforward to do using a POSIX shared memory object:
Whenever your program launches it can
shm_open
a new object with some consistent name and set the owner toroot
. The object needn't contain any particular value. POSIX requires that all shared memory objects persist until reboot unless manually destroyed (which only its owner or creator can do...which in this case is the root user).Whenever your program launches it first checks if such a shared memory object already exists having root as the owner. Since only root could create such an object, and only root or a reboot could destroy it, you can thus know for certain whether your program has been launched since the last reboot, save the only possible circumvention being the root user invoking
shm_unlink
on the object manually.I wrote a test-and-set function below that should do exactly what you need. And it works except for the ownership setting/detection: for some unknown reason both calls to
shmctl
are failing on my system, saying "invalid argument". Theman
page forshmctl
says theEINVAL
error indicates either an invalid memory object identifier or an invalid command. But theIPC_SET
andIPC_STAT
commands are certainly valid, and you can watch the program's output to see the valid object identifier that is being created and/or opened each time.If anyone has any clues, I'm stumped. Some information and examples for POSIX shared memory here.