| 
 | < Day Day Up > | 
 | 
As we’ve seen much of the C language’s networking API in Part I of this book, this chapter simply provides a summary of the API information to provide a pattern for future language chapters in this second part of the book.
The networking API for C provides a mixed set of functions for the development of client and server applications. The available functions are shown in Figure 9.1.
|  | 
| Function | Description | Server | Client | Stream | Dgram | Blocking | 
|---|---|---|---|---|---|---|
| socket | Create a new socket | • | • | • | • | |
| bind | Bind a name to a socket | • | • | • | • | |
| listen | Convert to a listening socket | • | • | |||
| connect | Connect to a peer socket | • | • | • | • | |
| accept | Accept a connection from a listening socket | • | • | • | ||
| send | Send data through a socket | • | • | • | • | • | 
| sendto | Send data through a socket to a named destination | • | • | • | • | |
| recv | Send data through a socket | • | • | • | • | • | 
| recvfrom | Send data through a socket to a named destination | • | • | • | • | |
| select | Notify upon socket event | • | • | • | • | • | 
| close | Close a socket | • | • | • | • | |
| shutdown | Close a socket with user control | • | • | • | • | |
| getsockopt | Get the value of a specified socket option | • | • | • | • | |
| setsockopt | Set the value of a specified socket option | • | • | • | • | |
| getpeername | Get peer information about a socket | • | • | • | • | |
| getsockname | Get local information about a socket | • | • | • | • | |
| gethostbyname | Get network host address | • | • | • | ||
| gethostbyaddr | Get network host name | • | • | • | ||
| inet_addr | Convert a string address to a 32-bit address | • | • | • | • | |
| inet_aton | Convert a string address to a 32-bit address | • | • | • | • | |
| inet_ntoa | Convert a 32-bit address to a string address | • | • | • | • | 
|  | 
For each of the functions shown in Figure 9.1, its classification is shown for a number of different attributes. Some functions are used by only server-side sockets, whereas others are used solely by client-side sockets (most are available to both). Another axis defines whether the function exists for stream sockets (TCP) or datagram sockets (UDP). Finally, some of the functions block until a satisfying event occurs, whereas others will never block.
Because much of this material has been discussed in Part I of this book, we present a light treatment of the Sockets API here to serve as a model for later chapters that discuss languages other than C.
A socket is necessary to be created as the first step of any socket-based application. The socket function provides the following prototype:
int socket( int domain, int type, int protocol );
The socket object is represented as a simple integer and is returned by the socket function. Three parameters must be passed to define the type of socket to be created. We’re interested primarily in stream (TCP) and datagram (UDP) sockets, but many other types of sockets may be created. In addition to stream and datagram, a raw socket is also illustrated by the following code snippets:
myStreamSocket = socket( AF_INET, SOCK_STREAM, 0 ); myDgramSocket = socket( AF_INET, SOCK_DGRAM, 0 ); myRawSocket = socket( AF_INET, SOCK_RAW, IPPROTO_RAW );
The AF_INET symbolic constant defines that we are using the IPv4 Internet protocol. After this, the second parameter (type) defines the semantics of communication. For stream communication (using TCP), we use the SOCK_STREAM type and for datagram communication (using UDP), we specify SOCK_DGRAM. The third parameter could define a particular protocol to use, but only the types exist for stream and datagram, so it’s left as zero.
When we’re finished with a socket, we must close it. The close prototype is defined as:
int close( sock );
After close is called, no further data may be received through the socket. Any data queued for transmission would be given some amount of time to be sent before the connection physically closes.
For socket communication over the Internet (domain AF_INET), the sockaddr_in structure is used for naming purposes.
  struct sockaddr_in {      int16_t sin_family;      uint16_t sin_port;      struct in_addr sin_addr;      char sin_zero[8];  }; struct in_addr {      uint32_t s_addr;  };  For Internet communication, we’ll use AF_INET solely for sin_family. Field sin_port defines our specified port number in network byte order. Therefore, we must use htons to load the port and ntohs to read it from this structure. Field sin_addr is, through s_addr, a 32-bit field that represents an IPv4 Internet address. Recall that IPv4 addresses are four-byte addresses. We’ll see quite often that the sin_addr is set to INADDR_ANY, which is the wildcard. When we’re accepting connections (server socket), this wildcard says we accept connections from any available interface on the host. For client sockets, this is commonly left blank. For a client, if we set sin_addr to the IP address of a local interface, this restricts outgoing connections to that interface.
