

Ingebedde systemen: project - controller voor servomotoren

Francis Begyn en Heleen Cauwels

2017

Begeleider: Dimitri Van Cauwelaert



## Inhoudstabel

| In       | houdstabel                               | 2  |
|----------|------------------------------------------|----|
| Li       | ijst met figuren                         | 3  |
| Li       | ijst met tabellen                        | 4  |
| Li       | ijst met afkortingen                     | 5  |
| In       | aleiding                                 | 6  |
| 1        | Vereisten en afspraken van de controller | 6  |
| <b>2</b> | Verslag code                             | 7  |
|          | 2.1 Entity                               | 7  |
|          | 2.2 Architecture                         | 7  |
|          | 2.3 Testbench                            | 11 |

# Lijst met figuren

# Lijst met tabellen

### Inleiding

De opdracht is het maken van een controller voor een servomotor in VHDL en een testbench om deze te kunnen uittesten. Hier is het ook belangrijk om rekening te houden met de vereisten van de klant en indien nodig het apparaat aan te passen.

Een servomotor is een motor die door middel van Pulse Width Modulation (PWM) aangestuurd kan worden naar een gewenste positie en deze in normale operatie zal behouden. De snelheid waarmee deze behouden wordt, is afhankelijk van de kloksnelheid die aan de controller meegeven wordt.

## 1 Vereisten en afspraken van de controller

De controller moet in staat om een servomotor over 180 aan te sturen met PWM signalen tussen 1,25ms en 1,75ms. De reset verloopt synchroon met de klok, dit is om te verhinderen dat het PWM signaal vervormt zou worden door de slechte timing van een reset. Moet men met een asynchrone reset werken, zou het PWM signaal onbedoeld langer kunnen worden dan de maximum pulsduur (1,75ms).

De controller werkt met een klok van 50Hz en een servoklok van 510kHz. Hiermee rekening houdende blijkt dat de minimum pulsduur an 1,25ms overeen komt met 637,5 servoklok ticks, het midden van 1.5ms komt overeen met 765 servoklok ticks en de maximum pulsduur van 1,75ms komt overeen met 892,5 servoklok ticks. In VHDL wordt gewerkt met usigned variabelen, dus 637,5 en 892,5 worden afgerond naar respectievelijk 637 en 892.

Elke controller heeft een adress waarop die moet luisteren. Als er opdrachten gestuurd worden die vooraf gegaan worden door het correcte address, dn moet de controller deze instructie uitvoeren. Indien een ander address vermeld wordt dan mag de controller de opdracht negeren.

Ook is er een broadcast adres waarop alle controller moeten luisteren, namelijk 255. Als een signaal uitgestuurd wordt met het broadcast adress dan moeten alle aangesloten controller naar dat signaal luisteren.

Onder normale operatie zal de controller de positie behouden, maar als er iets fout loopt zal de controller de servomotor terugzetten naar zijn neutrale positie. De neutrale positie wordt hier gedefinieerd als op 0 radialen, en komt overeen met een pulsduur van 1,5ms. De controller keert ook naar deze neutrale positie terug indien een reset opgeroepen wordt.

Het Done signaal werk met 3 state logic, dit betekend dat het op een bus aangesloten kan worden. Hierdoor kunnen meerdere controllers gebruik maken van nzelfde bus, waardoor er bespaard kan worden op het aantal kabels dat nodig is.

Als de aansturende microcontroller tijdens het aansturen van de controller beslist om een taak van hogere prioriteit uit te voeren, moet de controller dit kunnen afhandelen en naar de neutrale positie gaan.

## 2 Verslag code

### 2.1 Entity

```
library IEEE;
use IEEE.std_logic_1164.all;
use IEEE.numeric_std.all;
entity servocontrol is
  generic (
    address: unsigned(7 downto 0)
 port (
    \operatorname{clk}
           : in std_logic;
           : in std_logic;
    \operatorname{rst}
          : in std_logic;
           : in std_logic;
    \operatorname{set}
    data: in std_logic_vector (7 downto 0);
    done : out std_logic;
    pwm: out std_logic
  );
end entity;
```

#### 2.2 Architecture

In het begin van de archtectuur definieren we de benodigde signalen. Op basis van cht en pwmi kunnen we gemakkelijk een PWM singaal laten generen (verder meer details over hoe). Daarna worden de nodige types en signalen aangemaakt om een statenmachine te kunnen gebruiken. Eerst worden de verschillende staten gedefinieerd waarna de signalen voor het bijhouden van de staat ook aangemaakt worden.

```
library IEEE;
use IEEE.std_logic_1164. all;
use IEEE.numeric_std.all;
architecture control of servocontrol is
signal cnt: unsigned(9 downto 0);
```

```
signal pwmi: unsigned(9 downto 0);

--signal pwm_gen: std_logic;

type state is (idle,addr_rd,data_rd,move,hold);

signal currentState: state;

signal nextState: state;
```

