• No se han encontrado resultados

6. Fase de comprobaci´ on sem´ antica

6.3. Analizador sem´antico

El programa utilizado para realizar todas estas funciones es lo que hemos llamado Analizador Sem´antico y sus fuentes se muestran en el Ap´endice C. Lo m´as destacado de este c´odigo y lo que

6.3. ANALIZADOR SEM ´ANTICO 43

causa que no quede mucho que comentar sobre ´el, es su aproximaci´on casi exacta a las reglas descritas anteriormente, ya que al estar implementadas en un lenguaje funcional como Haskell, son simples, aunque no siempre triviales, “traducciones” de las reglas al lenguaje funcional, poniendo como condiciones las precondiciones de las reglas y como valor devuelto la postcondici´on adecuada. Aparte claro est´a de la implementaci´on de las operaciones requeridas, realizadas de forma muy natural al haberse especificado como listas, lo que se acercan mucho a su implementaci´on real.

Como gu´ıa para el lector comentaremos las diferentes reglas mostradas en las figuras. En primer lugar las reglas para expresiones de la figura 6.2. No hay renombramiento de literales enteros y booleanos verificandose autom´aticamente, as´ı como la validaci´on de la regi´on self. Las formas b´asicas, como son las variables la copia y el reuso simplemente buscan el renombramiento en la tabla y lo devuelven. El caso de let y case es algo m´as complicado, primero comprueban la expresi´on principal y luego los patrones para crear nuevos ´ambitos que servir´an de referencia para la aplicaci´on de las reglas a las expresiones correspondientes. Para las expresiones guardadas simplemente se aplican las reglas a un lado y a otro y se devuelven las expresiones renombradas.

Reglas para los patrones, ver 6.3. Las constantes se validan autom´aticamente y para las variables se crea un nombre fresco que es apuntado en la tabla de salida, las constructoras son un paso recursivo aplocando a cada patr´on de la constructora la trasnformaci´on y devolveindo la uni´on de todas las tablas de salida. La regla para renombramiento de patrones en uan cl´ausula where ha sido explicada en la secci´on anterior.

En la figura 6.4 s´olo queda comentar que para varias definiciones se vantransformando en orden utilizando como T0 para la siguiente la T0 original unida a la T00 de salida de la definici´on actual.

La regla para los tipos de datos declarados por el usuario, ver 6.5 es una traducci´on literal de la regla informal dada anteriormente para este efecto.

Para m´as informaci´on, se˜nalar que las relaciones de renombramiento y validaci´on de decla- raciones de datos, expresiones, patrones, definiciones globales, partes izquierdas y partes dere- chas se llaman respectivamente, renombraData, renombraExp, renombraPatron, renombraDefs, renombraIzq y renombraDer, siendo estas las funciones m´as importantes del programa, a la par que las m´as simples, mientras que el resto son funciones auxiliares que tratan con problemas con- cretos o con particularidades del lenguaje de implementaci´on.

En caso de que un programa dado no cumpla las restricciones sem´anticas especificadas, el proceso de compilaci´on se interrumpir´a, informando al programador de que hubo un error sem´antico, identificando la restricci´on violada.

TS Tabla de s´ımbolos anidada. Puede verse como una lista Tm : Tm-1 : ... : T0, donde a mayor n´umero m´as interno es el ´ambito,

siendo Tm el ´ambito actual.

T0 Tabla global de TS, donde se apuntaran las definiciones a nivel del

programa.

TR Tabla de regiones de la funci´on en curso. Sirve para validar que la regi´on sobre la que se construye alg´un dato est´a efectivamente declarada. La regi´on local y temporal self, siempre validar´a positivamente cualquier consulta.

T : TS Acci´on de a˜nadir un nuevo ´ambito T a TS.

[x 7→ x0] Tabla unitaria donde x es la clave, variable del programa, y x0 es su renombramiento, correspondiente a un nombre fresco.

S

i Ti Operaci´on de uni´on de todas las tablas de un ´ambito. Esta uni´on es exclusiva, es decir, s´olo est´a definida si no hay dos claves iguales.

TS (x) = x0 Operaci´on de obtenci´on de renombramiento. El identificador x est´a de- finido en alg´un ´ambito, recorrido de interno a externo, de TS y su re- nombramiento es x0.

UsoDer (d) Comprobaci´on de uso. Dada una declaraci´on de tipo del usuario, com- prueba que todos los identificadores utilizados en la definici´on tienen, al menos, una aparici´on de uso.

T0

0 = Rec(T0, d) Operaci´on de transformaciones recursivas. Dada una declaraci´on d com-

prueba si existe alguna aparici´on recursiva con los mismos par´ametros, en ese caso devuelve una tabla T00 que es T0 cambi´andole las aparicio-

