

# Passi da seguire



[https://wiki.lokar.fmf.uni-lj.si/diriwiki/index.php/Kak%C5%A1no\\_programsko\\_opremo\\_stranka\\_resni%C4%8Dno\\_potrebuje](https://wiki.lokar.fmf.uni-lj.si/diriwiki/index.php/Kak%C5%A1no_programsko_opremo_stranka_resni%C4%8Dno_potrebuje)



How the customer explained it



How the Project Leader understood it



How the Analyst designed it



How the Programmer wrote it



How the Business Consultant described it



How the project was documented



What operations installed



How the customer was billed



How it was supported



What the customer really needed

## Il «contratto»



**Requisiti chiari...**



## Esercizio

Descrivere requisiti (obiettivo e contratto) e metodo dell'algoritmo che riceve in ingresso due array ordinati in modo crescente ( $v$  e  $w$ ) e restituisce in uscita un nuovo array con all'interno l'insieme ordinato degli elementi dei due array originari.



# Esercizio

prototipo:

pre-condizione:

post-condizione:

`int[] merge (int[] v , int[] w)`  
v e w sono ordinati in modo crescente  
restituisce un nuovo array ordinato in modo crescente  
con gli elementi di v e w



# Esercizio



# Esercizio



## ALGORITMO

Un algoritmo è un procedimento che risolve un determinato problema attraverso un numero finito di passi elementari, chiari e non ambigui.

Un algoritmo per essere eseguito su una macchina deve essere tradotto nel linguaggio che quella macchina riconosce.

L'algoritmo è quindi indipendente dal linguaggio di programmazione utilizzato.

## Esercizio

|                          |               |
|--------------------------|---------------|
| i = 1                    |               |
| j = 1                    |               |
| for( k=1 ; k<n+m ; k++ ) |               |
|                          | $F(A, B, C)$  |
| +                        | -             |
| $z[k] = v[i]$            | $z[k] = w[j]$ |
| $i = i + 1$              | $j = j + 1$   |

$$\left\{ \begin{array}{l} A = v[i] < w[j] \\ B = i \leq n \\ C = j \leq m \end{array} \right.$$

## Esercizio

$v[i] < w[j]$     $i \leq n$     $j \leq m$

| A | B | C | Situazione                             | Azione           | F |
|---|---|---|----------------------------------------|------------------|---|
| 0 | 0 | 0 | $v$ e $w$ sono terminati               | caso impossibile | X |
| 0 | 0 | 1 | $v$ è terminato                        | $z[k] = w[j]$    | 0 |
| 0 | 1 | 0 | $w$ è terminato                        | $z[k] = v[i]$    | 1 |
| 0 | 1 | 1 | $v[i] \geq w[j]$ $i \leq n$ $j \leq m$ | $z[k] = w[j]$    | 0 |
| 1 | 0 | 0 | $v$ e $w$ sono terminati               | caso impossibile | X |
| 1 | 0 | 1 | $v$ è terminato                        | $z[k] = w[j]$    | 0 |
| 1 | 1 | 0 | $w$ è terminato                        | $z[k] = v[i]$    | 1 |
| 1 | 1 | 1 | $v[i] < w[j]$ $i \leq n$ $j \leq m$    | $z[k] = v[i]$    | 1 |

## Teorema di Böhm-Jacopini

Il teorema di Böhm-Jacopini, enunciato nel 1966 da due informatici italiani dai quale prende il nome, afferma che:

«qualsiasi algoritmo può essere implementato utilizzando tre sole strutture, la **sequenza**, la **selezione** e il **iterazione**, che possono essere tra loro innestati fino a giungere ad un qualsivoglia livello di profondità finito (come le scatole cinesi)»

In altre parole, qualsiasi procedimento risolutivo non strutturato («codice a spaghetti») può essere trasformato in un equivalente algoritmo strutturato.

# L'aspetto complementare



...d'altra parte un algoritmo destrutturato potrebbe non essere traducibile in un linguaggio di programmazione imperativo in assenza di istruzioni di salto incondizionato (goto, break, exit, end, loop,...)



## DNS – Diagrammi Nassi-Shneiderman



Isaac "Ike" Nassi  
<http://www.nassi.com/ike.html>



Ben Shneiderman  
<http://www.cs.umd.edu/~ben/>

## DNS – Diagrammi Nassi-Shneiderman

- sono un valido strumento didattico per scrivere codice strutturato
- hanno intrinsecamente il concetto di indentazione del codice quindi ne aumentano la leggibilità
- nascono con il linguaggio Pascal ma ne possono essere utilizzati in tutti i linguaggi procedurali (C++, java o assembly)
- si integrano perfettamente nelle metodologie di progettazione del software (es. UML) e costituiscono una valida alternativa ai diagrammi di flusso (che generano la programmazione “a spaghetti”)
- la fase di traduzione nel linguaggio di programmazione è estremamente semplice

## DNS – Definizione ricorsiva



# Forma di un algoritmo DNS complesso



## Blocchi di selezione

Selezionare significa poter intraprendere vie risolutive differenti a seconda di condizioni che sono verificate durante l'esecuzione. La selezione può essere:

- **Singola**: se la condizione è verificata (+) viene eseguito un blocco, altrimenti (-) si procede con l'algoritmo;
- **Binaria**: se la condizione è verificata (+) viene eseguito blocco, altrimenti (-) viene eseguito un secondo blocco;
- **Multipla**: controlla il valore di una variabile all'interno di un intervallo di valori predefiniti. Quando la variabile corrisponde a uno dei valori della selezione viene eseguito il blocco ad esso associato, altrimenti (se presente) il blocco di default.

# Blocco di selezione



Selezione singola



Selezione binaria



Selezione multipla

# Trasformazione dei blocchi di selezione

## Selezione binaria



## Selezione multipla



## Blocchi iterativi

Un blocco di iterazione permette di eseguire ripetutamente un blocco di istruzioni fino a quando una condizione logica non è verificata.

Per via della loro natura ripetitiva un blocco di iterazione è anche detto **ciclo**.

for (istruzioni)

Blocco

Ciclo for

while (condizione)

Blocco

Ciclo while

do

Blocco

while (condizione)

Ciclo do - while

# Trasformazione dei blocchi di selezione



## Architettura ARM (Advanced Risc Machine)

Un processore con architettura ARM fa parte di una famiglia di processori di tipo **RISC** (Reduced Instruction Set Computer) generalmente con **registri a 32-bit**.

L'architettura di tipo RISC garantisce ottime prestazioni con un basso consumo energetico, per questo i processori ARM trovano un largo uso nel mercato dei dispositivi mobili dove il risparmio energetico è di fondamentale importanza. I processori ARM si trovano nei più comuni dispositivi come cellulari, tablet, televisori, videogiochi portatili, raspberry, ecc...

## Registri

I principali registri dei processori ARM sono 16, nominati da R0 a R15, e si distinguono in base al loro utilizzo:

- **R0 - R12** sono i registri di **uso generale**;
- **R13** (o **SP**) è generalmente il registro dello **Stack Pointer**, cioè l'indirizzo di memoria dello stack. Non è però obbligatorio il suo utilizzo come Stack pointer; viene usato solo per convenzione;
- **R14** (o **LR**) è il registro del **Link Register**, nel quale viene salvato l'indirizzo di **ritorno** nel momento in cui viene chiamata una procedura;
- **R15** (o **PC**) è il **Program Counter**, il registro contiene l'**indirizzo di memoria** della prossima istruzione da eseguire.

# Registri

|     |            |
|-----|------------|
| R0  | 0x00000000 |
| R1  | 0x00000000 |
| R2  | 0x00000000 |
| R3  | 0x00000000 |
| R4  | 0x00000000 |
| R5  | 0x00000000 |
| R6  | 0x00000000 |
| R7  | 0x00000000 |
| R8  | 0x00000000 |
| R9  | 0x00000000 |
| R10 | 0x00000000 |
| R11 | 0x00000000 |
| R12 | 0x00000000 |
| SP  | 0x00000000 |
| LR  | 0x00000000 |
| PC  | 0x00000000 |

## Registro di stato

Un altro registro molto importante è il **registro di stato CPSR** (Current Program Status Register) il quale tiene traccia dello stato del programma. Molto utili sono i **4 bit di stato** (detti **flags di stato**) i quali esprimono le seguenti condizioni aritmetiche:

- **Bit N** negativo;
- **Bit Z** zero;
- **Bit C** carry/overflow;
- **Bit V** overflow.



# Tabella delle condizioni aritmetico-logiche

| Suffisso       | Descrizione                     | Flag testato    | Condition Code |
|----------------|---------------------------------|-----------------|----------------|
| <b>EQ</b>      | Equal ( == )                    | Z = 1           | 0000           |
| <b>NE</b>      | Not Equal ( != )                | Z = 0           | 0001           |
| <b>CS / HS</b> | Unsigned maggiore uguale ( >= ) | C = 1           | 0010           |
| <b>CC / LO</b> | Unisgned minore stretto ( < )   | C = 0           | 0011           |
| <b>MI</b>      | Negativo ( < 0 )                | N = 1           | 0100           |
| <b>PL</b>      | Positivo o 0 ( >= 0 )           | N = 0           | 0101           |
| <b>VS</b>      | Overflow                        | V = 1           | 0110           |
| <b>VC</b>      | No overflow                     | V = 0           | 0111           |
| <b>HI</b>      | Unsigned maggiore stretto ( > ) | C = 1 AND Z = 0 | 1000           |
| <b>LS</b>      | Unsigned minore uguale ( <= )   | C = 0 OR Z = 1  | 1001           |
| <b>GE</b>      | Maggiore uguale ( >= )          | N = V           | 1010           |
| <b>LT</b>      | Minore stretto ( < )            | N != V          | 1011           |
| <b>GT</b>      | Maggiore stretto ( > )          | Z = 0 AND N = V | 1100           |
| <b>LE</b>      | Minore uguale ( <= )            | Z = 1 OR N != V | 1101           |
| <b>AL</b>      | Always                          | ---             | 1110           |

Istruzione:  
ADDGT



Condition Code: GT

In questo esempio l'istruzione **ADD** verrà eseguita solo se la condizione **GT** (maggiore stretto) sarà verificata. La condizione GT è identificata dal codice **1100** nella sezione del registro chiamata **condition code**.

In questo esempio la condizione è verificata in quanto nel **registro di stato CPSR** i flags NZCV rispettano le condizioni in tabella ( $Z = 1 \text{ OR } N \neq V$ ).

Registro CPSR

| N | Z | C | V | 0   |
|---|---|---|---|-----|
| 1 | 0 | 1 | 0 | ... |

$$Z = 1 \text{ OR } N \neq V$$

## Tipi di dati

I tipi di dati sulle quali le istruzioni assembly ARM operano sono:

- **Byte** 1-byte, 8-bit
- **Half word** 2-byte, 16-bit
- **Full word** o **word** 4-byte, 32-bit

Il linguaggio assemblativo può interpretare questi tipi di dati ed esprimere diversi formati quali:

- Numeri interi espressi in diverse basi
- Numeri in virgola mobile
- Valori booleani
- Singoli caratteri
- Stringhe di caratteri

## Set di istruzioni

Le istruzioni dell'architettura ARM possono essere suddivise in tre insiemi:

- **Istruzioni aritmetico-logiche** implementano le operazioni matematiche. Queste possono essere di tipo **aritmetiche** (somma, differenza, prodotto), **logiche** (operazioni booleane) o **relazionali** (comparazioni tra due valori).
- **Istruzioni di Branch** cambiano il controllo del flusso delle istruzioni, modificando il contenuto del Program Counter (R15). Le istruzioni di branch sono necessarie per l'implementazione di istruzioni di condizione (if-then-else), cicli e chiamate di funzioni.
- **Istruzioni di Load/Store** muovono dati da (load) e verso (store) la memoria principale.

## Istruzioni aritmetico-logiche

In linguaggio assembly ARM le istruzioni aritmetico-logiche hanno una **sintassi** del tipo

opcode{S}{condition} dest, op1, op2

- Per **opcode** si intende l'istruzione che deve essere eseguita.
- Il campo **{S}** è un suffisso opzionale che, se specificato, carica nel registro di stato CPSR il risultato dell'operazione compresi i flag di stato.

## Istruzioni aritmetico-logiche

- Anche il campo **{condition}** è un suffisso opzionale il quale definisce la condizione logica necessaria all'esecuzione dell'istruzione; se la condizione è verificata l'istruzione verrà eseguita altrimenti verrà ignorata.
- Il campo **dest** indica il **registro destinatario** nel quale verrà salvato il risultato dell'operazione.
- In fine **op1** e **op2** sono gli **operandi** dell'operazione che si desidera eseguire.

Bisogna specificare però che op1 fa riferimento all'operando soltanto mediante **indirizzamento a registro**, mentre op2 può indirizzare sia in maniera **immediata** che a **registro**.

## Esempi istruzioni aritmetico-logiche

ADD R0, R1, R2

Carico in R0 la somma tra il contenuto del registro R1 e del registro R2 (indirizzamento a registro).

ADD R0, R1, #16

Carico in R0 la somma tra il contenuto del registro R1 e il numero decimale 16 (indirizzamento immediato).

ADD R0, R1, #0xF

Carico in R0 la somma tra il contenuto del registro R1 e il numero esadecimale F (indirizzamento immediato).

## Esempi istruzioni aritmetico-logiche

ADDLT R0, R1, R2

Carico in R0 la somma tra il contenuto del registro R1 e del registro R2 solamente se la condizione indicata con il suffisso LT è verificata.

ADDS R0, R1, R2

Carico in R0 la somma tra il contenuto del registro R1 e del registro R2, inoltre carico lo stato del risultato nei flags NZCV del registro di stato CPSR.

Per esempio, se il risultato della somma va in overflow i flag C (carry) e V (overflow) verranno settati a 1.

# Principali istruzioni aritmetico-logiche

| Descrizione                              | Opcode | Sintassi            | Semantica                               |
|------------------------------------------|--------|---------------------|-----------------------------------------|
| Addizione                                | ADD    | ADD Rd, R1, R2/#imm | $R_d = R_1 + R_2/\#imm$                 |
| Sottrazione                              | SUB    | SUB Rd, R1, R2/#imm | $R_d = R_1 - R_2/\#imm$                 |
| Moltiplicazione                          | MUL    | MUL Rd, R1, R2/#imm | $R_d = R_1 \times R_2/\#imm$            |
| Carica nel registro                      | MOV    | MOV Rd, R1/#imm     | $R_d \leftarrow R_1/\#imm$              |
| Carica negato nel registro               | MVN    | MVN Rd, R1/#imm     | $R_d \leftarrow -(R_1/\#imm)$           |
| AND logico                               | AND    | AND Rd, R1, R2/#imm | $R_d = R_1 \wedge R_2/\#imm$            |
| OR logico                                | ORR    | ORR Rd, R1, R2/#imm | $R_d = R_1 \vee R_2/\#imm$              |
| OR esclusivo                             | EOR    | EOR Rd, R1, R2/#imm | $R_d = R_1 \oplus R_2/\#imm$            |
| AND con complemento del secondo operando | BIC    | BIC Rd, R1, R2/#imm | $R_d = R_1 \wedge \neg R_2/\#imm$       |
| Shift logico sinistro                    | LSL    | LSL Rd, R1, R2/#imm | $R_d = R_1 \ll R_2/\#imm$               |
| Shift logico destro                      | LSR    | LSR Rd, R1, R2/#imm | $R_d = R_1 \gg R_2/\#imm$               |
| Rotazione destra                         | ROR    | ROR Rd, R1, R2/#imm | $R_d = R_1 \circlearrowright R_2/\#imm$ |
| Confronto                                | CMP    | CMP R1, R2          | $NZCV \leftarrow R_1 - R_2$             |
| Negato del confronto                     | CMN    | CMN R1, R2          | $NZCV \leftarrow R_1 + R_2$             |

## Istruzioni di branch

Le istruzioni di **branch** (o **jump**) sono utili per il controllo del flusso di esecuzione delle istruzioni.

Le principali istruzioni sono **B** (branch) e **BL** (branch with link).

L'istruzione **B** carica nel PC (R15) l'indirizzo della prima istruzione della procedura che si desidera eseguire, causando così un salto nel flusso di esecuzione delle istruzioni. Per comodità le procedure vengono identificate univocamente con dei nomi detti **label** (o **etichetta**).

L'istruzione **BL** è simile all'istruzione **B**, in più però carica nel registro **LR** (R14) l'indirizzo di ritorno della procedura, ovvero il valore del PC nel momento in cui viene eseguita l'istruzione **BL**.

## Esempio chiamata procedura con istruzione BL

All'istante di tempo **T1** si sta eseguendo l'istruzione di ADD memoria che conterrà l'istruzione successiva (**BL Procedura**).

### Istante di tempo T1

|            |        |       |            |
|------------|--------|-------|------------|
| Esecuzione | -----> | [...] |            |
| PC         | -----> | ADD   | R3, R0, #1 |
|            |        | BL    | Procedura  |
|            |        | MOV   | R0, #2     |
|            |        | [...] |            |

All'istante di tempo **T2** verrà prelevata l'istruzione **BL** (precedentemente puntata dal Program Counter). A questo punto il PC punterà all'istruzione successiva (**MOV R0,#2**).

### Istante di tempo T2

|                 |       |            |
|-----------------|-------|------------|
|                 | [...] |            |
|                 | ADD   | R3, R0, #1 |
| Prelievo -----> | BL    | Procedura  |
| PC ----->       | MOV   | R0, #2     |
|                 | [...] |            |

All'istante di tempo **T3** viene eseguita l'istruzione di Branch with link e accadono i seguenti eventi nel seguente ordine:

1. Il **Link Register** verrà caricato con il valore del Program Counter;
2. Il Program Counter verrà caricato con l'indirizzo della prima istruzione di un segmento di codice identificato con l'etichetta **Procedura**.

Così facendo avverrà un **salto** nel flusso di esecuzione delle istruzioni.

Dall'istante **T4** in poi verrà eseguito il segmento di codice identificato con l'etichetta **Procedura**. All'istante **Tn**, quando si arriverà all'ultima istruzione di **Procedura**, per tornare al normale flusso di istruzioni al momento del **salto** basterà caricare nel Program Counter il contenuto del Link Registr. Così dall'istante **Tn+1** in poi il programma riprenderà il normale svolgimento.

## Istante di tempo Tn



## Istante di tempo T<sub>n+1</sub>

Esecuzione ----->    PC -----> [...]

[...]

|     |            |
|-----|------------|
| ADD | R3, R0, #1 |
| BL  | Procedura  |
| MOV | R0, #2     |

## Istruzione di Load e Store

L'istruzione **LDR (Load Register)** carica in un registro destinatario un byte, una half word o una full word contenuta in una data locazione della memoria principale. La sintassi è

LDR{type}{condition} dest, [pointer]

LDR{type}{condition} dest, [pointer,offset]

Il campo **dest** è il registro destinatario nel quale caricare il contenuto prelevato dalla memoria. Il campo **[pointer]** indica un **registro puntare** il quale definisce contenuto referenziare. È possibile inoltre definire un **offset** il quale verrà sommato all puntatore.

LDR R0, [R1, #2]

Carico in R0 la **parola** in memoria puntata dal registro R1



**LDRH R0, [R1, #2]**

Carico in R0 la HalfWord (**H**) 2 byte a partire dalla cella puntata dal registro R1



L'istruzione **STR** (**Store Register**) esegue esattamente l'operazione inversa di LDR; carica in memoria il contenuto di un registro **sorgente** all'indirizzo definito dal **puntatore**.

STR{condition} source, [pointer,offset]

STR R0, [R1]

Carico all'indirizzo puntato da R1 la parola contenuta nel registro R0



0x01F0 = 0000 0001 1111 0000

# **Traduzione dei blocchi DNS in linguaggio Assembly ARM**

# Sequenza di blocchi

Inizialmente si cercano tutte le variabili e costanti che compaiono nel DNS. Si assegna ogni variabile ai registri disponibili, se il loro numero è tale da contenere tutte le variabili (ed il numero di byte tali da poter essere memorizzati in un registro), altrimenti occorre memorizzarle in memoria.



# Sequenza di blocchi

Se il blocco contiene una istruzione semplice, si traduce con la corrispondente istruzione assembly, altrimenti si segue la procedura che segue per ciascuna tipologia di blocco:



# Selezione singola



Se la condizione logica è verificata viene eseguita l'istruzione ADD, altrimenti il flusso di esecuzione proseguirà il normale corso.

# Selezione singola



Se la condizione logica è verificata viene eseguita l'istruzione ADD, altrimenti il flusso di esecuzione proseguirà il normale corso.

# Selezione singola con branch

La selezione singola con esecuzione condizionata funziona nel caso ci sia una sola istruzione nel blocco, altrimenti è necessario utilizzare l'istruzione di branch.



## Selezione binaria

La selezione binaria verso altri blocchi richiedono una particolare attenzione, in quanto bisogna garantire il corretto proseguimento del flusso di istruzioni del programma.



# Selezione binaria



# Selezione multipla



## Variabili

| x  | Valore1 | Valore2 |
|----|---------|---------|
| R0 | R1      | R2      |

# Selezione multipla



# Blocchi iterativi

```
while (i < j)
    i = i * 2
```



ciclo  
CMP R0, R1  
BGE end  
MUL R0, R0,#2  
B ciclo

```
do
    i = i * 2
    while (i < j)
```



end  
ciclo  
MUL R0, R0,#2  
CMP R0, R1  
BLT ciclo

```
for (i = 0; i < n; i++)
    j = j + i
```



MOV R0, #0  
ciclo  
CMP R0, Rn  
BGE end  
ADD R1, R1, R0  
ADD R0, R0, #1  
B ciclo

end