fork()
System Callmain
function:int main( int argc, char* argv[]);
argc
Specifies the number of command-line argumentsargv
is an array of string pointers that provide access to the command-line argument values (these are populated during start-up by a routine that obtains the values from the kernel)int
main(int argc, char *argv[])
{
int i;
for (i = 0; i < argc; i++) /* echo all command-line args */
printf("argv[%d]: %s\n", i, argv[i]);
exit(0);
}
argc
is used as a loop termination condition to print the contents of argv
to standard output.main
function or via a global variable extern char **environ;
.int main(int argc, char *argv[], char *envp[]);
char *envp[]
.extern char **environ;
extern char **environ;
.center[]
malloc
, calloc
).center[]
#include <stdlib.h>
// size_t is an integer (or long actually) type defined in stdlib
//Allocate a single block of size bytes
void *malloc(size_t size);
//Allocate an noObj lenth array of size bytes
void *calloc (size_t noObj, size_t size);
//Re-allocate the memory at ptr to be of newSize
void *realloc(void *ptr, size_T newSize);
//De-allocate and return the memory to the heap
void free(void *ptr)
pid_t
, which will either be a long or an int. (IS pid_t signed or unsigned? why/why not?)pid_t getpid(void)
#include <sys/types.h>
#include <unistd.h>
pid_t getpid(void);
getpid(...)
is part of an entire family of system calls:#include <sys/types.h>
#include <unistd.h>
// Calling Process ID
pid_t getpid(void);
// Parent of calling processes ID
pid_t getppid(void);
// Calling processes user ID
uid_t getuid(void);
// Calling processes effective user ID
uid_t geteuid(void);
// Calling processes group ID
gid_t getgid(void);
// Calling processes effective group ID
gig_t getegid(void);
getppid(void)
)getuid(void)
)uid_t geteuid(void);
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>
int main(void){
printf("Process ID: %ld\n", (long)getpid());
printf("Parent process ID: %ld\n", (long)getppid());
printf("Owner user ID: %ld\n", (long)getuid());
printf("Effective user ID: %ld\n", (long)geteuid());
return 0;
}
State | Menaing |
---|---|
New | Process is being created |
Running | The process' instructions are currently being executed |
Blocked | The process is waiting for an event (e.g. i/o). |
Ready | The process is waiting to run (i.e. can execute but currently scheduled of the processor) |
Done | The process is finished and is being de-allocated |
.center[]
fork()
System Call#include <sys/types.h>
#include <unistd.h>
pid_t fork ( void ); //Returns -1 on failure
fork()
System Callfork()
creates a new child process.fork()
call.fork()
System Callfork()
fork()
System Callfork()
can fail, resulting in a fork()
ing mess if you don’t check for errorsfork()
only returns once, with a -1errno
is also set:
EAGAIN
- you already have too many processes goingENOMEM
- you haven’t got enough space left for this processAlways check for fork()
ing failures!
make
is a UNIX utility for building application from source.make
utility uses the instructions in the makefile
to compile the codeCOMPILER = gcc
CFLAGS = -Wall -pedantic
EXES = example01 forkex01 forkex02 forkex03 forkex04 forkex05 forkex06
all: ${EXES}
example01: example01.c
${COMPILER} ${CFLAGS} example01.c -o example01
forkex01: forkex01.c
${COMPILER} ${CFLAGS} forkex01.c -o forkex01
forkex02: forkex02.c
${COMPILER} ${CFLAGS} forkex02.c -o forkex02
forkex03: forkex03.c
${COMPILER} ${CFLAGS} forkex03.c -o forkex03
forkex04: forkex04.c
${COMPILER} ${CFLAGS} forkex04.c -o forkex04
forkex05: forkex05.c
${COMPILER} ${CFLAGS} forkex05.c -o forkex05
forkex06: forkex06.c
${COMPILER} ${CFLAGS} forkex06.c -o forkex06
clean:
rm -f *~ *.o ${EXES}
Constants defined in the first two line
Then the instructions for the compilation targets
rm -f *~ *.o ${EXES}
to remove the object files and executables for a fresh buildAll of your assignments WILL have a makefile submitted with them otherwise they won’t be marked.
fork()
call#include <sys/types.h>
#include <stdio.h>
#include <unistd.h>
int main(void){
pid_t foo;
foo = fork();
printf("hey i'm %ld and my foo is %ld\n",
(long)getpid(),
(long)foo);
return 0;
}
foo = fork();
if(foo)
{ /* parent code goes here */ }
else { /* child code goes here */ };
if((foo = fork()) < 0){
/* fork error */
perror("error in fork");
exit(1);
} else if(foo){
/* parent code here */
} else {
/* child code here */
}
/* code here will be executed by both */
/* child and parent as long as they do */
/* not execute an call to exit or */
/* abort in their code above */
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
int main(void){
int i, n = 4;
pid_t childpid;
for (i = 1; i < n; ++i)
if((childpid = fork()) < 0){
/* fork error */
perror("error in fork");
exit(EXIT_FAILURE);
} else if(childpid){
/* parent code */
* break;
} else {
/* child code */
}
/* mutual code */
printf("This is process %ld with parent %ld\n",
(long)getpid(), (long)getppid());
return 0;
}
*Running this looks like:
turing.Examples> forkex02
This is process 25906 with parent 21711
This is process 25907 with parent 25906
This is process 25909 with parent 25908
turing.Examples> This is process 25908 with parent 25907
Note that the prompt returned before one of the spawned processes.
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
int main(void){
int i, n = 4;
pid_t childpid;
for (i = 1; i < n; ++i)
if((childpid = fork()) < 0){
/* fork error */
perror("error in fork");
exit(EXIT_FAILURE);
} else if(childpid){
/* parent code */
} else {
/* child code */
* break;
}
/* mutual code */
printf("This is process %ld with parent %ld\n",
(long)getpid(), (long)getppid());
return 0;
}
turing.Examples> forkex03
This is process 25977 with parent 25975
This is process 25976 with parent 25975
This is process 25975 with parent 21711
This is process 25978 with parent 25975
turing.Examples>
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
int main(void){
int i, n = 4;
pid_t childpid;
for (i = 1; i < n; i++)
if((childpid = fork()) < 0){
/* fork error */
perror("error in fork");
exit(EXIT_FAILURE);
} else if(childpid){
* /* parent code */
} else {
* /* child code */
}
/* mutual code */
printf("This is process %ld with parent %ld\n",
(long)getpid(), (long)getppid());
return 0;
}
turing.Examples> forkex04
This is process 26140 with parent 21711
This is process 26142 with parent 26140
This is process 26143 with parent 26141
This is process 26141 with parent 26140
turing.Examples> This is process 26146 with parent 1
This is process 26147 with parent 1
This is process 26145 with parent 1
This is process 26144 with parent 1
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
int main(void){
int i, n = 4;
pid_t childpid;
for (i = 1; i < n; ++i)
if((childpid = fork()) < 0){
/* fork error */
perror("error in fork");
exit(EXIT_FAILURE);
} else if(childpid){
/* parent code */
break;
} else {
/* child code */
}
/* mutual code */
printf("This is process %d with parent %d\n",
getpid(), getppid());
return 0;
}
turing.Examples> forkex05
This is process 27533 with parent 21711
This is process 27535 with parent 27534
This is process 27536 with parent 27535
This is process 27534 with parent 27533
turing.Examples>
They finish and exit in a random fashion.
int globvar = 6; /* external variable in initialized data */
char buf[] = "a write to stdout\n";
int main(void){
int var; /* automatic variable on the stack */
pid_t pid;
var = 88;
if (write(STDOUT_FILENO, buf, sizeof(buf)-1) != sizeof(buf)-1)
err_sys("write error");
printf("before fork\n"); /* we don't flush stdout */
if ((pid = fork()) < 0) {
err_sys("fork error");
} else if (pid == 0) { /* child */
globvar++; /* modify variables */
var++;
} else {
sleep(2); /* parent */
}
printf("pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar,var);
exit(0);
}
fork()
System Call
class: middle, center, inverse