Alma Mater Studiorum · Universit`
a di
Bologna
SCUOLA DI SCIENZE Corso di Laurea in Informatica
VSNLib: UN’ INTERFACCIA UNICA
DI CONFIGURAZIONE PER STACK VIRTUALI
TRAMITE PACCHETTI NETLINK
Relatore:
Chiar.mo Prof.
RENZO DAVOLI
Presentata da:
SIMONE PREITE
Sessione II
Anno Accademico 2016/17
Introduzione
Non vi `e dubbio che i nodi nella rete internet siano cresciuti in maniera
esponenziale fino ad oggi. Nonostante l’enorme dimensione raggiunta, que-sta rete ha mantenuto gli stessi principi costruttivi che impiegava quando collegava poche macchine sperimentali: i nodi sono considerati ancora le
in-terfacce di rete degli elaboratori. Questa situazione non `e pi`u coerente con
il funzionamento odierno di internet, dove il ruolo fondamentale non `e pi`u
quello delle macchine che erogano servizi bens`ı dei servizi stessi.
La crescita smisurata dei nodi di rete e la necessit`a di passare da
un’indiriz-zamento di nodi a quello di processi genera il bisogno di avere una grande
quantit`a di indirizzi per poter accedere a questi processi individualmente. La
famiglia di protocolli oggi pi`u in uso, IP vesione 4 (IPv4), risulta insufficiente
per questo scopo, infatti vengono usati solamente 4 byte per indirizzare tutti i nodi accedibili direttamente.
Per questo, gi`a da tempo, `e in atto la migrazione verso la versione 6 del
procollo IP (IPv6) che consente di avere una quantit`a di indirizzi maggiore,
sia per la problematica che abbiamo appena visto sia perch´e sempre pi`u si
affacciano sulla rete dispositivi automatici capaci di agire come entit`a
auto-nome, oggetti provvisti di collegamenti di rete: `e il fenomeno del cos`ı detto
Internet of Things o IoT.
In questa ottica si colloca il presente lavoro di tesi. La realizzazione di
stru-menti dell’Internet of Threads (IoTh), cio`e la possibilit`a di assegnare indirizzi
anche a singoli processi e thread, richiede l’utilizzo di stack di rete a livello del singolo processo, ovvero implementati come librerie e funzionanti in
ii Introduzione
zio utente.
Se l’interfaccia di uso comune da parte delle applicazioni della rete ha come standard la API di Berkeley socket non ne esiste una altrettanto standar-dizzata per la configurazione e la gestione delle interfacce di rete, essendo questa, nell’ottica degli elaboratori come nodi di rete, prerogative riservate agli amministratori di sistema e non comunemente accessibili come API da parte di processi.
In questa tesi si studier`a come realizzare una libreria che sia in grado di
in-terfacciarsi ad applicazioni che usano il protocollo netlink, usato dal kernel linux per la configurazione e la gestione delle interfacce di rete, in modo da poter utilizzare, all’interno dei programmi, gli strumenti per gli amministra-tori di rete.
Il primo capitolo presenta una rassegna dello stato dell’arte nell’ambito dei
protocolli di rete e del paradigma di IoTh; nel secondo capitolo verr`a
trat-tato il progetto VSNLib, ovvero la libreria oggetto della tesi; infine il terzo
Indice
Introduzione i 1 Internet of Threads 1 1.1 Definizione . . . 1 1.2 Stack Implementati . . . 3 1.2.1 PicoTCP . . . 3 1.2.2 LWIP . . . 3 1.2.3 LWIPv6 . . . 4 1.3 Netlink . . . 51.3.1 Inter Process Communication(IPC) . . . 5
1.3.2 Il Sistema IPC Netlink . . . 6
1.3.3 Libnl . . . 7 2 VSNLib 9 2.1 Il Progetto . . . 9 2.2 VSNLib . . . 10 2.2.1 Preambolo . . . 10 2.2.2 Ambiente di Sperimentazione . . . 11 2.2.3 Schema . . . 11 2.2.4 VSNLib . . . 13 2.2.5 Moduli . . . 13 3 Casi d’Uso 15 3.1 Environment . . . 15 iii
iv INDICE 3.2 Esempi . . . 16 3.2.1 Test . . . 19 3.2.2 Replica dell’Esperimento . . . 19 Conclusioni 23 Sviluppi Futuri 25 A Core 27 B Modules 35 C Test 39 Bibliografia 49
Elenco delle figure
1.1 Differenze tra stack reali e virtuali . . . 2
1.2 comunicazione netlink . . . 6
1.3 struct nlmsghdr . . . 7
2.1 struct nlmserror . . . 10
2.2 mappa concettuale libreria . . . 12
3.1 start/include mod vuos . . . 16
3.2 netcat IPv4 . . . 22
3.3 netcat IPv6 . . . 22
Elenco delle tabelle
1.1 IPv4 to IPv6 conversion . . . 5
1.2 IPv4 to IPv6 mask conversion . . . 5
Capitolo 1
Internet of Threads
1.1
Definizione
L’Internet of Things, ovvero l’internet delle cose, vuole rappresentare la diffusione dei sistemi embedded come veri e propri nodi di reti. Nell’ottica dell’IoT, oggi, apparecchi comuni in un’abitazione, quali condizionatore, la caldaia o addirittura il tostapane, possono essere nodi di rete ed essere quindi configurati e controllati attraverso di essa.
Il concetto di Internet of Threads `e una evoluzione di questa astrazione.
L’i-dea alla base di IoTh `e quella di avere non solo oggetti ma anche processi
come nodi di rete.
Per poter capire pienamente i vantaggi dell’IoTh si pu`o pensare ad
un’ana-logia con quanto `e successo nel mondo della telefonia. In passato la telefonia
collegava apparecchi fissi connessi via cavo. In questo modello il numero di telefono individuava un luogo e non una persona e quindi era comune, per trovare una persona, doversi interrogare sul dove questa potesse essere; al contrario l’avvento dei telefoni cellulari ha cambiato l’idea di significato di
numero di telefono, non pi`u associato a luoghi fisici ma ad una persona infatti
i telefoni cellulari sono normalmente telefoni personali[1, 2].
Come si pu`o notare infatti, con questa tecnologia lo stack `e direttamente
integrato nel programma. Nello stesso modo IoTh indirizza i processi che 1
2 1. Internet of Threads
(a) stack associato all’interfaccia fisica (b) stack associato ai processi
Figura 1.1: Differenze tra stack reali e virtuali
forniscono un servizio. Significa che questi non sono pi`u vincolati a risiedere
in un luogo specifico, o in un elaboratore specifico nell’ambito delle reti, ma `e
possibile trasparentemente farli migrare da una macchina all’altra senza che
questo crei difficolt`a di indirizzamento.
Una nota va fatta in termini di sicurezza. La possibilit`a che ogni servizio
uti-lizzi propri indirizzi IP e uno stack di rete privato rende molto pi`u difficile, se
non praticamente impossibile, sfruttarne le fragilit`a per scalare l’intrusione
1.2 Stack Implementati 3
1.2
Stack Implementati
Diversi sono i progetti che si occupano di offrire ai processi il proprio
stack di rete. Tra questi ne verranno presi in esame tre, considerati pi`u
indicati per uno studio in quanto open source. Ognuno di essi offre la propria implementazione di stack di rete.
Uno stack di rete legato al processo permette di connettere lo stesso ad una rete reale, tramite le interfacce fornite dal sistema, o virtuale (ad esempio
VDE) come se fosse una macchina fisica a s`e stante. Grazie a questo ogni
processo pu`o gestire la rete con le proprie regole che possono differire da
quelle in uso sulla macchina che lo ospita.
1.2.1
PicoTCP
PicoTCP[4] `e supportato da Tass Belgium (Altran); `e lo stack pi`u
cono-sciuto e diffuso per sistemi embedded ed esistono anche dei progetti, basati su IP, di reti mesh realizzati con picoTCP[14].
PicoTCP presenta una struttura modulare (schema molto utilizzato per
que-sto tipo di progetti come si vedr`a in seguito), che permette di avere il minimo
indispensabile all’interno dell’applicazione.
Vanta la compatibilit`a con un gran numero di dispositivi costruiti con diversi
processori e interfacce di rete, pu`o essere compilato con diversi compilatori
ed `e gi`a integrato in alcuni sistemi nati per ridurre al minimo il consumo di
risorse.
Purtroppo a causa di incompatibilit`a con la licenza GPL alcune parti del
codice non sono state rese pubbliche anche se rappresentano una piccola percentuale dell’intero progetto.
1.2.2
LWIP
Nasce come progetto di Adam Dunkels pensato per sistemi embedded ma che inizialmente non forniva supporto ad IPv6.
4 1. Internet of Threads
eseguire anche senza sistema oprativo (single-threaded) e con il minor
consu-mo di RAM possibile, inoltre `e modulare ed offre la possibilit`a di aggiungere
protocolli come DHCP e DNS solo in caso di necessit`a.
Supporta sia little che big endian e gira su processori a 8 e 32 bit, quindi funziona anche su un semplice ATMEGA.
Il protocollo IPv6 `e ancora in fase sperimentale: il supporto IPv6 `e
imple-mentato solamente nella versione disponibile da GITHub.
1.2.3
LWIPv6
LWIPv6[6] nasce per colmare la mancanza del supporto IPv6 in LWIP,
col tempo si `e differenziato al punto tale da essere considerato un progetto a
s`e stante.
Quando al team di virtualsquare1[15] `e sopravvenuta la necessit`a di avere
uno stack per la rete VDE non esisteva ancora nulla di maturo o comunque plug and play, da questa esigenza il team ha pensato di realizzare uno stack versatile in versione libreria tale per cui ogni processo potesse avere il proprio stack di rete.
Caratteristica principale di LWIPv6 `e il suo motore ibrido che, di fatto
fun-ziona solo con IPv6.
Gli indirizzi usati dal LWIPv6 sono descritti nel RFC 4291[17] al paragrafo 2.5.5.2. IPv4-Mapped IPv6 Address. La conversione degli indirizzi ip da
IPv4 ad IPv6 `e particolare ma intuitiva. I 32 bit finali dell’indirizzo IPv6
rappresentano quello in versione 4, i 16 bit antecedenti sono settati a 1 e la parte restante, i primi 80 bit, a 0. In tabella 1.1 un esempio sull’indirizzo 192.168.1.2.
Per quanto riguarda le maschere per`o questa configurazione non pu`o
esse-re adottata. Se i primi 80 bit venissero settati a 0 non saesse-rebbero considerati e nel caso in cui due reti differissero proprio nei primi 80 bit non si potrebbero
1un progetto ideato da Renzo Davoli e Michael Goldweber. Comprende una gamma di
tool pensati per sperimentare e creare strumenti di uso comune in tema di virtualizzazione nell’ambito dei sistemi operativi e delle reti.
1.3 Netlink 5
IP hversioni 80 bit 16 bit 32 bit
IP v4 unused unused 192.168.1.2
IP v6 00000000.00000000.0000 F F F F 192.168.1.2
IP v6 00000000.00000000.0000 F F F F C0A80102
Tabella 1.1: rappresentazione di un indirizzo IPv4 in IPv6
pi`u smistare i pacchetti verso la corretta rete di destinazione.
LWIPv6 si caratterizza perch´e, tramite una modifica della conversione, non
ha bisogno di usare due motori diversi per IPv4 ed IPv6 ma uno che `e ibrido.
La modifica `e semplice ed `e stato sufficiente ribaltare i primi 80 bit. Come
prima, suggeriamo un esempio di conversione di una maschera all’interno di LWIPv6 nella tabella 1.2
IP hversioni 80 bit 16 bit 32 bit
IP v4 unused unused 255.255.255.0
IP v6 F F F F F F F F.F F F F F F F F.F F F F F F F F 255.255.255.0
IP v6 F F F F F F F F.F F F F F F F F.F F F F F F F F F F F F F F 00
Tabella 1.2: rappresentazione delle maschere in LWIPv6
1.3
Netlink
1.3.1
Inter Process Communication(IPC)
Molti processi necessitano di scambiare informazioni per i motivi pi`u
di-sparati, dalla comunicazione di rete a quella interna ad una sola macchina. `
E banalmente limitativo pensare di costruire solo programmi fini a se stessi e che quindi non prevedano nessuna interazione con altri programmi.
Con IPC intendiamo quindi l’insieme delle tecnologie adottate per permettere ai processi di comunicare tra essi, sia che siano ospitati sulla stessa macchina
6 1. Internet of Threads
sia che siano distribuiti sulla rete. Tra queste tecnologie esiste appunto quel-la utilizzata all’interno delquel-la libreria VSNLib di cui parleremo nello specifico del prossimo capitolo.
1.3.2
Il Sistema IPC Netlink
Netlink `e un sistema IPC adottato dal kernel linux perch´e in grado di
mettere in comunicazione diversi task. Solitamente serve per far comunicare
task in user-space con task in kernel-space ma pu`o far interagire anche
pro-cessi entrambi in user-space.
Studiato per essere il successore di ioctl per le configurazioni ed il
monito-raggio, si propone di essere pi`u flessibile. Ma come avviene esattamente la
comunicazione attraverso netlink?
Netlink utilizza un sistema di socket indicizzato in base ai processi. Ogni
processo pu`o definire socket diversi di tipo netlink (AF NETLINK,
NE-TLINK GENERIC, NENE-TLINK XFRM) per inviare e ricevere messaggi con e da altri processi; implementa un sistema di porte basato sul process ID. Ad esempio per la comunicazione con il kernel viene usata la porta 0. In figura 1.2 possiamo vedere uno schema esplicativo di come avvengono le comnucazioni netlink, siano esse tra processi o tra processi e kernel
1.3 Netlink 7
L’interfaccia di comunicazione `e abbastanza standardizzata ma il payload `e
personalizzabile ed `e possibile definire la propria struttura da utilizzare per
inviare dati (informazioni), tra processi diversi, attraverso i socket netlink.
Figura 1.3: schema della struttura nlmsghdr
1.3.3
Libnl
Netlink Protocol Library Suite (libnl) [10], `e una raccolta di librerie ed
utility che forniscono API di comunicazione netlink basate su quelle del kernel linux e comprende:
libnl Core della libreria, offre uno strato di unificazione delle interfacce sulle quali poi si basano le altre librerie che pertanto dipendono da questa; libnl-route Questa libreria si occupa di fornire API per la configurazione
degli elementi della famiglia NETLINK ROUTE;
libnl-genl genl significa generic netlink e questa libreria offre una versione estesa del protocollo netlink;
Capitolo 2
VSNLib
Interfaccia di configurazione degli stack di rete attraverso l’utilizzo di tecnologia netlink.
2.1
Il Progetto
Gli stack analizzati in precedenza offrono a grandi linee la stessa tipologia di servizio seppur ognuna con le proprie caratteristice.
Nessuno dei progetti ha per`o tenuto in considerazione l’idea di utilizzare
un’interfaccia di configurazione che fosse standard, `e pertanto necessario
usa-re le funzioni specifiche per ognuno di questi, costringendo il programmatousa-re
a cambiare modalit`a di interazione da stack a stack.
L’idea di creare questa VSNLib nasce proprio dall’ esigenza di uniformare le interfacce di comunicazione in modo tale che l’utente non sia costretto ad adattarsi ogni qual volta decida di cambiare stack.
10 2. VSNLib
2.2
VSNLib
2.2.1
Preambolo
Per configurare uno stack di rete programmi come ip1 generano un
pac-chetto netlink con le informazioni necessarie lo inviano al kernel che lo elabo-ra, esegue le operazioni e genera un messaggio contenente un flag attraverso il quale viene comunicato il tipo di errore (0 in caso di successo). A questo
messaggio `e associato un payload di risposta generalmente usato per
compor-re un feedback da mostracompor-re all’utente che sta interagendo con il programma, in figura 2.1 lo schema dell’header per il messaggio di risposta.
VSNLib `e un progetto in fase di sviluppo scritto in C e completamente
open-Figura 2.1: schema della struttura per il messaggio di risposta
source, liberamente scaricabile attraverso la piattaforma github al seguente link https://github.com/simonepreite/vsnlib.
1il programma usato per configurare lo stack di rete tramite comandi quali ip addr...,
2.2 VSNLib 11
2.2.2
Ambiente di Sperimentazione
Per poter sperimentare e valutare le funzionalita’ della libreria VSNLib
`e necessario intercettare le comunicazioni Netlink fra i processi e il kernel.
Per farlo si `e reso necessario intercettare la system call di invio della richiesta
contenente il payload; ovvero l’esecuzione della sendto, verso il socket netlink sotto la nostra attenzione, contenente il buffer della richiesta.
In questo punto interviene la libreria che si interpone e, di fatto, fa le veci
del kernel, al quale il pacchetto netlink non arriver`a mai realmente.
In partenza VSNLib utilizzava come motore di cattura delle system call la
libreria purelibc[8], in seguito per`o si `e giunti alla conclusione che costruire
un modulo ad hoc all’interno del sistema vuos fosse un’alternativa pi`u
ver-satile, immediata e pulita, inoltre vuos ha un sistema di debug built in che
rende pi`u semplice l’individuazione dei problemi legati allo sviluppo.
Vuos[5] `e un progetto di virtualsquare, ha come obiettivo quello di
virtualiz-zare parti del sistema operativo senza negare all’utilizzatore la scelta degli
elementi da virtualizzare, `e costruito con un sistema di moduli che possono
essere aggiunti a discrezione dell’utente.
La versione di sviluppo ufficiale `e liberamente scaricabile attraverso la
piatta-forma github al seguente link https://github.com/virtualsquare/vuos.
Per il testing `e stato creato un fork di vuos contenente il modulo relativo
alla cattura delle system call di interesse per VSNLib, anche questa versione reperibile risiede su github https://github.com/simonepreite/vuos. Il modulo in questione si chiama unrealvsnlib per analogia alla libreria ma
ognuno `e libero di costruire un proprio modulo che si adatti alle esigenze di
sviluppo personali.
2.2.3
Schema
La libreria costituisce uno strato intermedio di compatibilit`a tra netlink
12 2. VSNLib
Vediamo come `e composta:
VSNLib: Il primo strato si occupa di inizializzare la libreria in base allo stack che si intende utilizzare (informazione richiesta all’utente) e di mediare la comunicazione tra netlink ed il modulo scelto.
Indicare esplicitamente il modulo desiderato `e una scelta
implemen-tativa dettata dal principio di avere un core che sia il pi`u contenuto
possibile.
Modules: Sono la parte specifica della libreria, essi contengono l’intestazio-ne delle funzioni gel’intestazio-neriche che si occupano di chiamare quelle particolari per ogni stack.
2.2 VSNLib 13
2.2.4
VSNLib
Espone le interfacce di comunicazione della libreria, che di fatto sono due, una per inizializzare l’ambiente ed una per inviare il pacchetto da gestire.
int i n i t _ v s n l i b(c h a r* s t a c k) ;
v o i d h a n d l e _ v s n l i b(v o i d* buf , s i z e _ t len , v o i d* nif ,
v o i d* s t a c k) ;
La prima funzione `e di inizializzazione della libreria e tramite dlopen ed il
nome del modulo da caricare si incarica di popolare l’array di puntatori con i corrispettivi delle funzioni nel modulo.
La seconda funzione contiene un loop che cicla sui messaggi del paccheto
ne-tlink, che di fatto potrebbero essere pi`u di uno. Per ogni messaggio ne viene
analizzata la tipologia ed in base al tipo si accede all’elemento dell’array che contiene il puntatore alla funzione adeguata a svolgere quel compito.
L’utente non ha accesso ad altre funzioni perch´e `e la libreria stessa ad
ana-lizzare il pacchetto, riempire i campi della struttura generica ed inviarlo al modulo relspecifico dello stack.
In risposta si ricever`a una struttura adatta alla costruzione del paccehetto
netlink da spedire alla system call recvfrom del programma in attesa, questo
`e l’ultimo compito della libreria.
2.2.5
Moduli
La presenza di moduli per gestire i vari stack lascia la libert`a a chiunque
necessiti di uno stack non supportato, o personalizzato, di creare il proprio modulo rispettando le specifiche della libreria.
Ogni modulo, infatti, ha la stessa struttura e contiene azioni generiche comuni in tutti gli stack. Vengono proposti i prototipi di queste funzioni qui di seguito.
s t r u c t r e s p o n s e* a d d d e l a d d r(s t r u c t c o n f i g* cfg ) ;
14 2. VSNLib
s t r u c t r e s p o n s e* a d d d e l l i n k(s t r u c t c o n f i g* cfg ) ;
s t r u c t r e s p o n s e* g e t s e t l i n k(s t r u c t c o n f i g* cfg ) ;
s t r u c t r e s p o n s e* a d d d e l r o u t e(s t r u c t c o n f i g* cfg ) ;
s t r u c t r e s p o n s e* g e t r o u t e(s t r u c t c o n f i g* cfg ) ;
In breve ogni modulo `e composto dallo stesso numero di funzioni che si
occupano di mediare la comunicazione tra la libreria e il vero stack. Ognuna di queste ha un’implementazione diversa a seconda del modulo.
Grazie alla tecnica dei puntatori a funzione, si riesce ad essere abbastanza generali e possiamo evitare di specificarne il nome completo.
L’idea `e stata quella di inizializzare, in fase di caricamento della libreria (che
avviene con dlopen), un array di puntatori a funzione (attraverso dlsym) in modo tale da poter effettuare le chiamate attraverso di esso. Dlopen e dlsym permettono di caricare dinamicamente solo il modulo di cui si necessita. All’ interno dei moduli le funzioni ricevono in input lo stesso tipo di struttura.
Durante l’esecuzione di un’azione loro compito `e la raccola di informazioni
dalle funzioni reali dello stack che verranno poi usate per riempire i campi
della struttura di risposta che a sua volta servir`a a creare il payload per il
Capitolo 3
Casi d’Uso
Di seguito alcuni esempi di utilizzo della libreria. Le dimostrazioni se-guenti sono state effettuate su una macchina con architettura amd64 e con installato il sistema operativo debian in versione sid (quindi unstable) ma sono state testate e riprodotte anche su altre configurazioni e distribuzioni GNU/Linux.
3.1
Environment
Come precedentemente accennato `e possibile integrare la libreria in vuos
attraverso il modulo unrealvsnlib o creandone uno alternativo che dipenda da VSNLib.
Creazione moduli: La creazione di un modulo `e molto semplice e basta
seguire le linee guida di quelli preesisteni, inoltre grazie ad una
routi-ne in cmake non `e necessario aggiungere alcuna riga manualmente al
makefile e sar`a, appunto, cmake ad occuparsi della compilazione.
Start Vuos: Vuos si avvia attaverso il comando umvu ed `e in grado di
inter-pretare comandi successivi come argomenti, come mostrato
nell’esem-pio infatti, umvu `e affiancato dal comando konsole e pertanto verr`a
16 3. Casi d’Uso
ciata una nuova istanza di terminale con ambiente vuos. In alternativa
`e possibile utilizzare altri comandi come xterm, bash o zsh.
Include dei Moduli vuos I moduli in vuos vengono inclusi attraverso la direttiva
vu_insmode <nome_modulo>
Un esempio di avvio e inclusione del modulo `e quello della figura
seguente.
Figura 3.1: start and include vuos
3.2
Esempi
Nella parte successiva verr`a mostrato un esempio di configurazione per
LWIPv6 attraverso VSNLib.
Per semplificare la dimostrazione si `e deciso di non utilizzare l’ambiente sopra
descritto o altre tecniche di cattura delle system call ma i pacchetti netlink sono stati costruiti all’interno del codice di test. Sono esempi statici di come
li creerebbe il programma ip ma utili a mostrare sia come `e formato un
pacchetto di questa tipologia sia per tracciare i passi compiuti per il parsing e la configurazione.
Init VSNLib: L’unico header di riferimento della libreria necessario `e
vsn-shared.h, all’interno sono dichiarate le funzioni necessarie all’utilizzo della libreria.
3.2 Esempi 17
e x t e r n int i n i t _ v s n l i b(c h a r* s t a c k) ;
e x t e r n v o i d h a n d l e _ v s n l i b(v o i d* buf , s i z e _ t len ,
v o i d* nif , v o i d* s t a c k) ;
Come gi`a spiegato la prima funzione serve per indicare alla libreria di
quale modulo fare il load pertanto richiede come input la stringa con
nome del file da caricare comprensivo di .so (si noti che `e necessario
specificare, tra i PATH di ricerca delle librerie, quello in cui risiede
il modulo .so. Senza questo accorgimento il file non verr`a trovato
da dlopen provocando la terminazione del programma. La variabile
dell’environment da settare `e LD LIBRARY PATH, un esempio `e mostrato
in figura 3.2). La seconda `e l’unico “handle” per passare il pacchetto
netlink che dovr`a essere poi interpretato, inoltre prende come argomeni
un puntatore allo stack e all’interfaccia oggetto della modifica.
Strutture: I pacchetti netlink sono accomunati dallo stesso header, come
mostrato in precedenza, ma presentano strutture diverse per il resto. In questo esempio sono stati creati i pacchetti per l’aggiunta e la rimozione di un indirizzo da un’interfaccia, l’aggiunta e la rimozione di una rotta e l’accensione/spegnimento di un’interfaccia. s t r u c t i n f o{ s t r u c t r t a t t r a t t r ; u n s i g n e d c h a r i p 6 _ a d d r e s s [s i z e o f(s t r u c t i n 6 _ a d d r) ]; }; s t r u c t a d d r _ p a y l o a d{ s t r u c t n l m s g h d r h e a d e r ; s t r u c t i f a d d r m s g ifa ; s t r u c t i n f o a t t r i b u t e [ 2 ] ; }; s t r u c t r o u t e _ p a y l o a d{
18 3. Casi d’Uso s t r u c t n l m s g h d r h e a d e r ; s t r u c t r t m s g rtm ; s t r u c t i n f o a t t r i b u t e ; }; s t r u c t l i n k _ p a y l o a d{ s t r u c t n l m s g h d r h e a d e r ; s t r u c t i f i n f o m s g ifi ; }; rispettivamente:
• info contiene il payload con gli indirizzi.
• addr payload viene usato per aggiungere/rimuovere un indirizzo a/da un interfaccia.
• route payload usato per aggiungere/rimuovere rotte. • link payload per attivare/disattivare l’interfaccia.
La descrizione `e ristretta a questo esempio, ma queste strutture vengono
impiegate per compiere pi`u azioni di quelle presentate.
Creazione pacchetti: E eseguita dalle funzioni nel test.`
v o i d f i l l _ b u f _ a d d r(s t r u c t l i n k _ p a y l o a d* buffer , int mode , int m o d e I P ) ; v o i d f i l l _ b u f _ r o u t e(s t r u c t l i n k _ p a y l o a d* buffer , int mode , int m o d e I P ) ; v o i d f i l l _ b u f _ l i n k(s t r u c t l i n k _ p a y l o a d* buffer , int m o d e ) ;
Il compito di queste funzioni `e riempire i campi delle strutture sopra
elencate in base all’operazione richiesta, `e possibile modificarle per
ot-tenere comportamenti diversi e compiere altri test.
co-3.2 Esempi 19
me ip in maniera semplificata, questo rende pi`u semplice al lettore la
comprensione, la modifica e l’utilizzo degli strumenti proposti.
VSNLib: Per tutte le operazioni viene chiamata sempre la stessa funzione
“handle” di VSNLib cambiando solo il paccehtto inviato. Questo dimo-stra come la libreria sia in grado di interpretare in maniera autonoma i pacchetti netlink ed utilizzarne le informazioni per completare con successo (o con fallimento in caso le informazioni non siano coerenti) le operazioni richieste.
3.2.1
Test
Ispirato da un programma esplicativo per LWIPv6 questo test dovrebbe avere lo stesso comportamento di netcat ma la configurazione dello stack av-viene attraverso VSNLib e i pacchetti netlink, come mostrato nell’Appendice C. Procediamo in sequenza:
Come prima operazione viene inizializzata VSNLib, successivamente viene eseguito un controllo per accertarsi che non sia necessario caricare LWIPv6 dinamicamente ed una volta superato questo step vengono creati stack e in-terfaccia di rete (nel caso specifico viene collegata ad uno switch vde). In ulti-ma analisi vengono eseguite le configurazioni necessarie a rendere l’interfaccia attiva.
3.2.2
Replica dell’Esperimento
• Preparazione: Il contesto prevede che sulla macchina siano installati
vde2 e la libreria LWIPv6, vde2 `e reperibile nei repository ufficiali di
debian mentre LWIPv6 `e disponibile nella versione sid sotto il nome
di liblwipv6-dev (si ricorda che tutto il codice `e open source ed `e
di-sponibile online, ai link di riferimento in bibliografia, per lo studio, lo sviluppo e l’utilizzo).
20 3. Casi d’Uso
• Installazione VSNLib:
Si `e utilizzato cmake per automatizzare il processo di compilazione e
installazione della libreria.
1 git c l o n e h t t p :// g i t h u b . com / s i m o n e p r e i t e / v s n l i b .git 2 cd v s n l i b 3 m k d i r b u i l d 4 cd b u i l d 5 c m a k e .. 6 m a k e 7 s u d o m a k e i n s t a l l
• Uso del test:
All’interno della directory vsnlib se ne trova un’altra chiamata test-dir. Al suo interno troviamo due file di esempio di un client netcat che utilizza uno stack LWIPv6 sia in configurazione IPv4 sia IPv6. Per compilare il test basta eseguire i comandi:
1 cd test - dir 2 m a k e
I comandi generano gli eseguibili nella subdirectory build. • Set vde2:
Per questo esempio `e stato usato uno switch vde il cui path `e passato
al programma di test per creare l’interfaccia. Bisogna creare, in primo luogo, lo switch 1 v d e _ s w i t c h - s </ p a t h / of / switch >
Questo path deve essere passato al programma test, ovviamente pu`o
essere modificato a piacimento.
• Set vdens:
3.2 Esempi 21
che da la possibilit`a di agganciarsi allo switch, per utilizzarlo `e possibile scaricare vdens direttamente dal suo repo[16].
La sintassi `e abbastanza intuitiva, per quanto riguarda il test:
1 v d e n s vde :// </ p a t h / of / switch >
Si noti che se il path del file con lo switch parte da root gli “/” diventano 3, ad esempio
1 v d e n s vde : / / / tmp / s w i t c h 1
Inoltre si ricordi, come spiegato nel readme di vdens, che l’uso dei namespace agli utenti non privilegiati deve essere abilitato in debian 1 s u d o e c h o 1 > / p r o c / sys / k e r n e l /
u n p r i v i l e g e d _ u s e r n s _ c l o n e
Entrati nell’ambiente vdens bisogna configurarlo per il server che si vuole con il nostro test: IPv4:
1 ip a d d r add 1 9 2 . 1 6 8 . 2 5 0 . 1 / 2 4 dev v d e 0 2 ip l i n k set v d e 0 up 3 nc - l - p 9 9 9 9 IPv6: 1 ip -6 a d d r add 2 0 0 1 : db8 :0: f 1 0 1 : : 1 / 6 4 dev v d e 0 2 ip l i n k set v d e 0 up 3 nc - l6 - p 9 9 9 9
Si presti attenzione alla versione di netcat installata in quanto netcat-traditional non supporta IPv6, si suggerisce netcat-openbsd.
• Start test:
Ora non resta che lanciare il programma di test in base alla configu-razione scelta e controllare che netcat permetta di scambiare messaggi come nelle figure 3.2 e 3.3.
22 3. Casi d’Uso
Figura 3.2: netcat IPv4
Conclusioni
Lo state dell’arte al momento in cui si scrive `e una libreria in grado di
configurare uno stack per LWIPv6, mentre gli altri moduli sono in fase di
sviluppo ma esistono gi`a dei proof of concept (moduli in fase sperimentale)
all’interno del repo su github1. I risultati dei test effettuati, nonostante il
progetto sia tutt’ora in fase di sviluppo, sono significativi: si riesce a configu-rare uno stack specifico in maniera generica ed usando le interfacce proposte dal kernel Linux.
L’utilizzo di queste tipologie di stack dipende strettamente dalla diffusione del protocollo IPv6 e dei sistemi embedded.
Lo sviluppo di tali tecnologie `e un punto chiave per il testing del procollo
IPv6.
La ricerca e lo sviluppo di questi meccanismi sono necessari per ottenere un
sistema performante e stabile quando IPv6 sar`a l’ordinario. Vanno inoltre
considerati i vantaggi conseguenti nelle reti virtuali che gi`a ora vengono
uti-lizzate per la sperimentazione e che in futuro saranno sicuramente di larga diffusione.
L’indipendenza dalle infrastrutture fisiche `e un vantaggio non indifferente,
significa che le reti ed i sistemi possono essere creati, modificati o ricostruiti
da zero con estrema facilit`a. Basti pensare ad una rete di macchine virtuali
che, trattandosi di software, pu`o essere ampliata con nuovi elaboratori
vir-tuali, spenta e modificata senza dover invesitire in nuovo hardware.
Quelli esposti sono solo alcuni dei vantaggi ma sono sufficienti a lasciare
1http://github.com/simonepreite/vsnlib
24 Conclusioni
Sviluppi Futuri
In fase di progettazione non sono stati posti limiti alla crescita del
pro-getto e la sua modularit`a `e legata principalmente a questo aspetto.
L’augurio `e che questo progetto continui a crescere arricchendosi di
funzio-nalit`a.
In questa fase ci si `e occupati delle configurazioni e del routing che deve
es-sere completata e testata prima di poter affermarne la stabilit`a.
Senz’altro aumentare il supporto agli stack meno conosciuti ed a quelli che in
futuro saranno sviluppati `e il prosieguo pi`u ovvio per VSNLib. Tra le altre
ipotesi di ampliamento c’`e quella di gestione del filtering, l’intento in questo
caso sar`a fornire un’interfaccia identica a quella di iptables ma per settare le
regole di filtraggio sugli stack virtuali.
Appendice A
Core
vsnshared.h # i f n d e f _ V S N S H A R E D _ H # d e f i n e _ V S N S H A R E D _ H # i n c l u d e < asm / t y p e s . h > # i n c l u d e < l i n u x / n e t l i n k . h > # i n c l u d e < l i n u x / r t n e t l i n k . h > # i n c l u d e < a r p a / i n e t . h > s t r u c t r e s p o n s e{ int e r r o r ; // 0 f o r s u c c e s s };e x t e r n v o i d h a n d l e _ v s n l i b(v o i d* buf , s i z e _ t len , v o i d* nif , v o i d* s t a c k) ;
e x t e r n int i n i t _ v s n l i b(c h a r* s t a c k) ; # e n d i f vsnlib.h # i f n d e f _ V S N L I B _ H # d e f i n e _ V S N L I B _ H # i n c l u d e " v s n s h a r e d . h " # d e f i n e A P I _ T A B L E _ S I Z E 6 /* s t r u c t d e f i n i t i o n */ s t r u c t c o n f i g{ c h a r* a d d r ; int m a s k ; 27
28 A Core int f a m i l y ; v o i d* i n t e r f a c e ; v o i d* s t a c k; // s t a c k f i l e d e s c r i p t o r int o p e r a t i o n ; }; t y p e d e f s t r u c t r e s p o n s e* (*g e n e r i c _ a p i) (s t r u c t c o n f i g* cfg ) ; s t r u c t g e n _ a p i{ c h a r* f u n _ n a m e ; g e n e r i c _ a p i r e a l _ f u n ; }; s t a t i c s t r u c t g e n _ a p i a p i _ t a b l e [ ] = { {" a d d d e l l i n k ", N U L L} , {" g e t s e t l i n k ",N U L L} , {" a d d d e l a d d r ", N U L L} , {" g e t a d d r ", N U L L} , {" a d d d e l r o u t e ", N U L L} , {" g e t r o u t e ", N U L L} }; /* c l i e n t s i d e */ s t r u c t n l m s g h d r* r t m _ n e w r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ d e l r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ g e t r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ n e w a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ d e l a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ g e t a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ n e w l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ d e l l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ g e t l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t r u c t n l m s g h d r* r t m _ s e t l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; t y p e d e f s t r u c t n l m s g h d r* v s n _ c l i(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) ; s t a t i c v s n _ c l i * v s n c l i _ f u n _ t a b l e [ ] = {
A Core 29 /* N E W / D E L / G E T / S E T l i n k */ r t m _ n e w l i n k _ c, r t m _ d e l l i n k _ c, r t m _ g e t l i n k _ c, r t m _ s e t l i n k _ c, /* N E W / D E L / G E T a d d r */ r t m _ n e w a d d r _ c, r t m _ d e l a d d r _ c, r t m _ g e t a d d r _ c, NULL, /* N E W / D E L / G E T r o u t e */ r t m _ n e w r o u t e _ c, r t m _ d e l r o u t e _ c, r t m _ g e t r o u t e _ c, N U L L }; # e n d i f vsnlib.c # i n c l u d e < s t d i o . h > # i n c l u d e < s t d l i b . h > # i n c l u d e < f c n t l . h > # i n c l u d e < u n i s t d . h > # i n c l u d e < d i r e n t . h > # i n c l u d e < s t r i n g . h > # i n c l u d e < d l f c n . h > # i n c l u d e < v s n l i b . h > # i n c l u d e < v s n s h a r e d . h > s t r u c t r e t _ v a l{ s t r u c t n l m s g h d r* res ; s t r u c t r e t _ v a l* n e x t ; }; c h a r* c o m b i n e _ s t a c k _ h a n d l e r(c h a r* f u n _ n a m e , c h a r* s t a c k) { // r i m u o v e il . so c h a r* c o m b o ; int l e n _ s t a c k = s t r l e n(s t a c k) -1; c o m b o = m a l l o c(s i z e o f(c h a r) *(s t r l e n( f u n _ n a m e ) +( l e n _ s t a c k -1) ) ) ; s t r c p y( combo , s t a c k) ; c o m b o [ l e n _ s t a c k -2] = ’ _ ’; c o m b o [ l e n _ s t a c k -1] = ’ \0 ’; s t r c a t( combo , f u n _ n a m e ) ; r e t u r n c o m b o ;
30 A Core } int i n i t _ v s n l i b(c h a r* s t a c k) { c o n s t c h a r * e r r o r ; c h a r* f ; v o i d * s t a c k _ m o d u l e ; s t a c k _ m o d u l e = d l o p e n(stack,R T L D _ L A Z Y) ; if(! s t a c k _ m o d u l e ) { f p r i n t f( stderr , " C a n n o t o p e n s t a c k l i b r a r y : % s \ n ", d l e r r o r() ) ; r e t u r n -1; } for(int i =0; i < A P I _ T A B L E _ S I Z E; i ++) { a p i _ t a b l e [ i ]. r e a l _ f u n = d l s y m( s t a c k _ m o d u l e , a p i _ t a b l e [ i ]. f u n _ n a m e ) ; if (( e r r o r = d l e r r o r() ) ) { f p r i n t f( stderr , " C o u l d n ’ t f i n d % s : % s \ n ", a p i _ t a b l e [ i ]. f u n _ n a m e , e r r o r ) ; e x i t(1) ; } } r e t u r n 0; // if s u c c e s s }
v o i d h a n d l e _ v s n l i b(v o i d* buf , s i z e _ t len , v o i d* nif , v o i d* s t a c k) {
s t r u c t n l m s g h d r* h e a d e r = (s t r u c t n l m s g h d r*) buf ; s t r u c t n l m s g h d r* r e t _ p k g ; w h i l e (N L M S G _ O K( header , len ) ) { /* i n t e r r ; */ int t y p e ; if ( header - > n l m s g _ t y p e == N L M S G _ D O N E) b r e a k; t y p e = header - > n l m s g _ t y p e - R T M _ B A S E; header - > n l m s g _ p i d = g e t p i d () ; if ( t y p e >= 0 && v s n c l i _ f u n _ t a b l e [ t y p e ] != N U L L) { s t r u c t r e t _ v a l* app = m a l l o c(s i z e o f(s t r u c t r e t _ v a l) ) ; ( v s n c l i _ f u n _ t a b l e [ t y p e ]( header , nif , s t a c k) ) ; } h e a d e r = N L M S G _ N E X T( header , len ) ; } } s t r u c t n l m s g h d r* r t m _ n e w l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f i n f o m s g* ifi = (s t r u c t i f i n f o m s g*) (N L M S G _ D A T A( h e a d e r ) ) ;
A Core 31 s t r u c t c o n f i g g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ c o n f . i n t e r f a c e = nif ; g e n e r i c _ c o n f . o p e r a t i o n = ifi - > i f i _ f l a g s ; g e n e r i c _ r e s p = a p i _ t a b l e [ 0 ] . r e a l _ f u n (& g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ d e l l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f i n f o m s g* ifi = (s t r u c t i f i n f o m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g* g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ r e s p = a p i _ t a b l e [ 0 ] . r e a l _ f u n ( g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ g e t l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f i n f o m s g* ifi = (s t r u c t i f i n f o m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g* g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ r e s p = a p i _ t a b l e [ 1 ] . r e a l _ f u n ( g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ s e t l i n k _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f i n f o m s g* ifi = (s t r u c t i f i n f o m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g* g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ r e s p = a p i _ t a b l e [ 1 ] . r e a l _ f u n ( g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ n e w a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f a d d r m s g* ifa = (s t r u c t i f a d d r m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; c h a r s t r 6 [I N E T 6 _ A D D R S T R L E N]; c h a r* s t r 4 ; g e n e r i c _ c o n f . o p e r a t i o n = header - > n l m s g _ t y p e ; g e n e r i c _ c o n f . i n t e r f a c e = nif ; g e n e r i c _ c o n f . m a s k = ifa - > i f a _ p r e f i x l e n ; g e n e r i c _ c o n f . f a m i l y = ifa - > i f a _ f a m i l y ; s t r u c t r t a t t r* a t t r = (s t r u c t r t a t t r*) ((v o i d*) (((c h a r*) ifa ) + s i z e o f(s t r u c t i f a d d r m s g) ) ) ; if( ifa - > i f a _ f a m i l y == A F _ I N E T 6) { s t r u c t i n 6 _ a d d r* i p 6 _ a = (s t r u c t i n 6 _ a d d r*)R T A _ D A T A( a t t r ) ; i n e t _ n t o p( ifa - > i f a _ f a m i l y , ip6_a , str6 , I N E T 6 _ A D D R S T R L E N) ;
32 A Core i n e t _ p t o n( ifa - > i f a _ f a m i l y , str6 , i p 6 _ a ) ; g e n e r i c _ c o n f . a d d r = s t r 6 ; } e l s e{ s t r u c t i n _ a d d r i p 4 _ a = *((s t r u c t i n _ a d d r*)R T A _ D A T A( a t t r ) ) ; s t r 4 = i n e t _ n t o a( i p 4 _ a ) ; g e n e r i c _ c o n f . a d d r = s t r 4 ; } g e n e r i c _ c o n f .s t a c k=s t a c k; g e n e r i c _ r e s p = a p i _ t a b l e [ 2 ] . r e a l _ f u n (& g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ d e l a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { r t m _ n e w a d d r _ c( header , nif , s t a c k) ; } s t r u c t n l m s g h d r* r t m _ g e t a d d r _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t i f a d d r m s g* ifa = (s t r u c t i f a d d r m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g* g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ r e s p = a p i _ t a b l e [ 3 ] . r e a l _ f u n ( g e n e r i c _ c o n f ) ; } s t r u c t n l m s g h d r* r t m _ n e w r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t r t m s g* rtm = (s t r u c t r t m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; c h a r s t r 6 [I N E T 6 _ A D D R S T R L E N]; c h a r* s t r 4 ; g e n e r i c _ c o n f . o p e r a t i o n = header - > n l m s g _ t y p e ; g e n e r i c _ c o n f . i n t e r f a c e = nif ; g e n e r i c _ c o n f . f a m i l y = rtm - > r t m _ f a m i l y ; g e n e r i c _ c o n f .s t a c k=s t a c k; s t r u c t r t a t t r* a t t r = (s t r u c t r t a t t r*) ((v o i d*) (((c h a r*) rtm ) + s i z e o f(s t r u c t r t m s g) ) ) ; if( rtm - > r t m _ f a m i l y == A F _ I N E T 6) { s t r u c t i n 6 _ a d d r* i p 6 _ a = (s t r u c t i n 6 _ a d d r*)R T A _ D A T A( a t t r ) ; i n e t _ n t o p( rtm - > r t m _ f a m i l y , ip6_a , str6 , I N E T 6 _ A D D R S T R L E N) ; g e n e r i c _ c o n f . a d d r = s t r 6 ; e l s e{ s t r u c t i n _ a d d r i p 4 _ a = *((s t r u c t i n _ a d d r*)R T A _ D A T A( a t t r ) ) ; s t r 4 = i n e t _ n t o a( i p 4 _ a ) ; g e n e r i c _ c o n f . a d d r = s t r 4 ;
A Core 33 } g e n e r i c _ r e s p = a p i _ t a b l e [ 4 ] . r e a l _ f u n (& g e n e r i c _ c o n f ) ; } } s t r u c t n l m s g h d r* r t m _ d e l r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { r t m _ n e w r o u t e _ c( header , nif , s t a c k) ; } s t r u c t n l m s g h d r* r t m _ g e t r o u t e _ c(s t r u c t n l m s g h d r* header , v o i d* nif , v o i d* s t a c k) { s t r u c t r t m s g* rtm = (s t r u c t r t m s g*) (N L M S G _ D A T A( h e a d e r ) ) ; s t r u c t c o n f i g* g e n e r i c _ c o n f ; s t r u c t r e s p o n s e* g e n e r i c _ r e s p ; g e n e r i c _ r e s p = a p i _ t a b l e [ 5 ] . r e a l _ f u n ( g e n e r i c _ c o n f ) ; }
Appendice B
Modules
vsnmodules.h LWIPv6.c # i n c l u d e < s t d i o . h > # i n c l u d e < v s n l i b . h > # i n c l u d e < l w i p v 6 . h > s t r u c t r e s p o n s e* a d d d e l a d d r(s t r u c t c o n f i g* cfg ) { s t r u c t i p _ a d d r a d d r ; s t r u c t i p _ a d d r m a s k ; u i n t 1 6 _ t* p _ m a s k = N U L L; if( cfg - > f a m i l y == A F _ I N E T 6) { u i n t 1 6 _ t m a s k _ a p p [8]={0 ,0 ,0 ,0 ,0 ,0 ,0 ,0}; s t r u c t i n 6 _ a d d r ip6 ; u i n t 1 6 _ t* p = (u i n t 1 6 _ t*) (& ip6 ) ; p _ m a s k = (u i n t 1 6 _ t*) (& m a s k _ a p p ) ; for(int i =0; i <8; i ++) { * p =0; p ++; } p = (u i n t 1 6 _ t*) (& ip6 ) ;i n e t _ p t o n( cfg - > family , cfg - > addr , & ip6 ) ;
w h i l e( cfg - > m a s k > 0) { * p _ m a s k = 0 x f f f f ; cfg - > mask -= 16;
36 B Modules p _ m a s k ++; } p _ m a s k = (u i n t 1 6 _ t*) (& m a s k _ a p p ) ; p r i n t f(" i p v 6 a d d r : %04 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X \ n ", h t o n s(*( p ) ) ,h t o n s(*( p +1) ) ,h t o n s(*( p +2) ) ,h t o n s(*( p +3) ) ,h t o n s(*( p +4) ) , h t o n s(*( p +5) ) ,h t o n s(*( p +6) )h t o n s(*( p +7) ) ) ; I P 6 _ A D D R(& addr ,h t o n s(*( p ) ) ,h t o n s(*( p +1) ) ,h t o n s(*( p +2) ) ,h t o n s(*( p +3) ) , h t o n s(*( p +4) ) ,h t o n s(*( p +5) ) ,h t o n s(*( p +6) ) ,h t o n s(*( p +7) ) ) ; p r i n t f(" m a s k : %04 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X \ n ",*( p _ m a s k ) ,*( p _ m a s k +1) ,*( p _ m a s k +2) ,*( p _ m a s k +3) ,*( p _ m a s k +4) ,*( p _ m a s k +5) ,*( p _ m a s k +6) ,*( p _ m a s k +7) ) ; I P 6 _ A D D R(& mask ,*( p _ m a s k ) ,*( p _ m a s k +1) ,*( p _ m a s k +2) ,*( p _ m a s k +3) ,*( p _ m a s k +4) ,*( p _ m a s k +5) ,*( p _ m a s k +6) ,*( p _ m a s k +7) ) ; } e l s e{ s t r u c t i n _ a d d r ip4 ; u i n t 1 6 _ t l w i p _ m a s k 4 [4] = {0 ,0 ,0 ,0}; p _ m a s k = (u i n t 1 6 _ t*) (& l w i p _ m a s k 4 ) ; w h i l e( cfg - > m a s k > 0) { * p _ m a s k = 2 5 5 ; cfg - > m a s k -= 8; p _ m a s k ++; }
i n e t _ a t o n( cfg - > addr , & ip4 ) ;
u n s i g n e d c h a r i p 4 _ d o t [ 4 ] ; int j =0; for(int i =0; i <4; i ++) { i p 4 _ d o t [ i ]=( ip4 . s _ a d d r > > j ) & 0 xFF ; j = j +8; } p r i n t f(" i p v 4 a d d r e s s : % d .% d .% d .% d \ n ", i p 4 _ d o t [0] , i p 4 _ d o t [1] , i p 4 _ d o t [2] , i p 4 _ d o t [ 3 ] ) ; I P 6 4 _ A D D R(& addr , i p 4 _ d o t [0] , i p 4 _ d o t [1] , i p 4 _ d o t [2] , i p 4 _ d o t [ 3 ] ) ; p r i n t f(" i p v 4 m a s k : % d .% d .% d .% d \ n ", l w i p _ m a s k 4 [0] , l w i p _ m a s k 4 [1] , l w i p _ m a s k 4 [2] , l w i p _ m a s k 4 [ 3 ] ) ; I P 6 4 _ M A S K A D D R(& mask , l w i p _ m a s k 4 [0] , l w i p _ m a s k 4 [1] , l w i p _ m a s k 4 [2] , l w i p _ m a s k 4 [ 3 ] ) ; } if( cfg - > o p e r a t i o n == R T M _ N E W A D D R) { p r i n t f(" e r r o r add a d d r l w i p v 6 : % d \ n ", l w i p _ a d d _ a d d r((s t r u c t n e t i f*) ( cfg - > i n t e r f a c e ) ,& addr ,& m a s k ) ) ;
}
e l s e if ( cfg - > o p e r a t i o n == R T M _ D E L A D D R) {
p r i n t f(" e r r o r del a d d r l w i p v 6 : % d \ n ", l w i p _ d e l _ a d d r((s t r u c t n e t i f*) ( cfg - > i n t e r f a c e ) ,& addr ,& m a s k ) ) ;
B Modules 37 } } s t r u c t r e s p o n s e* g e t a d d r(s t r u c t c o n f i g* cfg ) { } s t r u c t r e s p o n s e* a d d d e l l i n k(s t r u c t c o n f i g* cfg ) { p r i n t f(" a d d d e l l i n k \ n ") ; if( cfg - > o p e r a t i o n == 1) p r i n t f(" i f u p : % d \ n ", l w i p _ i f u p((s t r u c t n e t i f*) ( cfg - > i n t e r f a c e ) ) ) ; e l s e p r i n t f(" i f d o w n : % d \ n ", l w i p _ i f d o w n((s t r u c t n e t i f*) ( cfg - > i n t e r f a c e ) ) ) ; } s t r u c t r e s p o n s e* g e t s e t l i n k(s t r u c t c o n f i g* cfg ) { } s t r u c t r e s p o n s e* a d d d e l r o u t e(s t r u c t c o n f i g* cfg ) { s t r u c t i p _ a d d r a d d r ; if( cfg - > f a m i l y ==A F _ I N E T 6) { s t r u c t i n 6 _ a d d r a d d r e s s ; u i n t 1 6 _ t* p = (u i n t 1 6 _ t*) (& a d d r e s s ) ; p = (u i n t 1 6 _ t*) (& a d d r e s s ) ;
i n e t _ p t o n( cfg - > family , cfg - > addr , & a d d r e s s ) ;
p r i n t f(" i p v 6 a d d r : %04 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X : : % 0 4 X \ n ", h t o n s(*( p ) ) ,h t o n s(*( p +1) ) ,h t o n s(*( p +2) ) ,h t o n s(*( p +3) ) ,h t o n s(*( p +4) ) , h t o n s(*( p +5) ) ,h t o n s(*( p +6) )h t o n s(*( p +7) ) ) ; I P 6 _ A D D R(& addr ,h t o n s(*( p ) ) ,h t o n s(*( p +1) ) ,h t o n s(*( p +2) ) ,h t o n s(*( p +3) ) , h t o n s(*( p +4) ) ,h t o n s(*( p +5) ) ,h t o n s(*( p +6) ) ,h t o n s(*( p +7) ) ) ; } e l s e{ s t r u c t i n _ a d d r ip4 ;
i n e t _ a t o n( cfg - > addr , & ip4 ) ;
u n s i g n e d c h a r i p 4 _ d o t [ 4 ] ; int j =0; for(int i =0; i <4; i ++) { i p 4 _ d o t [ i ]=( ip4 . s _ a d d r > > j ) & 0 xFF ; j = j +8; } p r i n t f(" i p v 4 r o u t e : % d .% d .% d .% d \ n ", i p 4 _ d o t [0] , i p 4 _ d o t [1] , i p 4 _ d o t [2] , i p 4 _ d o t [3] ) ; I P 6 4 _ A D D R(& addr , i p 4 _ d o t [0] , i p 4 _ d o t [1] , i p 4 _ d o t [2] , i p 4 _ d o t [ 3 ] ) ; }
38 B Modules
if( cfg - > o p e r a t i o n ==R T M _ N E W R O U T E) {
p r i n t f(" e r r o r add r o u t e l w i p v 6 % d \ n ", l w i p _ a d d _ r o u t e((s t r u c t s t a c k*) cfg - >stack, I P _ A D D R _ A N Y, I P _ A D D R _ A N Y, & addr , (s t r u c t n e t i f*) ( cfg - > i n t e r f a c e ) ,0) ) ;
}
e l s e if( cfg - > o p e r a t i o n ==R T M _ D E L R O U T E) {
p r i n t f(" e r r o r del r o u t e l w i p v 6 : % d \ n ", l w i p _ d e l _ r o u t e((s t r u c t s t a c k*) cfg - >stack, I P _ A D D R _ A N Y, I P _ A D D R _ A N Y, & addr , ( s t r u c t n e t i f *) ( cfg - > i n t e r f a c e ) , 0) ) ;
} }
s t r u c t r e s p o n s e* g e t r o u t e(s t r u c t c o n f i g* cfg ) { }
Appendice C
Test
minlibnetlink.h # i f n d e f _ M I N L I B N E T L I N K _ H # d e f i n e _ M I N L I B N E T L I N K _ H # i n c l u d e < s t d i o . h > # i n c l u d e < f c n t l . h > # i n c l u d e < s t d l i b . h > # i n c l u d e < u n i s t d . h > # i n c l u d e < s t r i n g . h > # i n c l u d e < l w i p v 6 . h > # i n c l u d e < sys / s o c k e t . h > # i n c l u d e < sys / s e l e c t . h > # i n c l u d e < n e t i n e t / in . h > # i n c l u d e < asm / t y p e s . h > # i n c l u d e < l i n u x / n e t l i n k . h > # i n c l u d e < l i n u x / r t n e t l i n k . h > # i n c l u d e < v s n s h a r e d . h > s t r u c t i n f o{ s t r u c t r t a t t r a t t r ; u n s i g n e d c h a r i p 6 _ a d d r e s s [s i z e o f(s t r u c t i n 6 _ a d d r) ]; }; s t r u c t a d d r _ p a y l o a d{ s t r u c t n l m s g h d r h e a d e r ; s t r u c t i f a d d r m s g ifa ; s t r u c t i n f o a t t r i b u t e [ 2 ] ; }; 3940 C Test s t r u c t r o u t e _ p a y l o a d{ s t r u c t n l m s g h d r h e a d e r ; s t r u c t r t m s g rtm ; s t r u c t i n f o a t t r i b u t e ; }; s t r u c t l i n k _ p a y l o a d{ s t r u c t n l m s g h d r h e a d e r ; s t r u c t i f i n f o m s g ifi ; }; v o i d f i l l _ b u f _ l i n k(s t r u c t l i n k _ p a y l o a d* buffer , int m o d e ) ;
v o i d f i l l _ b u f _ r o u t e(s t r u c t r o u t e _ p a y l o a d* buffer , c h a r* IPv6 , int mode , int
m o d e I P ) ;
v o i d f i l l _ b u f _ a d d r(s t r u c t a d d r _ p a y l o a d* buffer , c h a r* IP , int mask , int mode , int m o d e I P ) ; # e n d i f minlibnetlink.c # i n c l u d e " m i n l i b n e t l i n k . h " v o i d f i l l _ b u f _ l i n k(s t r u c t l i n k _ p a y l o a d* buffer , int m o d e ) { // n l m s g h d r i n i t buffer - > h e a d e r . n l m s g _ l e n = 3 2 ; buffer - > h e a d e r . n l m s g _ t y p e =R T M _ N E W L I N K; buffer - > h e a d e r . n l m s g _ f l a g s =N L M _ F _ R E Q U E S T|N L M _ F _ A C K; buffer - > ifi . i f i _ f l a g s = m o d e ; buffer - > h e a d e r . n l m s g _ s e q = 1 0 2 ; buffer - > h e a d e r . n l m s g _ p i d =0; // i f a d d r i n i t buffer - > ifi . i f i _ f a m i l y =A F _ U N S P E C; buffer - > ifi . i f i _ t y p e =A F _ N E T R O M; // m a s c h e r a buffer - > ifi . i f i _ i n d e x =0; buffer - > ifi . i f i _ c h a n g e =0 x1 ; }
v o i d f i l l _ b u f _ r o u t e(s t r u c t r o u t e _ p a y l o a d* buffer , c h a r* IPv6 , int mode , int
m o d e I P ) {
i n e t _ p t o n( modeIP , IPv6 , buffer - > a t t r i b u t e . i p 6 _ a d d r e s s ) ;
// r t a t t r i n i t
buffer - > a t t r i b u t e . a t t r . r t a _ l e n = 2 0 ;
buffer - > a t t r i b u t e . a t t r . r t a _ t y p e =R T A _ G A T E W A Y;
C Test 41 buffer - > h e a d e r . n l m s g _ l e n = 6 4 ; buffer - > h e a d e r . n l m s g _ t y p e = m o d e ; buffer - > h e a d e r . n l m s g _ s e q = 1 0 1 ; buffer - > h e a d e r . n l m s g _ p i d =0; // r t m s g i n i t if( m o d e I P ==A F _ I N E T 6) { buffer - > rtm . r t m _ f a m i l y =A F _ I N E T 6; } e l s e{ buffer - > rtm . r t m _ f a m i l y =A F _ I N E T; } buffer - > rtm . r t m _ d s t _ l e n =0; // m a s c h e r a buffer - > rtm . r t m _ s r c _ l e n =0; buffer - > rtm . r t m _ t o s =0; buffer - > rtm . r t m _ t a b l e =R T _ T A B L E _ M A I N; buffer - > rtm . r t m _ p r o t o c o l =R T P R O T _ U N S P E C; buffer - > rtm . r t m _ s c o p e =R T _ S C O P E _ N O W H E R E; buffer - > rtm . r t m _ t y p e =R T N _ U N S P E C; buffer - > rtm . r t m _ f l a g s =0; if( m o d e ==R T M _ N E W R O U T E) { buffer - > h e a d e r . n l m s g _ f l a g s =N L M _ F _ R E Q U E S T|N L M _ F _ A C K|N L M _ F _ E X C L| N L M _ F _ C R E A T E; } e l s e if( m o d e ==R T M _ D E L R O U T E) { buffer - > h e a d e r . n l m s g _ f l a g s =N L M _ F _ R E Q U E S T|N L M _ F _ A C K; } } // g e n e r e a lo s t e s s o p a c c h e t t o n e t l i n k g e n e r a t o da ip a d d r a d d < I P v 6 / mask > d e v < d e v i c e _ n a m e >
v o i d f i l l _ b u f _ a d d r(s t r u c t a d d r _ p a y l o a d* buffer , c h a r* IP , int mask , int mode , int m o d e I P ) { i n e t _ p t o n( modeIP , IP , buffer - > a t t r i b u t e [ 0 ] . i p 6 _ a d d r e s s ) ; i n e t _ p t o n( modeIP , IP , buffer - > a t t r i b u t e [ 1 ] . i p 6 _ a d d r e s s ) ; if( m o d e I P == A F _ I N E T 6) { // r t a t t r i n i t buffer - > a t t r i b u t e [ 0 ] . a t t r . r t a _ l e n = 2 0 ; buffer - > a t t r i b u t e [ 1 ] . a t t r . r t a _ l e n = 2 0 ; buffer - > h e a d e r . n l m s g _ l e n = 6 4 ; } e l s e{ buffer - > a t t r i b u t e [ 0 ] . a t t r . r t a _ l e n =8; buffer - > a t t r i b u t e [ 1 ] . a t t r . r t a _ l e n =8; buffer - > h e a d e r . n l m s g _ l e n = 4 0 ; }
42 C Test buffer - > a t t r i b u t e [ 0 ] . a t t r . r t a _ t y p e =I F A _ L O C A L; buffer - > a t t r i b u t e [ 1 ] . a t t r . r t a _ t y p e =I F A _ A D D R E S S; buffer - > ifa . i f a _ f a m i l y = m o d e I P ; buffer - > h e a d e r . n l m s g _ s e q = 1 0 0 ; buffer - > h e a d e r . n l m s g _ p i d =0; // i f a d d r i n i t buffer - > ifa . i f a _ p r e f i x l e n = m a s k ; // m a s c h e r a buffer - > ifa . i f a _ f l a g s =0; buffer - > ifa . i f a _ s c o p e =R T _ S C O P E _ U N I V E R S E; buffer - > ifa . i f a _ i n d e x = 0; buffer - > h e a d e r . n l m s g _ t y p e = m o d e ; if( m o d e ==R T M _ N E W A D D R) { buffer - > h e a d e r . n l m s g _ f l a g s =N L M _ F _ R E Q U E S T|N L M _ F _ A C K|N L M _ F _ E X C L| N L M _ F _ C R E A T E; } e l s e{ buffer - > h e a d e r . n l m s g _ f l a g s =N L M _ F _ R E Q U E S T|N L M _ F _ A C K; } } test4 vsnlib.c # i n c l u d e " m i n l i b n e t l i n k . h " # d e f i n e B U F S I Z E 1 0 2 4 c h a r buf [B U F S I Z E];
int m a i n(int argc , c h a r** a r g v ) {
s t r u c t s o c k a d d r _ i n s e r v _ a d d r ; int fd ; s t r u c t s t a c k *s t a c k; s t r u c t n e t i f * nif ; v o i d* buf ; v o i d* b u f _ r ; v o i d* b u f _ l ; s t r u c t r e s p o n s e* ret ; s t r u c t i p _ a d d r a d d r _ 6 ; s t r u c t a d d r _ p a y l o a d b u f _ a d d r ; s t r u c t r o u t e _ p a y l o a d b u f _ r o u t e ; s t r u c t l i n k _ p a y l o a d b u f _ l i n k ; s t r u c t i p _ a d d r addr , m a s k ; buf =& b u f _ a d d r . h e a d e r ; b u f _ r =& b u f _ r o u t e . h e a d e r ;
C Test 43 // c a l l v s n l i b if(i n i t _ v s n l i b( a r g v [ 1 ] ) == -1) { p e r r o r(" C a n n o t i n i t v s n l i b ") ; } # i f d e f L W I P V 6 D L /* Run - t i m e l o a d t h e l i b r a r y ( if r e q u e s t e d ) */ if (( h a n d l e =l o a d l w i p v 6 d l() ) == N U L L) { p e r r o r(" L W I P lib not l o a d e d ") ; e x i t( -1) ; } # e n d i f /* d e f i n e a n e w s t a c k */ if((s t a c k=l w i p _ s t a c k _ n e w() ) ==N U L L) { p e r r o r(" L w i p s t a c k not c r e a t e d ") ; e x i t( -1) ; } /* a d d an i n t e r f a c e */ if(( nif =l w i p _ v d e i f _ a d d(stack, a r g v [ 2 ] ) ) ==N U L L) { p e r r o r(" I n t e r f a c e not l o a d e d ") ; e x i t( -1) ; } f i l l _ b u f _ l i n k(& b u f _ l i n k , 1) ; h a n d l e _ v s n l i b(& b u f _ l i n k , b u f _ l i n k . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ; f i l l _ b u f _ a d d r(& b u f _ a d d r , " 1 9 2 . 1 6 8 . 2 5 0 . 2 0 ", 24 , R T M _ N E W A D D R, A F _ I N E T) ; h a n d l e _ v s n l i b( buf , b u f _ a d d r . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; f i l l _ b u f _ r o u t e(& b u f _ r o u t e , " 1 9 2 . 1 6 8 . 2 5 0 . 2 ", R T M _ N E W R O U T E, A F _ I N E T) ; h a n d l e _ v s n l i b( buf_r , b u f _ r o u t e . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; m e m s e t((c h a r *) & s e r v _ a d d r ,0 ,s i z e o f( s e r v _ a d d r ) ) ; s e r v _ a d d r . s i n _ f a m i l y = A F _ I N E T; s e r v _ a d d r . s i n _ a d d r . s _ a d d r = i n e t _ a d d r(" 1 9 2 . 1 6 8 . 2 5 0 . 1 ") ; s e r v _ a d d r . s i n _ p o r t = h t o n s(a t o i(" 9 9 9 9 ") ) ; /* c r e a t e a T C P l w i p v 6 s o c k e t */ if(( fd =l w i p _ m s o c k e t(stack,PF_INET,S O C K _ S T R E A M,0) ) <0) { p e r r o r(" S o c k e t o p e n i n g e r r o r ") ; e x i t( -1) ; }
44 C Test /* c o n n e c t it to t h e a d d r e s s s p e c i f i e d as a r g v [ 1 ] p o r t a r g v [ 2 ] */ if (l w i p _ c o n n e c t( fd ,(s t r u c t s o c k a d d r *) (& s e r v _ a d d r ) ,s i z e o f( s e r v _ a d d r ) ) < 0) { p e r r o r(" S o c k e t c o n n e c t i n g e r r o r ") ; e x i t( -1) ; } w h i l e(1) { f d _ s e t r f d s ; int n ; F D _ Z E R O(& r f d s ) ; F D _ S E T(S T D I N _ F I L E N O,& r f d s ) ; F D _ S E T( fd ,& r f d s ) ; /* w a i t f o r i n p u t b o t h f r o m s t d i n a n d f r o m t h e s o c k e t */
l w i p _ s e l e c t( fd +1 ,& rfds ,NULL,NULL,N U L L) ;
/* c o p y d a t a f r o m t h e s o c k e t to s t d o u t */ if(F D _ I S S E T( fd ,& r f d s ) ) { if(( n =l w i p _ r e a d( fd , buf ,B U F S I Z E) ) == 0) e x i t(0) ; w r i t e (S T D O U T _ F I L E N O, buf , n ) ; } /* c o p y d a t a f r o m s t d i n to t h e s o c k e t */ if(F D _ I S S E T(S T D I N _ F I L E N O,& r f d s ) ) { if(( n =r e a d(S T D I N _ F I L E N O, buf ,B U F S I Z E) ) == 0) e x i t(0) ; l w i p _ w r i t e( fd , buf , n ) ; } } f i l l _ b u f _ r o u t e(& b u f _ r o u t e , " 1 9 2 . 1 6 8 . 2 5 0 . 1 ", R T M _ D E L R O U T E, A F _ I N E T) ; h a n d l e _ v s n l i b( buf_r , b u f _ r o u t e . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ; f i l l _ b u f _ a d d r(& b u f _ a d d r , " 1 9 2 . 1 6 8 . 2 5 0 . 2 0 ", 64 , R T M _ D E L A D D R, A F _ I N E T) ; h a n d l e _ v s n l i b( buf , b u f _ a d d r . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; f i l l _ b u f _ l i n k(& b u f _ l i n k , 0) ; h a n d l e _ v s n l i b(& b u f _ l i n k , b u f _ l i n k . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ; } test6 vsnlib.c # i n c l u d e " m i n l i b n e t l i n k . h " # d e f i n e B U F S I Z E 1 0 2 4 c h a r buf [B U F S I Z E];
C Test 45 s t r u c t s o c k a d d r _ i n 6 s e r v _ a d d r ; int fd ; s t r u c t s t a c k *s t a c k; s t r u c t n e t i f * nif ; v o i d* buf ; v o i d* b u f _ r ; v o i d* b u f _ l ; s t r u c t r e s p o n s e* ret ; s t r u c t i p _ a d d r a d d r _ 6 ; s t r u c t a d d r _ p a y l o a d b u f _ a d d r ; s t r u c t r o u t e _ p a y l o a d b u f _ r o u t e ; s t r u c t l i n k _ p a y l o a d b u f _ l i n k ; buf =& b u f _ a d d r . h e a d e r ; b u f _ r =& b u f _ r o u t e . h e a d e r ; // c a l l v s n l i b if(i n i t _ v s n l i b( a r g v [ 1 ] ) == -1) { p e r r o r(" C a n n o t i n i t v s n l i b ") ; } # i f d e f L W I P V 6 D L /* Run - t i m e l o a d t h e l i b r a r y ( if r e q u e s t e d ) */ if (( h a n d l e =l o a d l w i p v 6 d l() ) == N U L L) { p e r r o r(" L W I P lib not l o a d e d ") ; e x i t( -1) ; } # e n d i f /* d e f i n e a n e w s t a c k */ if((s t a c k=l w i p _ s t a c k _ n e w() ) ==N U L L) { p e r r o r(" L w i p s t a c k not c r e a t e d ") ; e x i t( -1) ; } /* a d d an i n t e r f a c e */ if(( nif =l w i p _ v d e i f _ a d d(stack, a r g v [ 2 ] ) ) ==N U L L) { p e r r o r(" I n t e r f a c e not l o a d e d ") ; e x i t( -1) ; } f i l l _ b u f _ l i n k(& b u f _ l i n k , 1) ; h a n d l e _ v s n l i b(& b u f _ l i n k , b u f _ l i n k . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ;
46 C Test f i l l _ b u f _ a d d r(& b u f _ a d d r , " 2 0 0 1 : db8 :0: f 1 0 1 ::3 ", 64 , R T M _ N E W A D D R, A F _ I N E T 6) ; h a n d l e _ v s n l i b( buf , b u f _ a d d r . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; f i l l _ b u f _ r o u t e(& b u f _ r o u t e , " 2 0 0 1 : db8 :0: f 1 0 1 ::2 ", R T M _ N E W R O U T E, A F _ I N E T 6) ; h a n d l e _ v s n l i b( buf_r , b u f _ r o u t e . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; m e m s e t((c h a r *) & s e r v _ a d d r ,0 ,s i z e o f( s e r v _ a d d r ) ) ; s e r v _ a d d r . s i n 6 _ f a m i l y = P F _ I N E T 6; c h a r str [I N E T 6 _ A D D R S T R L E N]; int r e t o = i n e t _ p t o n(P F _ I N E T 6, " 2 0 0 1 : db8 :0: f 1 0 1 ::1 ", s e r v _ a d d r . s i n 6 _ a d d r . s 6 _ a d d r ) ; i n e t _ n t o p(P F _ I N E T 6, s e r v _ a d d r . s i n 6 _ a d d r . s6_addr , str , I N E T 6 _ A D D R S T R L E N) ; s e r v _ a d d r . s i n 6 _ p o r t = h t o n s(a t o i(" 9 9 9 9 ") ) ; /* c r e a t e a T C P l w i p v 6 s o c k e t */ if(( fd =l w i p _ m s o c k e t(stack,P F _ I N E T 6,S O C K _ S T R E A M,0) ) <0) { p e r r o r(" S o c k e t o p e n i n g e r r o r ") ; e x i t( -1) ; } /* c o n n e c t it to t h e a d d r e s s s p e c i f i e d as a r g v [ 1 ] p o r t a r g v [ 2 ] */ if (l w i p _ c o n n e c t( fd ,(s t r u c t s o c k a d d r *) (& s e r v _ a d d r ) ,s i z e o f( s e r v _ a d d r ) ) < 0) { p e r r o r(" S o c k e t c o n n e c t i n g e r r o r ") ; e x i t( -1) ; } w h i l e(1) { f d _ s e t r f d s ; int n ; F D _ Z E R O(& r f d s ) ; F D _ S E T(S T D I N _ F I L E N O,& r f d s ) ; F D _ S E T( fd ,& r f d s ) ; /* w a i t f o r i n p u t b o t h f r o m s t d i n a n d f r o m t h e s o c k e t */
l w i p _ s e l e c t( fd +1 ,& rfds ,NULL,NULL,N U L L) ;
/* c o p y d a t a f r o m t h e s o c k e t to s t d o u t */ if(F D _ I S S E T( fd ,& r f d s ) ) { if(( n =l w i p _ r e a d( fd , buf ,B U F S I Z E) ) == 0) e x i t(0) ; w r i t e (S T D O U T _ F I L E N O, buf , n ) ; } /* c o p y d a t a f r o m s t d i n to t h e s o c k e t */ if(F D _ I S S E T(S T D I N _ F I L E N O,& r f d s ) ) { if(( n =r e a d(S T D I N _ F I L E N O, buf ,B U F S I Z E) ) == 0) e x i t(0) ; l w i p _ w r i t e( fd , buf , n ) ; }
C Test 47 } f i l l _ b u f _ r o u t e(& b u f _ r o u t e , " 2 0 0 1 : db8 :0: f 1 0 1 ::2 ", R T M _ D E L R O U T E, A F _ I N E T 6) ; h a n d l e _ v s n l i b( buf_r , b u f _ r o u t e . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ; f i l l _ b u f _ a d d r(& b u f _ a d d r , " 2 0 0 1 : db8 :0: f 1 0 1 ::3 ", 64 , R T M _ D E L A D D R, A F _ I N E T 6 ) ; h a n d l e _ v s n l i b( buf , b u f _ a d d r . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*)s t a c k) ; f i l l _ b u f _ l i n k(& b u f _ l i n k , 0) ; h a n d l e _ v s n l i b(& b u f _ l i n k , b u f _ l i n k . h e a d e r . n l m s g _ l e n , (v o i d*) nif , (v o i d*) s t a c k) ; } Makefile 1 CC= cc 2 F L A G _ C C= - ldl - l p t h r e a d - l l w i p v 6 - l v s n - g 3 B U I L D D I R=./ b u i l d / 4 OBJ=$ (B U I L D D I R)m i n l i b n e t l i n k . o 5 F L A G _ P R E P= - c - o 6 7 all : t e s t 4 t e s t 6 8 9 t e s t 4 : b u i l d l i b $ (OBJ) 10 $ (CC) $ (F L A G _ P R E P) $ (B U I L D D I R)t e s t 4 . o t e s t 4 _ v s n l i b . c 11 $ (CC) - o $ (B U I L D D I R)t e s t 4 .exe $ (B U I L D D I R)t e s t 4 . o $ (OBJ) $ (F L A G _ C C) 12 13 t e s t 6 : b u i l d l i b $ (OBJ) 14 $ (CC) $ (F L A G _ P R E P) $ (B U I L D D I R) t e s t 6 . o t e s t 6 _ v s n l i b . c 15 $ (CC) - o $ (B U I L D D I R)t e s t 6 .exe $ (B U I L D D I R)t e s t 6 . o $ (OBJ) $ (F L A G _ C C) 16 17 b u i l d l i b : 18 m k d i r - p $ (B U I L D D I R) 19 $ (CC) $ (F L A G _ P R E P) $ (B U I L D D I R)m i n l i b n e t l i n k . o m i n l i b n e t l i n k . c 20 21 c l e a n : 22 rm - rf b u i l d
Bibliografia
[1] Renzo Davoli. Internet of Threads. http://www.cs.unibo.it/~renzo/ papers/2013.iciw.pdf, 2013.
[2] Renzo Davoli. Internet of Threads: Processes as Internet Nodes. http: //www.cs.unibo.it/~renzo/papers/2014.IntTechIoTh.pdf, 2014. [3] Renzo Davoli. Internet of Threads. http://www.cs.unibo.it/~renzo/
papers/ConfGARR11_SelectedPapers_Davoli.pdf.
[4] Altran. picoTCP. https://github.com/tass-belgium/picotcp.
[5] Virtualsquare Team. vuos. https://github.com/virtualsquare/
vuos.
[6] Virtualsquare Team. LWIPv6. http://wiki.v2.cs.unibo.it/wiki/ index.php/LWIPV6.
[7] Virtualsquare Team. UMview. http://wiki.v2.cs.unibo.it/wiki/ index.php/UMview.
[8] Virtualsquare Team. purelibc. http://wiki.virtualsquare.org/
wiki/index.php/PureLibc.
[9] NetLink associazione LUGMan (Linux Users Group Mantova). http: //lugman.org/images/4/43/NetLink.pdf, 2009.
[10] Thomas Graf. https://www.infradead.org/~RFC4291RFC4291tgr/ libnl/.
50 BIBLIOGRAFIA
[11] Thomas Graf. https://github.com/tgraf/libnl. [12] Thomas Haller. https://github.com/thom311/libnl. [13] LWIP. https://savannah.nongnu.org/projects/lwip/.
[14] Reti mesh picoTCP. http://www.picotcp.com/mesh-design-guide. [15] Virtual Square. http://www.virtualsquare.org.
[16] Renzo Davoli. vdens. https://github.com/rd235/vdens. [17] RFC 4291, https://tools.ietf.org/html/rfc4291.