state\_trans beschrijft de transities tussen verschillende staten. Hierbij hoort volgende flowchart.



Hierbij moet rekening gehouden worden met een aantal dingen. Zo is er een reset met hoogste prioriteit, deze heeft altijd voorrang. Moet er gekeken worden of het address juist is of het address het broadcast address is. Indien de set onderbroken wordt, moet de controller naar de neutrale positie gaan.

```
— State_trans describes the transitions of the states.
state_trans: process(rst,currentState,set,clk) begin
  if rst = '1' then
    nextState <= idle;
  else
    case currentState is
      when idle =>
        if set = '1' then
         nextState <= addr_rd;
         nextState \le idle;
        end if:
     when addr_rd =>
        if falling_edge (clk) then
          if (set = '1' and (unsigned(data) = address or unsigned(data) = 255))
              then
            nextState <= data_rd;
```

```
elsif set ='0' then
           nextState <= idle;
           nextState \le hold;
         end if:
       end if;
     when data_rd =>
       nextState <= move;
      when move =>
       nextState <= hold;
      when hold =>
        if set ='1' then
         nextState <= addr_rd;
        elsif set ='0' then
         nextState <= hold;
       end if:
      when others =>
       nextState <= idle;
     end case;
    end if;
end process state_trans;
```

transition is gewoon het synchroon proces dat de overgangen van de staten dicteert.

```
-- Transition is the actual transition beteen states
transition: process(clk) begin
  if rising_edge(clk) then
    currentState <= nextState;
  end if;
end process transition;</pre>
```

set\_output dicteert hoe de output signalen zich gedragen in elke staat. In deze setup stelt zich dit zo dat Done een 3 state logic heeft. Het PWM signaal wordt hier niet gedfinieerd omdat er een oneliner gebruikt wordt om dit aan de uitgang te koppelen (zie verder).

```
-- set_output determines which output correspont with a state
-- The done is defined so it works on bus structure, 3 state logic
set_output: process(currentState) begin
  case currentState is
  when idle =>
    done <= 'H';
    --pwm <= pwm_gen;
  when addr_rd =>
    done <= 'H';</pre>
```

```
when data_rd =>
    done <= 'L';
when move =>
    done <= 'L';
when hold =>
    done <= 'H';
    --pwm <= pwm_gen;
when others =>
    -- done <= '-';
end case;
end process set_output;</pre>
```

pwm\_data is het proces dat de berekeningen en lengte van de puls bepaalt. Op basis van de servoklok wordt berekend welke waarden nodig zijn om de gewenste pulsen te bekomen. In de commentaar zijn de berekende waarden voor een servoklok van 510kHz te vinden. Indien de servoklok aangepast wordt, met deze code aangepast worden zodat de juiste waarden gebruikt worden.

```
-- pwm_data sets the amount of ticks needed to generate a correct duration with
    servo clock = 510kHz
--1.25ms = 637 ticks, 1.5ms = 765 ticks, 1.75ms = 892 ticks
pwm_data: process(currentState,clk) begin
  case currentState is
    when idle =>
      pwmi <= unsigned(765); -- values according to 510kHz servo clock.
    when move =>
      if falling_edge (clk) then
        if data >= 255 then
          pwmi \leq unsigned(892);
         pwmi \leq unsigned('0' & data) + 637;
        end if;
      end if:
    when others =>
 end case;
end process pwm_data;
```

gen\_pwm is het proces dat de eigenlijk puls genereert. Het aantal servoklok ticks wordt bijgehouden en vergelijken met de benodigde aantallen die is pwm\_data bepaald zijn. cnt moet gerest wordne bij elke klok tick.

Nadien wordt met een oneliner eigenlijk het pwm signaal gegenereert door een klein procesje dat reageert op de veranderingen in gen\_pwm.

```
— gen_pwm is a combination of the proces below and the one—liners below it.
— gen_pwm counts the amount of ticks according to sclk and resets every clk
```

```
gen_pwm: process(clk, sclk) begin

if rising_edge(clk) then

cnt <= (others => '0');

elsif rising_edge(sclk) then

if cnt < 1023 then

cnt <= cnt + 1;

end if;

end if;

end process gen_pwm;

-- one-liner is the actual code that generates the output signal

-- pwm_gen <= '1' when (cnt < pwmi) else '0'; -- this only sends pwm signal

when explicitly asked by set_output

pwm <= '1' when (cnt < pwmi) else '0'; -- this holds the pwm signal at all times
```

#### 2.3 Testbench

De testbench is verantwoordelijk om de geschreven architecture uit te testen. De vereiste van de testbench is dat deze een klok en servoklok genereert, de servocontroller uittest in stappen van 32, de positie van de servomotor opvraagt en controleert. De servoklok mag niet gebruikt worden om het PWM signaal te controleren.

#### Conclusie