Let’s now look at a quick example of addressing for both a client and a server. First, we’ll create the socket address (later to be bound to our server socket) that permits incoming connections on any interface and port 48000.
int servsock; struct sockaddr_in servaddr; servsock = socket( AF_INET, SOCK_STREAM, 0); memset( &servaddr, 0, sizeof(servaddr) ); servaddr.sin_family = AF_INET; servaddr.sin_port = htons( 48000 ); servaddr.sin_addr.s_addr = inet_addr( INADDR_ANY ); Next, we’ll create a socket address that permits a client socket to connect to our previously created server socket. int clisock; struct sockaddr_in servaddr; clisock = socket( AF_INET, SOCK_STREAM, 0); memset( &servaddr, 0, sizeof(servaddr) ); servaddr.sin_family = AF_INET; servaddr.sin_port = htons( 48000 ); servaddr.sin_addr.s_addr = inet_addr( "192.168.1.1" );
Note the similarities between these two code segments. The difference, as we’ll see later, is that the server uses the address to bind to itself as an advertisement. The client uses this information to define to whom it wants to connect.
In this section, we look at a number of other important socket control primitives.
The bind function provides a local naming capability to a socket. This can be used to name either client or server sockets, but is used most often in the server case. The bind function is provided by the following prototype:
int bind( int sock, struct sockaddr *addr, int addrLen );
The socket to be named is provided by the sock argument and the address structure previously defined is defined by addr. Note that the structure here differs from our address structure discussed previously. The bind function may be used with a variety of different protocols, but when using a socket created with AF_INET, the sockaddr_in structure must be used. Therefore, as shown in the following example, we cast our sockaddr_in structure as sockaddr.
err = bind( servsock, (struct sockaddr *)&servaddr, sizeof(servaddr));
Using our address structure created in our server example in the previous address section, we bind the name defined by servaddr to our server socket servsock.
Recall that a client application can also call bind in order to name the client socket. This isn’t used often, as the Sockets API will dynamically assign a port to us.
Before a server socket can accept incoming client connections, it must call the listen function to declare this willingness. The listen function is provided by the following function prototype:
int listen( int sock, int backlog );
The sock argument represents the previously created server socket and the backlog argument represents the number of outstanding client connections that may be queued. Within Linux, the backlog parameter (post 2.2 kernel version) represents the numbers of established connections waiting for by the Application layer protocol. Other operating systems may treat this differently.
The accept call is the final call made by servers to accept incoming client connections. Before accept can be called, the server socket must be created, a name must be bound to it, and listen must be called. The accept function returns a socket descriptor for a client connection, and is provided by the following function prototype:
int accept( int sock, struct sockaddr *addr, int *addrLen );
In practice, two examples of accept are commonly seen. The first represents the case in which we need to know who connected to us. This requires the creation of an address structure that will not be initialized.
struct sockaddr_in cliaddr; int cliLen; cliLen = sizeof( struct sockaddr_in ); clisock = accept( servsock, (struct sockaddr *)cliaddr, &cliLen );
The call to accept will block until a client connection is available. Upon return, the clisock return value will contain the value of the new client socket and cliaddr will represent the address for the client peer (host address and port number).
The alternate example is commonly found when the server application isn’t interested in the client information. This one typically appears as:
cliSock = accept( servsock, (struct sockaddr *)NULL, NULL );
In this case, NULL is passed for the address structure and length. The accept function will then ignore these parameters.
The connect function is used by client Sockets applications to connect to a server. Clients must have created a socket and then defined an address structure containing the host and port number to which they want to connect. The connect function is provided by the following function prototype:
int connect( int sock, (struct sockaddr *)servaddr, int addrLen );
The sock argument represents our client socket, created previously with the Sockets API function. The servaddr structure is the server peer to which we want to connect (as illustrated previously in the Socket Addresses section). Finally, we must pass in the length of our servaddr structure so that connect knows we’re passing in a sockaddr_in structure. The following code shows a complete example of connect:
int clisock; struct sockaddr_in servaddr; clisock = socket( AF_INET, SOCK_STREAM, 0); memset( &servaddr, 0, sizeof(servaddr) ); servaddr.sin_family = AF_INET; servaddr.sin_port = htons( 48000 ); servaddr.sin_addr.s_addr = inet_addr( "192.168.1.1" ); connect( clisock, (struct sockaddr_in *)&servaddr, sizeof(servaddr) );
The connect function blocks until either an error occurs, or the three-way handshake with the server completes. Any error is returned by the connect function.
A variety of API functions exists to read data from a socket or write data to a socket. Two of the API functions (recv, send) are used exclusively by sockets that are connected (such as stream sockets), whereas an alternative pair (recvfrom, sendto) is used exclusively by sockets that are unconnected (such as datagram sockets).
The send and recv functions are used to send a message to the peer socket endpoint and to receive a message from the peer socket endpoint. These functions have the following prototypes:
int send( int sock, const void *msg, int len, unsigned int flags ); int recv( int sock, void *buf, int len, unsigned int flags );
The send function takes as its first argument the socket descriptor from which to send the msg. The msg is defined as a (const void *) because the object referenced by msg will not be altered by the send function. The number of bytes to be sent in msg is contained by the len argument. Finally, a flags argument can be used to alter the behavior of the send call. An example of sending a string through a previously created stream socket is shown as:
strcpy( buf, "Hello\n"); send( sock, (void *)buf, strlen(buf), 0);
In this example, our character array is initialized by the strcpy function. This buffer is then sent through sock to the peer endpoint, with a length defined by the string length function, strlen. To illustrate flags usage, let’s look at one side effect of the send call. When send is called, it may block until all of the data contained within buf has been placed on the socket’s send queue. If not enough space is available to do this, the send function blocks until space is available. If we want to avoid this blocking behavior, and instead want the send call to simply return if sufficient space was available, we could set the MSG_DONTWAIT flag, such as:
send( sock, (void *)buf, strlen(buf), MSG_DONTWAIT);
The return value from send represents either an error (less than 0) or the number of bytes that were queued to be sent. Completion of the send function does not imply that the data was actually transmitted to the host, only queued on the socket’s send queue waiting to be transferred.
The recv function mirrors the send function in terms of argument list. Instead of sending the data pointed to be msg, the recv function fills the buf argument with the bytes read from the socket. We must define the size of the buffer, so that the network protocol stack doesn’t overwrite the buffer, which is defined by the len argument. Finally, we can alter the behavior of the read call using the flags argument. The value returned by the recv function is the number of bytes now contained in the msg buffer or -1 on error. An example of the recv function is:
#define MAX_BUFFER_SIZE 50 char buffer[MAX_BUFFER_SIZE+1]; ... numBytes = recv( sock, buffer, MAX_BUFFER_SIZE, 0);
At completion of this example, numBytes will contain the number of bytes that are contained within the buffer argument.
We could peek at the data that’s available to read by using the MSG_PEEK flag. This performs a read, but doesn’t consume the data at the socket. This requires another recv to actually consume the available data. An example of this type of read is illustrated as:
numBytes = recv( sock, buffer, MAX_BUFFER_SIZE, MSG_PEEK);
This call requires an extra copy (the first to peek at the data, and the second to actually read and consume it). More often than not, this behavior is handled instead at the Application layer by actually reading the data and then determining what action to take.
The sendto and recvfrom functions are used to send a message to the peer socket endpoint and receive a message from the peer socket endpoint. These functions have the following prototypes:
int sendto( int sock, const void *msg, int len, unsigned int flags, const struct sockaddr *to, int tolen ); int recvfrom( int sock, void *buf, int len, unsigned int flags, struct sockaddr *from, int *fromlen );
The sendto function is used by an unconnected socket to send a datagram to a destination defined by an initialized address structure. The sendto function is similar to the previously discussed send function, except that the recipient is defined by the to structure. An example of the sendto function is shown in the following code example:
 struct sockaddr_in destaddr; int sock; char *buf; ... memset( &destaddr, 0, sizeof(destaddr) ); destaddr.sin_family = AF_INET; destaddr.sin_port = 581; destaddr.sin_addr.s_addr = inet_addr("192.168.1.1"); sendto( sock, buf, strlen(buf), 0,            (struct sockaddr *)&destaddr, sizeof(destaddr) );  In this example, our datagram (contained with buf) is sent to an application on host 192.168.1.1, port number 581. The destaddr structure defines the intended recipient for our datagram.
