| #ifndef _XOPEN_SOURCE |
| #define _XOPEN_SOURCE 700 |
| #endif |
| #include <errno.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <time.h> |
| #include <sys/types.h> |
| #include <sys/shm.h> |
| #include <sys/wait.h> |
| #include <unistd.h> |
| #include "test.h" |
| |
| static const char path[] = "."; |
| static const int id = 'h'; |
| |
| #define T(f) do{ \ |
| if ((f)+1 == 0) \ |
| t_error("%s failed: %s\n", #f, strerror(errno)); \ |
| }while(0) |
| |
| #define EQ(a,b,fmt) do{ \ |
| if ((a) != (b)) \ |
| t_error("%s == %s failed: " fmt "\n", #a, #b, a, b); \ |
| }while(0) |
| |
| static void set() |
| { |
| time_t t; |
| key_t k; |
| int shmid; |
| struct shmid_ds shmid_ds; |
| void *p; |
| |
| T(t = time(0)); |
| T(k = ftok(path, id)); |
| |
| |
| T(shmid = shmget(k, 100, IPC_CREAT|0666)); |
| T(shmctl(shmid, IPC_RMID, 0)); |
| T(shmid = shmget(k, 100, IPC_CREAT|IPC_EXCL|0666)); |
| |
| if (t_status) |
| exit(t_status); |
| |
| |
| errno = 0; |
| if (shmget(k, 100, IPC_CREAT|IPC_EXCL|0666) != -1 || errno != EEXIST) |
| t_error("shmget(IPC_CREAT|IPC_EXCL) should have failed with EEXIST, got %s\n", strerror(errno)); |
| |
| |
| T(shmctl(shmid, IPC_STAT, &shmid_ds)); |
| EQ(shmid_ds.shm_perm.cuid, geteuid(), "got %d, want %d"); |
| EQ(shmid_ds.shm_perm.uid, geteuid(), "got %d, want %d"); |
| EQ(shmid_ds.shm_perm.cgid, getegid(), "got %d, want %d"); |
| EQ(shmid_ds.shm_perm.gid, getegid(), "got %d, want %d"); |
| EQ(shmid_ds.shm_perm.mode & 0x1ff, 0666, "got %o, want %o"); |
| EQ(shmid_ds.shm_segsz, 100, "got %d, want %d"); |
| EQ(shmid_ds.shm_lpid, 0, "got %d, want %d"); |
| EQ(shmid_ds.shm_cpid, getpid(), "got %d, want %d"); |
| EQ((int)shmid_ds.shm_nattch, 0, "got %d, want %d"); |
| EQ((long)shmid_ds.shm_atime, 0, "got %ld, want %d"); |
| EQ((long)shmid_ds.shm_dtime, 0, "got %ld, want %d"); |
| if (shmid_ds.shm_ctime < t) |
| t_error("shmid_ds.shm_ctime >= t failed: got %ld, want >= %ld\n", (long)shmid_ds.shm_ctime, (long)t); |
| if (shmid_ds.shm_ctime > t+5) |
| t_error("shmid_ds.shm_ctime <= t+5 failed: got %ld, want <= %ld\n", (long)shmid_ds.shm_ctime, (long)t+5); |
| |
| |
| if ((p=shmat(shmid, 0, 0)) == 0) |
| t_error("shmat failed: %s\n", strerror(errno)); |
| T(shmctl(shmid, IPC_STAT, &shmid_ds)); |
| EQ((int)shmid_ds.shm_nattch, 1, "got %d, want %d"); |
| EQ(shmid_ds.shm_lpid, getpid(), "got %d, want %d"); |
| if (shmid_ds.shm_atime < t) |
| t_error("shm_atime is %ld want >= %ld\n", (long)shmid_ds.shm_atime, (long)t); |
| if (shmid_ds.shm_atime > t+5) |
| t_error("shm_atime is %ld want <= %ld\n", (long)shmid_ds.shm_atime, (long)t+5); |
| strcpy(p, "test data"); |
| T(shmdt(p)); |
| } |
| |
| static void get() |
| { |
| key_t k; |
| int shmid; |
| void *p; |
| |
| T(k = ftok(path, id)); |
| T(shmid = shmget(k, 0, 0)); |
| |
| errno = 0; |
| if ((p=shmat(shmid, 0, SHM_RDONLY)) == 0) |
| t_error("shmat failed: %s\n", strerror(errno)); |
| |
| if (strcmp(p, "test data") != 0) |
| t_error("reading shared mem failed: got \"%.100s\" want \"test data\"\n", p); |
| |
| |
| T(shmdt(p)); |
| T(shmctl(shmid, IPC_RMID, 0)); |
| } |
| |
| int main(void) |
| { |
| int p; |
| int status; |
| |
| set(); |
| p = fork(); |
| if (p == -1) |
| t_error("fork failed: %s\n", strerror(errno)); |
| else if (p == 0) |
| get(); |
| else { |
| T(waitpid(p, &status, 0)); |
| if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) |
| t_error("child exit status: %d\n", status); |
| } |
| return t_status; |
| } |