\include{app_b}
\include{fdl}
+
% at the end put the bibliography
programma.
-
\subsection{Allocazione della memoria per i programmi C}
\label{sec:proc_mem_alloc}
\texttt{sbrk}), solo che a questo punto sarà possibile usarlo solo in maniera
indiretta attraverso dei puntatori.
+
+\subsection{Le funzioni \texttt{malloc}, \texttt{calloc}, \texttt{realloc} e
+ \texttt{free}}
+\label{sec:proc_mem_malloc}
+
Le funzioni previste dallo standard ANSI C per la gestione della memoria sono
quattro, i prototipi sono i seguenti:
\begin{prototype}{stdlib.h}{void *calloc(size\_t size)}
tipo.
+\subsection{La funzione \texttt{alloca}}
+\label{sec:proc_mem_alloca}
-
+\subsection{Le funzioni \texttt{brk} e \texttt{sbrk}}
+\label{sec:proc_mem_sbrk}
+\subsection{Il controllo della memoria virtuale}
+\label{sec:proc_mem_sbrk}
+
+\section{Il controllo di flusso non locale}
+\label{sec:proc_flux}
+
\section{La gestione di parametri e opzioni}
-\label{sec:parameter_options}
+\label{sec:proc_options}
Il passaggio dei parametri e delle variabili di ambiente dalla riga di comando
al singolo programma quando viene lanciato è effettuato attraverso le
versione estesa di \texttt{getopt}.
-
\subsection{Le variabili di ambiente}
\label{sec:proc_env_var}
per scrivere una prima applicazione client/server che usi i socket TCP per una
comunicazione in entrambe le direzioni.
-L'applicazione sarà una implementazione elementare, ma completa, del servizio
-\texttt{echo}. Si è scelto di usare questo servizio, seguendo lo Stevens, in
-quanto esso costituisce il prototipo ideale di una generica applicazione di
-rete; pertanto attraverso questo esempio potremo illustrare i fondamenti con i
-quali si può costruire una qualunque applicazione di rete.
-
Inoltre prenderemo in esame, oltre al comportamento in condizioni normali,
anche tutti i possibili scenari particolari (errori, sconnessione della rete,
crash del client o del server durante la connessione) che possono avere luogo
\section{Il servizio \texttt{echo}}
\label{sec:TCPsimp_echo}
-Il servizio \texttt{echo} è uno dei servizi standard solitamente provvisti
-direttamente dal superserver \texttt{inetd}, definito dall'RFC~862. Come dice
-il nome il servizio deve semplicemente rimandare indietro i dati che gli
-vengono inviati; l'RFC specifica che per il TCP una volta stabilita la
-connessione ogni dato in ingresso deve essere rimandato in uscita, fintanto
-che il chiamante non ha chiude la connessione; il servizio opera sulla porta
-TCP numero 7.
-
-Nel nostro caso l'esempio sarà strutturato scrivendo un client che legge una
-linea dallo standard input e la scrive sul server, il server leggerà una linea
-dalla connessione e la riscriverà all'indietro; sarà compito del client
+L'applicazione scelta come esempio sarà una implementazione elementare, ma
+completa, del servizio \texttt{echo}. Il servizio \texttt{echo} è uno dei
+servizi standard solitamente provvisti direttamente dal superserver
+\texttt{inetd}, ed è definito dall'RFC~862. Come dice il nome il servizio deve
+rimandare indietro sulla connessione i dati che gli vengono inviati; l'RFC
+descrive le specifiche sia per TCP che UDP, e per il primo stabilisce che una
+volta stabilita la connessione ogni dato in ingresso deve essere rimandato in
+uscita, fintanto che il chiamante non ha chiude la connessione; il servizio
+opera sulla porta 7.
+
+Nel nostro caso l'esempio sarà costituito da un client che legge una linea di
+caratteri dallo standard input e la scrive sul server, il server leggerà una
+linea dalla connessione e la riscriverà all'indietro; sarà compito del client
leggere la risposta del server e stamparla sullo standard output.
+Si è scelto di usare questo servizio, seguendo lo Stevens, perché costituisce
+il prototipo ideale di una generica applicazione di rete in cui un server
+risponde alle richieste di un client; tutto quello che cambia nel caso si una
+applicazione più complessa è la elaborazione dell'input del client da parte
+del server nel fornire le risposte in uscita.
+
+
--- /dev/null
+#
+# Simple Makefile to build examples
+#
+# C flags
+CC=gcc
+CFLAGS= -Wall
+CFLADJ=-c
+
+OBJ = SockRead.o SockWrite.o
+
+
+echod: SimpleEchoTCPServer.c $(OBJ)
+ $(CC) $(CFLAGS) SimpleEchoTCPServer.c $(OBJ) -o echod
+
+daytimed: ElemDaytimeTCPCuncServ.c
+ $(CC) $(CFLAGS) ElemDaytimeTCPCuncServ.c -o daytimed
+
+iterdaytimed: SimpleDaytimeTCPServer.c
+ $(CC) $(CFLAGS) SimpleDaytimeTCPServer.c -o iterdaytimed
+
+daytime: SimpleDaytimeTCPClient.c
+ $(CC) $(CFLAGS) SimpleDaytimeTCPClient.c -o daytime
+
+$(OBJ): wrappers.h
+
+
+
+.PHONY : clean
+clean:
+ rm -f *~
+ rm -f *.o
+ rm -f daytime
+
+
+
+
+
+
+
+
+
+
+
+
--- /dev/null
+/****************************************************************
+ *
+ * Program echo_tcp_server.c:
+ * Elementary TCP server for echo service (port 7)
+ *
+ * Author: Simone Piccardi
+ * Jun. 2001
+ *
+ * Usage: echod
+ *
+ * $Id: SimpleEchoTCPServer.c,v 1.1 2001/06/10 11:47:17 piccardi Exp $
+ *
+ ****************************************************************/
+/*
+ * Include needed headers
+ */
+#include <sys/types.h> /* predefined types */
+#include <unistd.h> /* include unix standard library */
+#include <arpa/inet.h> /* IP addresses conversion utiliites */
+#include <sys/socket.h> /* socket library */
+#include <stdio.h> /* include standard I/O library */
+#include <time.h>
+
+#include "wrappers.h"
+
+#define BACKLOG 10
+#define MAXLINE 256
+
+/* Subroutine declaration */
+void usage(void);
+void SockEcho(int sockfd);
+
+/* Program begining */
+int main(int argc, char *argv[])
+{
+/*
+ * Variables definition
+ */
+ int list_fd, conn_fd;
+ pid_t pid;
+ struct sockaddr_in serv_add;
+ /*
+ * Input section: decode parameters passed in the calling
+ * Use getopt function
+ */
+ int i;
+ opterr = 0; /* don't want writing to stderr */
+ while ( (i = getopt(argc, argv, "h")) != -1) {
+ switch (i) {
+ /*
+ * Handling options
+ */
+ case 'h':
+ printf("Wrong -h option use\n");
+ usage();
+ return(0);
+ break;
+ case '?': /* unrecognized options */
+ printf("Unrecognized options -%c\n",optopt);
+ usage();
+ default: /* should not reached */
+ usage();
+ }
+ }
+ /* ***********************************************************
+ *
+ * Options processing completed
+ *
+ * Main code beginning
+ *
+ * ***********************************************************/
+ /* create socket */
+ if ( (list_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
+ perror("Socket creation error");
+ exit(-1);
+ }
+ /* initialize address */
+ memset((void *)&serv_add, 0, sizeof(serv_add)); /* clear server address */
+ serv_add.sin_family = AF_INET; /* address type is INET */
+ serv_add.sin_port = htons(13); /* daytime port is 13 */
+ serv_add.sin_addr.s_addr = htonl(INADDR_ANY); /* connect from anywhere */
+ /* bind socket */
+ if (bind(list_fd, (struct sockaddr *)&serv_add, sizeof(serv_add)) < 0) {
+ perror("bind error");
+ exit(-1);
+ }
+ /* listen on socket */
+ if (listen(list_fd, BACKLOG) < 0 ) {
+ perror("listen error");
+ exit(-1);
+ }
+ /* handle echo to client */
+ while (1) {
+ /* accept connection */
+ if ( (conn_fd = accept(list_fd, NULL, NULL)) < 0) {
+ perror("accept error");
+ exit(-1);
+ }
+ /* fork to handle connection */
+ if ( (pid = fork()) < 0 ){
+ perror("fork error");
+ exit(-1);
+ }
+ if (pid == 0) { /* child */
+ close(list_fd); /* close listening socket */
+ SockEcho(conn_fd); /* handle echo */
+ exit(0);
+ } else { /* parent */
+ close(conn_fd); /* close connected socket */
+ }
+ }
+ /* normal exit, never reached */
+ exit(0);
+}
+/*
+ * routine to print usage info and exit
+ */
+void usage(void) {
+ printf("Simple daytime server\n");
+ printf("Usage:\n");
+ printf(" daytimed [-h] \n");
+ printf(" -h print this help\n");
+ exit(1);
+}
+/*
+ * routine to handle echo for connection
+ */
+void SockEcho(int sockfd) {
+ char buffer[MAXLINE];
+ int nread, nwrite;
+
+ /* main loop, reading 0 char means client close connection */
+ while ( (nread = SockRead(sockfd, buffer, MAXLINE)) != 0) {
+ nwrite = SockWrite(sockfd, buffer, nread);
+ }
+ return;
+}
#include <unistd.h>
+#include <errno.h>
ssize_t SockRead(int fd, void *buf, size_t count)
{
#include <unistd.h>
+#include <errno.h>
ssize_t SockWrite(int fd, const void *buf, size_t count)
{
*
* Author: S. Piccardi
*
- * $Id: wrappers.h,v 1.1 2001/03/05 22:20:08 piccardi Exp $
+ * $Id: wrappers.h,v 1.2 2001/06/10 11:47:17 piccardi Exp $
*
***************************************************************/
#include <sys/sem.h> /* IPC semaphore declarations */
* Return: the previous sigaction structure
*/
typedef void SigFunc(int);
-SigFunc * Signal(int signo, SigFunc *func)
+inline SigFunc * Signal(int signo, SigFunc *func)
{
struct sigaction new_handl, old_handl;
new_handl.sa_handler=func;
return (old_handl.sa_handler);
}
+/**
+ ** Defining prototypes for the all other functions
+ **/
+ssize_t SockRead(int fd, void *buf, size_t count);
+ssize_t SockWrite(int fd, const void *buf, size_t count);
+