• Non ci sono risultati.

3.2 I plugin scriptabili

4.3.2 Una prima soluzione

Guardando il sorgente della libreria SDL si pu`o notare che in varie parti all’interno del codice vengono eseguite alcune operazioni soltanto se l’applicazione SDL `e eseguita in una propria finestra.

Una prima soluzione potrebbe essere quella di cercare di far eseguire queste opera- zioni all’applicazione SDL. Attraverso la funzione SDL GetWMInfo() `e possibile ac- cedere alle informazioni private relative alla finestra e al Display assegnate da SDL. Una volta ottenute queste informazioni, `e possibile effettuare una chiamata alla funzio- ne XSelectInput() per chiedere al server X di inviare gli eventi relativi alla finestra all’applicazione SDL.

La nuova applicazione si chiama sdlEv. E stata creata la funzione´

setwinevents() che viene eseguita quando ci si accorge che l’applicazione deve usare una finestra gi`a esistente. Il codice4.2mostra tale funzione pi`u in dettaglio.

Listing 4.2: Parte della funzione setwinevents()

1 s t a t i c v o i d s e t w i n e v e n t s () { 2 [ . . . ] 3 X W i n d o w A t t r i b u t e s a t t r ; 4 l o n g e v e n t _ m a s k ; 5 S D L _ S y s W M i n f o i n f o ; 6 S D L _ V E R S I O N (& i n f o . v e r s i o n ) ; 7 D i s p l a y * S D L _ D i s p l a y = i n f o . i n f o . x11 . d i s p l a y ; 8 W i n d o w win = i n f o . i n f o . x11 . w i n d o w ; 9 10 e v e n t _ m a s k = K e y P r e s s M a s k | K e y R e l e a s e M a s k | B u t t o n P r e s s M a s k | 11 B u t t o n R e l e a s e M a s k | [ . . . ] ; 12 13 X G e t W i n d o w A t t r i b u t e s ( S D L _ D i s p l a y , win , & a t t r ) ; 14 if ( n o t _ y o u r s (& attr , B u t t o n P r e s s M a s k ) ) { 15 f p r i n t f ( stderr , " B u t t o n s a l r e a d y taken , r e m o v i n g \ n ") ; 16 e v e n t _ m a s k &= ~ B u t t o n P r e s s M a s k ; 17 } [ . . . ] 18 e v e n t _ m a s k |= a t t r . y o u r _ e v e n t _ m a s k ; 19 X S e l e c t I n p u t ( S D L _ D i s p l a y , win , e v e n t _ m a s k ) ; 20 }

Esistono alcuni eventi11 che possono essere gestiti soltanto da un’applicazione. La

