《CSAPP》(第3版)答案(第八章)
P9
process pair | cocurrent? |
---|---|
AB | No |
AC | Yes |
AD | Yes |
BC | Yes |
BD | Yes |
CD | Yes |
P10
A. call once, return twice
fork
B. call once, never return
longjmp, execve
C. call once, return 1 or more times
setjmp
P11
4行
c
+-------+
| "hello"
|
c | p
+-------+-------+
| fork "hello"
|
| c
| +-------+
| | "hello"
| |
| p | p
+------+-------+-------+
main fork fork "hello"
P12
8行
c
+-------+---------+
| "hello" "hello"
|
c | p
+-------+-------+---------+
| fork "hello" "hello"
|
| c
| +-------+---------+
| | "hello" "hello"
| |
| p | p
+------+-------+-------+---------+
main fork fork "hello" "hello"
P13
a = 4 a=4 a=4
a = 3 a=3 a=3
a = 2 a=2 a=2
P14
3行
c
+-------+
| "hello"
|
c | p
+-------+-------+
| fork "hello"
|
| p
+------+-------+-------+
main fork return "hello"
P15
5行
c
+-------+---------+
| "hello" "hello"
|
c | p
+-------+-------+---------+
| fork "hello" "hello"
|
|
| p
+------+-------+-------+
main fork return "hello"
P16
c o u n t e r = 2 counter = 2 counter=2
注意子进程有自己的计数器。
P17
hello 0 1 Bye 2 Bye
hello 1 0 Bye 2 Bye
hello 1 Bye 0 2 Bye
P18
c
+-------+---------+
| "0" exit "2"
|
c | p
+-------+-------+---------+
| fork "1" exit "2"
| (atexit)
| c
| +-------+---------+
| | "0" exit
| |
| p | p
+------+-------+-------+---------+
main fork fork "1" exit
2一定在0/1后面,所以是B、D
P19
2 n = 2 6 = 64 2^n=2^6=64 2n=26=64
P20
#include <stdio.h>
#include "csapp.h"
int main(int argc, char* argv[], char* env[]) {
if (execve("/bin/ls", argv, env) == -1) {
fprintf(stderr, "execve error: %s\n", strerror(errno));
exit(1);
}
}
P21
abc或bac
P22
#include <stdio.h>
#include "csapp.h"
int mysystem(char* command) {
pid_t pid;
int status;
if ((pid = Fork()) == 0) {
char* argv[4] = {
"", "-c", command, NULL };
execve("/bin/sh", argv, environ);
}
printf("child pid: %d\n", pid);
if (Waitpid(pid, &status, 0) > 0) {
if (WIFEXITED(status))
return WEXITSTATUS(status);
if (WIFSIGNALED(status))
return WTERMSIG(status);
}
}
int main(int argc, char* argv[]) {
int code;
code = mysystem("./exit-code");
printf("normally exit, code: %d\n", code); fflush(stdout);
code = mysystem("./wait-sig");
printf("exit caused by signal, code: %d\n", code); fflush(stdout);
return 0;
}
csapp.h
#ifndef __CSAPP_H__
#define __CSAPP_H__
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <unistd.h>
#include <string.h>
#include <ctype.h>
#include <setjmp.h>
#include <signal.h>
#include <dirent.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <errno.h>
#include <math.h>
#include <pthread.h>
#include <semaphore.h>
#include <sys/socket.h>
#include <netdb.h>
#include <netinet/in.h>
#include <arpa/inet.h>
/* Default file permissions are DEF_MODE & ~DEF_UMASK */
/* $begin createmasks */
#define DEF_MODE S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH
#define DEF_UMASK S_IWGRP|S_IWOTH
/* $end createmasks */
/* Simplifies calls to bind(), connect(), and accept() */
/* $begin sockaddrdef */
typedef struct sockaddr SA;
/* $end sockaddrdef */
/* Persistent state for the robust I/O (Rio) package */
/* $begin rio_t */
#define RIO_BUFSIZE 8192
typedef struct {
int rio_fd; /* Descriptor for this internal buf */
int rio_cnt; /* Unread bytes in internal buf */
char *rio_bufptr; /* Next unread byte in internal buf */
char rio_buf[RIO_BUFSIZE]; /* Internal buffer */
} rio_t;
/* $end rio_t */
/* External variables */
extern int h_errno; /* Defined by BIND for DNS errors */
extern char **environ; /* Defined by libc */
/* Misc constants */
#define MAXLINE 8192 /* Max text line length */
#define MAXBUF 8192 /* Max I/O buffer size */
#define LISTENQ 1024 /* Second argument to listen() */
/* Our own error-handling functions */
void unix_error(char *msg);
void posix_error(int code, char *msg);
void dns_error(char *msg);
void gai_error(int code, char *msg);
void app_error(char *msg);
/* Process control wrappers */
pid_t Fork(void);
void Execve(const char *filename, char *const argv[], char *const envp[]);
pid_t Wait(int *status);
pid_t Waitpid(pid_t pid, int *iptr, int options);
void Kill(pid_t pid, int signum);
unsigned int Sleep(unsigned int secs);
void Pause(void);
unsigned int Alarm(unsigned int seconds);
void Setpgid(pid_t pid, pid_t pgid);
pid_t Getpgrp();
/* Signal wrappers */
typedef void handler_t(int);
handler_t *Signal(int signum, handler_t *handler);
void Sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
void Sigemptyset(sigset_t *set);
void Sigfillset(sigset_t *set);
void Sigaddset(sigset_t *set, int signum);
void Sigdelset(sigset_t *set, int signum);
int Sigismember(const sigset_t *set, int signum);
int Sigsuspend(const sigset_t *set);
/* Sio (Signal-safe I/O) routines */
ssize_t sio_puts(char s[]);
ssize_t sio_putl(long v);
void sio_error(char s[]);
/* Sio wrappers */
ssize_t Sio_puts(char s[]);
ssize_t Sio_putl(long v);
void Sio_error(char s[]);
/* Unix I/O wrappers */
int Open(const char *pathname, int flags, mode_t mode);
ssize_t Read(int fd, void *buf, size_t count);
ssize_t Write(int fd, const void *buf, size_t count);
off_t Lseek(int fildes, off_t offset, int whence);
void Close(int fd);
int Select(int n, fd_set *readfds, fd_set *writefds, fd_set *exceptfds,
struct timeval *timeout);
int Dup2(int fd1, int fd2);
void Stat(const char *filename, struct stat *buf);
void Fstat(int fd, struct stat *buf) ;
/* Directory wrappers */
DIR *Opendir(const char *name);
struct dirent *Readdir(DIR *dirp);
int Closedir(DIR *dirp);
/* Memory mapping wrappers */
void *Mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);
void Munmap(void *start, size_t length);
/* Standard I/O wrappers */
void Fclose(FILE *fp);
FILE *Fdopen(int fd, const char *type);
char *Fgets(char *ptr, int n, FILE *stream);
FILE *Fopen(const char *filename, const char *mode);
void Fputs(const char *ptr, FILE *stream);
size_t Fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
void Fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);
/* Dynamic storage allocation wrappers */
void *Malloc(size_t size);
void *Realloc(void *ptr, size_t size);
void *Calloc(size_t nmemb, size_t size);
void Free(void *ptr);
/* Sockets interface wrappers */
int Socket(int domain, int