Like the send function, the number of characters queued for transmission is returned, or –1 if an error occurred.
The recvfrom function provides the ability for an unconnected socket to receive datagrams. The recvfrom function is again similar to the recv function, but an address structure and length are provided. The address structure is used to return the sender of the datagram to the function caller. This information can be used with the sendto function to return a response datagram to the original sender.
An example of the recvfrom function is shown in the following code:
#define MAX_LEN 100 struct sockaddr_in fromaddr; int sock, len, fromlen; char buf[MAX_LEN+1]; ... fromlen = sizeof(fromaddr); len = recvfrom( sock, buf, MAX_LEN, 0, (struct sockaddr *)&fromaddr, &fromlen );
This blocking call returns when either an error occurs (represented by a –1 return), or a datagram is received (return value of 0 or greater). The datagram will be contained within buf and have a length of len. The fromaddr will contain the datagram sender, specifically the host address and port number of the originating application.
Socket options permit an application to change some of the modifiable behaviors of sockets and the functions that manipulate them. For example, an application can modify the sizes of the send or receive socket buffers or the size of the maximum segment used by the TCP layer for a given socket.
The functions for setting or retrieving options for a given socket is provided by the following function prototypes:
int getsockopt( int sock, int level, int optname, void *optval, socklen_t *optlen ); int setsockopt( int sock, int level, int optname, const void *optval, socklen_t optlen );
First, we define the socket of interest using the sock argument. Next, we must define the level of the socket option that is being applied. The level argument can be SOL_SOCKET for Sockets layer options, IPPROTO_IP for IP layer options, and IPPROTO_TCP for TCP layer options. The specific option within the level is applied using the optname argument. Arguments optval and optlen define the specifics of the value of the option. The optval is used to get or set the option value and optlen defines the length of the option. This slightly complicated structure is used because structures can be used to define options.
Let’s now look at an example for both setting and retrieving an option. In the first example, we’ll retrieve the size of the send buffer for a socket.
int sock, size, len; ... getsockopt( sock, SOL_SOCKET, SO_SNDBUF, (void *)&size, (socklen_t *)&len ); printf( "Send buffer size is &d\$$\n", size );
Now we’ll look at a slightly more complicated example. In this case, we’re going to set the linger option. Socket linger allows us to change the behavior of a stream socket when the socket is closed and data is remaining to be sent. After close is called, any data remaining will attempt to be sent for some amount of time. If after some duration, the data cannot be sent, then the data to be sent is abandoned. The time after the close to when the data is removed from the send queue is defined as the linger time. This can be set using a special structure called linger, as shown in the following example:
struct linger ling; int sock; ... ling.l_onoff = 1; /* Enable */ ling.l_linger = 10; /* 10 seconds */ setsockopt( sock, SOL_SOCKET, SO_LINGER, (void *)&ling, sizeof(struct linger) );
After this call is performed, the socket will wait 10 seconds after the socket close before aborting the send.
| 
 | < Day Day Up > | 
 | 
