

# Semestrální práce z předmětu 36PJ

## Překlad VHDL do HTML

Michal Trs, [trsm1@fel.cvut.cz](mailto:trsm1@fel.cvut.cz)

### Zadání

Převaděč zdrojových souborů syntetizovatelného VHDL do HTML. Převaděč kontroluje syntaxi, obarvuje výstup a odkazuje od použití signálů a proměnných na jejich deklaraci. Vstupem je tedy *soubor.vhd* a výstupem *soubor.vhd.html*.

### Rozbor řešení

Lexikální analyzátor jsem realizoval tabulkou. Pomocí automatu identifikuje speciální znaky jako jsou závorky, dvojtečka, středník, logické operátory, dále čísla, řetězce, atribut, bit a identifikátor. Pokud je načtený symbol identifikátor, prohledám tabulku klíčových slov a případně vrátím token kw<klicove\_slovo>. Dále vracím token eof, pokud je načten konec souboru a token err pokud došlo k chybě (např.: špatný formát čísla, identifikátor začíná číslem, atd...).

Syntaktický analyzátor jsem realizoval funkciemi rekursivního sestupu. Jedná se o překladovou gramatiku. Pouze v několika málo případech jsem musel použít dědičný atribut, abych dokázal rozhodnout, zda pro načtený identifikátor budu vytvářet v HTML souboru návěští, nebo naopak se na návěští budu odkazovat. Dále jsem potřeboval tabulku Globálních deklarací a tabulku lokálních deklarací, aby při odkazování na proměnné docházelo k zastiňování.

### Gramatika