nes recursivas por el s´ımbolo especial REC utilizado posteriormente en el an´alisis de compartici´on

T0 ` C, T0 ` f Equivale a C ∈ dom(T0), f ∈ dom(T0).

T0 `ddd | T00 Comprobaci´on de declaraciones de datos del usuario. Los identificadores

de la declaraci´on d est´an bien definidos y adem´as todos son utilizados al menos una vez en la tabla derecha, las apariciones recursivas de d se hacen sobre los mismos argumentos y las regiones externas de todas las alternativas son la misma. T0

0 es T0 a˜nadi´endole la informaci´on que da

la declaraci´on.

TS ` e | e0 Renombramiento de expresiones. Los identificadores de la expresi´on e est´an bien definidos en TS y e0 es el resultado de sustituir en e dichos s´ımbolos por sus renombramientos.

TS `dp | p0, T Renombramiento de patrones. Los identificadores del patr´on p est´an bien definidos con respecto a TS, el renombramiento es p0 y se genera una tabla de un nuevo ´ambito T .

T0 `gdef | def0, T, T0

0 Renombramiento de definiciones globales. La definici´on global def

est´a bien construida con respecto a la tabla global T0, def’ es su re-

nombramiento y se generan dos tablas de ´ambito: T de un ´ambito m´as interno y T0

0 a˜nadi´endole a T0 los nuevos identificadores globales.

T S ` i | i T S ` b | b T S(x) = x0 T S ` x | x0 T S(x) = x0 T S ` x! | x0! TR` self T S(x) = x0 , TR` r T S ` x@r | x0@r T0 ` f i ( T S ` ei | e0i ) TR` r T S ` f ei@r | f e0i@r T0 ` C i ( T S ` ei| e0i ) TR` r T S ` C ei@r | C e0i@r T S ` e1 | e01 T S `dp1 | p01 , T1 T1 : T S ` e | e0 T S ` let p1 = e1 in e | let p01= e01 in e0 T S ` e | e0 ∀i ( T S `dpi | pi0 , Ti Ti : T S ` ei | e0i ) T S ` case e of pi → ei | case e0 of p0i→ e0i ∀i ( T S ` ei | e0i T S ` e00i | e000i T S ` | ei= e00i | | e0i = e000i

Figura 6.2: Reglas sem´anticas para expresiones

T S `di | i , ∅ T S `db | b , ∅ fresca(x0) T S `dx | x0 , [x 7→ x0] ∀i ( T S `dpi | p0i , Ti ) T0 ` C T S `dC pi | C p0 i , ∪iTi T S ` e1 | e01 T S `dp1 | p01 , T1 T1 : T S ` e2 | e02 . . . Tn−1: . . . : T1: T S ` en| e0n Tn−1: . . . : T1: T S `dpn| p0n , Tn Tn: . . . : T1 : T S ` e | e0 T S ` e where pi= ei | e0 where p0i = e0i Figura 6.3: Reglas sem´anticas para patrones

T0 6` f ∀i ( T0 `dpi | p0i , Ti)

T0 `g f pi@r | f p0i@r , (∪iTi) ∪ [r 7→ ()] , [f 7→ ()]

T0 ` f ∀i ( T0 `dpi | p0i , Ti)

T0`gf pi@r | f p0i@r , (∪iTi) ∪ [r 7→ ()] , ∅

T0`g izq | izq0 , T , T00 T : (T00∪ T0) ` der | der0

T0 `g izq der | izq0 der0 , ∅ , T00

T0`g def1| def01 , ∅ , T1 T0∪ T1`g def2 | def02 , ∅ , T2. . . ∪n−1i=0 Ti `g defn| def0n , ∅ , Tn T0 `gdef1 ; . . . ; defn| def01 ; . . . ; def0n , ∅ , ∪ni=0Ti

Figura 6.4: Reglas sem´anticas para definiciones globales

T0 6` d U soDer(d) T00 = Rec(T0, d)

T0`dd d | T00

Cap´ıtulo 7

Transformaciones de

desazucaramiento

En este cap´ıtulo se detallan las reglas que se siguen para transformar la sintaxis dulce, con los identificadores ya renombrados y los tipos ya inferidos, a la sintaxis amarga. Para cada posi- ble constructora del ´arbol abstracto hay una regla para eliminar las construcciones complejas y transformarlas en combinaciones de las construcciones m´as b´asicas. En el ap´endice D se muestran los detalles de la implementaci´on de c´omo se realizan dichas transformaciones. En esta secci´on, la transformada de una expresi´on e se denota por e. En la secci´on 7.1 pueden verse las reglas m´as sencillas, mientras que en la secci´on 7.2 se dedicar´an varias subsecciones para explicar en detalle las reglas m´as complejas.

Documento similar