# $\begin{array}{c} {\bf Manipulation~5:~Servo\text{-}moteur~command\'e~par} \\ {\bf PWM} \end{array}$

Département: **TIC** 

Unité d'enseignement:  $\mathbf{CSN}$ 

# Auteur(s):

- PILLONEL Bastien
- BOUGNON-PEIGNE Kévin

#### Professeur:

• MESSERLI Etienne

# Assistant:

• JACCARD Anthony

#### Date:

• 2023

# Introduction

Le but de ce laboratoire est de réaliser un système qui pilote un servo-moteur (abrégé *servo* pour le reste du rapport), sur commande par PWM (Pulse Width Modulation).

# Objectif

Les objectifs sont de concevoir, développer, simuler et tester un contrôleur de servo-moteur, sous la forme d'un système séquentiel simple.

Le système utilise le principe d'un PWM ("Pulse Width Modulation" ou "modulation à largeur d'impulsion") qui permet de transmettre une information analogique via un signal binaire. Ce signal PWM est responsable de la transmission de la consigne de position au servo.

Le laboratoire est décomposé en deux parties. Dans la première partie il s'agit de générer un signal PWM à l'aide d'un compteur en "dent de scie" et d'une comparaison (plus d'explications dans la section "Première partie: Création du PWM"). Dans la deuxième partie il s'agit de gérer la position courante du servo, selon le mode de fonctionnement et l'état des signaux de commande (voir la section "Deuxième partie: Gestion de la position").

# Spécification

#### PWM: Pulse Width Modulation

Un PWM est un signal carré de période fixe, à rapport cyclique changeant. Pour réaliser ce genre de signal, l'on se base sur un signal triangulaire (ou en dent de scie, dans le cas présent) et on le compare avec un signal de contrôle.

#### Voici une démonstration:



## Comportement du servo

Dans le cadre de ce laboratoire, le PWM du servo fonctionne selon ces informations:



# Analyse

# Première partie: Génération du PWM

Soit le bloc de cette partie représentée par:



Il est souhaité de générer un PWM selon les caractérstiques suivantes:



Pour générer ce PWM, voici les fonctions nécessaires:

- asynchrone: Reset
- synchrone : Un compteur de la période du PWM
- synchrone : Un élément mémoire pour le compteur précédent
- synchrone : Un rebouclement de ce compteur (chargement à 0)
- synchrone : Un comparateur entre le compteur de la période du PWM et le seuil d'entrée, pour fixer la sortie pwm\_o, soit à '1', soit à '0'.

Selon la liste précédente, on voit que la table de fonctions synchrones ne peut faire intervenir que les éléments liés au compteur. Car l'entrée **seuil\_i** et la sortie **pwm.o** sont régies par la règle:

On obtient alors le décodeur d'états futurs du compteur:

| top_1MHz_i | cpt_period | cpt_fut_period                        | Commentaires               |
|------------|------------|---------------------------------------|----------------------------|
| 0          | -          | $=$ cpt $\_$ period                   | Maintien des valeurs       |
| 1          | =19999     | =0                                    | Rebouclement de la période |
| 1          | /          | $=\!\!\operatorname{cpt\_period} + 1$ | Incrémentation du compteur |

Le schéma attendu sera montré en chapitre "Réalisation et Implémentation", afin de le comparer avec la vue RTL du bloque.

# Deuxième partie: Gestion de la position

Voici tout d'abord la liste des entrées et sortie de notre bloc "gestion de position":



Les fonctions nécessaires sont:

- asynchrone: reset
- synchrone : un compteur/décompteur pour le temps de l'impulsion haute  $(T_{on})$
- synchrone : un élément mémoire pour le compteur précédent
- synchrone : un rebouclement du compteur, lorsqu'il au maximum de  $T_{on}$  (pour le mode automatique)
- synchrone : un détecteur des limites "min" et "max" de  $T_{on}$ , avec maintien de la valeur une fois arrivé en buté (pour le mode manuel)
  - De plus, il faut charger une valeur de  $T_{on}$  correspondant à la position centrale, si  $T_{on}$  est hors limites

Note: Le compteur définit le temps haut  $T_{on}$ . Le compteur peut alors prendre comme valeur minimale 999, correspondant à un  $T_{on}$  de 1[ms] et une valeur maximale de 1999, correspondant à un  $T_{on}$  de 2[ms].

Si l'on reprend la donnée du labo, on remarque qu'il est important de donner des priorités pour chaque fonction:

- 1. Chargement pos. centrale si hors limite
- 2. Chargement pos. centrale si center i actif
- 3. Boucle d'incrémentation si mode\_i est actif (mode auto)
- 4. Incrément jusqu'à  $T_{on}$  max puis maintien si up\_i actif
- 5. Décrément jusqu'à  $T_{on}$  min puis maintien si down\_i

Voici alors la table de fonctions synchrones:

| center_i | $mode\_i$ | up_i | down_i | reg_pres | $reg\_fut$             | Commentaires     |
|----------|-----------|------|--------|----------|------------------------|------------------|
|          | /         | /    | /      | <999 ou  | =1499                  | Chargement       |
|          |           |      |        | >1999    |                        | pos. centrale si |
|          |           |      |        |          |                        | hors limite      |
| 1        | /         | /    | /      | /        | =1499                  | Chargement       |
|          |           |      |        |          |                        | pos. centrale    |
| /        | 1         | /    | /      | =1999    | =999                   | Rebouclement     |
| /        | 1         | /    | /      | autres   | $=$ reg $\_$ pres $+1$ | Incrément        |
|          |           |      |        |          |                        | (mode auto.)     |
| /        | /         | 1    | /      | =1999    | $=$ reg $\_$ pres      | Maintien         |
| /        | /         | 1    | /      | autres   | $=$ reg $\_$ pres $+1$ | Incrément        |
|          |           |      |        |          |                        | (mode man.)      |
| /        | /         | /    | 1      | =999     | $=$ reg $\_$ pres      | Maintien         |
| /        | /         | /    | 1      | autres   | =reg_pres-1            | Soustraction     |

Un premier regroupement des valeurs de  $reg\_pres$ , égalent à "autres", peut être effectuer, on obtient donc la table suivante:

| center_ | _i modei | up_i | ${\rm down\_i}$ | $reg\_pres$ | $reg\_fut$          | Commentaires     |
|---------|----------|------|-----------------|-------------|---------------------|------------------|
|         | /        | /    | /               | <999 ou     | =1499               | Chargement       |
|         |          |      |                 | > 1999      |                     | pos. centrale si |
|         |          |      |                 |             |                     | hors limite      |
| 1       | /        | /    | /               | /           | =1499               | Chargement       |
|         |          |      |                 |             |                     | pos. centrale    |
| /       | 1        | /    | /               | =1999       | =999                | Rebouclement     |
| /       | /        | 1    | /               | =1999       | $=$ reg $\_$ pres   | Maintien         |
| /       | /        | /    | 1               | =999        | $=$ reg $\_$ pres   | Maintien         |
| /       | /        | /    | 1               | autres      | $=$ reg $\_$ pres-1 | Soustraction     |
| /       | /        | /    | /               | autres      | $=$ reg_pres $+1$   | Incrément        |

L'addition et la soustraction peuvent être effectuer par le même bloc additionneur (pour une soustraction, on met le report d'entrée à '1' et on inverse le second nombre d'entrée).

On peut donc coupler ces états et terminer avec la table suivante:

| center_ | i mode_i | up_i | down_i | $reg\_pres$      | $reg\_fut$        | Commentaires                            |
|---------|----------|------|--------|------------------|-------------------|-----------------------------------------|
| /       | /        | /    | /      | <999 ou<br>>1999 | =1499             | Chargement pos. centrale si hors limite |
| 1       | /        | /    | /      | /                | =1499             | Chargement pos. centrale                |
| /       | 1        | /    | /      | =1999            | =999              | Rebouclement                            |
| /       | /        | 1    | /      | =1999            | $=$ reg $\_$ pres | Maintien                                |
| /       | /        | /    | 1      | =999             | =reg_pres         | Maintien                                |
| /       | /        | /    | /      | autres           | =reg_pres "op."   | Incrément/Soustraction                  |

Comme pour la partie de génération PWM, le schéma estimé sera présenté en titre "Réalisation et Implémentation" pour être comparé avec la vue RTL.

# Réalisation et implantation

# Première partie: Génération du PWM

## Comparaison des schémas: Esquisse papier VS vue RTL





On voit que la vue RTL correspond bien au schéma attendu.

Remarque: La valeur héxadécimal 0x4E20 correspond à 20'000 et la condition est strictement plus petite, soit bien 19'999 comme énoncé avec la table de fonctions synchrones

# Deuxième partie: Gestion de la position

# Comparaison des schémas: Esquisse papier VS vue RTL





Ici encore, la vue RTL correspond bien à l'esquisse effectuée.

# Simulation

## Log de simulation

```
# vsim work.servo_pwm_tb
# Start time: 22:25:21 on May 02,2023
# ** Note: (vsim-8009) Loading existing optimized design _opt2
# Loading std.standard
# Loading std.textio(body)
# Loading ieee.std_logic_1164(body)
# Loading ieee.numeric_std(body)
# Loading ieee.fixed_float_types
# Loading ieee.math_real(body)
# Loading ieee.fixed_generic_pkg(body)
# Loading ieee.float_generic_pkg(body)
# Loading ieee.fixed_pkg
# Loading ieee.float_pkg
# Loading work.servo_pwm_tb(test_bench)#1
# Loading work.ilog_pkg(body)
# Loading work.servo_pwm_top(struct)#1
# Loading work.top_gen(calc)#1
# Loading work.top_gen(calc)#2
# Loading work.gestion_position(logic)#1
# Loading work.pwm(comp)#1
```

# Intégration/Mesure

L'intégration nous fourni l'analyse et le système complet comporte alors  ${\bf 140}$  éléments logiques:



## Prise de mesure

Les mesures ont étés prises en branchant une sonde d'oscilloscope sur la sortie du PWM, afin de mesurer le temps d'impulsion haute du signal.

Soit la pseudo représentation de la Mezzanine et le branchement de l'oscilloscope:

# Angle maximale





# Angle milieu





# Angle minimale





# Conclusion

Date: Date de rendu

- PILLONEL Bastien
- BOUGNON-PEIGNE Kévin

# Annexe(s)

- Résultat de simulation
- pwm.vhd
- gestion position.vhd
- Gestion position optimisée
  - Schéma attendu
  - Vue RTL
  - gestion\_position.vhd (optimisé)

## Résultat de la simulation

```
# ** Note: Debut de la simulation

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb

# ** Warning: NUMERIC_STD. TO_INTEGER: metavalue detected, returning 0

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/pum_inst

# ** Error: (vsim-86) Argument value -2147483647 is not in bounds of subtype NATURAL.

Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/pum_inst

# Warning: NUMERIC_STD.TO_UNSIGNED: vector truncated

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/pum_inst

# Warning: NUMERIC_STD.TO_INTEGER: metavalue detected, returning 0

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/pum_inst

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gest_pos

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gest_pos

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gest_pos

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gest_pos

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gest_pos

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gen_top_Zms

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gen_top_Zms

# Warning: NUMERIC_STD.** metavalue detected, returning TRUE

# Time: 0 ns Iteration: 0 Instance: /servo_pum_tb/dut/gen_top_Zms

# Warning: NUMERIC_STD.** metavalue detected, returning O

# Time: 0 ns Iteration: 1 Instance: /servo_pum_tb/dut/gen_top_IMIz

# Warning: NUMERIC_STD.** metavalue detected, returning FALSE

# Time: 0 ns Iteration: 1 Instance: /servo_pum_tb

# Warning: NUMERIC_STD.** metavalue detected, returning O

# Time: 30 ns Iteration: 3 Instance: /servo_pum_tb

# Warning: NUMERIC_STD.** instance: /servo_pum_tb
```

# pwm.vhd

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity pwm is
   port (-- Sync
         clock_i
                    : in std_logic;
         reset_i
                    : in std_logic;
          -- Inputs
         top_1MHz_i : in std_logic;
                     : in std_logic_vector(14 downto 0); -- range: [0, 2000]
         seuil_i
          -- Outputs
         pwm_o
                     : out std_logic
   ):
end entity pwm;
architecture comp of pwm is
    -- TO COMPLETE: Signals declaration
    -- | Signals |----
    -- Period counter with range [0, 20000] <=> 15bits needed
   signal cpt_period_reg_pres_s : unsigned(seuil_i'range);
   signal cpt_period_reg_fut_s : unsigned(seuil_i'range);
                               : std_logic;
   signal pwm_s
   begin
   -- TO COMPLETE: Sawtooth counter generation
-- Hold current value when top 1MHz ("enable" like) is down
    -- Add1 or Loop period's counter otherwise
   to_unsigned(0, cpt_period_reg_fut_s'length);
   -- D Flip-Flop / Register
   process(reset_i, clock_i)
   begin
     if reset_i = '1' then
       cpt_period_reg_pres_s <= (others => '0');
     elsif rising_edge(clock_i) then
      cpt_period_reg_pres_s <= cpt_period_reg_fut_s;</pre>
     end if;
   end process;
    -- TO COMPLETE: PWM signal generation and output
   -- Comparator
    -- Variant1: Compare directly period counter with threshold
   pwm_s <= '1' when cpt_period_reg_pres_s <= unsigned(seuil_i) else
'0';</pre>
    -- Variant2: First comparison with cpt_period > 20'000
               and then compare specific range with threshold
    --pwm_s <= '0' when cpt_period_reg_pres_s
                                                        >= 2000
              '0' when cpt_period_reg_pres_s(10 downto 0) > unsigned(seuil_i(10 downto 0)) else
   pwm_o <= pwm_s;</pre>
end architecture;
```

# gestion\_position.vhd

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity gestion_position is
    port (-- Sync
           clock_i
                          : in std_logic;
           reset_i
                         : in std_logic;
            -- Inputs
           down_i
                         : in std_logic;
                         : in std_logic;
           up i
           mode_i
                          : in std_logic;
           top_2ms_i
                        : in std_logic;
           center_i
                         : in std_logic;
            -- Outputs
           position
                         : out std_logic_vector(10 downto 0)
    ):
end entity gestion_position;
architecture logic of gestion_position is
    signal Q_pres_s, Q_fut_s, add_1_s, sub_1_s, mode_select_s, going_up_s, manual_mode_s : std_logic_vector(10 downto 0);
    signal det_min_s, det_max_s, det_out_range_s, center_s, enable_count_s, hold_value_s : std_logic;
    signal count_value_s : unsigned(10 downto 0);
     --/ Constants /----
    constant COUNT_MAX : unsigned(10 downto 0) := "11111001111"; -- unsigned 1999
    constant COUNT_MIN: unsigned(10 downto 0) := "11111001111"; -- unsigned 1999 constant COUNT_MID: unsigned(10 downto 0) := "101111101111"; -- unsigned 999 constant COUNT_MID: unsigned(10 downto 0) := "10111011011"; -- unsigned 1499
    begin
    - TO COMPLETE: Calculate position
- Intern signal to enable the flipflop during max Ton of the pwm
    enable_count_s <= top_2ms_i;</pre>
                       <= '1' when count_value_s = COUNT_MAX else
    det_max_s
                       '0';
<= '1' when count_value_s = COUNT_MIN else
    det_min_s
                          '0';
    det_out_range_s <= '1' when (count_value_s < COUNT_MIN OR count_value_s > COUNT_MAX) else
                          '0';
    count_value_s <= unsigned(Q_pres_s);
add_1_s <= std_logic_vector(count_value_s + 1);
sub_1_s <= std_logic_vector(count_value_s - 1);</pre>
                       <= det_out_range_s or center_i;</pre>
    hold_value_s <= (up_i and det_max_s) or (down_i and det_min_s);
```

```
-- Decoder of futur state
                   <= std_logic_vector(COUNT_MIN) when det_max_s = '1' else</pre>
    going_up_s
                       add_1_s;
   Q_pres_s;
   <= std_logic_vector(COUNT_MID) when center_s = '1' else
    Q_fut_s
                        mode_select_s;
   -- Process of a enabled flipflop process(reset_i, clock_i)
    begin
       -- reset the flip flop
if reset_i = '1' then
Q_pres_s <= (others => '0');
        elsif rising_edge(clock_i) then
  if enable_count_s = '1' then
    Q_pres_s <= Q_fut_s;
end if;</pre>
        end if;
    end process;
    -- TO COMPLETE: Position output
   position <= Q_pres_s;</pre>
end logic;
```

# Gestion position optimisée

Une seconde version, plus optimale, de la gestion de position a été réalisée.

À l'intégration, cette solution est complètement fonctionnelle, les mesures sont identiques aux précédentes inclues dans le rapport et est constitué de **120** éléments logiques, voici plus de détails:



Cette dernière n'a pas été vérifiée par l'assistant, car il restait 7 erreurs à la simulation et le temps manquait pour vérifier l'origine de chacune d'entre-elles.

#### Voici les logs de simulation:

```
# vsim -voptargs=""+acc"" work.servo_pwm_tb
# Start time: 11:44:16 on May 03,2023
\# ** Note: (vsim-3812) Design is being optimized...
# Loading std.standard
# Loading std.textio(body)
# Loading ieee.std_logic_1164(body)
# Loading ieee.numeric_std(body)
# Loading ieee.fixed_float_types
# Loading ieee.math_real(body)
# Loading ieee.fixed_generic_pkg(body)
# Loading ieee.float_generic_pkg(body)
# Loading ieee.fixed_pkg
# Loading ieee.float_pkg
# Loading work.servo_pwm_tb(test_bench)#1
# Loading work.ilog_pkg(body)
# Loading work.servo_pwm_top(struct)#1
# Loading work.top_gen(calc)#1
# Loading work.top_gen(calc)#2
# Loading work.gestion_position(logic)#1
# Loading work.pwm(comp)#1
```

#### Erreurs obtenues:

```
# ** Note: Debut de la simulation
# Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb
# ** Warning: NUMERIC_STD."<=": metavalue detected, returning FALSE</pre>
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/pwm_inst
# ** Warning: NUMERIC_STD."<": metavalue detected, returning FALSE</pre>
# Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/pwm_inst
# ** Warning: NUMERIC_STD."=": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/gest_pos
# ** Warning: NUMERIC_STD.">": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/gest_pos
# ** Warning: NUMERIC_STD."=": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/gest_pos
# ** Warning: NUMERIC_STD."<": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/gest_pos
# ** Warning: NUMERIC_STD."=": metavalue detected, returning FALSE
     Time: O ns Iteration: O Instance: /servo_pwm_tb/dut/gen_top_2ms
# ** Warning: NUMERIC_STD."/=": metavalue detected, returning TRUE
     Time: 0 ns Iteration: 0 Instance: /servo_pwm_tb/dut/gen_top_2ms
# ** Warning: NUMERIC_STD."=": metavalue detected, returning FALSE
     Time: O ns Iteration: O Instance: /servo_pwm_tb/dut/gen_top_1MHz
# ** Warning: NUMERIC_STD."/=": metavalue detected, returning TRUE
     Time: O ns Iteration: O Instance: /servo_pwm_tb/dut/gen_top_1MHz
# ** Warning: NUMERIC_STD."<=": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 1 Instance: /servo_pwm_tb/dut/pwm_inst
# ** Warning: NUMERIC_STD."=": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 1 Instance: /servo_pwm_tb/dut/gen_top_2ms
     Warning: NUMERIC_STD."<=": metavalue detected, returning FALSE
     Time: 0 ns Iteration: 4 Instance: /servo_pwm_tb/dut/pwm_inst
# ** Note: top_2ms_o OK
     Time: 666734 ns Iteration: 0 Instance: /servo_pwm_tb
# ** Error: >>>ERROR in pwm_o PWM duty cycle: expected between 49450 ns and 49483 ns but got 49500 ns
     Time: 1369615 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Error: >>>ERROR incorrect freq on pwm_o: expected between 600000 ns and 733333 ns but got 57717 ns
     Time: 166386082 ns Iteration: 5 Instance: /servo_pwm_tb
# ** Error: >>>ERROR in pwm_o PWM duty cycle: expected between 57733 ns and 57766 ns but got 33 ns # Time: 166386115 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Note: testcase upanddown done
     Time: 1321980082 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Error: >>>ERROR incorrect freq on pwm_o: expected between 600000 ns and 733333 ns but got 48543 ns
# Time: 1669932082 ns Iteration: 5 Instance: /servo_pwm_tb
# ** Error: >>>ERROR in pwm_o PWM duty cycle: expected between 49483 ns and 49516 ns but got 957 ns
# Time: 1669933039 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Error: >>>ERROR incorrect freq on pwm_o: expected between 600000 ns and 733333 ns but got 596442 ns # Time: 2689634524 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Error: >>>ERROR in pwm_o PWM duty cycle: expected between 49483 ns and 49516 ns but got 54219 ns
     Time: 2876478082 ns Iteration: 5 Instance: /servo_pwm_tb
# ** Note: testcase_tocenter done
     Time: 3143580082 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Note: testcase automode done
     Time: 3244626082 ns Iteration: 3 Instance: /servo pwm tb
# ** Note: testcase_errors done
     Time: 3346926082 ns Iteration: 3 Instance: /servo pwm tb
# ** Note: Fin de la simulation
     Time: 3346926082 ns Iteration: 3 Instance: /servo pwm tb
# ** Note: NOMBRE D'ERREURS : 7
     Time: 3346926082 ns Iteration: 3 Instance: /servo_pwm_tb
# ** Note: IL FAUT CONTINUER...
     Time: 3346926082 ns Iteration: 3 Instance: /servo_pwm_tb
```

Toutefois, certaines erreurs ont été analysées et se trouvent être, comme le cas où on a une erreur de rapport cyclique à 33[ns], dû à des glitchs de simulations.

# Schéma attendu VS vue RTL





Les deux vues sont identiques, si ce n'est la gestion du carry. Cette partie étant gérée avec des unsigned, en utilisant le addn d'un laboratoire précédent, il serait possible de mieux se rapprocher de l'ébauche papier.

# gestion\_position.vhd (optimisé)

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity gestion_position is
   port (-- Sync
          clock_i
                       : in std_logic;
                       : in std_logic;
          reset_i
           -- Inputs
          down_i
                      : in std_logic;
                       : in std_logic;
          up_i
          mode_i
                       : in std_logic;
                      : in std_logic;
          top_2ms_i
          center_i
                      : in std_logic;
          -- Outputs
                      : out std_logic_vector(10 downto 0)
          position
   ):
end entity gestion_position;
{\tt architecture}\ {\tt logic}\ {\tt of}\ {\tt gestion\_position}\ {\tt is}
    -- | Signals |--
    signal reg_pres_s, reg_fut_s : unsigned(position range);
   signal le_999_s, eq_999_s : std_logic; signal gt_1999_s, eq_1999_s : std_logic;
                           : std_logic;
: unsigned(position'range);
    signal sub_carry_s
    signal cst_one_s
    signal reg_plus_minus_one_s : unsigned(position'range);
    signal center_out_limits_s
                                    : std_logic;
    signal up_limit_s, down_limit_s : unsigned(position'range);
                                 : unsigned(position'range);
    signal loop_auto_mode_s
    --/ Constants /----
    constant LIMIT_UPPER_BOUND : unsigned(position'range) :=
                                                to_unsigned(1999, position'length);
    {\tt constant} \ {\tt CENTER\_VAL}
                                 : unsigned(position'range) :=
                                                to_unsigned(1499, position'length);
    constant LIMIT_LOWER_BOUND : unsigned(position'range) :=
                                                to_unsigned( 999, position'length);
```

```
begin
    -- TO COMPLETE: Calculate position
    -- Add 1 / Sub 1 part
    reg_plus_minus_one_s <= reg_pres_s + cst_one_s;</pre>
    -- In between signals
    center_out_limits_s <= center_i or le_999_s or gt_1999_s;</pre>
    {\tt loop\_auto\_mode\_s} \ \mathrel{<=} \ {\tt LIMIT\_LOWER\_BOUND} \ \ {\tt when} \ \ {\tt eq\_1999\_s} \ = \ {\tt '1'} \ \ {\tt else}
                       reg_plus_minus_one_s;
    up_limit_s <= reg_pres_s
                                        when eq_1999_s = '1' else
                 reg_plus_minus_one_s;
                                       when eq_999_s = '1' else
    down_limit_s <= reg_pres_s
                   reg_plus_minus_one_s;
    -- Main separation of Futur States Decoder
    reg_fut_s <= CENTER_VAL when center_out_limits_s = '1' else
                 loop_auto_mode_s when mode_i = '1'
up_limit_s when up_i = '1'
down_limit_s when down_i = '1'
                                                                     else
                                                                     else
                                                                     else
                 reg_pres_s;
    -- D Flip-Flop / Register
    process(reset_i, clock_i)
      if reset_i = '1' then
       reg_pres_s <= (others => '0');
      elsif rising_edge(clock_i) then
       if top_2ms_i = '1' then
        reg_pres_s <= reg_fut_s;
      end if;
    end process;
    -- TO COMPLETE: Position output
    position <= std_logic_vector(reg_pres_s);</pre>
    le_999_s <= '1' when reg_pres_s < LIMIT_LOWER_BOUND else</pre>
               '0';
    eq_999_s <= '1' when reg_pres_s = LIMIT_LOWER_BOUND else
    gt_1999_s <= '1' when reg_pres_s > LIMIT_UPPER_BOUND else
    eq_1999_s <= '1' when reg_pres_s = LIMIT_UPPER_BOUND else
                 '0';
end logic;
```