Vycházel jsem z popisu jazyka v notaci EBNF na této adrese [http://opensource.ethz.ch/emacs/vhdl93\\_syntax.html](http://opensource.ethz.ch/emacs/vhdl93_syntax.html), kterou jsem dále upravil (zjednodušil). Není podporována většina deklarací v entitě, jako např.: port map. Zde uvádím upravenou gramatiku kterou jsem implementoval.

```
adding_operator ::=      symbSign | symbOpAnd
architecture_body ::=      symbKwArch symbIDENT symbKwOf
                           symbIDENT symbKwIs declaration_part symbKwBegin
                           architecture_statement_part symbKwEnd [ symbKwArch ]
                           [ symbIDENT ] symbSemicolon
architecture_statement_part ::= { concurrent_statement }
```

assertion\_statement ::= **symbKwAssert** expression  
     [ **symbKwReport** expression ] [ **symbKwSever** expression ]  
     **symbSemicolon**

attribute\_name ::=       **symbATTRIB**  
     [ **symbBraOp** expression **symbBraCl** ]

block\_declarative\_item ::= type\_declaration | subtype\_declaration  
     | constant\_declaration | signal\_declaration  
     | variable\_declaration | use\_clause | component\_declaration

case\_statement ::=       **symbKwCase** expression **symbKwIs**  
     case\_statement\_alternative { case\_statement\_alternative }  
     **symbKwEnd symbKwCase** [ **symbIDENT** ]  
     **symbSemicolon**

case\_statement\_alternative ::=   **symbKwWhen** choice **symbChoice**  
     sequence\_of\_statements

component\_declaration ::= **symbKwComp symbKwIDENT**  
     [ **symbKwIs** ] [ generic\_clause ] [ port\_clause ]  
     **symbKwEnd symbKwComp** [ **symbIDENT** ]  
     **symbSemicolon**

concurrent\_statement ::= process\_statement | assertion\_statement  
     | concurrent\_statement2

concurrent\_statement2 ::= **symbIDENT symbCOLON**  
     concurrent\_statement3 | concurrent\_statement4

concurrent\_statement3 ::= process\_statement | assertion\_statement  
     | generate\_statement

concurrent\_statement4 ::= name3 signal\_assignment\_statement

constant\_declaration ::= **symbKwConst** name\_list **symbColon** name  
     [ **symbVarAss** expression ] **symbSemicolon**

context\_item ::=       Library\_clause | Use\_clause

declaration\_part ::=   { block\_declarative\_item }

design\_file ::=       Design\_unit { Design\_unit }

design\_unit ::=       { Context\_item }   Library\_unit

element\_association ::= **symbKwOthers symbChoice** expression

entity\_declaration ::=   **symbKwEnt symbIDENT symbKwIs**  
     entity\_headerdeclaration\_part

```

[ symbKwBegin entity_statement_part ] symbKwEnd
[ symbKwEnt ] [ symbIDENT ] symbSemicolon

entity_header ::= [ generic_clause ] [ port_clause ]
entity_statement ::= assertion_statement | process_statement
entity_statement_part ::= { entity_statement }

enumeration_type_definition ::= symbBraOp symbIDENT
{ symbCOMMA symbIDENT } symbBraCl

expression ::= relation { symbOpLog relation }
| relation [ symbOpLogSpec relation ]

factor ::= [ symbKwAbsNot ] primary

generate_statement ::= generation_scheme symbKwGen
[ declaration_part symbKwBegin ] { concurrent_statement }
symbKwEnd symbKwGen [ symbIDENT ]
symbSemicolon

generation_scheme ::= symbKwFor parameter_specification
| symbKwIf expression

generic_clause ::= symbKwGenic symbBraOp interface_list
symbBraCl symbSemicolon

choice ::= symbIDENT | symbKwOth
| symbSTRINGorVECTOR

identifier_list ::= symbIDENT { symbCOMMA symbIDENT }

if_statement ::= symbKwIf expression symbKwThen
sequence_of_statements { symbKwElsif expression
symbKwThen sequence_of_statements }
[ symbKwElse sequence_of_statements ] symbKwEnd
symbKwIf [ symbIDENT ] symbSemicolon

interface_declaration ::= [ symbKwConst | symbKwSignal
| symbKwVar ] identifier_list symbColon [ mode ] name
[ symbKwBus ] [ symbVarAss expression ]

interface_list ::= interface_declaration
{ symbSemicolon interface_declaration }

library_clause ::= symbKwLib name_list symbSemicolon

library_unit ::= entity_declaration | architecture_body

```

literal ::= **symbINT** | **symbREAL**  
 | **symbSTRINGorVECTOR** | **symbKwNull** | **symbBIT**

mode ::= **symbKwIn** | **symbKwMode**

name ::= name2 | **symbSTRINGorVECTOR**

name\_list ::= **symbIDENT** { **symbCOMMA symbIDENT** }

name2 ::= **symbIDENT** name3

name3 ::= [attribute\_name | selected\_name | slice\_name]

null\_statement ::= **symbKwNull symbSemicolon**

parameter\_specification ::= **symbIDENT symbKwIn** range

port\_clause ::= **symbKwPort symbBraOp** interface\_list  
**symbBraCl symbSemicolon**

prefix ::= **symbIDENT symbDot** prefix | **symbKwAll**

primary ::= name | literal | **symbBraOp** expression  
**symbBraCl** | element\_association

process\_declarative\_item ::= type\_declaration | subtype\_declaration  
 | constant\_declaration | variable\_declaration | use\_clause

process\_declarative\_part ::= { process\_declarative\_item }

process\_statement ::= [ **symbKwPost** ] **symbKwProc**  
 [ **symbBraOp** name\_list **symbBraCl** ] [ **symbKwIs** ]  
 process\_declarative\_part **symbKwBegin**  
 sequence\_of\_statements **symbKwEnd** [ **symbKwPost** ]  
**symbKwProc** [ **symbIDENT** ] **symbSemicolon**

range ::= simple\_expression  
 [ **symbKwDir** simple\_expression ]

relation ::= shift\_expression  
 [ relational\_operator shift\_expression ]

relational\_operator ::= **symbOpRel** | **symbSigAss**

selected\_name ::= **symbDot** prefix

sequence\_of\_statements ::= { sequential\_statement }

sequential\_statement ::= wait\_statement | assertion\_statement  
 | if\_statement | case\_statement | null\_statement  
 | sequential\_statement2

```

sequential_statement2 ::= symbIDENT symbColon
                      sequential_statement3 | sequential_statement4

sequential_statement3 ::= | wait_statement | assertion_statement
                      | if_statement | case_statement | null_statement

sequential_statement4 ::= name3 | signal_assignment_statement
                      | variable_assignment_statement

shift_expression ::= simple_expression
                   [ symbOpShift simple_expression ]

signal_assignment_statement ::= symbSigAss waveform
                             symbSemicolon

signal_declaration ::= symbKwSignal name_list symbColon name
                     [ symbKwReg | symbKwBus ] [ symbVarAss expression ]
                     symbSemicolon

simple_expression ::= [ symbSign ] term { adding_operator term }

slice_name ::= symbBraOp range symbBraCl

subtype_declaration ::= symbKwSubtype symbIDENT symbKwIs
                      name symbSemicolon

term ::= factor { symbOpMult factor }

type_declaration ::= symbKwType symbIDENT
                     [ symbKwIs enumeration_type_definition ]
                     symbSemicolon

use_clause ::= symbKwUse symbIDENT selected_name
               {symbCOMMA symbIDENT selected_name }
               symbSemicolon

variable_assignment_statement ::= symbVarAss expression
                               symbSemicolon

variable_declaration ::= [ symbKwShared ] symbKwVar name_list
                      symbColon name [ symbVarAss expression ]
                      symbSemicolon

wait_statement ::= symbKwWait [ symbKwOn name_list ]
                  [ symbKwUntil expresion ] [ symbKwFor expresion ]
                  symbSemicolon

waveform ::= expression [ symbKwAfter expression ]
            | symbKwNull [ symbKwAfter expression ]

```

## **Seznam klíčových slov**

abs, not, after, use, architekture, assert, begin, bus, case, component, constant ,to, downto, else, elsif, end, entity, for, generate, generic, if, in, is, library, out, inout, buffer, linkage, name, null, of, on, others, port, postponed, process, register, report, return, severity, shared, signal, subtype, then, type, unaffected, until, all, variable, wait, when

## **Závěr**

Překladač jsme implementoval v jazyce Java jako konzolovou aplikaci. Testoval jsem ji na zdrojových souborech komponenty DRAM scheduler z projektu Liberouter. Pro praktické použití by bylo potřeba implementovat celou gramatiku VHDL, aby bylo možné překládat sofistikovanější deklarace a propojení komponent a dále testbenche, které nejsou syntetizovatelné.