variabile event mask contiene tutti gli eventi che si desidera siano gentiti da SDL, mentre in attr ci sono gli attributi attuali della finestra (e quindi anche gli eventi gi`a catturati dall’applicazione host). La funzione not yours() confronta tali maschere e nel caso in cui l’evento single recipient sia gi`a gestito viene eliminato dalla masche- re degli eventi da far gestire ad SDL. La chiamata alla funzione XSelectoInput() effettua l’assegnamento di tali eventi all’applicazione SDL.

Esecuzione

Vediamo adesso come si comporta questa applicazione quando eseguita nella finestra del plugin. Il plugin `e lo stesso dell’esempio precedente (ovvero plugin2), e nell’attri-

buto progtorun mettiamo questa volta sdlEv. L’applicazione SDL si accorger`a12che

viene eseguita in una finestra gi`a esistente ed eseguir`a la funzione setwinevents() quando necessario.

Gli eventi della tastiera sono gestiti correttamente dall’applicazione SDL. Premen- do i tasti freccia su/giu `e possibile cambiare il colore del rettangolo. Gli eventi del mouse sono gestiti soltanto in parte. Infatti se guardiamo le informazioni di debug mostrate sul terminale ci si accorge che l’evento pressione del mouse `e gi`a gestito dal- la finestra del plugin e quindi non pu`o essere gestito da SDL. Infatti premendo con il mouse all’interno del rettangolo, il fuoco passa temporaneamente al plugin. Uscen- do dal rettangolo e tornando sopra sar`a nuovamente possibile gestire gli eventi della tastiera da parte dell’applicazione SDL.

Per consentire una corretta e completa gestione del mouse `e necessario modificare il plugin e rilasciare la gestione degli eventi di pressione del mouse. Questa modifica `e stata applicata, insieme ad altre modifiche, nel plugin3.

4.3.3

La soluzione finale

Con l’esempio precedente non `e ancora possibile gestire il ridimensionamento della finestra. Infatti guardando ancora il sorgente della libreria SDL si nota che la funzio- ne di resize non `e effettuata se non si usa una propria finestra. Un altro problema `e informare la finestra del plugin quando deve essere ridimensionata.

La prima modifica la faremo al plugin. Per consentire il ridimensionamento met- teremo il plugin in attesa della ricezione di un evento di ConfigureNotify che viene in genere mandato dalle applicazioni quando `e necessario cambiare le geometrie del-

la finestra13. Per effettuare il resize del plugin utilizzeremo la capacit`a di scripting

del plugin e andremo a modificare le variabili width e height associate al plugin attraverso chiamate Javascript.

Per prima cosa alla prima esecuzione della funzione NPP SetWindow() viene in- stallato un handler per gestire l’evento di ConfigureNotify. Alla ricezione di un tale evento, se le dimensioni della finestra richiesta differiscono da quella attuale, viene chiamata la funzione doResize(). Questa funzione, attraverso chiamate a Javascript, modifica le variabili width e height appropriatamente. In questo modo il plugin `e in grado di ridimensionare la propria finestra.

12poich`e trova la variabile di ambiente SDL WINDOWIS settata. 13dimensione, posizione, . . .

Come secondo passo `e necessario modificare l’applicazione SDL e renderla in gra- do di inviare un evento di ConfigureNotify ogniqualvolta `e necessario cambiare dimen- sioni. Per far ci`o utilizziamo l’applicazione sdlCom. Come nell’applicazione sdlEv `e stata applicata la modifica per consentire la gestione degli eventi di tastiera e mouse. Ogni volta che `e necessario ridimensionare la finestra SDL, cio`e quando viene chiama- ta la funzione SDL SetVideoMode(), viene invocata una nuova funzione resize(). Questa funzione ha lo scopo di creare un nuovo evento di ConfigureNotify con le nuove dimensioni della finestra, e di inviarlo, attraverso la funzione di libreria X11

XSendEvent() al plugin.14 Inoltre viene effettuato il ridimensionamento della finestra

SDL che la libreria non effettua con la chiamata alla funzione XResizeWindow(). Esecuzione

Eseguento il plugin3 con l’attributo progtorun settato con l’applicazione sdlCom si pu`o notare come gli eventi di mouse e tastiera siano gestiti correttamente dal- l’applicazione anche all’interno del plugin. Inoltre premendo i tasti destra/sinistra il ridimensionamento funziona correttamente.

In conclusione, per integrare un’applicazione SDL in un plugin `e necessario effettuare delle modifiche all’applicazione SDL stessa.

Il plugin app-wrapper

app-wrapper `e un plugin che consente di inglobare applicazione arbitrarie all’interno

di una pagina web. Questa versione gira soltanto su Linux/FreeBSD in quanto nel codice si fa ampio uso di primitive a basso livello dipendente dalla piattarfoma in uso (X11).1

Le applicazioni che possono essere inglobate senza eccessivi problemi sono quelle sviluppate usando la libreria SDL, debitamente patchate, e varie applicazioni basate sul sistema X Window, come xterm, mplayer, Xnest, ecc.

5.1

Il file di configurazione

Il plugin app-wrapper supporta l’integrazione di varie applicazioni. Per fare ci`o `e necessario registrare nel browser un MIME type per ogni applicazione che si vuo-

le supportare. Il file app-wrapper.conf (codice 5.1) contiene la configurazione

di questi MIME types. Ogni riga del file contiene un MIME type nel formato

type/subtype:suffixes:description:[options:]command line, dove type/subtype `e il MIME type;

1Una versione per l’ambiente Microsoft Windows `e attualmente in sviluppo.

Listing 5.1: File di configurazione app-wrapper.conf di esempio.

# - - - s t a r t of e x a m p l e s - - - - a p p l i c a t i o n / x - x t e r m :: x t e r m i n t o a w i n d o w : x t e r m - i n t o % w a p p l i c a t i o n / x - a s t e r i s k : a s t e r i s k : a s t e r i s k i n t o a w i n d o w :/ s b i n / a s t e r i s k - vd v i d e o / x - flv : flv : m p l a y e r p l a y s flv : s t r e a m : s a f e : m p l a y e r - wid % w - v i d e o / d i v x : d i v x : d i v x m p l a y e r : s t r e a m : s a f e : m p l a y e r - wid % w - 44

Listing 5.2: Struttura dati per il plugin app-wrapper. 1 s t r u c t _ i n s t a n c e { 2 int l o c a l _ h r e f ; /* set if h r e f is f i l e :// ... */ 3 int s t r e a m _ o n _ s t d i n ; /* can h a n d l e s t r e a m on s t d i n */ 4 int p i p e _ f d [ 2 ] ; /* in c a s e we o p e n a p i p e on a s t r e a m ... */ 5 p i d _ t c h i l d _ p i d ; 6 l o n g w i n d o w ; /* the w i n d o w u s e d by s e t w i n d o w */ 7 c h a r c m d b u f [ 2 5 6 ] ; /* a b u f f e r to c o p y the e x t e r n a l c o m m a n d */ 8 int width , h e i g h t ; /* w i d t h and h e i g h t of p l u g i n w i n d o w */ 9 };

suffixes `e una lista di estensioni, separate da una virgola, supportate dal MIME type; description `e la descrizione del MIME type che `e mostrata dal browser accedendo

alla pagina about:plugins;

option `e un parametro opzionale (vedi sotto);

command line `e il comando da eseguire, dove a %w `e sostituito il window id della finestra del plugin.

Le opzioni disponibili attualmente sono due:

safe: informa il plugin che il MIME type indicato `e considerato sicuro e pu`o eseguire qualsiasi URL. Per questioni di sicurezza un MIME type `e attivo per default soltanto per URL locali2.

stream: l’oggetto indicato nell’attributo src o data `e passato allo stdin dell’applicazione.

Come detto in precedenza all’avvio del browser viene controllato nelle directory predefinite l’esistenza di nuovi plugin e in caso affermativo viene chiamata la funzione NP GetMIMEDescription(). La modifica del file di configurazione richiede che il browser richiami tale funzione in modo da poter caricare eventuali nuovi MIME types. Per forzare ci`o `e necessario ad ogni modifica del file di configurazione aggiornare il

timestamp del plugin3 in modo tale che il browser richiami correttamente la funzione

NP GetMIMEDescription() all’avvio.

2nella forma file:///

Listing 5.3: La funzione NP GetMIMEDescription() per il plugin app-wrapper. 1 c h a r * N P _ G e t M I M E D e s c r i p t i o n ( v o i d ) { 2 s t a t i c c h a r m i m e _ b u f [ 2 0 4 8 ] ; 3 c h a r * s ; 4 b z e r o ( m i m e _ b u f , s i z e o f ( m i m e _ b u f ) ) ; 5 s = c h e c k _ c o n f i g ( NULL , NULL , m i m e _ b u f , s i z e o f ( m i m e _ b u f ) ) ; 6 if ( s == N U L L ) 7 s = ""; 8 r e t u r n s ; 9 }

5.2

Struttura dati

La struttura dati per il plugin app-wrapper `e mostrata nel codice5.2.

local href `e un boolean che indica se l’url `e locale o remota. `E usato per testare se

stiamo eseguendo un comando sicuro o no;

stream on stdin `e un boolean che indica se `e necessario inviare il file specificato nell’attributo src o data allo stdin;

window `e un intero che contiene il window id della finestra del plugin dove verr`a eseguita l’applicazione esterna.

Gli altri campi sono autoesplicativi e non necessitano ulteriore aprofindimento.

5.3

A grandi linee

Nella figura5.1 `e mostrato uno schema con il funzionamento a grandi linee del plugin

app-wrapper. In linea di principio il flusso del plugin `e simile a quello degli esempi mostrati in precedenza, con la sostanziale differenza che molte funzioni delle NPAPI hanno degli scopi ben precisi.

• La funzione NP GetMIMEDescription() ha lo scopo di interpretare il file di configurazione alla ricerca di tutti i MIME types gestiti dal plugin.

• L’inizializzazione globale non effettua niente di particolare, mentre nella fun- zione NPP New() si effettuano importanti operazioni, quali controllare se stiamo eseguendo un url locale o remota, individuare il MIME type associato all’istan- za del plugin e scorrere il file di configurazione alla ricerca del comando per eseguire l’applicazione associata.

• La funzione NPP SetWindow() prepara la finestra associata al plugin ad esegui- re l’applicazione. Vengono deselezionati gli eventi gestiti dal plugin per default cosicch`e possano essere gestiti dall’applicazione esterna, viene installato l’hand- ler per la gestione del ridimensionamento della finestra, e viene poi chiamata la funzione run cmd() che avvier`a l’applicazione.

• Dopo l’avvio dell’applicazione, il plugin si mantiene in attesa di eventuali eventi di resize, invocando quando necessario la funzione doResize().

• La distruzione dell’istanza `e effettuata dalla funzione NPP Destroy(), che con- trolla se l’applicazione `e ancora in esecuzione e in caso affermativo gli viene

inviato un signale di terminazione.4

5.4

Compilazione e Installazione

Per la compilazione del plugin sono necessari, oltre ai file forniti nel pacchetto5, i

file di sviluppo di X11 e SDL. Tipicamente questi file sono rilevati correttamente dal Makefile fornito, in caso di necessit`a `e sufficiente modificare le linee -I... e -L... con i percorsi di suddetti files.

Le macro utilizzate sono le seguenti: • PLUGINDEBUG

• USEINITIAL • WANT X11 • WANT SDL • WANT STREAM

Una volta dato il comando Make, verr`a creato il file app-wrapper.so che `e la libreria contenente il plugin, da copiare, insieme al file di configurazione app-wrapper.conf nella directory dei plugin.

4viene inviato un segnale di SIGTERM per consentire una chiusura regolare dell’applicazione. 5come specificato nel capitolo 2

Figura 5.1: Schema del funzionamento del plugin app-wrapper. Sulla sinistra sono mostrate le principali funzioni facenti parti di NPAPI, mentre sulla destra sono specificate le ulteriori funzioni chiamate, o le operazioni che tale funzione esegue.

NP_GetMIMEDescription()

NP_GetMIMEDescription() check MIME types

NP_Initialize() NPP_New() NPP_SetWindow() NPP_Destroy() NP_Shutdown() deselect events Search command Check MIME Check url add handler run_cmd() sdl hack create command line

open pipe if stream fork() run the program Application is running

receive resize event run handler

doResize() resize window

kill program if running

Listing 5.4: La funzione NPP New() per il plugin app-wrapper. 1 N P E r r o r N P P _ N e w ( N P M I M E T y p e p l u g i n T y p e , NPP i n s t a n c e , 2 u i n t 1 6 mode , i n t 1 6 argc , c h a r * a r g n [] , 3 c h a r * a r g v [] , N P S a v e d D a t a * s a v e d ) { 4 [ . . . ] 5 if (! p l u g i n _ c a l l s _ j s ( i n s t a n c e , " d o c u m e n t . l o c a t i o n . h r e f " , & res ) ) 6 f p r i n t f ( stdedd , " E r r o r on j a v a s c r i p t c a l l \ n ") ; 7 e l s e { 8 f p r i n t f ( stderr , " f r e t u r n s [% s ]\ n " , 9 N P V A R I A N T _ T O _ S T R I N G ( res ) . u t f 8 c h a r a c t e r s ) ; 10 if ( N P V A R I A N T _ I S _ S T R I N G ( res ) ) { 11 c o n s t c h a r * x = N P V A R I A N T _ T O _ S T R I N G ( res ) . u t f 8 c h a r a c t e r s ; 12 if (! s t r n c a s e c m p ( x , " f i l e ://" , 7) ) 13 m y _ i n s t a n c e - > l o c a l _ h r e f = 1; 14 } 15 N P N _ R e l e a s e V a r i a n t V a l u e (& res ) ; 16 } 17 for ( i = 0; i < a r g c ; i ++) { 18 int l = s i z e o f ( m y _ i n s t a n c e - > c m d b u f ) - 1; 19 c h a r * s ; 20 f p r i n t f ( stderr , " A r g u m e n t % d % s = ’% s ’\ n " , 21 i , a r g n [ i ] , a r g v [ i ]) ; 22 /* s t o r e the t r a n s l a t e d t y p e in a l o c a l s t r i n g */ 23 if (! s t r c m p ( a r g n [ i ] , " t y p e ") && 24 ( s = c h e c k _ c o n f i g ( a r g v [ i ] , m y _ i n s t a n c e , 25 m y _ i n s t a n c e - > cmdbuf , l ) ) ) { 26 s t r n c p y ( m y _ i n s t a n c e - > cmdbuf , s , l ) ; 27 m y _ i n s t a n c e - > c m d b u f [ l ] = ’\0 ’; 28 } 29 } 30 r e t u r n N P E R R _ N O _ E R R O R ; 31 }

5.5

Inizializzazione

5.5.1

La funzione NP GetMIMEDescription()

La funzione NP GetMIMEDescription() (codice 5.3) si occupa di caricare i vari

MIME types associati al plugin. Come detto in precedenza `e necessario forzare il browser a eseguire questa funzione ogni volta che vengono aggiunti o eliminati MI- ME types dal file di configurazione. Questa funzione chiama un’ulteriore funzione (check config()) che si occupa di interpretare il file di configurazione alla ricerca di MIME types. In questo caso viene passato un puntatore nullo all’istanza, in modo

tale da richiedere l’elenco dei MIME types supportati. Si veda l’appendice A.1 per

Listing 5.5: La funzione NPP SetWindow() per il plugin app-wrapper. 1 N P E r r o r N P P _ S e t W i n d o w ( NPP i n s t a n c e , N P W i n d o w * w i n d o w ) { 2 s t r u c t _ i n s t a n c e * me ; 3 N P S e t W i n d o w C a l l b a c k S t r u c t * w s _ i n f o ; 4 X W i n d o w A t t r i b u t e s a t t r ; 5 me = i n s t a n c e - > p d a t a ; 6 w s _ i n f o = window - > w s _ i n f o ; 7 if ( me - > w i n d o w == 0) { 8 l o n g p r i v a t e _ e v e n t s = B u t t o n P r e s s M a s k | S u b s t r u c t u r e R e d i r e c t M a s k ; 9 me - > w i n d o w = ( l o n g ) window - > w i n d o w ; 10 b z e r o (& attr , s i z e o f ( a t t r ) ) ;

11 X G e t W i n d o w A t t r i b u t e s ( ws_info - > display , me - > window , & a t t r ) ; 12

13 if ( a t t r . y o u r _ e v e n t _ m a s k & p r i v a t e _ e v e n t s ) { 14 a t t r . y o u r _ e v e n t _ m a s k &= ~ p r i v a t e _ e v e n t s ; 15 X S e l e c t I n p u t ( ws_info - > display , me - > window , 16 a t t r . y o u r _ e v e n t _ m a s k ) ; 17 } 18 19 W i d g e t x t w i d g e t = X t W i n d o w T o W i d g e t ( ws_info - > display , me - > w i n d o w ) ; 20 if ( x t w i d g e t ) { 21 X t A d d E v e n t H a n d l e r ( x t w i d g e t , S t r u c t u r e N o t i f y M a s k , False , 22 ( X t E v e n t H a n d l e r ) x t _ e v e n t _ h a n d l e r , i n s t a n c e ) ; 23 } 24 r u n _ c m d ( me ) ; 25 } 26 [ . . . ] 27 r e t u r n N P E R R _ N O _ E R R O R ; 28 }

5.5.2

Creazione dell’istanza

La funzione NPP New() (codice 5.4) dopo l’allocazione e l’inizializzazione del-

la memoria per l’istanza, controlla se l’url `e locale o meno attraverso una chia- mata document.location.href di Javascript. Dopo viene chiamata la funzione

check config() (vediA.1) passandogli il MIME type in modo che la questa funzio-

ne ritorni il comando da eseguire per avviare l’applicazione corrispondente al MIME types passatogli.

Documenti correlati