Download presentation
Presentation is loading. Please wait.
1
Introduction to Unix System Programming
Lecturer: Prof. Andrzej (AJ) Bieszczad Phone: “UNIX for Programmers and Users” Third Edition, Prentice-Hall, GRAHAM GLASS, KING ABLES Slides partially adapted from Kumoh National University of Technology (Korea) and NYU
2
System and Library Calls
- This section contains information on the following system calls and library calls, listed in alphabetical order: accept alarm bind bzero chdir chmod chown close connect dup dup2 execl execlp getdents getegid geteuid getgid gethostname gethostbyname getpgid getpid getppid getuid htonl htons inet_addr nice ntohl ntohs open pause perror pipe read setegid seteuid setgid setpgid setuid Prof. Andrzej (AJ) Bieszczad Phone:
3
System and Library Calls
execv execvp exit fchmod fchown fcntl fork fstat ftruncate mknod signal socket stat sync truncate unlink wait write inet_ntoa ioctl kill lchown link listen lseek lstat memset mknod Prof. Andrzej (AJ) Bieszczad Phone:
4
Introduction process duplication, and interprocess communication,
- In order to make use of services such as file creation, process duplication, and interprocess communication, application programs must talk to the operating system. - a collection of routines called system calls, which are the programmer’s functional interface to the UNIX kernel. - System calls are just like library routines, except that they perform a subroutine call directly into the heart of UNIX. - The UNIX system calls can be loosely grouped into three main categories: file management process management error handling Prof. Andrzej (AJ) Bieszczad Phone:
5
File-management system calls hierarchy
Files open close read write lseek unlink chown dup2 special Directory chmod fcntl fstat ftruncate truncate stat sync dup link mknod ioctl pipe Sockets getdents Internet sockets accept bind connect listen socket gethostbyname gethostname htonl htons inet_addr inet_ntoa Prof. Andrzej (AJ) Bieszczad Phone:
6
Process management system call hierarchy
nice chdir wait fork exec exit Signals setgid setpgrp getpgrp getppid setuid getgid getrgid getuid getruid alarm signal kill pause Prof. Andrzej (AJ) Bieszczad Phone:
7
Error-handling hierarchy
perror Prof. Andrzej (AJ) Bieszczad Phone:
8
KERNEL BASICS - The UNIX kernel the part of the UNIX operating system that contains the code for: sharing the CPU and RAM between competing processes processing all system calls handling peripherals - The kernel is a program that is loaded from disk into RAM when the computer is first tuned on. - Kernel Subsystems Memory management Process management Interprocess communication(IPC) Input/output File management Prof. Andrzej (AJ) Bieszczad Phone:
9
KERNEL BASICS and it runs until the system is turned off or crashes.
- It always stays in RAM. and it runs until the system is turned off or crashes. - User programs make use of the kernel via the system call interface. Prof. Andrzej (AJ) Bieszczad Phone:
10
Covers the system calls
- Error Handling: perror() - Regular file management: how to create, open, close, read, and write regular files. - Process management: how to duplicate, differentiate, suspend, and terminate processes and briefly discusses multithreaded processes. - Signals: the signal facility could come under the heading of either process management or interprocess communication. - IPC: describes interprocess communication via pipes (both unnamed pipes and named pipes) and sockets( including information about Internet sockets ) Prof. Andrzej (AJ) Bieszczad Phone:
11
ERROR HANDLING: PERROR()
- all system calls return a value of -1 if an error occurs. the “open()” system call can fail for one of several different reasons. - “errno”, a global varible that holds the numerical code of the last system call error - “perror()”, a subroutine that describes system-call errors. - Every process contains a global variable called “errno”, that is originally set to zero when the process is created. - The file “/usr/include/sys/errno.h” contains a list of the predefined error codes. #define EPERM /* Not owner */ #define ENOENT /* No such file or directory */ #define ESRCH /* No such process */ #define EINSR /* Interrupted System call */ #define EIO /* I/O error */ Prof. Andrzej (AJ) Bieszczad Phone:
12
ERROR HANDLING: PERROR()
- A successful system call never affects the current value of “errno” and unsuccessful system call always overwrites the current value of “errno”. - To access “errno” from your program, include “errno.h”. Library Routine: void perror(char* str) “perror()” displays the string str, followed by a colon, followed by a description of the last system call error. If there is no error to report, it displays the string “Error 0”. Actually, “perror()” isn’t a system call - it’s a standard C library routine. Prof. Andrzej (AJ) Bieszczad Phone:
13
ERROR HANDLING: PERROR()
$ cat showErrno.c #include <stdio.h> #include <sys/file.h> #include <errno.h> main() { int fd; /* Open a nonexistent file to cause an error */ fd = open(“nonexist.txt”, O_RDONLY); if ( fd==-1 ) /* fd == -1 , an error occurred */ printf( “errno = %d \n”, errno ); perror(“main”); } fd=open( “/”, O_WRONLY ); /* Force a different error */ if ( fd== -1 ) { printf(“errno=%d\n”, errno); perror(“main”); } Prof. Andrzej (AJ) Bieszczad Phone:
14
ERROR HANDLING: PERROR()
/* Execute a successful system call */ fd = open(“nonexist.txt”, O_RDONLY | O_CREAE, 0644 ); printf(“errno=%d\n”, errno); /* Display after successful call */ perror(“main”); errno=0; /* Manually reset error variable */ } Here’s the output from this program: $ showErrno > run the program. errno=2 main: No such file or directory errno= > even after a successful call.0 main: Is a directory errno=21 main: Error 0 $ _ Prof. Andrzej (AJ) Bieszczad Phone:
15
REGULAR FILE MANAGEMENT
- Four main subsections of file-management system calls a primer that describes the main concepts behind UNIX files and file descriptors a description of the basic file-management system calls an explanation of a few advanced system calls a description of the remaining file-management system calls Prof. Andrzej (AJ) Bieszczad Phone:
16
A FILE MANAGEMENT Primer
- File-management system calls allow you to manipulate the full collection of regular, directory, and special files, including: disk-based files terminals printers interprocess communication facilities, such as pipes and sockets - open() is initially used to access or create a file. If system call succeeds, it returns a small integer called a file descriptor that is used in subsequent I/O operations on that file. Prof. Andrzej (AJ) Bieszczad Phone:
17
- A typical sequence of events: int fd; /* File descriptor */ …
fd = open( fileName, … ); /* Open file, return file descriptor */ if ( fd==-1 ) { /* Deal with error condition */ } fcntl( fd, … ); /* Set some I/O flags if necessary */ read( fd, … ); /* Read from file */ write( fd, …); /* Write to file */ lseek( fd, …); /* Seek within file */ close(fd); /* Close the file, freeing file descriptor */ Prof. Andrzej (AJ) Bieszczad Phone:
18
- When a process no longer needs to access an open file,
it should close it using the “close” system call. - All of a process’ open files are automatically closed when the process terminates. but, it’s better programming practice to explicitly close your files. - File descriptors are numbered sequentially, starting from zero. By convention, the first three file descriptor values have a special meaning: Value Meaning standard input standard output standard error Prof. Andrzej (AJ) Bieszczad Phone:
19
Many file descriptors, one file Fd3
- For Example, the “printf()” library function always sends its output using file descriptor 1. the “scanf()” always reads its input using file descriptor 0. Fd1 File Fd2 Many file descriptors, one file Fd3 Prof. Andrzej (AJ) Bieszczad Phone:
20
- A single file may be opened several times and, thus may have several
• File Pointer - A single file may be opened several times and, thus may have several file descriptors associated with it: Each file descriptor has its own private set of properties: A file pointer that records the offset in the file where it is reading and or writing. When a file descriptor is created, its file pointer is positioned at offset 0 in the file (the first character) by default. As the process reads and/or writes, the file pointer is updated accordingly. Prof. Andrzej (AJ) Bieszczad Phone:
21
• THE MOST BASIC I/O SYSTEM CALLS
Name Function open opens/creates a file read reads bytes from a file into a buffer write writes bytes from a buffer to a file lseek moves to a particular offset in a file close closes a file unlink removes a file Prof. Andrzej (AJ) Bieszczad Phone:
22
Utility : reverse -c [ fileName ]
reverse reverses the lines of its input read from fileName and displays them to standard output. If no filename is specified, reverse reverses its standard input. When the -c option is used, reverse also reverses the characters in each line. - Here’s an example of reverse in action: $ cc revese.c -o reverse > compile the program. $ cat test > list the test file. Christmas is coming. The days that grow shorter. Remind me of seasons I knew in the past. Prof. Andrzej (AJ) Bieszczad Phone:
23
$ reverse test ---> reverse the file.
Remind me of seasons I knew in the past. The days that grow shorter. Christmas is coming. $ reverse -c test > reverse the lines too. .tsap eht ni wenk I snosaes fo em dnimeR ,retrohs worg taht syad ehT ,gnimoc si samtshirhC $ cat test | reverse > pipe output to “reverse”. The days that grow shorter, Christmas is coming, $ _ Prof. Andrzej (AJ) Bieszczad Phone:
24
- An overview of the program flow
• How reverse Works - An overview of the program flow Step Action Functions System calls Parse command line parseCommandLine, open processOptions If reading from standard input, pass open create a temporary file to store input; otherwise, open the input file for reading. Read from the file in chunks, pass1, trackLines read,write storing the starting offset of each line in an array. If reading from standard input, copy each chunk to the temporary file. Read the input file again, but pass2, processLine lseek backward this time, copying each reverseLine line to standard output. Reverse the line if the “-c” option was chosen. Close the file, removing it if it was pass close a temporary file. Prof. Andrzej (AJ) Bieszczad Phone:
25
#include <fcntl.h> /* For file-mode definitions */
• “reverse.c” : Listing #include <fcntl.h> /* For file-mode definitions */ #include <stdio.h> #include <stdlib.h> /* Enumerator */ enum { FALSE, TRUE }; /* Standard false and true values */ enum { STDIN, STDOUT, STDERR }; /* Standard I/O-channel indices */ /* #define Statements */ #define BUFFER_SIZE /* Copy buffer size */ #define NAME_SIZE #define MAX_LINES /* Max lines in file */ /* Globals */ char *fileName = NULL; /* Points to file name */ char tmpName[NAME_SIZE]; int charOption = FALSE; /* Set to true if “-c” option is used */ int standardInput = FALSE; /* Set to true if reading stdin */ int lineCount=0; /* Total number of lines in input */ int lineStart[MAX_LINES]; /* Store offsets of each line */ Prof. Andrzej (AJ) Bieszczad Phone:
26
int fileOffset = 0; /* Current position in input */
int fd; /* File descriptor of input */ /**************************************************************/ main( argc, argv ) int argc; char* argv[]; { parseCommandLine(argc, argv); /* Parse command line */ pass1(); /* Perform first pass through input */ pass2(); /* Perform second pass through input */ return ( /* EXITSUCCESS */ 0 ); /* Done */ } parseCommandLine( argc, argv ) /* Parse command-line arguments */ int i; for ( i=1; i<argc; i++) Prof. Andrzej (AJ) Bieszczad Phone:
27
processOptions( argv[i] ); else if ( fileName == NULL )
if ( argv[i][0] == ‘-’ ) processOptions( argv[i] ); else if ( fileName == NULL ) fileName = argv[i]; else usageError(); /* An error occurred */ } standardInput = ( fileName == NULL ); /**************************************************************/ processOptions(str) char* str; /* Parse options */ { int j; for( j=1; str[j] !=NULL; j++) switch(str[j]) /* Switch on command-line flag */ case ‘c’ : charOption = TRUE; break; Prof. Andrzej (AJ) Bieszczad Phone:
28
/**********************************************************/
default: usageError(); break; } /**********************************************************/ usageError() { fprintf(stderr, “Usage: reverse -c [filename] \n”); exit( /* EXITFAILURE */ 1 ); /*********************************************************/ pass1() /* perform first scan through file */ int tmpfd, charsRead, charsWritten; char buffer[BUFFER_SIZE]; if ( standarInput ) /* Read from standard input */ fd = STDIN; sprintf( tmpName, “.rev.%d”, getpid()); /* Random name */ Prof. Andrzej (AJ) Bieszczad Phone:
29
/* Create temporary file to store copy of input */
tmpfd = open( tmpName, O_CREAT | O_RDWR, 0600 ); if ( tmpfd == -1 ) fatalError(); } else /* Open named file for reading */ { fd = open( fileName, O_RDONLY ); if ( fd==-1 ) fatalError(); lineStart[0] = 0; /* Offset of first line */ while( TRUE ) /* Read all input */ /* Fill buffer */ charsRead = read( fd, buffer, BUFFER_SIZE ); if ( charsRead == 0 ) break; /* EOF */ if ( charsRead == -1 ) fatalError(); /* Error */ trackLines( buffer, charsRead ); /* Process line */ /* Copy line to temporary file if reading from stdin */ if ( standardInput ) Prof. Andrzej (AJ) Bieszczad Phone:
30
charsWritten = write( tmpfd, buffer, charsRead );
if ( charsWritten != charsRead ) fatalError(); } /* Store offset of trailing line, if present */ lineStart[lineCount+1] = fileOffset; /* If reading from standard input, prepare fd for pass2 */ if ( standardInput ) fd = tmpfd; /*************************************************************/ trackLines( buffer, charsRead ) char* buffer; int charsRead; /* Store offsets of each line start in buffer */ { int i; for ( i=0; i<charsRead; i++) ++fileOffset; /* Update current file position */ if ( buffer[i] == ‘\n’ ) lineStart[++lineCount] = fileOffset; Prof. Andrzej (AJ) Bieszczad Phone:
31
/************************************************/ int pass2()
/* Scan input file again, displaying lines in reverse order */ { int i; for ( i=lineCount -1; i>= 0; i-- ) processLine(i); close(fd); /* Close input file */ if ( standardInput ) unlink( tmpName ); /* Remove temp file */ } /*************************************************/ processLine(i) /* Read a line and display it */ int charsRead; char buffer[BUFFER_SIZE]; lseek( fd, lineStart[i], SEEK_SET ); /* Find and read the line */ charsRead = read( fd, buffer, lineStart[i+1]-lineStart[i] ); /* Reverse line if “-c” optione was selected */ if ( charOption ) reverseLine( buffer, charsRead ); Prof. Andrzej (AJ) Bieszczad Phone:
32
write( 1, buffer, charsRead ); /* Write it to standard output */ }
/*********************************************************/ reverseLine( buffer, size ) char* buffer; int size; /* Reverse all the characters in the buffer */ { int start = 0, end = size -1; char tmp; if ( buffer[end] == ‘\n’ ) --end; /* Leave trailing new line */ /* Swap characters in a pairwise fashion */ while( start < end ) tmp = buffer[start]; buffer[start] = buffer[end]; buffer[end] = tmp; ++start; /* Increment start index */ --end; /* Decrement end index */ Prof. Andrzej (AJ) Bieszczad Phone:
33
/*********************************************************/
fatalError() { perror( “reverse:”) ; /* Describe error */ exit(1); } Prof. Andrzej (AJ) Bieszczad Phone:
34
• Opening a File: open()
System Call: int open( char* fileName, int mode[, int permissions]) “open()” allows you to open or create a file for reading and/or writing. fileName : an absolute or relative pathname, mode : a bitwise or’ing of a read/write flag together with zero or more miscellaneous flags. permission : a number that encodes the value of the file’s permission flags. - The permissions value is affected by the process’umask value that we studied earlier. Prof. Andrzej (AJ) Bieszczad Phone:
35
- The read/write flags are as follows: FLAG MEANING
O_RDONLY Open for read only. O_WRONLY Open for write only. O_RDWR Open for both read and write. - The miscellaneous flags are as follows: FLAG MEANING O_APPEND Position the file pointer at the end of the file before each “write()”. O_CREAT If the file doesn’t exist, create the file and set the owner ID to the process’ effective UID. O_EXCL If O_CREAT is set and the file exists, then “open()” fails. O_TRUNC If the file exists, it is truncated to length zero. Prof. Andrzej (AJ) Bieszczad Phone:
36
supply the initial file-permission flag settings as an octal value.
• Creating a File - To create a file, use the O_CREAT flags as part of the mode flags and supply the initial file-permission flag settings as an octal value. sprintf( tmpName, “.rev.%d”, getpid() ); /* Random name */ /* Create temporary file to store copy of input */ tmpfd = open( tmpName, O_CREAT | O_RDWR, 0600); if ( tmpfd== -1 ) fatalError(); - The “getpid()” function is a system call that returns the process’ID(PID), which is guaranteed to be unique. - Files that begin with a period are sometimes known as hidden files. it doesn’t show up in an ls listing. Prof. Andrzej (AJ) Bieszczad Phone:
37
- To open an existing file, specify the mode flags only.
fd = open( fileName, O_RDONLY ); if ( fd== -1 ) fatalError(); Prof. Andrzej (AJ) Bieszczad Phone:
38
- Reading From a File : read()
To read bytes from a file, it uses the “read()” system call, System Call: ssize_t read( int fd, void* buf, size_t count) “read()” copies count bytes from the file referenced by the file descriptor fd into the buffer buf. - if we read one character of input at a time, a large number of system calls, thus slowing down the execution of our program considerably. to read up to “BUFFER_SIZE” characters at a time. charsRead = read( fd, buffer, BUFFER_SIZE ); if ( charsRead == 0 ) break; /* EOF */ if ( charsRead == -1 ) fatalError(); /* Error */ Prof. Andrzej (AJ) Bieszczad Phone:
39
- Writing to a File: write()
To write bytes to a file, it uses the “write()” system call, which works as follows: System Call: ssize_t write( int fd, void* buf, size_t count) “write()” copies count bytes from a buffer buf to the file referenced by the file descriptor fd. If the O_APPEND flag was set for fd, the file position is set to the end of the file before each “write”. If successful, “write()” returns the number of bytes that were written; otherwise, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
40
- Perform the “write” operation:
/* Copy line to temporary file if reading standard input */ if ( standardInput ) { charsWritten = write(tmpfd, buffer, charsRead ); if ( charsWritten != charsRead ) fatalError(); } Prof. Andrzej (AJ) Bieszczad Phone:
41
- Moving in a File : lseek()
System Call: off_t lseek( int fd, off_t offset, int mode ) “lseek()” allows you to change a descriptor’s current file position. fd : the file descriptor, offset : a long integer, mode : how offset should be interpreted. - The three possible values of mode VALUE MEANING SEEK_SET offset is relative to the start of the file. SEEK_CUR offset is relative to the current file position. SEEK_END offset is relative to the end of the file. Prof. Andrzej (AJ) Bieszczad Phone:
42
- The numbers of characters to read
calculated by subtracting the offset value of the start of the next line from the offset value of the start of the current line: lseek( fd, lineStart[i], SEEK_SET ); /* Find line and read it */ charsRead = read ( fd, buffer, lineStart[i+1] - lineStart[i] ); - To find out your current location without moving, use an offset value of zero relative to the current position: currentOffset = lseek( fd, 0, SEEK_CUR ); Prof. Andrzej (AJ) Bieszczad Phone:
43
$ cat sparse.c ---> list the test file. #include <fcntl.h>
#include <stdio.h> #include <stdlib.h> /*************************************************/ main() { int i, fd; /* Creates a sparse file */ fd = open(“sparse.txt”, O_CREAT | O_RDWR, 0600 ); write( fd, “space”, 6); lseek( fd, 60006, SEEK_SET ); write( fd, “file”, 4 ); close(fd); /* Create a normal file */ fd = open(“normal.txt”, O_CREAT | O_RDWR, 0600 ); write( fd, “normal”, 6 ); for ( i=1; i<=60000; i++ ) write( fd, “/0”, 1 ); Prof. Andrzej (AJ) Bieszczad Phone:
44
$ sparse ---> execute the file.
write( fd, “file”, 4 ); close(fd); } $ sparse > execute the file. $ ls -l *.txt > look at the files. -rw-r--r glass Feb :06 normal.txt -rw-r--r glass Feb :06 sparse.txt $ ls -s *.txt > list their block usage. 60 normal.txt* > uses a full 60 blocks. 8 sparse.txt* > only uses eight blocks. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
45
Closing a File: “close()”
- uses the “close()” system call to free the file descriptor of the input. System Call : int close(int fd) “close()” frees the file descriptor fd. If fd is the last file descriptor associated with a particular open file, the kernel resources associated with the file are deallocated. If successful, “close()” returns a value of 0; otherwise, it returns a value of -1. close(fd); /* Close input file */ Prof. Andrzej (AJ) Bieszczad Phone:
46
Jump To Process Management
Deleting a File: unlink() System Call: int unlink( const char* fileName ) “unlink()” removes the hard link from the name fileName to its file. If fileName is the last link to the file, the file’s resources are deallocated. An executable file can “unlink” itself during execution and still continue to completion. If successful, “unlink()” returns a value of 0; otherwise, it returns a value of -1. If ( standardInput ) unlink( tmpName ); /* Remove temp file */ Jump To Process Management Prof. Andrzej (AJ) Bieszczad Phone:
47
Second Example: Monitor
- This section contains a description of some more advanced system calls. Name Function stat obtains status information about a file fstat works just like stat getdents obtains directory entries Prof. Andrzej (AJ) Bieszczad Phone:
48
Second Example: Monitor
Utility : monitor [-t delay] [ -l count ] {fileName }+ monitors scans all of the specified files every delay seconds and displays information about any of the specified files that were modified since the last scan. If fileName is a directory, all of the files inside that directory are scanned. LABEL MEANING ADDED Indicates that the file was created since the last scan. Every file in the file list is given this label during the first scan. CHANGED Indicates that the file was modified since the last scan. DELETED Indicates that the file was deleted since the last scan. Prof. Andrzej (AJ) Bieszczad Phone:
49
Second Example: Monitor
- monitored an individual file and a directory, storing the output of monitor into a temporary file. - the contents of the “monitor.out” file reflected the additions, modifications, and deletions of the monitored file and directory: % ls > look at home directory. monitor.c monitor tmp/ % ls tmp > look at “tmp” directory. b % monitor tmp myFile.txt >& monitor.out & ---> start monitoring. [1] 12841 % cat > tmp/a > create a file in “y/tmp”. hi there ^D % cat > myFile.txt > create “myFile.txt”. Prof. Andrzej (AJ) Bieszczad Phone:
50
Second Example: Monitor
% rm tmp/a > delete “tmp/a”. % jobs > look at jobs. [1] + Running monitor tmp myFile.txt >& monitor.out % kill % > kill “monitor” job. [1] Terminated monitor tmp myFile.txt >& monitor.out % cat monitor.out > look at output. ADDED tmp/b size 9 bytes, mod. time = Sun Jan 18 00:38: ADDED tmp/a size 9 bytes, mod. Time = Fri Feb :51: ADDED myFile.txt size 9 bytes, mod. Time = Fri Feb :51: CHANGED myFile.txt size 18 bytes, mod. Time = Fri Feb 13 18:51: DELETED tmp/a % - Prof. Andrzej (AJ) Bieszczad Phone:
51
How monitor Works - The monitor utility continually scans the specified files and directories for modifications. - uses the “stat()” system call to obtain status information about named files, including their types and last modification time, and the “getdents()” system call to scan directories. - maintains a status table called “stats”, that holds the following information about each file that it finds: the name of the file the status information obtained by the “stat()” system call a record of whether the file was present during the present scan and the previous scan. Prof. Andrzej (AJ) Bieszczad Phone:
52
How monitor Works - During a scan, monitor processes each file as follows: If the file isn’t currently in the scan table, it’s added and the message “ADDED” is displayed. If the file is already in the scan table and has been modified since the last scan, the message “CHANGED” is displayed. Prof. Andrzej (AJ) Bieszczad Phone:
53
monitor.c: Listing - a complete listing of “monitor.c”, the source code of monitor. #include <stdio.h> /* For printf, fprintf */ #include <string.h> /* For strcmp */ #include <ctype.h> /* For isdigit */ #include <fcntl.h> /* For O_RDONLY */ #include <sys/dirent.h> /* For getdents */ #include <sys/stat.h> /* For IS macros */ #include <sys/types.h> /* For modet */ #include <time.h> /* For localtime, asctime */ /* #define Statements */ #define MAX_FILES #define MAX_FILENAME #define NOT_FOUND #define FOREVER #define DEFAULT_DELAY_TIME 10 #define DEFAULT_LOOP_COUNT FOREVER Prof. Andrzej (AJ) Bieszczad Phone:
54
/* Status structure, one per file. */ struct statStruct {
/* Booleans */ enum { FALSE, TRUE }; /* Status structure, one per file. */ struct statStruct { char fileName[MAX_FILENAME]; /* File name */ int lastCycle, thisCycle; /* To detect changes */ struct stat status; /* Information from stat() */ }; /* Globals */ char* fileNames[MAX_FILES]; /* One per file on command line */ int fileCount; /* Count of files on command line */ struct statStruct stats[MAX_FILES]; /* One per matching file */ int loopCount = DEFAULT_LOOP_COUNT; /* Number of times to loop */ int delayTime = DEFAULT_DELAY_TIME; /* Seconds between loops */ Prof. Andrzej (AJ) Bieszczad Phone:
55
/****************************************************/
processOptions( str ) char* str; /* Parse options */ { int j; for (j=1; str[j] != NULL; j++) switch( str[j] ) /* Switch on option letter */ case ‘t’: delayTime = getNumber( str, &j ); break; case ‘l’: loopCount = getNumber( str, &j ); } Prof. Andrzej (AJ) Bieszczad Phone:
56
{ parseCommandLine( argc, argv ); /* Parse command line */
main( argc, argv ) int argc; char* argv[]; { parseCommandLine( argc, argv ); /* Parse command line */ monitorLoop(); /* Execute main monitor loop */ return ( /* EXIT_SUCCESS */ 0 ); } /****************************************************/ parseCommandLine( argc, argv ) /* Parse command-line arguments */ { int i; for ( i=1; ( ( i < argc ) && ( i < MAX_FILES) ); i++ ) { if ( argv[I][0] == ‘-’ ) processOptions( argv[I] ); else fileNames[fileCount++] = argv[I]; if ( fileCount == 0 ) usageError(); Prof. Andrzej (AJ) Bieszczad Phone:
57
/****************************************************/
getNumber( str, i ) char* str; int* i; /* Convert a numeric ASCII option to a number */ { int number = 0; int digits = 0; /* Count the digits in the number */ while ( isdigit( str[(*i)+1] ) ) /* Convert chars to ints */ number = number * 10 + str[++(*i)] - ‘0’; ++digits; } if ( digits == 0 ) usageError(); /* There must be a number */ return(number); Prof. Andrzej (AJ) Bieszczad Phone:
58
/****************************************************/ usageError( ) {
fprintf(stderr,“Usage: monitor -t<seconds> -l<loops> {filename}+\n”); exit( /* EXIT_FAILURE */ 1 ); } monitorLoop() /* The main monitor loop */ do monitorFiles(); /* Scan all files */ fflush(stdout); /* Flush standard output */ fflush(stderr); /* Flush standard error */ sleep(delayTime); /* Wait until next loop */ while( loopCount == FOREVER || --loopCount > 0 ); Prof. Andrzej (AJ) Bieszczad Phone:
59
/****************************************************/ monitorFiles()
/* Process all files */ { int i; for ( i=0; i<fileCount; i++) monitorFiles( fileNames[i] ); for ( i=0; i<MAX_FILES; i++ ) /* Update stat array */ if ( stats[i].lastCycle && !stat[i].thisCycle ) printf(“DELETED %s\n”, stats[i].fileName); stats[i].lastCycle = stats[i].thisCycle; stats[i].thisCycle = FALSE; } Prof. Andrzej (AJ) Bieszczad Phone:
60
/****************************************************/
monitorFile( fileName ) char* fileName; /* Process a single file/directory */ { struct stat statBuf; mode_t mode; int result; result = stat(fileName, &statBuf); /* Obtain file status */ if ( result == -1 ) /* Status was not available */ fprintf( stderr, “Cannot stat %s \n”, fileName ); return; } mode = statBuf.st_mode; /* Mode of file */ Prof. Andrzej (AJ) Bieszczad Phone:
61
if ( S_ISDIR( mode ) ) /* Directory */ processDirectory( fileName );
else if ( S_ISREG( mode ) || S_ISCHR(mode) || S_ISBLCK(mode) ) updateStat( fileName, &statBuf ); /* Regular file */ } /******************************************************/ processDirectory( dirName ) char* dirName; /* Process all files in the named directory */ { int fd, charsRead; struct dirent dirEntry; char fileName[MAX_FILENAME]; fd=open(dirName, O_RDONLY); /* Open for reading */ if ( fd == -1 ) fatalError(); while( TRUE ) /* Read all directory entries */ Prof. Andrzej (AJ) Bieszczad Phone:
62
charsRead = getdents(fd, &dirEntry, sizeof(struct dirent) );
if ( charsRead == -1 ) fatalError(); if ( charsRead == 0 ) break; /* EOF */ if ( strcmp(dirEntry.d_name, “.”) != 0 && strcmp( dirEntry.d_name, “..”) != 0 ) /* Skip . and .. */ { sprintf( fileName, “%s/%s”, dirName, dirEntry.d_name ) monitorFile( fileName ); /* Call recursively */ } lseek( fd, dirEntry.d_off, SEEK_SET ); /* Find next entry */ close(fd); /* Close directory */ /*******************************************************/ updateStat( fileName, statBuf ) char* fileName; struct stat* statBuf; /* Add a status entry if necessary */ int entryIndex; Prof. Andrzej (AJ) Bieszczad Phone:
63
entryIndex = findEntry( fileName ); /* Find existing entry */
if ( entryIndex == NOT_FOUND ) entryIndex = addEntry( fileName, statBuf ); /* Add new entry */ else updateEntry( entryIndex, statBuf ); /* Update existing entry */ if ( entryIndex != NOT_FOUND ) stats[entryIndex].thisCycle = TRUE; /* Update status array */ } /******************************************************/ findEntry( fileName ) char* fileName; /* Locate the index of a named file in the status array */ { int i; for(i=0; i<MAX_FILES; i++) if ( stats[i].lastCycle && strcmp( stats[i].fileName, fileName ) == 0 ) return(i); return ( NOT_FOUND ); Prof. Andrzej (AJ) Bieszczad Phone:
64
addEntry( fileName, statBuf ) char* fileName; struct stat* statBuf;
/* Add a new Entry into the status array */ { int index; index = nextFree(); /* Find the next free entry */ if ( index == NOT_FOUND ) return( NOT_FOUND ); /* None left */ strcpy( stats[index].fileName, fileName ); /* Add filename */ stats[index].status = *statBuf; /* Add status information */ printf(“ADDED”); /* Notify standard output */ printEntry(index); /* Display status information */ return ( index ); } Prof. Andrzej (AJ) Bieszczad Phone:
65
/****************************************************/ nextFree()
/* Return the next free index in the status array */ { int i; for (i=0; i<MAX_FILES; i++) if ( !stats[i].lastCycle && !stats[i].thisCycle ) return (i); return( NOT_FOUND ); } /*****************************************************/ updateEntry( index, statBuf ) int index; struct stat* statBuf; /* Display information if the file has been modified */ if ( stats[index].status.st_mtime != statBuf->st_mtime ) stats[index].status = *statBuf; /* Store stat information */ printf(“CHANGED”); /* Notify standard output */ Prof. Andrzej (AJ) Bieszczad Phone:
66
/*************************************************/
printEntry( index ); } /*************************************************/ printEntry( index ) int index; /* Display an entry of the status array */ { printf(“%s “, stats[index].fileName ); printStat( &stats[index].status ); printStat( statBuf ) struct stat* statBuf; /* Display a status buffer */ printf( “size %lu bytes, mod. time = %s”, statBuf-> st_size, ascTime( localtime(&statBuf->st_mtime) ) ); Prof. Andrzej (AJ) Bieszczad Phone:
67
/*************************************************/ fatalError() {
perror(“monitor: “); exit( /* EXIT_FAILURE */ 1 ); } Prof. Andrzej (AJ) Bieszczad Phone:
68
Obtaining File Information: stat(), lstat(), and fstat()
- monitor obtains its file information by calling “stat()”, which works as follows: System Call : int stat( const char* name, struct stat* buf ) int lstat( const char* name, struct stat* buf ) int fstat( int fd, struct stat* buf ) “stat()” fills the buffer buf with information about the file name. The “stat” structure is defined in “/usr/include/sys/stat.h”. “lstat()” returns information about a symbolic link itself rather than the file that it references. “fstat()” performs the same function as “stat()”, except that it takes the file descriptor of the file to be “stat” end as its first parameter. Prof. Andrzej (AJ) Bieszczad Phone:
69
Obtaining File Information: stat(), lstat(), and fstat()
- The “stat” structure contains the following members: NAME MEANING st_dev the device number st_ino the inode number st_mode the permission flags st_nlink the hard-link count st_uid the user ID st_gid the group ID st_size the file size st_atime the last access time st_mtime the last modification time st_ctime the last status-change time Prof. Andrzej (AJ) Bieszczad Phone:
70
Obtaining File Information: stat(), lstat(), and fstat()
- some predefined macros defined in “/usr/include/sys/stat.h” “st_mode” as their argument and return true( a value of 1 ) for the following file types: MACRO RETURNS TRUE FOR FILE TYPE S_IFDIR directory S_IFCHR character special device S_IFBLK block special device S_IFREG regular file S_IFFIFO pipe - The time fields decoded using the standard C library “asctime()” and “localtime()” subroutines. Prof. Andrzej (AJ) Bieszczad Phone:
71
The monitor utility - invokes “stat()” from “monitorFile()”
result = stat( fileName, &statBuf ); /* Obtain file status */ - examines the mode of the file using the S_ISDIR, S_ISREG, S_ISCHR, and S_ISBLK macros, processing directory files and other files as follows: 1) If the file is a directory file, calls “processDirectory()”, which applies “monitorFile() to each of its directory entries. 2) If the file is a regular file, character special file, or block special file, calls “updateStat()”, which either adds or updates the file’s status entry. Prof. Andrzej (AJ) Bieszczad Phone:
72
Reading Directory Information: getdents()
- “processDirectory()” opens a directory file for reading and then uses “getdents()” to obtain every entry in the directory. System call: int getdents( int fd, struct dirent* buf, int structSize ) reads the directory file with descriptor fd from its current position and fills the strcuture pointed to be by buf with the next entry. - The structure “dirent” is defined in “/usr/include/sys/dirent.h” and contains the following fields: NAME MEANING d_ino the inode number d_off the offset of the next directory entry d_reclen the length of the directory entry structure d_nam the length of the filename Prof. Andrzej (AJ) Bieszczad Phone:
73
Inodes - UNIX uses a structure called an inode,
which stands for “index node”, to store information about each file. - contains the locations of its disk blocks. - holds other information associated with a file, such as its permission flags, owner, group, and last modification time. - a structure of fixed size containing pointers to disk blocks and additional indirect pointers( for large files ). - allocated a unique inode number, every file has exactly one inode. - stored in a special area at the start of the disk called the inode list. Prof. Andrzej (AJ) Bieszczad Phone:
74
Inode Contents - a list of the file information contained within each inode: the type of the file : regular, directory, block special, character special, etc. file permissions the owner and group Ids a hard-link count the last modification and access times if it’s a regular or directory file, the location of the blocks if it’s a special file, the major and minor device numbers if it’s a symbolic link, the value of the symbolic link Prof. Andrzej (AJ) Bieszczad Phone:
75
Miscellaneous File Management System Calls
- a brief description of the following miscellaneous file management system calls: NAME Function chown changes a file’s owner and/or group chmod changes a file’s permission settings dup duplicates a file descriptor dup similar to dup fchown works just like chown fchmod works just like chmod fcntl gives access to miscellaneous file characteristics ftruncate works just like truncate ioctl controls a device link creates a hard link mknod creates a special file sync schedulers all file buffers to be flushed to disk truncate truncates a file Prof. Andrzej (AJ) Bieszczad Phone:
76
Changing a File’s Owner and/or Group: chown(), ichown() and fchown()
- “chown()” and “fchown()” change the owner and/or group of a file and work like this: System Call: int chown( const char* fileName, uid_t ownerId, gid_t groupId ) int lchown( const char* fileName, uid_t ownerId, git_t groupId ) int fchown( int fd, uid_t ownerId, gid_t groupId ) - “chown()” causes the owner and group IDs of fileName to be changed to ownerId and groupId, respectively. A value of -1 in a particular field means that its associated value should remain unchanged. lchown():changes the ownership of a symbolic link fchown():takes an open descriptor as an argument instead of a filename. Prof. Andrzej (AJ) Bieszczad Phone:
77
Changing a File’s Owner and/or Group: chown(), ichown() and fchown()
- Example, changed the group of the file “test.txt” from “music” to “cs”. which has a group ID number of 62. $ cat mychown.c > list the program. main() { int flag; flag = chown(“test.txt”, -1, 62 ); /* Leave user ID uchanged */ if ( flag == -1 ) perror(“mychown.c”); } $ ls -lg test.txt > examine file before the change. -rw-r--r glass music May 25 11:42 test.txt $ mychown > run program. $ ls -lg test.txt > examine file after the change. -rw-r--r glass cs May :42 test.txt $ - Prof. Andrzej (AJ) Bieszczad Phone:
78
Changing a File’s Permissions: chmod() and fchmod()
- “chmod()” and “fchmod()” change a file’s permission flags System Call : int chmod( const char* fileName, int mode ) int fchmod( int fd, mode_t mode ); “chmod()” changes the mode of fileName to mode, where mode is usually supplied as an octal number, To change a file’s mode, you must either own it or be a super-user. “fchmod()” works just like “chmod()” except that it takes an open file descriptor as an argument instead of a filename. They both return a value of -1 if unsuccessful, and a value of 0 otherwise. Prof. Andrzej (AJ) Bieszczad Phone:
79
Changing a File’s Permissions: chmod() and fchmod()
- changed the permission flags of the file “test.txt” to 600 octal, which corresponds to read and write permission for the owner only: $ cat mychmod.c ---> list the program. main() { int flag; flag = chmod(“test.txt”, 0600); /* Use an octal encoding */ if ( flag==-1 ) perror(“mychmod.c”); } $ ls -l test.txt > examine file before the change. -rw-r--r glass May :42 test.txt $ mychmod > run the program. $ ls -l test.txt > examine file after the change. -rw glass May :42 test.txt $ - Prof. Andrzej (AJ) Bieszczad Phone:
80
Duplicating a File Descriptor: dup() and dup2()
- “dup()”, “dup2()” to duplicate file descriptors, and they work like this: System call: int dup( int oldFd ) int dup2( int oldFd, int newFd ) “dup()” finds the smallest free file-descriptor entry and points it to the same file to which oldFd points. “dup2()” closes newFd if it’s currently active and then points it - the original and copied file descriptors share the same file pointer and access mode. - return the index of the new file descriptor if successful and a value of -1 otherwise. Prof. Andrzej (AJ) Bieszczad Phone:
81
Duplicating a File Descriptor: dup() and dup2()
- I created a file called “test.txt” and wrote to it via four different file descriptors: The first file descriptor was the original descriptor. The second descriptor was a copy of the first, allocated in slot 4. The third descriptor was a copy of the first, allocated in slot 0 ( the standard input channel ), which was freed by the “close(0)” statement. The fourth descriptor was a copy of the third descriptor, copied over the existing descriptor in slot 2 ( the standard error channel ). Prof. Andrzej (AJ) Bieszczad Phone:
82
Duplicating a File Descriptor: dup() and dup2()
$ cat mydup.c > list the file. #include <stdio.h> #include <fcntl.h> main() { int fd1, fd2, fd3; fd1 = open( “test.txt”, O_RDWR | O_TRUNC ); printf(“fd1 = %d”\n”, fd1 ); write( fd1, “what’s “, 6 ); fd2 = dup(fd1); /* Make a copy of fd1 */ printf( “fd2=%d\n”, fd2); write( fd2, “up”, 3 ); close(0); /* Close standard input */ fd3 = dup(fd1); /* Make another copy of fd1 */ printf(“fd3 = %d\n”, fd3); write(0, “ doc“, 4); dup2(3,2); /* Duplicate channel 3 to channel 2 */ write(2, “?\n”, 2 ); } Prof. Andrzej (AJ) Bieszczad Phone:
83
Duplicating a File Descriptor: dup() and dup2()
$ mydup > run the program. fd1 = 3 fd2 = 4 fd3 = 0 $ cat test.txt > list the output file. what’s up doc? $ - Prof. Andrzej (AJ) Bieszczad Phone:
84
File Descriptor Operations: fcntl()
- “fcntl()” directly controls the settings of the flags associated with a file descriptor, and it works as follows: System Call : int fcntl( int fd, int cmd, int arg ) “fcntl()” performs the operation encoded by cmd on the file associated with the file descriptor fd. arg is an optional argument for cmd. Prof. Andrzej (AJ) Bieszczad Phone:
85
File Descriptor Operations: fcntl()
- Here are the most common values of cmd: VALUE OPERATION F_SETFD set the close-on-exec flag to the lowest bit of arg ( 0 or 1 ) F_GETFD return a number whose lowest bit is 1 if the close-on-exec flag is set and 0 otherwise. F_GETFL return a number corresponding to the current file status flags and access modes. F_SETFL set the current file-status flags to arg. F_GETOWN return the process ID or process group that is currently set to receive SIGIO/SIGURG signals. F_SETOWN set the process ID or process group that should receive SIGIO/SIGURG signals to arg. Prof. Andrzej (AJ) Bieszczad Phone:
86
File Descriptor Operations: fcntl()
$ cat myfcntl.c > list the program. #include <stdio.h> #include <fcntl.h> main() { int fd; fd = open(“test.txt”, O_WRONLY ); /* Open file for writing */ write( fd, “hi there\n”, 9 ); lseek( fd, 0, SEEK_SET ); /* Seek to beginning of file */ fcntl(fd, F_SETFL, O_WRONLY | O_APPEND ); /* Set APPEND flag */ write( fd, “ guys\n”, 6 ); close( fd ); } $ cat test.txt > list the original file. here are the contents of the original file. $ myfcntl > run the program. Prof. Andrzej (AJ) Bieszczad Phone:
87
File Descriptor Operations: fcntl()
$ cat test.txt > list the new contents. hi there the contents of the original file. guys > note that “guys” is at the end. $ - Prof. Andrzej (AJ) Bieszczad Phone:
88
Controlling Devices: ioctl()
- “ioctl()” performs many controlling functions on an input/output channel and works System Call: int ioctl( int fd, int cmd, int arg ) “ioctl()” performs the operation encoded by cmd on the file associated with the file descriptor fd. arg is an optional argument for cmd. - “ioctl()” returns a value of -1 if unsuccessful. Prof. Andrzej (AJ) Bieszczad Phone:
89
Creating Hard Links: Link
- “link()” creates a hard link to an exsiting file and works as follows: System Call : int link( const char* oldPath, const char* newPath ) “link()” creates a new label newPath and links it to the same file to which the label oldPath is linked. The hard-link count of the associated file is incremented by one. If oldPath and newPath reside on different physical devices, a hard link cannot be made and “link()” fails. “link()” returns a value of -1 if unsuccessful and a value of 0 otherwise. Prof. Andrzej (AJ) Bieszczad Phone:
90
$ cat mylink.c ---> list the program. main() {
link(“orginal.txt”, “another.txt”); } $ cat original.txt ---> list original file. this is a file. $ ls -l original.txt another.txt > examine the files before. another.txt not found -rw-r--r glass May 25 12:18 original.txt $ mylink > run the program. $ ls -l original.txt another.txt ---> examine files after. -rw-r--r glass May 25 12:18 another.txt -rw-r--r glass May 25 12:18 original.txt $ cat >> another.txt > alter “another.txt”. hi ^D Prof. Andrzej (AJ) Bieszczad Phone:
91
-rw-r--r-- 2 glass 20 May 25 12:19 another.txt
$ ls -l original.txt another.txt ---> both labels reflect the change. -rw-r--r glass May :19 another.txt -rw-r--r glass May :19 original.txt $ rm original.txt > remove original label. $ ls -l original.txt another.txt ---> examine labels. original.txt not found -rw-r--r glass May :19 another.txt $ cat another.txt > list contents via other label. this is a file. hi $ - Prof. Andrzej (AJ) Bieszczad Phone:
92
Creating Special Files: mknod()
- “mknod()” allows you to create a special file, and it works like this: System Call: int mknod( const char* fileName, mode_t type, dev_t device ) “mknod()” creates a new regular, directory, or special file called fileName whose type can be one of the following: VALUE MEANING S_IFDIR directory S_IFCHR character-oriented file S_IFBLK block-oriented file S_IFREG regular file S_IFIFO named pipe Prof. Andrzej (AJ) Bieszczad Phone:
93
Creating Special Files: mknod()
- If the file is a character-or block-oriented file, the low-order byte of device should specify the minor device number, the high-order byte should specify the major device number. In other cases, the value of device is ignored. - Only a super-user can use “mknod()” to create directories, character-oriented files, or block-oriented special files. - It is typical now to use the “mkdir()” system call to create directories. “mknod()” returns a value of -1 if unsuccessful and a value of 0 otherwise. Prof. Andrzej (AJ) Bieszczad Phone:
94
Flushing the File-System Buffer: sync()
- “sync()” flushes the file-system buffers and works as follow: System Call : void sync() “sync()” schedules all of the file system buffers to be written to disk. “sync()” should be performed by any programs that bypass the file system buffers and examine the raw file system. “sync()” always succeeds. Prof. Andrzej (AJ) Bieszczad Phone:
95
Truncating a File: truncate() and ftruncate()
- “truncate()” and “ftruncate()” set the length of a file, System Call: int truncate( const char* fileName, off_t length ) int ftruncate( int fd, off_t length ) “truncate()” sets the length of the file fileName to be length bytes, If the file is longer than length, it is truncated. If it is shorter than length, it is padded with ASCII NULLS. “ftruncate()” works just like “truncate()” does, except that it takes an open file descriptor as an argument instead of a filename. They both return a value of -1 if unsuccessful and a value of 0 otherwise. Prof. Andrzej (AJ) Bieszczad Phone:
96
Truncating a File: truncate() and ftruncate()
$ cat truncate.c > list the program. main() { truncate(“file1.txt”, 10); truncate(“file2.txt”, 10); } $ cat file1.txt > list “file1.txt”. short $ cat file2.txt > list “file2.txt”. long file with lots of letters $ ls -l file*.txt > examine both files. -rw-r--r glass May :16 file1.txt -rw-r--r glass May :17 file2.txt $ truncate > run the program. $ ls -l file*.txt > examine both files again. -rw-r--r glass May :16 file1.txt -rw-r--r glass May :17 file2.txt Prof. Andrzej (AJ) Bieszczad Phone:
97
Truncating a File: truncate() and ftruncate()
$ cat file1.txt > “file1.txt” is longer. short $ cat file2.txt > “file2.txt” is shorter. long file $ - Prof. Andrzej (AJ) Bieszczad Phone:
98
STREAMS - involves sending and receiving streams messages rather than just doing raw, character-by-character I/O. - added flow control and priority processing. - Anatomy of a STREAM Each STREAM has three parts: stream head : the access point for a user application, functions, and data structures representing the STREAM modules : code to process data being read or written stream driver : the back-end code that communicates with the specific device. Prof. Andrzej (AJ) Bieszczad Phone:
99
STREAMS - The stream head
provides the system-call interface for a user application. is created by using the “open()” system call. - The kernel manages any memory allocation required, the upstream and downstream flow of data, queue scheduling, flow control, and error logging. - STREAM System Calls the following additional system calls are useful with a stream: “getmsg()”: get a message from a stream “putmsg()”: put a message on a stream “poll()”: poll one or more streams for activity “isastream()”: find out if a given file descriptor is a stream Prof. Andrzej (AJ) Bieszczad Phone:
100
PROCESS MANAGEMENT - Every process in a UNIX system has the following attributes: some code( a.k.a. text ) some data a stack a unique process ID number(PID) - When UNIX is first started, there’s only one visible process in the system. This process is called “init”, and it has a process ID of 1. - The only way to create a new process in UNIX is to duplicate an existing process, so “init” is the ancestor of all subsequent processes. Prof. Andrzej (AJ) Bieszczad Phone:
101
PROCESS STATES - Every process in the system can be in one of six states. The six possible states are as follows: 1) Running, which means that the process is currently using the CPU. 2) Runnable, which means that the process can make use of the CPU as soon as it becomes available. 3) Sleeping, which means that the process is waiting for an event to occur. For example, if a process executes a “read()” system call, it sleeps until the I/O request completes. 4) Suspended, which means that the process has been “frozen” by a signal such as SIGSTOP. It will resume only when sent a SIGCONT signal. For example, a Control-Z from the keyboard suspends all of the processes in the foreground job. 5) Idle, which means that the process is being created by a “fork() system call and is not yet runnable. Prof. Andrzej (AJ) Bieszczad Phone:
102
6) Zombified, which means that the process has terminated but has
not yet returned its exit code to its parent. A process remains a zombie until its parent accepts its return code using the “wait()” system call. Suspended Signal Signal Initialize Allocated Exit CPU Idle Runnable Running Zombified Event Waits on occurs event Sleeping [ Process States ] Prof. Andrzej (AJ) Bieszczad Phone:
103
PROCESS COMPOSITION - Every process is composed of several different pieces: a code area, which contains the executable(text) portion of a process a data area, which is used by a process to contain static data a stack area, which is used by a process to store temporary data a user area, which holds housekeeping information about page tables, which are used by the memory management system Prof. Andrzej (AJ) Bieszczad Phone:
104
User Area - Every process in the system has some associated “housekeeping” information that is used by the kernel for process management. This information is stored in a data structure called a user area. Every process has its own user area. User areas are created in the kernel’s data region and are only accessible by the kernel; user processes may not access their user areas. Fields within a process’ user area include: a record of how the process should react to each kind of signal a record of the process’ open file descriptors a record of how much CPU time the process has used recently Prof. Andrzej (AJ) Bieszczad Phone:
105
The Process Table - There is a single kernel data structure of fixed size called the process table that contains one entry for every process in the system. The process table is created in the kernel’s data region and is accessible only by the kernel. Each entry contains the following information about each process: its process ID(PID) and parent process ID(PPID) its real and effective user ID(UID) and group ID(GID) its state ( running, runnable, sleeping, suspended, idle, or zombified ) the location of its code, data, stack, and user areas a list of all pending signals Prof. Andrzej (AJ) Bieszczad Phone:
106
Shared by processes The Process Table Process table Process 34
PID PPID Stat R Process 48 CPU R Process 12 S R Process 1 R : Running, S : Sleeping Shared by processes Prof. Andrzej (AJ) Bieszczad Phone:
107
PROCESS MANAGEMENT - When a process duplicates,
the parent and child processes are virtually identical ( except for aspects like PIDs, PPIDs, and runtimes); the child’s code, data, and stack are a copy of the parent’s, and the processes even continue to execute the same code. - A child process may replace its code with that of another executable file, there by differentiating itself from its parent. - When “init” starts executing, it quickly duplicates several times. Each of the duplicate child processes then replaces its code from the executable file called “getty”, which is responsible for handling user logins. Prof. Andrzej (AJ) Bieszczad Phone:
108
Process hierarchy PROCESS MANAGEMENT - The process hierarchy Parent
init(PID 1) Duplicate: fork(), then differentiate: exec() Child Child Child getty(PID 4) getty(PID5) getty(PID 6) handle a login handle a login handle a login Process hierarchy Prof. Andrzej (AJ) Bieszczad Phone:
109
PROCESS MANAGEMENT - When a child process terminates,
its death is communicated to its parent so that the parent may take some appropriate action. - When a shell executes a utility in the foreground, it duplicates into two shell processes; - the child-shell process replaces its code with that of the utility, whereas the parent shell waits for the child process to terminate. - When the child process terminates, the original parent process awakens and presents the user with the next shell prompt. Prof. Andrzej (AJ) Bieszczad Phone:
110
How a shell runs a utility
PROCESS MANAGEMENT Parent process PID 34 running shell Duplicate: fork() Parent process PID Child process PID 35 running shell, running shell Differentiate: exec() waiting for child Wait for child:wait() Child process PID 35 running utility Terminate: exit() Parent process PID Child process PID 35 running shell, terminates awakens Signal How a shell runs a utility Prof. Andrzej (AJ) Bieszczad Phone:
111
PROCESS MANAGEMENT Name Function fork duplicates a process
getpid obtains a process’ ID number getppid obtains a parent process’ID number exit terminates a process wait waits for a child process exec… replaces the code, data, and stack of a process. Prof. Andrzej (AJ) Bieszczad Phone:
112
Creating a New Process: fork()
- A process may duplicate itself by using “fork()”, which works like this: System Call: pid_t fork(void) “fork()” causes a process to duplicate. The child process is an almost-exact duplicate of the original parent process; it inherits a copy of its parent’s code, data, stack, open file descriptors, and signal table. the parent and child processes have different process ID numbers and parent process ID numbers. If “fork()” succeeds, it returns the PID of the child to the parent process and returns a value of 0 to the child process. Prof. Andrzej (AJ) Bieszczad Phone:
113
PROCESS MANAGEMENT - A process may obtain its own process ID and parent process ID numbers by using the “getpid()” and “getppid()” system calls, respectively. - Here’s a synopsis of these system calls: System Call : pid_t getpid(void) pid_t getppid(void) “getpid()” and “getppid()” return a process’ID number and parent process’ ID number, respectively. The parent process ID number of PID 1 (i.e., “init”) is 1. Prof. Andrzej (AJ) Bieszczad Phone:
114
PROCESS MANAGEMENT $ cat myfork.c ---> list the program.
#include <stdio.h> main() { int pid; printf(“I’m the original process with PID %d and PPID %d. \n”, getpid(), getppid() ); pid = fork(); /* Duplicate. Child and parent continue from here */ if ( pid!= 0 ) /* pid is non-zero, so I must be the parent */ { printf(“I’m the parent process with PID %d and PPID %d. \n”, printf(“My child’s PID is %d \n”, pid ); } else /* pid is zero, so I must be the child */ printf(“I’m the child process with PID %d and PPID %d. \n”, Prof. Andrzej (AJ) Bieszczad Phone:
115
PROCESS MANAGEMENT printf(“PID %d terminates. \n”, getpid() ); /* Both processes */ /* execute this */ } $ myfork > run the program. I’m the original process with PID and PPID I’m the parent process with PID and PPID My child’s PID is I’m the child process with PID and PPID PID terminates. ---> child terminates. PID terminates. ---> parent terminates. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
116
PROCESS MANAGEMENT - The PPID of the parent process refers to the PID of the shell that executed the “myfork” program. WARNING: it is dangerous for a parent process to terminate without waiting for the death of its child. The only reason our program doesn’t wait for its child to terminate is because we haven’t yet described the “wait()” system call!. Prof. Andrzej (AJ) Bieszczad Phone:
117
User process user functions exit handler main exit function
PROCESS MANAGEMENT User process _exit user functions exit exit handler return call call return _exit main function exit exit function exit handler return call exit C startup routine standard I/O cleanup _exit exec kernel Prof. Andrzej (AJ) Bieszczad Phone:
118
PROCESS MANAGEMENT - init process ID 1 init forks once
fork per terminal each child exec execs getty getty Prof. Andrzej (AJ) Bieszczad Phone:
119
PROCESS MANAGEMENT - login process ID 1 init forks once
fork per terminal each child exec execs getty getty opens terminal device ( file descriptors 0,1,2); exec reads username; initial environment set login Prof. Andrzej (AJ) Bieszczad Phone:
120
PROCESS MANAGEMENT process ID 1 init through getty and login fork
login shell fd 0,1,2 terminal device driver RS-232 connection user at a terminal Prof. Andrzej (AJ) Bieszczad Phone:
121
PROCESS MANAGEMENT - TELNET daemon process ID 1
init fork/exec of /bin/sh which executes shell script /etc/rc when system comes up multiuser TCP connection request from TELNET client inetd fork when connection request arrives from TELNET client inetd exec telnetd Prof. Andrzej (AJ) Bieszczad Phone:
122
PROCESS MANAGEMENT - Network process ID 1
init through inetd, telnetd and login TCP connection request from TELNET client login shell fd 0,1,2 pseudo-terminal device driver network connection through telnetd server and telnet client User at a terminal Prof. Andrzej (AJ) Bieszczad Phone:
123
- If a parent dies before its child terminates,
Orphan Processes - If a parent dies before its child terminates, the child is automatically adopted by the original “init” process, PID 1. - the parent process terminated before the child did. $ cat orphan.c > list the program. #include <stdio.h> main() { int pid; printf(“I’m the original process with PID %d and PPID %d. \n”, getpid(), getppid() ); pid = fork(); /* Duplicate. Child and parent continue from here */ if ( pid!= 0 ) /* Branch based on return value from fork() */ /* pid is non-zero, so I must be the parent */ printf(“I’m the parent process with PID %d and PPID %d. \n”, Prof. Andrzej (AJ) Bieszczad Phone:
124
Orphan Processes printf(“My child’s PID is %d \n”, pid ); } else {
/* pid is zero, so I must be the child */ sleep(5); /* Make sure that the parent terminates first */ printf(“I’m the child process with PID %d and PPID %d. \n”, getpid(), getppid() ); printf(“PID %d terminates. \n”, getpid() ); /* Both processes */ /* execute this */ $ orphan > run the program. I’m the original process with PID and PPID I’m the parent process with PID and PPID PID terminates. I’m the child process with PID and PPID > orphaned! PID terminates. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
125
Adoption Orphan Processes - illustration of the orphaning effect:
init Parent dies Adopt first child Child survives the parent Adoption Prof. Andrzej (AJ) Bieszczad Phone:
126
Terminating a Process: exit()
- A process may terminate at any time by executing “exit()”, which works as follows: System Call: void exit( int status ) “exit()” closes all of a process’ file descriptors; deallocates its code, data, and stack; and then terminates the process. When a child process terminates, it sends its parent a SIGCHLD signal and waits for its termination code status to be accepted. A process that is waiting for its parent to accept its return code is called a zombie process. A parent accepts a child’s termination code by executing “wait()”. Prof. Andrzej (AJ) Bieszczad Phone:
127
- The kernel ensures that all of a terminating process’ children
Orphan Processes - The kernel ensures that all of a terminating process’ children are orphaned and adopted by “init” by setting their PPID to 1. The “init” process always accept its childrens’ termination codes. “exit()” never returns. - The termination code of a child process may be used for a variety of purposes by the parent process. Shells may access the termination code of their last child process via one of their special variables. bash stores the termination code of the last command in the variable $?: Prof. Andrzej (AJ) Bieszczad Phone:
128
$ cat myexit.c ---> list the program. #include <stdio.h>
Orphan Processes $ cat myexit.c ---> list the program. #include <stdio.h> main() { printf(“I’m going to exit with return code 42 \n”); exit(42); } $ myexit ---> run the program. I’m going to exit with return code 42 $ echo $? ---> display the termination code. 42 $ _ Prof. Andrzej (AJ) Bieszczad Phone:
129
- A process that terminates cannot leave the system
Zombie Processes - A process that terminates cannot leave the system until its parent accepts its return code. - If its parent process is already dead, it’ll already have been adopted by the “init” process, which always accepts its childrens’ return codes. - If a process’ parent is alive, but the parent never executes a “wait()” system call, the process’ return code will never be accepted and the process will remain a zombie. - A zombie process doesn’t have any code, data, or stack, so it doesn’t use up many system resources, Prof. Andrzej (AJ) Bieszczad Phone:
130
- it does continue to inhabit the system’s fixed-size process table.
Zombie Processes - it does continue to inhabit the system’s fixed-size process table. - The next program created a zombie process, which is shown in the output from the ps utility. $ cat zombie.c > list the program. #include <stdio.h> main() { int pid; pid = fork(); /* Duplicate */ if ( pid!= 0 ) /* Branch based on return value from fork() */ while (1) /* Never terminate, and never execute a wait() */ sleep(1000); } else exit(42); /* Exit with a silly number */ Prof. Andrzej (AJ) Bieszczad Phone:
131
$ zombie & ---> execute the program in the background. [1] 13545
Zombie Processes $ zombie & > execute the program in the background. [1] 13545 $ ps > obtain process status. PID TT STAT TIME COMMAND p S : ksh (ksh) > the shell. p S : zombie > the parent process. p Z : <defunct> ---> the zombile child. p R : ps $ kill > kill the parent process. [1] Terminated zombie $ ps > notice that the zombie is gone now. PID TT STAT TIME COMMAND p S : ksh( ksh ) p R : ps $ _ Prof. Andrzej (AJ) Bieszczad Phone:
132
Waiting for a child(): wait()
- A parent process may wait for one of its children to terminate and then accept its child’s termination code by executing “wait()”. System Call : pid_t wait( int* status ) “wait()” causes a process to suspend until one of its children terminates. A successful call to “wait()” returns the PID of the child that terminated and places a status code into status that is encoded as follows: If the rightmost byte of status is zero, the leftmost byte contains the low eight bits of the value returned by the child’s “exit()” or “return()” system call. Prof. Andrzej (AJ) Bieszczad Phone:
133
Waiting for a child(): wait()
If the rightmost byte is nonzero, the rightmost seven bits are equal to the number of the signal that caused the child to terminate, and the remaining bit of the rightmost byte is set to 1 if the child produced a core dump. - If a process executes a “wait()” system call and has no children, “wait()” returns immediately with a value of -1. - If a process executes a “wait()” system call and one or more of its children are already zombies, “wait()” returns immediately with the status of one of the zombies. Prof. Andrzej (AJ) Bieszczad Phone:
134
Waiting for a child(): wait()
$ cat mywait.c > list the program. #include <stdio.h> main() { int pid, status, childPid; printf(“I’m the parent process and my PID is %d\n”, getpid() ); pid = fork(); /* Duplicate */ if ( pid!=0 ) /* Branch based on return value from fork() */ { printf(“I’m the parent process with PID %d and PPID %d\n”, getpid(), getppid() ); childPid = wait( &status ); /* Wait for a child to terminate. */ printf(“ A child with PID %d terminated with exit code %d \n”, childPid, status >> 8 ); } else { printf(“I’m the child process with PID %d and PPID %d \n”, exit(42); /* Exit with a silly number */ printf(“PID %d terminates \n”, getpid() ); Prof. Andrzej (AJ) Bieszczad Phone:
135
Waiting for a child(): wait()
$ mywait > run the program. I’m the parent process with PID and PPID 13464 I’m the parent process with PID and PPID 13409 A child with PID terminated with exit code 42 PID terminates $ _ Prof. Andrzej (AJ) Bieszczad Phone:
136
Differentiating a Process: exec()
- A process may replace its current code, data, and stack with those of another executable by using one of the “exec()” family of system calls. - When a process executes an “exec()” system call, its PID and PPID numbers stay the same - only the code that the process is executing changes. System Call: int execl( const char* path, const char* arg0, const char* arg1,…, const char* argn, NULL ) int execv( const char* path, const char* argv[] ) int execlp( const char* path, const char* arg0, const char* arg1, …, const char* argn, NULL) int execvp( const char* path, const char* argv[] ) The “exec()” family of system calls replaces the calling process’ code, data, and stack with those of the executable whose pathname is stored in path. Prof. Andrzej (AJ) Bieszczad Phone:
137
Differentiating a Process: exec()
- “execlp()” and “execvp()” use the $PATH environment variable to find path. - If the executable is not found, the system call returns a value of -1; otherwise, the calling process replaces its code, data, and stack with those of the executable and starts to execute the new code. - “execl()” and “execlp()” invoke the executable with the string arguments pointed to by arg1 through argn. arg0 must be the name of the executable file itself, and the list of arguments must be null terminated. - “execv()” and “execvp()” invoke the executable with the string arguments pointed to by argv[1] to argv[n], where argv[n+1] is NULL. argv[0] must be the name of the executable file itself. Prof. Andrzej (AJ) Bieszczad Phone:
138
Differentiating a Process: exec()
- In the next example, the program displays a small message and then replaces its code with that of the “ls”. $ cat myexec.c > list the program. #include <stdio.h> main() { printf(“I’m process %d and I’m about to exec an ls -l \n”, getpid() ); execl( “/bin/ls”, “ls”, “-l”, NULL ); /* Execute ls */ printf(“This line should never be executed \n”); } $ myexec > run the program. I’m process and I’m about to exec an ls -l total 125 -rw-r--r glass Feb :47 myexec.c -rwxr-xr-x glass Feb :48 myexec $ _ Prof. Andrzej (AJ) Bieszczad Phone:
139
Sample Program: Background Processing
- makes use of “fork()” and “exec()” to execute a program in the background. - The original process creates a child to “exec” the specified executable and then terminates. - The orphaned child is automatically adopted by “init”. - used “execvp()” so that the program could use $PATH to find the executable file. Prof. Andrzej (AJ) Bieszczad Phone:
140
Sample Program: Background Processing
$ cat background.c ---> list the program. #include <stdio.h> main( argc, argv ) int argc; char* argv[]; { if ( fork() == 0 ) /* Child */ execvp( argv[1], &argv[1] ); /* Execute other program */ fprintf( stderr, “Could not execute %s \n”, argv[1] ); } $ background.exe cc wait.c ---> run the program. Prof. Andrzej (AJ) Bieszczad Phone:
141
Sample Program: Background Processing
$ ps > confirm that “cc” is in. PID TT STAT TIME COMMAND p S : csh ( csh ) P R : ps p D : cc wait.c $ _ To Signals Prof. Andrzej (AJ) Bieszczad Phone:
142
Changing Directories: chdir()
- Every process has a current working directory that is used when processing a relative pathname. - A child process inherits its current working directory from its parent. - For example, when a utility is executed from a shell, its process inherits the shell’s current working directory. - To change a process’ current working directory, use “chdir()”, which works as follows: System Call: int chdir( const char* pathname ) “chdir()” sets a process’ current working directory to the directory pathname. The process must have execute permission from the directory to succeed. Prof. Andrzej (AJ) Bieszczad Phone:
143
Changing Directories: chdir()
- the process prints its current working directory before and after executing “chdir()” by executing pwd using the “system()” library routine: $ cat mychdir.c > list the source code #include <stdio.h> main() { system(“pwd”); /* Display current working directory */ chdir(“/”); /* Change working directory to root directory */ system(“pwd”); /* Display new working directory */ chdir(“/home/glass”); /* Change again */ system(“pwd”); /* Display again */ } $ mychdir > execute the program. /home/glass / $ _ Prof. Andrzej (AJ) Bieszczad Phone:
144
Changing Priorities: nice()
- Every process has a priority value between -20 and +19 that affects the amount of CPU time that it’s allocated. - In general, the smaller the priority value, the faster the process will run. Only super-user and kernel processes can have a negative priority value, and login shells start with a priority value of zero. - A child process inherits its priority value from its parent and may change it by using “nice()”: System Call: int nice(int delta) “nice()” adds delta to a process’ current priority value. Only a superuser may specify a delta that leads to a negative priority value. Prof. Andrzej (AJ) Bieszczad Phone:
145
Changing Priorities: nice()
Legal priority values lie between -20 and +19. If a delta beyond one of these limits is specified, the priority value is truncated to the limit. If “nice()” succeeds, it returns the new priority value; otherwise it returns a value of -1. this return value causes an ambiguity, since a value of -1 is a legal priority value. Prof. Andrzej (AJ) Bieszczad Phone:
146
Changing Priorities: nice()
- In the next example, the process executed ps commands before and after a couple of “nice()” calls. - When the process’ priority value became nonzero, it was flagged with an “N” by ps, together with the sh and ps commands that it created due to the “system()” library call. Prof. Andrzej (AJ) Bieszczad Phone:
147
Changing Priorities: nice()
$ cat mynice.c > list the source code. #include <stdio.h> main() { printf(“original priority \n”); system(“ps”); /* Execute a ps */ nice(0); /* Add 0 to my priority */ printf(“running at priority 0 \n”); system(“ps”); /* Execute another ps */ nice(10); /* Add 10 to my priority */ printf(“running at priority 10 \n”); system(“ps”); /* Execute the last ps */ } Prof. Andrzej (AJ) Bieszczad Phone:
148
Changing Priorities: nice()
$ mynice > execute the program. original priority PID TT STAT TIME COMMAND p S :00 -sh(sh) p S :00 a.out p S :00 sh -c ps p R :00 ps running at priority > adding a priority value of zero doesn’t ---> change the priority. PID TT STAT TIME COMMAND p S :00 sh -c ps p R :00 ps running at priority > adding a priority value of 10 makes them ---> run slower. Prof. Andrzej (AJ) Bieszczad Phone:
149
Changing Priorities: nice()
PID TT STAT TIME COMMAND p S :00 -sh(sh) p S N :00 a.out p S N :00 sh -c ps p R N :00 ps $ _ Prof. Andrzej (AJ) Bieszczad Phone:
150
Accessing User and Group IDs
- the system calls that allow you to read a process’ real and effective IDs: System Call : uid_t getuid() uid_t geteuid() gid_t getgid() gid_t getegid() “getuid()” and “geteuid()” return the calling process’ real and effective user IDs, respectively. “getgid()” and “getegid()” return the calling process’ real and effective group IDs, respectively. The ID numbers correspond to the user and group IDs listed in “/etc/passwd” and “/etc/group” files. These calls always succeed. Prof. Andrzej (AJ) Bieszczad Phone:
151
Accessing User and Group IDs
- the system calls that allow you to set a process’ real and effective IDs: System Call : int setuid(uid_t id) int seteuid(uid_t id) int setgid(gid_t id) int setegid(gid_t id) “setuid()” and “setgid()” set the calling process’ effective user ID and group ID, respectively. “setuid()” and “setgid()” set the calling process’ effective and real user IDs and group IDs, respectively, to the specified value. These calls succeed only if executed by a super-user, or if id is the real or effective user or group ID of the calling process. They return a value of 0 if successful; otherwise, they return a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
152
Sample Program: Disk Usage
- uses a novel technique for counting the number of non-directory files in hierarchy. - When the program is started, its first argument must be the name of the directory to search. - It searches through each entry in the directory, spawning off a new process for each entry. - Each child process either exits with 1 if its associated file is a non-directory file, or repeats the process, summing up the exit codes of its children and exiting with the total count. - creates a large number of processes, which is not particularly efficient. Prof. Andrzej (AJ) Bieszczad Phone:
153
Sample Program: Disk Usage
$ cat count.c > list the program #include <stdio.h> #include <sys/file.h> #include <sys/dir.h> #include <sys/stat.h> long processFile(); long processDirectory(); main( argc, argv ) int argc; char* argv[]; { long count; count = processFile( argv[1] ); printf(“Total number of non-directory files is %ld \n”, count ); return ( /* EXIT_SUCCESS */ 0 ); } Prof. Andrzej (AJ) Bieszczad Phone:
154
Sample Program: Disk Usage
long processFile( name ) char* name; { struct stat statBuf; /* To hold the return data from stat() */ mode_t mode; int result; result = stat( name, &statBuf ); /* Stat the specified file */ if ( result == -1 ) return(0); /* Error */ mode = statBuf.st_mode; /* Look at the file’s mode */ if ( S_ISDIR( mode ) ) /* Directory */ return ( processDirectory(name) ); else return ( 1 ); /* A nondirectory file was processed */ } Prof. Andrzej (AJ) Bieszczad Phone:
155
Sample Program: Disk Usage
long processDirectory( dirName ) char* dirName; { int fd, children, i, charsRead, childPid, status; long count, totalCount; char fileName[100]; struct dirent dirEntry; fd = open( dirName, O_RDONLY ); /* Open directory for reading */ children = 0; /* Initialize child process count */ while(1) /* Scan directory */ charsRead = getdents( fd, &dirEntry, sizeof( struct dirent ) ); if ( charsRead == 0 ) break; /* End of directory */ if ( strcmp( dirEntry.d_name, “.” ) != 0 && strcmp( dirEntry.d_name, “..” ) != 0 ) Prof. Andrzej (AJ) Bieszczad Phone:
156
Sample Program: Disk Usage
if ( fork() == 0 ) /* Creates a child to process dir. entry */ { sprintf( fileName, “%s/%s”, dirName, dirEntry.d_name ); count = processFile( fileName ); exit( count ); } else ++children; /* Increment count of child processes */ lseek( fd, dirEntry.d_off, SEEK_SET ); /* Jump to next dir. entry */ close(fd); /* Close directory */ totalCount = 0; /* Initialize file count */ for ( i=1; i<= children; i++ ) /* Wait for children to terminate */ childPid = wait( &status ); /* Accept child’s termination code */ totalCount += ( status >> 8 ); /* Update file count */ return ( totalCount ); /* Return number of files in directory */ Prof. Andrzej (AJ) Bieszczad Phone:
157
Sample Program: Disk Usage
$ ls -F > list current directory. a.out* disk.c fork tmp/ zombie* background myexec.c myfork.c mywait.c background.c myexit.c orphan.c mywait* count* myexit* orphan* zombie.c $ ls tmp > list only subdirectory a.out* disk.c myexit.c orphan.c background.c myexec.c myfork.c mywait.c zombie.c $ count > count regular files from “.”. Total number of non-directory files is 25 $ - Prof. Andrzej (AJ) Bieszczad Phone:
158
Threads - Multiple processes are expensive to create, either a new or by copying an existing process with the “fork()” system call. - A thread is an abstraction allows multiple “threads of control” in a single process space. - separate tasks to be able to share some resources in a process, such as memory space or an open device. - the term “light-weight processes” used interchangeably with “thread”. Prof. Andrzej (AJ) Bieszczad Phone:
159
Thread Management - Four major functions make up the common thread-management capabilities in most implementations: Create : Create a thread. Join : Suspend and wait for a created thread to terminate. Detach : Allow the thread to release its resources to the system when it finishes and not to require a join. Terminate : Return resources to the process. Prof. Andrzej (AJ) Bieszczad Phone:
160
Redirection - When a process forks, the child inherits a copy of its parent’s file descriptors. - When a process execs, all nonclose-on-exec file descriptors remain unaffected, including the standard input, output, and error channels. ls > ls.out - To perform the redirection, the shell performs the following series of actions: The parent shell forks and then waits for the child shell to terminate. The child shell opens the file “ls.out”, creating it or truncating it as necessary. Prof. Andrzej (AJ) Bieszczad Phone:
161
Redirection The child shell then duplicates the file descriptor of “ls.out” to the standard-output file descriptor, number 1, and the closes the original descriptor of “ls.out”. All standard output is therefore redirected to “ls.out”. The child shell then exec’s the ls utility. Since the file descriptors are inherited during an “exec()”, all of the standard output of ls goes to “ls.out”. When the child shell terminates, the parent resumes. The parent’s file descriptors are unaffected by the child’s actions, as each process maintains its own private descriptor table. Prof. Andrzej (AJ) Bieszczad Phone:
162
Redirection - a small program that does approximately the same kind of redirection as a UNIX shell does. $ cat redirect.c > list the program. #include <stdio.h> #include <fcntl.h> main( argc, argv ) int argc; char* argv[]; { int fd; /* Open file for redirection */ fd = open( argv[1], O_CREAT | O_TRUNC | O_WRONLY, 0600 ); dup2( fd, 1 ); /* Duplicate descriptor to standard output */ close(fd); /* Close original descriptor to save descriptor space */ execvp( argv[2], &argv[2] ); /* Invoke program; will inherit stdout */ perror(“main”); /* Should never execute */ } Prof. Andrzej (AJ) Bieszczad Phone:
163
Redirection $ redirect ls.out ls -l > redirect “ls -l” to “ls.out”. $ cat ls.out > list the output file. total 5 -rw-r-xr-x gglass Feb :35 ls.out -rw-r-xr-x gglass Feb :35 redirect.c -rwxr-xr-x gglass Feb :33 redirect $ - Prof. Andrzej (AJ) Bieszczad Phone:
164
SIGNALS - Program must sometimes deal with unexpected or unpredictable events, such as : a floating point error a power failure an alarm clock “ring” the death of a child process a termination request from a user ( i.e., Control-C ) a suspend request from a user ( i.e., Control-Z ) - These kind of events are sometimes called interrupts, as they must interrupt the regular flow of a program in order to be processed. - When UNIX recognizes that such an event has occurred, it sends the corresponding process a signal. Prof. Andrzej (AJ) Bieszczad Phone:
165
SIGNALS - There is a unique, numbered signal for each possible event. For example, if a process causes a floating point error, the kernel sends the offending process signal number 8: Signal Process #8 FIGURE Floating point error signal Prof. Andrzej (AJ) Bieszczad Phone:
166
SIGNALS - any process can send any other process a signal,
as long as it has permission to do so. - A programmer may arrange for a particular signal to be ignored or to be processed by a special piece of code called a signal handler. - the process that receives the signal 1) suspends its current flow of control, 2) executes the signal handler, 3) and then resumes the original flow of control when the signal handler finishes. Prof. Andrzej (AJ) Bieszczad Phone:
167
SIGNALS - Signals are defined in “/usr/include/sys/signal.h”.
The default handler usually performs one of the following actions: terminate the process and generate a core file ( dump ) terminate the process without generating a core image file ( quit ) ignore and discard the signal ( ignore ) suspend the process ( suspend ) resume the process Prof. Andrzej (AJ) Bieszczad Phone:
168
A List of Signals - Here’s a list of the System V predefined signals, along with their respective macro definitions, numerical values, and default actions, as well as a brief description of each: Macro # Default Description SIGHUP quit hang up SIGINT quit interrupt SIGQUIT dump quit SIGILL dump illegal instruction SIGTRAP dump trace trap( used by debuggers ) SIGABRT dump abort SIGEMT dump emulator trap instruction SIGFPE dump arithmetic execution SIGKILL quit kill( cannot be caught, blocked, or ignored ) SIGBUS dump bus error( bad format address ) Prof. Andrzej (AJ) Bieszczad Phone:
169
A List of Signals Macro # Default Description
SIGSEGV dump segmentation violation(out-of-range address) SIGSYS dump bad argument to system call SIGPIPE quit write on a pipe or other socket with no one to read it. SIGALRM quit alarm clock SIGTERM quit software termination signal(default signal sent by kill ) SIGUSR quit user signal 1 SIGUSR quit user signal 2 SIGCHLD ignore child status changed SIGPWR ignore power fail or restart SIGWINCH 20 ignore window size change SIGURG ignore urgent socket condition SIGPOLL ignore pollable event Prof. Andrzej (AJ) Bieszczad Phone:
170
A List of Signals Macro # Default Description
SIGSTOP quit stopped(signal) SIGSTP quit stopped(user) SIGCONT ignore continued SIGTTIN quit stopped( tty input ) SIGTTOU quit stopped( tty output ) SIGVTALRM 28 quit virtual timer expired SIGPROF quit profiling timer expired SIGXCPU dump CPU time limit exceeded SIGXFSZ dump file size limit exceeded Prof. Andrzej (AJ) Bieszczad Phone:
171
Terminal Signals - The easiest way to send a signal to a foreground process is by pressing Control-C or Control-Z from the keyboard. - When the terminal driver ( the piece of software that supports the terminal ) recognizes that Control-C was pressed. it sends a SIGINT signal to all of the processes in the current foreground job. - Similarly, Control-Z causes it to send a SIGTSTP signal to all of the processes in the current foreground job. - By default, SIGINT terminates a process and SIGTSTP suspends a process. Prof. Andrzej (AJ) Bieszczad Phone:
172
Requesting an Alarm Signal : alarm()
- SIGALRM, by using “alarm()”. The default handler for this signal displays the message “Alarm clock” and terminates the process. Here’s how “alarm()” works: System Call : unsigned int alarm( unsigned int count ) “alarm()” instructs the kernel to send the SIGALRM signal to the calling process after count seconds. If an alarm had already been scheduled, that alarm is overwritten. If count is 0, any pending alarm requests are cancelled. “alarm()” returns the number of seconds that remain until the alarm signal is sent. Prof. Andrzej (AJ) Bieszczad Phone:
173
Requesting an Alarm Signal : alarm()
- a small program that uses “alarm()”, together with its output: $ cat alarm.c > list the program. #include <stdio.h> main() { alarm(3); /* Schedule an alarm signal in three seconds */ printf(“Looping forever… \n”); while(1) printf(“This line should never be executed \n”); } $ alarm > run the program. Looping forever… Alarm clock > occurs three seconds later. $ - Prof. Andrzej (AJ) Bieszczad Phone:
174
Handling Signals : signal()
System Call: void(*signal(int sigCode, void (*func)(int))) (int) “signal()” allows a process to specify the action that it will take when a particular signal is received. The parameter sigCode specifies the number of the signal that is to be reprogrammed, and func may be one of several values: SIG_IGN, which indicates that the specified signal should be ignored and discarded. SIG_DFL, which indicates that the kernel’s default handler should be used. an address of a user-defined function, which indicates that the function should be executed when the specified signal arrives. Prof. Andrzej (AJ) Bieszczad Phone:
175
Handling Signals : signal()
The valid signal numbers are stored in “/usr/include/signal.h”. The signals SIGKILL and SIGSTP may not be reprogrammed. A child process inherits the signal settings from its parent during a “fork()”. “signal()” returns the previous func value associated with sigCode if successful; otherwise, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
176
Handling Signals: signal()
- The “signal()” system call may be used to override the default action. - a couple of changes to the previous program so that it caught and processed the SIGALRM signal efficiently: installed my own signal handler, “alarmHandler()”, by using “signal()”. Prof. Andrzej (AJ) Bieszczad Phone:
177
Handling Signals: signal()
System Call: int pause(void) “pause()” suspends the calling process and returns when the calling process receives a signal. It is most often used to wait efficiently for an alarm signal. “pause()” doesn’t return anything useful. Prof. Andrzej (AJ) Bieszczad Phone:
178
Handling Signals: signal()
$ cat handler.c > list the program. #include <stdio.h> #include <signal.h> int alarmFlag = 0; /* Global alarm flag */ void alarmHandler(); /* Forward declaration of alarm handler */ /***************************************************/ main() { signal( SIGALRM, alarmHandler ); /* Install signal handler */ alarm(3); /* Schedule an alarm signal in three seconds */ printf(“Looping…\n”); while( !alarmFlag ) /* Loop until flag set */ pause(); /* Wait for a signal */ } printf(“Loop ends due to alarm signal \n”); Prof. Andrzej (AJ) Bieszczad Phone:
179
Handling Signals: signal()
/*************************************************/ void alarmHandler() { printf(“An alarm clock signal was received \n”); alarmFlag=1; } $ handler > run the program. Looping… An alarm clock signal was received ---> occurs three seconds later. Loop ends due to alarm signal $ _ Prof. Andrzej (AJ) Bieszczad Phone:
180
Protecting Critical Code and Chaining Interrupt Handlers
- The same techniques that I just described may be used to protect critical pieces of code against Control-C attacks and other such signals. - it can be restored after the critical code has executed. Here’s the source code of a program that protects itself against SIGINT signals: $ cat critical.c > list the program. #include <stdio.h> #include <signal.h> main() { void (*oldHandler) () /* To hold old handler value */ printf(“I can be Control-C’ed \n”); sleep(3); oldHandler = signal(SIGINT, SIG_IGN); /* Ignore Control-C */ printf(“I’m protected from Control-C now\n”); Prof. Andrzej (AJ) Bieszczad Phone:
181
Protecting Critical Code and Chaining Interrupt Handlers
sleep(3); signal(SIGINT, oldHandler); /* Restore old handler */ printf(“I can be Control-C’ed again \n”); printf(“Bye! \n”); } $ critical > run the program. I can be Control-C’ed ^C > Control-C works here. $ critical > run the program again. I’m protected from Control-C now ^C I can be Control-C’ed again Bye! $ _ To Pipes and Sockets Prof. Andrzej (AJ) Bieszczad Phone:
182
Sending Signals: kill()
- A process may send a signal to another process by using the “kill()” system call. - “kill()” is a misnomer, since many of the signals that it can send to do not terminate a process. - It’s called “kill()” because of historical reasons; the main use of signals when UNIX was first designed was to terminate processes. Prof. Andrzej (AJ) Bieszczad Phone:
183
Sending Signals: kill()
System Call: int kill( pid_t pid, int sigCode ) “kill()” sends the signal with value sigCode to the process with PID pid. “kill()” succeeds and the signal is sent as long as at least one of the following conditions is satisfied: The sending process and the receiving process have the same owner. The sending process is owned by a super-user. There are a few variations on the way that “kill()” works: If pid is zero, the signal is sent to all of the processes in the sender’s process group. If pid is -1 and the sender is owned by a super-user, the signal is sent to all processes, including the sender. Prof. Andrzej (AJ) Bieszczad Phone:
184
Sending Signals: kill()
If pid is -1 and the sender is not owned by a super-user, the signal is sent to all of the processes owned by the same owner as that of the sender, excluding the sending process. If the pid is negative, but not -1, the signal is sent to all of the processes in the process group. Prof. Andrzej (AJ) Bieszczad Phone:
185
Death of Children - When a parent’s child terminates, the child process sends its parent a SIGCHLD signal. - A parent process often installs a handler to deal with this signal; this handler typically executes a “wait()” system call to accept the child’s termination code and let the child dezombify. - the parent can choose to ignore SIGCHLD signals, in which case the child dezombifies automatically. Prof. Andrzej (AJ) Bieszczad Phone:
186
Death of Children - The program works by performing the following steps: 1. The parent process installs a SIGCHLD handler that is executed when its child process terminates. 2. The parent process forks a child process to execute the command. 3. The parent process sleeps for the specified number of seconds. when it wakes up, it sends its child process a SIGINT signal to kill it. 4. If the child terminates before its parent finishes sleeping, the parent’s SIGCHLD handler is executed, causing the parent to terminate immediately. Prof. Andrzej (AJ) Bieszczad Phone:
187
Death of Children - Here’s the source code for and sample output from the program. $ cat limit.c > list the program. #include <stdio.h> #include <signal.h> int delay; void childHandler(); /************************************************/ main( argc, argv ) int argc; char* argv[]; { int pid; signal( SIGCHLD, childHandler ); /* Install death-of-child handler */ pid = fork(); /* Duplicate */ if ( pid == 0 ) /* Child */ execvp( argv[2], &argv[2] ); /* Execute command */ Prof. Andrzej (AJ) Bieszczad Phone:
188
Death of Children perror(“limit”); /* Should never execute */ }
else /* Parent */ { sscanf( argv[1], “%d”, &delay ); /* Read delay from command-line */ sleep(delay); /* Sleep for the specified number of seconds */ printf(“Child %d exceeded limit and is being killed \n”, pid ); kill( pid, SIGINT ); /* Kill the child */ /***************************************************/ void childHandler() /* Executed if the child dies before the parent */ int childPid, childStatus; childPid = wait(&childStatus); /* Accept child’s termination code */ printf(“Child %d terminated within %d seconds \n”, childPid, delay); exit(/* EXIT SUCCESS */ 0); Prof. Andrzej (AJ) Bieszczad Phone:
189
Death of Children $ limit 5 ls ---> run the program; command finishes OK. a.out alarm critical handler limit alarm.c critical.c handler.c limit.c Child terminated within 5 seconds. $ limit 4 sleep > run it again; command takes too long Child 4032 exceeded limit and is being killed. $ - Prof. Andrzej (AJ) Bieszczad Phone:
190
Suspending and Resuming Processes
- The SIGSTOP and SIGCONT signals suspend and resume a process, respectively. They are used by the UNIX shells that support job control to implement built-in commands like stop, fg, and bg. - The main program created two children that both entered an infinite loop and displayed a message every second. - The main program waited for three seconds and then suspended the first child. - After another three seconds, the parent restarted the first child, waited a little while longer, and then terminated both children. Prof. Andrzej (AJ) Bieszczad Phone:
191
Suspending and Resuming Processes
$ cat pulse.c > list the program. #include <signal.h> #include <stdio.h> main() { int pid1; int pid2; pid1 = fork(); if (pid1== 0) /* First child */ { while(1) /* Infinite loop */ printf(“pid1 is alive \n”); sleep(1); } pid2 = fork(); /* Second child */ Prof. Andrzej (AJ) Bieszczad Phone:
192
Suspending and Resuming Processes
if ( pid2 == 0 ) { while(1) /* Infinite loop */ printf(“pid2 is alive \n”); sleep(1); } sleep(3); kill( pid1, SIGSTOP ); /* Suspend first child */ kill( pid1, SIGCONT ); /* Resume first child */ kill( pid1, SIGINT ); /* Kill first child */ kill( pid2, SIGINT ); /* Kill second child */ Prof. Andrzej (AJ) Bieszczad Phone:
193
Suspending and Resuming Processes
$ pulse > run the program. pid1 is alive > both run in first three seconds. pid2 is alive pid1 is alive pid2 is alive > just the second child runs now. pid1 is alive > the first child is resumed. Prof. Andrzej (AJ) Bieszczad Phone:
194
Process Groups and Control Terminals
- When you’re in a shell and you execute a program that creates several children, a single Control-C from the keyboard will normally terminate the program and its children and then return you to the shell. - In order to support this kind of behavior, UNIX introduced a few new concepts. In addition to having a unique process ID number, every process is also a member of a process group. Several processes can be members of the same process group. When a process forks, the child inherits its process group from its parent. Prof. Andrzej (AJ) Bieszczad Phone:
195
Process Groups and Control Terminals
A process may change its process group to a new value by using “setpgid()”. When a process execs, its process group remains the same. Every process can have an associated control terminal, which is typically the terminal where the process was started. When a process forks, the child inherits its control terminal from its parent. When a process execs, its control terminal stays the same. Prof. Andrzej (AJ) Bieszczad Phone:
196
Process Groups and Control Terminals
Every terminal can be associated with a single control process. When a metacharacter such as a Control-C is detected, the terminal sends the appropriate signal to all of the processes in the process group of its control process. If a process attempts to read from its control terminal and is not a member of the same process group as the terminal’s control process, the process is sent a SIGTTIN signal, which normally suspends the process. Prof. Andrzej (AJ) Bieszczad Phone:
197
Process Groups and Control Terminals
Here’s how a shell uses these features: When an interactive shell begins, it is the control process of a terminal and has that terminal as its control terminal. When a shell executes a foreground process, the child shell places itself in a different process group before exec’ing the command and takes control of the terminal. Any signals generated from the terminal thus go to the foreground command rather than to the original parent shell. When the foreground command terminates, the original parent shell takes back control of the terminal. Prof. Andrzej (AJ) Bieszczad Phone:
198
Process Groups and Control Terminals
When a shell executes a background process, the child shell places itself in a different process group before exec’ing, but does not take control of the terminal. Any signals generated from the terminal continue to go to the shell. If the background process tries to read from its control terminal, it is suspended by a SIGTTIN signal. Prof. Andrzej (AJ) Bieszczad Phone:
199
The terminal’s control
Process Groups and Control Terminals FIGURE Control terminals and process groups Process group 145 Process group 171 Process group 230 231 233 148 150 174 176 230 145 171 Processes in groups 145,171, and 230 share the same controlling terminal Signals The terminal’s control process is 171 Prof. Andrzej (AJ) Bieszczad Phone:
200
Process Groups and Control Terminals
”setpgid()” changes a process’ group, System Call : pid_t setpgid( pid_t pid, pid_t pgrpId) “setpgid()” sets the process-group ID of the process with PID pid to pgrpId. If pid is zero, the caller’s process group ID is set to pgrpId. In order for “setpgid()” to succeed and set the process group ID, at least one of the following conditions must be met: • The caller and the specified process must have the same owner. • The caller must be owned by a super-user. If “setpgid()” fails, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
201
Process Groups and Control Terminals
A process may find out its current process-group ID by using “getpgid()”, System Call: pid_t getpgid( pid_t pid ) “getpgid()” returns the process group ID of the process with PID pid. If pid is zero, the process group ID of the caller is returned. Prof. Andrzej (AJ) Bieszczad Phone:
202
Process Groups and Control Terminals
The next example illustrates the fact that a terminal distributes signals to all of the processes in its control process’ process group. Since the child inherited its process group from its parent, both the parent and child caught the SIGINT signal. $ cat pgrp1.c > list program. #include <signal.h> #include <stdio.h> void sigintHanler(); main() { signal( SIGINT, singintHandler); /* Handle Control-C */ if ( fork() == 0 ) printf(“Child PID %d PGRP %d waits \n”, getpid(), getpgid(0)); else printf(“Parent PID %d PGRP %d waits \n”, getpid(), getpgid(0)); Prof. Andrzej (AJ) Bieszczad Phone:
203
Process Groups and Control Terminals
pause(); /* Wait for a signal */ } void sigintHandler() { printf(“Process %d got a SIGINT \n”, getpid() ); $ pgrp > run the program. Parent PID PGRP waits Child PID PGRP waits ^C Process got a SIGINT Process got a SIGINT $ - Prof. Andrzej (AJ) Bieszczad Phone:
204
Process Groups and Control Terminals
If a process places itself into a different process group, then it is no longer associated with the terminal’s control process and does not receive signals from the terminal. In the following example, the child process was not affected by Control-C. $ cat pgrp2.c > list the program. #include <signals.h> #include <stdio.h> void sigintHandler(); main() { int i; signal( SIGINT, sigintHandler ); /* Install signal handler */ if ( fork()==0 ) setpgid( 0, getpid() ); /* Place child in its own process group */ Prof. Andrzej (AJ) Bieszczad Phone:
205
Process Groups and Control Terminals
printf(“Process PID %d PGRP %d waits \n”, getpid(), getpgid(0) ); for (i=0; i<=3; i++) /* Loop three times */ { printf(“Process %d is alive \n”, getpid() ); sleep(1); } void sigintHandler() printf(“Process %d got a SIGINT\n”, getpid() ); exit(1); Prof. Andrzej (AJ) Bieszczad Phone:
206
Process Groups and Control Terminals
$ pgrp > run the program. Process PID PGRP waits Process PID PGRP waits ^C > type Control-C Process got a SIGINT > parent receives signal. Process is alive $- Prof. Andrzej (AJ) Bieszczad Phone:
207
Process Groups and Control Terminals
- If a process attempts to read from its control terminal after it disassociates itself with the terminal’s control process, it is sent a SIGTTIN signal, which suspends the receiver by default. In the following example, I trapped SIGTTIN with my own handler to make the effect a little clearer: $ cat pgrp3.c > list the program. #include <signal.h> #include <stdio.h> #include <sys/termio.h> #include <fcntl.h> void sigttinHandler(); main() { int status; char str[100]; Prof. Andrzej (AJ) Bieszczad Phone:
208
Process Groups and Control Terminals
if ( fork() == 0 ) /* Child */ { signal( SIGTTIN, sigttinHandler); /* Install handler */ setpgid( 0, getpid() ); /* Place myself in a new process group */ printf(“Enter a string: “); scanf(“%s”, str); /* Try to read from control terminal */ printf(“You entered %s \n”, str); } else /* Parent */ wait(&status); /* Wait for child to terminate */ void sigttinHandler() printf(“Attempted inappropriate read from control terminal \n”); exit(1); Prof. Andrzej (AJ) Bieszczad Phone:
209
Process Groups and Control Terminals
$ pgrp > run the program. Enter a string: Attempted inappropriate read from control terminal $ - Prof. Andrzej (AJ) Bieszczad Phone:
210
IPC - Interprocess Communication(IPC) is the generic term describing
how two processes may exchange information with each other. - In general, the two processes may be running on the same machine or on different machines, although some IPC mechanisms may only support local usage ( e.g., signals and pipes ) - This communication may be an exchange of data for which two or more processes are cooperatively processing the data or synchronization information to help two independent, but related, processes schedule work so that they do not destructively overlap. Prof. Andrzej (AJ) Bieszczad Phone:
211
Pipes - Pipes are an interprocess communication mechanism
that allow two or more processes to send information to each other. - commonly used from within shells to connect the standard output of one utility to the standard input of another. - For example, here’s a simple shell command that determines how many users there are on the system: $ who | wc -l - The who utility generates one line of output per user. This output is then “piped” into the wc utility, which, when invoked with the “-l” option, outputs the total number of lines in its input. Prof. Andrzej (AJ) Bieszczad Phone:
212
Bytes from “who” flow through the pipe to “wc”
Pipes who pipe wc Bytes from “who” flow through the pipe to “wc” A simple pipe Prof. Andrzej (AJ) Bieszczad Phone:
213
Pipes - It’s important to realize that both the writer process and the reader process of a pipeline execute concurrently; - a pipe automatically buffers the output of the writer and suspends the writer if the pipe gets too full. - Similarly, if a pipe empties, the reader is suspended until some more output becomes available. - All versions of UNIX support unnamed pipes, which are the kind of pipes that shells use. - System V also supports a more powerful kind of pipe called a named pipe. Prof. Andrzej (AJ) Bieszczad Phone:
214
Unnamed Pipes: “pipe()”
- An unnamed pipe is a unidirectional communications link that automatically buffers its input ( the maximum size of the input varies with different versions of UNIX, but is approximately 5K ) and may be created using the “pipe()” system call. - Each end of a pipe has an associated file descriptor. The “write” end of the pipe may be written to using “write()”, and the “read” end may be read from using “read()”. - When a process has finished with a pipe’s file descriptor. it should close it using “close()”. Prof. Andrzej (AJ) Bieszczad Phone:
215
Unnamed Pipes: “pipe()”
System Call : int pipe( int fd[2] ) “pipe()” creates an unnamed pipe and returns two file descriptors: The descriptor associated with the “read” end of the pipe is stored in fd[0], and the descriptor associated with the “write” end of the pipe is stored in fd[1]. If a process reads from a pipe whose “write” end has been closed, the “read()” call returns a value of zero, indicating the end of input. If a process reads from an empty pipe whose “write” end is still open, it sleeps until some input becomes available. Prof. Andrzej (AJ) Bieszczad Phone:
216
Unnamed Pipes: “pipe()”
If a process tries to read more bytes from a pipe than are present, all of the current contents are returned and “read()” returns the number of bytes actually read. If a process writes to a pipe whose “read” end has been closed, the write fails and the writer is sent a SIGPIPE signal. the default action of this signal is to terminate the receiver. If a process writes fewer bytes to a pipe than the pipe can hold, the “write()” is guaranteed to be atomic; that is, the writer process will complete its system call without being preempted by another process. If the kernel cannot allocate enough space for a new pipe, “pipe()” returns a value of -1; otherwise, it returns a value of 0. Prof. Andrzej (AJ) Bieszczad Phone:
217
Unnamed Pipes: “pipe()”
- Assume that the following code was executed: int fd[2]; pipe(fd); the data structures shown in Figure would be created. fd[0] Write end Pipe fd[1] Read end Figure An unnamed pipe Prof. Andrzej (AJ) Bieszczad Phone:
218
Unnamed Pipes: “pipe()”
- Unnamed pipes are usually used for communication between a parent process and its child, with one process writing and the other process reading. The typical sequence of events for such a communication is as follows: 1. The parent process creates an unnamed pipe using “pipe()”. 2. The parent process forks. 3. The writer closes its “read” end of the pipe, and the designated reader closes its “write” end of the pipe. 4. The processes communicate by using “write()” and “read()” calls. 5. Each process closes its active pipe descriptor when it’s finished with it. Prof. Andrzej (AJ) Bieszczad Phone:
219
Unnamed Pipes: “pipe()”
- Bidirectional communiation is only possible by using two pipes. Here’s a small program that uses a pipe to allow the parent to read a message from its child: $ cat talk.c > list the program. #include <stdio.h> #define READ /* The index of the “read” end of the pipe */ #define WRITE /* The index of the “write” end of the pipe */ char* phrase =“Stuff this in your pipe and smoke it”; main() { int fd[2], bytesRead; char message[100]; /* Parent process’ message buffer */ pipe(fd); /* Create an unnamed pipe */ Prof. Andrzej (AJ) Bieszczad Phone:
220
Unnamed Pipes: “pipe()”
if ( fork() == 0 ) /* Child, write */ { close(fd[READ]); /* Close unused end */ write(fd[WRITE], phrase, strlen(phrase)+1); /* Send */ close(fd[WRITE]); /* Close used end */ } else /* Parent, reader */ close(fd[WRITE]); /* Close unused end */ bytesRead = read( fd[READ], message, 100 ); /* Receive */ printf(“Read %d bytes: %s \n”, bytesRead, message ); close(fd[READ]); /* Close used end */ $ talk > run the program. Read 37 bytes: Stuff this in your pipe and smoke it $ _ Prof. Andrzej (AJ) Bieszczad Phone:
221
Unnamed Pipes: “pipe()”
- The child included the phrase’s NULL terminator as part of the message so that the parent could easily display it. - When a writer process sends more than one variable-length message into a pipe, it must use a protocol to indicate to the reader the location for the end of the message. Methods for such indication include : • sending the length of a message(in bytes) before sending the message itself • ending a message with a special character such as a new line or a NULL Prof. Andrzej (AJ) Bieszczad Phone:
222
Unnamed Pipes: “pipe()”
- UNIX shells use unnamed pipes to build pipelines. connecting the standard output of the first to the standard input of the second. $ cat connect.c ---> list the program. #include <stdio.h> #define READ 0 #define WRITE 1 main( argc, argv ) int argc; char* argv[]; { int fd[2]; pipe(fd); /* Create an unnamed pipe */ if ( fork()!=0 ) /* Parent, writer */ close( fd[READ] ); /* Close unused end */ Prof. Andrzej (AJ) Bieszczad Phone:
223
Unnamed Pipes: “pipe()”
dup2( fd[WRITE], 1); /* Duplicate used end to stdout */ close( fd[WRITE] ); /* Close original used end */ execlp( argv[1], argvp[1], NULL ); /* Execute writer program */ perror( “connect” ); /* Should never execute */ } else /* Child, reader */ { close( fd[WRITE] ); /* Close unused end */ dup2( fd[READ], 0 ); /* Duplicate used end to stdin */ close( fd[READ] ); /* Close original used end */ execlp( argv[2], argv[2], NULL ); /* Execute reader program */ Prof. Andrzej (AJ) Bieszczad Phone:
224
Unnamed Pipes: “pipe()”
$ who > execute “who” by itself. gglass ttyp Feb :15 (xyplex_3) $ connect who wc > pipe “who” through “wc”. …1 line, 6 words, 57 chars. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
225
Named Pipes - Named pipes, often referred to as FIFOs( first in, first out ), are less restricted than unnamed pipes and offer the following advantages: They have a name that exists in the file system. They may be used by unrelated processes. They exist until explicitly deleted. - Unfortunately, they are only supported by System V. named pipes have a larger buffer capacity, typically about 40K. - Named pipes exist as special files in the file system and may be created in one of two ways: by using the UNIX mknod utility by using the “mknod()” system call Prof. Andrzej (AJ) Bieszczad Phone:
226
Named Pipes - To create a named pipe using mknod, use the “p” option. The mode of the named pipe may be set using chmod, allowing others to access the pipe that you create. Here’s an example of this procedure: $ mknod myPipe p > create pipe. $ chmod ug+rw myPipe ---> update permissions. $ ls -lg myPipe > examine attributes. prw-rw glass cs Feb :38 myPipe $ _ Prof. Andrzej (AJ) Bieszczad Phone:
227
Named Pipes - To create a named pipe using “mknod()”,
specify “S_IFIFO” as the file mode. The mode of the pipe can then be changed using “chmod()”. - C code that creates a name pipe with read and write permissions for the owner and group: mknod(“myPipe”, SIFIFO, 0); /* Create a named pipe */ chmod(“myPipe”, 0660); /* Modify its permission flags */ - Once a named pipe is opened using “open()”, “write()” adds data at the start of the FIFO queue, and “read()” removes data from the end of the FIFO queue. Prof. Andrzej (AJ) Bieszczad Phone:
228
Named Pipes - When a process has finished using a named pipe,
it should close it using “close()”, and - when a named pipe is no longer needed, it should be removed from the file system using “unlink()”. - Like an unnamed pipe, a named pipe is intended only for use as a unidirectional link. - Writer processes should open a named pipe for writing only, and reader processes should open a pipe for reading only. Although a process can open a named pipe for both reading and writing, this usage doesn’t have much practical application. Prof. Andrzej (AJ) Bieszczad Phone:
229
Named Pipes - an example program that uses named pipes,
here are a couple of special rules concerning their use: If a process tries to open a named pipe for reading only and no process currently has it open for writing, the reader will wait until a process opens it for writing, unless O_NONBLOCK or O_NDELAY is set, in which case “open()” succeeds immediately. If a process tries to open a named pipe for writing only and no process currently has it open for reading, the writer will wait until a process opens it for reading, “open()” fails immediately. Named pipes will not work across a network. Prof. Andrzej (AJ) Bieszczad Phone:
230
Named Pipes - The next examples uses two programs, “reader” and “writer”, to demonstrate the use of named pipes, A single reader process that creates a named pipe called “aPipe” is executed. It then reads and displays NULL-terminated lines from the pipe until the pipe is closed by all of the writing processes. One or more writer processes are executed, each of which opens the named pipe called “aPipe” and sends three messages to it. If the pipe does not exist when a writer tries to open it, the writer retries every second until it succeeds. When all of a writer’s messages are sent, the writer closes the pipe and exits. Prof. Andrzej (AJ) Bieszczad Phone:
231
Named Pipes - Sample Output
$ reader & writer & writer & > start 1 reader, 2 writers. [1] > reader process. [2] > first writer process. [3] > second writer process. Hello from PID 4699 Hello from PID 4700 [2] Done writer > first writer exists. [3] Done writer > second writer exists. [4] Done reader > reader exists. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
232
Named Pipes - Reader Program #include <stdio.h>
#include <sys/types.h> #include <sys/stat.h> /* For SIFIFO */ #include <fcntl.h> /*********************************************/ main() { int fd; char str[100]; unlink(“aPipe”); /* Remove named pipe if it already exists */ mknod(“aPipe”, S_IFIFO, 0); /* Create name pipe */ chmod(“aPipe”, 0660); /* Change its permissions */ fd = open(“aPipe”, O_RDONLY); /* Open it for reading */ while(readLine(fd, str) ); /* Display received messages */ printf(“%s\n”, str); close(fd); /* Close pipe */ } Prof. Andrzej (AJ) Bieszczad Phone:
233
Named Pipes /**********************************************/
readLine( fd, str ) int fd; char* str; /* Read s single NULL-terminated line into str from fd */ /* Return 0 when the end of input is reached and 1 otherwise */ { int n; do /* Read characters until NULL or end of input */ n = read( fd, str, 1); /* Read one character */ } while ( n>0 && *str++ != NULL ); return ( n> 0 ); /* Return false if end of input */ Prof. Andrzej (AJ) Bieszczad Phone:
234
Named Pipes - Writer Program #include <stdio.h>
#include <fcntl.h> /**************************************************/ main() { int fd, messageLen, i; char message[100]; /* Prepare message */ sprintf( message, “Hello from PID %d”, getpid() ); messageLen = strlen( message ) +1; do /* Keep trying to open the file until successful */ fd = open( “aPipe”, O_WRONLY ); /*Open named pipe for writing */ if ( fd == -1 ) sleep(1); /* Try again in 1 second */ } while ( fd == -1 ); Prof. Andrzej (AJ) Bieszczad Phone:
235
Named Pipes for ( i=1; i<=3; i++) /* Send three messages */ {
write( fd, message, messageLen ); /* Write message down pipe */ sleep(3); /* Pause a while */ } close(fd); /* Close pipe descriptor */ Prof. Andrzej (AJ) Bieszczad Phone:
236
Sockets - Sockets are the traditional UNIX interprocess communication mechanism that allows processes to talk to each other, even if they’re on different machines. - Common uses of sockets include: transferring data from one machine to another machine transferring voice/video from one machine to another machine transferring files from one machine to another machine printing a file on one machine from another machine - Process communication via sockets is based on the client-server model: (SMTP, IMAP, POP), Web (HTTP), file transfer (FTP, scp), remote login (telnet, ssh), network management (SNMP), network operation (DNS, BGP, OSPF), VoIP (SIP), etc., etc. One process, known as a server process, creates a socket whose name is known by other client processes. Prof. Andrzej (AJ) Bieszczad Phone:
237
Sockets - These client processes can talk to the server process via a connection to its named socket. 1) A client process first creates an unnamed socket and then requests that it be connected to the server’s named socket. 2) A successful connection returns one file descriptor to the client and one to the server, both of which may be used for reading and writing. - Note that unlike pipes, socket connections are bidirectional. Prof. Andrzej (AJ) Bieszczad Phone:
238
Sockets 1) Server creates a named socket. Server “Name”
2) Client creates an unnamed socket and request a connection. Server “Name” Client Prof. Andrzej (AJ) Bieszczad Phone:
239
Sockets 3) Client makes a connection.
Server retains original named socket. Server “Name” Client Completed connection Prof. Andrzej (AJ) Bieszczad Phone:
240
Sockets - Once a socket connection is made, it’s quite common for the server process to fork a child process to converse with the client while the original parent process continues to accept other client connections. the different kinds of sockets how a server creates a named socket and waits for connections how a client creates an unnamed socket and requests a connection from a server how a server and client communicate after a socket connection is made how a socket connection is closed how a server can create a child process to converse with the client Prof. Andrzej (AJ) Bieszczad Phone:
241
Different Kinds of Sockets
- The various kinds of sockets may be classified according to three attributes: the domain the type the protocol - Domains The domain of a socket indicates where the server and client sockets may reside; some of the domains that are currently supported included: AF_UNIX ( the clients and server must be on the same machine ) AF_INET ( the clients and server may be anywhere on the Internet) AF_NS ( the clients and server may be on a XEROX network system) Prof. Andrzej (AJ) Bieszczad Phone:
242
Different Kinds of Sockets
- Domains The domain of a socket indicates where the server and client sockets may reside; some of the domains that are currently supported included: AF_UNIX ( the clients and server must be on the same machine ) AF_INET ( the clients and server may be anywhere on the Internet) AF_NS ( the clients and server may be on a XEROX network system) - AF stands for “address family”. a similar set of constants that begin with PF, which stands for “protocol family”( e.g. PF_UNIX, PF_INET, etc. ) Prof. Andrzej (AJ) Bieszczad Phone:
243
Different Kinds of Sockets
- Types The type of a socket determines the type of communication that can exist between the client and server; the two main types that are currently supported are: SOCK_STREAM: sequenced, reliable, based on a two-way connection, streams of bytes of variable length SOCK_DGRAM: like telegrams; connectionless, unreliable, fixed-length messages SOCK_SEQPACKET: packets of bytes of a fixed length SOCK_RAW: provides access to internal network protocols and interfaces. Prof. Andrzej (AJ) Bieszczad Phone:
244
Different Kinds of Sockets
- Protocols The protocol value specifies the low-level means by which the socket type is implemented. System calls that expect a protocol parameter accept zero as meaning “the correct protocol”; Most systems support protocols other than zero only as an optional extra, so we will use the default protocol in all of the examples. Prof. Andrzej (AJ) Bieszczad Phone:
245
Writing Socket Programs
- Any program that uses sockets must include the header files “/usr/include/sys/types.h” and “/usr/include/sys/socket.h”. - Additional header files must be included based on the socket domain that you wish to use. The most commonly used domains are: Domain Additional header files AF_UNIX /usr/include/sys/un.h AF_INET /usr/include/netinet/in.h /usr/include/arpa/inet.h /usr/include/netdb.h Prof. Andrzej (AJ) Bieszczad Phone:
246
Different Kinds of Sockets
- description of the socket-oriented system calls around a small client- server example that uses AF_UNIX sockets. - The AF_UNIX examples is made up of two programs: 1) “chef”, the server, which creates a named socket called “recipe” and writes the recipe to any clients who request it. The recipe is a collection of NULL-terminated strings of variable length. 2) “cook”, the client, which connects to the named socket called “recipe” and reads the recipe from the server. It displays the recipe to standard output as it reads it and then terminates. Prof. Andrzej (AJ) Bieszczad Phone:
247
Different Kinds of Sockets
- The “chef” server process runs in the background. Any client “cook” processes that connect to the server cause the server to fork a duplicate server to handle the recipe transfer, allowing the original server to accept other incoming connections. - Here’s some sample output from the “chef” and “cook” example: $ chef & > run the server in the background. [1] 5684 $ cook > run a client to display the recipe. spam, spam, spam, spam, spam, and spam. $ kill % > kill the server. [1] Terminated chef $ _ Prof. Andrzej (AJ) Bieszczad Phone:
248
“Chef” and “Cook” Listing
- “Chef” Server #include <stdio.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/un.h> /* For AFUNIX sockets */ #define DEFAULT_PROTOCOL 0 /**************************************************/ main() { int serverFd, clientFd, serverLen, clientLen; struct sockaddr_un serverUNIXAddress; /* Server address */ struct sockaddr_un clientUNIXAddress; /* Client address */ struct sockaddr* serverSockAddrPtr; /* Ptr to server address */ struct sockaddr* clientSockAddrptr; /* Ptr to client address */ Prof. Andrzej (AJ) Bieszczad Phone:
249
/* Ignore death-of-child signals to prevent zombies */
signal(SIGCHLD, SIG_IGN); serverSockAddrPtr = (struct sockaddr*) &serverUNIXAddress; serverLen = sizeof( serverUNIXAddress ); clientSockAddrPtr = (struct sockaddr*) &clientUNIXAddress; clientLen = sizeof( clientUNIXAddress ); /* Create a UNIX socket, bidirectional, default protocol */ serverFd = socket( AF_UNIX, SOCK_STREAM, DEFAULT_PROTOCOL ); serverUNIXAddress.sun_family = AF_UNIX; /*Set domain type */ strcpy( serverUNIXAddress.sun_path, “recipe”); /* Set name */ unlink(“recipe”); /* Remove file if it already exists */ bind(serverFd, serverSockAddrPtr, serverLen); /* Create file */ listen( serverFd, 5 ); /* Maximum pending connection length */ Prof. Andrzej (AJ) Bieszczad Phone:
250
while(1) /* Loop forever */ { /* Accept a client connection */
clientFd = accept( serverFd, clientSockAddrPtr, &clientLen); if ( fork() == 0 ) /* Create child to send receipe */ writeRecipe( clientFd ) /* Send the recipe */ close(clientFd); /* Close the socket */ exit( /* EXIT_SUCCESS */ 0 ); /* Terminate */ } else close(clientFd); /* Close the client descriptor */ Prof. Andrzej (AJ) Bieszczad Phone:
251
/*************************************************/ writeRecipe( fd )
int fd; { static char* line1 = “spam, spam, spam, spam,”; static char* line2 = “spam, and spam.”; write(fd, line1, strlen(line1)+1); /* Write first line */ write(fd, line2, strlen(line2)+1); /* Write second line */ } Prof. Andrzej (AJ) Bieszczad Phone:
252
“Chef” and “Cook” Listing
- “Cook” Client #include <stdio.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/un.h> /* For AFUNIX sockets */ #define DEFAULT_PROTOCOL 0 /*************************************************/ main() { int clientFd, serverLen, result; struct sockaddr_un serverUNIXAddress; struct sockaddr* serverSockAddrPtr; serverSockAddrPtr = ( struct sockaddr*) &serverUNIXAddress; serverLen = sizeof( serverUNIXAddress ); Prof. Andrzej (AJ) Bieszczad Phone:
253
serverLen = sizeof( serverUNIXAddress );
/* Create a UNIX socket, bidirectional, default protocol */ clientFd = socket( AF_UNIX, SOCK_STREAM, DEFAULT_PROTOCOL); serverUNIXAddress.sun_family = AF_UNIX; /* Server domain */ strcpy( serverUNIXAddress.sun_path, “recipe” ); /* Server name */ do /* Loop until a connection is made with the server */ { result = connect( clientFd, serverSockAddrPtr, serverLen); if ( result == -1 ) sleep(1); /* Wait and then try again */ } while ( result == -1 ); readRecipe( clientFd ); /* Read the recipe */ close( clientFd ); /* Close the socket */ exit( /* EXIT_SUCCESS */ 0 ); /* Done */ } Prof. Andrzej (AJ) Bieszczad Phone:
254
/************************************************/ readRecipe( fd )
int fd; { char str[200]; while ( readLine(fd, str) ) /* Read lines until end of input */ printf(“%s\n”, str); /* Echo line from socket */ } readLine(fd, str) int fd; char* str; /* Read a single NULL-terminated line */ int n; do /* Read characters until NULL or end of input */ n = read( fd, str, 1 ); /* Read one character */ } while ( n > 0 && *str++ != NULL ); return( n>0 ); /* Return false if end of input */ Prof. Andrzej (AJ) Bieszczad Phone:
255
Analyzing the Source Code
an overview of a server - creating a server socket - naming a server socket - specifying the maximum number of pending connections to a sever socket - accepting connections on a server socket - serving a client an overview of a client - creating a client socket - connecting a client socket to the server socket - communicating via sockets Prof. Andrzej (AJ) Bieszczad Phone:
256
The Server - A server is the process that’s responsible for creating a names socket and accepting connections to it. To accomplish this task, it must use the following system calls in the order in which they presented : Name Meaning Socket creates an unnamed socket bind gives the socket a name listen specifies the maximum number of pending connections accept accepts a socket connection from a client Prof. Andrzej (AJ) Bieszczad Phone:
257
Creating a Server: socket()
- A process may create a socket by using “socket()”, which works like this: System Call : int socket( int domain, int type, int protocol ) “socket()” creates an unnamed socket of the specified domain, type, and protocol. The legal values of these parameters were described earlier in this section. If “socket()” is successful, it returns a file descriptor associated with the newly created socket; otherwise, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
258
Creating a Server: socket()
- The “chef” server creates its unnamed socket on line 30: 30 serverFd=socket(AF_UNIX, SOCK_STREAN,DEFAULT_PROTOCOL); Naming a Socket : bind() Once the server has created an unnamed socket, it must be bind it to a name by using “bind()”, which works like this: System Call : int bind( int fd, const struct sockaddr* address, size_t addressLen ) “bind()” associates the unnamed socket represented by file descriptor fd with the socket address stored in address. addressLen must contain the length of the address structure. The type and value of the incoming address depend on the socket domain. Prof. Andrzej (AJ) Bieszczad Phone:
259
Creating a Server: socket()
- If the socket is in the AF_UNIX domain, a pointer to a “sock-addr_un” structure must be cast to a “sockaddr*” and passed in as address. This structure has two fields that should be set as follows: FIELD ASSIGN THE VALUE sun_family AF_UNIX sun_path the full UNIX pathname of the socket( absolute or relative ) If the named AF_UNIX socket already exists, an error occurs, so it’s a good idea to “unlink()” a name before attempting to bind to it. If the socket is in the AF_INET domain, a pointer to a “socket-addr_in” structure must be cast to a “sockaddr” and passed in as address. Prof. Andrzej (AJ) Bieszczad Phone:
260
Creating a Server: socket()
- This structure has four fields that should be set as follows: FIELD ASSIGN THE VALUE sin_family AF_INET sin_port the port number of the Internet socket sin_addr a structure of type “in_addr” that holds the Internet address sin_zero leave empty - For more information about Internet ports and addresses, please consult the Internet-specific part of this section. If “bind()” succeeds, it returns a value of 0; otherwise, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
261
Creating a Server: socket()
- The “chef” server assigns the “sockaddr_un” fields and performs a “bind()” on lines 31 to 34: serverUNIXAddress.sun_family = AF_UNIX; /* Set domain type */ strcpy( serverUNIXAddress.sun_path, “recipe”); /* Set name */ unlink(“recipe”); /* Remove file if it already exists */ bind( serverFd, serverSockAddrPtr, serverLen ) /* Create file */ Prof. Andrzej (AJ) Bieszczad Phone:
262
Creating a Socket Queue: listen()
- When a sever process is servicing a client connection, it’s always possible that another client will also attempt a connection. The “listen()” system call allows a process to specify the number of pending connections that may be queued, and it works like this: System Call: int listen( int fd, int queueLength ) “listen()” allows you to specify the maximum number of pending connections on a socket. If a client attempts a connection to a socket whose queue is full, it is denied. Prof. Andrzej (AJ) Bieszczad Phone:
263
Creating a Socket Queue: listen()
- The “chef” server listens to its named socket, where it specifies a maximum queue length of five: listen( serverFd, 5 ); /* Maximum pending connection length */ Prof. Andrzej (AJ) Bieszczad Phone:
264
Accepting a Client : accept()
- Once a socket has been created, named, and its queue size has been specified, the final step is to accept connection requests from clients. - To do so, the server must use “accept()”, which works as follows: System Call : int accept( int fd, struct sockaddr* address, int * addressLen ) “accept()” listens to the named server socket referenced by fd and waits until a connection request from a client is received. When this event occurs, “accept()” creates an unnamed socket with the same attributes as those of the original named server socket, connects it to the client’s socket. and returns a new file descriptor that may be used for communication with the client. Prof. Andrzej (AJ) Bieszczad Phone:
265
Accepting a Client : accept()
- The original named server socket may be used to accept more connections. The address structure is filled with the address of the client and is normally only used in conjuction with Internet connections. The addressLen field should be initially set to point to an integer containing the size of the structure pointed to by address. When a connection is made, the integer that it points to is set to the actual size, in bytes, of the resulting address. If “accept()” succeeds, it returns a new file descriptor that may be used to talk with the client; otherwise, it returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
266
Accepting a Client : accept()
- The “chef” sever accepts a connection clientFd = accept( serverFd, clientSockAddrPtr, &clientLen ); - Serving a client when a client connection succeeds, the most common sequence of events is as follows: The server process forks. The parent process closes the newly formed client file descriptor and loops back to “accept()”, ready to service new connection requests from clients. The child process talks to the client using “read()” and “write()”. When the conversation is complete, the child process closes the client file descriptor and exits. Prof. Andrzej (AJ) Bieszczad Phone:
267
Accepting a Client : accept()
- The “chef” sever process while( 1 ) /* Loop forever */ { /* Accept a client connection */ clientFd = accept( serverFd, clientSockAddrPtr, &clientLen ); if ( fork()==0 ) /* Create child to send receipe */ writeRecipe( clientFd ); /* Send the recipe */ close(clientFd); /* Close the socket */ exit( /* EXIT_SUCCESS */ 0 ); /* Terminate */ } else close(clientFd); /* Close the client descriptor */ Prof. Andrzej (AJ) Bieszczad Phone:
268
Accepting a Client : accept()
- The Client A client is a process that’s responsible for creating an unnamed socket and then attaching it to a named server socket. To accomplish this task, it muse use the following system calls in order in which they are presented: Name Meaning socket creates an unnamed socket connect attaches an unnamed client socket to a named server socket Prof. Andrzej (AJ) Bieszczad Phone:
269
Accepting a Client : accept()
- a client uses “socket()” to create an unnamed socket is the same as the way that the server uses it. The domain, type, and protocol of the client socket must match those of the targeted server socket. The “cook” client process creates its unnamed socket clientFd = socket( AF_UNIX, SOCK_STREAM, DEFAULT_PROTOCOL); Prof. Andrzej (AJ) Bieszczad Phone:
270
Making the Connection : connect()
To connect to a sever’s socket, a client process must fill a structure with the address of the server’s socket and then use “connect”, which works like this: System Call: int connect(int fd, struct sockaddr* address,int addressLen) “connect()” attempts to connect to a sever socket whose address is contained within a structure pointed to by address. If “connect()” successful, fd may be used to communicate with the server’s socket. The type of structure to which address points must follow the same rules as those stated in the description of “bind()”: Prof. Andrzej (AJ) Bieszczad Phone:
271
Making the Connection : connect()
If the socket is in the AF_UNIX domain, a pointer to a “sock-addr_un” structure must be cast to a(“sockaddr*”) and passed in as address. If the socket is in the AF_INET domain, a pointer t a “sock-addr_un” structure must be cast to a (“sockaddr*”) and passed in as address. addressLen must be equal to the size of the address structure. For examples of Internet clients, see the next examples of a socket and the Internet-shell program at the end of this chapter. If the connection is made, “connect()” returns a value of 0. If the server socket doesn’t exist or its pending queue is currently filled, “connect()” returns a value of -1. Prof. Andrzej (AJ) Bieszczad Phone:
272
Making the Connection : connect()
- The “cook” client process calls “connect()” until a successful connection is made. do /* Loop until a connection is made with the server */ { result = connect( clientFd, serverSockAddrPtr, serverLen ); if ( result == -1 ) sleep(1); /* Wait and then try again */ } while ( result == -1 ); Prof. Andrzej (AJ) Bieszczad Phone:
273
Communicating Via Sockets
- Once the server socket and client socket have connected, their file descriptors may be used by “write()” and read(). writeRecipe(fd) int fd; { static char* line1=“spam, spam, spam, spam,”; static char* line2=“spam, and spam.”; write(fd, line1, strlen(line1)+1); /* Write first line */ write(fd, line2, strlen(line2)+1); /* Write second line */ } Prof. Andrzej (AJ) Bieszczad Phone:
274
Communicating Via Sockets
- The client uses “read()” readLine( fd, str ) int fd; char* str; /* Read a single NULL-terminated line */ { int n; do /* Read characters until NULL or end of input */ n=read(fd, str, 1); /* Read one character */ } while ( n>0 && *str++ != NULL ); return ( n > 0 ); /* Return false if end of input */ - The server and client should be careful to close their socket file descriptors when they are no longer needed. Prof. Andrzej (AJ) Bieszczad Phone:
275
Internet Sockets - The AF_UNIX sockets that you’ve seen so far are OK for learning about sockets, but they aren’t where the action is. - An Internet socket is specified by two values: a 32-bit IP address, which specifies a single unique Internet host, and a 16-bit port number, which specifies a particular port on the host. - These specifications mean that an Internet client must know not only the IP address of the server, but also the server’s port number. - Several standard port numbers are reserved for system use. For example, port 13 is always served by a process that echoes the host’s time of day to any client that’s interested. Prof. Andrzej (AJ) Bieszczad Phone:
276
Internet Sockets - The first example, Three kinds of Internet addresses: • If you enter “s”, it automatically means the local host. • If you enter something that starts with a digit, it’s assumed to be an A.B.C.D-format IP address and is converted into a 32-bit IP address by software. • If you enter a string, it’s assumed to be a symbolic host name and is converted into a 32-bit IP address by software. Prof. Andrzej (AJ) Bieszczad Phone:
277
Sample Output $ inettime ---> run the program.
Host name ( q=quit, s=self ) : s > what’s my time? Self host name is csservr2 Internet Address = The time on the target port is Fri Mar 27 17:03: Host name ( q=quit, s=self ): wotan ---> what’s the time on “wotan”? Internet Address= The time on the target port is Fri Mar 27 17:03: Host name ( q=quit, s=self ): > what’s the time at ---> ddn.nic.mil. The time on the target port is Fri Mar 27 18:02: Host name ( q=quit, s=self ): q > quit program. $ _ Prof. Andrzej (AJ) Bieszczad Phone:
278
#include <stdio.h> #include <signal.h>
#include <ctype.h> #include <sys/types.h> #include <sys/socket.h> #include <netinet/in.h> /* For AFINET sockets */ #include <arpa/inet.h> #include <netdb.h> #define DAYTIME_PORT /* Standard port 0 */ #define DEFAULT_PROTOCOL 0 unsigned long promptForINETAddress(); unsigned long nameToAddr(); /**************************************************/ main() { int clientFd; /* Client socket file descriptor */ Prof. Andrzej (AJ) Bieszczad Phone:
279
int serverLen; /* Length of server address structure */
int result; /* From connect() call */ struct sockaddr_in severINETAddress; /* Server address */ struct sockaddr* serverSockAddrPtr; /* Pointer to address */ unsigned long inetAddress; /* 32-bit IP address */ /* Set the two server variables */ serverSockAddrPtr = ( struct sockaddr* ) &serverINETAddress; serverLen = sizeof ( serverINETAddress ); /* Length of address */ while( 1 ) /* Loop until break */ { inetAddress = promptForINETAddress(); /* Get 32-bit IP */ if ( inetAddress == 0 ) break; /* Done */ /* Start by zeroing out the entire address structure */ bzero( (char*)&serverINETAddress, sizeof( serverINETAddress)); Prof. Andrzej (AJ) Bieszczad Phone:
280
serverINETAddress.sin_family = AF_INET; /* Use Internet */
serverINETAddress.sin_addr.s_addr=inetAddress; /* IP */ serverINETAddress.sin_port = htons(DAYTIME_PORT); /* Now create the client socket */ clientFd = socket( AF_INET, SOCK_STREAM, DEFAULT_PROTOCOL); do /* Loop until a connection is made with the server */ { result = connect( clientFd, severSockAddrPtr, serverLen ); if ( result== -1 ) sleep(1); /* Try again in 1 second */ } while ( result==-1 ); readTime( clientFd ); /* Read the time from the server */ close(clientFd); /* Close the socket */ } exit( /* EXIT_SUCCESS */ 0 ); Prof. Andrzej (AJ) Bieszczad Phone:
281
/************************************************/
unsigned long promptForINETAddress() { char hostName[100]; /* Name from user: numeric or symbolic */ unsigned long inetAddress; /* 32-bit IP format */ /* Loop until “quit” or a legal name is entered */ /* If quit, return 0; else return host’s IP address */ do printf(“Host name( q=quit, s=self) : “); scanf(“%s”, hostName); /* Get name from keyboard */ if ( strcmp(hostName, “q”) == 0 ) return (0); /* Quit */ inetAddress = nameToAddr(hostName); /* Convert to IP */ if ( inetAddress == 0 ) printf(“Host name not found \n”); } while( inetAddress==0 ); } Prof. Andrzej (AJ) Bieszczad Phone:
282
/************************************************/
unsigned long nameToAddr(name) char* name; { char hostName[100]; struct hostent* hostStruct; struct in_addr* hostNode; /* Convert name into a 32-bit IP address */ /* If name begins with a digit, assume it’s a valid numeric */ /* Internet address of the form A.B.C.D and convert directly */ if ( isdigit(name[0]) ) return ( inet_addr(name) ); if ( strcmp(name, “s” ) ==0 ) /* Get host name from database */ gethostname(hostName, 100); printf(“Self host name is %s\n”, hostName); } Prof. Andrzej (AJ) Bieszczad Phone:
283
else /* Assume name is a valid symbolic host name */
strcpy( hostName, name); /* Now obtain address information from database */ hostStruct = gethostbyname(hostName); if ( hostStruct == NULL ) return(0); /* Not Found */ /* Extract the IP Address from the hostent structure */ hostNode = ( struct in_addr* ) hostStruct->h_addr; /* Display a readable version for fun */ printf(“Internet Address=%s\n”, inet_ntoa(*hostNode)); return ( hostNode->s_addr ); /* Return IP address */ } Prof. Andrzej (AJ) Bieszczad Phone:
284
char str[200]; /* Line buffer */
readTime( fd ) int fd; { char str[200]; /* Line buffer */ printf(“The time on the target port is “); while( readLine(fd,str) ) /* Read lines until end of input */ printf(“%s\n”, str); /* Echo line from server to user */ } /***************************************************/ readLine( fd, str ) char* str; /* Read a single line terminated by a new line */ int n; Prof. Andrzej (AJ) Bieszczad Phone:
285
do /* Read characters until NULL or end of input */ {
n = read(fd, str, 1); /* Read one character */ } while( n>0 && *str++ != ‘\n’ ) ; return ( n>0 ); /* Return false if end of input */ Prof. Andrzej (AJ) Bieszczad Phone:
286
Analyzing the Source Code
- Internet Clients The procedure for creating an Internet client is the same as that for creating an AF_UNIX client, Internet socket address structure is of the type “struct sockaddr_in” and has four fields: “sin_family”, the domain of the socket, which should be set to AF_INET “sin_port”, the port number, which is 13 in this case “sin_port”, the 32-bit IP number “sin_zero”, which is padding and is not set Prof. Andrzej (AJ) Bieszczad Phone:
287
Analyzing the Source Code
- “promptForINETAddress()” gets the host’s name from the user and then invokes “nameToAddr()” to convert into an IP address. If the user enters a string starting with a digit, “inet_addr()” is invoked to perform the conversion. Library Call: in_addr_t inet_addr(const char* string) “inet_addr()” returns the 32-bit IP address that corresponds to the A.B.C.D-format string. The IP address is in network-byte order. Prof. Andrzej (AJ) Bieszczad Phone:
288
Analyzing the Source Code
- Network-byte order a host-neutral ordering of bytes in the IP address. This ordering is necessary, since regular byte ordering can differ from machine to machine, which would make IP addresses nonportable. If the string doesn’t start with a digit, the next step is to see if it’s “s”, which signifies the local host. The name of the local host is obtained by “gethostname()”, which works as follows: System Call: int gethostname( char* name, int nameLen ) “gethostname()” sets the character array pointed to by name of length nameLen to a NULL-terminated string equal to the local host’s name. Prof. Andrzej (AJ) Bieszczad Phone:
289
Analyzing the Source Code
- Once the symbolic name of the host is determined, the next stage is to look it up in the network-host file, “/etc/hosts”. This task is performed by “gethostbyname()”, which works like this: Library Call : struct hostent* gethostbyname( const char* name ) “gethostbyname()” searches the “/etc/hosts” file and returns a pointer to a hostent structure that describes the file entry associated with the string name. If name is not found in the “/etc/hosts” file, NULL is returned. Prof. Andrzej (AJ) Bieszczad Phone:
290
Analyzing the Source Code
- The “hostent” structure has several fields, but the only one we’re interested in is a field of type (“struct in_addr*) called “h_addr.” This field contains the host’s associated IP number in a subfield called “s_addr.” Before returning this IP number, the program displays a string description of the IP address by calling “inetntoa()” Library Call: char* inet_ntoa( struct in_addr address ) “inet_ntoa()” takes a structure of type “in_addr” as its argument and returns a pointer to a string that describes the address in the format A.B.C.D. Prof. Andrzej (AJ) Bieszczad Phone:
291
Analyzing the Source Code
- Once the IP address “inetAddress” has been determined, the client’s socket address fields are filled by lines 37 to 40. 37 bzero((char*)&serverINETAddress, sizeof(serverINETAddress)); 38 serverINETAddress.sin_family = AF_INET; /* Use Internet */ 39 serverINETAddress.sin_addr.s_addr = inetAddress; /* IP */ 40 serverINETAddress.sin_port = htons(DAYTIME_PORT); “bzero()” clears the socket address structure’s contents before its fields are assigned: Library Call: void bzero(void* buffer, size_t length) “bzero()” fills the array buffer or size length with zeroes( ASCII NULL). Prof. Andrzej (AJ) Bieszczad Phone:
292
Analyzing the Source Code
- The “bzero()” call had its origins in the Berkeley version of UNIX. System V’s equivalent is “memset()”: Library Call: void memset(void* buffer, int value, size_t length) “memset()” fills the array buffer of size length with the value of value. Like the IP address, the port number is also converted to a network- byte ordering by “htons()”, which works like this: Prof. Andrzej (AJ) Bieszczad Phone:
293
Analyzing the Source Code
- Like the IP address, the port number is also converted to a network- byte ordering by “htons()”, which works like this: Library Call: in_addr_t htonl(in_addr_t hostLong) in_port_t htons(in_port_t hostShort) in_addr_t ntohl(in_addr_t networkLong) in_port_t ntohs(in_port_t networkShort) Each of these functions performs a conversion between a host-format number and a network-format number. For example, “htonl()” returns the network-format equivalent of the host-format unsigned long hostLong, and “ntohs()” returns the host-format equivalent of the network- format unsigned short networkShort. Prof. Andrzej (AJ) Bieszczad Phone:
294
Analyzing the Source Code
- The final step is to create the client socket and attempt the connection. The code for this task is almost the same as that for AF_UNIX sockets: 42 clientFd = socket( AF_INET, SOCK_STREAM, DEFAULT_PROTOCOL); 43 do /* Loop until a connection is made with the server */ 44 { result = connect( clientFd, serverSockAddrPtr, serverLen ); if ( result == -1 ) sleep(1); /* Try again in 1 second */ 47 } 48 while ( result == - 1 ); Prof. Andrzej (AJ) Bieszczad Phone:
295
Internet Servers - Constructing an Internet server is actually pretty easy. The “sin_family”, “sin_port”, and “sin_zero” fields of the socket_address structure should be filled in as they were in the client example. The only difference is that the “s_addr” field should be set to the network-byte-ordered value of the constant “INADDR_ANY”, which means “accept any incoming client requests.” The following example of how to create a server socket address is a slightly modified version of some code taken from the Internet shell program that ends this chapter: int serverFd; /* Server socket */ struct sockaddr_in serverINETAddress; /* Server Internet address */ struct sockaddr* serverSockAddrPtr; /* Pointer to server address */ struct sockaddr_in clientINETAddress; /* Client Internet address */ Prof. Andrzej (AJ) Bieszczad Phone:
296
Internet Servers struct scokaddr* clientSockAddrPtr; /* Pointer to client address */ int port=13; /* Set to the port that you wish to serve */ int serverLen; /* Length of address structure */ serverFd = socket(AF_INET, SOCK_STREAM, DEFAULT_PROTOCOL); /* Create */ serverLen = sizeof(serverINETAddress); /* Length of structure */ bzero((char*) &serverINETAddress, serverLen); /* Clear structure */ serverINETAddress.sin_family = AF_INET; /* Internet domain */ serverINETAddress.sin_addr.s_addr = htonl(INADDR_ANY); /* Accept all */ serverINETAddress.sin_port = htons(port); /* Server port number */ Prof. Andrzej (AJ) Bieszczad Phone:
297
Internet Servers - When the address is created, the socket is bound to the address, and its queue size is specified in the usual way: serverSockAddrPtr = (struct sockaddr*) &serverINETAddress; bind( serverFd, serverSockAddrPtr, serverLen); listen(serverFd, 5); The final step is to accept client connections. When a successful connection is made, the client socket address is filled with the client’s IP address and a new file descriptor is returned: clientLen = sizeof( clientINETAddress ); clientSockAddrPtr = ( struct sockaddr* ) clientINETAddress; clientFd = accept(serverFd, clientSockAddrPtr, &clientLen); - an Internet server’s code is very similar to that of an AF_UNIX server. Prof. Andrzej (AJ) Bieszczad Phone:
298
Shared Memory - Sharing a segment of memory is a straightforward and intuitive method of allowing two processes on the same machine to share data. Accessing a shared memory segment is the fasted form of IPC, since no data has to copied or sent anywhere else. - some of the common system calls used to allocate and use shared memory segments in System V-based versions of UNIX are: “shmget()”, which allocates a shared memory segment and returns the segment ID number “shmat()”, which attaches a shared memory segment to the virtual address space of the calling process “shmdt()”, which detaches an attached segment from the address space “shmctl()”, which allows you to modify attributes associated with the shared memory segment( e.g., access permissions ) Prof. Andrzej (AJ) Bieszczad Phone:
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.